@open-press/cli 0.8.0 → 1.1.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 (250) hide show
  1. package/README.md +33 -23
  2. package/dist/cli.js +320 -252
  3. package/package.json +9 -8
  4. package/template/core/AGENTS.md +0 -126
  5. package/template/core/CHANGELOG.md +0 -215
  6. package/template/core/README.md +0 -40
  7. package/template/core/engine/cli.mjs +0 -96
  8. package/template/core/engine/commands/_shared.mjs +0 -177
  9. package/template/core/engine/commands/deploy.mjs +0 -31
  10. package/template/core/engine/commands/dev.mjs +0 -49
  11. package/template/core/engine/commands/doctor.mjs +0 -229
  12. package/template/core/engine/commands/export.mjs +0 -8
  13. package/template/core/engine/commands/init.mjs +0 -24
  14. package/template/core/engine/commands/inspect.mjs +0 -35
  15. package/template/core/engine/commands/pdf.mjs +0 -26
  16. package/template/core/engine/commands/preview.mjs +0 -26
  17. package/template/core/engine/commands/render.mjs +0 -17
  18. package/template/core/engine/commands/replace.mjs +0 -41
  19. package/template/core/engine/commands/search.mjs +0 -33
  20. package/template/core/engine/commands/typecheck.mjs +0 -5
  21. package/template/core/engine/commands/upgrade.mjs +0 -159
  22. package/template/core/engine/commands/validate.mjs +0 -17
  23. package/template/core/engine/document-export.mjs +0 -15
  24. package/template/core/engine/init.mjs +0 -90
  25. package/template/core/engine/output/chrome-pdf.d.mts +0 -34
  26. package/template/core/engine/output/chrome-pdf.mjs +0 -358
  27. package/template/core/engine/output/deploy-sync.mjs +0 -15
  28. package/template/core/engine/output/fonts.mjs +0 -62
  29. package/template/core/engine/output/katex-assets.mjs +0 -45
  30. package/template/core/engine/output/page-block.mjs +0 -30
  31. package/template/core/engine/output/pdf-media.mjs +0 -45
  32. package/template/core/engine/output/public-assets.mjs +0 -19
  33. package/template/core/engine/output/static-server.mjs +0 -532
  34. package/template/core/engine/react/caption-numbering.mjs +0 -73
  35. package/template/core/engine/react/comment-endpoint.d.mts +0 -11
  36. package/template/core/engine/react/comment-endpoint.mjs +0 -102
  37. package/template/core/engine/react/comment-marker.mjs +0 -374
  38. package/template/core/engine/react/document-entry.mjs +0 -324
  39. package/template/core/engine/react/document-export.mjs +0 -373
  40. package/template/core/engine/react/http-json.mjs +0 -24
  41. package/template/core/engine/react/mdx-compile.mjs +0 -599
  42. package/template/core/engine/react/measurement-css.mjs +0 -136
  43. package/template/core/engine/react/object-entities.mjs +0 -119
  44. package/template/core/engine/react/pagination/allocator.mjs +0 -122
  45. package/template/core/engine/react/pagination/regions.mjs +0 -81
  46. package/template/core/engine/react/pagination-constants.mjs +0 -3
  47. package/template/core/engine/react/pagination.mjs +0 -9
  48. package/template/core/engine/react/pipeline/allocate.mjs +0 -251
  49. package/template/core/engine/react/pipeline/final-render.mjs +0 -94
  50. package/template/core/engine/react/pipeline/frame-measurement.mjs +0 -302
  51. package/template/core/engine/react/pipeline/press-tree.mjs +0 -135
  52. package/template/core/engine/react/project-asset-endpoint.d.mts +0 -10
  53. package/template/core/engine/react/project-asset-endpoint.mjs +0 -361
  54. package/template/core/engine/react/section-css.mjs +0 -56
  55. package/template/core/engine/react/source-edit-endpoint.d.mts +0 -10
  56. package/template/core/engine/react/source-edit-endpoint.mjs +0 -75
  57. package/template/core/engine/react/sources/heading-numbering.mjs +0 -132
  58. package/template/core/engine/react/sources/mdx-resolver.mjs +0 -439
  59. package/template/core/engine/react/style-discovery.mjs +0 -142
  60. package/template/core/engine/runtime/config.d.mts +0 -40
  61. package/template/core/engine/runtime/config.mjs +0 -175
  62. package/template/core/engine/runtime/file-utils.mjs +0 -106
  63. package/template/core/engine/runtime/file-walk.mjs +0 -22
  64. package/template/core/engine/runtime/inspection.mjs +0 -328
  65. package/template/core/engine/runtime/issue-report.mjs +0 -44
  66. package/template/core/engine/runtime/path-utils.mjs +0 -20
  67. package/template/core/engine/runtime/source-text-tools.d.mts +0 -102
  68. package/template/core/engine/runtime/source-text-tools.mjs +0 -832
  69. package/template/core/engine/runtime/source-workspace.mjs +0 -159
  70. package/template/core/engine/runtime/validation.mjs +0 -174
  71. package/template/core/index.html +0 -13
  72. package/template/core/openpress.config.mjs +0 -12
  73. package/template/core/package.json +0 -91
  74. package/template/core/src/main.tsx +0 -16
  75. package/template/core/src/openpress/app/OpenPressApp.tsx +0 -140
  76. package/template/core/src/openpress/app/OpenPressRuntime.tsx +0 -94
  77. package/template/core/src/openpress/app/index.ts +0 -2
  78. package/template/core/src/openpress/core/Frame.tsx +0 -78
  79. package/template/core/src/openpress/core/FrameContext.tsx +0 -24
  80. package/template/core/src/openpress/core/MdxArea.tsx +0 -34
  81. package/template/core/src/openpress/core/Press.tsx +0 -34
  82. package/template/core/src/openpress/core/cn.ts +0 -4
  83. package/template/core/src/openpress/core/index.tsx +0 -40
  84. package/template/core/src/openpress/core/primitives.tsx +0 -44
  85. package/template/core/src/openpress/core/types.ts +0 -191
  86. package/template/core/src/openpress/core/useSource.ts +0 -28
  87. package/template/core/src/openpress/document-model/anchorMapModel.ts +0 -27
  88. package/template/core/src/openpress/document-model/documentIndexes.ts +0 -329
  89. package/template/core/src/openpress/document-model/documentTypes.ts +0 -138
  90. package/template/core/src/openpress/document-model/index.ts +0 -6
  91. package/template/core/src/openpress/document-model/objectEntityModel.ts +0 -51
  92. package/template/core/src/openpress/document-model/projectIdentityModel.ts +0 -15
  93. package/template/core/src/openpress/document-model/reactDocumentMetadataModel.ts +0 -27
  94. package/template/core/src/openpress/manuscript/index.tsx +0 -238
  95. package/template/core/src/openpress/mdx/index.ts +0 -88
  96. package/template/core/src/openpress/numbering/index.ts +0 -294
  97. package/template/core/src/openpress/reader/PublicReaderPage.tsx +0 -267
  98. package/template/core/src/openpress/reader/ReaderNavigationPanel.tsx +0 -123
  99. package/template/core/src/openpress/reader/index.ts +0 -10
  100. package/template/core/src/openpress/reader/pageViewportScaleModel.ts +0 -73
  101. package/template/core/src/openpress/reader/readerPageRegistry.ts +0 -41
  102. package/template/core/src/openpress/reader/readerPageRoute.ts +0 -21
  103. package/template/core/src/openpress/reader/readerScroll.ts +0 -92
  104. package/template/core/src/openpress/reader/readerStateModel.ts +0 -15
  105. package/template/core/src/openpress/reader/readerTypes.ts +0 -4
  106. package/template/core/src/openpress/reader/usePageViewportScale.ts +0 -119
  107. package/template/core/src/openpress/reader/usePanelState.ts +0 -56
  108. package/template/core/src/openpress/reader/useReaderHashSync.ts +0 -61
  109. package/template/core/src/openpress/reader/useReaderKeyboardNav.ts +0 -48
  110. package/template/core/src/openpress/reader/useReaderRuntime.ts +0 -146
  111. package/template/core/src/openpress/reader/useReaderScrollAnchor.ts +0 -64
  112. package/template/core/src/openpress/shared/Panel.tsx +0 -77
  113. package/template/core/src/openpress/shared/frameScheduler.ts +0 -32
  114. package/template/core/src/openpress/shared/index.ts +0 -4
  115. package/template/core/src/openpress/shared/numberUtils.ts +0 -3
  116. package/template/core/src/openpress/shared/runtimeMode.ts +0 -11
  117. package/template/core/src/openpress/workbench/Workbench.tsx +0 -407
  118. package/template/core/src/openpress/workbench/actions/DeploymentControl.tsx +0 -157
  119. package/template/core/src/openpress/workbench/actions/PageZoomControl.tsx +0 -182
  120. package/template/core/src/openpress/workbench/actions/SearchControl.tsx +0 -345
  121. package/template/core/src/openpress/workbench/actions/deploymentStatusModel.ts +0 -112
  122. package/template/core/src/openpress/workbench/actions/index.ts +0 -5
  123. package/template/core/src/openpress/workbench/actions/useDeploymentWorkbench.ts +0 -136
  124. package/template/core/src/openpress/workbench/dialog/WorkbenchDialog.tsx +0 -72
  125. package/template/core/src/openpress/workbench/dialog/index.ts +0 -1
  126. package/template/core/src/openpress/workbench/document/components/DocumentPanel.tsx +0 -127
  127. package/template/core/src/openpress/workbench/document/components/InlineSourceEditorLayer.tsx +0 -207
  128. package/template/core/src/openpress/workbench/document/components/ReaderStage.tsx +0 -9
  129. package/template/core/src/openpress/workbench/document/hooks/useDocumentWorkbenchModel.ts +0 -34
  130. package/template/core/src/openpress/workbench/document/hooks/useInlineDocumentEditor.ts +0 -525
  131. package/template/core/src/openpress/workbench/document/index.ts +0 -10
  132. package/template/core/src/openpress/workbench/index.ts +0 -2
  133. package/template/core/src/openpress/workbench/inspector/InlineInspectorLayer.tsx +0 -459
  134. package/template/core/src/openpress/workbench/inspector/index.ts +0 -5
  135. package/template/core/src/openpress/workbench/inspector/inlineCommentModel.ts +0 -125
  136. package/template/core/src/openpress/workbench/inspector/inspectorGeometryModel.ts +0 -160
  137. package/template/core/src/openpress/workbench/inspector/inspectorModel.ts +0 -408
  138. package/template/core/src/openpress/workbench/inspector/useInspectorComments.ts +0 -248
  139. package/template/core/src/openpress/workbench/mentions/MentionSuggestionList.tsx +0 -41
  140. package/template/core/src/openpress/workbench/mentions/index.ts +0 -2
  141. package/template/core/src/openpress/workbench/mentions/useComposerMentions.ts +0 -185
  142. package/template/core/src/openpress/workbench/panels/Panel.tsx +0 -1
  143. package/template/core/src/openpress/workbench/panels/PendingCommentsPanel.tsx +0 -76
  144. package/template/core/src/openpress/workbench/panels/WorkbenchControlPanel.tsx +0 -29
  145. package/template/core/src/openpress/workbench/panels/index.ts +0 -3
  146. package/template/core/src/openpress/workbench/project/ProjectEntryPanel.tsx +0 -523
  147. package/template/core/src/openpress/workbench/project/ProjectPreviewDialog.tsx +0 -35
  148. package/template/core/src/openpress/workbench/project/index.ts +0 -2
  149. package/template/core/src/openpress/workbench/project/projectPreviewTypes.ts +0 -11
  150. package/template/core/src/openpress/workbench/project/projectSourceModel.ts +0 -24
  151. package/template/core/src/openpress/workbench/shell/WorkbenchShell.tsx +0 -167
  152. package/template/core/src/openpress/workbench/shell/index.ts +0 -1
  153. package/template/core/src/openpress/workbench/workbenchFormatters.ts +0 -120
  154. package/template/core/src/openpress/workbench/workbenchTypes.ts +0 -35
  155. package/template/core/src/styles/openpress/app-shell.css +0 -251
  156. package/template/core/src/styles/openpress/media-workspace.css +0 -230
  157. package/template/core/src/styles/openpress/print-route.css +0 -184
  158. package/template/core/src/styles/openpress/project-preview-panel.css +0 -924
  159. package/template/core/src/styles/openpress/public-viewer.css +0 -688
  160. package/template/core/src/styles/openpress/reader-runtime.css +0 -980
  161. package/template/core/src/styles/openpress/responsive.css +0 -245
  162. package/template/core/src/styles/openpress/workbench-panels.css +0 -594
  163. package/template/core/src/styles/openpress/workbench.css +0 -1255
  164. package/template/core/src/styles/openpress.css +0 -14
  165. package/template/core/src/vite-env.d.ts +0 -9
  166. package/template/core/tsconfig.json +0 -40
  167. package/template/core/vite.config.ts +0 -584
  168. package/template/packs/academic-paper/document/chapters/01-introduction/content/01-introduction.mdx +0 -35
  169. package/template/packs/academic-paper/document/chapters/02-methods/content/01-methods.mdx +0 -50
  170. package/template/packs/academic-paper/document/chapters/03-results-and-discussion/content/01-results.mdx +0 -47
  171. package/template/packs/academic-paper/document/chapters/04-acknowledgment/content/01-acknowledgment.mdx +0 -26
  172. package/template/packs/academic-paper/document/chapters/05-references/content/01-references.mdx +0 -32
  173. package/template/packs/academic-paper/document/components/ChapterOpenerVisual/index.tsx +0 -76
  174. package/template/packs/academic-paper/document/components/Page.tsx +0 -60
  175. package/template/packs/academic-paper/document/components/TokenSwatchGrid/index.tsx +0 -46
  176. package/template/packs/academic-paper/document/components/TokenSwatchGrid/style.css +0 -63
  177. package/template/packs/academic-paper/document/components/TypeSpecimen/index.tsx +0 -38
  178. package/template/packs/academic-paper/document/components/TypeSpecimen/style.css +0 -111
  179. package/template/packs/academic-paper/document/design.md +0 -279
  180. package/template/packs/academic-paper/document/index.tsx +0 -123
  181. package/template/packs/academic-paper/document/media/README.md +0 -13
  182. package/template/packs/academic-paper/document/media/figure-placeholder.svg +0 -9
  183. package/template/packs/academic-paper/document/openpress.config.mjs +0 -26
  184. package/template/packs/academic-paper/document/theme/README.md +0 -11
  185. package/template/packs/academic-paper/document/theme/base/page-contract.css +0 -522
  186. package/template/packs/academic-paper/document/theme/base/print.css +0 -93
  187. package/template/packs/academic-paper/document/theme/base/typography.css +0 -333
  188. package/template/packs/academic-paper/document/theme/fonts.css +0 -3
  189. package/template/packs/academic-paper/document/theme/page-surfaces/back-cover.css +0 -43
  190. package/template/packs/academic-paper/document/theme/page-surfaces/chapter-opener.css +0 -205
  191. package/template/packs/academic-paper/document/theme/page-surfaces/cover.css +0 -294
  192. package/template/packs/academic-paper/document/theme/page-surfaces/toc.css +0 -149
  193. package/template/packs/academic-paper/document/theme/patterns/_chart-frame.css +0 -49
  194. package/template/packs/academic-paper/document/theme/patterns/figure-grid.css +0 -68
  195. package/template/packs/academic-paper/document/theme/patterns/table-utilities.css +0 -66
  196. package/template/packs/academic-paper/document/theme/shell/reader-controls.css +0 -761
  197. package/template/packs/academic-paper/document/theme/tokens.css +0 -80
  198. package/template/packs/academic-paper/openpress.config.mjs +0 -5
  199. package/template/packs/claude-document/document/chapters/01-document-shape/content/01-document-shape.mdx +0 -51
  200. package/template/packs/claude-document/document/chapters/02-review-loop/content/01-review-loop.mdx +0 -31
  201. package/template/packs/claude-document/document/components/ChapterOpenerVisual.tsx +0 -96
  202. package/template/packs/claude-document/document/components/Page.tsx +0 -37
  203. package/template/packs/claude-document/document/design.md +0 -142
  204. package/template/packs/claude-document/document/index.tsx +0 -94
  205. package/template/packs/claude-document/document/media/README.md +0 -13
  206. package/template/packs/claude-document/document/openpress.config.mjs +0 -26
  207. package/template/packs/claude-document/document/theme/README.md +0 -15
  208. package/template/packs/claude-document/document/theme/base/page-contract.css +0 -525
  209. package/template/packs/claude-document/document/theme/base/print.css +0 -93
  210. package/template/packs/claude-document/document/theme/base/typography.css +0 -612
  211. package/template/packs/claude-document/document/theme/fonts.css +0 -4
  212. package/template/packs/claude-document/document/theme/page-surfaces/back-cover.css +0 -72
  213. package/template/packs/claude-document/document/theme/page-surfaces/chapter-opener.css +0 -236
  214. package/template/packs/claude-document/document/theme/page-surfaces/cover.css +0 -309
  215. package/template/packs/claude-document/document/theme/page-surfaces/toc.css +0 -225
  216. package/template/packs/claude-document/document/theme/patterns/_chart-frame.css +0 -53
  217. package/template/packs/claude-document/document/theme/patterns/figure-grid.css +0 -68
  218. package/template/packs/claude-document/document/theme/patterns/table-utilities.css +0 -66
  219. package/template/packs/claude-document/document/theme/shell/reader-controls.css +0 -789
  220. package/template/packs/claude-document/document/theme/tokens.css +0 -89
  221. package/template/packs/claude-document/openpress.config.mjs +0 -5
  222. package/template/packs/editorial-monograph/document/chapters/01-product-and-use-cases/content/01-product-and-use-cases.mdx +0 -31
  223. package/template/packs/editorial-monograph/document/chapters/02-workflow/content/01-workflow.mdx +0 -89
  224. package/template/packs/editorial-monograph/document/chapters/03-agent-skills-contributors/content/01-agent-skills-contributors.mdx +0 -51
  225. package/template/packs/editorial-monograph/document/chapters/04-validation-deploy/content/01-validation-deploy.mdx +0 -39
  226. package/template/packs/editorial-monograph/document/components/ChapterOpenerVisual/index.tsx +0 -76
  227. package/template/packs/editorial-monograph/document/components/Page.tsx +0 -37
  228. package/template/packs/editorial-monograph/document/components/TokenSwatchGrid/index.tsx +0 -46
  229. package/template/packs/editorial-monograph/document/components/TokenSwatchGrid/style.css +0 -63
  230. package/template/packs/editorial-monograph/document/components/TypeSpecimen/index.tsx +0 -38
  231. package/template/packs/editorial-monograph/document/components/TypeSpecimen/style.css +0 -111
  232. package/template/packs/editorial-monograph/document/design.md +0 -279
  233. package/template/packs/editorial-monograph/document/index.tsx +0 -97
  234. package/template/packs/editorial-monograph/document/media/README.md +0 -13
  235. package/template/packs/editorial-monograph/document/openpress.config.mjs +0 -26
  236. package/template/packs/editorial-monograph/document/theme/README.md +0 -11
  237. package/template/packs/editorial-monograph/document/theme/base/page-contract.css +0 -505
  238. package/template/packs/editorial-monograph/document/theme/base/print.css +0 -93
  239. package/template/packs/editorial-monograph/document/theme/base/typography.css +0 -336
  240. package/template/packs/editorial-monograph/document/theme/fonts.css +0 -3
  241. package/template/packs/editorial-monograph/document/theme/page-surfaces/back-cover.css +0 -43
  242. package/template/packs/editorial-monograph/document/theme/page-surfaces/chapter-opener.css +0 -205
  243. package/template/packs/editorial-monograph/document/theme/page-surfaces/cover.css +0 -147
  244. package/template/packs/editorial-monograph/document/theme/page-surfaces/toc.css +0 -149
  245. package/template/packs/editorial-monograph/document/theme/patterns/_chart-frame.css +0 -49
  246. package/template/packs/editorial-monograph/document/theme/patterns/figure-grid.css +0 -68
  247. package/template/packs/editorial-monograph/document/theme/patterns/table-utilities.css +0 -66
  248. package/template/packs/editorial-monograph/document/theme/shell/reader-controls.css +0 -761
  249. package/template/packs/editorial-monograph/document/theme/tokens.css +0 -80
  250. package/template/packs/editorial-monograph/openpress.config.mjs +0 -5
