From 450f208327a0f428a5a90329b0587fdc735458c7 Mon Sep 17 00:00:00 2001
From: Bhav Beri <43399374+bhavberi@users.noreply.github.com>
Date: Thu, 18 May 2023 20:33:15 +0530
Subject: [PATCH 01/10] Added Total Reviews Trophy (#200)
---
src/github_api_client.ts | 1 +
src/trophy.ts | 50 ++++++++++++++++++++++++++++++++++++++++
src/trophy_list.ts | 2 ++
src/user_info.ts | 3 +++
4 files changed, 56 insertions(+)
diff --git a/src/github_api_client.ts b/src/github_api_client.ts
index 3c5d6c80..c23bc60c 100644
--- a/src/github_api_client.ts
+++ b/src/github_api_client.ts
@@ -38,6 +38,7 @@ export class GithubAPIClient {
contributionsCollection {
totalCommitContributions
restrictedContributionsCount
+ totalPullRequestReviewContributions
}
organizations(first: 1) {
totalCount
diff --git a/src/trophy.ts b/src/trophy.ts
index ff602e98..66d9f94f 100644
--- a/src/trophy.ts
+++ b/src/trophy.ts
@@ -216,6 +216,56 @@ export class OGAccountTrophy extends Trophy{
}
}
+export class TotalReviewsTrophy extends Trophy {
+ constructor(score: number) {
+ const rankConditions = [
+ new RankCondition(
+ RANK.SSS,
+ "God Reviewer",
+ 70,
+ ),
+ new RankCondition(
+ RANK.SS,
+ "Deep Reviewer",
+ 57,
+ ),
+ new RankCondition(
+ RANK.S,
+ "Super Reviewer",
+ 45,
+ ),
+ new RankCondition(
+ RANK.AAA,
+ "Ultra Reviewer",
+ 30,
+ ),
+ new RankCondition(
+ RANK.AA,
+ "Hyper Reviewer",
+ 20,
+ ),
+ new RankCondition(
+ RANK.A,
+ "Active Reviewer",
+ 8,
+ ),
+ new RankCondition(
+ RANK.B,
+ "Intermediate Reviewer",
+ 3,
+ ),
+ new RankCondition(
+ RANK.C,
+ "New Reviewer",
+ 1,
+ ),
+ ];
+ super(score, rankConditions);
+ this.title = "Reviews";
+ this.filterTitles = ["Review", "Reviews"];
+ }
+}
+
export class TotalStarTrophy extends Trophy {
constructor(score: number) {
const rankConditions = [
diff --git a/src/trophy_list.ts b/src/trophy_list.ts
index 6d99519e..7b5b9c51 100644
--- a/src/trophy_list.ts
+++ b/src/trophy_list.ts
@@ -6,6 +6,7 @@ import {
TotalIssueTrophy,
TotalPullRequestTrophy,
TotalRepositoryTrophy,
+ TotalReviewsTrophy,
MultipleLangTrophy,
LongTimeAccountTrophy,
AncientAccountTrophy,
@@ -28,6 +29,7 @@ export class TrophyList {
new TotalIssueTrophy(userInfo.totalIssues),
new TotalPullRequestTrophy(userInfo.totalPullRequests),
new TotalRepositoryTrophy(userInfo.totalRepositories),
+ new TotalReviewsTrophy(userInfo.totalReviews),
);
// Secret trophies
this.trophies.push(
diff --git a/src/user_info.ts b/src/user_info.ts
index e65f7eb2..c2e47c3b 100644
--- a/src/user_info.ts
+++ b/src/user_info.ts
@@ -31,6 +31,7 @@ export type GitHubUserActivity = {
contributionsCollection: {
totalCommitContributions: number;
restrictedContributionsCount: number;
+ totalPullRequestReviewContributions: number;
};
organizations: {
totalCount: number;
@@ -45,6 +46,7 @@ export class UserInfo {
public readonly totalIssues: number;
public readonly totalOrganizations: number;
public readonly totalPullRequests: number;
+ public readonly totalReviews: number;
public readonly totalStargazers: number;
public readonly totalRepositories: number;
public readonly languageCount: number;
@@ -94,6 +96,7 @@ export class UserInfo {
this.totalIssues = userIssue.openIssues.totalCount + userIssue.closedIssues.totalCount;
this.totalOrganizations = userActivity.organizations.totalCount;
this.totalPullRequests = userPullRequest.pullRequests.totalCount;
+ this.totalReviews = userActivity.contributionsCollection.totalPullRequestReviewContributions;
this.totalStargazers = totalStargazers;
this.totalRepositories = userRepository.repositories.totalCount;
this.languageCount = languages.size;
From 33b4105d8af3abeb5559286aec9093945a52efa4 Mon Sep 17 00:00:00 2001
From: ryo-ma
Date: Fri, 19 May 2023 08:57:42 +0900
Subject: [PATCH 02/10] Update README.md
---
README.md | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/README.md b/README.md
index 86379e0b..4056815f 100644
--- a/README.md
+++ b/README.md
@@ -41,7 +41,7 @@ Add the following code to your readme. When pasting the code into your profile's
```
-
+
## Use theme
From 22e46c69e7f9981ac152b10af329f2839378095c Mon Sep 17 00:00:00 2001
From: Alexandro Castro <10711649+AlexcastroDev@users.noreply.github.com>
Date: Tue, 26 Sep 2023 14:59:20 +0100
Subject: [PATCH 03/10] feat: implement staticRenderRegeneration (#221)
* feat: implement staticRenderRegeneration
* chore: remove the check in create
* chore: ignore read permission throw
* chore: safe read file
* fix: stream save
* chore: change file folder
---
CONTRIBUTING.md | 2 +-
index.ts | 26 +++++++---
src/StaticRenderRegeneration/cache_manager.ts | 49 +++++++++++++++++++
src/StaticRenderRegeneration/index.ts | 29 +++++++++++
src/StaticRenderRegeneration/types.ts | 6 +++
src/StaticRenderRegeneration/utils.ts | 42 ++++++++++++++++
src/utils.ts | 3 ++
7 files changed, 148 insertions(+), 9 deletions(-)
create mode 100644 src/StaticRenderRegeneration/cache_manager.ts
create mode 100644 src/StaticRenderRegeneration/index.ts
create mode 100644 src/StaticRenderRegeneration/types.ts
create mode 100644 src/StaticRenderRegeneration/utils.ts
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 355ef0a4..95eaad16 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -23,7 +23,7 @@ GITHUB_API=https://github.example.com/api/graphql
Run local server.
```sh
-deno run --allow-net --allow-read --allow-env debug.ts
+deno run --allow-net --allow-read --allow-env --allow-write debug.ts
```
Open localhost from your browser.
diff --git a/index.ts b/index.ts
index 2498b876..5f7f6cda 100644
--- a/index.ts
+++ b/index.ts
@@ -4,11 +4,25 @@ import { CONSTANTS, parseParams } from "./src/utils.ts";
import { COLORS, Theme } from "./src/theme.ts";
import { Error400, Error404 } from "./src/error_page.ts";
import "https://deno.land/x/dotenv@v0.5.0/load.ts";
-
+import {staticRenderRegeneration} from "./src/StaticRenderRegeneration/index.ts";
const apiEndpoint = Deno.env.get("GITHUB_API") || CONSTANTS.DEFAULT_GITHUB_API;
const client = new GithubAPIClient(apiEndpoint);
-export default async (req: Request) => {
+const defaultHeaders = new Headers(
+ {
+ "Content-Type": "image/svg+xml",
+ "Cache-Control": `public, max-age=${CONSTANTS.CACHE_MAX_AGE}`,
+ },
+)
+
+export default (request: Request) => staticRenderRegeneration(request, {
+ revalidate: CONSTANTS.REVALIDATE_TIME,
+ headers: defaultHeaders
+}, function (req: Request) {
+ return app(req);
+});
+
+async function app (req: Request): Promise{
const params = parseParams(req);
const username = params.get("username");
const row = params.getNumberValue("row", CONSTANTS.DEFAULT_MAX_ROW);
@@ -69,6 +83,7 @@ export default async (req: Request) => {
},
);
}
+
// Success Response
return new Response(
new Card(
@@ -83,12 +98,7 @@ export default async (req: Request) => {
noFrame,
).render(userInfo, theme),
{
- headers: new Headers(
- {
- "Content-Type": "image/svg+xml",
- "Cache-Control": `public, max-age=${CONSTANTS.CACHE_MAX_AGE}`,
- },
- ),
+ headers: defaultHeaders,
},
);
};
diff --git a/src/StaticRenderRegeneration/cache_manager.ts b/src/StaticRenderRegeneration/cache_manager.ts
new file mode 100644
index 00000000..3a93e4b4
--- /dev/null
+++ b/src/StaticRenderRegeneration/cache_manager.ts
@@ -0,0 +1,49 @@
+import { existsSync } from './utils.ts'
+
+export class CacheManager {
+ constructor(private revalidateTime: number, private cacheFile: string) {}
+
+ // Reason to use /tmp/:
+ // https://github.com/orgs/vercel/discussions/314
+ get cacheFilePath(): string {
+ return `/tmp/${this.cacheFile}`;
+ }
+ get cacheFileExists(): boolean {
+ return existsSync(this.cacheFilePath);
+ }
+
+ get cacheFileLastModified(): Date | null {
+ if (!this.cacheFileExists) {
+ return null;
+ }
+ const fileInfo = Deno.statSync(this.cacheFilePath);
+ return fileInfo.mtime ?? null;
+ }
+
+ get cacheFileLastModifiedGetTime(): number | null {
+ const lastModified = this.cacheFileLastModified;
+ if (lastModified === null) {
+ return null;
+ }
+ return lastModified.getTime();
+ }
+
+ get isCacheValid(): boolean {
+ if (this.cacheFileLastModifiedGetTime === null) {
+ return false;
+ }
+ const currentTime = new Date().getTime();
+ return currentTime - this.cacheFileLastModifiedGetTime < this.revalidateTime;
+ }
+
+ async save (response: Response): Promise {
+ if(response === null) return
+ // Prevent TypeError: ReadableStream is locked
+ const text = await response.clone().text()
+ const data = new TextEncoder().encode(text)
+
+ Deno.writeFile(this.cacheFilePath, data, { create: true }).catch(() => {
+ console.warn("Failed to save cache file")
+ });
+ }
+}
\ No newline at end of file
diff --git a/src/StaticRenderRegeneration/index.ts b/src/StaticRenderRegeneration/index.ts
new file mode 100644
index 00000000..d93fc61f
--- /dev/null
+++ b/src/StaticRenderRegeneration/index.ts
@@ -0,0 +1,29 @@
+import { CacheManager } from "./cache_manager.ts";
+import { StaticRegenerationOptions } from "./types.ts";
+import { getUrl, readCache, generateUUID } from "./utils.ts";
+
+export async function staticRenderRegeneration(request: Request, options: StaticRegenerationOptions, render: (request: Request) => Promise) {
+ // avoid TypeError: Invalid URL at deno:core
+ const url = getUrl(request)
+
+ // if more conditions are added, make sure to create a variable to skipCache
+ if (url.pathname === "/favicon.ico") {
+ return await render(request);
+ }
+
+ const cacheFile = await generateUUID(url.pathname + (url.search ?? ""));
+ const cacheManager = new CacheManager(options.revalidate ?? 0, cacheFile);
+ if (cacheManager.isCacheValid) {
+ const cache = readCache(cacheManager.cacheFilePath)
+ if(cache !== null) {
+ return new Response(cache, {
+ headers: options.headers ?? new Headers({}),
+ });
+ }
+ }
+
+ const response = await render(request)
+ cacheManager.save(response)
+
+ return response
+}
diff --git a/src/StaticRenderRegeneration/types.ts b/src/StaticRenderRegeneration/types.ts
new file mode 100644
index 00000000..e9b79076
--- /dev/null
+++ b/src/StaticRenderRegeneration/types.ts
@@ -0,0 +1,6 @@
+export interface StaticRegenerationOptions {
+ // The number of milliseconds before the page should be revalidated
+ revalidate?: number
+ // The headers to be sent with the response
+ headers?: Headers
+}
\ No newline at end of file
diff --git a/src/StaticRenderRegeneration/utils.ts b/src/StaticRenderRegeneration/utils.ts
new file mode 100644
index 00000000..ebc19c4b
--- /dev/null
+++ b/src/StaticRenderRegeneration/utils.ts
@@ -0,0 +1,42 @@
+export function getUrl(request: Request) {
+ try {
+ return new URL(request.url)
+ } catch {
+ return {
+ pathname: request.url,
+ search: request.url
+ }
+ }
+}
+
+export function readCache(cacheFilePath: string): Uint8Array | null {
+ try {
+ return Deno.readFileSync(cacheFilePath)
+ } catch {
+ return null
+ }
+}
+
+// https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/digest
+export async function generateUUID(message: string): Promise {
+ const encoder = new TextEncoder();
+ const data = encoder.encode(message);
+ const hashBuffer = await crypto.subtle.digest("SHA-256", data);
+
+ const hashArray = Array.from(new Uint8Array(hashBuffer));
+ const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
+
+ return hashHex;
+}
+
+
+export const existsSync = (filename: string): boolean => {
+ try {
+ Deno.statSync(filename);
+ // successful, file or directory must exist
+ return true;
+ } catch {
+ return false;
+ }
+};
+
\ No newline at end of file
diff --git a/src/utils.ts b/src/utils.ts
index 9f868f76..3b650435 100644
--- a/src/utils.ts
+++ b/src/utils.ts
@@ -53,6 +53,8 @@ export function abridgeScore(score: number): string {
return (Math.sign(score) * Math.abs(score)).toString() + "pt";
}
+const HOUR_IN_MILLISECONDS = 60 * 60 * 1000;
+
export const CONSTANTS = {
CACHE_MAX_AGE: 7200,
DEFAULT_PANEL_SIZE: 110,
@@ -63,6 +65,7 @@ export const CONSTANTS = {
DEFAULT_NO_BACKGROUND: false,
DEFAULT_NO_FRAME: false,
DEFAULT_GITHUB_API: "https://api.github.com/graphql",
+ REVALIDATE_TIME: HOUR_IN_MILLISECONDS,
};
export enum RANK {
From 85f806480710de6f70b4f6c15f9991a068159aee Mon Sep 17 00:00:00 2001
From: ryo-ma
Date: Tue, 26 Sep 2023 23:29:10 +0900
Subject: [PATCH 04/10] Update README.md
From a4d50a11dbd6560527243202cd5ca5c9782daf9a Mon Sep 17 00:00:00 2001
From: ryo-ma
Date: Thu, 28 Sep 2023 00:56:48 +0900
Subject: [PATCH 05/10] Fixed the number of Github API requests that were being
executed twice.
---
index.ts | 2 +-
src/github_api_client.ts | 51 +++++++++++++++++++++++++++++-----------
src/utils.ts | 1 +
3 files changed, 39 insertions(+), 15 deletions(-)
diff --git a/index.ts b/index.ts
index 5f7f6cda..0bbbcce8 100644
--- a/index.ts
+++ b/index.ts
@@ -101,4 +101,4 @@ async function app (req: Request): Promise{
headers: defaultHeaders,
},
);
-};
+}
diff --git a/src/github_api_client.ts b/src/github_api_client.ts
index c23bc60c..c4b8361e 100644
--- a/src/github_api_client.ts
+++ b/src/github_api_client.ts
@@ -24,6 +24,7 @@ export class GithubAPIClient {
this.requestUserRepository(username),
]);
if (results.some((r) => r == null)) {
+ console.error(`Can not find a user with username:'${username}'`);
return null;
}
return new UserInfo(results[0]!, results[1]!, results[2]!, results[3]!);
@@ -109,28 +110,50 @@ export class GithubAPIClient {
private async request(
query: string,
username: string,
+ retryDelay = CONSTANTS.DEFAULT_GITHUB_RETRY_DELAY,
) {
const tokens = [
Deno.env.get("GITHUB_TOKEN1"),
Deno.env.get("GITHUB_TOKEN2"),
];
+ const maxRetries = tokens.length;
+
const variables = { username: username };
let response;
- for (const token of tokens) {
- response = await soxa.post(
- this.apiEndpoint,
- {},
- {
- data: { query: query, variables },
- headers: { Authorization: `bearer ${token}` },
- },
- ).catch((error) => {
- console.error(error.response.data);
- });
- if (response.data.data !== undefined) {
- break;
+
+ for (let attempt = 0; attempt < maxRetries; attempt++) {
+ try {
+ response = await soxa.post(
+ this.apiEndpoint,
+ {},
+ {
+ data: { query: query, variables },
+ headers: { Authorization: `bearer ${tokens[attempt]}` },
+ },
+ );
+ if (response.data.errors !== undefined) {
+ throw new Error(
+ response.data.errors.map((e: { message: string; type: string }) =>
+ e.message
+ ).join("\n"),
+ );
+ }
+ if (response.data.data !== undefined) {
+ return response.data.data.user;
+ } else {
+ return null;
+ }
+ } catch (error) {
+ console.error(
+ `Attempt ${attempt} failed with GITHUB_TOKEN${attempt + 1}:`,
+ error,
+ );
}
+
+ console.log(`Retrying in ${retryDelay / 1000} seconds...`);
+ await new Promise((resolve) => setTimeout(resolve, retryDelay));
}
- return response.data.data.user;
+
+ throw new Error(`Max retries (${maxRetries}) exceeded.`);
}
}
diff --git a/src/utils.ts b/src/utils.ts
index 3b650435..0ed5cd30 100644
--- a/src/utils.ts
+++ b/src/utils.ts
@@ -65,6 +65,7 @@ export const CONSTANTS = {
DEFAULT_NO_BACKGROUND: false,
DEFAULT_NO_FRAME: false,
DEFAULT_GITHUB_API: "https://api.github.com/graphql",
+ DEFAULT_GITHUB_RETRY_DELAY: 1000,
REVALIDATE_TIME: HOUR_IN_MILLISECONDS,
};
From 5dc3379214adb0b9e53b9d7a87238e6d880cea87 Mon Sep 17 00:00:00 2001
From: Alexandro Castro <10711649+AlexcastroDev@users.noreply.github.com>
Date: Fri, 29 Sep 2023 17:54:23 +0100
Subject: [PATCH 06/10] Improvements on workflow and files (#222)
* chore: improvements
* chore: add testing workflow
* fix: types and lint
* chore: change dynamic
* chore: improvements
* chore: no need repository to check status
---
.github/workflows/testing.yml | 22 +++
README.md | 6 +
index.ts => api/index.ts | 37 ++--
debug.ts | 2 +-
deps.ts | 22 ++-
src/Helpers/Retry.ts | 44 +++++
src/Helpers/__tests__/Retry.test.ts | 66 ++++++++
src/Repository/GithubRepository.ts | 25 +++
src/Schemas/index.ts | 61 +++++++
src/Services/GithubApiService.ts | 120 +++++++++++++
src/StaticRenderRegeneration/cache_manager.ts | 81 ++++-----
src/Types/Request.ts | 5 +
src/Types/index.ts | 1 +
src/github_api_client.ts | 159 ------------------
vercel.json | 19 ++-
15 files changed, 446 insertions(+), 224 deletions(-)
create mode 100644 .github/workflows/testing.yml
rename index.ts => api/index.ts (73%)
create mode 100644 src/Helpers/Retry.ts
create mode 100644 src/Helpers/__tests__/Retry.test.ts
create mode 100644 src/Repository/GithubRepository.ts
create mode 100644 src/Schemas/index.ts
create mode 100644 src/Services/GithubApiService.ts
create mode 100644 src/Types/Request.ts
create mode 100644 src/Types/index.ts
delete mode 100644 src/github_api_client.ts
diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml
new file mode 100644
index 00000000..b8215530
--- /dev/null
+++ b/.github/workflows/testing.yml
@@ -0,0 +1,22 @@
+name: Check PR Test
+
+on:
+ pull_request:
+ branches:
+ - master
+jobs:
+ install-dependencies:
+ runs-on: ubuntu-latest
+ steps:
+ - name: Check if PR is in draft mode
+ run: |
+ if [ "${{ github.event.pull_request.draft }}" == "true" ]; then
+ echo "PR is in draft mode, skipping workflow"
+ exit 78
+ fi
+ - name: Set up Deno
+ uses: denolib/setup-deno@v2
+ with:
+ deno-version: "1.37.0"
+ - name: Run tests
+ run: deno test --allow-env
diff --git a/README.md b/README.md
index 4056815f..b4727c8b 100644
--- a/README.md
+++ b/README.md
@@ -498,3 +498,9 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&no-frame=true
# Contribution Guide
Check [CONTRIBUTING.md](./CONTRIBUTING.md) for more details.
+
+# Testing
+
+```bash
+deno test --allow-env
+```
diff --git a/index.ts b/api/index.ts
similarity index 73%
rename from index.ts
rename to api/index.ts
index 0bbbcce8..4a095c78 100644
--- a/index.ts
+++ b/api/index.ts
@@ -1,28 +1,31 @@
-import { GithubAPIClient } from "./src/github_api_client.ts";
-import { Card } from "./src/card.ts";
-import { CONSTANTS, parseParams } from "./src/utils.ts";
-import { COLORS, Theme } from "./src/theme.ts";
-import { Error400, Error404 } from "./src/error_page.ts";
+import { Card } from "../src/card.ts";
+import { CONSTANTS, parseParams } from "../src/utils.ts";
+import { COLORS, Theme } from "../src/theme.ts";
+import { Error400, Error404 } from "../src/error_page.ts";
import "https://deno.land/x/dotenv@v0.5.0/load.ts";
-import {staticRenderRegeneration} from "./src/StaticRenderRegeneration/index.ts";
-const apiEndpoint = Deno.env.get("GITHUB_API") || CONSTANTS.DEFAULT_GITHUB_API;
-const client = new GithubAPIClient(apiEndpoint);
+import { staticRenderRegeneration } from "../src/StaticRenderRegeneration/index.ts";
+import { GithubRepositoryService } from "../src/Repository/GithubRepository.ts";
+import { GithubApiService } from "../src/Services/GithubApiService.ts";
+
+const serviceProvider = new GithubApiService();
+const client = new GithubRepositoryService(serviceProvider).repository;
const defaultHeaders = new Headers(
{
"Content-Type": "image/svg+xml",
"Cache-Control": `public, max-age=${CONSTANTS.CACHE_MAX_AGE}`,
},
-)
+);
-export default (request: Request) => staticRenderRegeneration(request, {
- revalidate: CONSTANTS.REVALIDATE_TIME,
- headers: defaultHeaders
-}, function (req: Request) {
- return app(req);
-});
+export default (request: Request) =>
+ staticRenderRegeneration(request, {
+ revalidate: CONSTANTS.REVALIDATE_TIME,
+ headers: defaultHeaders,
+ }, function (req: Request) {
+ return app(req);
+ });
-async function app (req: Request): Promise{
+async function app(req: Request): Promise {
const params = parseParams(req);
const username = params.get("username");
const row = params.getNumberValue("row", CONSTANTS.DEFAULT_MAX_ROW);
@@ -83,7 +86,7 @@ async function app (req: Request): Promise{
},
);
}
-
+
// Success Response
return new Response(
new Card(
diff --git a/debug.ts b/debug.ts
index ab64cc8a..4d1ac0b0 100644
--- a/debug.ts
+++ b/debug.ts
@@ -1,4 +1,4 @@
import { serve } from "https://deno.land/std@0.125.0/http/server.ts";
-import requestHandler from "./index.ts";
+import requestHandler from "./api/index.ts";
serve(requestHandler, { port: 8080 });
diff --git a/deps.ts b/deps.ts
index 5d6175aa..4b51bfce 100644
--- a/deps.ts
+++ b/deps.ts
@@ -1 +1,21 @@
-export { soxa } from "https://deno.land/x/soxa@1.4/mod.ts";
+import { Soxa as ServiceProvider } from "https://deno.land/x/soxa@1.4/src/core/Soxa.ts";
+import { defaults } from "https://deno.land/x/soxa@1.4/src/defaults.ts";
+import {
+ assertEquals,
+ assertRejects,
+} from "https://deno.land/std@0.203.0/assert/mod.ts";
+import {
+ assertSpyCalls,
+ spy,
+} from "https://deno.land/std@0.203.0/testing/mock.ts";
+
+import { CONSTANTS } from "./src/utils.ts";
+
+const baseURL = Deno.env.get("GITHUB_API") || CONSTANTS.DEFAULT_GITHUB_API;
+
+const soxa = new ServiceProvider({
+ ...defaults,
+ baseURL,
+});
+
+export { assertEquals, assertRejects, assertSpyCalls, soxa, spy };
diff --git a/src/Helpers/Retry.ts b/src/Helpers/Retry.ts
new file mode 100644
index 00000000..be1eb9fc
--- /dev/null
+++ b/src/Helpers/Retry.ts
@@ -0,0 +1,44 @@
+export type RetryCallbackProps = {
+ attempt: number;
+};
+
+type callbackType = (data: RetryCallbackProps) => Promise | T;
+
+async function* createAsyncIterable(
+ callback: callbackType,
+ retries: number,
+ delay: number,
+) {
+ for (let i = 0; i < retries; i++) {
+ try {
+ const data = await callback({ attempt: i });
+ yield data;
+ return;
+ } catch (e) {
+ yield null;
+ console.error(e);
+ await new Promise((resolve) => setTimeout(resolve, delay));
+ }
+ }
+}
+
+export class Retry {
+ constructor(private maxRetries = 2, private retryDelay = 1000) {}
+ async fetch(
+ callback: callbackType,
+ ) {
+ for await (
+ const callbackResult of createAsyncIterable(
+ callback,
+ this.maxRetries,
+ this.retryDelay,
+ )
+ ) {
+ if (callbackResult) {
+ return callbackResult as T;
+ }
+ }
+
+ throw new Error(`Max retries (${this.maxRetries}) exceeded.`);
+ }
+}
diff --git a/src/Helpers/__tests__/Retry.test.ts b/src/Helpers/__tests__/Retry.test.ts
new file mode 100644
index 00000000..7fcad13e
--- /dev/null
+++ b/src/Helpers/__tests__/Retry.test.ts
@@ -0,0 +1,66 @@
+import { Retry } from "../Retry.ts";
+import {
+ assertEquals,
+ assertRejects,
+ assertSpyCalls,
+ spy,
+} from "../../../deps.ts";
+
+type MockResponse = {
+ value: number;
+};
+
+Deno.test("Retry.fetch", () => {
+ const retryInstance = new Retry();
+ const callback = spy(retryInstance, "fetch");
+
+ retryInstance.fetch(() => {
+ return { value: 1 };
+ });
+
+ assertSpyCalls(callback, 1);
+});
+
+Deno.test("Should retry", async () => {
+ let countErrors = 0;
+
+ const callbackError = () => {
+ countErrors++;
+ throw new Error("Panic! Threw Error");
+ };
+ const retries = 3;
+ const retryInstance = new Retry(retries);
+
+ await assertRejects(
+ () => {
+ return retryInstance.fetch(callbackError);
+ },
+ Error,
+ `Max retries (${retries}) exceeded.`,
+ );
+
+ assertEquals(countErrors, 3);
+});
+
+Deno.test("Should retry the asyncronous callback", async () => {
+ let countErrors = 0;
+
+ const callbackError = async () => {
+ countErrors++;
+ // Mock request in callback
+ await new Promise((_, reject) => setTimeout(reject, 100));
+ };
+
+ const retries = 3;
+ const retryInstance = new Retry(retries);
+
+ await assertRejects(
+ () => {
+ return retryInstance.fetch(callbackError);
+ },
+ Error,
+ `Max retries (${retries}) exceeded.`,
+ );
+
+ assertEquals(countErrors, 3);
+});
diff --git a/src/Repository/GithubRepository.ts b/src/Repository/GithubRepository.ts
new file mode 100644
index 00000000..b345fd35
--- /dev/null
+++ b/src/Repository/GithubRepository.ts
@@ -0,0 +1,25 @@
+import {
+ GitHubUserActivity,
+ GitHubUserIssue,
+ GitHubUserPullRequest,
+ GitHubUserRepository,
+ UserInfo,
+} from "../user_info.ts";
+
+export abstract class GithubRepository {
+ abstract requestUserInfo(username: string): Promise;
+ abstract requestUserActivity(
+ username: string,
+ ): Promise;
+ abstract requestUserIssue(username: string): Promise;
+ abstract requestUserPullRequest(
+ username: string,
+ ): Promise;
+ abstract requestUserRepository(
+ username: string,
+ ): Promise;
+}
+
+export class GithubRepositoryService {
+ constructor(public repository: GithubRepository) {}
+}
diff --git a/src/Schemas/index.ts b/src/Schemas/index.ts
new file mode 100644
index 00000000..2c7e3f34
--- /dev/null
+++ b/src/Schemas/index.ts
@@ -0,0 +1,61 @@
+export const queryUserActivity = `
+ query userInfo($username: String!) {
+ user(login: $username) {
+ createdAt
+ contributionsCollection {
+ totalCommitContributions
+ restrictedContributionsCount
+ totalPullRequestReviewContributions
+ }
+ organizations(first: 1) {
+ totalCount
+ }
+ followers(first: 1) {
+ totalCount
+ }
+ }
+ }
+`;
+
+export const queryUserIssue = `
+ query userInfo($username: String!) {
+ user(login: $username) {
+ openIssues: issues(states: OPEN) {
+ totalCount
+ }
+ closedIssues: issues(states: CLOSED) {
+ totalCount
+ }
+ }
+ }
+`;
+
+export const queryUserPullRequest = `
+ query userInfo($username: String!) {
+ user(login: $username) {
+ pullRequests(first: 1) {
+ totalCount
+ }
+ }
+ }
+`;
+
+export const queryUserRepository = `
+ query userInfo($username: String!) {
+ user(login: $username) {
+ repositories(first: 100, ownerAffiliations: OWNER, orderBy: {direction: DESC, field: STARGAZERS}) {
+ totalCount
+ nodes {
+ languages(first: 3, orderBy: {direction:DESC, field: SIZE}) {
+ nodes {
+ name
+ }
+ }
+ stargazers {
+ totalCount
+ }
+ }
+ }
+ }
+ }
+`;
diff --git a/src/Services/GithubApiService.ts b/src/Services/GithubApiService.ts
new file mode 100644
index 00000000..4d7a9ccd
--- /dev/null
+++ b/src/Services/GithubApiService.ts
@@ -0,0 +1,120 @@
+import { GithubRepository } from "../Repository/GithubRepository.ts";
+import {
+ GitHubUserActivity,
+ GitHubUserIssue,
+ GitHubUserPullRequest,
+ GitHubUserRepository,
+ UserInfo,
+} from "../user_info.ts";
+import {
+ queryUserActivity,
+ queryUserIssue,
+ queryUserPullRequest,
+ queryUserRepository,
+} from "../Schemas/index.ts";
+import { soxa } from "../../deps.ts";
+import { Retry } from "../Helpers/Retry.ts";
+import { QueryDefaultResponse } from "../Types/index.ts";
+import { CONSTANTS } from "../utils.ts";
+
+// Need to be here - Exporting from another file makes array of null
+export const TOKENS = [
+ Deno.env.get("GITHUB_TOKEN1"),
+ Deno.env.get("GITHUB_TOKEN2"),
+];
+
+export class GithubApiService extends GithubRepository {
+ async requestUserRepository(
+ username: string,
+ ): Promise {
+ return await this.executeQuery(queryUserRepository, {
+ username,
+ });
+ }
+ async requestUserActivity(
+ username: string,
+ ): Promise {
+ return await this.executeQuery(queryUserActivity, {
+ username,
+ });
+ }
+ async requestUserIssue(username: string): Promise {
+ return await this.executeQuery(queryUserIssue, {
+ username,
+ });
+ }
+ async requestUserPullRequest(
+ username: string,
+ ): Promise {
+ return await this.executeQuery(
+ queryUserPullRequest,
+ { username },
+ );
+ }
+ async requestUserInfo(username: string): Promise {
+ // Avoid to call others if one of them is null
+ const repository = await this.requestUserRepository(username);
+ if (repository === null) return null;
+
+ const promises = Promise.allSettled([
+ this.requestUserActivity(username),
+ this.requestUserIssue(username),
+ this.requestUserPullRequest(username),
+ ]);
+ const [activity, issue, pullRequest] = await promises;
+ const status = [
+ activity.status,
+ issue.status,
+ pullRequest.status,
+ ];
+
+ if (status.includes("rejected")) {
+ console.error(`Can not find a user with username:' ${username}'`);
+ return null;
+ }
+
+ return new UserInfo(
+ (activity as PromiseFulfilledResult).value,
+ (issue as PromiseFulfilledResult).value,
+ (pullRequest as PromiseFulfilledResult).value,
+ repository,
+ );
+ }
+
+ async executeQuery(
+ query: string,
+ variables: { [key: string]: string },
+ ) {
+ try {
+ const retry = new Retry(
+ TOKENS.length,
+ CONSTANTS.DEFAULT_GITHUB_RETRY_DELAY,
+ );
+ const response = await retry.fetch>(async ({ attempt }) => {
+ return await soxa.post("", {}, {
+ data: { query: query, variables },
+ headers: {
+ Authorization: `bearer ${TOKENS[attempt]}`,
+ },
+ });
+ }) as QueryDefaultResponse<{ user: T; errors?: unknown[] }>;
+
+ if (response.data.data.errors) {
+ throw new Error("Error from Github API", {
+ cause: response.data.data.errors,
+ });
+ }
+
+ return response?.data?.data?.user ?? null;
+ } catch (error) {
+ // TODO: Move this to a logger instance later
+ if (error instanceof Error && error.cause) {
+ console.error(JSON.stringify(error.cause, null, 2));
+ } else {
+ console.error(error);
+ }
+
+ return null;
+ }
+ }
+}
diff --git a/src/StaticRenderRegeneration/cache_manager.ts b/src/StaticRenderRegeneration/cache_manager.ts
index 3a93e4b4..2f11479c 100644
--- a/src/StaticRenderRegeneration/cache_manager.ts
+++ b/src/StaticRenderRegeneration/cache_manager.ts
@@ -1,49 +1,50 @@
-import { existsSync } from './utils.ts'
+import { existsSync } from "./utils.ts";
export class CacheManager {
- constructor(private revalidateTime: number, private cacheFile: string) {}
-
- // Reason to use /tmp/:
- // https://github.com/orgs/vercel/discussions/314
- get cacheFilePath(): string {
- return `/tmp/${this.cacheFile}`;
- }
- get cacheFileExists(): boolean {
- return existsSync(this.cacheFilePath);
- }
+ constructor(private revalidateTime: number, private cacheFile: string) {}
- get cacheFileLastModified(): Date | null {
- if (!this.cacheFileExists) {
- return null;
- }
- const fileInfo = Deno.statSync(this.cacheFilePath);
- return fileInfo.mtime ?? null;
- }
+ // Reason to use /tmp/:
+ // https://github.com/orgs/vercel/discussions/314
+ get cacheFilePath(): string {
+ return `/tmp/${this.cacheFile}`;
+ }
+ get cacheFileExists(): boolean {
+ return existsSync(this.cacheFilePath);
+ }
- get cacheFileLastModifiedGetTime(): number | null {
- const lastModified = this.cacheFileLastModified;
- if (lastModified === null) {
- return null;
- }
- return lastModified.getTime();
+ get cacheFileLastModified(): Date | null {
+ if (!this.cacheFileExists) {
+ return null;
}
+ const fileInfo = Deno.statSync(this.cacheFilePath);
+ return fileInfo.mtime ?? null;
+ }
- get isCacheValid(): boolean {
- if (this.cacheFileLastModifiedGetTime === null) {
- return false;
- }
- const currentTime = new Date().getTime();
- return currentTime - this.cacheFileLastModifiedGetTime < this.revalidateTime;
+ get cacheFileLastModifiedGetTime(): number | null {
+ const lastModified = this.cacheFileLastModified;
+ if (lastModified === null) {
+ return null;
}
+ return lastModified.getTime();
+ }
- async save (response: Response): Promise {
- if(response === null) return
- // Prevent TypeError: ReadableStream is locked
- const text = await response.clone().text()
- const data = new TextEncoder().encode(text)
-
- Deno.writeFile(this.cacheFilePath, data, { create: true }).catch(() => {
- console.warn("Failed to save cache file")
- });
+ get isCacheValid(): boolean {
+ if (this.cacheFileLastModifiedGetTime === null) {
+ return false;
}
-}
\ No newline at end of file
+ const currentTime = new Date().getTime();
+ return currentTime - this.cacheFileLastModifiedGetTime <
+ this.revalidateTime;
+ }
+
+ async save(response: Response): Promise {
+ if (response === null) return;
+ // Prevent TypeError: ReadableStream is locked
+ const text = await response.clone().text();
+ const data = new TextEncoder().encode(text);
+
+ Deno.writeFile(this.cacheFilePath, data, { create: true }).catch(() => {
+ console.warn("Failed to save cache file");
+ });
+ }
+}
diff --git a/src/Types/Request.ts b/src/Types/Request.ts
new file mode 100644
index 00000000..5ef517e7
--- /dev/null
+++ b/src/Types/Request.ts
@@ -0,0 +1,5 @@
+export type QueryDefaultResponse = {
+ data: {
+ data: T;
+ };
+};
diff --git a/src/Types/index.ts b/src/Types/index.ts
new file mode 100644
index 00000000..07f4ba9c
--- /dev/null
+++ b/src/Types/index.ts
@@ -0,0 +1 @@
+export * from './Request.ts'
\ No newline at end of file
diff --git a/src/github_api_client.ts b/src/github_api_client.ts
deleted file mode 100644
index c4b8361e..00000000
--- a/src/github_api_client.ts
+++ /dev/null
@@ -1,159 +0,0 @@
-import { soxa } from "../deps.ts";
-import { UserInfo } from "./user_info.ts";
-import { CONSTANTS } from "./utils.ts";
-import type {
- GitHubUserActivity,
- GitHubUserIssue,
- GitHubUserPullRequest,
- GitHubUserRepository,
-} from "./user_info.ts";
-
-export class GithubAPIClient {
- constructor(
- private apiEndpoint: string = CONSTANTS.DEFAULT_GITHUB_API,
- ) {
- }
- async requestUserInfo(
- username: string,
- ): Promise {
- // Avoid timeout for the Github API
- const results = await Promise.all([
- this.requestUserActivity(username),
- this.requestUserIssue(username),
- this.requestUserPullRequest(username),
- this.requestUserRepository(username),
- ]);
- if (results.some((r) => r == null)) {
- console.error(`Can not find a user with username:'${username}'`);
- return null;
- }
- return new UserInfo(results[0]!, results[1]!, results[2]!, results[3]!);
- }
- private async requestUserActivity(
- username: string,
- ): Promise {
- const query = `
- query userInfo($username: String!) {
- user(login: $username) {
- createdAt
- contributionsCollection {
- totalCommitContributions
- restrictedContributionsCount
- totalPullRequestReviewContributions
- }
- organizations(first: 1) {
- totalCount
- }
- followers(first: 1) {
- totalCount
- }
- }
- }
- `;
- return await this.request(query, username);
- }
- private async requestUserIssue(
- username: string,
- ): Promise {
- const query = `
- query userInfo($username: String!) {
- user(login: $username) {
- openIssues: issues(states: OPEN) {
- totalCount
- }
- closedIssues: issues(states: CLOSED) {
- totalCount
- }
- }
- }
- `;
- return await this.request(query, username);
- }
- private async requestUserPullRequest(
- username: string,
- ): Promise {
- const query = `
- query userInfo($username: String!) {
- user(login: $username) {
- pullRequests(first: 1) {
- totalCount
- }
- }
- }
- `;
- return await this.request(query, username);
- }
- private async requestUserRepository(
- username: string,
- ): Promise {
- const query = `
- query userInfo($username: String!) {
- user(login: $username) {
- repositories(first: 100, ownerAffiliations: OWNER, orderBy: {direction: DESC, field: STARGAZERS}) {
- totalCount
- nodes {
- languages(first: 3, orderBy: {direction:DESC, field: SIZE}) {
- nodes {
- name
- }
- }
- stargazers {
- totalCount
- }
- }
- }
- }
- }
- `;
- return await this.request(query, username);
- }
- private async request(
- query: string,
- username: string,
- retryDelay = CONSTANTS.DEFAULT_GITHUB_RETRY_DELAY,
- ) {
- const tokens = [
- Deno.env.get("GITHUB_TOKEN1"),
- Deno.env.get("GITHUB_TOKEN2"),
- ];
- const maxRetries = tokens.length;
-
- const variables = { username: username };
- let response;
-
- for (let attempt = 0; attempt < maxRetries; attempt++) {
- try {
- response = await soxa.post(
- this.apiEndpoint,
- {},
- {
- data: { query: query, variables },
- headers: { Authorization: `bearer ${tokens[attempt]}` },
- },
- );
- if (response.data.errors !== undefined) {
- throw new Error(
- response.data.errors.map((e: { message: string; type: string }) =>
- e.message
- ).join("\n"),
- );
- }
- if (response.data.data !== undefined) {
- return response.data.data.user;
- } else {
- return null;
- }
- } catch (error) {
- console.error(
- `Attempt ${attempt} failed with GITHUB_TOKEN${attempt + 1}:`,
- error,
- );
- }
-
- console.log(`Retrying in ${retryDelay / 1000} seconds...`);
- await new Promise((resolve) => setTimeout(resolve, retryDelay));
- }
-
- throw new Error(`Max retries (${maxRetries}) exceeded.`);
- }
-}
diff --git a/vercel.json b/vercel.json
index 9bceb939..5288cb87 100644
--- a/vercel.json
+++ b/vercel.json
@@ -1,7 +1,14 @@
{
- "builds":[{
- "src": "**/index.ts",
- "use": "vercel-deno@1.1.1"
- }],
- "routes": [{ "src": "/.*", "dest": "/" }]
-}
+ "public": true,
+ "functions": {
+ "api/**/*.[jt]s": {
+ "runtime": "vercel-deno@3.0.4"
+ }
+ },
+ "rewrites": [
+ {
+ "source": "/(.*)",
+ "destination": "/api/$1"
+ }
+ ]
+}
\ No newline at end of file
From 495eba6a751c1d68d9bccb5498db578951f2535d Mon Sep 17 00:00:00 2001
From: Alexandro Castro <10711649+AlexcastroDev@users.noreply.github.com>
Date: Sat, 30 Sep 2023 06:49:18 +0100
Subject: [PATCH 07/10] chore: types improvements (#226)
chore: types improvements
chore: types improvements
---
.github/workflows/testing.yml | 25 +--
.gitignore | 1 +
CONTRIBUTING.md | 15 +-
README.md | 159 +++++++++---------
api/index.ts | 13 +-
deno.json | 9 +
src/.DS_Store | Bin 0 -> 6148 bytes
src/Helpers/Logger.ts | 19 +++
src/Helpers/Retry.ts | 4 +-
src/Helpers/__tests__/Retry.test.ts | 1 -
src/Repository/GithubRepository.ts | 13 +-
src/Services/GithubApiService.ts | 64 ++++---
src/StaticRenderRegeneration/cache_manager.ts | 3 +-
src/StaticRenderRegeneration/index.ts | 49 +++---
src/StaticRenderRegeneration/types.ts | 10 +-
src/StaticRenderRegeneration/utils.ts | 60 +++----
src/Types/EServiceKindError.ts | 4 +
src/Types/Request.ts | 6 +
src/Types/ServiceError.ts | 20 +++
src/Types/index.ts | 4 +-
src/error_page.ts | 5 +
src/icons.ts | 2 +-
src/pages/Error.ts | 23 +++
src/theme.ts | 5 +-
src/trophy.ts | 51 +++---
src/trophy_list.ts | 31 ++--
src/user_info.ts | 11 +-
vercel.json | 26 +--
28 files changed, 387 insertions(+), 246 deletions(-)
create mode 100644 deno.json
create mode 100644 src/.DS_Store
create mode 100644 src/Helpers/Logger.ts
create mode 100644 src/Types/EServiceKindError.ts
create mode 100644 src/Types/ServiceError.ts
create mode 100644 src/pages/Error.ts
diff --git a/.github/workflows/testing.yml b/.github/workflows/testing.yml
index b8215530..8d42acdc 100644
--- a/.github/workflows/testing.yml
+++ b/.github/workflows/testing.yml
@@ -7,16 +7,21 @@ on:
jobs:
install-dependencies:
runs-on: ubuntu-latest
+
+ strategy:
+ matrix:
+ deno-version: [1.36.1]
+
steps:
- - name: Check if PR is in draft mode
- run: |
- if [ "${{ github.event.pull_request.draft }}" == "true" ]; then
- echo "PR is in draft mode, skipping workflow"
- exit 78
- fi
- - name: Set up Deno
+ - name: Git Checkout Deno Module
+ uses: actions/checkout@v2
+ - name: Use Deno Version ${{ matrix.deno-version }}
uses: denolib/setup-deno@v2
with:
- deno-version: "1.37.0"
- - name: Run tests
- run: deno test --allow-env
+ deno-version: ${{ matrix.deno-version }}
+ - name: Deno format check
+ run: deno fmt --check
+ - name: Deno lint check
+ run: deno task lint
+ - name: Test Deno Module
+ run: deno task test
diff --git a/.gitignore b/.gitignore
index 3c497e35..2aa5d60a 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,3 +1,4 @@
.vscode
.env
.idea
+.lock
diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md
index 95eaad16..e8cf04a8 100644
--- a/CONTRIBUTING.md
+++ b/CONTRIBUTING.md
@@ -2,14 +2,14 @@
## Environment
-* Deno >= v1.9.2
-* [Vercel](https://vercel.com/)
-* GitHub API v4
+- Deno >= v1.9.2
+- [Vercel](https://vercel.com/)
+- GitHub API v4
## Local Run
-Create `.env` file to project root directory, and write your GitHub token to the `.env` file.
-Please select the authority of `repo` when creating token.
+Create `.env` file to project root directory, and write your GitHub token to the
+`.env` file. Please select the authority of `repo` when creating token.
```properties
GITHUB_TOKEN1=XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
@@ -23,7 +23,7 @@ GITHUB_API=https://github.example.com/api/graphql
Run local server.
```sh
-deno run --allow-net --allow-read --allow-env --allow-write debug.ts
+deno task start
```
Open localhost from your browser.
@@ -36,7 +36,8 @@ Read the [.editorconfig](./.editorconfig)
## Run deno lint
-If you want to contribute to my project, you should check the lint with the following command.
+If you want to contribute to my project, you should check the lint with the
+following command.
```sh
deno lint --unstable
diff --git a/README.md b/README.md
index b4727c8b..96bc8cfd 100644
--- a/README.md
+++ b/README.md
@@ -1,26 +1,26 @@
-
+
GitHub Profile Trophy
🏆 Add dynamically generated GitHub Stat Trophies on your readme
-
+
-
-
+
+
-
+
-
+
-
+
@@ -28,13 +28,14 @@
-
+
# Quick Start
-Add the following code to your readme. When pasting the code into your profile's readme, change the `?username=` value to your GitHub's username.
+Add the following code to your readme. When pasting the code into your profile's
+readme, change the `?username=` value to your GitHub's username.
```
[![trophy](https://github-profile-trophy.vercel.app/?username=ryo-ma)](https://github.com/ryo-ma/github-profile-trophy)
@@ -51,6 +52,7 @@ Add optional parameter of the theme.
```
[![trophy](https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=onedark)](https://github.com/ryo-ma/github-profile-trophy)
```
+
@@ -61,22 +63,25 @@ Add optional parameter of the theme.
Ranks are `SSS` `SS` `S` `AAA` `AA` `A` `B` `C` `UNKNOWN` `SECRET`.
-| Rank | Description |
-| ---- | ---- |
-| SSS, SS, S | You are at a hard to reach rank. You can brag. |
-| AAA, AA, A | You will reach this rank if you do your best. Let's aim here first. |
-| B, C | You are currently making good progress. Let's aim a bit higher. |
-| UNKNOWN | You have not taken action yet. Let's act first. |
-| SECRET | This rank is very rare. The trophy will not be displayed until certain conditions are met. |
+| Rank | Description |
+| ---------- | ------------------------------------------------------------------------------------------ |
+| SSS, SS, S | You are at a hard to reach rank. You can brag. |
+| AAA, AA, A | You will reach this rank if you do your best. Let's aim here first. |
+| B, C | You are currently making good progress. Let's aim a bit higher. |
+| UNKNOWN | You have not taken action yet. Let's act first. |
+| SECRET | This rank is very rare. The trophy will not be displayed until certain conditions are met. |
## Secret Rank
-The acquisition condition is secret, but you can know the condition by reading this code.
+
+The acquisition condition is secret, but you can know the condition by reading
+this code.
-There are only a few secret trophies. Therefore, if you come up with interesting conditions, I will consider adding a trophy. I am waiting for contributions.
+There are only a few secret trophies. Therefore, if you come up with interesting
+conditions, I will consider adding a trophy. I am waiting for contributions.
# About Display details
@@ -90,23 +95,21 @@ There are only a few secret trophies. Therefore, if you come up with interesting
4. Target aggregation result.
5. Next Rank Bar. The road from the current rank to the next rank.
-
# Optional Request Parameters
-* [title](#filter-by-titles)
-* [rank](#filter-by-ranks)
-* [column](#specify-the-maximum-row--column-size)
-* [row](#specify-the-maximum-row--column-size)
-* [theme](#apply-theme)
-* [margin-w](#margin-width)
-* [margin-h](#margin-height)
-* [no-bg](#transparent-background)
-* [no-frame](#hide-frames)
-
+- [title](#filter-by-titles)
+- [rank](#filter-by-ranks)
+- [column](#specify-the-maximum-row--column-size)
+- [row](#specify-the-maximum-row--column-size)
+- [theme](#apply-theme)
+- [margin-w](#margin-width)
+- [margin-h](#margin-height)
+- [no-bg](#transparent-background)
+- [no-frame](#hide-frames)
## Filter by titles
-You can filter the display by specifying the titles of trophy.
+You can filter the display by specifying the titles of trophy.
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&title=Followers
@@ -124,12 +127,13 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&title=Stars,Followers
## Filter by ranks
-You can filter the display by specifying the ranks.
+You can filter the display by specifying the ranks.\
`Available values: SECRET SSS SS S AAA AA A B C`
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&rank=S
```
+
@@ -146,26 +150,28 @@ You can also exclude ranks.
https://github-profile-trophy.vercel.app/?username=ryo-ma&rank=-C,-B
```
-
## Specify the maximum row & column size
-You can specify the maximum row and column size.
+You can specify the maximum row and column size.\
Trophy will be hidden if it exceeds the range of both row and column.
-`Available value: number type`
+`Available value: number type`\
`Default: column=6 row=3`
Restrict only row
+
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&row=2
```
Restrict only column
+
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&column=2
```
Restrict row & column
+
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&row=2&column=3
```
@@ -175,47 +181,49 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&row=2&column=3
Adaptive column
+
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&column=-1
```
-You can set `columns` to `-1` to adapt the width to the number of trophies, the parameter `row` will be ignored.
+You can set `columns` to `-1` to adapt the width to the number of trophies, the
+parameter `row` will be ignored.
## Apply theme
Available themes.
-| theme |
-| ---- |
-| [flat](#flat) |
-| [onedark](#onedark) |
-| [gruvbox](#gruvbox) |
-| [dracula](#dracula) |
-| [monokai](#monokai) |
-| [chalk](#chalk) |
-| [nord](#nord) |
-| [alduin](#alduin) |
-| [darkhub](#darkhub) |
-| [juicyfresh](#juicyfresh) |
-| [buddhism](#buddhism) |
-| [oldie](#oldie) |
-| [radical](#radical) |
-| [onestar](#onestar) |
-| [discord](#discord) |
-| [algolia](#algolia) |
-| [gitdimmed](#gitdimmed) |
-| [tokyonight](#tokyonight) |
-| [matrix](#matrix) |
-| [apprentice](#apprentice) |
+| theme |
+| --------------------------- |
+| [flat](#flat) |
+| [onedark](#onedark) |
+| [gruvbox](#gruvbox) |
+| [dracula](#dracula) |
+| [monokai](#monokai) |
+| [chalk](#chalk) |
+| [nord](#nord) |
+| [alduin](#alduin) |
+| [darkhub](#darkhub) |
+| [juicyfresh](#juicyfresh) |
+| [buddhism](#buddhism) |
+| [oldie](#oldie) |
+| [radical](#radical) |
+| [onestar](#onestar) |
+| [discord](#discord) |
+| [algolia](#algolia) |
+| [gitdimmed](#gitdimmed) |
+| [tokyonight](#tokyonight) |
+| [matrix](#matrix) |
+| [apprentice](#apprentice) |
| [dark_dimmed](#dark_dimmed) |
-| [dark_lover](#dark_lover) |
-
+| [dark_lover](#dark_lover) |
### flat
```
https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=flat
```
+
@@ -280,7 +288,6 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=nord
-
### alduin
```
@@ -388,7 +395,7 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=tokyonight
```
-
+
### matrix
@@ -398,7 +405,7 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=matrix
```
-
+
### apprentice
@@ -408,7 +415,7 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=apprentice
```
-
+
### dark_dimmed
@@ -418,7 +425,7 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=dark_dimmed
```
-
+
### dark_lover
@@ -428,13 +435,13 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&theme=dark_lover
```
-
+
## Margin Width
-You can put a margin in the width between trophies.
-`Available value: number type`
+You can put a margin in the width between trophies.\
+`Available value: number type`\
`Default: margin-w=0`
```
@@ -447,8 +454,8 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&margin-w=15
## Margin Height
-You can put a margin in the height between trophies.
-`Available value: number type`
+You can put a margin in the height between trophies.\
+`Available value: number type`\
`Default: margin-h=0`
```
@@ -467,8 +474,8 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&column=3&margin-w=15&m
## Transparent background
-You can turn the background transparent.
-`Available value: boolean type (true or false)`
+You can turn the background transparent.\
+`Available value: boolean type (true or false)`\
`Default: no-bg=false`
```
@@ -479,12 +486,10 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&no-bg=true
-
-
## Hide frames
-You can hide the frames around the trophies.
-`Available value: boolean type (true or false)`
+You can hide the frames around the trophies.\
+`Available value: boolean type (true or false)`\
`Default: no-frame=false`
```
@@ -495,12 +500,12 @@ https://github-profile-trophy.vercel.app/?username=ryo-ma&no-frame=true
-
# Contribution Guide
+
Check [CONTRIBUTING.md](./CONTRIBUTING.md) for more details.
# Testing
```bash
-deno test --allow-env
+deno task test
```
diff --git a/api/index.ts b/api/index.ts
index 4a095c78..68d5215a 100644
--- a/api/index.ts
+++ b/api/index.ts
@@ -1,11 +1,13 @@
import { Card } from "../src/card.ts";
import { CONSTANTS, parseParams } from "../src/utils.ts";
import { COLORS, Theme } from "../src/theme.ts";
-import { Error400, Error404 } from "../src/error_page.ts";
+import { Error400 } from "../src/error_page.ts";
import "https://deno.land/x/dotenv@v0.5.0/load.ts";
import { staticRenderRegeneration } from "../src/StaticRenderRegeneration/index.ts";
import { GithubRepositoryService } from "../src/Repository/GithubRepository.ts";
import { GithubApiService } from "../src/Services/GithubApiService.ts";
+import { ServiceError } from "../src/Types/index.ts";
+import { ErrorPage } from "../src/pages/Error.ts";
const serviceProvider = new GithubApiService();
const client = new GithubRepositoryService(serviceProvider).repository;
@@ -74,14 +76,11 @@ async function app(req: Request): Promise {
);
}
const userInfo = await client.requestUserInfo(username);
- if (userInfo === null) {
- const error = new Error404(
- "Can not find a user with username: " + username,
- );
+ if (userInfo instanceof ServiceError) {
return new Response(
- error.render(),
+ ErrorPage({ error: userInfo, username }).render(),
{
- status: error.status,
+ status: userInfo.code,
headers: new Headers({ "Content-Type": "text" }),
},
);
diff --git a/deno.json b/deno.json
new file mode 100644
index 00000000..327d1ec1
--- /dev/null
+++ b/deno.json
@@ -0,0 +1,9 @@
+{
+ "tasks": {
+ "start": "deno run -A debug.ts",
+ "debug": "deno --inspect-brk -A debug.ts",
+ "format": "deno fmt",
+ "lint": "deno lint",
+ "test": "ENV_TYPE=test deno test --allow-env"
+ }
+}
diff --git a/src/.DS_Store b/src/.DS_Store
new file mode 100644
index 0000000000000000000000000000000000000000..dbcdcabe06f6c0b9f79fd341be39ab7c161065a8
GIT binary patch
literal 6148
zcmeH~%}T>i5QWcZ7X>$6c6ndHHwdM^fO!GMmLfYKm=w9c<-ex?^3mjfCz}d4*~x^G`ee7
zI;O^_gG-D6)CI#~ypCCd+B`t*O2<@2XqKhaEVUXjEXx^hmDiPysaX!I;lt|5RuhWF
z(|LZ2bXb?FRRl!fn!s&tSKj|0=)cVW*G1ZifC&660=C%p+YMi-dh6unyw^7R6aCwm
o8|hr46;q=XbK|XeeUaDvn)kcXF*V8=k8-Mh1e}XZ1pb1+7jJhOK>z>%
literal 0
HcmV?d00001
diff --git a/src/Helpers/Logger.ts b/src/Helpers/Logger.ts
new file mode 100644
index 00000000..896192a5
--- /dev/null
+++ b/src/Helpers/Logger.ts
@@ -0,0 +1,19 @@
+const enableLogging = Deno.env.get("ENV_TYPE") !== "test";
+
+export class Logger {
+ public static log(message: unknown): void {
+ if (!enableLogging) return;
+ console.log(message);
+ }
+
+ public static error(message: unknown): void {
+ if (!enableLogging) return;
+
+ console.error(message);
+ }
+ public static warn(message: unknown): void {
+ if (!enableLogging) return;
+
+ console.warn(message);
+ }
+}
diff --git a/src/Helpers/Retry.ts b/src/Helpers/Retry.ts
index be1eb9fc..a7ce9a95 100644
--- a/src/Helpers/Retry.ts
+++ b/src/Helpers/Retry.ts
@@ -1,3 +1,5 @@
+import { Logger } from "./Logger.ts";
+
export type RetryCallbackProps = {
attempt: number;
};
@@ -16,7 +18,7 @@ async function* createAsyncIterable(
return;
} catch (e) {
yield null;
- console.error(e);
+ Logger.error(e);
await new Promise((resolve) => setTimeout(resolve, delay));
}
}
diff --git a/src/Helpers/__tests__/Retry.test.ts b/src/Helpers/__tests__/Retry.test.ts
index 7fcad13e..a1fec42b 100644
--- a/src/Helpers/__tests__/Retry.test.ts
+++ b/src/Helpers/__tests__/Retry.test.ts
@@ -44,7 +44,6 @@ Deno.test("Should retry", async () => {
Deno.test("Should retry the asyncronous callback", async () => {
let countErrors = 0;
-
const callbackError = async () => {
countErrors++;
// Mock request in callback
diff --git a/src/Repository/GithubRepository.ts b/src/Repository/GithubRepository.ts
index b345fd35..c2a408c6 100644
--- a/src/Repository/GithubRepository.ts
+++ b/src/Repository/GithubRepository.ts
@@ -1,3 +1,4 @@
+import { ServiceError } from "../Types/index.ts";
import {
GitHubUserActivity,
GitHubUserIssue,
@@ -7,17 +8,19 @@ import {
} from "../user_info.ts";
export abstract class GithubRepository {
- abstract requestUserInfo(username: string): Promise;
+ abstract requestUserInfo(username: string): Promise;
abstract requestUserActivity(
username: string,
- ): Promise;
- abstract requestUserIssue(username: string): Promise;
+ ): Promise;
+ abstract requestUserIssue(
+ username: string,
+ ): Promise;
abstract requestUserPullRequest(
username: string,
- ): Promise;
+ ): Promise;
abstract requestUserRepository(
username: string,
- ): Promise;
+ ): Promise;
}
export class GithubRepositoryService {
diff --git a/src/Services/GithubApiService.ts b/src/Services/GithubApiService.ts
index 4d7a9ccd..685bac58 100644
--- a/src/Services/GithubApiService.ts
+++ b/src/Services/GithubApiService.ts
@@ -14,8 +14,11 @@ import {
} from "../Schemas/index.ts";
import { soxa } from "../../deps.ts";
import { Retry } from "../Helpers/Retry.ts";
-import { QueryDefaultResponse } from "../Types/index.ts";
+import { GithubError, QueryDefaultResponse } from "../Types/index.ts";
import { CONSTANTS } from "../utils.ts";
+import { EServiceKindError } from "../Types/EServiceKindError.ts";
+import { ServiceError } from "../Types/ServiceError.ts";
+import { Logger } from "../Helpers/Logger.ts";
// Need to be here - Exporting from another file makes array of null
export const TOKENS = [
@@ -26,35 +29,40 @@ export const TOKENS = [
export class GithubApiService extends GithubRepository {
async requestUserRepository(
username: string,
- ): Promise {
+ ): Promise {
return await this.executeQuery(queryUserRepository, {
username,
});
}
async requestUserActivity(
username: string,
- ): Promise {
+ ): Promise {
return await this.executeQuery(queryUserActivity, {
username,
});
}
- async requestUserIssue(username: string): Promise {
+ async requestUserIssue(
+ username: string,
+ ): Promise {
return await this.executeQuery(queryUserIssue, {
username,
});
}
async requestUserPullRequest(
username: string,
- ): Promise {
+ ): Promise {
return await this.executeQuery(
queryUserPullRequest,
{ username },
);
}
- async requestUserInfo(username: string): Promise {
+ async requestUserInfo(username: string): Promise {
// Avoid to call others if one of them is null
const repository = await this.requestUserRepository(username);
- if (repository === null) return null;
+ if (repository instanceof ServiceError) {
+ Logger.error(repository);
+ return repository;
+ }
const promises = Promise.allSettled([
this.requestUserActivity(username),
@@ -69,15 +77,34 @@ export class GithubApiService extends GithubRepository {
];
if (status.includes("rejected")) {
- console.error(`Can not find a user with username:' ${username}'`);
- return null;
+ Logger.error(`Can not find a user with username:' ${username}'`);
+ return new ServiceError("not found", EServiceKindError.NOT_FOUND);
}
return new UserInfo(
(activity as PromiseFulfilledResult).value,
(issue as PromiseFulfilledResult).value,
(pullRequest as PromiseFulfilledResult).value,
- repository,
+ repository as GitHubUserRepository,
+ );
+ }
+
+ private handleError(responseErrors: GithubError[]): ServiceError {
+ const errors = responseErrors ?? [];
+ const isRateLimitExceeded = (errors ?? []).some((error) => {
+ error.type.includes(EServiceKindError.RATE_LIMIT);
+ });
+
+ if (isRateLimitExceeded) {
+ throw new ServiceError(
+ "Rate limit exceeded",
+ EServiceKindError.RATE_LIMIT,
+ );
+ }
+
+ throw new ServiceError(
+ "unknown error",
+ EServiceKindError.NOT_FOUND,
);
}
@@ -97,24 +124,23 @@ export class GithubApiService extends GithubRepository {
Authorization: `bearer ${TOKENS[attempt]}`,
},
});
- }) as QueryDefaultResponse<{ user: T; errors?: unknown[] }>;
+ }) as QueryDefaultResponse<{ user: T }>;
- if (response.data.data.errors) {
- throw new Error("Error from Github API", {
- cause: response.data.data.errors,
- });
+ if (response?.data?.errors) {
+ return this.handleError(response?.data?.errors);
}
- return response?.data?.data?.user ?? null;
+ return response?.data?.data?.user ??
+ new ServiceError("not found", EServiceKindError.NOT_FOUND);
} catch (error) {
// TODO: Move this to a logger instance later
if (error instanceof Error && error.cause) {
- console.error(JSON.stringify(error.cause, null, 2));
+ Logger.error(JSON.stringify(error.cause, null, 2));
} else {
- console.error(error);
+ Logger.error(error);
}
- return null;
+ return new ServiceError("not found", EServiceKindError.NOT_FOUND);
}
}
}
diff --git a/src/StaticRenderRegeneration/cache_manager.ts b/src/StaticRenderRegeneration/cache_manager.ts
index 2f11479c..f0a3a87c 100644
--- a/src/StaticRenderRegeneration/cache_manager.ts
+++ b/src/StaticRenderRegeneration/cache_manager.ts
@@ -1,3 +1,4 @@
+import { Logger } from "../Helpers/Logger.ts";
import { existsSync } from "./utils.ts";
export class CacheManager {
@@ -44,7 +45,7 @@ export class CacheManager {
const data = new TextEncoder().encode(text);
Deno.writeFile(this.cacheFilePath, data, { create: true }).catch(() => {
- console.warn("Failed to save cache file");
+ Logger.warn("Failed to save cache file");
});
}
}
diff --git a/src/StaticRenderRegeneration/index.ts b/src/StaticRenderRegeneration/index.ts
index d93fc61f..bc846213 100644
--- a/src/StaticRenderRegeneration/index.ts
+++ b/src/StaticRenderRegeneration/index.ts
@@ -1,29 +1,36 @@
import { CacheManager } from "./cache_manager.ts";
import { StaticRegenerationOptions } from "./types.ts";
-import { getUrl, readCache, generateUUID } from "./utils.ts";
+import { getUrl, hashString, readCache } from "./utils.ts";
-export async function staticRenderRegeneration(request: Request, options: StaticRegenerationOptions, render: (request: Request) => Promise) {
- // avoid TypeError: Invalid URL at deno:core
- const url = getUrl(request)
+export async function staticRenderRegeneration(
+ request: Request,
+ options: StaticRegenerationOptions,
+ render: (request: Request) => Promise,
+) {
+ // avoid TypeError: Invalid URL at deno:core
+ const url = getUrl(request);
- // if more conditions are added, make sure to create a variable to skipCache
- if (url.pathname === "/favicon.ico") {
- return await render(request);
- }
+ // if more conditions are added, make sure to create a variable to skipCache
+ if (url.pathname === "/favicon.ico") {
+ return await render(request);
+ }
- const cacheFile = await generateUUID(url.pathname + (url.search ?? ""));
- const cacheManager = new CacheManager(options.revalidate ?? 0, cacheFile);
- if (cacheManager.isCacheValid) {
- const cache = readCache(cacheManager.cacheFilePath)
- if(cache !== null) {
- return new Response(cache, {
- headers: options.headers ?? new Headers({}),
- });
- }
+ const cacheFile = await hashString(url.pathname + (url.search ?? ""));
+ const cacheManager = new CacheManager(options.revalidate ?? 0, cacheFile);
+ if (cacheManager.isCacheValid) {
+ const cache = readCache(cacheManager.cacheFilePath);
+ if (cache !== null) {
+ return new Response(cache, {
+ headers: options.headers ?? new Headers({}),
+ });
}
-
- const response = await render(request)
- cacheManager.save(response)
+ }
+
+ const response = await render(request);
+
+ if (response.status >= 200 && response.status < 300) {
+ cacheManager.save(response);
+ }
- return response
+ return response;
}
diff --git a/src/StaticRenderRegeneration/types.ts b/src/StaticRenderRegeneration/types.ts
index e9b79076..c98b66c7 100644
--- a/src/StaticRenderRegeneration/types.ts
+++ b/src/StaticRenderRegeneration/types.ts
@@ -1,6 +1,6 @@
export interface StaticRegenerationOptions {
- // The number of milliseconds before the page should be revalidated
- revalidate?: number
- // The headers to be sent with the response
- headers?: Headers
-}
\ No newline at end of file
+ // The number of milliseconds before the page should be revalidated
+ revalidate?: number;
+ // The headers to be sent with the response
+ headers?: Headers;
+}
diff --git a/src/StaticRenderRegeneration/utils.ts b/src/StaticRenderRegeneration/utils.ts
index ebc19c4b..0c04cc5c 100644
--- a/src/StaticRenderRegeneration/utils.ts
+++ b/src/StaticRenderRegeneration/utils.ts
@@ -1,42 +1,42 @@
export function getUrl(request: Request) {
- try {
- return new URL(request.url)
- } catch {
- return {
- pathname: request.url,
- search: request.url
- }
- }
+ try {
+ return new URL(request.url);
+ } catch {
+ return {
+ pathname: request.url,
+ search: request.url,
+ };
+ }
}
export function readCache(cacheFilePath: string): Uint8Array | null {
- try {
- return Deno.readFileSync(cacheFilePath)
- } catch {
- return null
- }
+ try {
+ return Deno.readFileSync(cacheFilePath);
+ } catch {
+ return null;
+ }
}
// https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/digest
-export async function generateUUID(message: string): Promise {
- const encoder = new TextEncoder();
- const data = encoder.encode(message);
- const hashBuffer = await crypto.subtle.digest("SHA-256", data);
-
- const hashArray = Array.from(new Uint8Array(hashBuffer));
- const hashHex = hashArray.map(b => b.toString(16).padStart(2, '0')).join('');
+export async function hashString(message: string): Promise {
+ const encoder = new TextEncoder();
+ const data = encoder.encode(message);
+ const hashBuffer = await crypto.subtle.digest("SHA-256", data);
- return hashHex;
-}
+ const hashArray = Array.from(new Uint8Array(hashBuffer));
+ const hashHex = hashArray.map((b) => b.toString(16).padStart(2, "0")).join(
+ "",
+ );
+ return hashHex;
+}
export const existsSync = (filename: string): boolean => {
- try {
- Deno.statSync(filename);
- // successful, file or directory must exist
- return true;
- } catch {
- return false;
- }
+ try {
+ Deno.statSync(filename);
+ // successful, file or directory must exist
+ return true;
+ } catch {
+ return false;
+ }
};
-
\ No newline at end of file
diff --git a/src/Types/EServiceKindError.ts b/src/Types/EServiceKindError.ts
new file mode 100644
index 00000000..172ed05f
--- /dev/null
+++ b/src/Types/EServiceKindError.ts
@@ -0,0 +1,4 @@
+export const enum EServiceKindError {
+ RATE_LIMIT = "RATE_LIMITED",
+ NOT_FOUND = "NOT_FOUND",
+}
diff --git a/src/Types/Request.ts b/src/Types/Request.ts
index 5ef517e7..85f1362c 100644
--- a/src/Types/Request.ts
+++ b/src/Types/Request.ts
@@ -1,5 +1,11 @@
+export type GithubError = {
+ message: string;
+ type: string;
+};
+
export type QueryDefaultResponse = {
data: {
data: T;
+ errors: GithubError[];
};
};
diff --git a/src/Types/ServiceError.ts b/src/Types/ServiceError.ts
new file mode 100644
index 00000000..8a0db072
--- /dev/null
+++ b/src/Types/ServiceError.ts
@@ -0,0 +1,20 @@
+import { EServiceKindError } from "./EServiceKindError.ts";
+
+export class ServiceError extends Error {
+ constructor(message: string, kind: EServiceKindError) {
+ super(message);
+ this.name = "ServiceError";
+ this.cause = kind;
+ }
+
+ get code(): number {
+ switch (this.cause) {
+ case EServiceKindError.RATE_LIMIT:
+ return 419;
+ case EServiceKindError.NOT_FOUND:
+ return 404;
+ default:
+ return 400;
+ }
+ }
+}
diff --git a/src/Types/index.ts b/src/Types/index.ts
index 07f4ba9c..0e52b91c 100644
--- a/src/Types/index.ts
+++ b/src/Types/index.ts
@@ -1 +1,3 @@
-export * from './Request.ts'
\ No newline at end of file
+export * from "./Request.ts";
+export * from "./ServiceError.ts";
+export * from "./EServiceKindError.ts";
diff --git a/src/error_page.ts b/src/error_page.ts
index 0f3076a4..160bba49 100644
--- a/src/error_page.ts
+++ b/src/error_page.ts
@@ -14,6 +14,11 @@ export class Error400 extends BaseError {
readonly message = "Bad Request";
}
+export class Error419 extends BaseError {
+ readonly status = 419;
+ readonly message = "Rate Limit Exceeded";
+}
+
export class Error404 extends BaseError {
readonly status = 404;
readonly message = "Not Found";
diff --git a/src/icons.ts b/src/icons.ts
index 8e7dbf7f..7cabc097 100644
--- a/src/icons.ts
+++ b/src/icons.ts
@@ -2,7 +2,7 @@ import { RANK } from "./utils.ts";
import { Theme } from "./theme.ts";
const leafIcon = (laurel: string): string => {
- return `
+ return `
Created by potrace 1.15, written by Peter Selinger 2001-2017
diff --git a/src/pages/Error.ts b/src/pages/Error.ts
new file mode 100644
index 00000000..3e93d3dd
--- /dev/null
+++ b/src/pages/Error.ts
@@ -0,0 +1,23 @@
+import { EServiceKindError, ServiceError } from "../Types/index.ts";
+import { Error400, Error404, Error419 } from "../error_page.ts";
+
+interface ErrorPageProps {
+ error: ServiceError;
+ username: string;
+}
+
+export function ErrorPage({ error, username }: ErrorPageProps) {
+ let cause: Error400 | Error404 | Error419 = new Error400();
+
+ if (error.cause === EServiceKindError.RATE_LIMIT) {
+ cause = new Error419();
+ }
+
+ if (error.cause === EServiceKindError.NOT_FOUND) {
+ cause = new Error404(
+ "Can not find a user with username: " + username,
+ );
+ }
+
+ return cause;
+}
diff --git a/src/theme.ts b/src/theme.ts
index 04c4ac07..977938dd 100644
--- a/src/theme.ts
+++ b/src/theme.ts
@@ -1,4 +1,4 @@
-export const COLORS: {[name: string]: Theme} = {
+export const COLORS: { [name: string]: Theme } = {
default: {
BACKGROUND: "#FFF",
TITLE: "#000",
@@ -550,7 +550,7 @@ export const COLORS: {[name: string]: Theme} = {
DEFAULT_RANK_BASE: "#7f6ceb",
DEFAULT_RANK_SHADOW: "#a598ed",
DEFAULT_RANK_TEXT: "#7f6ceb",
- }
+ },
};
export interface Theme {
@@ -577,4 +577,3 @@ export interface Theme {
DEFAULT_RANK_SHADOW: string;
DEFAULT_RANK_TEXT: string;
}
-
diff --git a/src/trophy.ts b/src/trophy.ts
index 66d9f94f..778f30d1 100644
--- a/src/trophy.ts
+++ b/src/trophy.ts
@@ -1,5 +1,5 @@
-import { getTrophyIcon, getNextRankBar } from "./icons.ts";
-import { CONSTANTS, RANK, abridgeScore, RANK_ORDER } from "./utils.ts";
+import { getNextRankBar, getTrophyIcon } from "./icons.ts";
+import { abridgeScore, CONSTANTS, RANK, RANK_ORDER } from "./utils.ts";
import { Theme } from "./theme.ts";
class RankCondition {
@@ -10,7 +10,6 @@ class RankCondition {
) {}
}
-
export class Trophy {
rankCondition: RankCondition | null = null;
rank: RANK = RANK.UNKNOWN;
@@ -59,14 +58,16 @@ export class Trophy {
const result = progress / distance;
return result;
}
- render(theme: Theme,
+ render(
+ theme: Theme,
x = 0,
y = 0,
panelSize = CONSTANTS.DEFAULT_PANEL_SIZE,
noBackground = CONSTANTS.DEFAULT_NO_BACKGROUND,
noFrame = CONSTANTS.DEFAULT_NO_FRAME,
): string {
- const { BACKGROUND: PRIMARY, TITLE: SECONDARY, TEXT, NEXT_RANK_BAR } = theme;
+ const { BACKGROUND: PRIMARY, TITLE: SECONDARY, TEXT, NEXT_RANK_BAR } =
+ theme;
const nextRankBar = getNextRankBar(
this.title,
this.calculateNextRankPercentage(),
@@ -90,8 +91,8 @@ export class Trophy {
height="${panelSize - 1}"
stroke="#e1e4e8"
fill="${PRIMARY}"
- stroke-opacity="${noFrame ? '0' : '1'}"
- fill-opacity="${noBackground ? '0' : '1'}"
+ stroke-opacity="${noFrame ? "0" : "1"}"
+ fill-opacity="${noBackground ? "0" : "1"}"
/>
${getTrophyIcon(theme, this.rank)}
${this.title}
@@ -103,8 +104,8 @@ export class Trophy {
}
}
-export class MultipleLangTrophy extends Trophy{
- constructor(score: number){
+export class MultipleLangTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -119,8 +120,8 @@ export class MultipleLangTrophy extends Trophy{
}
}
-export class AllSuperRankTrophy extends Trophy{
- constructor(score: number){
+export class AllSuperRankTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -131,12 +132,12 @@ export class AllSuperRankTrophy extends Trophy{
super(score, rankConditions);
this.title = "AllSuperRank";
this.filterTitles = ["AllSuperRank"];
- this.bottomMessage = "All S Rank"
+ this.bottomMessage = "All S Rank";
this.hidden = true;
}
}
-export class Joined2020Trophy extends Trophy{
- constructor(score: number){
+export class Joined2020Trophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -147,12 +148,12 @@ export class Joined2020Trophy extends Trophy{
super(score, rankConditions);
this.title = "Joined2020";
this.filterTitles = ["Joined2020"];
- this.bottomMessage = "Joined 2020"
+ this.bottomMessage = "Joined 2020";
this.hidden = true;
}
}
-export class AncientAccountTrophy extends Trophy{
- constructor(score: number){
+export class AncientAccountTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -163,12 +164,12 @@ export class AncientAccountTrophy extends Trophy{
super(score, rankConditions);
this.title = "AncientUser";
this.filterTitles = ["AncientUser"];
- this.bottomMessage = "Before 2010"
+ this.bottomMessage = "Before 2010";
this.hidden = true;
}
}
-export class LongTimeAccountTrophy extends Trophy{
- constructor(score: number){
+export class LongTimeAccountTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -182,8 +183,8 @@ export class LongTimeAccountTrophy extends Trophy{
this.hidden = true;
}
}
-export class MultipleOrganizationsTrophy extends Trophy{
- constructor(score: number){
+export class MultipleOrganizationsTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -199,8 +200,8 @@ export class MultipleOrganizationsTrophy extends Trophy{
}
}
-export class OGAccountTrophy extends Trophy{
- constructor(score: number){
+export class OGAccountTrophy extends Trophy {
+ constructor(score: number) {
const rankConditions = [
new RankCondition(
RANK.SECRET,
@@ -211,7 +212,7 @@ export class OGAccountTrophy extends Trophy{
super(score, rankConditions);
this.title = "OGUser";
this.filterTitles = ["OGUser"];
- this.bottomMessage = "Joined 2008"
+ this.bottomMessage = "Joined 2008";
this.hidden = true;
}
}
diff --git a/src/trophy_list.ts b/src/trophy_list.ts
index 7b5b9c51..d8e406f3 100644
--- a/src/trophy_list.ts
+++ b/src/trophy_list.ts
@@ -1,22 +1,22 @@
import {
- Trophy,
- TotalStarTrophy,
+ AllSuperRankTrophy,
+ AncientAccountTrophy,
+ Joined2020Trophy,
+ LongTimeAccountTrophy,
+ MultipleLangTrophy,
+ MultipleOrganizationsTrophy,
+ OGAccountTrophy,
TotalCommitTrophy,
TotalFollowerTrophy,
TotalIssueTrophy,
TotalPullRequestTrophy,
TotalRepositoryTrophy,
TotalReviewsTrophy,
- MultipleLangTrophy,
- LongTimeAccountTrophy,
- AncientAccountTrophy,
- OGAccountTrophy,
- Joined2020Trophy,
- AllSuperRankTrophy,
- MultipleOrganizationsTrophy,
+ TotalStarTrophy,
+ Trophy,
} from "./trophy.ts";
import { UserInfo } from "./user_info.ts";
-import { RANK_ORDER, RANK } from "./utils.ts";
+import { RANK, RANK_ORDER } from "./utils.ts";
export class TrophyList {
private trophies = new Array();
@@ -49,7 +49,9 @@ export class TrophyList {
return this.trophies;
}
private get isAllSRank() {
- return this.trophies.every((trophy) => trophy.rank.slice(0, 1) == RANK.S) ? 1 : 0;
+ return this.trophies.every((trophy) => trophy.rank.slice(0, 1) == RANK.S)
+ ? 1
+ : 0;
}
filterByHideen() {
this.trophies = this.trophies.filter((trophy) =>
@@ -64,9 +66,9 @@ export class TrophyList {
filterByRanks(ranks: Array) {
if (ranks.filter((rank) => rank.includes("-")).length !== 0) {
this.trophies = this.trophies.filter((trophy) =>
- !ranks.map(rank => rank.substring(1)).includes(trophy.rank)
- )
- return
+ !ranks.map((rank) => rank.substring(1)).includes(trophy.rank)
+ );
+ return;
}
this.trophies = this.trophies.filter((trophy) =>
ranks.includes(trophy.rank)
@@ -78,4 +80,3 @@ export class TrophyList {
);
}
}
-
diff --git a/src/user_info.ts b/src/user_info.ts
index c2e47c3b..7bfdac88 100644
--- a/src/user_info.ts
+++ b/src/user_info.ts
@@ -88,15 +88,18 @@ export class UserInfo {
const joined2020 = new Date(userActivity.createdAt).getFullYear() == 2020
? 1
: 0;
- const ogAccount =
- new Date(userActivity.createdAt).getFullYear() <= 2008 ? 1 : 0;
+ const ogAccount = new Date(userActivity.createdAt).getFullYear() <= 2008
+ ? 1
+ : 0;
this.totalCommits = totalCommits;
this.totalFollowers = userActivity.followers.totalCount;
- this.totalIssues = userIssue.openIssues.totalCount + userIssue.closedIssues.totalCount;
+ this.totalIssues = userIssue.openIssues.totalCount +
+ userIssue.closedIssues.totalCount;
this.totalOrganizations = userActivity.organizations.totalCount;
this.totalPullRequests = userPullRequest.pullRequests.totalCount;
- this.totalReviews = userActivity.contributionsCollection.totalPullRequestReviewContributions;
+ this.totalReviews =
+ userActivity.contributionsCollection.totalPullRequestReviewContributions;
this.totalStargazers = totalStargazers;
this.totalRepositories = userRepository.repositories.totalCount;
this.languageCount = languages.size;
diff --git a/vercel.json b/vercel.json
index 5288cb87..64706355 100644
--- a/vercel.json
+++ b/vercel.json
@@ -1,14 +1,14 @@
{
- "public": true,
- "functions": {
- "api/**/*.[jt]s": {
- "runtime": "vercel-deno@3.0.4"
- }
- },
- "rewrites": [
- {
- "source": "/(.*)",
- "destination": "/api/$1"
- }
- ]
-}
\ No newline at end of file
+ "public": true,
+ "functions": {
+ "api/**/*.[jt]s": {
+ "runtime": "vercel-deno@3.0.4"
+ }
+ },
+ "rewrites": [
+ {
+ "source": "/(.*)",
+ "destination": "/api/$1"
+ }
+ ]
+}
From c8bbf60527d225bfec4ac17ec7d2567e83bdc59f Mon Sep 17 00:00:00 2001
From: Alex Oliveira
Date: Sat, 30 Sep 2023 09:14:43 +0100
Subject: [PATCH 08/10] fix: task run
---
deno.json | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/deno.json b/deno.json
index 327d1ec1..2212a01b 100644
--- a/deno.json
+++ b/deno.json
@@ -1,7 +1,7 @@
{
"tasks": {
"start": "deno run -A debug.ts",
- "debug": "deno --inspect-brk -A debug.ts",
+ "debug": "deno run --inspect-brk -A debug.ts",
"format": "deno fmt",
"lint": "deno lint",
"test": "ENV_TYPE=test deno test --allow-env"
From af753bf270ca148685814803263f0093a7256c23 Mon Sep 17 00:00:00 2001
From: Alexandro Castro <10711649+AlexcastroDev@users.noreply.github.com>
Date: Sat, 30 Sep 2023 10:15:58 +0100
Subject: [PATCH 09/10] fix: check of rate limit (#229)
* fix: task run
* fix: check of rate limit
---
.gitignore | 1 +
src/Services/GithubApiService.ts | 12 +++++++++---
2 files changed, 10 insertions(+), 3 deletions(-)
diff --git a/.gitignore b/.gitignore
index 2aa5d60a..91b922d4 100644
--- a/.gitignore
+++ b/.gitignore
@@ -2,3 +2,4 @@
.env
.idea
.lock
+*.sh
diff --git a/src/Services/GithubApiService.ts b/src/Services/GithubApiService.ts
index 685bac58..96627c1a 100644
--- a/src/Services/GithubApiService.ts
+++ b/src/Services/GithubApiService.ts
@@ -91,9 +91,11 @@ export class GithubApiService extends GithubRepository {
private handleError(responseErrors: GithubError[]): ServiceError {
const errors = responseErrors ?? [];
- const isRateLimitExceeded = (errors ?? []).some((error) => {
- error.type.includes(EServiceKindError.RATE_LIMIT);
- });
+
+ const isRateLimitExceeded = errors.some((error) =>
+ error.type.includes(EServiceKindError.RATE_LIMIT) ||
+ error.message.includes("rate limit")
+ );
if (isRateLimitExceeded) {
throw new ServiceError(
@@ -133,6 +135,10 @@ export class GithubApiService extends GithubRepository {
return response?.data?.data?.user ??
new ServiceError("not found", EServiceKindError.NOT_FOUND);
} catch (error) {
+ if (error instanceof ServiceError) {
+ Logger.error(error);
+ return error;
+ }
// TODO: Move this to a logger instance later
if (error instanceof Error && error.cause) {
Logger.error(JSON.stringify(error.cause, null, 2));
From b4aa73dea1f9b6a452d91498a3de4a5c5d03cff0 Mon Sep 17 00:00:00 2001
From: ryo-ma
Date: Sun, 1 Oct 2023 11:49:55 +0900
Subject: [PATCH 10/10] fix: Temporary fix for retry bug
---
src/Services/GithubApiService.ts | 12 ++++++------
1 file changed, 6 insertions(+), 6 deletions(-)
diff --git a/src/Services/GithubApiService.ts b/src/Services/GithubApiService.ts
index 96627c1a..8cdfd6c6 100644
--- a/src/Services/GithubApiService.ts
+++ b/src/Services/GithubApiService.ts
@@ -120,18 +120,18 @@ export class GithubApiService extends GithubRepository {
CONSTANTS.DEFAULT_GITHUB_RETRY_DELAY,
);
const response = await retry.fetch>(async ({ attempt }) => {
- return await soxa.post("", {}, {
+ const res = await soxa.post("", {}, {
data: { query: query, variables },
headers: {
Authorization: `bearer ${TOKENS[attempt]}`,
},
});
+ if (res?.data?.errors) {
+ return this.handleError(res?.data?.errors);
+ }
+ return res;
}) as QueryDefaultResponse<{ user: T }>;
- if (response?.data?.errors) {
- return this.handleError(response?.data?.errors);
- }
-
return response?.data?.data?.user ??
new ServiceError("not found", EServiceKindError.NOT_FOUND);
} catch (error) {
@@ -146,7 +146,7 @@ export class GithubApiService extends GithubRepository {
Logger.error(error);
}
- return new ServiceError("not found", EServiceKindError.NOT_FOUND);
+ return new ServiceError("Rate limit exceeded", EServiceKindError.RATE_LIMIT);
}
}
}