Skip to content

Commit 67ba749

Browse files
authored
fix: Refetch username after delay when github api rate limit is triggered (#891)
Fixes #859
1 parent 8299130 commit 67ba749

File tree

2 files changed

+48
-11
lines changed

2 files changed

+48
-11
lines changed

frontend/islands/GithubUserLink.tsx

+12-1
Original file line numberDiff line numberDiff line change
@@ -6,17 +6,28 @@ import { cachedGitHubLogin } from "../utils/github.ts";
66

77
export function GitHubUserLink({ user }: { user?: User }) {
88
const login = useSignal("");
9+
const error = useSignal(false);
910

1011
useEffect(() => {
1112
if (user) {
1213
cachedGitHubLogin(user)
1314
.then((login_) => {
1415
login.value = login_;
1516
})
16-
.catch(console.error);
17+
.catch((error_) => {
18+
console.error(error_);
19+
20+
error.value = true;
21+
});
1722
}
1823
});
1924

25+
if (error.value) {
26+
return (
27+
<span class="italic text-[0.625rem]">Could not load GitHub username</span>
28+
);
29+
}
30+
2031
return login.value == ""
2132
? <span>loading...</span>
2233
: <a class="link" href={"https://github.com/" + login.value}>GitHub</a>;

frontend/utils/github.ts

+36-10
Original file line numberDiff line numberDiff line change
@@ -5,15 +5,41 @@ import { getOrInsertItem } from "./client_cache.ts";
55
export async function cachedGitHubLogin(user: User): Promise<string> {
66
return await getOrInsertItem(
77
`gh-login-${user.githubId}`,
8-
() =>
9-
fetch(`https://api.github.com/user/${user.githubId}`, {
10-
headers: {
11-
"Content-Type": "application/json",
12-
},
13-
})
14-
.then((r) => r.json())
15-
.then((data) => {
16-
return data.login;
17-
}),
8+
() => {
9+
const MAX_RETRIES = 3;
10+
const fetchGithubUser = async (retryCount = 0) => {
11+
const response = await fetch(
12+
`https://api.github.com/user/${user.githubId}`,
13+
{
14+
headers: {
15+
"Content-Type": "application/json",
16+
},
17+
},
18+
);
19+
20+
if (
21+
response.status === 403 &&
22+
response.headers.get("x-ratelimit-remaining") === "0"
23+
) {
24+
throw new Error("Github API rate limit exceeded");
25+
}
26+
27+
const data = await response.json();
28+
29+
if (!data.login) {
30+
if (retryCount >= MAX_RETRIES) {
31+
throw new Error(
32+
"Failed to fetch GitHub login after maximum retries",
33+
);
34+
}
35+
36+
await new Promise((resolve) => setTimeout(resolve, 100));
37+
return fetchGithubUser(retryCount + 1);
38+
}
39+
return data.login;
40+
};
41+
42+
return fetchGithubUser();
43+
},
1844
);
1945
}

0 commit comments

Comments
 (0)