gazetta 0.4.0 → 0.6.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 (240) hide show
  1. package/admin-dist/assets/index-B6pVot0Y.css +1 -0
  2. package/admin-dist/assets/index-DniLwxJA.js +609 -0
  3. package/admin-dist/assets/{vendor-primevue-BnR1c_bQ.js → vendor-primevue-C0Q_YTCb.js} +330 -431
  4. package/admin-dist/assets/vendor-vue-D3wBSmDf.js +1 -0
  5. package/admin-dist/index.html +4 -4
  6. package/dist/admin-api/index.d.ts +26 -5
  7. package/dist/admin-api/index.d.ts.map +1 -1
  8. package/dist/admin-api/index.js +175 -13
  9. package/dist/admin-api/index.js.map +1 -1
  10. package/dist/admin-api/routes/compare.d.ts +3 -1
  11. package/dist/admin-api/routes/compare.d.ts.map +1 -1
  12. package/dist/admin-api/routes/compare.js +34 -24
  13. package/dist/admin-api/routes/compare.js.map +1 -1
  14. package/dist/admin-api/routes/fields.d.ts +2 -2
  15. package/dist/admin-api/routes/fields.d.ts.map +1 -1
  16. package/dist/admin-api/routes/fields.js +10 -3
  17. package/dist/admin-api/routes/fields.js.map +1 -1
  18. package/dist/admin-api/routes/fragments.d.ts +2 -2
  19. package/dist/admin-api/routes/fragments.d.ts.map +1 -1
  20. package/dist/admin-api/routes/fragments.js +94 -19
  21. package/dist/admin-api/routes/fragments.js.map +1 -1
  22. package/dist/admin-api/routes/history.d.ts +23 -0
  23. package/dist/admin-api/routes/history.d.ts.map +1 -0
  24. package/dist/admin-api/routes/history.js +143 -0
  25. package/dist/admin-api/routes/history.js.map +1 -0
  26. package/dist/admin-api/routes/pages.d.ts +2 -2
  27. package/dist/admin-api/routes/pages.d.ts.map +1 -1
  28. package/dist/admin-api/routes/pages.js +120 -20
  29. package/dist/admin-api/routes/pages.js.map +1 -1
  30. package/dist/admin-api/routes/preview.d.ts +2 -2
  31. package/dist/admin-api/routes/preview.d.ts.map +1 -1
  32. package/dist/admin-api/routes/preview.js +50 -15
  33. package/dist/admin-api/routes/preview.js.map +1 -1
  34. package/dist/admin-api/routes/publish.d.ts +3 -1
  35. package/dist/admin-api/routes/publish.d.ts.map +1 -1
  36. package/dist/admin-api/routes/publish.js +306 -65
  37. package/dist/admin-api/routes/publish.js.map +1 -1
  38. package/dist/admin-api/routes/site.d.ts +2 -2
  39. package/dist/admin-api/routes/site.d.ts.map +1 -1
  40. package/dist/admin-api/routes/site.js +27 -4
  41. package/dist/admin-api/routes/site.js.map +1 -1
  42. package/dist/admin-api/routes/templates.d.ts +2 -2
  43. package/dist/admin-api/routes/templates.d.ts.map +1 -1
  44. package/dist/admin-api/routes/templates.js +19 -9
  45. package/dist/admin-api/routes/templates.js.map +1 -1
  46. package/dist/admin-api/schemas/compare.d.ts +29 -0
  47. package/dist/admin-api/schemas/compare.d.ts.map +1 -0
  48. package/dist/admin-api/schemas/compare.js +30 -0
  49. package/dist/admin-api/schemas/compare.js.map +1 -0
  50. package/dist/admin-api/schemas/dependents.d.ts +15 -0
  51. package/dist/admin-api/schemas/dependents.d.ts.map +1 -0
  52. package/dist/admin-api/schemas/dependents.js +14 -0
  53. package/dist/admin-api/schemas/dependents.js.map +1 -0
  54. package/dist/admin-api/schemas/fetch.d.ts +12 -0
  55. package/dist/admin-api/schemas/fetch.d.ts.map +1 -0
  56. package/dist/admin-api/schemas/fetch.js +11 -0
  57. package/dist/admin-api/schemas/fetch.js.map +1 -0
  58. package/dist/admin-api/schemas/fields.d.ts +11 -0
  59. package/dist/admin-api/schemas/fields.d.ts.map +1 -0
  60. package/dist/admin-api/schemas/fields.js +11 -0
  61. package/dist/admin-api/schemas/fields.js.map +1 -0
  62. package/dist/admin-api/schemas/fragments.d.ts +27 -0
  63. package/dist/admin-api/schemas/fragments.d.ts.map +1 -0
  64. package/dist/admin-api/schemas/fragments.js +26 -0
  65. package/dist/admin-api/schemas/fragments.js.map +1 -0
  66. package/dist/admin-api/schemas/history.d.ts +73 -0
  67. package/dist/admin-api/schemas/history.d.ts.map +1 -0
  68. package/dist/admin-api/schemas/history.js +35 -0
  69. package/dist/admin-api/schemas/history.js.map +1 -0
  70. package/dist/admin-api/schemas/index.d.ts +32 -0
  71. package/dist/admin-api/schemas/index.d.ts.map +1 -0
  72. package/dist/admin-api/schemas/index.js +32 -0
  73. package/dist/admin-api/schemas/index.js.map +1 -0
  74. package/dist/admin-api/schemas/pages.d.ts +46 -0
  75. package/dist/admin-api/schemas/pages.d.ts.map +1 -0
  76. package/dist/admin-api/schemas/pages.js +47 -0
  77. package/dist/admin-api/schemas/pages.js.map +1 -0
  78. package/dist/admin-api/schemas/publish.d.ts +67 -0
  79. package/dist/admin-api/schemas/publish.d.ts.map +1 -0
  80. package/dist/admin-api/schemas/publish.js +60 -0
  81. package/dist/admin-api/schemas/publish.js.map +1 -0
  82. package/dist/admin-api/schemas/site.d.ts +28 -0
  83. package/dist/admin-api/schemas/site.d.ts.map +1 -0
  84. package/dist/admin-api/schemas/site.js +24 -0
  85. package/dist/admin-api/schemas/site.js.map +1 -0
  86. package/dist/admin-api/schemas/targets.d.ts +36 -0
  87. package/dist/admin-api/schemas/targets.d.ts.map +1 -0
  88. package/dist/admin-api/schemas/targets.js +19 -0
  89. package/dist/admin-api/schemas/targets.js.map +1 -0
  90. package/dist/admin-api/schemas/templates.d.ts +17 -0
  91. package/dist/admin-api/schemas/templates.d.ts.map +1 -0
  92. package/dist/admin-api/schemas/templates.js +16 -0
  93. package/dist/admin-api/schemas/templates.js.map +1 -0
  94. package/dist/admin-api/source-context.d.ts +165 -0
  95. package/dist/admin-api/source-context.d.ts.map +1 -0
  96. package/dist/admin-api/source-context.js +95 -0
  97. package/dist/admin-api/source-context.js.map +1 -0
  98. package/dist/app.js +1 -1
  99. package/dist/app.js.map +1 -1
  100. package/dist/assemble.d.ts.map +1 -1
  101. package/dist/assemble.js +4 -1
  102. package/dist/assemble.js.map +1 -1
  103. package/dist/cli/bootstrap.d.ts +48 -0
  104. package/dist/cli/bootstrap.d.ts.map +1 -0
  105. package/dist/cli/bootstrap.js +85 -0
  106. package/dist/cli/bootstrap.js.map +1 -0
  107. package/dist/cli/history.d.ts +45 -0
  108. package/dist/cli/history.d.ts.map +1 -0
  109. package/dist/cli/history.js +165 -0
  110. package/dist/cli/history.js.map +1 -0
  111. package/dist/cli/index.js +691 -113
  112. package/dist/cli/index.js.map +1 -1
  113. package/dist/compare.d.ts +15 -5
  114. package/dist/compare.d.ts.map +1 -1
  115. package/dist/compare.js +83 -45
  116. package/dist/compare.js.map +1 -1
  117. package/dist/concurrency.d.ts +63 -0
  118. package/dist/concurrency.d.ts.map +1 -0
  119. package/dist/concurrency.js +134 -0
  120. package/dist/concurrency.js.map +1 -0
  121. package/dist/content-root.d.ts +38 -0
  122. package/dist/content-root.d.ts.map +1 -0
  123. package/dist/content-root.js +29 -0
  124. package/dist/content-root.js.map +1 -0
  125. package/dist/editor/mount.d.ts +1 -1
  126. package/dist/editor/mount.d.ts.map +1 -1
  127. package/dist/editor/mount.js +61 -29
  128. package/dist/editor/mount.js.map +1 -1
  129. package/dist/hash.d.ts +47 -1
  130. package/dist/hash.d.ts.map +1 -1
  131. package/dist/hash.js +107 -10
  132. package/dist/hash.js.map +1 -1
  133. package/dist/history-provider.d.ts +49 -0
  134. package/dist/history-provider.d.ts.map +1 -0
  135. package/dist/history-provider.js +226 -0
  136. package/dist/history-provider.js.map +1 -0
  137. package/dist/history-recorder.d.ts +98 -0
  138. package/dist/history-recorder.d.ts.map +1 -0
  139. package/dist/history-recorder.js +160 -0
  140. package/dist/history-recorder.js.map +1 -0
  141. package/dist/history-restorer.d.ts +46 -0
  142. package/dist/history-restorer.d.ts.map +1 -0
  143. package/dist/history-restorer.js +105 -0
  144. package/dist/history-restorer.js.map +1 -0
  145. package/dist/history.d.ts +111 -0
  146. package/dist/history.d.ts.map +1 -0
  147. package/dist/history.js +25 -0
  148. package/dist/history.js.map +1 -0
  149. package/dist/index.d.ts +26 -4
  150. package/dist/index.d.ts.map +1 -1
  151. package/dist/index.js +16 -3
  152. package/dist/index.js.map +1 -1
  153. package/dist/locale.d.ts +74 -0
  154. package/dist/locale.d.ts.map +1 -0
  155. package/dist/locale.js +150 -0
  156. package/dist/locale.js.map +1 -0
  157. package/dist/manifest.d.ts.map +1 -1
  158. package/dist/manifest.js +16 -1
  159. package/dist/manifest.js.map +1 -1
  160. package/dist/providers/azure-blob.d.ts.map +1 -1
  161. package/dist/providers/azure-blob.js.map +1 -1
  162. package/dist/providers/r2.d.ts.map +1 -1
  163. package/dist/providers/r2.js +7 -4
  164. package/dist/providers/r2.js.map +1 -1
  165. package/dist/providers/s3.d.ts.map +1 -1
  166. package/dist/providers/s3.js +23 -15
  167. package/dist/providers/s3.js.map +1 -1
  168. package/dist/publish-locale.d.ts +44 -0
  169. package/dist/publish-locale.d.ts.map +1 -0
  170. package/dist/publish-locale.js +103 -0
  171. package/dist/publish-locale.js.map +1 -0
  172. package/dist/publish-rendered.d.ts +17 -5
  173. package/dist/publish-rendered.d.ts.map +1 -1
  174. package/dist/publish-rendered.js +114 -66
  175. package/dist/publish-rendered.js.map +1 -1
  176. package/dist/publish.d.ts +39 -3
  177. package/dist/publish.d.ts.map +1 -1
  178. package/dist/publish.js +166 -17
  179. package/dist/publish.js.map +1 -1
  180. package/dist/renderer.d.ts +14 -4
  181. package/dist/renderer.d.ts.map +1 -1
  182. package/dist/renderer.js +35 -23
  183. package/dist/renderer.js.map +1 -1
  184. package/dist/resolver.d.ts +7 -2
  185. package/dist/resolver.d.ts.map +1 -1
  186. package/dist/resolver.js +66 -15
  187. package/dist/resolver.js.map +1 -1
  188. package/dist/robots.d.ts +22 -0
  189. package/dist/robots.d.ts.map +1 -0
  190. package/dist/robots.js +25 -0
  191. package/dist/robots.js.map +1 -0
  192. package/dist/seo.d.ts +56 -0
  193. package/dist/seo.d.ts.map +1 -0
  194. package/dist/seo.js +72 -0
  195. package/dist/seo.js.map +1 -0
  196. package/dist/serve.d.ts +41 -3
  197. package/dist/serve.d.ts.map +1 -1
  198. package/dist/serve.js +206 -65
  199. package/dist/serve.js.map +1 -1
  200. package/dist/sidecars.d.ts +60 -0
  201. package/dist/sidecars.d.ts.map +1 -0
  202. package/dist/sidecars.js +231 -0
  203. package/dist/sidecars.js.map +1 -0
  204. package/dist/site-loader.d.ts +74 -6
  205. package/dist/site-loader.d.ts.map +1 -1
  206. package/dist/site-loader.js +149 -36
  207. package/dist/site-loader.js.map +1 -1
  208. package/dist/sitemap.d.ts +45 -0
  209. package/dist/sitemap.d.ts.map +1 -0
  210. package/dist/sitemap.js +67 -0
  211. package/dist/sitemap.js.map +1 -0
  212. package/dist/source-sidecars.d.ts +32 -0
  213. package/dist/source-sidecars.d.ts.map +1 -0
  214. package/dist/source-sidecars.js +98 -0
  215. package/dist/source-sidecars.js.map +1 -0
  216. package/dist/targets.d.ts +47 -1
  217. package/dist/targets.d.ts.map +1 -1
  218. package/dist/targets.js +78 -9
  219. package/dist/targets.js.map +1 -1
  220. package/dist/template-loader.d.ts +7 -3
  221. package/dist/template-loader.d.ts.map +1 -1
  222. package/dist/template-loader.js +27 -12
  223. package/dist/template-loader.js.map +1 -1
  224. package/dist/templates-scan-worker.js +1 -1
  225. package/dist/templates-scan-worker.js.map +1 -1
  226. package/dist/templates-scan.d.ts.map +1 -1
  227. package/dist/templates-scan.js +1 -1
  228. package/dist/templates-scan.js.map +1 -1
  229. package/dist/types.d.ts +116 -9
  230. package/dist/types.d.ts.map +1 -1
  231. package/dist/types.js +28 -5
  232. package/dist/types.js.map +1 -1
  233. package/dist/workers/cloudflare-r2.d.ts +11 -2
  234. package/dist/workers/cloudflare-r2.d.ts.map +1 -1
  235. package/dist/workers/cloudflare-r2.js +120 -55
  236. package/dist/workers/cloudflare-r2.js.map +1 -1
  237. package/package.json +11 -2
  238. package/admin-dist/assets/index-Bh_y1d_l.css +0 -1
  239. package/admin-dist/assets/index-DjGNi6yy.js +0 -608
  240. package/admin-dist/assets/vendor-vue-DSjyxCX6.js +0 -1
