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,252 @@
1
+ import { createFileRoute, Link } from '@tanstack/react-router'
2
+ import { useAction, useQuery } from 'convex/react'
3
+ import { useEffect, useMemo, useRef, useState } from 'react'
4
+ import { api } from '../../convex/_generated/api'
5
+ import { InstallSwitcher } from '../components/InstallSwitcher'
6
+ import { SkillCard } from '../components/SkillCard'
7
+ import { SoulCard } from '../components/SoulCard'
8
+ import { getSkillBadges } from '../lib/badges'
9
+ import type { PublicSkill, PublicSoul } from '../lib/publicUser'
10
+ import { getSiteMode } from '../lib/site'
11
+
12
+ export const Route = createFileRoute('/')({
13
+ component: Home,
14
+ })
15
+
16
+ function Home() {
17
+ const mode = getSiteMode()
18
+ return mode === 'souls' ? <OnlyCrabsHome /> : <SkillsHome />
19
+ }
20
+
21
+ function SkillsHome() {
22
+ const highlighted =
23
+ (useQuery(api.skills.list, {
24
+ batch: 'highlighted',
25
+ limit: 6,
26
+ }) as PublicSkill[]) ?? []
27
+ const latest = (useQuery(api.skills.list, { limit: 12 }) as PublicSkill[]) ?? []
28
+
29
+ return (
30
+ <main>
31
+ <section className="hero">
32
+ <div className="hero-inner">
33
+ <div className="hero-copy fade-up" data-delay="1">
34
+ <span className="hero-badge">Lobster-light. Agent-right.</span>
35
+ <h1 className="hero-title">PilotHub, the skill dock for sharp agents.</h1>
36
+ <p className="hero-subtitle">
37
+ Upload AgentSkills bundles, version them like npm, and make them searchable with
38
+ vectors. No gatekeeping, just signal.
39
+ </p>
40
+ <div style={{ display: 'flex', gap: 12, marginTop: 20 }}>
41
+ <Link to="/upload" search={{ updateSlug: undefined }} className="btn btn-primary">
42
+ Publish a skill
43
+ </Link>
44
+ <Link
45
+ to="/skills"
46
+ search={{
47
+ q: undefined,
48
+ sort: undefined,
49
+ dir: undefined,
50
+ highlighted: undefined,
51
+ view: undefined,
52
+ focus: undefined,
53
+ }}
54
+ className="btn"
55
+ >
56
+ Browse skills
57
+ </Link>
58
+ </div>
59
+ </div>
60
+ <div className="hero-card hero-search-card fade-up" data-delay="2">
61
+ <div className="hero-install" style={{ marginTop: 18 }}>
62
+ <div className="stat">Search skills. Versioned, rollback-ready.</div>
63
+ <InstallSwitcher exampleSlug="sonoscli" />
64
+ </div>
65
+ </div>
66
+ </div>
67
+ </section>
68
+
69
+ <section className="section">
70
+ <h2 className="section-title">Highlighted skills</h2>
71
+ <p className="section-subtitle">Curated signal — highlighted for quick trust.</p>
72
+ <div className="grid">
73
+ {highlighted.length === 0 ? (
74
+ <div className="card">No highlighted skills yet.</div>
75
+ ) : (
76
+ highlighted.map((skill) => (
77
+ <SkillCard
78
+ key={skill._id}
79
+ skill={skill}
80
+ badge={getSkillBadges(skill)}
81
+ summaryFallback="A fresh skill bundle."
82
+ meta={
83
+ <div className="stat">
84
+ ⭐ {skill.stats.stars} · ⤓ {skill.stats.downloads} · ⤒{' '}
85
+ {skill.stats.installsAllTime ?? 0}
86
+ </div>
87
+ }
88
+ />
89
+ ))
90
+ )}
91
+ </div>
92
+ </section>
93
+
94
+ <section className="section">
95
+ <h2 className="section-title">Latest drops</h2>
96
+ <p className="section-subtitle">Newest uploads across the registry.</p>
97
+ <div className="grid">
98
+ {latest.length === 0 ? (
99
+ <div className="card">No skills yet. Be the first.</div>
100
+ ) : (
101
+ latest.map((skill) => (
102
+ <SkillCard
103
+ key={skill._id}
104
+ skill={skill}
105
+ summaryFallback="Agent-ready skill pack."
106
+ meta={
107
+ <div className="stat">
108
+ {skill.stats.versions} versions · ⤓ {skill.stats.downloads} · ⤒{' '}
109
+ {skill.stats.installsAllTime ?? 0}
110
+ </div>
111
+ }
112
+ />
113
+ ))
114
+ )}
115
+ </div>
116
+ <div className="section-cta">
117
+ <Link
118
+ to="/skills"
119
+ search={{
120
+ q: undefined,
121
+ sort: undefined,
122
+ dir: undefined,
123
+ highlighted: undefined,
124
+ view: undefined,
125
+ focus: undefined,
126
+ }}
127
+ className="btn"
128
+ >
129
+ See all skills
130
+ </Link>
131
+ </div>
132
+ </section>
133
+ </main>
134
+ )
135
+ }
136
+
137
+ function OnlyCrabsHome() {
138
+ const navigate = Route.useNavigate()
139
+ const ensureSoulSeeds = useAction(api.seed.ensureSoulSeeds)
140
+ const latest = (useQuery(api.souls.list, { limit: 12 }) as PublicSoul[]) ?? []
141
+ const [query, setQuery] = useState('')
142
+ const seedEnsuredRef = useRef(false)
143
+ const trimmedQuery = useMemo(() => query.trim(), [query])
144
+
145
+ useEffect(() => {
146
+ if (seedEnsuredRef.current) return
147
+ seedEnsuredRef.current = true
148
+ void ensureSoulSeeds({})
149
+ }, [ensureSoulSeeds])
150
+
151
+ return (
152
+ <main>
153
+ <section className="hero">
154
+ <div className="hero-inner">
155
+ <div className="hero-copy fade-up" data-delay="1">
156
+ <span className="hero-badge">SOUL.md, shared.</span>
157
+ <h1 className="hero-title">SoulHub, where system lore lives.</h1>
158
+ <p className="hero-subtitle">
159
+ Share SOUL.md bundles, version them like docs, and keep personal system lore in one
160
+ public place.
161
+ </p>
162
+ <div style={{ display: 'flex', gap: 12, marginTop: 20 }}>
163
+ <Link to="/upload" search={{ updateSlug: undefined }} className="btn btn-primary">
164
+ Publish a soul
165
+ </Link>
166
+ <Link
167
+ to="/souls"
168
+ search={{
169
+ q: undefined,
170
+ sort: undefined,
171
+ dir: undefined,
172
+ view: undefined,
173
+ focus: undefined,
174
+ }}
175
+ className="btn"
176
+ >
177
+ Browse souls
178
+ </Link>
179
+ </div>
180
+ </div>
181
+ <div className="hero-card hero-search-card fade-up" data-delay="2">
182
+ <form
183
+ className="search-bar"
184
+ onSubmit={(event) => {
185
+ event.preventDefault()
186
+ void navigate({
187
+ to: '/souls',
188
+ search: {
189
+ q: trimmedQuery || undefined,
190
+ sort: undefined,
191
+ dir: undefined,
192
+ view: undefined,
193
+ focus: undefined,
194
+ },
195
+ })
196
+ }}
197
+ >
198
+ <span className="mono">/</span>
199
+ <input
200
+ className="search-input"
201
+ placeholder="Search souls, prompts, or lore"
202
+ value={query}
203
+ onChange={(event) => setQuery(event.target.value)}
204
+ />
205
+ </form>
206
+ <div className="hero-install" style={{ marginTop: 18 }}>
207
+ <div className="stat">Search souls. Versioned, readable, easy to remix.</div>
208
+ </div>
209
+ </div>
210
+ </div>
211
+ </section>
212
+
213
+ <section className="section">
214
+ <h2 className="section-title">Latest souls</h2>
215
+ <p className="section-subtitle">Newest SOUL.md bundles across the hub.</p>
216
+ <div className="grid">
217
+ {latest.length === 0 ? (
218
+ <div className="card">No souls yet. Be the first.</div>
219
+ ) : (
220
+ latest.map((soul) => (
221
+ <SoulCard
222
+ key={soul._id}
223
+ soul={soul}
224
+ summaryFallback="A SOUL.md bundle."
225
+ meta={
226
+ <div className="stat">
227
+ ⭐ {soul.stats.stars} · ⤓ {soul.stats.downloads} · {soul.stats.versions} v
228
+ </div>
229
+ }
230
+ />
231
+ ))
232
+ )}
233
+ </div>
234
+ <div className="section-cta">
235
+ <Link
236
+ to="/souls"
237
+ search={{
238
+ q: undefined,
239
+ sort: undefined,
240
+ dir: undefined,
241
+ view: undefined,
242
+ focus: undefined,
243
+ }}
244
+ className="btn"
245
+ >
246
+ See all souls
247
+ </Link>
248
+ </div>
249
+ </section>
250
+ </main>
251
+ )
252
+ }