@metaobjectsdev/codegen-ts 0.9.0 → 0.10.0
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/README.md +1 -1
- package/dist/column-mapper.d.ts.map +1 -1
- package/dist/column-mapper.js +24 -8
- package/dist/column-mapper.js.map +1 -1
- package/dist/constants.d.ts +8 -0
- package/dist/constants.d.ts.map +1 -1
- package/dist/constants.js +16 -0
- package/dist/constants.js.map +1 -1
- package/dist/docs-paths.d.ts +58 -0
- package/dist/docs-paths.d.ts.map +1 -0
- package/dist/docs-paths.js +89 -0
- package/dist/docs-paths.js.map +1 -0
- package/dist/enum-import.d.ts +14 -0
- package/dist/enum-import.d.ts.map +1 -0
- package/dist/enum-import.js +35 -0
- package/dist/enum-import.js.map +1 -0
- package/dist/enum-shared.d.ts +32 -0
- package/dist/enum-shared.d.ts.map +1 -0
- package/dist/enum-shared.js +83 -0
- package/dist/enum-shared.js.map +1 -0
- package/dist/generator-registry.d.ts +22 -0
- package/dist/generator-registry.d.ts.map +1 -0
- package/dist/generator-registry.js +161 -0
- package/dist/generator-registry.js.map +1 -0
- package/dist/generator.d.ts +6 -0
- package/dist/generator.d.ts.map +1 -1
- package/dist/generator.js.map +1 -1
- package/dist/generators/api-doc-render.d.ts +17 -0
- package/dist/generators/api-doc-render.d.ts.map +1 -0
- package/dist/generators/api-doc-render.js +431 -0
- package/dist/generators/api-doc-render.js.map +1 -0
- package/dist/generators/api-docs-file.d.ts +21 -0
- package/dist/generators/api-docs-file.d.ts.map +1 -0
- package/dist/generators/api-docs-file.js +112 -0
- package/dist/generators/api-docs-file.js.map +1 -0
- package/dist/generators/api-field-shape.d.ts +39 -0
- package/dist/generators/api-field-shape.d.ts.map +1 -0
- package/dist/generators/api-field-shape.js +92 -0
- package/dist/generators/api-field-shape.js.map +1 -0
- package/dist/generators/api-label.d.ts +3 -0
- package/dist/generators/api-label.d.ts.map +1 -0
- package/dist/generators/api-label.js +8 -0
- package/dist/generators/api-label.js.map +1 -0
- package/dist/generators/api-model.d.ts +122 -0
- package/dist/generators/api-model.d.ts.map +1 -0
- package/dist/generators/api-model.js +809 -0
- package/dist/generators/api-model.js.map +1 -0
- package/dist/generators/docs-data-builder.d.ts +26 -4
- package/dist/generators/docs-data-builder.d.ts.map +1 -1
- package/dist/generators/docs-data-builder.js +436 -164
- package/dist/generators/docs-data-builder.js.map +1 -1
- package/dist/generators/docs-data.d.ts +136 -27
- package/dist/generators/docs-data.d.ts.map +1 -1
- package/dist/generators/docs-data.js +1 -1
- package/dist/generators/docs-data.js.map +1 -1
- package/dist/generators/docs-file.d.ts +19 -0
- package/dist/generators/docs-file.d.ts.map +1 -1
- package/dist/generators/docs-file.js +154 -27
- package/dist/generators/docs-file.js.map +1 -1
- package/dist/generators/entity-file.d.ts.map +1 -1
- package/dist/generators/entity-file.js +29 -14
- package/dist/generators/entity-file.js.map +1 -1
- package/dist/generators/extractor-file.d.ts.map +1 -1
- package/dist/generators/extractor-file.js +2 -1
- package/dist/generators/extractor-file.js.map +1 -1
- package/dist/generators/field-anchor.d.ts +7 -0
- package/dist/generators/field-anchor.d.ts.map +1 -0
- package/dist/generators/field-anchor.js +23 -0
- package/dist/generators/field-anchor.js.map +1 -0
- package/dist/generators/index.d.ts +8 -1
- package/dist/generators/index.d.ts.map +1 -1
- package/dist/generators/index.js +6 -0
- package/dist/generators/index.js.map +1 -1
- package/dist/generators/mermaid-er.d.ts +14 -0
- package/dist/generators/mermaid-er.d.ts.map +1 -1
- package/dist/generators/mermaid-er.js +14 -0
- package/dist/generators/mermaid-er.js.map +1 -1
- package/dist/generators/output-parser-file.d.ts.map +1 -1
- package/dist/generators/output-parser-file.js +3 -4
- package/dist/generators/output-parser-file.js.map +1 -1
- package/dist/generators/output-prompt-file.d.ts.map +1 -1
- package/dist/generators/output-prompt-file.js +2 -2
- package/dist/generators/output-prompt-file.js.map +1 -1
- package/dist/generators/prompt-render-file.d.ts.map +1 -1
- package/dist/generators/prompt-render-file.js +3 -4
- package/dist/generators/prompt-render-file.js.map +1 -1
- package/dist/generators/queries-file.d.ts.map +1 -1
- package/dist/generators/queries-file.js +8 -3
- package/dist/generators/queries-file.js.map +1 -1
- package/dist/generators/render-helper-file.d.ts.map +1 -1
- package/dist/generators/render-helper-file.js +2 -2
- package/dist/generators/render-helper-file.js.map +1 -1
- package/dist/generators/routes-file-hono.d.ts.map +1 -1
- package/dist/generators/routes-file-hono.js +5 -1
- package/dist/generators/routes-file-hono.js.map +1 -1
- package/dist/generators/routes-file.d.ts +3 -0
- package/dist/generators/routes-file.d.ts.map +1 -1
- package/dist/generators/routes-file.js +6 -1
- package/dist/generators/routes-file.js.map +1 -1
- package/dist/generators/template-doc-builder.d.ts +19 -0
- package/dist/generators/template-doc-builder.d.ts.map +1 -0
- package/dist/generators/template-doc-builder.js +220 -0
- package/dist/generators/template-doc-builder.js.map +1 -0
- package/dist/generators/template-doc-data.d.ts +62 -0
- package/dist/generators/template-doc-data.d.ts.map +1 -0
- package/dist/generators/template-doc-data.js +16 -0
- package/dist/generators/template-doc-data.js.map +1 -0
- package/dist/generators/template-payload-tree.d.ts +15 -0
- package/dist/generators/template-payload-tree.d.ts.map +1 -0
- package/dist/generators/template-payload-tree.js +61 -0
- package/dist/generators/template-payload-tree.js.map +1 -0
- package/dist/generators/template-source-annotate.d.ts +74 -0
- package/dist/generators/template-source-annotate.d.ts.map +1 -0
- package/dist/generators/template-source-annotate.js +184 -0
- package/dist/generators/template-source-annotate.js.map +1 -0
- package/dist/generators/template-source-render.d.ts +24 -0
- package/dist/generators/template-source-render.d.ts.map +1 -0
- package/dist/generators/template-source-render.js +175 -0
- package/dist/generators/template-source-render.js.map +1 -0
- package/dist/generators/trace-helper-file.d.ts +9 -0
- package/dist/generators/trace-helper-file.d.ts.map +1 -0
- package/dist/generators/trace-helper-file.js +196 -0
- package/dist/generators/trace-helper-file.js.map +1 -0
- package/dist/index.d.ts +29 -4
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +28 -2
- package/dist/index.js.map +1 -1
- package/dist/metaobjects-config.d.ts +75 -2
- package/dist/metaobjects-config.d.ts.map +1 -1
- package/dist/metaobjects-config.js +43 -0
- package/dist/metaobjects-config.js.map +1 -1
- package/dist/naming.d.ts +19 -0
- package/dist/naming.d.ts.map +1 -1
- package/dist/naming.js +41 -0
- package/dist/naming.js.map +1 -1
- package/dist/payload-codegen.d.ts.map +1 -1
- package/dist/payload-codegen.js +12 -4
- package/dist/payload-codegen.js.map +1 -1
- package/dist/projection/extract-view-spec.d.ts.map +1 -1
- package/dist/projection/extract-view-spec.js +51 -25
- package/dist/projection/extract-view-spec.js.map +1 -1
- package/dist/relation-resolver.d.ts +16 -0
- package/dist/relation-resolver.d.ts.map +1 -1
- package/dist/relation-resolver.js +82 -1
- package/dist/relation-resolver.js.map +1 -1
- package/dist/render-context.d.ts +4 -0
- package/dist/render-context.d.ts.map +1 -1
- package/dist/render-context.js.map +1 -1
- package/dist/render-engine/embedded-templates.generated.d.ts +2 -0
- package/dist/render-engine/embedded-templates.generated.d.ts.map +1 -0
- package/dist/render-engine/embedded-templates.generated.js +15 -0
- package/dist/render-engine/embedded-templates.generated.js.map +1 -0
- package/dist/render-engine/framework-provider.d.ts.map +1 -1
- package/dist/render-engine/framework-provider.js +26 -13
- package/dist/render-engine/framework-provider.js.map +1 -1
- package/dist/runner.d.ts.map +1 -1
- package/dist/runner.js +17 -0
- package/dist/runner.js.map +1 -1
- package/dist/templates/docs-file.d.ts +2 -6
- package/dist/templates/docs-file.d.ts.map +1 -1
- package/dist/templates/docs-file.js +2 -5
- package/dist/templates/docs-file.js.map +1 -1
- package/dist/templates/drizzle-schema.d.ts.map +1 -1
- package/dist/templates/drizzle-schema.js +30 -2
- package/dist/templates/drizzle-schema.js.map +1 -1
- package/dist/templates/entity-constants.d.ts +7 -0
- package/dist/templates/entity-constants.d.ts.map +1 -1
- package/dist/templates/entity-constants.js +1 -1
- package/dist/templates/entity-constants.js.map +1 -1
- package/dist/templates/entity-file.d.ts.map +1 -1
- package/dist/templates/entity-file.js +16 -5
- package/dist/templates/entity-file.js.map +1 -1
- package/dist/templates/enums-file.d.ts +11 -0
- package/dist/templates/enums-file.d.ts.map +1 -0
- package/dist/templates/enums-file.js +44 -0
- package/dist/templates/enums-file.js.map +1 -0
- package/dist/templates/extract-delegate-emitter.d.ts.map +1 -1
- package/dist/templates/extract-delegate-emitter.js +5 -7
- package/dist/templates/extract-delegate-emitter.js.map +1 -1
- package/dist/templates/extract-schema-emitter.d.ts.map +1 -1
- package/dist/templates/extract-schema-emitter.js +5 -1
- package/dist/templates/extract-schema-emitter.js.map +1 -1
- package/dist/templates/extractor.d.ts.map +1 -1
- package/dist/templates/extractor.js +56 -39
- package/dist/templates/extractor.js.map +1 -1
- package/dist/templates/field-meta.d.ts.map +1 -1
- package/dist/templates/field-meta.js +1 -5
- package/dist/templates/field-meta.js.map +1 -1
- package/dist/templates/filter-allowlist.d.ts +7 -2
- package/dist/templates/filter-allowlist.d.ts.map +1 -1
- package/dist/templates/filter-allowlist.js +17 -9
- package/dist/templates/filter-allowlist.js.map +1 -1
- package/dist/templates/filter-type.d.ts +7 -1
- package/dist/templates/filter-type.d.ts.map +1 -1
- package/dist/templates/filter-type.js +9 -5
- package/dist/templates/filter-type.js.map +1 -1
- package/dist/templates/find-templates.d.ts +4 -0
- package/dist/templates/find-templates.d.ts.map +1 -0
- package/dist/templates/find-templates.js +15 -0
- package/dist/templates/find-templates.js.map +1 -0
- package/dist/templates/fr010-field-mapping.d.ts +2 -0
- package/dist/templates/fr010-field-mapping.d.ts.map +1 -1
- package/dist/templates/fr010-field-mapping.js +10 -6
- package/dist/templates/fr010-field-mapping.js.map +1 -1
- package/dist/templates/inferred-types.d.ts +44 -7
- package/dist/templates/inferred-types.d.ts.map +1 -1
- package/dist/templates/inferred-types.js +107 -16
- package/dist/templates/inferred-types.js.map +1 -1
- package/dist/templates/mermaid-er.d.ts +35 -2
- package/dist/templates/mermaid-er.d.ts.map +1 -1
- package/dist/templates/mermaid-er.js +174 -7
- package/dist/templates/mermaid-er.js.map +1 -1
- package/dist/templates/output-parser.d.ts.map +1 -1
- package/dist/templates/output-parser.js +30 -79
- package/dist/templates/output-parser.js.map +1 -1
- package/dist/templates/output-prompt.d.ts.map +1 -1
- package/dist/templates/output-prompt.js +2 -2
- package/dist/templates/output-prompt.js.map +1 -1
- package/dist/templates/queries-file.d.ts.map +1 -1
- package/dist/templates/queries-file.js +112 -4
- package/dist/templates/queries-file.js.map +1 -1
- package/dist/templates/queries.d.ts +5 -0
- package/dist/templates/queries.d.ts.map +1 -1
- package/dist/templates/queries.js +7 -7
- package/dist/templates/queries.js.map +1 -1
- package/dist/templates/recover-schema-emitter.d.ts +8 -0
- package/dist/templates/recover-schema-emitter.d.ts.map +1 -0
- package/dist/templates/recover-schema-emitter.js +64 -0
- package/dist/templates/recover-schema-emitter.js.map +1 -0
- package/dist/templates/relations-block.js +10 -0
- package/dist/templates/relations-block.js.map +1 -1
- package/dist/templates/render-helper.d.ts.map +1 -1
- package/dist/templates/render-helper.js +4 -4
- package/dist/templates/render-helper.js.map +1 -1
- package/dist/templates/routes-file.d.ts.map +1 -1
- package/dist/templates/routes-file.js +183 -6
- package/dist/templates/routes-file.js.map +1 -1
- package/dist/templates/tph-discriminator.d.ts +56 -0
- package/dist/templates/tph-discriminator.d.ts.map +1 -0
- package/dist/templates/tph-discriminator.js +180 -0
- package/dist/templates/tph-discriminator.js.map +1 -0
- package/dist/templates/value-object-file.d.ts +2 -1
- package/dist/templates/value-object-file.d.ts.map +1 -1
- package/dist/templates/value-object-file.js +32 -4
- package/dist/templates/value-object-file.js.map +1 -1
- package/dist/templates/zod-validators.d.ts +64 -1
- package/dist/templates/zod-validators.d.ts.map +1 -1
- package/dist/templates/zod-validators.js +181 -8
- package/dist/templates/zod-validators.js.map +1 -1
- package/package.json +103 -34
- package/src/column-mapper.ts +25 -8
- package/src/constants.ts +18 -0
- package/src/docs-paths.ts +128 -0
- package/src/enum-import.ts +43 -0
- package/src/enum-shared.ts +95 -0
- package/src/generator-registry.ts +204 -0
- package/src/generator.ts +6 -0
- package/src/generators/api-doc-render.ts +572 -0
- package/src/generators/api-docs-file.ts +146 -0
- package/src/generators/api-field-shape.ts +114 -0
- package/src/generators/api-label.ts +7 -0
- package/src/generators/api-model.ts +1067 -0
- package/src/generators/docs-data-builder.ts +479 -185
- package/src/generators/docs-data.ts +139 -28
- package/src/generators/docs-file.ts +205 -39
- package/src/generators/entity-file.ts +31 -15
- package/src/generators/extractor-file.ts +2 -1
- package/src/generators/field-anchor.ts +24 -0
- package/src/generators/index.ts +8 -1
- package/src/generators/mermaid-er.ts +14 -0
- package/src/generators/output-parser-file.ts +3 -4
- package/src/generators/output-prompt-file.ts +2 -1
- package/src/generators/prompt-render-file.ts +3 -4
- package/src/generators/queries-file.ts +9 -3
- package/src/generators/render-helper-file.ts +2 -1
- package/src/generators/routes-file-hono.ts +5 -1
- package/src/generators/routes-file.ts +7 -1
- package/src/generators/template-doc-builder.ts +306 -0
- package/src/generators/template-doc-data.ts +85 -0
- package/src/generators/template-payload-tree.ts +71 -0
- package/src/generators/template-source-annotate.ts +290 -0
- package/src/generators/template-source-render.ts +203 -0
- package/src/generators/trace-helper-file.ts +301 -0
- package/src/index.ts +55 -4
- package/src/metaobjects-config.ts +117 -2
- package/src/naming.ts +48 -0
- package/src/payload-codegen.ts +14 -3
- package/src/projection/extract-view-spec.ts +49 -30
- package/src/relation-resolver.ts +103 -1
- package/src/render-context.ts +4 -0
- package/src/render-engine/embedded-templates.generated.ts +14 -0
- package/src/render-engine/framework-provider.ts +25 -11
- package/src/runner.ts +21 -0
- package/src/templates/docs-file.ts +2 -9
- package/src/templates/drizzle-schema.ts +31 -1
- package/src/templates/entity-constants.ts +1 -1
- package/src/templates/entity-file.ts +16 -5
- package/src/templates/enums-file.ts +50 -0
- package/src/templates/extract-delegate-emitter.ts +5 -6
- package/src/templates/extractor.ts +68 -38
- package/src/templates/field-meta.ts +0 -6
- package/src/templates/filter-allowlist.ts +17 -10
- package/src/templates/filter-type.ts +8 -6
- package/src/templates/find-templates.ts +15 -0
- package/src/templates/fr010-field-mapping.ts +10 -8
- package/src/templates/inferred-types.ts +108 -18
- package/src/templates/mermaid-er.ts +176 -8
- package/src/templates/output-parser.ts +30 -79
- package/src/templates/output-prompt.ts +2 -1
- package/src/templates/queries-file.ts +132 -3
- package/src/templates/queries.ts +15 -7
- package/src/templates/relations-block.ts +17 -0
- package/src/templates/render-helper.ts +4 -3
- package/src/templates/routes-file.ts +233 -6
- package/src/templates/tph-discriminator.ts +232 -0
- package/src/templates/value-object-file.ts +38 -4
- package/src/templates/zod-validators.ts +204 -7
- package/templates/api/agent-api.md.mustache +30 -0
- package/templates/api/entity-api.md.mustache +69 -0
- package/templates/api/index.md.mustache +21 -0
- package/templates/docs/entity-page.md.mustache +33 -21
- package/templates/docs/template-page.md.mustache +56 -0
- package/src/templates/extract-schema-emitter.ts +0 -111
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
// apiDocsFile() — the `api-docs` GENERATOR (ADR-0022 Part 3, Tier-1 NATIVE).
|
|
2
|
+
//
|
|
3
|
+
// It documents the PUBLIC API surface the OTHER generators emit for a model —
|
|
4
|
+
// the generated code's own API, in two forms (a human reference + a condensed
|
|
5
|
+
// agent form). It is a thin wiring layer: it REUSES, without re-derivation,
|
|
6
|
+
// • Task-1's buildApiModel() — the accurate-by-construction IR, and
|
|
7
|
+
// • Task-2's renderEntityApiPage / renderApiIndex / renderAgentApi renderers,
|
|
8
|
+
// • the shared docs-paths placement (collision-safe, layout-aware) that
|
|
9
|
+
// docsFile() uses — so multi-package models never silently overwrite a page.
|
|
10
|
+
//
|
|
11
|
+
// Output (all under `docs/api/`):
|
|
12
|
+
// • one `<Node>.md` per entity + template.output unit (the human page),
|
|
13
|
+
// • `README.md` — the consolidated human index (GitHub treats it as landing),
|
|
14
|
+
// • `AGENT-API.md` — the token-frugal agent form.
|
|
15
|
+
//
|
|
16
|
+
// Unlike `docs`/`mermaid-er` (Tier-2 neutral, owned by `meta docs`), `api-docs`
|
|
17
|
+
// is a NATIVE generator: it is REGISTERED in the generator registry (ADR-0022
|
|
18
|
+
// Part 3) — so it appears in `gen --list` and is selectable by its stable name
|
|
19
|
+
// `api-docs`. It is NOT a `meta docs` mode.
|
|
20
|
+
//
|
|
21
|
+
// It is NOT (yet) part of the default `meta gen` scaffold suite: it is
|
|
22
|
+
// registry-listed but not auto-run. Turning it on by default in the scaffold,
|
|
23
|
+
// and surfacing the agent form (AGENT-API.md) to a coding agent via a pointer
|
|
24
|
+
// from the installed `.metaobjects/` context, are tracked as agent-context-
|
|
25
|
+
// coordination follow-ups — deliberately deferred here to avoid colliding with
|
|
26
|
+
// the live agent-context work.
|
|
27
|
+
|
|
28
|
+
import type { MetaObject } from "@metaobjectsdev/metadata";
|
|
29
|
+
import type { Generator, GeneratorFactory, EmittedFile } from "../generator.js";
|
|
30
|
+
import {
|
|
31
|
+
docPageOutputPath,
|
|
32
|
+
surfaceCrossHref,
|
|
33
|
+
assertNoDuplicateDocPaths,
|
|
34
|
+
type DocPageNode,
|
|
35
|
+
type DocPagePlacement,
|
|
36
|
+
} from "../docs-paths.js";
|
|
37
|
+
import { projectProvider } from "../render-engine/framework-provider.js";
|
|
38
|
+
import { buildApiModel } from "./api-model.js";
|
|
39
|
+
import {
|
|
40
|
+
renderEntityApiPage,
|
|
41
|
+
renderApiIndex,
|
|
42
|
+
renderAgentApi,
|
|
43
|
+
} from "./api-doc-render.js";
|
|
44
|
+
|
|
45
|
+
// All api-docs artifacts live under this sub-directory of the codegen out dir.
|
|
46
|
+
// Per-unit pages fold further under their package path (package layout); the
|
|
47
|
+
// index + agent form stay at the api root (their links are computed relative to
|
|
48
|
+
// it via the same docPageHref the renderers use).
|
|
49
|
+
//
|
|
50
|
+
// The DEFAULT prefix is `docs/api` — byte-identical to the historical `meta gen`
|
|
51
|
+
// behaviour + goldens. The unified `meta docs` command (which writes everything
|
|
52
|
+
// under one docs root, `./docs`) overrides it via `subDir:'api'` so the api
|
|
53
|
+
// surface emits `api/<Node>.md` rather than doubling to `./docs/docs/api`.
|
|
54
|
+
const DEFAULT_API_DIR = "docs/api";
|
|
55
|
+
|
|
56
|
+
export interface ApiDocsFileOpts {
|
|
57
|
+
filter?: (entity: MetaObject) => boolean;
|
|
58
|
+
target?: string;
|
|
59
|
+
/** Output prefix for all api-docs artifacts. Default `docs/api`. */
|
|
60
|
+
subDir?: string;
|
|
61
|
+
/** When true, the model surface is emitted alongside the api surface (at the
|
|
62
|
+
* docs root), so each api entity page cross-links back to its model page. The
|
|
63
|
+
* href is computed via the shared `surfaceCrossHref` so it resolves in BOTH
|
|
64
|
+
* layouts. ABSENT/false ⇒ default api output byte-identical. */
|
|
65
|
+
modelSurface?: boolean;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* @deprecated ADR-0025: `meta docs` is the single door for ALL docs. `apiDocsFile()`
|
|
70
|
+
* stays as the INTERNAL engine of the docs door's api surface — do NOT add it to a
|
|
71
|
+
* `meta gen` config / the generators array. Use `meta docs` (it emits the api surface
|
|
72
|
+
* alongside the model surface). A `meta gen` config that lists it is warned + skipped.
|
|
73
|
+
*/
|
|
74
|
+
export const apiDocsFile = function apiDocsFile(opts?: ApiDocsFileOpts): Generator {
|
|
75
|
+
const generator: Generator = {
|
|
76
|
+
name: "api-docs",
|
|
77
|
+
generate(ctx) {
|
|
78
|
+
const provider = projectProvider(ctx.projectRoot ?? process.cwd());
|
|
79
|
+
const layout = ctx.config.outputLayout ?? "flat";
|
|
80
|
+
|
|
81
|
+
// Per-call output prefix. Default `docs/api`; `meta docs` passes `api`.
|
|
82
|
+
const apiDir = opts?.subDir ?? DEFAULT_API_DIR;
|
|
83
|
+
const indexFilename = `${apiDir}/README.md`;
|
|
84
|
+
const agentFilename = `${apiDir}/AGENT-API.md`;
|
|
85
|
+
|
|
86
|
+
// ONE ApiModel feeds every form (Task-1 builder; Task-2 renderers). The
|
|
87
|
+
// pkMap is reused from the run's renderContext when present (the real gen
|
|
88
|
+
// run always provides it) and derived otherwise.
|
|
89
|
+
// Auto-detect: document the OPT-IN Hono CRUD surface iff the Hono routes
|
|
90
|
+
// generator is actually in the run. The runner aggregates each generator's
|
|
91
|
+
// `emitsHonoRoutes` marker into ctx.config.includeHonoRoutes, so api-docs
|
|
92
|
+
// "just works" — it documents Hono exactly when `routesFileHono` is wired,
|
|
93
|
+
// and omits it (Fastify-only) otherwise. No explicit opt needed.
|
|
94
|
+
const model = buildApiModel(ctx.loadedRoot, {
|
|
95
|
+
loadedRoot: ctx.loadedRoot,
|
|
96
|
+
outputLayout: layout,
|
|
97
|
+
includeHonoRoutes: ctx.config.includeHonoRoutes ?? false,
|
|
98
|
+
...(ctx.renderContext?.pkMap !== undefined && { pkMap: ctx.renderContext.pkMap }),
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
// Track (path, fqn) for the SAME hard collision backstop docsFile() uses —
|
|
102
|
+
// two units that resolve to one path (flat, cross-package short-name clash)
|
|
103
|
+
// throw rather than silently overwrite.
|
|
104
|
+
const placements: DocPagePlacement[] = [];
|
|
105
|
+
|
|
106
|
+
// Per-unit human page. Placement is collision-safe via docPageOutputPath
|
|
107
|
+
// off {name, effective package}, prefixed under the api dir; the index
|
|
108
|
+
// (renderApiIndex) computes its links from the SAME {name, package}, so a
|
|
109
|
+
// link always points at the page's real location in BOTH layouts.
|
|
110
|
+
const files: EmittedFile[] = model.units.map((unit) => {
|
|
111
|
+
const node: DocPageNode = { name: unit.node, package: unit.package };
|
|
112
|
+
const path = `${apiDir}/${docPageOutputPath(layout, node)}`;
|
|
113
|
+
placements.push({ path, fqn: unit.package ? `${unit.package}::${unit.node}` : unit.node });
|
|
114
|
+
// Cross-link back to the sibling model page, when emitted. The model page
|
|
115
|
+
// lives at the docs root at `<placement>`; this api page lives at
|
|
116
|
+
// `<apiDir>/<placement>`. The href is derived from the SAME
|
|
117
|
+
// docPageOutputPath placement via surfaceCrossHref so it resolves in both
|
|
118
|
+
// layouts. ABSENT otherwise.
|
|
119
|
+
const modelPageHref = opts?.modelSurface
|
|
120
|
+
? surfaceCrossHref(path, docPageOutputPath(layout, node))
|
|
121
|
+
: undefined;
|
|
122
|
+
return { path, content: renderEntityApiPage(unit, provider, modelPageHref) };
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
// The consolidated human index (README.md) + the condensed agent form,
|
|
126
|
+
// both at the api root. Only emitted when at least one unit page exists.
|
|
127
|
+
if (files.length > 0) {
|
|
128
|
+
placements.push({ path: indexFilename, fqn: "<the api-docs index page>" });
|
|
129
|
+
placements.push({ path: agentFilename, fqn: "<the api-docs agent form>" });
|
|
130
|
+
files.unshift(
|
|
131
|
+
{ path: indexFilename, content: renderApiIndex(model, layout, provider) },
|
|
132
|
+
{ path: agentFilename, content: renderAgentApi(model, provider) },
|
|
133
|
+
);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// Hard backstop against silent overwrite (ALL layouts): throw naming both
|
|
137
|
+
// colliding FQNs + the path. Same guard docsFile() reuses.
|
|
138
|
+
assertNoDuplicateDocPaths(placements);
|
|
139
|
+
|
|
140
|
+
return files;
|
|
141
|
+
},
|
|
142
|
+
};
|
|
143
|
+
if (opts?.filter) generator.filter = opts.filter;
|
|
144
|
+
if (opts?.target) generator.target = opts.target;
|
|
145
|
+
return generator;
|
|
146
|
+
} as GeneratorFactory<ApiDocsFileOpts>;
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
// api-field-shape.ts — the field SHAPES (name + TS type + optionality) the
|
|
2
|
+
// api-docs renderers attach to a unit's model / create-payload / update-payload /
|
|
3
|
+
// extractor-payload symbols. ACCURATE BY CONSTRUCTION: every shape REUSES the
|
|
4
|
+
// real generators' own field walks so the documented fields can never drift from
|
|
5
|
+
// the emitted code (the api-docs accuracy gate enforces the field-set match):
|
|
6
|
+
//
|
|
7
|
+
// • model fields → entity-file's inferred type. The TS type comes from
|
|
8
|
+
// `fieldTsTypeString` (the SINGLE source of truth the value-object interface
|
|
9
|
+
// emitter uses) and the optional/nullable rule from `isFieldRequired` (the
|
|
10
|
+
// SAME rule the docs Storage/Constraints nullable column uses).
|
|
11
|
+
// • create-payload → `insertSchemaFields` (the EXACT field set + optionality
|
|
12
|
+
// the zod InsertSchema emitter walks: auto-gen PK omitted, @readOnly omitted,
|
|
13
|
+
// TPH discriminator pinned, @autoSet optional, else `fieldWillBeOptional`).
|
|
14
|
+
// • update-payload → `updateSchemaFields` (the UpdateSchema walk: TPH
|
|
15
|
+
// discriminator + @autoSet-onCreate omitted, everything else optional).
|
|
16
|
+
// • extractor-payload → the referenced @payloadRef value-object's field
|
|
17
|
+
// interface — the type `extract<Name>` returns — via the SAME
|
|
18
|
+
// `fieldTsTypeString` mapping the VO interface emitter uses.
|
|
19
|
+
|
|
20
|
+
import type { MetaObject, MetaField, MetaRoot } from "@metaobjectsdev/metadata";
|
|
21
|
+
import { fieldTsTypeString } from "../templates/inferred-types.js";
|
|
22
|
+
import {
|
|
23
|
+
insertSchemaFields,
|
|
24
|
+
updateSchemaFields,
|
|
25
|
+
type SchemaFieldShape,
|
|
26
|
+
} from "../templates/zod-validators.js";
|
|
27
|
+
import { isFieldRequired } from "./docs-data-builder.js";
|
|
28
|
+
|
|
29
|
+
/** One documented field in a model / payload shape. Structured so BOTH renderers
|
|
30
|
+
* (the human field table + the agent inline shape) format it without re-deriving. */
|
|
31
|
+
export interface FieldShape {
|
|
32
|
+
/** The field name (the property key). */
|
|
33
|
+
name: string;
|
|
34
|
+
/** The TS type expression, e.g. `string`, `number`, `"active" | "archived"`,
|
|
35
|
+
* `Address[]`. Enum unions are inlined (self-contained for an agent). */
|
|
36
|
+
type: string;
|
|
37
|
+
/** Whether the property is optional in this shape (`name?: T`). */
|
|
38
|
+
optional: boolean;
|
|
39
|
+
/** A short note, e.g. `pinned "Bridge"` (TPH discriminator) or `server-set`
|
|
40
|
+
* (@autoSet). Undefined when there is nothing extra to say. */
|
|
41
|
+
note?: string;
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* The entity MODEL field shape — every field, with the value-object interface's
|
|
46
|
+
* TS type and the docs nullable rule (optional iff not required and not the PK).
|
|
47
|
+
* Mirrors `InferSelectModel` field-presence; the PK is reported required.
|
|
48
|
+
*/
|
|
49
|
+
export function modelFieldShapes(obj: MetaObject): FieldShape[] {
|
|
50
|
+
const pkNames = new Set<string>(obj.primaryIdentity()?.fields ?? []);
|
|
51
|
+
return obj.fields().map((field: MetaField): FieldShape => {
|
|
52
|
+
const isPk = pkNames.has(field.name);
|
|
53
|
+
const required = isPk || isFieldRequired(field);
|
|
54
|
+
return {
|
|
55
|
+
name: field.name,
|
|
56
|
+
type: fieldTsTypeString(obj.name, field),
|
|
57
|
+
optional: !required,
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
/** Resolve a schema-field walk (insert/update) into a documented shape by
|
|
63
|
+
* pairing each schema field with its TS type (same `fieldTsTypeString` map). A
|
|
64
|
+
* TPH-pinned discriminator field documents the literal as its type. */
|
|
65
|
+
function shapesFromSchemaFields(obj: MetaObject, walk: SchemaFieldShape[]): FieldShape[] {
|
|
66
|
+
const fieldByName = new Map<string, MetaField>(obj.fields().map((f) => [f.name, f]));
|
|
67
|
+
return walk.map((sf): FieldShape => {
|
|
68
|
+
const field = fieldByName.get(sf.name);
|
|
69
|
+
let type = field !== undefined ? fieldTsTypeString(obj.name, field) : "unknown";
|
|
70
|
+
let note: string | undefined;
|
|
71
|
+
if (sf.pinnedLiteral !== undefined) {
|
|
72
|
+
type = JSON.stringify(sf.pinnedLiteral);
|
|
73
|
+
note = `pinned ${type}`;
|
|
74
|
+
} else if (sf.autoSet === true) {
|
|
75
|
+
note = "server-set";
|
|
76
|
+
}
|
|
77
|
+
const shape: FieldShape = { name: sf.name, type, optional: sf.optional };
|
|
78
|
+
if (note !== undefined) shape.note = note;
|
|
79
|
+
return shape;
|
|
80
|
+
});
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
/** The create-payload (InsertSchema) field shape — what `create<Name>` / POST
|
|
84
|
+
* accepts. */
|
|
85
|
+
export function createFieldShapes(obj: MetaObject): FieldShape[] {
|
|
86
|
+
return shapesFromSchemaFields(obj, insertSchemaFields(obj));
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/** The update-payload (UpdateSchema) field shape — what `update<Name>` / PATCH
|
|
90
|
+
* accepts (typically all-optional partial). */
|
|
91
|
+
export function updateFieldShapes(obj: MetaObject): FieldShape[] {
|
|
92
|
+
return shapesFromSchemaFields(obj, updateSchemaFields(obj));
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* The extractor PAYLOAD field shape — the field interface of the value object
|
|
97
|
+
* `extract<Name>` returns (the `@payloadRef` target). Same `fieldTsTypeString`
|
|
98
|
+
* mapping + `isFieldRequired` optionality the VO interface emitter uses, so it
|
|
99
|
+
* matches the emitted payload type. Returns undefined when the ref does not
|
|
100
|
+
* resolve to a loaded object.
|
|
101
|
+
*/
|
|
102
|
+
export function payloadFieldShapes(root: MetaRoot, payloadRef: string): FieldShape[] | undefined {
|
|
103
|
+
const vo = root.findObject(payloadRef);
|
|
104
|
+
if (vo === undefined) return undefined;
|
|
105
|
+
return modelFieldShapes(vo);
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
/** Format a list of field shapes as a compact inline TS object type, e.g.
|
|
109
|
+
* `{ name: string; status?: "active" | "archived" }`. Empty list → `{}`. */
|
|
110
|
+
export function inlineShape(fields: FieldShape[]): string {
|
|
111
|
+
if (fields.length === 0) return "{}";
|
|
112
|
+
const parts = fields.map((f) => `${f.name}${f.optional ? "?" : ""}: ${f.type}`);
|
|
113
|
+
return `{ ${parts.join("; ")} }`;
|
|
114
|
+
}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
const LABELS: Record<string, string> = {
|
|
2
|
+
ts: "TypeScript", java: "Java", kotlin: "Kotlin", csharp: "C#", python: "Python",
|
|
3
|
+
};
|
|
4
|
+
/** Human label for an api-surface language key. Unknown → capitalized verbatim. */
|
|
5
|
+
export function apiLabel(lang: string): string {
|
|
6
|
+
return LABELS[lang] ?? (lang.length ? lang[0]!.toUpperCase() + lang.slice(1) : lang);
|
|
7
|
+
}
|