diff --git a/packages/package-registry/src/__tests__/github.test.ts b/packages/package-registry/src/__tests__/github.test.ts index 2e91c1e..69636a4 100644 --- a/packages/package-registry/src/__tests__/github.test.ts +++ b/packages/package-registry/src/__tests__/github.test.ts @@ -1,9 +1,15 @@ import { expect, test } from "bun:test" -import { parseGitHubRepoFromUri } from "../github" +import { getGitHubRepoMetadata, parseGitHubRepoFromUri } from "../github" test("parse github repo from uri", () => { - expect(parseGitHubRepoFromUri("https://github.com/huakunshen/kunkun-ext-ossinsight")).toEqual({ - owner: "huakunshen", + expect(parseGitHubRepoFromUri("https://github.com/kunkunsh/kunkun-ext-ossinsight")).toEqual({ + owner: "kunkunsh", repo: "kunkun-ext-ossinsight" }) + expect(() => parseGitHubRepoFromUri("invalid-uri")).toThrow("Invalid GitHub repository URI") +}) + +test("get github repo metadata", async () => { + const metadata = await getGitHubRepoMetadata("kunkunsh", "kunkun-ext-ossinsight") + expect(metadata).toBeDefined() }) diff --git a/packages/package-registry/src/github.ts b/packages/package-registry/src/github.ts index d07577f..6f94e0a 100644 --- a/packages/package-registry/src/github.ts +++ b/packages/package-registry/src/github.ts @@ -33,6 +33,12 @@ export function authenticatedUserIsMemberOfGitHubOrg( }) } +/** + * Parse a GitHub repository URI into owner and repo + * If not a valid GitHub repository URI, throw an error + * @param uri + * @returns owner and repo + */ export function parseGitHubRepoFromUri(uri: string): { owner: string repo: string @@ -46,3 +52,15 @@ export function parseGitHubRepoFromUri(uri: string): { const [, owner, repo] = match return { owner, repo } } + +/** + * Get GitHub repository metadata + * @param owner + * @param repo + * @param githubToken - Optional GitHub token to prevent rate limiting + * @returns repository metadata + */ +export function getGitHubRepoMetadata(owner: string, repo: string, githubToken?: string) { + const octokit = new Octokit({ auth: githubToken }) + return octokit.rest.repos.get({ owner, repo }).then((res) => res.data) +} diff --git a/packages/package-registry/src/jsr/index.ts b/packages/package-registry/src/jsr/index.ts index 66d12bb..e22bcdc 100644 --- a/packages/package-registry/src/jsr/index.ts +++ b/packages/package-registry/src/jsr/index.ts @@ -7,7 +7,11 @@ import { } from "@huakunshen/jsr-client/hey-api-client" import { ExtPackageJson, License } from "@kksh/api/models" import * as v from "valibot" -import { authenticatedUserIsMemberOfGitHubOrg, userIsPublicMemberOfGitHubOrg } from "../github" +import { + authenticatedUserIsMemberOfGitHubOrg, + getGitHubRepoMetadata, + userIsPublicMemberOfGitHubOrg +} from "../github" import type { ExtensionPublishValidationData } from "../models" import type { NpmPkgMetadata } from "../npm/models" import { getInfoFromRekorLog } from "../sigstore" @@ -205,6 +209,7 @@ export function jsrPackageExists(scope: string, name: string, version?: string): /** * Validate a Jsr package as a Kunkun extension + * !This function will also run in frontend, so if there is any verification logic that must be run in backend, do not add it here * - check if jsr pkg is linked to a github repo * - check if jsr pkg is signed with github action * - check if user's github username is the same as repo's owner name @@ -368,6 +373,7 @@ export async function validateJsrPackageAsKunkunExtension(payload: { } } const rekorInfo = await getInfoFromRekorLog(rekorLogId) + return { data: { pkgJson: parseResult.output, diff --git a/packages/package-registry/src/npm/index.ts b/packages/package-registry/src/npm/index.ts index 2624d49..3e29864 100644 --- a/packages/package-registry/src/npm/index.ts +++ b/packages/package-registry/src/npm/index.ts @@ -2,6 +2,7 @@ import { ExtPackageJson, License } from "@kksh/api/models" import * as v from "valibot" import { authenticatedUserIsMemberOfGitHubOrg, + getGitHubRepoMetadata, parseGitHubRepoFromUri, userIsPublicMemberOfGitHubOrg } from "../github" @@ -237,6 +238,7 @@ export async function validateNpmPackageAsKunkunExtension(payload: { provenance.summary.sourceRepositoryDigest, parseResult.output.readme ?? "README.md" ) + return { data: { pkgJson: parseResult.output, diff --git a/packages/supabase/src/models.ts b/packages/supabase/src/models.ts index 85535a7..8f8a6cc 100644 --- a/packages/supabase/src/models.ts +++ b/packages/supabase/src/models.ts @@ -20,7 +20,10 @@ export const ExtPublishMetadata = v.object({ repo: v.string("GitHub repo of the extension"), owner: v.string("GitHub owner of the extension"), commit: v.string("Commit hash of the extension"), - workflowPath: v.string("Workflow path of the extension") + workflowPath: v.string("Workflow path of the extension"), + repoNodeId: v.optional( + v.string("GitHub repo node ID of the extension (a string, not the number id)") + ) }) ) }) diff --git a/packages/ui/src/components/animation/Globe.svelte b/packages/ui/src/components/animation/Globe.svelte index 97991d6..4fc1cd8 100644 --- a/packages/ui/src/components/animation/Globe.svelte +++ b/packages/ui/src/components/animation/Globe.svelte @@ -12,8 +12,7 @@ // let className = "" // export { className as class } - let { locations = [], class: className }: { class?: string; locations?: [number, number][] } = - $props() + let { locations, class: className }: { class?: string; locations?: [number, number][] } = $props() let pointerInteracting: number | null = null let pointerInteractionMovement = 0 let canvas: HTMLCanvasElement @@ -54,24 +53,25 @@ baseColor: [0.3, 0.3, 0.3], markerColor: [251 / 255, 100 / 255, 21 / 255], glowColor: [1, 1, 1], - markers: locations.map((location) => { - return { - location: location, - size: 0.03 - } - }), - // [ - // { location: [14.5995, 120.9842], size: 0.03 }, - // { location: [19.076, 72.8777], size: 0.03 }, - // { location: [23.8103, 90.4125], size: 0.05 }, - // { location: [30.0444, 31.2357], size: 0.07 }, - // { location: [39.9042, 116.4074], size: 0.08 }, - // { location: [-23.5505, -46.6333], size: 0.05 }, - // { location: [19.4326, -99.1332], size: 0.04 }, - // { location: [40.7128, -74.006], size: 0.1 }, - // { location: [34.6937, 135.5022], size: 0.05 }, - // { location: [41.0082, 28.9784], size: 0.06 } - // ], + markers: locations + ? locations.map((location) => { + return { + location: location, + size: 0.03 + } + }) + : [ + { location: [14.5995, 120.9842], size: 0.03 }, + { location: [19.076, 72.8777], size: 0.03 }, + { location: [23.8103, 90.4125], size: 0.05 }, + { location: [30.0444, 31.2357], size: 0.07 }, + { location: [39.9042, 116.4074], size: 0.08 }, + { location: [-23.5505, -46.6333], size: 0.05 }, + { location: [19.4326, -99.1332], size: 0.04 }, + { location: [40.7128, -74.006], size: 0.1 }, + { location: [34.6937, 135.5022], size: 0.05 }, + { location: [41.0082, 28.9784], size: 0.06 } + ], // onRender: (state) => { // if (!pointerInteracting) { // // Called on every animation frame. diff --git a/packages/ui/src/components/common/TauriLink.svelte b/packages/ui/src/components/common/TauriLink.svelte index ce064d4..8a6bfa8 100644 --- a/packages/ui/src/components/common/TauriLink.svelte +++ b/packages/ui/src/components/common/TauriLink.svelte @@ -16,7 +16,7 @@ style?: HTMLAttributes["style"] class?: HTMLAttributes["class"] children: Snippet - ref: HTMLAnchorElement | HTMLButtonElement | null + ref?: HTMLAnchorElement | HTMLButtonElement | null } = $props() // @ts-expect-error window.__TAURI_INTERNALS__ is not defined in the browser diff --git a/packages/ui/src/components/extension/GitHubProvenanceCard.svelte b/packages/ui/src/components/extension/GitHubProvenanceCard.svelte index 8fd00c7..92e29e2 100644 --- a/packages/ui/src/components/extension/GitHubProvenanceCard.svelte +++ b/packages/ui/src/components/extension/GitHubProvenanceCard.svelte @@ -19,6 +19,7 @@ } = $props() const workflowRunId = githubActionInvocationId.split("/").at(-3) const workflowRunUrl = `https://github.com/${repoOwner}/${repoName}/actions/runs/${workflowRunId}/workflow` + const giteaMirrorUrl = `https://gitea.kunkun.sh/kunkun-extensions-mirror/${repoOwner}-${repoName}` @@ -60,6 +61,12 @@ class="underline">Transparentcy log entry

+

+ Mirror + + Mirror Repo + +

diff --git a/packages/ui/src/components/extension/StoreExtDetail.svelte b/packages/ui/src/components/extension/StoreExtDetail.svelte index 3b31517..2addcc4 100644 --- a/packages/ui/src/components/extension/StoreExtDetail.svelte +++ b/packages/ui/src/components/extension/StoreExtDetail.svelte @@ -4,7 +4,7 @@ import { ExtPackageJson, IconEnum, KunkunExtManifest } from "@kksh/api/models" import { ExtPublishMetadata, ExtPublishSourceTypeEnum } from "@kksh/supabase/models" import { type Tables } from "@kksh/supabase/types" - import { Badge, Button, ScrollArea, Separator } from "@kksh/svelte5" + import { Badge, Button, ScrollArea, Separator, Tooltip } from "@kksh/svelte5" import { Constants, IconMultiplexer } from "@kksh/ui" import { cn } from "@kksh/ui/utils" import { CircleCheckBigIcon, MoveRightIcon, Trash2Icon } from "lucide-svelte" @@ -186,7 +186,7 @@ {/if} -