@@ -1,177 +0,0 @@
1
- import { spawn, spawnSync } from "node:child_process";
2
- import fs from "node:fs/promises";
3
- import path from "node:path";
4
- import { fileURLToPath } from "node:url";
5
- import { printUrlToPdf, stopChildProcess, waitForPrintReady } from "../output/chrome-pdf.mjs";
6
- import { loadConfig, publicPdfHref } from "../runtime/config.mjs";
7
- import { exportDocument } from "../document-export.mjs";
8
- import { optimizePdfMediaForStaticRoot } from "../output/pdf-media.mjs";
9
-
10
- export const ENGINE_DIR = path.resolve(path.dirname(fileURLToPath(import.meta.url)), "..");
11
- export const CLI_ENTRY = path.join(ENGINE_DIR, "cli.mjs");
12
- export const STATIC_SERVER = path.join(ENGINE_DIR, "output", "static-server.mjs");
13
-
14
- export function parseOptions(argv) {
15
- const options = {};
16
- const positional = [];
17
- for (let i = 0; i < argv.length; i += 1) {
18
- const value = argv[i];
19
- if (value === "--renderer") options.renderer = argv[++i];
20
- else if (value === "--host") options.host = argv[++i];
21
- else if (value === "--port") options.port = argv[++i];
22
- else if (value === "--dry-run") options.dryRun = true;
23
- else if (value === "--force") options.force = true;
24
- else if (value === "--confirm") options.confirm = true;
25
- else if (value === "--json") options.json = true;
26
- else if (value === "--no-build") options.noBuild = true;
27
- else if (value === "--apply") options.apply = true;
28
- else if (value === "--include-code") options.includeCode = true;
29
- else if (value === "--case-sensitive") options.caseSensitive = true;
30
- else if (value === "--scope") options.scope = argv[++i];
31
- else if (value === "--source") options.source = argv[++i];
32
- else if (value === "--output") options.output = argv[++i];
33
- else if (value.startsWith("--")) throw new Error(`Unknown option: ${value}`);
34
- else positional.push(value);
35
- }
36
- options.path = positional[0];
37
- options.positional = positional;
38
- return options;
39
- }
40
-
41
- export function parseInitOptions(argv) {
42
- const options = { force: false };
43
- const positional = [];
44
- for (let i = 0; i < argv.length; i += 1) {
45
- const value = argv[i];
46
- if (value === "--skill") options.skill = argv[++i];
47
- else if (value === "--force") options.force = true;
48
- else if (value.startsWith("--")) throw new Error(`Unknown option: ${value}`);
49
- else positional.push(value);
50
- }
51
- options.target = positional[0];
52
- return options;
53
- }
54
-
55
- export function formatDisplayPath(absolutePath) {
56
- const relative = path.relative(process.cwd(), absolutePath);
57
- if (!relative || relative.startsWith("..")) return absolutePath;
58
- return relative;
59
- }
60
-
61
- export function runCommand(commandName, commandArgs, cwd) {
62
- const result = spawnSync(commandName, commandArgs, { cwd, stdio: "inherit" });
63
- return result.status ?? 1;
64
- }
65
-
66
- export function formatNodeScriptCommand(root, scriptPath) {
67
- const relative = path.relative(root, scriptPath).replaceAll("\\", "/");
68
- const displayPath = relative && !relative.startsWith("../") ? relative : scriptPath;
69
- return `node ${displayPath}`;
70
- }
71
-
72
- export async function buildReactStatic({ root, noBuild = false, recurse, silent = false }) {
73
- if (noBuild) return 0;
74
- if (!silent) {
75
- return await recurse("render", [root, "--renderer", "react"]);
76
- }
77
-
78
- await exportDocument(root);
79
- const result = spawnSync("npx", ["vite", "build", "--config", "vite.config.ts"], {
80
- cwd: root,
81
- encoding: "utf8",
82
- });
83
- return result.status ?? 1;
84
- }
85
-
86
- export async function buildReactPdf({
87
- root,
88
- config,
89
- outPath,
90
- host = "127.0.0.1",
91
- port = "5185",
92
- noBuild = false,
93
- recurse,
94
- }) {
95
- config ??= await loadConfig(root);
96
- outPath ??= config.paths.pdf;
97
- const renderCode = await buildReactStatic({ root, noBuild, recurse });
98
- if (renderCode !== 0) throw new Error(`React render failed with exit code ${renderCode}`);
99
- await optimizePdfMediaForStaticRoot(config.paths.outputDir);
100
- await fs.mkdir(path.dirname(outPath), { recursive: true });
101
-
102
- const server = await startStaticServer(root, config, host, port);
103
- try {
104
- const pageCount = await printUrlToPdf({
105
- root,
106
- url: `http://${host}:${port}/?print=1`,
107
- outPath,
108
- waitForReady: waitForPrintReady,
109
- debuggingPortBase: 9300,
110
- debuggingPortRange: 600,
111
- profilePrefix: "chrome-pdf",
112
- });
113
- console.log(`${pageCount} OpenPress pages printed to PDF`);
114
- } finally {
115
- await stopChildProcess(server);
116
- }
117
-
118
- return { pdfPath: outPath };
119
- }
120
-
121
- export function startStaticServer(root, config, host, port) {
122
- return new Promise((resolve, reject) => {
123
- const child = spawn("node", [STATIC_SERVER, config.outputDir, "--host", host, "--port", port, "--workspace", "."], {
124
- cwd: root,
125
- shell: false,
126
- stdio: ["ignore", "pipe", "pipe"],
127
- });
128
- let settled = false;
129
- let stderr = "";
130
- const timer = setTimeout(() => {
131
- if (settled) return;
132
- settled = true;
133
- child.kill();
134
- reject(new Error(`Timed out waiting for OpenPress static server on ${host}:${port}`));
135
- }, 10000);
136
-
137
- child.stdout.on("data", (chunk) => {
138
- const text = String(chunk);
139
- if (!settled && text.includes("OpenPress static preview:")) {
140
- settled = true;
141
- clearTimeout(timer);
142
- resolve(child);
143
- }
144
- });
145
- child.stderr.on("data", (chunk) => {
146
- stderr += String(chunk);
147
- });
148
- child.on("error", (error) => {
149
- if (settled) return;
150
- settled = true;
151
- clearTimeout(timer);
152
- reject(error);
153
- });
154
- child.on("close", (code) => {
155
- if (settled) return;
156
- settled = true;
157
- clearTimeout(timer);
158
- reject(new Error(`OpenPress static server exited with code ${code ?? 1}: ${stderr}`));
159
- });
160
- });
161
- }
162
-
163
- export async function writePdfStageDeployConfig(root, source, config) {
164
- const deployRoot = path.resolve(root, source);
165
- const openpressDir = path.join(deployRoot, "openpress");
166
- await fs.mkdir(openpressDir, { recursive: true });
167
- await fs.writeFile(
168
- path.join(openpressDir, "deploy.json"),
169
- `${JSON.stringify({ pdf: publicPdfHref(config), deployed_at: new Date().toISOString() }, null, 2)}\n`,
170
- "utf8",
171
- );
172
- await fs.writeFile(
173
- path.join(deployRoot, "_headers"),
174
- `${publicPdfHref(config)}\n Content-Type: application/pdf\n Content-Disposition: inline; filename="${config.pdf.filename}"\n`,
175
- "utf8",
176
- );
177
- }
@@ -1,31 +0,0 @@
1
- import path from "node:path";
2
- import { deploySync } from "../output/deploy-sync.mjs";
3
- import { CLI_ENTRY, buildReactPdf, formatNodeScriptCommand, runCommand, writePdfStageDeployConfig } from "./_shared.mjs";
4
-
5
- export async function run({ root, config, options, recurse }) {
6
- if (config.deploy.requiresConfirmation === true && !options.confirm) {
7
- console.error("OpenPress deploy requires --confirm before updating a public Cloudflare Pages site.");
8
- return 2;
9
- }
10
- const source = config.deploy.source;
11
- const projectName = config.deploy.projectName;
12
- const commitDirty = config.deploy.commitDirty;
13
- if (options.dryRun) {
14
- console.log("OpenPress deploy dry run");
15
- console.log(`Command: ${formatNodeScriptCommand(root, CLI_ENTRY)} render . --renderer react`);
16
- console.log(`Step: deploy-sync (copy ${config.outputDir} → ${source})`);
17
- console.log(`Command: ${formatNodeScriptCommand(root, CLI_ENTRY)} pdf . --output ${source}/${config.pdf.filename}`);
18
- console.log(`Step: write ${source}/openpress/deploy.json with deployment metadata`);
19
- console.log(`Command: npx wrangler pages deploy ${source}${projectName ? ` --project-name=${projectName}` : ""}${commitDirty ? " --commit-dirty=true" : ""}`);
20
- return 0;
21
- }
22
- const renderCode = await recurse("render", [root, "--renderer", "react"]);
23
- if (renderCode !== 0) return renderCode;
24
- await deploySync(root, config.outputDir, source);
25
- await buildReactPdf({ root, config, outPath: path.resolve(root, source, config.pdf.filename), noBuild: true, recurse });
26
- await writePdfStageDeployConfig(root, source, config);
27
- const wranglerArgs = ["wrangler", "pages", "deploy", source];
28
- if (projectName) wranglerArgs.push(`--project-name=${projectName}`);
29
- if (commitDirty) wranglerArgs.push("--commit-dirty=true");
30
- return runCommand("npx", wranglerArgs, root);
31
- }
@@ -1,49 +0,0 @@
1
- import { exportDocument } from "../document-export.mjs";
2
- import { diagnose } from "./doctor.mjs";
3
- import { CLI_ENTRY, formatNodeScriptCommand, runCommand } from "./_shared.mjs";
4
-
5
- export async function run({ root, options }) {
6
- const renderer = options.renderer ?? "react";
7
- if (renderer !== "react") {
8
- console.error(`Unknown renderer: ${renderer}`);
9
- return 2;
10
- }
11
- const host = options.host ?? "127.0.0.1";
12
- const port = options.port ?? "5173";
13
- const url = `http://${host}:${port}/?dev=1`;
14
- if (options.dryRun) {
15
- console.log(`OpenPress dev URL: ${url}`);
16
- if (!options.noBuild) {
17
- console.log(`Command: ${formatNodeScriptCommand(root, CLI_ENTRY)} export .`);
18
- }
19
- console.log(`Command: npx vite --force --config vite.config.ts --host ${host} --port ${port}`);
20
- return 0;
21
- }
22
- if (!options.noBuild) {
23
- await exportDocument(root);
24
- }
25
-
26
- // One-line update notice (24h cached, network failure is silent).
27
- await printDoctorNoticeIfStale(root);
28
-
29
- console.log(`OpenPress dev: ${url}`);
30
- return runCommand("npx", ["vite", "--force", "--config", "vite.config.ts", "--host", host, "--port", port], root);
31
- }
32
-
33
- async function printDoctorNoticeIfStale(root) {
34
- try {
35
- const report = await diagnose(root);
36
- if (!report.stale) return;
37
- const parts = [];
38
- if (report.coreUpdateAvailable) {
39
- parts.push(`@open-press/core ${report.coreVersion} → ${report.coreLatest}`);
40
- }
41
- if (report.pendingMigrations.length > 0) {
42
- parts.push(`${report.pendingMigrations.length} migration note(s)`);
43
- }
44
- if (parts.length === 0) return;
45
- console.log(`○ open-press: ${parts.join(" · ")} — run \`npx open-press doctor\` for details.`);
46
- } catch {
47
- // Doctor is informational only; never block dev.
48
- }
49
- }
@@ -1,229 +0,0 @@
1
- import { existsSync } from "node:fs";
2
- import { mkdir, readFile, stat, writeFile } from "node:fs/promises";
3
- import path from "node:path";
4
-
5
- const CACHE_TTL_MS = 24 * 60 * 60 * 1000; // 24h
6
- const CORE_PACKAGE = "@open-press/core";
7
-
8
- export async function run({ root, options }) {
9
- const json = Boolean(options?.json);
10
- const noCache = Boolean(options?.noCache);
11
-
12
- const report = await diagnose(root, { noCache });
13
-
14
- if (json) {
15
- process.stdout.write(JSON.stringify(report, null, 2) + "\n");
16
- } else {
17
- printHumanReport(report);
18
- }
19
-
20
- // Exit 0 even when stale — doctor is informational, not a gate.
21
- // Agents / CI can check report.stale or report.coreUpdateAvailable.
22
- return 0;
23
- }
24
-
25
- /**
26
- * Diagnose workspace against latest framework state.
27
- * Result shape:
28
- * {
29
- * coreVersion: "0.4.0", // installed
30
- * coreLatest: "0.5.0" | null, // null on network failure
31
- * coreUpdateAvailable: boolean,
32
- * skillsInstalled: ["openpress", ...],
33
- * skillsLockSource: "quan0715/open-press" | null,
34
- * pendingMigrations: ["0.5.0"], // versions with docs/migrations notes
35
- * stale: boolean, // either core or skills behind
36
- * cachedAt: ISO timestamp
37
- * }
38
- */
39
- export async function diagnose(root, { noCache = false } = {}) {
40
- const cachePath = path.join(root, ".openpress", "cache", "doctor.json");
41
-
42
- if (!noCache) {
43
- const cached = await readCached(cachePath);
44
- if (cached) return cached;
45
- }
46
-
47
- const coreVersion = await readCoreVersion(root);
48
- const coreLatest = await fetchCoreLatest();
49
- const skillsInstalled = await listInstalledSkills(root);
50
- const skillsLockSource = await readSkillsLockSource(root);
51
- const pendingMigrations = await listPendingMigrations(root, coreVersion, coreLatest);
52
-
53
- const coreUpdateAvailable = Boolean(
54
- coreVersion && coreLatest && coreVersion !== coreLatest && semverLt(coreVersion, coreLatest),
55
- );
56
-
57
- const report = {
58
- coreVersion,
59
- coreLatest,
60
- coreUpdateAvailable,
61
- skillsInstalled,
62
- skillsLockSource,
63
- pendingMigrations,
64
- stale: coreUpdateAvailable || pendingMigrations.length > 0,
65
- cachedAt: new Date().toISOString(),
66
- };
67
-
68
- await writeCached(cachePath, report).catch(() => {});
69
- return report;
70
- }
71
-
72
- async function readCached(cachePath) {
73
- try {
74
- const stats = await stat(cachePath);
75
- if (Date.now() - stats.mtimeMs > CACHE_TTL_MS) return null;
76
- return JSON.parse(await readFile(cachePath, "utf8"));
77
- } catch {
78
- return null;
79
- }
80
- }
81
-
82
- async function writeCached(cachePath, report) {
83
- await mkdir(path.dirname(cachePath), { recursive: true });
84
- await writeFile(cachePath, JSON.stringify(report, null, 2) + "\n", "utf8");
85
- }
86
-
87
- async function readCoreVersion(root) {
88
- // Try workspace package.json deps first; fall back to installed package.
89
- try {
90
- const pkg = JSON.parse(await readFile(path.join(root, "package.json"), "utf8"));
91
- const range = pkg.dependencies?.[CORE_PACKAGE] ?? pkg.devDependencies?.[CORE_PACKAGE];
92
- if (range) {
93
- // Try the installed version (more accurate than the range).
94
- try {
95
- const installed = JSON.parse(
96
- await readFile(path.join(root, "node_modules", CORE_PACKAGE, "package.json"), "utf8"),
97
- );
98
- return installed.version;
99
- } catch {
100
- return range.replace(/^[\^~>=<\s]+/, "");
101
- }
102
- }
103
- } catch {}
104
-
105
- // Self-bundled framework (cli scaffolded workspace): pkg.version is the framework version.
106
- try {
107
- const pkg = JSON.parse(await readFile(path.join(root, "package.json"), "utf8"));
108
- if (pkg.name === CORE_PACKAGE) return pkg.version;
109
- } catch {}
110
-
111
- return null;
112
- }
113
-
114
- async function fetchCoreLatest() {
115
- try {
116
- const res = await fetch(`https://registry.npmjs.org/${CORE_PACKAGE}/latest`, {
117
- headers: { Accept: "application/json" },
118
- signal: AbortSignal.timeout(5000),
119
- });
120
- if (!res.ok) return null;
121
- const data = await res.json();
122
- return typeof data.version === "string" ? data.version : null;
123
- } catch {
124
- return null;
125
- }
126
- }
127
-
128
- async function listInstalledSkills(root) {
129
- const skillsDir = path.join(root, ".agents", "skills");
130
- try {
131
- const { readdir } = await import("node:fs/promises");
132
- const entries = await readdir(skillsDir, { withFileTypes: true });
133
- return entries.filter((e) => e.isDirectory()).map((e) => e.name).sort();
134
- } catch {
135
- return [];
136
- }
137
- }
138
-
139
- async function readSkillsLockSource(root) {
140
- try {
141
- const lock = JSON.parse(await readFile(path.join(root, "skills-lock.json"), "utf8"));
142
- const sources = lock?.sources;
143
- if (Array.isArray(sources) && sources.length > 0) return sources[0]?.source ?? null;
144
- return null;
145
- } catch {
146
- return null;
147
- }
148
- }
149
-
150
- async function listPendingMigrations(root, currentVersion, latestVersion) {
151
- if (!currentVersion || !latestVersion || !semverLt(currentVersion, latestVersion)) return [];
152
- // Look for docs/migrations/<version>.md files for versions in (current, latest].
153
- const migrationsDir = path.join(root, "docs", "migrations");
154
- try {
155
- const { readdir } = await import("node:fs/promises");
156
- const files = await readdir(migrationsDir);
157
- return files
158
- .filter((f) => /^\d+\.\d+\.\d+\.md$/.test(f))
159
- .map((f) => f.replace(/\.md$/, ""))
160
- .filter((v) => semverGt(v, currentVersion) && !semverGt(v, latestVersion))
161
- .sort(semverCompare);
162
- } catch {
163
- return [];
164
- }
165
- }
166
-
167
- function semverParse(v) {
168
- const m = /^(\d+)\.(\d+)\.(\d+)/.exec(v);
169
- if (!m) return [0, 0, 0];
170
- return [Number(m[1]), Number(m[2]), Number(m[3])];
171
- }
172
- function semverCompare(a, b) {
173
- const A = semverParse(a);
174
- const B = semverParse(b);
175
- for (let i = 0; i < 3; i++) if (A[i] !== B[i]) return A[i] - B[i];
176
- return 0;
177
- }
178
- function semverLt(a, b) { return semverCompare(a, b) < 0; }
179
- function semverGt(a, b) { return semverCompare(a, b) > 0; }
180
-
181
- function printHumanReport(report) {
182
- const lines = [];
183
- lines.push("○ open-press doctor");
184
- lines.push("");
185
- lines.push("framework");
186
- if (report.coreVersion) {
187
- if (report.coreLatest === null) {
188
- lines.push(` ? @open-press/core: ${report.coreVersion} installed (couldn't check latest — offline?)`);
189
- } else if (report.coreUpdateAvailable) {
190
- lines.push(` ⚠ @open-press/core: ${report.coreVersion} installed → ${report.coreLatest} available`);
191
- } else {
192
- lines.push(` ✓ @open-press/core: ${report.coreVersion} (latest)`);
193
- }
194
- } else {
195
- lines.push(" ? @open-press/core: not detected in this workspace");
196
- }
197
- lines.push("");
198
- lines.push("skills");
199
- if (report.skillsInstalled.length === 0) {
200
- lines.push(" ? no skills installed under .agents/skills/");
201
- lines.push(" run: npx skills add quan0715/open-press");
202
- } else {
203
- lines.push(` ✓ ${report.skillsInstalled.length} skills installed`);
204
- if (report.skillsLockSource) {
205
- lines.push(` source: ${report.skillsLockSource}`);
206
- lines.push(" refresh: npx skills upgrade");
207
- }
208
- }
209
- lines.push("");
210
- lines.push("migrations");
211
- if (report.pendingMigrations.length === 0) {
212
- if (report.coreUpdateAvailable) {
213
- lines.push(` ✓ no breaking migrations documented for the ${report.coreLatest} window`);
214
- } else {
215
- lines.push(" ✓ up to date");
216
- }
217
- } else {
218
- lines.push(` ⚠ ${report.pendingMigrations.length} migration note(s) since your version:`);
219
- for (const v of report.pendingMigrations) lines.push(` - docs/migrations/${v}.md`);
220
- }
221
- lines.push("");
222
- if (report.stale) {
223
- lines.push("next");
224
- lines.push(" npx open-press upgrade # apply all updates (agent-driven)");
225
- lines.push(" npx open-press doctor --json # machine-readable output");
226
- lines.push("");
227
- }
228
- process.stdout.write(lines.join("\n"));
229
- }
@@ -1,8 +0,0 @@
1
- import path from "node:path";
2
- import { exportDocument } from "../document-export.mjs";
3
-
4
- export async function run({ root }) {
5
- const result = await exportDocument(root);
6
- console.log(`OpenPress export: ${path.relative(root, result.documentPath)} (${result.pageCount} pages)`);
7
- return 0;
8
- }
@@ -1,24 +0,0 @@
1
- import { initWorkspace, listStylePackSkills } from "../init.mjs";
2
- import { formatDisplayPath, parseInitOptions } from "./_shared.mjs";
3
-
4
- export const needsWorkspace = false;
5
-
6
- export async function run({ argv }) {
7
- const options = parseInitOptions(argv);
8
- if (!options.target) {
9
- console.error("openpress init: target path is required");
10
- console.error("Usage: openpress init <target> [--skill <name>] [--force]");
11
- const available = await listStylePackSkills();
12
- if (available.length) console.error(`Style packs available: ${available.join(", ")}`);
13
- return 1;
14
- }
15
- const result = await initWorkspace(options);
16
- const displayPath = formatDisplayPath(result.targetPath);
17
- console.log(`OpenPress init: created ${displayPath} from style pack "${result.skill}".`);
18
- console.log("Next steps:");
19
- console.log(` cd ${displayPath}`);
20
- console.log(" # 填入 openpress.config.mjs 的 title / subtitle / organization");
21
- console.log(" # 改 document/index.tsx 與 document/chapters/**/*.mdx 為實際內容");
22
- console.log(" node engine/cli.mjs validate");
23
- return 0;
24
- }
@@ -1,35 +0,0 @@
1
- import { inspectWorkspace } from "../runtime/inspection.mjs";
2
- import { exitCodeForIssueReport } from "../runtime/issue-report.mjs";
3
-
4
- export async function run({ root, config, options, recurse }) {
5
- const host = options.host ?? "127.0.0.1";
6
- const port = options.port ?? "5186";
7
- const url = `http://${host}:${port}/?print=1`;
8
-
9
- if (options.dryRun) {
10
- if (!options.noBuild) {
11
- console.log("Command: node engine/cli.mjs render . --renderer react");
12
- }
13
- console.log(`Command: node engine/output/static-server.mjs ${config.outputDir} --host ${host} --port ${port} --workspace .`);
14
- console.log(`Chrome inspection URL: ${url}`);
15
- return 0;
16
- }
17
-
18
- const report = await inspectWorkspace({ root, config, options, recurse });
19
- if (options.json) {
20
- console.log(JSON.stringify(report, null, 2));
21
- return exitCodeForIssueReport(report);
22
- }
23
-
24
- if (report.ok) {
25
- console.log(report.format());
26
- console.log(`Checked: ${report.checked.join(", ")}`);
27
- if (report.summary) {
28
- console.log(`Summary: ${JSON.stringify(report.summary)}`);
29
- }
30
- return 0;
31
- }
32
-
33
- console.log(report.format());
34
- return exitCodeForIssueReport(report);
35
- }
@@ -1,26 +0,0 @@
1
- import path from "node:path";
2
- import { CLI_ENTRY, STATIC_SERVER, buildReactPdf, formatNodeScriptCommand } from "./_shared.mjs";
3
-
4
- export async function run({ root, config, options, recurse }) {
5
- const outputPath = options.output ? path.resolve(root, options.output) : undefined;
6
- if (options.dryRun) {
7
- const relOutput = path.relative(root, outputPath ?? config.paths.pdf);
8
- const host = options.host ?? "127.0.0.1";
9
- const port = options.port ?? "5185";
10
- console.log(`Command: ${formatNodeScriptCommand(root, CLI_ENTRY)} render . --renderer react`);
11
- console.log(`Command: ${formatNodeScriptCommand(root, STATIC_SERVER)} ${config.outputDir} --host ${host} --port ${port} --workspace .`);
12
- console.log(`Command: Chrome --print-to-pdf=${relOutput} http://${host}:${port}/?print=1`);
13
- return 0;
14
- }
15
- const result = await buildReactPdf({
16
- root,
17
- config,
18
- outPath: outputPath,
19
- host: options.host,
20
- port: options.port,
21
- noBuild: options.noBuild,
22
- recurse,
23
- });
24
- console.log(`OpenPress PDF: ${path.relative(root, result.pdfPath)}`);
25
- return 0;
26
- }
@@ -1,26 +0,0 @@
1
- import { CLI_ENTRY, STATIC_SERVER, formatNodeScriptCommand, runCommand } from "./_shared.mjs";
2
-
3
- export async function run({ root, config, options, recurse }) {
4
- const renderer = options.renderer ?? "react";
5
- if (renderer !== "react") {
6
- console.error(`Unknown renderer: ${renderer}`);
7
- return 2;
8
- }
9
- const host = options.host ?? "127.0.0.1";
10
- const port = options.port ?? "5173";
11
- const url = `http://${host}:${port}`;
12
- if (options.dryRun) {
13
- console.log(`OpenPress preview URL: ${url}`);
14
- if (!options.noBuild) {
15
- console.log(`Command: ${formatNodeScriptCommand(root, CLI_ENTRY)} render . --renderer react`);
16
- }
17
- console.log(`Command: ${formatNodeScriptCommand(root, STATIC_SERVER)} ${config.outputDir} --host ${host} --port ${port} --workspace .`);
18
- return 0;
19
- }
20
- if (!options.noBuild) {
21
- const renderCode = await recurse("render", [root, "--renderer", renderer]);
22
- if (renderCode !== 0) return renderCode;
23
- }
24
- console.log(`OpenPress preview: ${url}`);
25
- return runCommand("node", [STATIC_SERVER, config.outputDir, "--host", host, "--port", port, "--workspace", "."], root);
26
- }
@@ -1,17 +0,0 @@
1
- import { exportDocument } from "../document-export.mjs";
2
- import { runCommand } from "./_shared.mjs";
3
-
4
- export async function run({ root, options }) {
5
- const renderer = options.renderer ?? "react";
6
- if (renderer !== "react") {
7
- console.error(`Unknown renderer: ${renderer}`);
8
- return 2;
9
- }
10
- if (options.dryRun) {
11
- console.log("Command: node engine/cli.mjs export .");
12
- console.log("Command: npx vite build --config vite.config.ts");
13
- return 0;
14
- }
15
- await exportDocument(root);
16
- return runCommand("npx", ["vite", "build", "--config", "vite.config.ts"], root);
17
- }
@@ -1,41 +0,0 @@
1
- import { replaceSourceText } from "../runtime/source-text-tools.mjs";
2
-
3
- export async function run({ config, options }) {
4
- const args = replaceArgsFromOptions(options);
5
- if (!args) {
6
- console.error("Usage: node engine/cli.mjs replace [path] <from> <to> [--json] [--apply] [--scope content|all] [--include-code] [--case-sensitive]");
7
- return 2;
8
- }
9
-
10
- const report = await replaceSourceText({
11
- config,
12
- from: args.from,
13
- to: args.to,
14
- scope: options.scope ?? "content",
15
- caseSensitive: options.caseSensitive === true,
16
- includeCode: options.includeCode === true,
17
- apply: options.apply === true,
18
- });
19
-
20
- if (options.json) {
21
- console.log(JSON.stringify(report, null, 2));
22
- return 0;
23
- }
24
-
25
- console.log(`OpenPress replace ${report.applied ? "applied" : "preview"}: "${args.from}" -> "${args.to}" (${report.matchCount} matches in ${report.fileCount} files)`);
26
- if (!report.applied) console.log("No files written. Re-run with --apply to update sources.");
27
- for (const change of report.changes) {
28
- console.log(`${change.path}: ${change.replacements.length} replacements`);
29
- }
30
- return 0;
31
- }
32
-
33
- function replaceArgsFromOptions(options) {
34
- const positional = options.positional ?? [];
35
- const args = positional.length >= 3 ? positional.slice(1) : positional;
36
- if (args.length < 2) return null;
37
- return {
38
- from: args[0],
39
- to: args[1],
40
- };
41
- }