package/dist/hash.d.ts CHANGED
@@ -1,8 +1,54 @@
1
1
  import type { FragmentManifest, PageManifest } from './types.js';
2
- export declare function sidecarNameFor(hash: string): string;
2
+ /** Build a hash sidecar filename, optionally locale-suffixed. */
3
+ export declare function sidecarNameFor(hash: string, locale?: string): string;
4
+ /** Parse a hash sidecar filename. Returns { hash, locale? } or null. */
3
5
  export declare function parseSidecarName(entryName: string): string | null;
6
+ /** Parse locale from a hash sidecar filename. */
7
+ export declare function parseSidecarLocale(entryName: string): string | undefined;
8
+ /**
9
+ * Filename-safe encoding for fragment/template names. Fragments can be
10
+ * subfolder-qualified (e.g. "buttons/primary"); we replace `/` with `.`
11
+ * so the name works as a filename component and stays readable in
12
+ * listings.
13
+ *
14
+ * Why `.`: per operations.md, refs are lowercase-kebab-case — `.` is
15
+ * explicitly avoided in ref names ("confuses URL routing"), so using
16
+ * it as the path separator in encoded form is collision-free with
17
+ * the legal input alphabet (letters, digits, `-`, `_`, `/`).
18
+ *
19
+ * The prior `/` ↔ `__` scheme was ambiguous for any input containing
20
+ * `_` (a legitimate character in names like `my_fragment`) — a lone
21
+ * `_` adjacent to `/` encoded to `___` and decoded ambiguously.
22
+ *
23
+ * We reject `.` in inputs at encode time. It's already documented as
24
+ * off-limits in ref names; enforcing it here turns a silent
25
+ * round-trip bug into a loud error for anyone who tries.
26
+ */
27
+ export declare function encodeRefName(name: string): string;
28
+ export declare function decodeRefName(name: string): string;
29
+ /** `.uses-header` for a page/fragment that references @header. */
30
+ export declare function usesSidecarNameFor(fragmentName: string): string;
31
+ export declare function parseUsesSidecarName(entryName: string): string | null;
32
+ /** `.tpl-page-default` for a page/fragment rendered with that template. */
33
+ export declare function templateSidecarNameFor(templateName: string): string;
34
+ export declare function parseTemplateSidecarName(entryName: string): string | null;
35
+ /** Compact ISO timestamp for `.pub-` sidecar filenames: `20260417T220000Z`. */
36
+ export declare function compactTimestamp(date?: Date): string;
37
+ /** Parse compact timestamp back to ISO string: `2026-04-17T22:00:00Z`. */
38
+ export declare function parseCompactTimestamp(compact: string): string;
39
+ export interface PubSidecar {
40
+ lastPublished: string;
41
+ noindex: boolean;
42
+ }
43
+ /** `.pub-20260417T220000Z`, `.pub-20260417T220000Z-noindex`, or `.pub-20260417T220000Z.fr` */
44
+ export declare function pubSidecarNameFor(date?: Date, noindex?: boolean, locale?: string): string;
45
+ export declare function parsePubSidecarName(entryName: string): PubSidecar | null;
46
+ /** Parse locale from a pub sidecar filename. */
47
+ export declare function parsePubSidecarLocale(entryName: string): string | undefined;
4
48
  export interface HashManifestOptions {
5
49
  templateHashes: Map<string, string>;
50
+ /** Fragment content hashes — include only for static-mode page hashing. */
51
+ fragmentHashes?: Map<string, string>;
6
52
  }
