@emeryld/rrroutes-openapi 2.2.15

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.
package/dist/index.mjs ADDED
@@ -0,0 +1,266 @@
1
+ // src/index.ts
2
+ import { randomBytes } from "crypto";
3
+ import path from "path";
4
+ import { fileURLToPath } from "url";
5
+ import { static as expressStatic } from "express";
6
+
7
+ // src/docs/LeafDocsPage.tsx
8
+ import { renderToStaticMarkup } from "react-dom/server";
9
+
10
+ // src/docs/schemaIntrospection.ts
11
+ import * as z from "zod";
12
+ function getDef(schema) {
13
+ if (!schema || typeof schema !== "object") return void 0;
14
+ const anySchema = schema;
15
+ return anySchema._zod?.def ?? anySchema._def;
16
+ }
17
+ function getDescription(schema) {
18
+ const anyZ = z;
19
+ const registry = anyZ.globalRegistry?.get ? anyZ.globalRegistry.get(schema) : void 0;
20
+ if (registry && typeof registry.description === "string") {
21
+ return registry.description;
22
+ }
23
+ const def = getDef(schema);
24
+ if (def && typeof def.description === "string") {
25
+ return def.description;
26
+ }
27
+ return void 0;
28
+ }
29
+ function unwrap(schema) {
30
+ let s = schema;
31
+ let optional = false;
32
+ let nullable = false;
33
+ const ZodEffectsCtor = z.ZodEffects;
34
+ while (true) {
35
+ if (ZodEffectsCtor && s instanceof ZodEffectsCtor) {
36
+ const def = getDef(s) || {};
37
+ const sourceType = typeof s.sourceType === "function" ? s.sourceType() : def.schema;
38
+ if (!sourceType) break;
39
+ s = sourceType;
40
+ continue;
41
+ }
42
+ if (s instanceof z.ZodOptional) {
43
+ optional = true;
44
+ const def = getDef(s);
45
+ s = def && def.innerType || s;
46
+ continue;
47
+ }
48
+ if (s instanceof z.ZodNullable) {
49
+ nullable = true;
50
+ const def = getDef(s);
51
+ s = def && def.innerType || s;
52
+ continue;
53
+ }
54
+ if (s instanceof z.ZodDefault) {
55
+ const def = getDef(s);
56
+ s = def && def.innerType || s;
57
+ continue;
58
+ }
59
+ break;
60
+ }
61
+ return { base: s, optional, nullable };
62
+ }
63
+ function introspectSchema(schema) {
64
+ if (!schema) return void 0;
65
+ const { base, optional, nullable } = unwrap(schema);
66
+ const def = getDef(base);
67
+ const node = {
68
+ kind: inferKind(base),
69
+ optional: optional || void 0,
70
+ nullable: nullable || void 0,
71
+ description: getDescription(base)
72
+ };
73
+ if (base instanceof z.ZodObject) {
74
+ const rawShape = base.shape ?? (def && typeof def.shape === "function" ? def.shape() : def?.shape);
75
+ const shape = typeof rawShape === "function" ? rawShape() : rawShape ?? {};
76
+ const props = {};
77
+ for (const key of Object.keys(shape)) {
78
+ const child = shape[key];
79
+ const childNode = introspectSchema(child);
80
+ if (childNode) props[key] = childNode;
81
+ }
82
+ node.properties = props;
83
+ }
84
+ if (base instanceof z.ZodArray) {
85
+ const inner = def && def.element || def && def.type || void 0;
86
+ if (inner) {
87
+ node.element = introspectSchema(inner);
88
+ }
89
+ }
90
+ if (base instanceof z.ZodUnion) {
91
+ const options = def && def.options || [];
92
+ node.union = options.map((opt) => introspectSchema(opt)).filter(Boolean);
93
+ }
94
+ if (base instanceof z.ZodLiteral) {
95
+ if (def) {
96
+ if (Array.isArray(def.values)) {
97
+ node.literal = def.values.length === 1 ? def.values[0] : def.values.slice();
98
+ } else {
99
+ node.literal = def.value;
100
+ }
101
+ }
102
+ }
103
+ if (base instanceof z.ZodEnum) {
104
+ if (def) {
105
+ if (Array.isArray(def.values)) {
106
+ node.enumValues = def.values.slice();
107
+ } else if (def.entries && typeof def.entries === "object") {
108
+ node.enumValues = Object.values(def.entries).map(
109
+ (v) => String(v)
110
+ );
111
+ }
112
+ }
113
+ }
114
+ return node;
115
+ }
116
+ function inferKind(schema) {
117
+ if (schema instanceof z.ZodString) return "string";
118
+ if (schema instanceof z.ZodNumber) return "number";
119
+ if (schema instanceof z.ZodBoolean) return "boolean";
120
+ if (schema instanceof z.ZodBigInt) return "bigint";
121
+ if (schema instanceof z.ZodDate) return "date";
122
+ if (schema instanceof z.ZodArray) return "array";
123
+ if (schema instanceof z.ZodObject) return "object";
124
+ if (schema instanceof z.ZodUnion) return "union";
125
+ if (schema instanceof z.ZodLiteral) return "literal";
126
+ if (schema instanceof z.ZodEnum) return "enum";
127
+ if (schema instanceof z.ZodRecord) return "record";
128
+ if (schema instanceof z.ZodTuple) return "tuple";
129
+ if (schema instanceof z.ZodUnknown) return "unknown";
130
+ if (schema instanceof z.ZodAny) return "any";
131
+ return "unknown";
132
+ }
133
+
134
+ // src/docs/serializer.ts
135
+ function serializeLeaf(leaf) {
136
+ const cfg = leaf.cfg;
137
+ const tags = Array.isArray(cfg.tags) ? cfg.tags.slice() : [];
138
+ return {
139
+ method: leaf.method,
140
+ // 'get' | 'post' | ...
141
+ path: leaf.path,
142
+ cfg: {
143
+ description: cfg.description,
144
+ summary: cfg.summary,
145
+ docsGroup: cfg.docsGroup,
146
+ tags,
147
+ deprecated: cfg.deprecated,
148
+ stability: cfg.stability,
149
+ feed: !!cfg.feed,
150
+ docsMeta: cfg.docsMeta,
151
+ hasBody: !!cfg.bodySchema || !!cfg.bodyFiles?.length,
152
+ hasQuery: !!cfg.querySchema,
153
+ hasParams: !!cfg.paramsSchema,
154
+ hasOutput: !!cfg.outputSchema,
155
+ bodySchema: introspectSchema(cfg.bodySchema),
156
+ querySchema: introspectSchema(cfg.querySchema),
157
+ paramsSchema: introspectSchema(cfg.paramsSchema),
158
+ outputSchema: introspectSchema(cfg.outputSchema)
159
+ }
160
+ };
161
+ }
162
+
163
+ // src/docs/LeafDocsPage.tsx
164
+ import { jsx, jsxs } from "react/jsx-runtime";
165
+ var DEFAULT_ASSET_BASE = "/__rrroutes/docs/assets";
166
+ function normalizeBase(base) {
167
+ if (!base) return DEFAULT_ASSET_BASE;
168
+ return base.endsWith("/") ? base.slice(0, -1) : base;
169
+ }
170
+ var DocsDocument = ({ leavesJson, assetBase, cspNonce }) => {
171
+ const cssHref = `${assetBase}/docs.css`;
172
+ const jsSrc = `${assetBase}/docs.js`;
173
+ return /* @__PURE__ */ jsxs("html", { lang: "en", children: [
174
+ /* @__PURE__ */ jsxs("head", { children: [
175
+ /* @__PURE__ */ jsx("meta", { charSet: "UTF-8" }),
176
+ /* @__PURE__ */ jsx("meta", { name: "viewport", content: "width=device-width, initial-scale=1.0" }),
177
+ /* @__PURE__ */ jsx("title", { children: "API Reference" }),
178
+ /* @__PURE__ */ jsx("link", { rel: "stylesheet", href: cssHref })
179
+ ] }),
180
+ /* @__PURE__ */ jsxs("body", { children: [
181
+ /* @__PURE__ */ jsx("div", { id: "docs-root" }),
182
+ /* @__PURE__ */ jsx(
183
+ "script",
184
+ {
185
+ id: "leaf-data",
186
+ type: "application/json",
187
+ nonce: cspNonce,
188
+ dangerouslySetInnerHTML: { __html: leavesJson }
189
+ }
190
+ ),
191
+ /* @__PURE__ */ jsx("script", { type: "module", src: jsSrc, nonce: cspNonce })
192
+ ] })
193
+ ] });
194
+ };
195
+ function serializeLeaves(leaves) {
196
+ return JSON.stringify(leaves.map(serializeLeaf)).replace(/<\//g, "<\\/");
197
+ }
198
+ function createLeafDocsDocument(leaves, options = {}) {
199
+ const assetBase = normalizeBase(options.assetBasePath ?? DEFAULT_ASSET_BASE);
200
+ const leavesJson = serializeLeaves(leaves);
201
+ return /* @__PURE__ */ jsx(DocsDocument, { leavesJson, assetBase, cspNonce: options.cspNonce });
202
+ }
203
+ function renderLeafDocsHTML(leaves, options = {}) {
204
+ const doc = createLeafDocsDocument(leaves, options);
205
+ const html = renderToStaticMarkup(doc);
206
+ return `<!DOCTYPE html>${html}`;
207
+ }
208
+
209
+ // src/docs/docs.ts
210
+ function renderLeafDocsHTML2(leaves, options = {}) {
211
+ return renderLeafDocsHTML(leaves, options);
212
+ }
213
+
214
+ // src/index.ts
215
+ var trimTrailingSlash = (value) => value.endsWith("/") && value.length > 1 ? value.slice(0, -1) : value;
216
+ function mountRRRoutesDocs({
217
+ router,
218
+ leaves,
219
+ options = {}
220
+ }) {
221
+ const docsPath = options.path ?? "/__rrroutes/docs";
222
+ const normalizedDocsPath = trimTrailingSlash(docsPath);
223
+ const assetsMountPath = trimTrailingSlash(
224
+ options.assetBasePath ?? `${normalizedDocsPath}/assets`
225
+ );
226
+ const publicDir = fileURLToPath(new URL("../public", import.meta.url));
227
+ const assetsDir = path.join(publicDir, "assets");
228
+ const cspEnabled = options.csp !== false;
229
+ router.use(assetsMountPath, expressStatic(assetsDir, { immutable: true, maxAge: "365d" }));
230
+ router.get(docsPath, (req, res) => {
231
+ const preparedLeaves = Array.isArray(leaves) ? leaves.filter((leaf) => leaf.cfg.docsHidden !== true) : [];
232
+ const onRequestResult = options.onRequest?.({ req, res, leaves: preparedLeaves }) ?? {};
233
+ const finalLeaves = onRequestResult.leaves ?? preparedLeaves;
234
+ const hasCustomHtml = typeof onRequestResult.html === "string";
235
+ let nonce = onRequestResult.nonce;
236
+ if (!nonce && cspEnabled && !hasCustomHtml) {
237
+ nonce = randomBytes(16).toString("base64");
238
+ }
239
+ const html = hasCustomHtml ? onRequestResult.html : renderLeafDocsHTML2(finalLeaves, {
240
+ cspNonce: nonce,
241
+ assetBasePath: assetsMountPath
242
+ });
243
+ if (cspEnabled && nonce) {
244
+ res.setHeader(
245
+ "Content-Security-Policy",
246
+ [
247
+ "default-src 'self'",
248
+ `script-src 'self' 'nonce-${nonce}'`,
249
+ `style-src 'self' 'nonce-${nonce}'`,
250
+ "img-src 'self' data:",
251
+ "connect-src 'self'",
252
+ "font-src 'self'",
253
+ "frame-ancestors 'self'"
254
+ ].join("; ")
255
+ );
256
+ }
257
+ res.send(html);
258
+ });
259
+ return { path: docsPath };
260
+ }
261
+ export {
262
+ mountRRRoutesDocs,
263
+ renderLeafDocsHTML2 as renderLeafDocsHTML,
264
+ serializeLeaf
265
+ };
266
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.ts","../src/docs/LeafDocsPage.tsx","../src/docs/schemaIntrospection.ts","../src/docs/serializer.ts","../src/docs/docs.ts"],"sourcesContent":["import { randomBytes } from 'crypto';\nimport path from 'node:path';\nimport { fileURLToPath } from 'node:url';\nimport { static as expressStatic } from 'express';\nimport type { Request, Response, Router } from 'express';\nimport type { AnyLeaf } from '@emeryld/rrroutes-contract';\nimport { renderLeafDocsHTML } from './docs/docs.js';\n\nexport type DocsRequestContext = {\n req: Request;\n res: Response;\n leaves: AnyLeaf[];\n};\n\nexport type DocsOnRequestResult = {\n leaves?: AnyLeaf[];\n nonce?: string;\n html?: string;\n};\n\nexport type OpenApiDocsOptions = {\n /** Path where docs are mounted. Defaults to `/__rrroutes/docs`. */\n path?: string;\n /** Whether to emit a CSP header + nonce. Defaults to true. */\n csp?: boolean;\n /** Override where static assets are served from. Defaults to `${path}/assets`. */\n assetBasePath?: string;\n /**\n * Hook that runs on every request. Use it to adjust leaves, override nonce, or\n * provide a fully custom HTML response.\n */\n onRequest?: (ctx: DocsRequestContext) => DocsOnRequestResult | void;\n};\n\nexport type MountDocsArgs = {\n router: Router;\n leaves: AnyLeaf[];\n options?: OpenApiDocsOptions;\n};\n\nconst trimTrailingSlash = (value: string) =>\n value.endsWith('/') && value.length > 1 ? value.slice(0, -1) : value;\n\nexport function mountRRRoutesDocs({\n router,\n leaves,\n options = {},\n}: MountDocsArgs) {\n const docsPath = options.path ?? '/__rrroutes/docs';\n const normalizedDocsPath = trimTrailingSlash(docsPath);\n const assetsMountPath = trimTrailingSlash(\n options.assetBasePath ?? `${normalizedDocsPath}/assets`,\n );\n const publicDir = fileURLToPath(new URL('../public', import.meta.url));\n const assetsDir = path.join(publicDir, 'assets');\n const cspEnabled = options.csp !== false;\n\n router.use(assetsMountPath, expressStatic(assetsDir, { immutable: true, maxAge: '365d' }));\n\n router.get(docsPath, (req, res) => {\n const preparedLeaves = Array.isArray(leaves)\n ? leaves.filter((leaf) => leaf.cfg.docsHidden !== true)\n : [];\n const onRequestResult = options.onRequest?.({ req, res, leaves: preparedLeaves }) ?? {};\n const finalLeaves = onRequestResult.leaves ?? preparedLeaves;\n\n const hasCustomHtml = typeof onRequestResult.html === 'string';\n\n let nonce = onRequestResult.nonce;\n if (!nonce && cspEnabled && !hasCustomHtml) {\n nonce = randomBytes(16).toString('base64');\n }\n\n const html = hasCustomHtml\n ? (onRequestResult.html as string)\n : renderLeafDocsHTML(finalLeaves, {\n cspNonce: nonce,\n assetBasePath: assetsMountPath,\n });\n\n if (cspEnabled && nonce) {\n res.setHeader(\n 'Content-Security-Policy',\n [\n \"default-src 'self'\",\n `script-src 'self' 'nonce-${nonce}'`,\n `style-src 'self' 'nonce-${nonce}'`,\n \"img-src 'self' data:\",\n \"connect-src 'self'\",\n \"font-src 'self'\",\n \"frame-ancestors 'self'\",\n ].join('; '),\n );\n }\n\n res.send(html);\n });\n\n return { path: docsPath };\n}\n\nexport { renderLeafDocsHTML } from './docs/docs.js';\nexport { serializeLeaf } from './docs/serializer.js';\nexport type { SerializableLeaf } from './docs/serializer.js';\n","import type { AnyLeaf } from '@emeryld/rrroutes-contract';\nimport type { ReactElement } from 'react';\nimport { renderToStaticMarkup } from 'react-dom/server';\nimport { serializeLeaf } from './serializer.js';\n\nexport interface RenderOptions {\n /** CSP nonce applied to data + script tags. */\n cspNonce?: string;\n /** Base URL where static assets are served (e.g. `/__rrroutes/docs/assets`). */\n assetBasePath?: string;\n}\n\nconst DEFAULT_ASSET_BASE = '/__rrroutes/docs/assets';\n\nfunction normalizeBase(base: string) {\n if (!base) return DEFAULT_ASSET_BASE;\n return base.endsWith('/') ? base.slice(0, -1) : base;\n}\n\ntype DocsDocumentProps = {\n leavesJson: string;\n assetBase: string;\n cspNonce?: string;\n};\n\nexport const DocsDocument = ({ leavesJson, assetBase, cspNonce }: DocsDocumentProps) => {\n const cssHref = `${assetBase}/docs.css`;\n const jsSrc = `${assetBase}/docs.js`;\n\n return (\n <html lang=\"en\">\n <head>\n <meta charSet=\"UTF-8\" />\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\" />\n <title>API Reference</title>\n <link rel=\"stylesheet\" href={cssHref} />\n </head>\n <body>\n <div id=\"docs-root\"></div>\n <script\n id=\"leaf-data\"\n type=\"application/json\"\n nonce={cspNonce}\n dangerouslySetInnerHTML={{ __html: leavesJson }}\n />\n <script type=\"module\" src={jsSrc} nonce={cspNonce} />\n </body>\n </html>\n );\n};\n\nfunction serializeLeaves(leaves: AnyLeaf[]) {\n return JSON.stringify(leaves.map(serializeLeaf)).replace(/<\\//g, '<\\\\/');\n}\n\nexport function createLeafDocsDocument(\n leaves: AnyLeaf[],\n options: RenderOptions = {},\n): ReactElement {\n const assetBase = normalizeBase(options.assetBasePath ?? DEFAULT_ASSET_BASE);\n const leavesJson = serializeLeaves(leaves);\n\n return <DocsDocument leavesJson={leavesJson} assetBase={assetBase} cspNonce={options.cspNonce} />;\n}\n\nexport function renderLeafDocsHTML(leaves: AnyLeaf[], options: RenderOptions = {}): string {\n const doc = createLeafDocsDocument(leaves, options);\n const html = renderToStaticMarkup(doc);\n return `<!DOCTYPE html>${html}`;\n}\n","// schemaIntrospection.ts\nimport * as z from \"zod\";\n\nexport type SerializableSchemaNode = {\n kind: string; // \"object\" | \"string\" | \"number\" | ...\n optional?: boolean;\n nullable?: boolean;\n description?: string;\n\n // object\n properties?: Record<string, SerializableSchemaNode>;\n // array\n element?: SerializableSchemaNode;\n // union\n union?: SerializableSchemaNode[];\n // literal\n literal?: unknown;\n // enum\n enumValues?: string[];\n};\n\ntype ZodAny = z.ZodTypeAny;\n\n/**\n * Zod 3 uses `schema._def`, Zod 4 uses `schema._zod.def`.\n */\nfunction getDef(schema: unknown): any | undefined {\n if (!schema || typeof schema !== \"object\") return undefined;\n const anySchema = schema as any;\n return anySchema._zod?.def ?? anySchema._def;\n}\n\n/**\n * Try to get a human-readable description.\n * Zod 4: use metadata/registry; fallback to internal def.description.\n * Zod 3: only internal def.description exists.\n */\nfunction getDescription(schema: ZodAny): string | undefined {\n const anyZ: any = z as any;\n\n // Zod 4 global registry metadata, if present\n const registry = anyZ.globalRegistry?.get\n ? anyZ.globalRegistry.get(schema)\n : undefined;\n if (registry && typeof registry.description === \"string\") {\n return registry.description;\n }\n\n // Legacy / internal description\n const def = getDef(schema);\n if (def && typeof def.description === \"string\") {\n return def.description;\n }\n\n return undefined;\n}\n\n/**\n * Peel off wrappers (effects, optional, nullable, default) and\n * return the inner schema + flags.\n *\n * Supports:\n * - Zod 3: ZodEffects, ZodOptional, ZodNullable, ZodDefault\n * - Zod 4: ZodOptional, ZodNullable, ZodDefault\n */\nfunction unwrap(schema: ZodAny): {\n base: ZodAny;\n optional: boolean;\n nullable: boolean;\n} {\n let s: ZodAny = schema;\n let optional = false;\n let nullable = false;\n\n // Zod 3 only (undefined in Zod 4)\n const ZodEffectsCtor: any = (z as any).ZodEffects;\n\n // eslint-disable-next-line no-constant-condition\n while (true) {\n // Zod 3: ZodEffects wrapper\n if (ZodEffectsCtor && s instanceof ZodEffectsCtor) {\n const def = getDef(s) || {};\n const sourceType =\n typeof (s as any).sourceType === \"function\"\n ? (s as any).sourceType()\n : def.schema;\n if (!sourceType) break;\n s = sourceType;\n continue;\n }\n\n // Zod 3 + 4: optional/nullable/default wrappers\n if (s instanceof z.ZodOptional) {\n optional = true;\n const def = getDef(s);\n s = (def && def.innerType) || s; // innerType exists in both 3 & 4\n continue;\n }\n\n if (s instanceof z.ZodNullable) {\n nullable = true;\n const def = getDef(s);\n s = (def && def.innerType) || s;\n continue;\n }\n\n if (s instanceof z.ZodDefault) {\n const def = getDef(s);\n s = (def && def.innerType) || s;\n continue;\n }\n\n break;\n }\n\n return { base: s, optional, nullable };\n}\n\nexport function introspectSchema(\n schema: ZodAny | undefined\n): SerializableSchemaNode | undefined {\n if (!schema) return undefined;\n\n const { base, optional, nullable } = unwrap(schema);\n const def = getDef(base);\n\n const node: SerializableSchemaNode = {\n kind: inferKind(base),\n optional: optional || undefined,\n nullable: nullable || undefined,\n description: getDescription(base),\n };\n\n // OBJECT\n if (base instanceof z.ZodObject) {\n // Zod 3: _def.shape() (function)\n // Zod 4: .shape getter returns an object\n const rawShape: any =\n (base as any).shape ?? (def && typeof def.shape === \"function\"\n ? def.shape()\n : def?.shape);\n\n const shape =\n typeof rawShape === \"function\" ? rawShape() : rawShape ?? {};\n\n const props: Record<string, SerializableSchemaNode> = {};\n for (const key of Object.keys(shape)) {\n const child = shape[key] as ZodAny;\n const childNode = introspectSchema(child);\n if (childNode) props[key] = childNode;\n }\n node.properties = props;\n }\n\n // ARRAY\n if (base instanceof z.ZodArray) {\n // Zod 3: def.type is inner schema\n // Zod 4: def.element is inner schema\n const inner =\n (def && (def.element as ZodAny)) ||\n (def && (def.type as ZodAny)) ||\n undefined;\n if (inner) {\n node.element = introspectSchema(inner);\n }\n }\n\n // UNION\n if (base instanceof z.ZodUnion) {\n const options: ZodAny[] = (def && def.options) || [];\n node.union = options\n .map((opt) => introspectSchema(opt))\n .filter(Boolean) as SerializableSchemaNode[];\n }\n\n // LITERAL\n if (base instanceof z.ZodLiteral) {\n if (def) {\n // Zod 4: def.values (multi-literal)\n if (Array.isArray(def.values)) {\n node.literal =\n def.values.length === 1 ? def.values[0] : def.values.slice();\n } else {\n // Zod 3: def.value\n node.literal = def.value;\n }\n }\n }\n\n // ENUM\n if (base instanceof z.ZodEnum) {\n if (def) {\n if (Array.isArray(def.values)) {\n // Zod 3\n node.enumValues = def.values.slice();\n } else if (def.entries && typeof def.entries === \"object\") {\n // Zod 4: entries is a { key: value } map\n node.enumValues = Object.values(def.entries).map((v: unknown) =>\n String(v)\n );\n }\n }\n }\n\n return node;\n}\n\nfunction inferKind(schema: ZodAny): string {\n // This path still uses instanceof; it works with Zod 4 Classic\n // (importing from \"zod\"). Anything unknown falls back to \"unknown\".\n if (schema instanceof z.ZodString) return \"string\";\n if (schema instanceof z.ZodNumber) return \"number\";\n if (schema instanceof z.ZodBoolean) return \"boolean\";\n if (schema instanceof z.ZodBigInt) return \"bigint\";\n if (schema instanceof z.ZodDate) return \"date\";\n if (schema instanceof z.ZodArray) return \"array\";\n if (schema instanceof z.ZodObject) return \"object\";\n if (schema instanceof z.ZodUnion) return \"union\";\n if (schema instanceof z.ZodLiteral) return \"literal\";\n if (schema instanceof z.ZodEnum) return \"enum\";\n if (schema instanceof z.ZodRecord) return \"record\";\n if (schema instanceof z.ZodTuple) return \"tuple\";\n if (schema instanceof z.ZodUnknown) return \"unknown\";\n if (schema instanceof z.ZodAny) return \"any\";\n\n return \"unknown\";\n}\n","// serializer.ts\nimport type { AnyLeaf, MethodCfg } from \"@emeryld/rrroutes-contract\";\nimport { introspectSchema, SerializableSchemaNode } from \"./schemaIntrospection.js\";\n\ntype SerializableMethodCfg = Pick<\n MethodCfg,\n | \"description\"\n | \"summary\"\n | \"docsGroup\"\n | \"tags\"\n | \"deprecated\"\n | \"stability\"\n | \"feed\"\n | \"docsMeta\"\n> & {\n hasBody: boolean;\n hasQuery: boolean;\n hasParams: boolean;\n hasOutput: boolean;\n\n // NEW: full Zod ASTs\n bodySchema?: SerializableSchemaNode;\n querySchema?: SerializableSchemaNode;\n paramsSchema?: SerializableSchemaNode;\n outputSchema?: SerializableSchemaNode;\n};\n\nexport type SerializableLeaf = {\n method: string;\n path: string;\n cfg: SerializableMethodCfg;\n};\n\nexport type { SerializableSchemaNode } from \"./schemaIntrospection.js\";\n\nexport function serializeLeaf(leaf: AnyLeaf): SerializableLeaf {\n const cfg = leaf.cfg;\n\n const tags = Array.isArray(cfg.tags) ? cfg.tags.slice() : [];\n\n return {\n method: leaf.method, // 'get' | 'post' | ...\n path: leaf.path,\n cfg: {\n description: cfg.description,\n summary: cfg.summary,\n docsGroup: cfg.docsGroup,\n tags,\n deprecated: cfg.deprecated,\n stability: cfg.stability,\n feed: !!cfg.feed,\n docsMeta: cfg.docsMeta,\n hasBody: !!cfg.bodySchema || !!cfg.bodyFiles?.length,\n hasQuery: !!cfg.querySchema,\n hasParams: !!cfg.paramsSchema,\n hasOutput: !!cfg.outputSchema,\n\n bodySchema: introspectSchema(cfg.bodySchema),\n querySchema: introspectSchema(cfg.querySchema),\n paramsSchema: introspectSchema(cfg.paramsSchema),\n outputSchema: introspectSchema(cfg.outputSchema),\n },\n };\n}\n","// renderLeafDocsHTML.ts\nimport type { AnyLeaf } from \"@emeryld/rrroutes-contract\";\nimport {\n createLeafDocsDocument,\n renderLeafDocsHTML as LeafDocsPage,\n RenderOptions,\n} from \"./LeafDocsPage.js\";\n\nexport function renderLeafDocsHTML(leaves: AnyLeaf[], options: RenderOptions = {}): string {\n return LeafDocsPage(leaves, options);\n}\n\nexport type { RenderOptions } from \"./LeafDocsPage.js\";\nexport { createLeafDocsDocument } from \"./LeafDocsPage.js\";\n"],"mappings":"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","names":["renderLeafDocsHTML","renderLeafDocsHTML"]}
@@ -0,0 +1 @@
1
+ :root{--bg-root: #020617;--bg-card: rgba(15, 23, 42, .6);--bg-card-hover: rgba(30, 41, 59, .7);--bg-glass: rgba(15, 23, 42, .9);--bg-panel: #0f172a;--border-subtle: rgba(148, 163, 184, .2);--text-main: #f8fafc;--text-muted: #94a3b8;--text-accent: #c4b5fd;--accent-primary: #a855f7;--accent-glow: rgba(168, 85, 247, .25);--accent-success: #4ade80;--accent-error: #f87171;--method-get: #4ade80;--method-get-bg: rgba(34, 197, 94, .15);--method-post: #60a5fa;--method-post-bg: rgba(59, 130, 246, .15);--method-put: #facc15;--method-put-bg: rgba(234, 179, 8, .15);--method-patch: #2dd4bf;--method-patch-bg: rgba(45, 212, 191, .15);--method-delete: #f87171;--method-delete-bg: rgba(248, 113, 113, .15);--radius-card: 12px;--shadow-card: 0 4px 20px rgba(0, 0, 0, .4);--shadow-panel: -5px 0 30px rgba(0,0,0,.5);--font-mono: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace;--tag-0-bg: rgba(254, 202, 202, .2);--tag-0-fg: #fca5a5;--tag-1-bg: rgba(253, 230, 138, .2);--tag-1-fg: #fcd34d;--tag-2-bg: rgba(187, 247, 208, .2);--tag-2-fg: #86efac;--tag-3-bg: rgba(165, 243, 252, .2);--tag-3-fg: #67e8f9;--tag-4-bg: rgba(191, 219, 254, .2);--tag-4-fg: #93c5fd;--tag-5-bg: rgba(233, 213, 255, .2);--tag-5-fg: #d8b4fe;--tag-6-bg: rgba(251, 207, 232, .2);--tag-6-fg: #f9a8d4}*{box-sizing:border-box}html,body{height:100%;margin:0;padding:0;background-color:var(--bg-root);color:var(--text-main);font-family:system-ui,-apple-system,sans-serif}body{background-image:radial-gradient(circle at 15% 10%,rgba(88,28,135,.15),transparent 40%),radial-gradient(circle at 85% 80%,rgba(15,23,42,1),transparent 40%);background-attachment:fixed}.page{min-height:100vh;padding:20px 40px 60px;max-width:1600px;margin:0 auto;display:flex;flex-direction:column;gap:24px}.header{display:flex;justify-content:space-between;padding-bottom:10px;border-bottom:1px solid var(--border-subtle)}.header-title h1{font-size:28px;margin:0;font-weight:700;letter-spacing:-.02em;background:linear-gradient(to right,#e2e8f0,#a855f7);-webkit-background-clip:text;-webkit-text-fill-color:transparent}.controls-container{position:sticky;top:0;z-index:50;background:var(--bg-glass);-webkit-backdrop-filter:blur(16px);backdrop-filter:blur(16px);border:1px solid var(--border-subtle);border-radius:var(--radius-card);padding:12px 16px;display:flex;flex-direction:column;gap:12px;box-shadow:0 10px 30px -10px #00000080}.filters-row{display:flex;align-items:flex-start;gap:16px;width:100%}.left-column{display:flex;flex-direction:column;gap:16px;flex-basis:300px;min-width:250px}.search-box{width:100%;position:relative}.search-input{width:100%;background:#02061799;border:1px solid var(--border-subtle);color:var(--text-main);padding:8px 12px 8px 32px;border-radius:6px;font-size:13px}.search-input:focus{outline:none;border-color:var(--accent-primary)}.search-icon{position:absolute;left:10px;top:50%;transform:translateY(-50%);color:var(--text-muted);font-size:12px}.filter-group{display:flex;flex-direction:column;gap:6px}.filter-label{font-size:10px;text-transform:uppercase;color:var(--text-muted);font-weight:700}.checkbox-group{display:flex;flex-wrap:wrap;gap:6px}.pill-checkbox input{display:none}.pill-checkbox span{display:inline-block;padding:3px 8px;border-radius:4px;font-size:11px;font-weight:600;background:#1e293b80;color:var(--text-muted);border:1px solid transparent;cursor:pointer}.pill-checkbox input:checked+span{background:#a855f726;color:var(--accent-primary);border-color:var(--accent-primary)}.pill-checkbox.colored-tag input:checked+.tag-filter-pill{border-color:var(--accent-primary);box-shadow:0 0 0 1px var(--accent-glow)}.overview-row{display:flex;flex-wrap:wrap;gap:8px;padding-top:8px;border-top:1px solid var(--border-subtle);align-items:center}.overview-label{font-size:10px;text-transform:uppercase;color:var(--text-muted);font-weight:700}.group-chip{font-size:11px;text-decoration:none;color:var(--text-accent);background:#a855f714;padding:3px 8px;border-radius:4px;border:1px solid transparent}.group-chip:hover{background:#a855f733;border-color:var(--accent-primary);color:#fff}.api-group{margin-bottom:40px;scroll-margin-top:180px}.group-header{display:flex;align-items:center;justify-content:space-between;margin-bottom:16px;border-bottom:1px solid var(--border-subtle)}.group-title{font-size:20px;font-weight:600;margin:0;padding-bottom:8px}.group-actions button{background:transparent;border:1px solid var(--border-subtle);color:var(--text-muted);font-size:11px;padding:4px 10px;border-radius:4px;cursor:pointer;margin-left:8px}.endpoint-card{background:var(--bg-card);border:1px solid var(--border-subtle);border-radius:var(--radius-card);overflow:hidden;transition:all .2s ease;margin-bottom:12px}.endpoint-card:hover{border-color:#94a3b866;box-shadow:var(--shadow-card)}.card-header{padding:12px 16px;display:flex;align-items:center;gap:12px;flex-wrap:wrap;cursor:pointer}.header-spacer{flex:1}.card-body{display:none;padding:0 16px 16px;border-top:1px solid var(--border-subtle);margin-top:4px}.endpoint-card[data-expanded=true] .card-body{display:block;animation:slideDown .2s ease-out}.endpoint-card[data-expanded=true] .expand-icon{transform:rotate(180deg);color:var(--text-main)}@keyframes slideDown{0%{opacity:0;transform:translateY(-5px)}to{opacity:1;transform:translateY(0)}}.method-badge{font-size:11px;font-weight:800;text-transform:uppercase;padding:4px 8px;border-radius:4px;min-width:55px;text-align:center}.m-GET{background:var(--method-get-bg);color:var(--method-get);border:1px solid rgba(74,222,128,.3)}.m-POST{background:var(--method-post-bg);color:var(--method-post);border:1px solid rgba(96,165,250,.3)}.m-PUT{background:var(--method-put-bg);color:var(--method-put);border:1px solid rgba(250,204,21,.3)}.m-PATCH{background:var(--method-patch-bg);color:var(--method-patch);border:1px solid rgba(45,212,191,.3)}.m-DELETE{background:var(--method-delete-bg);color:var(--method-delete);border:1px solid rgba(248,113,113,.3)}.path-container{font-family:var(--font-mono);font-size:13px;display:inline-flex;align-items:center;gap:8px;padding:2px 6px;border-radius:4px}.path-container:hover{background:#ffffff0d}.copy-icon{opacity:0;font-size:10px}.path-container:hover .copy-icon{opacity:1}.tags-container{display:flex;gap:6px}.status-badge{font-size:9px;text-transform:uppercase;padding:2px 6px;border-radius:3px;font-weight:600}.tag-0{background:var(--tag-0-bg);color:var(--tag-0-fg);border-color:var(--tag-0-fg)}.tag-1{background:var(--tag-1-bg);color:var(--tag-1-fg);border-color:var(--tag-1-fg)}.tag-2{background:var(--tag-2-bg);color:var(--tag-2-fg);border-color:var(--tag-2-fg)}.tag-3{background:var(--tag-3-bg);color:var(--tag-3-fg);border-color:var(--tag-3-fg)}.tag-4{background:var(--tag-4-bg);color:var(--tag-4-fg);border-color:var(--tag-4-fg)}.tag-5{background:var(--tag-5-bg);color:var(--tag-5-fg);border-color:var(--tag-5-fg)}.tag-6{background:var(--tag-6-bg);color:var(--tag-6-fg);border-color:var(--tag-6-fg)}.not-implemented{border-color:var(--method-delete);color:var(--method-delete);background:var(--method-delete-bg)}.btn-try-it{background:#a855f71a;border:1px solid rgba(168,85,247,.4);color:var(--text-accent);font-size:11px;font-weight:600;padding:4px 12px;border-radius:12px;cursor:pointer;transition:all .2s;margin-right:12px;display:flex;align-items:center;gap:6px}.btn-try-it:hover{background:#a855f740;color:#fff;box-shadow:0 0 10px var(--accent-glow)}.btn-try-it .play-icon{font-size:10px}.section-block{margin-top:18px}.section-title{font-size:11px;text-transform:uppercase;letter-spacing:.1em;color:var(--text-muted);margin-bottom:8px;border-bottom:1px solid var(--border-subtle);padding-bottom:4px}.schema-table{width:100%;border-collapse:collapse;font-size:12px}.schema-table th{text-align:left;color:var(--text-muted);font-weight:600;padding:6px 8px;border-bottom:1px solid var(--border-subtle);background:#0003}.schema-table td{padding:6px 8px;border-bottom:1px solid var(--border-subtle);vertical-align:top;color:var(--text-muted)}.col-name{font-family:var(--font-mono);color:var(--text-accent)!important;width:25%}.col-type{font-family:var(--font-mono);color:#93c5fd!important;width:15%}.req-badge{font-size:9px;text-transform:uppercase;padding:2px 4px;border-radius:2px}.req-true{color:#4ade80;background:#4ade801a}.req-false{color:#94a3b8;background:#94a3b81a}.schema-toggle{background:none;border:none;color:var(--text-muted);cursor:pointer;font-size:10px;padding:0 4px;margin-right:4px;width:16px;text-align:center}.schema-toggle:hover{color:var(--text-main)}tr[data-hidden=true]{display:none}.schema-name-cell{display:flex;align-items:center}.schema-name-text{opacity:.8}.schema-toggle-placeholder{width:20px;display:inline-block}.schema-depth-0{padding-left:0}.schema-depth-1{padding-left:16px}.schema-depth-2{padding-left:32px}.schema-depth-3{padding-left:48px}.schema-depth-4{padding-left:64px}.schema-depth-5{padding-left:80px}.schema-depth-6{padding-left:96px}.schema-depth-7{padding-left:112px}.schema-depth-8{padding-left:128px}.playground-overlay{position:fixed;top:0;right:0;bottom:0;width:600px;max-width:90vw;background:var(--bg-panel);border-left:1px solid var(--border-subtle);box-shadow:var(--shadow-panel);z-index:100;transform:translate(100%);transition:transform .3s cubic-bezier(.16,1,.3,1);display:flex;flex-direction:column}.playground-overlay.open{transform:translate(0)}.loading-placeholder{text-align:center;padding:40px;color:#64748b}.no-results{text-align:center;padding:40px;color:var(--text-muted)}.section-header-row{display:flex;justify-content:space-between;align-items:flex-end;border-bottom:1px solid var(--border-subtle);margin-bottom:8px}.section-title-no-border{border:none;margin:0}.schema-actions-inline{margin-bottom:4px}.schema-actions-inline button.expand-all{background:none;border:none;color:var(--text-accent);cursor:pointer;font-size:10px}.schema-actions-inline button.collapse-all{background:none;border:none;color:var(--text-muted);cursor:pointer;font-size:10px}.pg-header{padding:16px;border-bottom:1px solid var(--border-subtle);display:flex;justify-content:space-between;align-items:center;background:var(--bg-glass)}.pg-title{font-size:16px;font-weight:700;color:var(--text-main)}.pg-close{background:transparent;border:none;color:var(--text-muted);font-size:20px;cursor:pointer}.pg-close:hover{color:var(--text-main)}.pg-content{flex:1;overflow-y:auto;padding:20px;display:flex;flex-direction:column;gap:24px}.pg-section{display:flex;flex-direction:column;gap:8px}.pg-label{font-size:11px;text-transform:uppercase;color:var(--text-muted);font-weight:700;letter-spacing:.05em}.pg-input,.pg-textarea{background:#02061766;border:1px solid var(--border-subtle);color:var(--text-main);padding:8px;border-radius:6px;font-family:var(--font-mono);font-size:12px}.pg-input:focus,.pg-textarea:focus{outline:none;border-color:var(--accent-primary)}.pg-textarea{min-height:150px;resize:vertical}.pg-input-disabled-label{opacity:.7}.param-grid{display:grid;grid-template-columns:1fr 1fr;gap:8px}.param-row{display:contents}.btn-add-query{font-size:11px;background:none;border:1px dashed var(--border-subtle);color:var(--text-muted);padding:4px;cursor:pointer}.pg-footer{padding:16px;border-top:1px solid var(--border-subtle);background:var(--bg-glass);display:flex;justify-content:flex-end;gap:12px}.btn-send{background:var(--accent-primary);color:#fff;border:none;padding:8px 24px;border-radius:6px;font-weight:600;cursor:pointer;transition:opacity .2s}.btn-send:hover{opacity:.9}.btn-send:disabled{opacity:.5;cursor:not-allowed}.response-box{background:#0000004d;border-radius:8px;border:1px solid var(--border-subtle);overflow:hidden}.response-meta{background:#ffffff0d;padding:8px 12px;display:flex;gap:12px;font-size:11px;font-family:var(--font-mono);border-bottom:1px solid var(--border-subtle)}.meta-item{display:flex;align-items:center;gap:4px}.status-ok{color:var(--accent-success)}.status-err{color:var(--accent-error)}.response-body{padding:12px;overflow-x:auto;font-family:var(--font-mono);font-size:12px}.pg-response-hidden{display:none}.response-pre{margin:0}.json-tree{font-family:var(--font-mono);line-height:1.6}.jt-obj,.jt-arr{margin-left:14px}.jt-key{color:#93c5fd;margin-right:4px}.jt-str{color:#86efac}.jt-num{color:#fca5a5}.jt-bool{color:#fcd34d}.jt-null{color:#94a3b8}.jt-toggle{display:inline-block;width:12px;text-align:center;cursor:pointer;color:var(--text-muted);-webkit-user-select:none;user-select:none;margin-left:-14px}.jt-toggle:hover{color:var(--text-main)}.jt-collapsed .jt-content{display:none}.jt-collapsed:after{content:" ... ";color:var(--text-muted);font-size:10px}.json-tree-line{padding-left:16px}.hidden-template{display:none}