pilothub 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (272) hide show
  1. package/.env.local.example +19 -0
  2. package/.github/workflows/ci.yml +40 -0
  3. package/.oxlintrc.json +3 -0
  4. package/AGENTS.md +45 -0
  5. package/CHANGELOG.md +138 -0
  6. package/DEPRECATIONS.md +7 -0
  7. package/LICENSE +21 -0
  8. package/README.md +150 -0
  9. package/biome.json +41 -0
  10. package/convex/_generated/api.d.ts +153 -0
  11. package/convex/_generated/api.js +23 -0
  12. package/convex/_generated/dataModel.d.ts +60 -0
  13. package/convex/_generated/server.d.ts +143 -0
  14. package/convex/_generated/server.js +93 -0
  15. package/convex/auth.config.ts +8 -0
  16. package/convex/auth.ts +19 -0
  17. package/convex/comments.ts +88 -0
  18. package/convex/crons.ts +34 -0
  19. package/convex/devSeed.ts +459 -0
  20. package/convex/devSeedExtra.ts +541 -0
  21. package/convex/downloads.ts +78 -0
  22. package/convex/githubBackups.ts +170 -0
  23. package/convex/githubBackupsNode.ts +183 -0
  24. package/convex/githubImport.ts +317 -0
  25. package/convex/githubSoulBackups.ts +170 -0
  26. package/convex/githubSoulBackupsNode.ts +186 -0
  27. package/convex/http.ts +194 -0
  28. package/convex/httpApi.handlers.test.ts +488 -0
  29. package/convex/httpApi.test.ts +70 -0
  30. package/convex/httpApi.ts +305 -0
  31. package/convex/httpApiV1.handlers.test.ts +584 -0
  32. package/convex/httpApiV1.ts +1172 -0
  33. package/convex/leaderboards.ts +39 -0
  34. package/convex/lib/access.ts +36 -0
  35. package/convex/lib/apiTokenAuth.ts +36 -0
  36. package/convex/lib/badges.ts +50 -0
  37. package/convex/lib/changelog.test.ts +34 -0
  38. package/convex/lib/changelog.ts +278 -0
  39. package/convex/lib/embeddings.ts +38 -0
  40. package/convex/lib/githubBackup.ts +443 -0
  41. package/convex/lib/githubImport.test.ts +247 -0
  42. package/convex/lib/githubImport.ts +425 -0
  43. package/convex/lib/githubSoulBackup.ts +443 -0
  44. package/convex/lib/leaderboards.ts +103 -0
  45. package/convex/lib/moderation.ts +42 -0
  46. package/convex/lib/public.ts +89 -0
  47. package/convex/lib/searchText.test.ts +46 -0
  48. package/convex/lib/searchText.ts +27 -0
  49. package/convex/lib/skillBackfill.test.ts +34 -0
  50. package/convex/lib/skillBackfill.ts +67 -0
  51. package/convex/lib/skillPublish.test.ts +28 -0
  52. package/convex/lib/skillPublish.ts +284 -0
  53. package/convex/lib/skillStats.ts +80 -0
  54. package/convex/lib/skills.test.ts +197 -0
  55. package/convex/lib/skills.ts +273 -0
  56. package/convex/lib/soulChangelog.ts +273 -0
  57. package/convex/lib/soulPublish.ts +236 -0
  58. package/convex/lib/tokens.test.ts +33 -0
  59. package/convex/lib/tokens.ts +51 -0
  60. package/convex/lib/webhooks.test.ts +91 -0
  61. package/convex/lib/webhooks.ts +112 -0
  62. package/convex/maintenance.test.ts +270 -0
  63. package/convex/maintenance.ts +840 -0
  64. package/convex/rateLimits.ts +50 -0
  65. package/convex/schema.ts +472 -0
  66. package/convex/search.test.ts +12 -0
  67. package/convex/search.ts +254 -0
  68. package/convex/seed.test.ts +37 -0
  69. package/convex/seed.ts +254 -0
  70. package/convex/seedSouls.ts +111 -0
  71. package/convex/skillStatEvents.ts +568 -0
  72. package/convex/skills.ts +1606 -0
  73. package/convex/soulComments.ts +88 -0
  74. package/convex/soulDownloads.ts +14 -0
  75. package/convex/soulStars.ts +71 -0
  76. package/convex/souls.ts +570 -0
  77. package/convex/stars.ts +108 -0
  78. package/convex/statsMaintenance.ts +205 -0
  79. package/convex/telemetry.ts +434 -0
  80. package/convex/tokens.ts +88 -0
  81. package/convex/tsconfig.json +7 -0
  82. package/convex/uploads.ts +20 -0
  83. package/convex/users.ts +122 -0
  84. package/convex/webhooks.ts +50 -0
  85. package/convex.json +3 -0
  86. package/docs/README.md +32 -0
  87. package/docs/api.md +51 -0
  88. package/docs/architecture.md +61 -0
  89. package/docs/auth.md +54 -0
  90. package/docs/cli.md +117 -0
  91. package/docs/deploy.md +78 -0
  92. package/docs/diffing.md +84 -0
  93. package/docs/github-import.md +171 -0
  94. package/docs/http-api.md +187 -0
  95. package/docs/manual-testing.md +64 -0
  96. package/docs/mintlify.md +43 -0
  97. package/docs/quickstart.md +120 -0
  98. package/docs/skill-format.md +58 -0
  99. package/docs/soul-format.md +37 -0
  100. package/docs/spec.md +177 -0
  101. package/docs/telemetry.md +91 -0
  102. package/docs/troubleshooting.md +49 -0
  103. package/docs/webhook.md +51 -0
  104. package/e2e/menu-smoke.pw.test.ts +49 -0
  105. package/e2e/pilothub.e2e.test.ts +494 -0
  106. package/e2e/search-exact.pw.test.ts +97 -0
  107. package/package.json +84 -0
  108. package/packages/pilothub/LICENSE +22 -0
  109. package/packages/pilothub/README.md +57 -0
  110. package/packages/pilothub/bin/pilothub.js +2 -0
  111. package/packages/pilothub/package.json +41 -0
  112. package/packages/pilothub/src/browserAuth.test.ts +96 -0
  113. package/packages/pilothub/src/browserAuth.ts +174 -0
  114. package/packages/pilothub/src/cli/buildInfo.ts +94 -0
  115. package/packages/pilothub/src/cli/commands/auth.ts +97 -0
  116. package/packages/pilothub/src/cli/commands/delete.test.ts +73 -0
  117. package/packages/pilothub/src/cli/commands/delete.ts +83 -0
  118. package/packages/pilothub/src/cli/commands/publish.test.ts +122 -0
  119. package/packages/pilothub/src/cli/commands/publish.ts +108 -0
  120. package/packages/pilothub/src/cli/commands/skills.test.ts +191 -0
  121. package/packages/pilothub/src/cli/commands/skills.ts +380 -0
  122. package/packages/pilothub/src/cli/commands/star.ts +46 -0
  123. package/packages/pilothub/src/cli/commands/sync.test.ts +310 -0
  124. package/packages/pilothub/src/cli/commands/sync.ts +200 -0
  125. package/packages/pilothub/src/cli/commands/syncHelpers.test.ts +26 -0
  126. package/packages/pilothub/src/cli/commands/syncHelpers.ts +427 -0
  127. package/packages/pilothub/src/cli/commands/syncTypes.ts +27 -0
  128. package/packages/pilothub/src/cli/commands/unstar.ts +48 -0
  129. package/packages/pilothub/src/cli/helpStyle.ts +45 -0
  130. package/packages/pilothub/src/cli/pilotbotConfig.test.ts +159 -0
  131. package/packages/pilothub/src/cli/pilotbotConfig.ts +147 -0
  132. package/packages/pilothub/src/cli/registry.test.ts +63 -0
  133. package/packages/pilothub/src/cli/registry.ts +43 -0
  134. package/packages/pilothub/src/cli/scanSkills.test.ts +64 -0
  135. package/packages/pilothub/src/cli/scanSkills.ts +84 -0
  136. package/packages/pilothub/src/cli/slug.ts +16 -0
  137. package/packages/pilothub/src/cli/types.ts +12 -0
  138. package/packages/pilothub/src/cli/ui.ts +75 -0
  139. package/packages/pilothub/src/cli.ts +311 -0
  140. package/packages/pilothub/src/config.ts +36 -0
  141. package/packages/pilothub/src/discovery.test.ts +75 -0
  142. package/packages/pilothub/src/discovery.ts +19 -0
  143. package/packages/pilothub/src/http.test.ts +156 -0
  144. package/packages/pilothub/src/http.ts +301 -0
  145. package/packages/pilothub/src/schema/ark.ts +29 -0
  146. package/packages/pilothub/src/schema/index.ts +5 -0
  147. package/packages/pilothub/src/schema/routes.ts +22 -0
  148. package/packages/pilothub/src/schema/schemas.ts +260 -0
  149. package/packages/pilothub/src/schema/textFiles.test.ts +23 -0
  150. package/packages/pilothub/src/schema/textFiles.ts +66 -0
  151. package/packages/pilothub/src/skills.test.ts +191 -0
  152. package/packages/pilothub/src/skills.ts +172 -0
  153. package/packages/pilothub/src/types.ts +10 -0
  154. package/packages/pilothub/tsconfig.json +14 -0
  155. package/packages/schema/README.md +3 -0
  156. package/packages/schema/dist/ark.d.ts +4 -0
  157. package/packages/schema/dist/ark.js +26 -0
  158. package/packages/schema/dist/ark.js.map +1 -0
  159. package/packages/schema/dist/index.d.ts +5 -0
  160. package/packages/schema/dist/index.js +5 -0
  161. package/packages/schema/dist/index.js.map +1 -0
  162. package/packages/schema/dist/routes.d.ts +21 -0
  163. package/packages/schema/dist/routes.js +22 -0
  164. package/packages/schema/dist/routes.js.map +1 -0
  165. package/packages/schema/dist/schemas.d.ts +297 -0
  166. package/packages/schema/dist/schemas.js +243 -0
  167. package/packages/schema/dist/schemas.js.map +1 -0
  168. package/packages/schema/dist/textFiles.d.ts +5 -0
  169. package/packages/schema/dist/textFiles.js +66 -0
  170. package/packages/schema/dist/textFiles.js.map +1 -0
  171. package/packages/schema/package.json +26 -0
  172. package/packages/schema/src/ark.ts +29 -0
  173. package/packages/schema/src/index.ts +5 -0
  174. package/packages/schema/src/routes.ts +22 -0
  175. package/packages/schema/src/schemas.test.ts +123 -0
  176. package/packages/schema/src/schemas.ts +287 -0
  177. package/packages/schema/src/textFiles.test.ts +23 -0
  178. package/packages/schema/src/textFiles.ts +66 -0
  179. package/packages/schema/tsconfig.json +15 -0
  180. package/pilothub +46 -0
  181. package/playwright.config.ts +33 -0
  182. package/public/.well-known/pilothub.json +6 -0
  183. package/public/api/v1/openapi.json +379 -0
  184. package/public/favicon.ico +0 -0
  185. package/public/logo192.png +0 -0
  186. package/public/logo512.png +0 -0
  187. package/public/manifest.json +25 -0
  188. package/public/og.png +0 -0
  189. package/public/og.svg +98 -0
  190. package/public/pilot-logo.png +0 -0
  191. package/public/pilot-mark.png +0 -0
  192. package/public/robots.txt +3 -0
  193. package/public/tanstack-circle-logo.png +0 -0
  194. package/public/tanstack-word-logo-white.svg +1 -0
  195. package/scripts/check-peer-deps.ts +56 -0
  196. package/scripts/docs-list.ts +148 -0
  197. package/scripts/run-playwright-local.sh +14 -0
  198. package/server/og/fetchSkillOgMeta.ts +27 -0
  199. package/server/og/fetchSoulOgMeta.ts +27 -0
  200. package/server/og/ogAssets.ts +80 -0
  201. package/server/og/skillOgSvg.test.ts +59 -0
  202. package/server/og/skillOgSvg.ts +258 -0
  203. package/server/og/soulOgSvg.ts +209 -0
  204. package/server/routes/og/skill.png.ts +103 -0
  205. package/server/routes/og/soul.png.ts +111 -0
  206. package/src/__tests__/skill-detail-page.test.tsx +86 -0
  207. package/src/__tests__/skills-index.test.tsx +145 -0
  208. package/src/__tests__/upload.route.test.tsx +228 -0
  209. package/src/components/AppProviders.tsx +19 -0
  210. package/src/components/ClientOnly.tsx +18 -0
  211. package/src/components/Footer.tsx +29 -0
  212. package/src/components/Header.tsx +295 -0
  213. package/src/components/InstallSwitcher.tsx +53 -0
  214. package/src/components/SkillCard.tsx +36 -0
  215. package/src/components/SkillDetailPage.tsx +817 -0
  216. package/src/components/SkillDiffCard.tsx +485 -0
  217. package/src/components/SoulCard.tsx +19 -0
  218. package/src/components/SoulDetailPage.tsx +263 -0
  219. package/src/components/UserBootstrap.tsx +18 -0
  220. package/src/components/ui/dropdown-menu.tsx +67 -0
  221. package/src/components/ui/toggle-group.tsx +35 -0
  222. package/src/convex/client.ts +3 -0
  223. package/src/lib/badges.ts +29 -0
  224. package/src/lib/diffing.test.ts +163 -0
  225. package/src/lib/diffing.ts +106 -0
  226. package/src/lib/gravatar.test.ts +9 -0
  227. package/src/lib/gravatar.ts +158 -0
  228. package/src/lib/og.test.ts +142 -0
  229. package/src/lib/og.ts +156 -0
  230. package/src/lib/publicUser.ts +39 -0
  231. package/src/lib/roles.ts +19 -0
  232. package/src/lib/site.test.ts +130 -0
  233. package/src/lib/site.ts +84 -0
  234. package/src/lib/theme-transition.test.ts +134 -0
  235. package/src/lib/theme-transition.ts +134 -0
  236. package/src/lib/theme.test.tsx +88 -0
  237. package/src/lib/theme.ts +43 -0
  238. package/src/lib/uploadFiles.jsdom.test.ts +33 -0
  239. package/src/lib/uploadFiles.test.ts +123 -0
  240. package/src/lib/uploadFiles.ts +245 -0
  241. package/src/lib/uploadUtils.test.ts +78 -0
  242. package/src/lib/uploadUtils.ts +93 -0
  243. package/src/lib/useAuthStatus.ts +12 -0
  244. package/src/lib/utils.test.ts +9 -0
  245. package/src/lib/utils.ts +6 -0
  246. package/src/logo.svg +12 -0
  247. package/src/routeTree.gen.ts +345 -0
  248. package/src/router.tsx +17 -0
  249. package/src/routes/$owner/$slug.tsx +55 -0
  250. package/src/routes/__root.tsx +136 -0
  251. package/src/routes/admin.tsx +11 -0
  252. package/src/routes/cli/auth.tsx +168 -0
  253. package/src/routes/dashboard.tsx +97 -0
  254. package/src/routes/import.tsx +415 -0
  255. package/src/routes/index.tsx +252 -0
  256. package/src/routes/management.tsx +529 -0
  257. package/src/routes/settings.tsx +203 -0
  258. package/src/routes/skills/index.tsx +422 -0
  259. package/src/routes/souls/$slug.tsx +55 -0
  260. package/src/routes/souls/index.tsx +243 -0
  261. package/src/routes/stars.tsx +68 -0
  262. package/src/routes/u/$handle.tsx +307 -0
  263. package/src/routes/upload/utils.ts +81 -0
  264. package/src/routes/upload.tsx +499 -0
  265. package/src/styles.css +2718 -0
  266. package/tsconfig.json +24 -0
  267. package/tsconfig.oxlint.json +16 -0
  268. package/vercel.json +8 -0
  269. package/vite.config.ts +48 -0
  270. package/vitest.config.ts +47 -0
  271. package/vitest.e2e.config.ts +11 -0
  272. package/vitest.setup.ts +1 -0