7
53
  /**
8
54
  * Compute the content hash for a page or fragment manifest.
@@ -1 +1 @@
1
- {"version":3,"file":"hash.d.ts","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAkB,gBAAgB,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAIhF,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAEnD;AAED,wBAAgB,gBAAgB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAGjE;AAuBD,MAAM,WAAW,mBAAmB;IAClC,cAAc,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CACpC;AAED;;;;;;;GAOG;AACH,wBAAgB,YAAY,CAC1B,QAAQ,EAAE,YAAY,GAAG,gBAAgB,EACzC,IAAI,EAAE,mBAAmB,GACxB,MAAM,CASR"}
1
+ {"version":3,"file":"hash.d.ts","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAkB,gBAAgB,EAAE,YAAY,EAAE,MAAM,YAAY,CAAA;AAOhF,iEAAiE;AACjE,wBAAgB,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,MAAM,CAEpE;AAED,wEAAwE;AACxE,wBAAgB,gBAAgB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAGjE;AAED,iDAAiD;AACjD,wBAAgB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAGxE;AAED;;;;;;;;;;;;;;;;;;GAkBG;AACH,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAQlD;AACD,wBAAgB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAElD;AAED,kEAAkE;AAClE,wBAAgB,kBAAkB,CAAC,YAAY,EAAE,MAAM,GAAG,MAAM,CAE/D;AACD,wBAAgB,oBAAoB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAGrE;AAED,2EAA2E;AAC3E,wBAAgB,sBAAsB,CAAC,YAAY,EAAE,MAAM,GAAG,MAAM,CAEnE;AACD,wBAAgB,wBAAwB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAGzE;AAED,+EAA+E;AAC/E,wBAAgB,gBAAgB,CAAC,IAAI,GAAE,IAAiB,GAAG,MAAM,CAKhE;AAED,0EAA0E;AAC1E,wBAAgB,qBAAqB,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAG7D;AAED,MAAM,WAAW,UAAU;IACzB,aAAa,EAAE,MAAM,CAAA;IACrB,OAAO,EAAE,OAAO,CAAA;CACjB;AAED,8FAA8F;AAC9F,wBAAgB,iBAAiB,CAAC,IAAI,GAAE,IAAiB,EAAE,OAAO,UAAQ,EAAE,MAAM,CAAC,EAAE,MAAM,GAAG,MAAM,CAGnG;AAED,wBAAgB,mBAAmB,CAAC,SAAS,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI,CAOxE;AAED,gDAAgD;AAChD,wBAAgB,qBAAqB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAG3E;AAmCD,MAAM,WAAW,mBAAmB;IAClC,cAAc,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;IACnC,2EAA2E;IAC3E,cAAc,CAAC,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CACrC;AAED;;;;;;;GAOG;AACH,wBAAgB,YAAY,CAAC,QAAQ,EAAE,YAAY,GAAG,gBAAgB,EAAE,IAAI,EAAE,mBAAmB,GAAG,MAAM,CASzG"}
package/dist/hash.js CHANGED
@@ -1,28 +1,125 @@
1
1
  import { createHash } from 'node:crypto';
2
- const SIDECAR_RE = /^\.([0-9a-f]{8})\.hash$/;
3
- export function sidecarNameFor(hash) {
4
- return `.${hash}.hash`;
2
+ const SIDECAR_RE = /^\.([0-9a-f]{8})\.hash(?:\.([a-z]{2}(?:-[a-z]+)?))?$/;
3
+ const USES_SIDECAR_RE = /^\.uses-(.+)$/;
4
+ const TPL_SIDECAR_RE = /^\.tpl-(.+)$/;
5
+ const PUB_SIDECAR_RE = /^\.pub-(\d{8}T\d{6}Z)(-noindex)?(?:\.([a-z]{2}(?:-[a-z]+)?))?$/;
6
+ /** Build a hash sidecar filename, optionally locale-suffixed. */
7
+ export function sidecarNameFor(hash, locale) {
8
+ return locale ? `.${hash}.hash.${locale}` : `.${hash}.hash`;
5
9
  }
