e559218f98
Reintroduce Gitea/Forgejo as a skill import source on dev only, since the fork deploys against git.farh.net. Pasting a Gitea/Forgejo repo URL into the skills sidebar mirrors the existing GitHub experience: pin to a commit SHA, check for updates, read repo files. Server: new gitea-fetch.ts (URL builders, probe-cache helpers) and gitea-skills.ts (parse, probe, pin, tree, text, branch). Dispatch in readUrlSkillImports probes /api/v1/version and routes non-github.com hosts into the new readGiteaUrlSkillImports branch. updateStatus and readFile get a gitea arm alongside the github/skills_sh arm. Audit falls through to "remote not supported" the same way github does. UI: Server icon, Gitea source label, gitea in the "external" source class, Pin/Update UI gate widened to sourceType === "gitea". CLI help text updated. Existing github code is left byte-for-byte unchanged (wrapped in isGitHubDotCom) so dev <-> master syncs stay clean. PAT support, gitea portability descriptors, and gitea audit are deliberate follow-ups. Detection requires /api/v1/version to return Gitea-shaped JSON; the per-host result is cached for process lifetime with FIFO eviction at 1024 entries. Non-Gitea hosts fall through to the existing raw-markdown url branch. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
243 lines
8.3 KiB
TypeScript
243 lines
8.3 KiB
TypeScript
import { z } from "zod";
|
|
|
|
export const companySkillSourceTypeSchema = z.enum(["local_path", "github", "gitea", "url", "catalog", "skills_sh"]);
|
|
export const companySkillTrustLevelSchema = z.enum(["markdown_only", "assets", "scripts_executables"]);
|
|
export const companySkillCompatibilitySchema = z.enum(["compatible", "unknown", "invalid"]);
|
|
export const companySkillSourceBadgeSchema = z.enum(["paperclip", "github", "gitea", "local", "url", "catalog", "skills_sh"]);
|
|
|
|
export const companySkillFileInventoryEntrySchema = z.object({
|
|
path: z.string().min(1),
|
|
kind: z.enum(["skill", "markdown", "reference", "script", "asset", "other"]),
|
|
});
|
|
|
|
export const companySkillSchema = z.object({
|
|
id: z.string().uuid(),
|
|
companyId: z.string().uuid(),
|
|
key: z.string().min(1),
|
|
slug: z.string().min(1),
|
|
name: z.string().min(1),
|
|
description: z.string().nullable(),
|
|
markdown: z.string(),
|
|
sourceType: companySkillSourceTypeSchema,
|
|
sourceLocator: z.string().nullable(),
|
|
sourceRef: z.string().nullable(),
|
|
trustLevel: companySkillTrustLevelSchema,
|
|
compatibility: companySkillCompatibilitySchema,
|
|
fileInventory: z.array(companySkillFileInventoryEntrySchema).default([]),
|
|
metadata: z.record(z.string(), z.unknown()).nullable(),
|
|
createdAt: z.coerce.date(),
|
|
updatedAt: z.coerce.date(),
|
|
});
|
|
|
|
export const companySkillListItemSchema = companySkillSchema.extend({
|
|
attachedAgentCount: z.number().int().nonnegative(),
|
|
editable: z.boolean(),
|
|
editableReason: z.string().nullable(),
|
|
sourceLabel: z.string().nullable(),
|
|
sourceBadge: companySkillSourceBadgeSchema,
|
|
catalogKind: z.enum(["bundled", "optional"]).nullable(),
|
|
originHash: z.string().nullable(),
|
|
packageName: z.string().nullable(),
|
|
packageVersion: z.string().nullable(),
|
|
});
|
|
|
|
export const companySkillUsageAgentSchema = z.object({
|
|
id: z.string().uuid(),
|
|
name: z.string().min(1),
|
|
urlKey: z.string().min(1),
|
|
adapterType: z.string().min(1),
|
|
desired: z.boolean(),
|
|
actualState: z.string().nullable().describe(
|
|
"Runtime adapter skill state when explicitly fetched; company skill detail reads return null without probing agent runtimes.",
|
|
),
|
|
});
|
|
|
|
export const companySkillDetailSchema = companySkillSchema.extend({
|
|
attachedAgentCount: z.number().int().nonnegative(),
|
|
usedByAgents: z.array(companySkillUsageAgentSchema).default([]),
|
|
editable: z.boolean(),
|
|
editableReason: z.string().nullable(),
|
|
sourceLabel: z.string().nullable(),
|
|
sourceBadge: companySkillSourceBadgeSchema,
|
|
});
|
|
|
|
export const companySkillUpdateStatusSchema = z.object({
|
|
supported: z.boolean(),
|
|
reason: z.string().nullable(),
|
|
trackingRef: z.string().nullable(),
|
|
currentRef: z.string().nullable(),
|
|
latestRef: z.string().nullable(),
|
|
hasUpdate: z.boolean(),
|
|
installedHash: z.string().nullable(),
|
|
originHash: z.string().nullable(),
|
|
userModifiedAt: z.string().nullable(),
|
|
updateHoldReason: z.enum([
|
|
"local_modifications",
|
|
"audit_hard_stop",
|
|
"origin_unavailable",
|
|
"compatibility_invalid",
|
|
"operator_hold",
|
|
]).nullable(),
|
|
auditVerdict: z.enum(["pass", "warning", "fail"]).nullable(),
|
|
auditCodes: z.array(z.string()),
|
|
});
|
|
|
|
export const companySkillAuditFindingSchema = z.object({
|
|
code: z.string().min(1),
|
|
severity: z.enum(["warning", "error"]),
|
|
message: z.string().min(1),
|
|
path: z.string().nullable(),
|
|
});
|
|
|
|
export const companySkillAuditResultSchema = z.object({
|
|
skillId: z.string().uuid(),
|
|
installedHash: z.string().nullable(),
|
|
originHash: z.string().nullable(),
|
|
verdict: z.enum(["pass", "warning", "fail"]),
|
|
codes: z.array(z.string()),
|
|
findings: z.array(companySkillAuditFindingSchema),
|
|
scannedAt: z.string().min(1),
|
|
scanVersion: z.string().min(1),
|
|
});
|
|
|
|
export const companySkillInstallUpdateSchema = z.object({
|
|
force: z.boolean().optional(),
|
|
}).default({});
|
|
|
|
export const companySkillResetSchema = z.object({
|
|
force: z.boolean().optional(),
|
|
}).default({});
|
|
|
|
export const companySkillImportSchema = z.object({
|
|
source: z.string().min(1),
|
|
});
|
|
|
|
export const companySkillProjectScanRequestSchema = z.object({
|
|
projectIds: z.array(z.string().uuid()).optional(),
|
|
workspaceIds: z.array(z.string().uuid()).optional(),
|
|
});
|
|
|
|
export const companySkillProjectScanSkippedSchema = z.object({
|
|
projectId: z.string().uuid(),
|
|
projectName: z.string().min(1),
|
|
workspaceId: z.string().uuid().nullable(),
|
|
workspaceName: z.string().nullable(),
|
|
path: z.string().nullable(),
|
|
reason: z.string().min(1),
|
|
});
|
|
|
|
export const companySkillProjectScanConflictSchema = z.object({
|
|
slug: z.string().min(1),
|
|
key: z.string().min(1),
|
|
projectId: z.string().uuid(),
|
|
projectName: z.string().min(1),
|
|
workspaceId: z.string().uuid(),
|
|
workspaceName: z.string().min(1),
|
|
path: z.string().min(1),
|
|
existingSkillId: z.string().uuid(),
|
|
existingSkillKey: z.string().min(1),
|
|
existingSourceLocator: z.string().nullable(),
|
|
reason: z.string().min(1),
|
|
});
|
|
|
|
export const companySkillProjectScanResultSchema = z.object({
|
|
scannedProjects: z.number().int().nonnegative(),
|
|
scannedWorkspaces: z.number().int().nonnegative(),
|
|
discovered: z.number().int().nonnegative(),
|
|
imported: z.array(companySkillSchema),
|
|
updated: z.array(companySkillSchema),
|
|
skipped: z.array(companySkillProjectScanSkippedSchema),
|
|
conflicts: z.array(companySkillProjectScanConflictSchema),
|
|
warnings: z.array(z.string()),
|
|
});
|
|
|
|
export const companySkillCreateSchema = z.object({
|
|
name: z.string().min(1),
|
|
slug: z.string().min(1).nullable().optional(),
|
|
description: z.string().nullable().optional(),
|
|
markdown: z.string().nullable().optional(),
|
|
});
|
|
|
|
export const companySkillFileDetailSchema = z.object({
|
|
skillId: z.string().uuid(),
|
|
path: z.string().min(1),
|
|
kind: z.enum(["skill", "markdown", "reference", "script", "asset", "other"]),
|
|
content: z.string(),
|
|
language: z.string().nullable(),
|
|
markdown: z.boolean(),
|
|
editable: z.boolean(),
|
|
});
|
|
|
|
export const companySkillFileUpdateSchema = z.object({
|
|
path: z.string().min(1),
|
|
content: z.string(),
|
|
});
|
|
|
|
export const catalogSkillKindSchema = z.enum(["bundled", "optional"]);
|
|
|
|
export const catalogSkillFileSchema = z.object({
|
|
path: z.string().min(1),
|
|
kind: z.enum(["skill", "markdown", "reference", "script", "asset", "other"]),
|
|
sizeBytes: z.number().int().nonnegative(),
|
|
sha256: z.string().min(1),
|
|
});
|
|
|
|
export const catalogSkillSchema = z.object({
|
|
id: z.string().min(1),
|
|
key: z.string().min(1),
|
|
kind: catalogSkillKindSchema,
|
|
category: z.string().min(1),
|
|
slug: z.string().min(1),
|
|
name: z.string().min(1),
|
|
description: z.string(),
|
|
path: z.string().min(1),
|
|
entrypoint: z.literal("SKILL.md"),
|
|
trustLevel: companySkillTrustLevelSchema,
|
|
compatibility: companySkillCompatibilitySchema,
|
|
defaultInstall: z.boolean(),
|
|
recommendedForRoles: z.array(z.string()),
|
|
requires: z.array(z.string()),
|
|
tags: z.array(z.string()),
|
|
files: z.array(catalogSkillFileSchema),
|
|
contentHash: z.string().min(1),
|
|
packageName: z.string().min(1).optional(),
|
|
packageVersion: z.string().min(1).optional(),
|
|
});
|
|
|
|
export const catalogSkillListQuerySchema = z.object({
|
|
kind: catalogSkillKindSchema.optional(),
|
|
category: z.string().min(1).optional(),
|
|
q: z.string().min(1).optional(),
|
|
});
|
|
|
|
export const catalogSkillFileDetailSchema = z.object({
|
|
catalogSkillId: z.string().min(1),
|
|
path: z.string().min(1),
|
|
kind: z.enum(["skill", "markdown", "reference", "script", "asset", "other"]),
|
|
content: z.string(),
|
|
language: z.string().nullable(),
|
|
markdown: z.boolean(),
|
|
});
|
|
|
|
export const companySkillInstallCatalogSchema = z.object({
|
|
catalogSkillId: z.string().min(1),
|
|
slug: z.string().min(1).nullable().optional(),
|
|
force: z.boolean().optional(),
|
|
});
|
|
|
|
export const companySkillInstallCatalogResultSchema = z.object({
|
|
action: z.enum(["created", "updated", "unchanged"]),
|
|
skill: companySkillSchema,
|
|
catalogSkill: catalogSkillSchema,
|
|
warnings: z.array(z.string()),
|
|
});
|
|
|
|
export type CompanySkillImport = z.infer<typeof companySkillImportSchema>;
|
|
export type CompanySkillProjectScan = z.infer<typeof companySkillProjectScanRequestSchema>;
|
|
export type CompanySkillCreate = z.infer<typeof companySkillCreateSchema>;
|
|
export type CompanySkillFileUpdate = z.infer<typeof companySkillFileUpdateSchema>;
|
|
export type CatalogSkillListQuery = z.infer<typeof catalogSkillListQuerySchema>;
|
|
export type CompanySkillInstallCatalog = z.infer<typeof companySkillInstallCatalogSchema>;
|
|
export type CompanySkillInstallUpdate = z.infer<typeof companySkillInstallUpdateSchema>;
|
|
export type CompanySkillReset = z.infer<typeof companySkillResetSchema>;
|