@@ -0,0 +1,50 @@
1
+ import { v } from 'convex/values'
2
+ import { internalMutation } from './_generated/server'
3
+
4
+ export const checkRateLimitInternal = internalMutation({
5
+ args: {
6
+ key: v.string(),
7
+ limit: v.number(),
8
+ windowMs: v.number(),
9
+ },
10
+ handler: async (ctx, args) => {
11
+ const now = Date.now()
12
+ const windowStart = Math.floor(now / args.windowMs) * args.windowMs
13
+ const resetAt = windowStart + args.windowMs
14
+ if (args.limit <= 0) {
15
+ return { allowed: false, remaining: 0, limit: args.limit, resetAt }
16
+ }
17
+
18
+ const existing = await ctx.db
19
+ .query('rateLimits')
20
+ .withIndex('by_key_window', (q) => q.eq('key', args.key).eq('windowStart', windowStart))
21
+ .unique()
22
+
23
+ if (!existing) {
24
+ await ctx.db.insert('rateLimits', {
25
+ key: args.key,
26
+ windowStart,
27
+ count: 1,
28
+ limit: args.limit,
29
+ updatedAt: now,
30
+ })
31
+ return { allowed: true, remaining: Math.max(0, args.limit - 1), limit: args.limit, resetAt }
32
+ }
33
+
34
+ if (existing.count >= args.limit) {
35
+ return { allowed: false, remaining: 0, limit: args.limit, resetAt }
36
+ }
37
+
38
+ await ctx.db.patch(existing._id, {
39
+ count: existing.count + 1,
40
+ limit: args.limit,
41
+ updatedAt: now,
42
+ })
43
+ return {
44
+ allowed: true,
45
+ remaining: Math.max(0, args.limit - existing.count - 1),
46
+ limit: args.limit,
47
+ resetAt,
48
+ }
49
+ },
50
+ })
@@ -0,0 +1,472 @@
1
+ import { authTables } from '@convex-dev/auth/server'
2
+ import { defineSchema, defineTable } from 'convex/server'
3
+ import { v } from 'convex/values'
4
+ import { EMBEDDING_DIMENSIONS } from './lib/embeddings'
5
+
6
+ const authSchema = authTables as unknown as Record<string, ReturnType<typeof defineTable>>
7
+
8
+ const users = defineTable({
9
+ name: v.optional(v.string()),
10
+ image: v.optional(v.string()),
11
+ email: v.optional(v.string()),
12
+ emailVerificationTime: v.optional(v.number()),
13
+ phone: v.optional(v.string()),
14
+ phoneVerificationTime: v.optional(v.number()),
15
+ isAnonymous: v.optional(v.boolean()),
16
+ handle: v.optional(v.string()),
17
+ displayName: v.optional(v.string()),
18
+ bio: v.optional(v.string()),
19
+ role: v.optional(v.union(v.literal('admin'), v.literal('moderator'), v.literal('user'))),
20
+ deletedAt: v.optional(v.number()),
21
+ createdAt: v.optional(v.number()),
22
+ updatedAt: v.optional(v.number()),
23
+ })
24
+ .index('email', ['email'])
25
+ .index('phone', ['phone'])
26
+ .index('handle', ['handle'])
27
+
28
+ const skills = defineTable({
29
+ slug: v.string(),
30
+ displayName: v.string(),
31
+ summary: v.optional(v.string()),
32
+ ownerUserId: v.id('users'),
33
+ canonicalSkillId: v.optional(v.id('skills')),
34
+ forkOf: v.optional(
35
+ v.object({
36
+ skillId: v.id('skills'),
37
+ kind: v.union(v.literal('fork'), v.literal('duplicate')),
38
+ version: v.optional(v.string()),
39
+ at: v.number(),
40
+ }),
41
+ ),
42
+ latestVersionId: v.optional(v.id('skillVersions')),
43
+ tags: v.record(v.string(), v.id('skillVersions')),
44
+ softDeletedAt: v.optional(v.number()),
45
+ badges: v.object({
46
+ redactionApproved: v.optional(
47
+ v.object({
48
+ byUserId: v.id('users'),
49
+ at: v.number(),
50
+ }),
51
+ ),
52
+ highlighted: v.optional(
53
+ v.object({
54
+ byUserId: v.id('users'),
55
+ at: v.number(),
56
+ }),
57
+ ),
58
+ official: v.optional(
59
+ v.object({
60
+ byUserId: v.id('users'),
61
+ at: v.number(),
62
+ }),
63
+ ),
64
+ deprecated: v.optional(
65
+ v.object({
66
+ byUserId: v.id('users'),
67
+ at: v.number(),
68
+ }),
69
+ ),
70
+ }),
71
+ moderationStatus: v.optional(
72
+ v.union(v.literal('active'), v.literal('hidden'), v.literal('removed')),
73
+ ),
74
+ moderationNotes: v.optional(v.string()),
75
+ moderationReason: v.optional(v.string()),
76
+ moderationFlags: v.optional(v.array(v.string())),
77
+ lastReviewedAt: v.optional(v.number()),
78
+ hiddenAt: v.optional(v.number()),
79
+ hiddenBy: v.optional(v.id('users')),
80
+ reportCount: v.optional(v.number()),
81
+ lastReportedAt: v.optional(v.number()),
82
+ batch: v.optional(v.string()),
83
+ statsDownloads: v.optional(v.number()),
84
+ statsStars: v.optional(v.number()),
85
+ statsInstallsCurrent: v.optional(v.number()),
86
+ statsInstallsAllTime: v.optional(v.number()),
87
+ stats: v.object({
88
+ downloads: v.number(),
89
+ installsCurrent: v.optional(v.number()),
90
+ installsAllTime: v.optional(v.number()),
91
+ stars: v.number(),
92
+ versions: v.number(),
93
+ comments: v.number(),
94
+ }),
95
+ createdAt: v.number(),
96
+ updatedAt: v.number(),
97
+ })
98
+ .index('by_slug', ['slug'])
99
+ .index('by_owner', ['ownerUserId'])
100
+ .index('by_updated', ['updatedAt'])
101
+ .index('by_stats_downloads', ['statsDownloads', 'updatedAt'])
102
+ .index('by_stats_stars', ['statsStars', 'updatedAt'])
103
+ .index('by_stats_installs_current', ['statsInstallsCurrent', 'updatedAt'])
104
+ .index('by_stats_installs_all_time', ['statsInstallsAllTime', 'updatedAt'])
105
+ .index('by_batch', ['batch'])
106
+ .index('by_active_updated', ['softDeletedAt', 'updatedAt'])
107
+
108
+ const souls = defineTable({
109
+ slug: v.string(),
110
+ displayName: v.string(),
111
+ summary: v.optional(v.string()),
112
+ ownerUserId: v.id('users'),
113
+ latestVersionId: v.optional(v.id('soulVersions')),
114
+ tags: v.record(v.string(), v.id('soulVersions')),
115
+ softDeletedAt: v.optional(v.number()),
116
+ stats: v.object({
117
+ downloads: v.number(),
118
+ stars: v.number(),
119
+ versions: v.number(),
120
+ comments: v.number(),
121
+ }),
122
+ createdAt: v.number(),
123
+ updatedAt: v.number(),
124
+ })
125
+ .index('by_slug', ['slug'])
126
+ .index('by_owner', ['ownerUserId'])
127
+ .index('by_updated', ['updatedAt'])
128
+
129
+ const skillVersions = defineTable({
130
+ skillId: v.id('skills'),
131
+ version: v.string(),
132
+ fingerprint: v.optional(v.string()),
133
+ changelog: v.string(),
134
+ changelogSource: v.optional(v.union(v.literal('auto'), v.literal('user'))),
135
+ files: v.array(
136
+ v.object({
137
+ path: v.string(),
138
+ size: v.number(),
139
+ storageId: v.id('_storage'),
140
+ sha256: v.string(),
141
+ contentType: v.optional(v.string()),
142
+ }),
143
+ ),
144
+ parsed: v.object({
145
+ frontmatter: v.record(v.string(), v.any()),
146
+ metadata: v.optional(v.any()),
147
+ pilotbot: v.optional(v.any()),
148
+ clawdis: v.optional(v.any()),
149
+ }),
150
+ createdBy: v.id('users'),
151
+ createdAt: v.number(),
152
+ softDeletedAt: v.optional(v.number()),
153
+ })
154
+ .index('by_skill', ['skillId'])
155
+ .index('by_skill_version', ['skillId', 'version'])
156
+
157
+ const soulVersions = defineTable({
158
+ soulId: v.id('souls'),
159
+ version: v.string(),
160
+ fingerprint: v.optional(v.string()),
161
+ changelog: v.string(),
162
+ changelogSource: v.optional(v.union(v.literal('auto'), v.literal('user'))),
163
+ files: v.array(
164
+ v.object({
165
+ path: v.string(),
166
+ size: v.number(),
167
+ storageId: v.id('_storage'),
168
+ sha256: v.string(),
169
+ contentType: v.optional(v.string()),
170
+ }),
171
+ ),
172
+ parsed: v.object({
173
+ frontmatter: v.record(v.string(), v.any()),
174
+ metadata: v.optional(v.any()),
175
+ clawdis: v.optional(v.any()),
176
+ }),
177
+ createdBy: v.id('users'),
178
+ createdAt: v.number(),
179
+ softDeletedAt: v.optional(v.number()),
180
+ })
181
+ .index('by_soul', ['soulId'])
182
+ .index('by_soul_version', ['soulId', 'version'])
183
+
184
+ const skillVersionFingerprints = defineTable({
185
+ skillId: v.id('skills'),
186
+ versionId: v.id('skillVersions'),
187
+ fingerprint: v.string(),
188
+ createdAt: v.number(),
189
+ })
190
+ .index('by_version', ['versionId'])
191
+ .index('by_fingerprint', ['fingerprint'])
192
+ .index('by_skill_fingerprint', ['skillId', 'fingerprint'])
193
+
194
+ const skillBadges = defineTable({
195
+ skillId: v.id('skills'),
196
+ kind: v.union(
197
+ v.literal('highlighted'),
198
+ v.literal('official'),
199
+ v.literal('deprecated'),
200
+ v.literal('redactionApproved'),
201
+ ),
202
+ byUserId: v.id('users'),
203
+ at: v.number(),
204
+ })
205
+ .index('by_skill', ['skillId'])
206
+ .index('by_skill_kind', ['skillId', 'kind'])
207
+ .index('by_kind_at', ['kind', 'at'])
208
+
209
+ const soulVersionFingerprints = defineTable({
210
+ soulId: v.id('souls'),
211
+ versionId: v.id('soulVersions'),
212
+ fingerprint: v.string(),
213
+ createdAt: v.number(),
214
+ })
215
+ .index('by_version', ['versionId'])
216
+ .index('by_fingerprint', ['fingerprint'])
217
+ .index('by_soul_fingerprint', ['soulId', 'fingerprint'])
218
+
219
+ const skillEmbeddings = defineTable({
220
+ skillId: v.id('skills'),
221
+ versionId: v.id('skillVersions'),
222
+ ownerId: v.id('users'),
223
+ embedding: v.array(v.number()),
224
+ isLatest: v.boolean(),
225
+ isApproved: v.boolean(),
226
+ visibility: v.string(),
227
+ updatedAt: v.number(),
228
+ })
229
+ .index('by_skill', ['skillId'])
230
+ .index('by_version', ['versionId'])
231
+ .vectorIndex('by_embedding', {
232
+ vectorField: 'embedding',
233
+ dimensions: EMBEDDING_DIMENSIONS,
234
+ filterFields: ['visibility'],
235
+ })
236
+
237
+ const skillDailyStats = defineTable({
238
+ skillId: v.id('skills'),
239
+ day: v.number(),
240
+ downloads: v.number(),
241
+ installs: v.number(),
242
+ updatedAt: v.number(),
243
+ })
244
+ .index('by_skill_day', ['skillId', 'day'])
245
+ .index('by_day', ['day'])
246
+
247
+ const skillLeaderboards = defineTable({
248
+ kind: v.string(),
249
+ generatedAt: v.number(),
250
+ rangeStartDay: v.number(),
251
+ rangeEndDay: v.number(),
252
+ items: v.array(
253
+ v.object({
254
+ skillId: v.id('skills'),
255
+ score: v.number(),
256
+ installs: v.number(),
257
+ downloads: v.number(),
258
+ }),
259
+ ),
260
+ }).index('by_kind', ['kind', 'generatedAt'])
261
+
262
+ const skillStatBackfillState = defineTable({
263
+ key: v.string(),
264
+ cursor: v.optional(v.string()),
265
+ doneAt: v.optional(v.number()),
266
+ updatedAt: v.number(),
267
+ }).index('by_key', ['key'])
268
+
269
+ const skillStatEvents = defineTable({
270
+ skillId: v.id('skills'),
271
+ kind: v.union(
272
+ v.literal('download'),
273
+ v.literal('star'),
274
+ v.literal('unstar'),
275
+ v.literal('install_new'),
276
+ v.literal('install_reactivate'),
277
+ v.literal('install_deactivate'),
278
+ v.literal('install_clear'),
279
+ ),
280
+ delta: v.optional(
281
+ v.object({
282
+ allTime: v.number(),
283
+ current: v.number(),
284
+ }),
285
+ ),
286
+ occurredAt: v.number(),
287
+ processedAt: v.optional(v.number()),
288
+ })
289
+ .index('by_unprocessed', ['processedAt'])
290
+ .index('by_skill', ['skillId'])
291
+
292
+ const skillStatUpdateCursors = defineTable({
293
+ key: v.string(),
294
+ cursorCreationTime: v.optional(v.number()),
295
+ updatedAt: v.number(),
296
+ }).index('by_key', ['key'])
297
+
298
+ const soulEmbeddings = defineTable({
299
+ soulId: v.id('souls'),
300
+ versionId: v.id('soulVersions'),
301
+ ownerId: v.id('users'),
302
+ embedding: v.array(v.number()),
303
+ isLatest: v.boolean(),
304
+ isApproved: v.boolean(),
305
+ visibility: v.string(),
306
+ updatedAt: v.number(),
307
+ })
308
+ .index('by_soul', ['soulId'])
309
+ .index('by_version', ['versionId'])
310
+ .vectorIndex('by_embedding', {
311
+ vectorField: 'embedding',
312
+ dimensions: EMBEDDING_DIMENSIONS,
313
+ filterFields: ['visibility'],
314
+ })
315
+
316
+ const comments = defineTable({
317
+ skillId: v.id('skills'),
318
+ userId: v.id('users'),
319
+ body: v.string(),
320
+ createdAt: v.number(),
321
+ softDeletedAt: v.optional(v.number()),
322
+ deletedBy: v.optional(v.id('users')),
323
+ })
324
+ .index('by_skill', ['skillId'])
325
+ .index('by_user', ['userId'])
326
+
327
+ const skillReports = defineTable({
328
+ skillId: v.id('skills'),
329
+ userId: v.id('users'),
330
+ reason: v.optional(v.string()),
331
+ createdAt: v.number(),
332
+ })
333
+ .index('by_skill', ['skillId'])
334
+ .index('by_user', ['userId'])
335
+ .index('by_skill_user', ['skillId', 'userId'])
336
+
337
+ const soulComments = defineTable({
338
+ soulId: v.id('souls'),
339
+ userId: v.id('users'),
340
+ body: v.string(),
341
+ createdAt: v.number(),
342
+ softDeletedAt: v.optional(v.number()),
343
+ deletedBy: v.optional(v.id('users')),
344
+ })
345
+ .index('by_soul', ['soulId'])
346
+ .index('by_user', ['userId'])
347
+
348
+ const stars = defineTable({
349
+ skillId: v.id('skills'),
350
+ userId: v.id('users'),
351
+ createdAt: v.number(),
352
+ })
353
+ .index('by_skill', ['skillId'])
354
+ .index('by_user', ['userId'])
355
+ .index('by_skill_user', ['skillId', 'userId'])
356
+
357
+ const soulStars = defineTable({
358
+ soulId: v.id('souls'),
359
+ userId: v.id('users'),
360
+ createdAt: v.number(),
361
+ })
362
+ .index('by_soul', ['soulId'])
363
+ .index('by_user', ['userId'])
364
+ .index('by_soul_user', ['soulId', 'userId'])
365
+
366
+ const auditLogs = defineTable({
367
+ actorUserId: v.id('users'),
368
+ action: v.string(),
369
+ targetType: v.string(),
370
+ targetId: v.string(),
371
+ metadata: v.optional(v.any()),
372
+ createdAt: v.number(),
373
+ })
374
+ .index('by_actor', ['actorUserId'])
375
+ .index('by_target', ['targetType', 'targetId'])
376
+
377
+ const apiTokens = defineTable({
378
+ userId: v.id('users'),
379
+ label: v.string(),
380
+ prefix: v.string(),
381
+ tokenHash: v.string(),
382
+ createdAt: v.number(),
383
+ lastUsedAt: v.optional(v.number()),
384
+ revokedAt: v.optional(v.number()),
385
+ })
386
+ .index('by_user', ['userId'])
387
+ .index('by_hash', ['tokenHash'])
388
+
389
+ const rateLimits = defineTable({
390
+ key: v.string(),
391
+ windowStart: v.number(),
392
+ count: v.number(),
393
+ limit: v.number(),
394
+ updatedAt: v.number(),
395
+ })
396
+ .index('by_key_window', ['key', 'windowStart'])
397
+ .index('by_key', ['key'])
398
+
399
+ const githubBackupSyncState = defineTable({
400
+ key: v.string(),
401
+ cursor: v.optional(v.string()),
402
+ updatedAt: v.number(),
403
+ }).index('by_key', ['key'])
404
+
405
+ const userSyncRoots = defineTable({
406
+ userId: v.id('users'),
407
+ rootId: v.string(),
408
+ label: v.string(),
409
+ firstSeenAt: v.number(),
410
+ lastSeenAt: v.number(),
411
+ expiredAt: v.optional(v.number()),
412
+ })
413
+ .index('by_user', ['userId'])
414
+ .index('by_user_root', ['userId', 'rootId'])
415
+
416
+ const userSkillInstalls = defineTable({
417
+ userId: v.id('users'),
418
+ skillId: v.id('skills'),
419
+ firstSeenAt: v.number(),
420
+ lastSeenAt: v.number(),
421
+ activeRoots: v.number(),
422
+ lastVersion: v.optional(v.string()),
423
+ })
424
+ .index('by_user', ['userId'])
425
+ .index('by_user_skill', ['userId', 'skillId'])
426
+ .index('by_skill', ['skillId'])
427
+
428
+ const userSkillRootInstalls = defineTable({
429
+ userId: v.id('users'),
430
+ rootId: v.string(),
431
+ skillId: v.id('skills'),
432
+ firstSeenAt: v.number(),
433
+ lastSeenAt: v.number(),
434
+ lastVersion: v.optional(v.string()),
435
+ removedAt: v.optional(v.number()),
436
+ })
437
+ .index('by_user', ['userId'])
438
+ .index('by_user_root', ['userId', 'rootId'])
439
+ .index('by_user_root_skill', ['userId', 'rootId', 'skillId'])
440
+ .index('by_user_skill', ['userId', 'skillId'])
441
+ .index('by_skill', ['skillId'])
442
+
443
+ export default defineSchema({
444
+ ...authSchema,
445
+ users,
446
+ skills,
447
+ souls,
448
+ skillVersions,
449
+ soulVersions,
450
+ skillVersionFingerprints,
451
+ skillBadges,
452
+ soulVersionFingerprints,
453
+ skillEmbeddings,
454
+ soulEmbeddings,
455
+ skillDailyStats,
456
+ skillLeaderboards,
457
+ skillStatBackfillState,
458
+ skillStatEvents,
459
+ skillStatUpdateCursors,
460
+ comments,
461
+ skillReports,
462
+ soulComments,
463
+ stars,
464
+ soulStars,
465
+ auditLogs,
466
+ apiTokens,
467
+ rateLimits,
468
+ githubBackupSyncState,
469
+ userSyncRoots,
470
+ userSkillInstalls,
471
+ userSkillRootInstalls,
472
+ })
@@ -0,0 +1,12 @@
1
+ /* @vitest-environment node */
2
+
3
+ import { describe, expect, it } from 'vitest'
4
+ import { __test } from './search'
5
+
6
+ describe('search helpers', () => {
7
+ it('advances candidate limit until max', () => {
8
+ expect(__test.getNextCandidateLimit(50, 1000)).toBe(100)
9
+ expect(__test.getNextCandidateLimit(800, 1000)).toBe(1000)
10
+ expect(__test.getNextCandidateLimit(1000, 1000)).toBeNull()
11
+ })
12
+ })