diff --git a/package-lock.json b/package-lock.json index 659fb60..feb6c0d 100644 --- a/package-lock.json +++ b/package-lock.json @@ -9,6 +9,7 @@ "version": "0.0.1", "dependencies": { "gsap": "^3.12.5", + "ofetch": "^1.4.0", "sveltekit-i18n": "^2.4.2" }, "devDependencies": { @@ -2231,6 +2232,11 @@ "node": ">=0.10.0" } }, + "node_modules/destr": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/destr/-/destr-2.0.3.tgz", + "integrity": "sha512-2N3BOUU4gYMpTP24s5rF5iP7BDr7uNTCs4ozw3kf/eKfvWSIu93GEBi5m427YoyJoeOzQ5smuu4nNAPGb8idSQ==" + }, "node_modules/detect-libc": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz", @@ -3286,6 +3292,11 @@ "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==", "dev": true }, + "node_modules/node-fetch-native": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/node-fetch-native/-/node-fetch-native-1.6.4.tgz", + "integrity": "sha512-IhOigYzAKHd244OC0JIMIUrjzctirCmPkaIfhDeGcEETWof5zKYUW7e7MYvChGWh/4CJeXEgsRyGzuF334rOOQ==" + }, "node_modules/node-releases": { "version": "2.0.18", "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.18.tgz", @@ -3328,6 +3339,16 @@ "node": ">= 6" } }, + "node_modules/ofetch": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/ofetch/-/ofetch-1.4.0.tgz", + "integrity": "sha512-MuHgsEhU6zGeX+EMh+8mSMrYTnsqJQQrpM00Q6QHMKNqQ0bKy0B43tk8tL1wg+CnsSTy1kg4Ir2T5Ig6rD+dfQ==", + "dependencies": { + "destr": "^2.0.3", + "node-fetch-native": "^1.6.4", + "ufo": "^1.5.4" + } + }, "node_modules/optionator": { "version": "0.9.4", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz", @@ -4526,8 +4547,7 @@ "node_modules/ufo": { "version": "1.5.4", "resolved": "https://registry.npmjs.org/ufo/-/ufo-1.5.4.tgz", - "integrity": "sha512-UsUk3byDzKd04EyoZ7U4DOlxQaD14JUKQl6/P7wiX4FNvUfm3XL246n9W5AmqwW5RSFJ27NAuM0iLscAOYUiGQ==", - "dev": true + "integrity": "sha512-UsUk3byDzKd04EyoZ7U4DOlxQaD14JUKQl6/P7wiX4FNvUfm3XL246n9W5AmqwW5RSFJ27NAuM0iLscAOYUiGQ==" }, "node_modules/unplugin": { "version": "1.14.1", diff --git a/package.json b/package.json index 39ed7d3..73235b9 100644 --- a/package.json +++ b/package.json @@ -36,6 +36,7 @@ "type": "module", "dependencies": { "gsap": "^3.12.5", + "ofetch": "^1.4.0", "sveltekit-i18n": "^2.4.2" } } diff --git a/src/lib/getStars.ts b/src/lib/getStars.ts new file mode 100644 index 0000000..6d99c7c --- /dev/null +++ b/src/lib/getStars.ts @@ -0,0 +1,37 @@ +import { ofetch } from "ofetch"; + +export default async function getStarsForUser(username: string, auth?: string) { + const userResponse = await request(`/users/${username}`, auth); + if (!userResponse || !userResponse.public_repos) { + throw new Error(userResponse ? userResponse.message : 'User not found.'); + } + + const pages = Math.ceil(userResponse.public_repos / 100); + let repos = []; + const fetchReposPromises = []; + + for (let i = 1; i <= pages; i++) { + fetchReposPromises.push(request(`/users/${username}/repos?per_page=100&page=${i}`, auth)); + } + + const reposResponses = await Promise.all(fetchReposPromises); + repos = reposResponses.flat(); + + return starSum(repos); +} + +async function request(url, auth) { + const headers = { + 'User-Agent': 'GitHub StarCounter', + }; + + if (auth) { + headers.Authorization = `Basic ${Buffer.from(auth).toString('base64')}`; + } + + return ofetch(`https://api.github.com${url}`, { headers }); +} + +async function starSum(repos) { + return repos.reduce((acc, curr) => acc + curr.stargazers_count, 0); +} diff --git a/src/routes/+page.server.ts b/src/routes/+page.server.ts new file mode 100644 index 0000000..3a4dbbf --- /dev/null +++ b/src/routes/+page.server.ts @@ -0,0 +1,8 @@ +import type {PageServerLoad} from "./$types"; +import getStars from "$lib/getStars"; + +export const load: PageServerLoad = async () => { + return { + starCount: await getStars('anatolykopyl') + }; +} diff --git a/src/routes/+page.svelte b/src/routes/+page.svelte index ab40598..2d9b232 100644 --- a/src/routes/+page.svelte +++ b/src/routes/+page.svelte @@ -4,11 +4,13 @@ import About from "./About.svelte"; import Projects from "./Projects/Projects.svelte"; import Footer from "./Footer.svelte"; + + export let data;
- + diff --git a/src/routes/Hero.svelte b/src/routes/Hero.svelte index 4e43212..c3810af 100644 --- a/src/routes/Hero.svelte +++ b/src/routes/Hero.svelte @@ -6,6 +6,7 @@ import Telegram from "$lib/icons/Telegram.svelte"; import Linkedin from "$lib/icons/Linkedin.svelte"; import MaterialSymbolsStarRounded from '~icons/material-symbols/star-rounded'; +export let starCount: number; let logo: HTMLElement; onMount(async () => { @@ -73,7 +74,7 @@ const goTop = () => { >
- 123 + {starCount}