|
1 | 1 | import type { DocsResponse } from '#shared/types' |
2 | 2 | import { fetchNpmPackage } from '#server/utils/npm' |
3 | | -import { assertValidPackageName } from '#shared/utils/npm' |
| 3 | +import { assertValidPackageName, parsePackageParam } from '#shared/utils/npm' |
4 | 4 | import { generateDocsWithDeno } from '#server/utils/docs' |
5 | 5 |
|
6 | 6 | export default defineCachedEventHandler( |
7 | 7 | async (event) => { |
8 | | - const segments = getRouterParam(event, 'pkg')?.split('/') ?? [] |
9 | | - if (segments.length === 0) { |
| 8 | + const pkgParam = getRouterParam(event, 'pkg') |
| 9 | + if (!pkgParam) { |
10 | 10 | throw createError({ statusCode: 400, message: 'Package name is required' }) |
11 | 11 | } |
12 | 12 |
|
13 | | - // Parse package name and optional version from URL segments |
14 | | - // Patterns: [pkg] or [pkg, 'v', version] or [@scope, pkg] or [@scope, pkg, 'v', version] |
15 | | - let packageName: string |
16 | | - let version: string | undefined |
17 | | - |
18 | | - const vIndex = segments.indexOf('v') |
19 | | - if (vIndex !== -1 && vIndex < segments.length - 1) { |
20 | | - packageName = segments.slice(0, vIndex).join('/') |
21 | | - version = segments.slice(vIndex + 1).join('/') |
22 | | - } |
23 | | - else { |
24 | | - packageName = segments.join('/') |
25 | | - } |
| 13 | + const { packageName, version: requestedVersion } = parsePackageParam(pkgParam) |
26 | 14 |
|
27 | 15 | if (!packageName) { |
28 | 16 | throw createError({ statusCode: 400, message: 'Package name is required' }) |
29 | 17 | } |
30 | 18 | assertValidPackageName(packageName) |
31 | 19 |
|
32 | 20 | const packument = await fetchNpmPackage(packageName) |
33 | | - |
34 | | - if (!version) { |
35 | | - version = packument['dist-tags']?.latest |
36 | | - } |
| 21 | + const version = requestedVersion ?? packument['dist-tags']?.latest |
37 | 22 |
|
38 | 23 | if (!version) { |
39 | 24 | throw createError({ statusCode: 404, message: 'No latest version found' }) |
40 | 25 | } |
41 | 26 |
|
42 | | - try { |
43 | | - const generated = await generateDocsWithDeno(packageName, version) |
44 | | - |
45 | | - if (!generated) { |
46 | | - return { |
47 | | - package: packageName, |
48 | | - version, |
49 | | - html: '', |
50 | | - toc: null, |
51 | | - breadcrumbs: null, |
52 | | - status: 'missing', |
53 | | - message: 'Docs are not available for this package. It may not have TypeScript types.', |
54 | | - } satisfies DocsResponse |
55 | | - } |
56 | | - |
57 | | - return { |
58 | | - package: packageName, |
59 | | - version, |
60 | | - html: generated.html, |
61 | | - toc: generated.toc, |
62 | | - breadcrumbs: null, |
63 | | - status: 'ok', |
64 | | - } satisfies DocsResponse |
65 | | - } |
66 | | - catch (error) { |
67 | | - const message = error instanceof Error ? error.message : 'Unknown error' |
68 | | - console.error(`[docs] Error generating docs for ${packageName}@${version}:`, message) |
| 27 | + const generated = await generateDocsWithDeno(packageName, version) |
69 | 28 |
|
| 29 | + if (!generated) { |
70 | 30 | return { |
71 | 31 | package: packageName, |
72 | 32 | version, |
73 | 33 | html: '', |
74 | 34 | toc: null, |
75 | | - breadcrumbs: null, |
76 | | - status: 'error', |
77 | | - message: 'Failed to generate docs. The package may not export TypeScript types.', |
| 35 | + status: 'missing', |
| 36 | + message: 'Docs are not available for this package. It may not have TypeScript types.', |
78 | 37 | } satisfies DocsResponse |
79 | 38 | } |
| 39 | + |
| 40 | + return { |
| 41 | + package: packageName, |
| 42 | + version, |
| 43 | + html: generated.html, |
| 44 | + toc: generated.toc, |
| 45 | + status: 'ok', |
| 46 | + } satisfies DocsResponse |
80 | 47 | }, |
81 | 48 | { |
82 | 49 | maxAge: 60 * 60, // 1 hour cache |
|
0 commit comments