@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.
Files changed (323) hide show
  1. package/README.md +1 -1
  2. package/dist/column-mapper.d.ts.map +1 -1
  3. package/dist/column-mapper.js +24 -8
  4. package/dist/column-mapper.js.map +1 -1
  5. package/dist/constants.d.ts +8 -0
  6. package/dist/constants.d.ts.map +1 -1
  7. package/dist/constants.js +16 -0
  8. package/dist/constants.js.map +1 -1
  9. package/dist/docs-paths.d.ts +58 -0
  10. package/dist/docs-paths.d.ts.map +1 -0
  11. package/dist/docs-paths.js +89 -0
  12. package/dist/docs-paths.js.map +1 -0
  13. package/dist/enum-import.d.ts +14 -0
  14. package/dist/enum-import.d.ts.map +1 -0
  15. package/dist/enum-import.js +35 -0
  16. package/dist/enum-import.js.map +1 -0
  17. package/dist/enum-shared.d.ts +32 -0
  18. package/dist/enum-shared.d.ts.map +1 -0
  19. package/dist/enum-shared.js +83 -0
  20. package/dist/enum-shared.js.map +1 -0
  21. package/dist/generator-registry.d.ts +22 -0
  22. package/dist/generator-registry.d.ts.map +1 -0
  23. package/dist/generator-registry.js +161 -0
  24. package/dist/generator-registry.js.map +1 -0
  25. package/dist/generator.d.ts +6 -0
  26. package/dist/generator.d.ts.map +1 -1
  27. package/dist/generator.js.map +1 -1
  28. package/dist/generators/api-doc-render.d.ts +17 -0
  29. package/dist/generators/api-doc-render.d.ts.map +1 -0
  30. package/dist/generators/api-doc-render.js +431 -0
  31. package/dist/generators/api-doc-render.js.map +1 -0
  32. package/dist/generators/api-docs-file.d.ts +21 -0
  33. package/dist/generators/api-docs-file.d.ts.map +1 -0
  34. package/dist/generators/api-docs-file.js +112 -0
  35. package/dist/generators/api-docs-file.js.map +1 -0
  36. package/dist/generators/api-field-shape.d.ts +39 -0
  37. package/dist/generators/api-field-shape.d.ts.map +1 -0
  38. package/dist/generators/api-field-shape.js +92 -0
  39. package/dist/generators/api-field-shape.js.map +1 -0
  40. package/dist/generators/api-label.d.ts +3 -0
  41. package/dist/generators/api-label.d.ts.map +1 -0
  42. package/dist/generators/api-label.js +8 -0
  43. package/dist/generators/api-label.js.map +1 -0
  44. package/dist/generators/api-model.d.ts +122 -0
  45. package/dist/generators/api-model.d.ts.map +1 -0
  46. package/dist/generators/api-model.js +809 -0
  47. package/dist/generators/api-model.js.map +1 -0
  48. package/dist/generators/docs-data-builder.d.ts +26 -4
  49. package/dist/generators/docs-data-builder.d.ts.map +1 -1
  50. package/dist/generators/docs-data-builder.js +436 -164
  51. package/dist/generators/docs-data-builder.js.map +1 -1
  52. package/dist/generators/docs-data.d.ts +136 -27
  53. package/dist/generators/docs-data.d.ts.map +1 -1
  54. package/dist/generators/docs-data.js +1 -1
  55. package/dist/generators/docs-data.js.map +1 -1
  56. package/dist/generators/docs-file.d.ts +19 -0
  57. package/dist/generators/docs-file.d.ts.map +1 -1
  58. package/dist/generators/docs-file.js +154 -27
  59. package/dist/generators/docs-file.js.map +1 -1
  60. package/dist/generators/entity-file.d.ts.map +1 -1
  61. package/dist/generators/entity-file.js +29 -14
  62. package/dist/generators/entity-file.js.map +1 -1
  63. package/dist/generators/extractor-file.d.ts.map +1 -1
  64. package/dist/generators/extractor-file.js +2 -1
  65. package/dist/generators/extractor-file.js.map +1 -1
  66. package/dist/generators/field-anchor.d.ts +7 -0
  67. package/dist/generators/field-anchor.d.ts.map +1 -0
  68. package/dist/generators/field-anchor.js +23 -0
  69. package/dist/generators/field-anchor.js.map +1 -0
  70. package/dist/generators/index.d.ts +8 -1
  71. package/dist/generators/index.d.ts.map +1 -1
  72. package/dist/generators/index.js +6 -0
  73. package/dist/generators/index.js.map +1 -1
  74. package/dist/generators/mermaid-er.d.ts +14 -0
  75. package/dist/generators/mermaid-er.d.ts.map +1 -1
  76. package/dist/generators/mermaid-er.js +14 -0
  77. package/dist/generators/mermaid-er.js.map +1 -1
  78. package/dist/generators/output-parser-file.d.ts.map +1 -1
  79. package/dist/generators/output-parser-file.js +3 -4
  80. package/dist/generators/output-parser-file.js.map +1 -1
  81. package/dist/generators/output-prompt-file.d.ts.map +1 -1
  82. package/dist/generators/output-prompt-file.js +2 -2
  83. package/dist/generators/output-prompt-file.js.map +1 -1
  84. package/dist/generators/prompt-render-file.d.ts.map +1 -1
  85. package/dist/generators/prompt-render-file.js +3 -4
  86. package/dist/generators/prompt-render-file.js.map +1 -1
  87. package/dist/generators/queries-file.d.ts.map +1 -1
  88. package/dist/generators/queries-file.js +8 -3
  89. package/dist/generators/queries-file.js.map +1 -1
  90. package/dist/generators/render-helper-file.d.ts.map +1 -1
  91. package/dist/generators/render-helper-file.js +2 -2
  92. package/dist/generators/render-helper-file.js.map +1 -1
  93. package/dist/generators/routes-file-hono.d.ts.map +1 -1
  94. package/dist/generators/routes-file-hono.js +5 -1
  95. package/dist/generators/routes-file-hono.js.map +1 -1
  96. package/dist/generators/routes-file.d.ts +3 -0
  97. package/dist/generators/routes-file.d.ts.map +1 -1
  98. package/dist/generators/routes-file.js +6 -1
  99. package/dist/generators/routes-file.js.map +1 -1
  100. package/dist/generators/template-doc-builder.d.ts +19 -0
  101. package/dist/generators/template-doc-builder.d.ts.map +1 -0
  102. package/dist/generators/template-doc-builder.js +220 -0
  103. package/dist/generators/template-doc-builder.js.map +1 -0
  104. package/dist/generators/template-doc-data.d.ts +62 -0
  105. package/dist/generators/template-doc-data.d.ts.map +1 -0
  106. package/dist/generators/template-doc-data.js +16 -0
  107. package/dist/generators/template-doc-data.js.map +1 -0
  108. package/dist/generators/template-payload-tree.d.ts +15 -0
  109. package/dist/generators/template-payload-tree.d.ts.map +1 -0
  110. package/dist/generators/template-payload-tree.js +61 -0
  111. package/dist/generators/template-payload-tree.js.map +1 -0
  112. package/dist/generators/template-source-annotate.d.ts +74 -0
  113. package/dist/generators/template-source-annotate.d.ts.map +1 -0
  114. package/dist/generators/template-source-annotate.js +184 -0
  115. package/dist/generators/template-source-annotate.js.map +1 -0
  116. package/dist/generators/template-source-render.d.ts +24 -0
  117. package/dist/generators/template-source-render.d.ts.map +1 -0
  118. package/dist/generators/template-source-render.js +175 -0
  119. package/dist/generators/template-source-render.js.map +1 -0
  120. package/dist/generators/trace-helper-file.d.ts +9 -0
  121. package/dist/generators/trace-helper-file.d.ts.map +1 -0
  122. package/dist/generators/trace-helper-file.js +196 -0
  123. package/dist/generators/trace-helper-file.js.map +1 -0
  124. package/dist/index.d.ts +29 -4
  125. package/dist/index.d.ts.map +1 -1
  126. package/dist/index.js +28 -2
  127. package/dist/index.js.map +1 -1
  128. package/dist/metaobjects-config.d.ts +75 -2
  129. package/dist/metaobjects-config.d.ts.map +1 -1
  130. package/dist/metaobjects-config.js +43 -0
  131. package/dist/metaobjects-config.js.map +1 -1
  132. package/dist/naming.d.ts +19 -0
  133. package/dist/naming.d.ts.map +1 -1
  134. package/dist/naming.js +41 -0
  135. package/dist/naming.js.map +1 -1
  136. package/dist/payload-codegen.d.ts.map +1 -1
  137. package/dist/payload-codegen.js +12 -4
  138. package/dist/payload-codegen.js.map +1 -1
  139. package/dist/projection/extract-view-spec.d.ts.map +1 -1
  140. package/dist/projection/extract-view-spec.js +51 -25
  141. package/dist/projection/extract-view-spec.js.map +1 -1
  142. package/dist/relation-resolver.d.ts +16 -0
  143. package/dist/relation-resolver.d.ts.map +1 -1
  144. package/dist/relation-resolver.js +82 -1
  145. package/dist/relation-resolver.js.map +1 -1
  146. package/dist/render-context.d.ts +4 -0
  147. package/dist/render-context.d.ts.map +1 -1
  148. package/dist/render-context.js.map +1 -1
  149. package/dist/render-engine/embedded-templates.generated.d.ts +2 -0
  150. package/dist/render-engine/embedded-templates.generated.d.ts.map +1 -0
  151. package/dist/render-engine/embedded-templates.generated.js +15 -0
  152. package/dist/render-engine/embedded-templates.generated.js.map +1 -0
  153. package/dist/render-engine/framework-provider.d.ts.map +1 -1
  154. package/dist/render-engine/framework-provider.js +26 -13
  155. package/dist/render-engine/framework-provider.js.map +1 -1
  156. package/dist/runner.d.ts.map +1 -1
  157. package/dist/runner.js +17 -0
  158. package/dist/runner.js.map +1 -1
  159. package/dist/templates/docs-file.d.ts +2 -6
  160. package/dist/templates/docs-file.d.ts.map +1 -1
  161. package/dist/templates/docs-file.js +2 -5
  162. package/dist/templates/docs-file.js.map +1 -1
  163. package/dist/templates/drizzle-schema.d.ts.map +1 -1
  164. package/dist/templates/drizzle-schema.js +30 -2
  165. package/dist/templates/drizzle-schema.js.map +1 -1
  166. package/dist/templates/entity-constants.d.ts +7 -0
  167. package/dist/templates/entity-constants.d.ts.map +1 -1
  168. package/dist/templates/entity-constants.js +1 -1
  169. package/dist/templates/entity-constants.js.map +1 -1
  170. package/dist/templates/entity-file.d.ts.map +1 -1
  171. package/dist/templates/entity-file.js +16 -5
  172. package/dist/templates/entity-file.js.map +1 -1
  173. package/dist/templates/enums-file.d.ts +11 -0
  174. package/dist/templates/enums-file.d.ts.map +1 -0
  175. package/dist/templates/enums-file.js +44 -0
  176. package/dist/templates/enums-file.js.map +1 -0
  177. package/dist/templates/extract-delegate-emitter.d.ts.map +1 -1
  178. package/dist/templates/extract-delegate-emitter.js +5 -7
  179. package/dist/templates/extract-delegate-emitter.js.map +1 -1
  180. package/dist/templates/extract-schema-emitter.d.ts.map +1 -1
  181. package/dist/templates/extract-schema-emitter.js +5 -1
  182. package/dist/templates/extract-schema-emitter.js.map +1 -1
  183. package/dist/templates/extractor.d.ts.map +1 -1
  184. package/dist/templates/extractor.js +56 -39
  185. package/dist/templates/extractor.js.map +1 -1
  186. package/dist/templates/field-meta.d.ts.map +1 -1
  187. package/dist/templates/field-meta.js +1 -5
  188. package/dist/templates/field-meta.js.map +1 -1
  189. package/dist/templates/filter-allowlist.d.ts +7 -2
  190. package/dist/templates/filter-allowlist.d.ts.map +1 -1
  191. package/dist/templates/filter-allowlist.js +17 -9
  192. package/dist/templates/filter-allowlist.js.map +1 -1
  193. package/dist/templates/filter-type.d.ts +7 -1
  194. package/dist/templates/filter-type.d.ts.map +1 -1
  195. package/dist/templates/filter-type.js +9 -5
  196. package/dist/templates/filter-type.js.map +1 -1
  197. package/dist/templates/find-templates.d.ts +4 -0
  198. package/dist/templates/find-templates.d.ts.map +1 -0
  199. package/dist/templates/find-templates.js +15 -0
  200. package/dist/templates/find-templates.js.map +1 -0
  201. package/dist/templates/fr010-field-mapping.d.ts +2 -0
  202. package/dist/templates/fr010-field-mapping.d.ts.map +1 -1
  203. package/dist/templates/fr010-field-mapping.js +10 -6
  204. package/dist/templates/fr010-field-mapping.js.map +1 -1
  205. package/dist/templates/inferred-types.d.ts +44 -7
  206. package/dist/templates/inferred-types.d.ts.map +1 -1
  207. package/dist/templates/inferred-types.js +107 -16
  208. package/dist/templates/inferred-types.js.map +1 -1
  209. package/dist/templates/mermaid-er.d.ts +35 -2
  210. package/dist/templates/mermaid-er.d.ts.map +1 -1
  211. package/dist/templates/mermaid-er.js +174 -7
  212. package/dist/templates/mermaid-er.js.map +1 -1
  213. package/dist/templates/output-parser.d.ts.map +1 -1
  214. package/dist/templates/output-parser.js +30 -79
  215. package/dist/templates/output-parser.js.map +1 -1
  216. package/dist/templates/output-prompt.d.ts.map +1 -1
  217. package/dist/templates/output-prompt.js +2 -2
  218. package/dist/templates/output-prompt.js.map +1 -1
  219. package/dist/templates/queries-file.d.ts.map +1 -1
  220. package/dist/templates/queries-file.js +112 -4
  221. package/dist/templates/queries-file.js.map +1 -1
  222. package/dist/templates/queries.d.ts +5 -0
  223. package/dist/templates/queries.d.ts.map +1 -1
  224. package/dist/templates/queries.js +7 -7
  225. package/dist/templates/queries.js.map +1 -1
  226. package/dist/templates/recover-schema-emitter.d.ts +8 -0
  227. package/dist/templates/recover-schema-emitter.d.ts.map +1 -0
  228. package/dist/templates/recover-schema-emitter.js +64 -0
  229. package/dist/templates/recover-schema-emitter.js.map +1 -0
  230. package/dist/templates/relations-block.js +10 -0
  231. package/dist/templates/relations-block.js.map +1 -1
  232. package/dist/templates/render-helper.d.ts.map +1 -1
  233. package/dist/templates/render-helper.js +4 -4
  234. package/dist/templates/render-helper.js.map +1 -1
  235. package/dist/templates/routes-file.d.ts.map +1 -1
  236. package/dist/templates/routes-file.js +183 -6
  237. package/dist/templates/routes-file.js.map +1 -1
  238. package/dist/templates/tph-discriminator.d.ts +56 -0
  239. package/dist/templates/tph-discriminator.d.ts.map +1 -0
  240. package/dist/templates/tph-discriminator.js +180 -0
  241. package/dist/templates/tph-discriminator.js.map +1 -0
  242. package/dist/templates/value-object-file.d.ts +2 -1
  243. package/dist/templates/value-object-file.d.ts.map +1 -1
  244. package/dist/templates/value-object-file.js +32 -4
  245. package/dist/templates/value-object-file.js.map +1 -1
  246. package/dist/templates/zod-validators.d.ts +64 -1
  247. package/dist/templates/zod-validators.d.ts.map +1 -1
  248. package/dist/templates/zod-validators.js +181 -8
  249. package/dist/templates/zod-validators.js.map +1 -1
  250. package/package.json +103 -34
  251. package/src/column-mapper.ts +25 -8
  252. package/src/constants.ts +18 -0
  253. package/src/docs-paths.ts +128 -0
  254. package/src/enum-import.ts +43 -0
  255. package/src/enum-shared.ts +95 -0
  256. package/src/generator-registry.ts +204 -0
  257. package/src/generator.ts +6 -0
  258. package/src/generators/api-doc-render.ts +572 -0
  259. package/src/generators/api-docs-file.ts +146 -0
  260. package/src/generators/api-field-shape.ts +114 -0
  261. package/src/generators/api-label.ts +7 -0
  262. package/src/generators/api-model.ts +1067 -0
  263. package/src/generators/docs-data-builder.ts +479 -185
  264. package/src/generators/docs-data.ts +139 -28
  265. package/src/generators/docs-file.ts +205 -39
  266. package/src/generators/entity-file.ts +31 -15
  267. package/src/generators/extractor-file.ts +2 -1
  268. package/src/generators/field-anchor.ts +24 -0
  269. package/src/generators/index.ts +8 -1
  270. package/src/generators/mermaid-er.ts +14 -0
  271. package/src/generators/output-parser-file.ts +3 -4
  272. package/src/generators/output-prompt-file.ts +2 -1
  273. package/src/generators/prompt-render-file.ts +3 -4
  274. package/src/generators/queries-file.ts +9 -3
  275. package/src/generators/render-helper-file.ts +2 -1
  276. package/src/generators/routes-file-hono.ts +5 -1
  277. package/src/generators/routes-file.ts +7 -1
  278. package/src/generators/template-doc-builder.ts +306 -0
  279. package/src/generators/template-doc-data.ts +85 -0
  280. package/src/generators/template-payload-tree.ts +71 -0
  281. package/src/generators/template-source-annotate.ts +290 -0
  282. package/src/generators/template-source-render.ts +203 -0
  283. package/src/generators/trace-helper-file.ts +301 -0
  284. package/src/index.ts +55 -4
  285. package/src/metaobjects-config.ts +117 -2
  286. package/src/naming.ts +48 -0
  287. package/src/payload-codegen.ts +14 -3
  288. package/src/projection/extract-view-spec.ts +49 -30
  289. package/src/relation-resolver.ts +103 -1
  290. package/src/render-context.ts +4 -0
  291. package/src/render-engine/embedded-templates.generated.ts +14 -0
  292. package/src/render-engine/framework-provider.ts +25 -11
  293. package/src/runner.ts +21 -0
  294. package/src/templates/docs-file.ts +2 -9
  295. package/src/templates/drizzle-schema.ts +31 -1
  296. package/src/templates/entity-constants.ts +1 -1
  297. package/src/templates/entity-file.ts +16 -5
  298. package/src/templates/enums-file.ts +50 -0
  299. package/src/templates/extract-delegate-emitter.ts +5 -6
  300. package/src/templates/extractor.ts +68 -38
  301. package/src/templates/field-meta.ts +0 -6
  302. package/src/templates/filter-allowlist.ts +17 -10
  303. package/src/templates/filter-type.ts +8 -6
  304. package/src/templates/find-templates.ts +15 -0
  305. package/src/templates/fr010-field-mapping.ts +10 -8
  306. package/src/templates/inferred-types.ts +108 -18
  307. package/src/templates/mermaid-er.ts +176 -8
  308. package/src/templates/output-parser.ts +30 -79
  309. package/src/templates/output-prompt.ts +2 -1
  310. package/src/templates/queries-file.ts +132 -3
  311. package/src/templates/queries.ts +15 -7
  312. package/src/templates/relations-block.ts +17 -0
  313. package/src/templates/render-helper.ts +4 -3
  314. package/src/templates/routes-file.ts +233 -6
  315. package/src/templates/tph-discriminator.ts +232 -0
  316. package/src/templates/value-object-file.ts +38 -4
  317. package/src/templates/zod-validators.ts +204 -7
  318. package/templates/api/agent-api.md.mustache +30 -0
  319. package/templates/api/entity-api.md.mustache +69 -0
  320. package/templates/api/index.md.mustache +21 -0
  321. package/templates/docs/entity-page.md.mustache +33 -21
  322. package/templates/docs/template-page.md.mustache +56 -0
  323. 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
+ }