your personal website on atproto - mirror blento.app
at fix-cached-posts 107 lines 3.7 kB view raw
1import type { CardDefinition } from '../../types'; 2import CreateGitHubProfileCardModal from './CreateGitHubProfileCardModal.svelte'; 3import GitHubProfileCard from './GitHubProfileCard.svelte'; 4import type { GitHubContributionsData } from './types'; 5import { fetchGitHubContributions } from './api.remote'; 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, GitHubContributionsData> = {}; 16 for (const item of items) { 17 const user = item.cardData.user; 18 if (!user) continue; 19 try { 20 const data = await fetchGitHubContributions(user); 21 if (data) githubData[user] = data; 22 } catch (error) { 23 console.error('Failed to fetch GitHub contributions:', error); 24 } 25 } 26 return githubData; 27 }, 28 loadDataServer: async (items) => { 29 const githubData: Record<string, GitHubContributionsData> = {}; 30 for (const item of items) { 31 const user = item.cardData.user; 32 if (!user) continue; 33 try { 34 const data = await fetchGitHubContributions(user); 35 if (data) githubData[user] = data; 36 } catch (error) { 37 console.error('Failed to fetch GitHub contributions:', error); 38 } 39 } 40 return githubData; 41 }, 42 onUrlHandler: (url, item) => { 43 const username = getGitHubUsername(url); 44 45 if (!username) return; 46 47 item.cardData.href = url; 48 item.cardData.user = username; 49 50 item.w = 6; 51 item.mobileW = 8; 52 item.h = 3; 53 item.mobileH = 6; 54 return item; 55 }, 56 urlHandlerPriority: 5, 57 minH: 2, 58 minW: 2, 59 60 canChange: (item) => Boolean(getGitHubUsername(item.cardData.href)), 61 change: (item) => { 62 item.cardData.user = getGitHubUsername(item.cardData.href); 63 64 return item; 65 }, 66 name: 'Github Profile', 67 68 keywords: ['developer', 'code', 'repos', 'contributions'], 69 groups: ['Social'], 70 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>` 71} as CardDefinition & { type: 'githubProfile' }; 72 73function getGitHubUsername(url: string | undefined): string | undefined { 74 if (!url) return; 75 76 try { 77 const parsed = new URL(url); 78 79 // Must be github.com (optionally with www.) 80 if (!/^(www\.)?github\.com$/.test(parsed.hostname)) { 81 return undefined; 82 } 83 84 // Remove empty segments 85 const segments = parsed.pathname.split('/').filter(Boolean); 86 87 // Profile URLs have exactly one path segment: /username 88 if (segments.length !== 1) { 89 return undefined; 90 } 91 92 const username = segments[0]; 93 94 // GitHub username rules (simplified but accurate) 95 // - Alphanumeric or hyphens 96 // - Cannot start or end with a hyphen 97 // - Max length 39 98 if (!/^[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,37}[a-zA-Z0-9])?$/.test(username)) { 99 return undefined; 100 } 101 102 return username; 103 } catch { 104 // Invalid URL 105 return undefined; 106 } 107}