your personal website on atproto - mirror
blento.app
1import type { CardDefinition } from '../types';
2import CreateGitHubProfileCardModal from './CreateGitHubProfileCardModal.svelte';
3import type GithubContributionsGraph from './GithubContributionsGraph.svelte';
4import GitHubProfileCard from './GitHubProfileCard.svelte';
5import type { GitHubContributionsData } from './types';
6
7export type GithubProfileLoadedData = Record<string, GitHubContributionsData | undefined>;
8
9export const GithubProfileCardDefitition = {
10 type: 'githubProfile',
11 contentComponent: GitHubProfileCard,
12 creationModalComponent: CreateGitHubProfileCardModal,
13
14 loadData: async (items) => {
15 const githubData: Record<string, GithubContributionsGraph> = {};
16 for (const item of items) {
17 try {
18 const response = await fetch(
19 `https://blento.app/api/github?user=${encodeURIComponent(item.cardData.user)}`
20 );
21 if (response.ok) {
22 githubData[item.cardData.user] = await response.json();
23 }
24 } catch (error) {
25 console.error('Failed to fetch GitHub contributions:', error);
26 }
27 }
28 return githubData;
29 },
30 onUrlHandler: (url, item) => {
31 const username = getGitHubUsername(url);
32
33 console.log(username);
34 if (!username) return;
35
36 item.cardData.href = url;
37 item.cardData.user = username;
38
39 item.w = 6;
40 item.mobileW = 8;
41 item.h = 3;
42 item.mobileH = 6;
43 return item;
44 },
45 urlHandlerPriority: 5,
46 minH: 2,
47 minW: 2,
48
49 canChange: (item) => Boolean(getGitHubUsername(item.cardData.href)),
50 change: (item) => {
51 item.cardData.user = getGitHubUsername(item.cardData.href);
52
53 return item;
54 },
55 name: 'Github Profile',
56
57 groups: ['Social'],
58 icon: `<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="currentColor" class="size-4"><path d="M12 .297c-6.63 0-12 5.373-12 12 0 5.303 3.438 9.8 8.205 11.385.6.113.82-.258.82-.577 0-.285-.01-1.04-.015-2.04-3.338.724-4.042-1.61-4.042-1.61C4.422 18.07 3.633 17.7 3.633 17.7c-1.087-.744.084-.729.084-.729 1.205.084 1.838 1.236 1.838 1.236 1.07 1.835 2.809 1.305 3.495.998.108-.776.417-1.305.76-1.605-2.665-.3-5.466-1.332-5.466-5.93 0-1.31.465-2.38 1.235-3.22-.135-.303-.54-1.523.105-3.176 0 0 1.005-.322 3.3 1.23.96-.267 1.98-.399 3-.405 1.02.006 2.04.138 3 .405 2.28-1.552 3.285-1.23 3.285-1.23.645 1.653.24 2.873.12 3.176.765.84 1.23 1.91 1.23 3.22 0 4.61-2.805 5.625-5.475 5.92.42.36.81 1.096.81 2.22 0 1.606-.015 2.896-.015 3.286 0 .315.21.69.825.57C20.565 22.092 24 17.592 24 12.297c0-6.627-5.373-12-12-12" /></svg>`
59
60} as CardDefinition & { type: 'githubProfile' };
61
62function getGitHubUsername(url: string | undefined): string | undefined {
63 if (!url) return;
64
65 try {
66 const parsed = new URL(url);
67
68 // Must be github.com (optionally with www.)
69 if (!/^(www\.)?github\.com$/.test(parsed.hostname)) {
70 return undefined;
71 }
72
73 // Remove empty segments
74 const segments = parsed.pathname.split('/').filter(Boolean);
75
76 // Profile URLs have exactly one path segment: /username
77 if (segments.length !== 1) {
78 return undefined;
79 }
80
81 const username = segments[0];
82
83 // GitHub username rules (simplified but accurate)
84 // - Alphanumeric or hyphens
85 // - Cannot start or end with a hyphen
86 // - Max length 39
87 if (!/^[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,37}[a-zA-Z0-9])?$/.test(username)) {
88 return undefined;
89 }
90
91 return username;
92 } catch {
93 // Invalid URL
94 return undefined;
95 }
96}