10
+ /** Parse a hash sidecar filename. Returns { hash, locale? } or null. */
6
11
  export function parseSidecarName(entryName) {
7
12
  const m = SIDECAR_RE.exec(entryName);
8
13
  return m ? m[1] : null;
9
14
  }
15
+ /** Parse locale from a hash sidecar filename. */
16
+ export function parseSidecarLocale(entryName) {
17
+ const m = SIDECAR_RE.exec(entryName);
18
+ return m?.[2] ?? undefined;
19
+ }
20
+ /**
21
+ * Filename-safe encoding for fragment/template names. Fragments can be
22
+ * subfolder-qualified (e.g. "buttons/primary"); we replace `/` with `.`
23
+ * so the name works as a filename component and stays readable in
24
+ * listings.
25
+ *
26
+ * Why `.`: per operations.md, refs are lowercase-kebab-case — `.` is
27
+ * explicitly avoided in ref names ("confuses URL routing"), so using
28
+ * it as the path separator in encoded form is collision-free with
29
+ * the legal input alphabet (letters, digits, `-`, `_`, `/`).
30
+ *
31
+ * The prior `/` ↔ `__` scheme was ambiguous for any input containing
32
+ * `_` (a legitimate character in names like `my_fragment`) — a lone
33
+ * `_` adjacent to `/` encoded to `___` and decoded ambiguously.
34
+ *
35
+ * We reject `.` in inputs at encode time. It's already documented as
36
+ * off-limits in ref names; enforcing it here turns a silent
37
+ * round-trip bug into a loud error for anyone who tries.
38
+ */
39
+ export function encodeRefName(name) {
40
+ if (name.includes('.')) {
41
+ throw new Error(`Invalid reference name "${name}": dot is reserved for path encoding. ` +
42
+ `Use lowercase-kebab-case with / for subfolders (e.g. "buttons/primary").`);
43
+ }
44
+ return name.replace(/\//g, '.');
45
+ }
46
+ export function decodeRefName(name) {
47
+ return name.replace(/\./g, '/');
48
+ }
49
+ /** `.uses-header` for a page/fragment that references @header. */
50
+ export function usesSidecarNameFor(fragmentName) {
51
+ return `.uses-${encodeRefName(fragmentName)}`;
52
+ }
53
+ export function parseUsesSidecarName(entryName) {
54
+ const m = USES_SIDECAR_RE.exec(entryName);
55
+ return m ? decodeRefName(m[1]) : null;
56
+ }
57
+ /** `.tpl-page-default` for a page/fragment rendered with that template. */
58
+ export function templateSidecarNameFor(templateName) {
59
+ return `.tpl-${encodeRefName(templateName)}`;
60
+ }
61
+ export function parseTemplateSidecarName(entryName) {
62
+ const m = TPL_SIDECAR_RE.exec(entryName);
63
+ return m ? decodeRefName(m[1]) : null;
64
+ }
65
+ /** Compact ISO timestamp for `.pub-` sidecar filenames: `20260417T220000Z`. */
66
+ export function compactTimestamp(date = new Date()) {
67
+ return date
68
+ .toISOString()
69
+ .replace(/[-:]/g, '')
70
+ .replace(/\.\d{3}/, '');
71
+ }
72
+ /** Parse compact timestamp back to ISO string: `2026-04-17T22:00:00Z`. */
73
+ export function parseCompactTimestamp(compact) {
74
+ // 20260417T220000Z → 2026-04-17T22:00:00Z
75
+ return `${compact.slice(0, 4)}-${compact.slice(4, 6)}-${compact.slice(6, 8)}T${compact.slice(9, 11)}:${compact.slice(11, 13)}:${compact.slice(13, 15)}Z`;
76
+ }
77
+ /** `.pub-20260417T220000Z`, `.pub-20260417T220000Z-noindex`, or `.pub-20260417T220000Z.fr` */
78
+ export function pubSidecarNameFor(date = new Date(), noindex = false, locale) {
79
+ const base = `.pub-${compactTimestamp(date)}${noindex ? '-noindex' : ''}`;
80
+ return locale ? `${base}.${locale}` : base;
81
+ }
82
+ export function parsePubSidecarName(entryName) {
83
+ const m = PUB_SIDECAR_RE.exec(entryName);
84
+ if (!m)
85
+ return null;
86
+ return {
87
+ lastPublished: parseCompactTimestamp(m[1]),
88
+ noindex: !!m[2],
89
+ };
90
+ }
91
+ /** Parse locale from a pub sidecar filename. */
92
+ export function parsePubSidecarLocale(entryName) {
93
+ const m = PUB_SIDECAR_RE.exec(entryName);
94
+ return m?.[3] ?? undefined;
95
+ }
10
96
  /**
11
- * Walk the component tree and substitute `template: "name"` with `template: "name#hash"`
12
- * using the provided template hashes. Returns a new normalized structure — input is not mutated.
97
+ * Walk the component tree and substitute template/fragment refs with their
98
+ * hashed forms (`"name#hash"`) using the provided maps. Returns a new
99
+ * normalized structure — input is not mutated.
100
+ *
101
+ * `fragmentHashes` is only provided for static-mode targets where fragments
102
+ * are baked into pages at publish time — a fragment content change must
103
+ * invalidate every page that uses it. In ESI mode fragments are published
104
+ * separately, so pages don't need fragment hashes in their own hash.
13
105
  */
14
- function substituteTemplateHashes(components, templateHashes) {
106
+ function substituteHashes(components, templateHashes, fragmentHashes) {
15
107
  if (!components)
16
108
  return undefined;
17
109
  return components.map(entry => {
18
- if (typeof entry === 'string')
19
- return entry;
110
+ if (typeof entry === 'string') {
111
+ if (!fragmentHashes || !entry.startsWith('@'))
112
+ return entry;
113
+ const name = entry.slice(1);
114
+ const h = fragmentHashes.get(name);
115
+ return h ? `@${name}#${h}` : entry;
116
+ }
20
117
  const hash = templateHashes.get(entry.template);
21
118
  return {
22
119
  name: entry.name,
23
120
  template: hash ? `${entry.template}#${hash}` : entry.template,
24
121
  content: entry.content,
25
- components: substituteTemplateHashes(entry.components, templateHashes),
122
+ components: substituteHashes(entry.components, templateHashes, fragmentHashes),
26
123
  };
27
124
  });
28
125
  }
@@ -39,7 +136,7 @@ export function hashManifest(manifest, opts) {
39
136
  const normalized = {
40
137
  template: rootHash ? `${manifest.template}#${rootHash}` : manifest.template,
41
138
  content: manifest.content ?? null,
42
- components: substituteTemplateHashes(manifest.components, opts.templateHashes) ?? null,
139
+ components: substituteHashes(manifest.components, opts.templateHashes, opts.fragmentHashes) ?? null,
43
140
  };
44
141
  const json = JSON.stringify(normalized, sortedReplacer);
45
142
  return createHash('md5').update(json).digest('hex').slice(0, 8);
package/dist/hash.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"hash.js","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAA;AAGxC,MAAM,UAAU,GAAG,yBAAyB,CAAA;AAE5C,MAAM,UAAU,cAAc,CAAC,IAAY;IACzC,OAAO,IAAI,IAAI,OAAO,CAAA;AACxB,CAAC;AAED,MAAM,UAAU,gBAAgB,CAAC,SAAiB;IAChD,MAAM,CAAC,GAAG,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAA;IACpC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAA;AACxB,CAAC;AAED;;;GAGG;AACH,SAAS,wBAAwB,CAC/B,UAAwC,EACxC,cAAmC;IAEnC,IAAI,CAAC,UAAU;QAAE,OAAO,SAAS,CAAA;IACjC,OAAO,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;QAC5B,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAA;QAC3C,MAAM,IAAI,GAAG,cAAc,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAA;QAC/C,OAAO;YACL,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,QAAQ,EAAE,IAAI,CAAC,CAAC,CAAC,GAAG,KAAK,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,QAAQ;YAC7D,OAAO,EAAE,KAAK,CAAC,OAAO;YACtB,UAAU,EAAE,wBAAwB,CAAC,KAAK,CAAC,UAAU,EAAE,cAAc,CAAC;SACvE,CAAA;IACH,CAAC,CAAC,CAAA;AACJ,CAAC;AAMD;;;;;;;GAOG;AACH,MAAM,UAAU,YAAY,CAC1B,QAAyC,EACzC,IAAyB;IAEzB,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAA;IAC3D,MAAM,UAAU,GAAG;QACjB,QAAQ,EAAE,QAAQ,CAAC,CAAC,CAAC,GAAG,QAAQ,CAAC,QAAQ,IAAI,QAAQ,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,QAAQ;QAC3E,OAAO,EAAE,QAAQ,CAAC,OAAO,IAAI,IAAI;QACjC,UAAU,EAAE,wBAAwB,CAAC,QAAQ,CAAC,UAAU,EAAE,IAAI,CAAC,cAAc,CAAC,IAAI,IAAI;KACvF,CAAA;IACD,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,UAAU,EAAE,cAAc,CAAC,CAAA;IACvD,OAAO,UAAU,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAA;AACjE,CAAC;AAED,SAAS,cAAc,CAAC,IAAY,EAAE,KAAc;IAClD,IAAI,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;QAChE,MAAM,GAAG,GAA4B,EAAE,CAAA;QACvC,KAAK,MAAM,CAAC,IAAI,MAAM,CAAC,IAAI,CAAC,KAAgC,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;YACrE,GAAG,CAAC,CAAC,CAAC,GAAI,KAAiC,CAAC,CAAC,CAAC,CAAA;QAChD,CAAC;QACD,OAAO,GAAG,CAAA;IACZ,CAAC;IACD,OAAO,KAAK,CAAA;AACd,CAAC"}
1
+ {"version":3,"file":"hash.js","sourceRoot":"","sources":["../src/hash.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,49 @@
1
+ /**
2
+ * HistoryProvider implementation on top of any StorageProvider.
3
+ *
4
+ * Layout per target (inside `.gazetta/history/` under the target's
5
+ * storage root):
6
+ *
7
+ * index.json { nextId: N, revisions: ['rev-0001', ...] }
8
+ * revisions/rev-NNNN.json one manifest per revision — metadata +
9
+ * items map (itemPath → blob hash)
10
+ * objects/<hh>/<rest> content-addressed blobs (SHA-256, sharded
11
+ * by first 2 hex chars)
12
+ *
13
+ * Design-decisions.md #18:
14
+ * - One uniform approach across all providers (no native versioning).
15
+ * - Content-addressed blobs → unchanged items share storage across
16
+ * revisions; storage scales with unique content, not revision count.
17
+ * - Soft undo only — every restore writes a new forward revision.
18
+ * - Retention default 50; oldest evicted on write.
19
+ *
20
+ * SRP: this module owns .gazetta/history/ layout and retention. Nothing
21
+ * else. The write pipeline (save/publish) calls `recordRevision`; undo/
22
+ * rollback call `readRevision` + `readBlob`. Restore happens outside —
23
+ * this module doesn't touch the content tree.
24
+ */
25
+ import type { StorageProvider } from './types.js';
26
+ import type { HistoryProvider } from './history.js';
27
+ export interface CreateHistoryProviderOptions {
28
+ /** Storage under which `.gazetta/history/` lives. Usually the target's storage. */
29
+ storage: StorageProvider;
30
+ /**
31
+ * Path prefix where history lives. Default: `.gazetta/history`. Callers
32
+ * with a rooted storage (filesystem target with `path:`) pass the
33
+ * default; anything more exotic (e.g. history under a sub-prefix)
34
+ * overrides.
35
+ */
36
+ rootPath?: string;
37
+ /**
38
+ * Maximum number of revisions to keep; older ones evicted on write.
39
+ * Default: `DEFAULT_HISTORY_RETENTION` (50). Pass < 1 → clamped to 1
40
+ * (zero retention would self-evict every write).
41
+ */
42
+ retention?: number;
43
+ }
44
+ /**
45
+ * Build a HistoryProvider backed by the given storage. No I/O happens
46
+ * at construction time — everything is lazy on first call.
47
+ */
48
+ export declare function createHistoryProvider(opts: CreateHistoryProviderOptions): HistoryProvider;
49
+ //# sourceMappingURL=history-provider.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history-provider.d.ts","sourceRoot":"","sources":["../src/history-provider.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAGH,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAA;AACjD,OAAO,KAAK,EAAE,eAAe,EAA6C,MAAM,cAAc,CAAA;AAG9F,MAAM,WAAW,4BAA4B;IAC3C,mFAAmF;IACnF,OAAO,EAAE,eAAe,CAAA;IACxB;;;;;OAKG;IACH,QAAQ,CAAC,EAAE,MAAM,CAAA;IACjB;;;;OAIG;IACH,SAAS,CAAC,EAAE,MAAM,CAAA;CACnB;AAgBD;;;GAGG;AACH,wBAAgB,qBAAqB,CAAC,IAAI,EAAE,4BAA4B,GAAG,eAAe,CAyMzF"}
@@ -0,0 +1,226 @@
1
+ /**
2
+ * HistoryProvider implementation on top of any StorageProvider.
3
+ *
4
+ * Layout per target (inside `.gazetta/history/` under the target's
5
+ * storage root):
6
+ *
7
+ * index.json { nextId: N, revisions: ['rev-0001', ...] }
8
+ * revisions/rev-NNNN.json one manifest per revision — metadata +
9
+ * items map (itemPath → blob hash)
10
+ * objects/<hh>/<rest> content-addressed blobs (SHA-256, sharded
11
+ * by first 2 hex chars)
12
+ *
13
+ * Design-decisions.md #18:
14
+ * - One uniform approach across all providers (no native versioning).
15
+ * - Content-addressed blobs → unchanged items share storage across
16
+ * revisions; storage scales with unique content, not revision count.
17
+ * - Soft undo only — every restore writes a new forward revision.
18
+ * - Retention default 50; oldest evicted on write.
19
+ *
20
+ * SRP: this module owns .gazetta/history/ layout and retention. Nothing
21
+ * else. The write pipeline (save/publish) calls `recordRevision`; undo/
22
+ * rollback call `readRevision` + `readBlob`. Restore happens outside —
23
+ * this module doesn't touch the content tree.
24
+ */
25
+ import { createHash } from 'node:crypto';
26
+ import { DEFAULT_HISTORY_RETENTION } from './types.js';
27
+ /**
28
+ * Build a HistoryProvider backed by the given storage. No I/O happens
29
+ * at construction time — everything is lazy on first call.
30
+ */
31
+ export function createHistoryProvider(opts) {
32
+ const { storage } = opts;
33
+ const root = opts.rootPath ?? '.gazetta/history';
34
+ const retention = Math.max(1, opts.retention ?? DEFAULT_HISTORY_RETENTION);
35
+ const indexPath = join(root, 'index.json');
36
+ /** Read the index or return an empty one if it doesn't exist yet. */
37
+ async function readIndex() {
38
+ if (!(await storage.exists(indexPath))) {
39
+ return { revisions: [] };
40
+ }
41
+ const parsed = JSON.parse(await storage.readFile(indexPath));
42
+ // Forward-compat: tolerate legacy `nextId` by ignoring it. Old indexes
43
+ // (from the numeric-id era) continue to read cleanly, new writes use
44
+ // the timestamp scheme. No migration pass needed — retention naturally
45
+ // evicts the legacy ids over time.
46
+ return { revisions: parsed.revisions ?? [] };
47
+ }
48
+ /**
49
+ * Ensure the parent directory exists before writing. Object-store
50
+ * providers (R2, S3) ignore mkdir. Filesystem requires it because
51
+ * `writeFile` fails on missing parents, and our sharded blob paths
52
+ * (objects/<hh>/<rest>) plus revisions/rev-NNNN.json live in dirs
53
+ * that don't exist until the first write.
54
+ */
55
+ async function writeWithParents(path, content) {
56
+ const parent = path.substring(0, path.lastIndexOf('/'));
57
+ if (parent)
58
+ await storage.mkdir(parent);
59
+ await storage.writeFile(path, content);
60
+ }
61
+ async function writeIndex(idx) {
62
+ await writeWithParents(indexPath, JSON.stringify(idx, null, 2) + '\n');
63
+ }
64
+ function blobPath(hash) {
65
+ // Shard by first two hex chars — keeps any one `objects/` subdirectory
66
+ // from ballooning past a few thousand entries on large sites.
67
+ return join(root, 'objects', hash.slice(0, 2), hash.slice(2));
68
+ }
69
+ function revisionPath(id) {
70
+ return join(root, 'revisions', `${id}.json`);
71
+ }
72
+ /** SHA-256 hex of the content — strong enough that collisions can be
73
+ * ignored for practical purposes (blob identity), cheap enough at our
74
+ * scale (tens of KB per item, hundreds of items per revision). */
75
+ function hashContent(content) {
76
+ return createHash('sha256').update(content).digest('hex');
77
+ }
78
+ /**
79
+ * Allocate a fresh revision id. Scheme: `rev-<unixMillis>`, with a
80
+ * `-<seq>` suffix on same-millisecond collisions ("rev-1760...050",
81
+ * "rev-1760...050-2", ...). Collision tracking uses the current
82
+ * index's revisions list — assumes no concurrent writers against
83
+ * the same target (already true: Gazetta never has two admins
84
+ * writing the same target's history simultaneously).
85
+ *
86
+ * Why millis + suffix rather than a monotonic counter:
87
+ * - Retention evictions leave you with a window of revisions you
88
+ * can date-read from the filename alone.
89
+ * - No counter to overflow or reset when history is disabled then
90
+ * re-enabled.
91
+ * - Lex-sort = chrono sort (13-digit ms are fixed-width through
92
+ * year 5138 AD).
93
+ *
94
+ * `_clock` is injectable for deterministic tests — production uses
95
+ * Date.now(). Stays private to createHistoryProvider.
96
+ */
97
+ function formatId(existing, now) {
98
+ const base = `rev-${now}`;
99
+ if (!existing.some(id => id === base || id.startsWith(`${base}-`)))
100
+ return base;
101
+ // Same-millisecond collision: bump the suffix. Start from 2 so the
102
+ // first duplicate gets `-2` (matches the mental model "base, then
103
+ // the second, then the third").
104
+ let seq = 2;
105
+ while (existing.includes(`${base}-${seq}`))
106
+ seq += 1;
107
+ return `${base}-${seq}`;
108
+ }
109
+ /**
110
+ * Write any blob that's not already stored. Returns the hash. Dedup
111
+ * check is a single `exists()` — cheaper than reading the existing
112
+ * blob to confirm equal content (hashes collide vanishingly).
113
+ */
114
+ async function writeBlob(content) {
115
+ const hash = hashContent(content);
116
+ const path = blobPath(hash);
117
+ if (!(await storage.exists(path))) {
118
+ await writeWithParents(path, content);
119
+ }
120
+ return hash;
121
+ }
122
+ async function recordRevision(input) {
123
+ const idx = await readIndex();
124
+ const id = formatId(idx.revisions, Date.now());
125
+ // Write blobs (dedup via content-addressing) and build the
126
+ // path → hash snapshot.
127
+ const snapshot = {};
128
+ // Deterministic order so rev manifests diff cleanly on inspection.
129
+ const sortedEntries = [...input.items.entries()].sort((a, b) => a[0].localeCompare(b[0]));
130
+ for (const [path, content] of sortedEntries) {
131
+ snapshot[path] = await writeBlob(content);
132
+ }
133
+ const manifest = {
134
+ id,
135
+ timestamp: new Date().toISOString(),
136
+ operation: input.operation,
137
+ author: input.author,
138
+ source: input.source,
139
+ items: [...input.items.keys()].sort(),
140
+ message: input.message,
141
+ restoredFrom: input.restoredFrom,
142
+ snapshot,
143
+ };
144
+ await writeWithParents(revisionPath(id), JSON.stringify(manifest, null, 2) + '\n');
145
+ // Update the index (append) then apply retention. Do the index
146
+ // write last so a mid-write failure leaves orphan blobs and an
147
+ // orphan manifest (both harmless) rather than a dangling index
148
+ // entry pointing at a missing manifest.
149
+ idx.revisions.push(id);
150
+ await writeIndex(idx);
151
+ await applyRetention(idx);
152
+ // Return the public Revision shape (no snapshot).
153
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
154
+ const { snapshot: _snapshot, ...revision } = manifest;
155
+ return revision;
156
+ }
157
+ /**
158
+ * Evict oldest revisions to fit `retention`. Deletes manifests; blobs
159
+ * become eligible for GC if no remaining revision references them.
160
+ * GC is lazy — blob files stay until an explicit GC pass, which is
161
+ * fine for v1 (disk is cheap; a future `gazetta gc` command can walk
162
+ * all manifests and prune orphans).
163
+ */
164
+ async function applyRetention(idx) {
165
+ const excess = idx.revisions.length - retention;
166
+ if (excess <= 0)
167
+ return;
168
+ const toEvict = idx.revisions.slice(0, excess);
169
+ idx.revisions = idx.revisions.slice(excess);
170
+ for (const id of toEvict) {
171
+ const path = revisionPath(id);
172
+ if (await storage.exists(path))
173
+ await storage.rm(path);
174
+ }
175
+ await writeIndex(idx);
176
+ }
177
+ async function listRevisions(limit) {
178
+ const idx = await readIndex();
179
+ const ids = [...idx.revisions].reverse(); // newest first
180
+ const sliced = typeof limit === 'number' ? ids.slice(0, limit) : ids;
181
+ // Read manifests in parallel; strip snapshot for the summary list.
182
+ return Promise.all(sliced.map(async (id) => {
183
+ const m = await readManifest(id);
184
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
185
+ const { snapshot: _snapshot, ...rev } = m;
186
+ return rev;
187
+ }));
188
+ }
189
+ async function readManifest(id) {
190
+ return JSON.parse(await storage.readFile(revisionPath(id)));
191
+ }
192
+ async function readRevision(id) {
193
+ return readManifest(id);
194
+ }
195
+ async function readBlob(hash) {
196
+ return storage.readFile(blobPath(hash));
197
+ }
198
+ async function deleteRevision(id) {
199
+ const idx = await readIndex();
200
+ const at = idx.revisions.indexOf(id);
201
+ if (at === -1)
202
+ return;
203
+ idx.revisions.splice(at, 1);
204
+ await writeIndex(idx);
205
+ const path = revisionPath(id);
206
+ if (await storage.exists(path))
207
+ await storage.rm(path);
208
+ // Orphan blobs left for lazy GC — see applyRetention rationale.
209
+ }
210
+ return {
211
+ recordRevision,
212
+ listRevisions,
213
+ readRevision,
214
+ readBlob,
215
+ deleteRevision,
216
+ };
217
+ }
218
+ /**
219
+ * `posix.join` behavior without importing it — keeps the module self-
220
+ * contained and works identically across platforms. Storage providers
221
+ * normalize separators internally, but our stored paths are POSIX.
222
+ */
223
+ function join(...parts) {
224
+ return parts.filter(Boolean).join('/').replace(/\/+/g, '/');
225
+ }
226
+ //# sourceMappingURL=history-provider.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history-provider.js","sourceRoot":"","sources":["../src/history-provider.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,98 @@
1
+ /**
2
+ * Higher-level helper for recording revisions on a target.
3
+ *
4
+ * The bare `HistoryProvider.recordRevision` takes a full
5
+ * `items: Map<path, content>` snapshot. That's fine for testing but
6
+ * wasteful at runtime: every save of one page would re-hash and
7
+ * re-list every page + fragment on the target. This helper does the
8
+ * right thing:
9
+ *
10
+ * - First revision on a target: walks the content tree once to
11
+ * snapshot every manifest (page.json, fragment.json, site.yaml).
12
+ * - Subsequent revisions: reads the previous snapshot and overlays
13
+ * the delta (changed items the caller passes in). `readBlob` for
14
+ * each carried-over path gives us the content for the new
15
+ * revision's items map — at which point `recordRevision` dedupes
16
+ * the unchanged blobs via content-addressing, so no new storage.
17
+ *
18
+ * SRP: this module owns the "what goes in a revision snapshot"
19
+ * decision. `HistoryProvider` owns layout. Callers (admin-api save /
20
+ * admin-api publish / CLI publish) just describe *what they wrote*
21
+ * and we construct the revision.
22
+ */
23
+ import type { HistoryProvider, RevisionOperation } from './history.js';
24
+ import type { ContentRoot } from './content-root.js';
25
+ /** A single item that was written in this save/publish. */
26
+ export interface WrittenItem {
27
+ /** Path relative to the content root, e.g. `pages/home/page.json`. */
28
+ path: string;
29
+ /** Current content as stored. `null` marks a deletion. */
30
+ content: string | null;
31
+ }
32
+ /**
33
+ * Location to scan when building the first revision's baseline
34
+ * snapshot. Each entry names a directory under the content root and
35
+ * the manifest filename to capture from every subdirectory.
36
+ */
37
+ export interface ScanLocation {
38
+ /** Directory relative to the content root, e.g. `pages` or `fragments`. */
39
+ dir: string;
40
+ /** Manifest filename to capture, e.g. `page.json` or `fragment.json`. */
41
+ manifest: string;
42
+ }
43
+ /**
44
+ * Built-in content locations Gazetta knows about today. Callers can
45
+ * pass a superset (e.g. for future data/*, templates/*) — the list is
46
+ * part of `RecordWriteOptions` so this module stays open for extension
47
+ * without changes when new content kinds land.
48
+ */
49
+ export declare const DEFAULT_SCAN_LOCATIONS: readonly ScanLocation[];
50
+ /**
51
+ * Flat files at the content root to capture in the baseline snapshot
52
+ * (no per-subdirectory recursion). `site.yaml` is the only one today.
53
+ */
54
+ export declare const DEFAULT_SCAN_ROOT_FILES: readonly string[];
55
+ export interface RecordWriteOptions {
56
+ /** HistoryProvider for the target we're recording on. */
57
+ history: HistoryProvider;
58
+ /** Content root of the target — used to scan on first revision. */
59
+ contentRoot: ContentRoot;
60
+ operation: RevisionOperation;
61
+ /** Items the save/publish wrote (and optionally deleted). */
62
+ items: WrittenItem[];
63
+ /** Author identifier passed through to the manifest. */
64
+ author?: string;
65
+ /** Source target name (for publish). */
66
+ source?: string;
67
+ /** Optional human-readable note. */
68
+ message?: string;
69
+ /** For rollback/restore: the revision id this one restored from. */
70
+ restoredFrom?: string;
71
+ /**
72
+ * Override the directories walked during the first-revision baseline
73
+ * scan. Defaults to `DEFAULT_SCAN_LOCATIONS` (pages + fragments).
74
+ * Pass a superset if the site has extra authored content (e.g.
75
+ * custom `data/*.json` dirs); pass `[]` to skip directory scanning
76
+ * entirely (only root files + explicit items are captured).
77
+ */
78
+ scanLocations?: readonly ScanLocation[];
79
+ /**
80
+ * Override the flat files captured from the content root. Defaults
81
+ * to `DEFAULT_SCAN_ROOT_FILES` (`site.yaml`). Missing files are
82
+ * silently skipped so empty publish-targets still record cleanly.
83
+ */
84
+ scanRootFiles?: readonly string[];
85
+ }
86
+ /**
87
+ * Build + record a revision for the given write. Reads the previous
88
+ * snapshot (if any), overlays the delta, and calls
89
+ * `history.recordRevision`. Returns the recorded Revision.
90
+ *
91
+ * Callers are expected to have already written the items to the
92
+ * target's storage before invoking this; the recorder reads back via
93
+ * the HistoryProvider's dedup path (blobs it has already seen just
94
+ * `exists()` and skip) so the happy path is cheap on repeated saves
95
+ * of the same item.
96
+ */
97
+ export declare function recordWrite(opts: RecordWriteOptions): Promise<import("./history.js").Revision>;
98
+ //# sourceMappingURL=history-recorder.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"history-recorder.d.ts","sourceRoot":"","sources":["../src/history-recorder.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;GAqBG;AAIH,OAAO,KAAK,EAAE,eAAe,EAAiB,iBAAiB,EAAE,MAAM,cAAc,CAAA;AACrF,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAA;AAEpD,2DAA2D;AAC3D,MAAM,WAAW,WAAW;IAC1B,sEAAsE;IACtE,IAAI,EAAE,MAAM,CAAA;IACZ,0DAA0D;IAC1D,OAAO,EAAE,MAAM,GAAG,IAAI,CAAA;CACvB;AAED;;;;GAIG;AACH,MAAM,WAAW,YAAY;IAC3B,2EAA2E;IAC3E,GAAG,EAAE,MAAM,CAAA;IACX,yEAAyE;IACzE,QAAQ,EAAE,MAAM,CAAA;CACjB;AAED;;;;;GAKG;AACH,eAAO,MAAM,sBAAsB,EAAE,SAAS,YAAY,EAGzD,CAAA;AAED;;;GAGG;AACH,eAAO,MAAM,uBAAuB,EAAE,SAAS,MAAM,EAAkB,CAAA;AAEvE,MAAM,WAAW,kBAAkB;IACjC,yDAAyD;IACzD,OAAO,EAAE,eAAe,CAAA;IACxB,mEAAmE;IACnE,WAAW,EAAE,WAAW,CAAA;IACxB,SAAS,EAAE,iBAAiB,CAAA;IAC5B,6DAA6D;IAC7D,KAAK,EAAE,WAAW,EAAE,CAAA;IACpB,wDAAwD;IACxD,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,wCAAwC;IACxC,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,oCAAoC;IACpC,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,oEAAoE;IACpE,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB;;;;;;OAMG;IACH,aAAa,CAAC,EAAE,SAAS,YAAY,EAAE,CAAA;IACvC;;;;OAIG;IACH,aAAa,CAAC,EAAE,SAAS,MAAM,EAAE,CAAA;CAClC;AAED;;;;;;;;;;GAUG;AACH,wBAAsB,WAAW,CAAC,IAAI,EAAE,kBAAkB,4CAmCzD"}