@unbrained/pm-cli 2026.5.24 → 2026.5.28

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 (246) hide show
  1. package/CHANGELOG.md +953 -522
  2. package/README.md +2 -10
  3. package/dist/cli/bootstrap-args.d.ts +18 -1
  4. package/dist/cli/bootstrap-args.js +143 -3
  5. package/dist/cli/bootstrap-args.js.map +1 -1
  6. package/dist/cli/commander-usage.js +134 -11
  7. package/dist/cli/commander-usage.js.map +1 -1
  8. package/dist/cli/commands/append.js +4 -3
  9. package/dist/cli/commands/append.js.map +1 -1
  10. package/dist/cli/commands/claim.js +5 -4
  11. package/dist/cli/commands/claim.js.map +1 -1
  12. package/dist/cli/commands/close.d.ts +3 -0
  13. package/dist/cli/commands/close.js +26 -3
  14. package/dist/cli/commands/close.js.map +1 -1
  15. package/dist/cli/commands/completion.d.ts +2 -2
  16. package/dist/cli/commands/completion.js +109 -56
  17. package/dist/cli/commands/completion.js.map +1 -1
  18. package/dist/cli/commands/config.d.ts +1 -1
  19. package/dist/cli/commands/config.js +82 -4
  20. package/dist/cli/commands/config.js.map +1 -1
  21. package/dist/cli/commands/create.js +7 -272
  22. package/dist/cli/commands/create.js.map +1 -1
  23. package/dist/cli/commands/delete.js +4 -3
  24. package/dist/cli/commands/delete.js.map +1 -1
  25. package/dist/cli/commands/docs.d.ts +1 -12
  26. package/dist/cli/commands/docs.js +8 -312
  27. package/dist/cli/commands/docs.js.map +1 -1
  28. package/dist/cli/commands/extension/bundled-catalog.d.ts +14 -0
  29. package/dist/cli/commands/extension/bundled-catalog.js +268 -0
  30. package/dist/cli/commands/extension/bundled-catalog.js.map +1 -0
  31. package/dist/cli/commands/extension/doctor.d.ts +31 -0
  32. package/dist/cli/commands/extension/doctor.js +345 -0
  33. package/dist/cli/commands/extension/doctor.js.map +1 -0
  34. package/dist/cli/commands/extension/install-sources.d.ts +37 -0
  35. package/dist/cli/commands/extension/install-sources.js +384 -0
  36. package/dist/cli/commands/extension/install-sources.js.map +1 -0
  37. package/dist/cli/commands/extension/managed-state.d.ts +48 -0
  38. package/dist/cli/commands/extension/managed-state.js +172 -0
  39. package/dist/cli/commands/extension/managed-state.js.map +1 -0
  40. package/dist/cli/commands/extension/scaffold.d.ts +14 -0
  41. package/dist/cli/commands/extension/scaffold.js +202 -0
  42. package/dist/cli/commands/extension/scaffold.js.map +1 -0
  43. package/dist/cli/commands/extension/shared.d.ts +14 -0
  44. package/dist/cli/commands/extension/shared.js +106 -0
  45. package/dist/cli/commands/extension/shared.js.map +1 -0
  46. package/dist/cli/commands/extension.d.ts +36 -68
  47. package/dist/cli/commands/extension.js +143 -1422
  48. package/dist/cli/commands/extension.js.map +1 -1
  49. package/dist/cli/commands/files.d.ts +1 -12
  50. package/dist/cli/commands/files.js +11 -308
  51. package/dist/cli/commands/files.js.map +1 -1
  52. package/dist/cli/commands/get.js +4 -3
  53. package/dist/cli/commands/get.js.map +1 -1
  54. package/dist/cli/commands/health.js +17 -3
  55. package/dist/cli/commands/health.js.map +1 -1
  56. package/dist/cli/commands/history-redact.js +23 -18
  57. package/dist/cli/commands/history-redact.js.map +1 -1
  58. package/dist/cli/commands/history-repair.js +24 -18
  59. package/dist/cli/commands/history-repair.js.map +1 -1
  60. package/dist/cli/commands/legacy-none-tokens.d.ts +3 -0
  61. package/dist/cli/commands/legacy-none-tokens.js +39 -0
  62. package/dist/cli/commands/legacy-none-tokens.js.map +1 -0
  63. package/dist/cli/commands/linked-artifacts.d.ts +96 -0
  64. package/dist/cli/commands/linked-artifacts.js +335 -0
  65. package/dist/cli/commands/linked-artifacts.js.map +1 -0
  66. package/dist/cli/commands/linked-test-parsers.d.ts +28 -0
  67. package/dist/cli/commands/linked-test-parsers.js +192 -0
  68. package/dist/cli/commands/linked-test-parsers.js.map +1 -0
  69. package/dist/cli/commands/list.js +19 -5
  70. package/dist/cli/commands/list.js.map +1 -1
  71. package/dist/cli/commands/normalize.js +4 -3
  72. package/dist/cli/commands/normalize.js.map +1 -1
  73. package/dist/cli/commands/plan.d.ts +5 -0
  74. package/dist/cli/commands/plan.js +56 -8
  75. package/dist/cli/commands/plan.js.map +1 -1
  76. package/dist/cli/commands/recurrence-parsers.d.ts +26 -0
  77. package/dist/cli/commands/recurrence-parsers.js +98 -0
  78. package/dist/cli/commands/recurrence-parsers.js.map +1 -0
  79. package/dist/cli/commands/restore.js +19 -8
  80. package/dist/cli/commands/restore.js.map +1 -1
  81. package/dist/cli/commands/search.js +5 -8
  82. package/dist/cli/commands/search.js.map +1 -1
  83. package/dist/cli/commands/test/linked-command-detection.d.ts +37 -0
  84. package/dist/cli/commands/test/linked-command-detection.js +200 -0
  85. package/dist/cli/commands/test/linked-command-detection.js.map +1 -0
  86. package/dist/cli/commands/test.d.ts +1 -2
  87. package/dist/cli/commands/test.js +8 -350
  88. package/dist/cli/commands/test.js.map +1 -1
  89. package/dist/cli/commands/update-many.js +4 -3
  90. package/dist/cli/commands/update-many.js.map +1 -1
  91. package/dist/cli/commands/update.js +83 -356
  92. package/dist/cli/commands/update.js.map +1 -1
  93. package/dist/cli/commands/validate.js +32 -12
  94. package/dist/cli/commands/validate.js.map +1 -1
  95. package/dist/cli/error-guidance.d.ts +1 -0
  96. package/dist/cli/error-guidance.js +6 -2
  97. package/dist/cli/error-guidance.js.map +1 -1
  98. package/dist/cli/main.d.ts +11 -0
  99. package/dist/cli/main.js +76 -28
  100. package/dist/cli/main.js.map +1 -1
  101. package/dist/cli/register-list-query.d.ts +4 -1
  102. package/dist/cli/register-list-query.js +242 -203
  103. package/dist/cli/register-list-query.js.map +1 -1
  104. package/dist/cli/register-mutation.js +73 -11
  105. package/dist/cli/register-mutation.js.map +1 -1
  106. package/dist/cli/register-operations.js +3 -3
  107. package/dist/cli/register-operations.js.map +1 -1
  108. package/dist/cli/register-setup.js +12 -7
  109. package/dist/cli/register-setup.js.map +1 -1
  110. package/dist/cli/registration-helpers.js +3 -2
  111. package/dist/cli/registration-helpers.js.map +1 -1
  112. package/dist/cli.js +4 -3
  113. package/dist/cli.js.map +1 -1
  114. package/dist/core/config/positional-value.d.ts +44 -0
  115. package/dist/core/config/positional-value.js +109 -0
  116. package/dist/core/config/positional-value.js.map +1 -0
  117. package/dist/core/extensions/extension-capability-aliases.d.ts +14 -0
  118. package/dist/core/extensions/extension-capability-aliases.js +159 -0
  119. package/dist/core/extensions/extension-capability-aliases.js.map +1 -0
  120. package/dist/core/extensions/extension-hook-runtime.d.ts +13 -0
  121. package/dist/core/extensions/extension-hook-runtime.js +414 -0
  122. package/dist/core/extensions/extension-hook-runtime.js.map +1 -0
  123. package/dist/core/extensions/extension-policy.d.ts +69 -0
  124. package/dist/core/extensions/extension-policy.js +481 -0
  125. package/dist/core/extensions/extension-policy.js.map +1 -0
  126. package/dist/core/extensions/extension-registries.d.ts +8 -0
  127. package/dist/core/extensions/extension-registries.js +52 -0
  128. package/dist/core/extensions/extension-registries.js.map +1 -0
  129. package/dist/core/extensions/extension-runtime-helpers.d.ts +6 -0
  130. package/dist/core/extensions/extension-runtime-helpers.js +29 -0
  131. package/dist/core/extensions/extension-runtime-helpers.js.map +1 -0
  132. package/dist/core/extensions/extension-types.d.ts +13 -39
  133. package/dist/core/extensions/extension-types.js +34 -2
  134. package/dist/core/extensions/extension-types.js.map +1 -1
  135. package/dist/core/extensions/index.d.ts +7 -0
  136. package/dist/core/extensions/index.js +11 -2
  137. package/dist/core/extensions/index.js.map +1 -1
  138. package/dist/core/extensions/loader.d.ts +4 -22
  139. package/dist/core/extensions/loader.js +22 -1139
  140. package/dist/core/extensions/loader.js.map +1 -1
  141. package/dist/core/history/drift-scan.d.ts +11 -0
  142. package/dist/core/history/drift-scan.js +114 -32
  143. package/dist/core/history/drift-scan.js.map +1 -1
  144. package/dist/core/history/history-rewrite.d.ts +43 -0
  145. package/dist/core/history/history-rewrite.js +48 -0
  146. package/dist/core/history/history-rewrite.js.map +1 -0
  147. package/dist/core/history/history.js +5 -4
  148. package/dist/core/history/history.js.map +1 -1
  149. package/dist/core/history/replay.js +4 -3
  150. package/dist/core/history/replay.js.map +1 -1
  151. package/dist/core/item/item-record.d.ts +19 -0
  152. package/dist/core/item/item-record.js +24 -0
  153. package/dist/core/item/item-record.js.map +1 -0
  154. package/dist/core/output/mutation-projection.d.ts +31 -0
  155. package/dist/core/output/mutation-projection.js +103 -0
  156. package/dist/core/output/mutation-projection.js.map +1 -0
  157. package/dist/core/output/output.d.ts +2 -0
  158. package/dist/core/output/output.js +5 -3
  159. package/dist/core/output/output.js.map +1 -1
  160. package/dist/core/schema/runtime-schema.js +8 -38
  161. package/dist/core/schema/runtime-schema.js.map +1 -1
  162. package/dist/core/search/vector-stores.js +46 -9
  163. package/dist/core/search/vector-stores.js.map +1 -1
  164. package/dist/core/sentry/helpers.d.ts +1 -1
  165. package/dist/core/sentry/helpers.js +20 -3
  166. package/dist/core/sentry/helpers.js.map +1 -1
  167. package/dist/core/shared/command-types.d.ts +1 -0
  168. package/dist/core/shared/command-types.js +2 -2
  169. package/dist/core/shared/command-types.js.map +1 -1
  170. package/dist/core/shared/constants.d.ts +10 -1
  171. package/dist/core/shared/constants.js +56 -58
  172. package/dist/core/shared/constants.js.map +1 -1
  173. package/dist/core/shared/levenshtein.js +23 -7
  174. package/dist/core/shared/levenshtein.js.map +1 -1
  175. package/dist/core/shared/primitives.d.ts +23 -0
  176. package/dist/core/shared/primitives.js +39 -2
  177. package/dist/core/shared/primitives.js.map +1 -1
  178. package/dist/core/store/front-matter-cache.d.ts +16 -2
  179. package/dist/core/store/front-matter-cache.js +99 -33
  180. package/dist/core/store/front-matter-cache.js.map +1 -1
  181. package/dist/core/store/item-store.js +8 -73
  182. package/dist/core/store/item-store.js.map +1 -1
  183. package/dist/mcp/server.js +76 -28
  184. package/dist/mcp/server.js.map +1 -1
  185. package/dist/sdk/cli-contracts/enum-contracts.d.ts +20 -0
  186. package/dist/sdk/cli-contracts/enum-contracts.js +156 -0
  187. package/dist/sdk/cli-contracts/enum-contracts.js.map +1 -0
  188. package/dist/sdk/cli-contracts/tool-option-contracts.d.ts +14 -0
  189. package/dist/sdk/cli-contracts/tool-option-contracts.js +243 -0
  190. package/dist/sdk/cli-contracts/tool-option-contracts.js.map +1 -0
  191. package/dist/sdk/cli-contracts/tool-parameter-tables.d.ts +11 -0
  192. package/dist/sdk/cli-contracts/tool-parameter-tables.js +901 -0
  193. package/dist/sdk/cli-contracts/tool-parameter-tables.js.map +1 -0
  194. package/dist/sdk/cli-contracts.d.ts +11 -33
  195. package/dist/sdk/cli-contracts.js +30 -1356
  196. package/dist/sdk/cli-contracts.js.map +1 -1
  197. package/dist/sdk/package-import-adapters.d.ts +74 -0
  198. package/dist/sdk/package-import-adapters.js +186 -0
  199. package/dist/sdk/package-import-adapters.js.map +1 -0
  200. package/dist/sdk/package-runtime-options.d.ts +26 -0
  201. package/dist/sdk/package-runtime-options.js +71 -0
  202. package/dist/sdk/package-runtime-options.js.map +1 -0
  203. package/dist/sdk/runtime.d.ts +2 -0
  204. package/dist/sdk/runtime.js +4 -2
  205. package/dist/sdk/runtime.js.map +1 -1
  206. package/docs/AGENT_GUIDE.md +6 -10
  207. package/docs/CLAUDE_CODE_PLUGIN.md +5 -28
  208. package/docs/CODEX_PLUGIN.md +5 -5
  209. package/docs/COMMANDS.md +19 -3
  210. package/docs/CONFIGURATION.md +15 -0
  211. package/docs/EXTENSIONS.md +4 -63
  212. package/docs/RELEASING.md +4 -4
  213. package/marketplace.json +7 -3
  214. package/package.json +9 -6
  215. package/packages/pm-beads/extensions/beads/index.js +2 -49
  216. package/packages/pm-beads/extensions/beads/index.ts +2 -54
  217. package/packages/pm-beads/extensions/beads/runtime-loader.js +86 -0
  218. package/packages/pm-beads/extensions/beads/runtime-loader.ts +88 -0
  219. package/packages/pm-beads/extensions/beads/runtime.js +26 -115
  220. package/packages/pm-beads/extensions/beads/runtime.ts +33 -132
  221. package/packages/pm-calendar/extensions/calendar/index.js +47 -2
  222. package/packages/pm-calendar/extensions/calendar/index.ts +52 -2
  223. package/packages/pm-calendar/extensions/calendar/runtime.js +1 -0
  224. package/packages/pm-calendar/extensions/calendar/runtime.ts +1 -0
  225. package/packages/pm-governance-audit/extensions/governance-audit/runtime.js +14 -41
  226. package/packages/pm-governance-audit/extensions/governance-audit/runtime.ts +25 -41
  227. package/packages/pm-guide-shell/extensions/guide-shell/runtime.js +10 -50
  228. package/packages/pm-guide-shell/extensions/guide-shell/runtime.ts +17 -50
  229. package/packages/pm-linked-test-adapters/extensions/linked-test-adapters/runtime.js +8 -40
  230. package/packages/pm-linked-test-adapters/extensions/linked-test-adapters/runtime.ts +10 -40
  231. package/packages/pm-search-advanced/extensions/search-advanced/index.js +1 -1
  232. package/packages/pm-search-advanced/extensions/search-advanced/runtime.js +4 -37
  233. package/packages/pm-search-advanced/extensions/search-advanced/runtime.ts +6 -37
  234. package/packages/pm-todos/extensions/todos/index.js +3 -50
  235. package/packages/pm-todos/extensions/todos/index.ts +3 -55
  236. package/packages/pm-todos/extensions/todos/runtime-loader.js +86 -0
  237. package/packages/pm-todos/extensions/todos/runtime-loader.ts +88 -0
  238. package/packages/pm-todos/extensions/todos/runtime.js +24 -117
  239. package/packages/pm-todos/extensions/todos/runtime.ts +32 -129
  240. package/plugins/pm-claude/README.md +2 -2
  241. package/plugins/pm-claude/commands/pm-planner.md +1 -15
  242. package/plugins/pm-claude/scripts/pm-mcp-server.mjs +5 -2
  243. package/plugins/pm-claude/skills/pm-planner/SKILL.md +3 -21
  244. package/plugins/pm-codex/scripts/pm-mcp-server.mjs +15 -6
  245. package/plugins/pm-codex/skills/pm-native/SKILL.md +1 -13
  246. package/PRD.md +0 -1734
@@ -0,0 +1,172 @@
1
+
2
+ !function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="f7dc1712-641e-52f3-bbd9-7f20eafddd5f")}catch(e){}}();
3
+ import fs from "node:fs/promises";
4
+ import path from "node:path";
5
+ import { EXIT_CODE } from "../../../core/shared/constants.js";
6
+ import { PmCliError } from "../../../core/shared/errors.js";
7
+ import { nowIso } from "../../../core/shared/time.js";
8
+ import { normalizeExtensionNameForMatch, normalizeStringList } from "./shared.js";
9
+ const MANAGED_EXTENSION_STATE_FILENAME = ".managed-extensions.json";
10
+ const MANAGED_EXTENSION_STATE_VERSION = 1;
11
+ export function resolveManagedExtensionStatePath(extensionsRoot) {
12
+ return path.join(extensionsRoot, MANAGED_EXTENSION_STATE_FILENAME);
13
+ }
14
+ export function createEmptyManagedExtensionState() {
15
+ return {
16
+ version: MANAGED_EXTENSION_STATE_VERSION,
17
+ updated_at: nowIso(),
18
+ entries: [],
19
+ };
20
+ }
21
+ export function sortManagedEntries(entries) {
22
+ return [...entries].sort((left, right) => {
23
+ const byScope = left.scope.localeCompare(right.scope);
24
+ if (byScope !== 0) {
25
+ return byScope;
26
+ }
27
+ const byName = left.name.localeCompare(right.name);
28
+ if (byName !== 0) {
29
+ return byName;
30
+ }
31
+ return left.directory.localeCompare(right.directory);
32
+ });
33
+ }
34
+ export function managedExtensionSourcesEquivalent(left, right) {
35
+ if (left.kind !== right.kind || left.input !== right.input || left.location !== right.location) {
36
+ return false;
37
+ }
38
+ if (left.kind === "npm" && right.kind === "npm") {
39
+ return left.package === right.package && left.version === right.version;
40
+ }
41
+ if (left.kind === "github" && right.kind === "github") {
42
+ return left.repository === right.repository && left.ref === right.ref && left.subpath === right.subpath && left.commit === right.commit;
43
+ }
44
+ if (left.kind === "builtin" && right.kind === "builtin") {
45
+ return left.name === right.name;
46
+ }
47
+ return true;
48
+ }
49
+ export function normalizeManagedState(raw) {
50
+ if (typeof raw !== "object" || raw === null) {
51
+ return null;
52
+ }
53
+ const candidate = raw;
54
+ if (candidate.version !== MANAGED_EXTENSION_STATE_VERSION || !Array.isArray(candidate.entries)) {
55
+ return null;
56
+ }
57
+ const entries = [];
58
+ for (const rawEntry of candidate.entries) {
59
+ if (typeof rawEntry !== "object" || rawEntry === null) {
60
+ continue;
61
+ }
62
+ const entry = rawEntry;
63
+ if (typeof entry.name !== "string" ||
64
+ entry.name.trim().length === 0 ||
65
+ typeof entry.directory !== "string" ||
66
+ entry.directory.trim().length === 0 ||
67
+ (entry.scope !== "project" && entry.scope !== "global") ||
68
+ typeof entry.manifest_version !== "string" ||
69
+ typeof entry.manifest_entry !== "string" ||
70
+ !Array.isArray(entry.capabilities) ||
71
+ entry.capabilities.some((value) => typeof value !== "string") ||
72
+ typeof entry.installed_at !== "string" ||
73
+ typeof entry.updated_at !== "string" ||
74
+ typeof entry.source !== "object" ||
75
+ entry.source === null) {
76
+ continue;
77
+ }
78
+ const source = entry.source;
79
+ if ((source.kind !== "local" && source.kind !== "github" && source.kind !== "npm" && source.kind !== "builtin") ||
80
+ typeof source.input !== "string" ||
81
+ typeof source.location !== "string") {
82
+ continue;
83
+ }
84
+ entries.push({
85
+ name: entry.name.trim(),
86
+ directory: entry.directory.trim(),
87
+ scope: entry.scope,
88
+ manifest_version: entry.manifest_version,
89
+ manifest_entry: entry.manifest_entry,
90
+ capabilities: normalizeStringList(entry.capabilities),
91
+ installed_at: entry.installed_at,
92
+ updated_at: entry.updated_at,
93
+ source: {
94
+ kind: source.kind,
95
+ input: source.input,
96
+ location: source.location,
97
+ name: typeof source.name === "string" ? source.name : undefined,
98
+ package: typeof source.package === "string" ? source.package : undefined,
99
+ version: typeof source.version === "string" ? source.version : undefined,
100
+ repository: typeof source.repository === "string" ? source.repository : undefined,
101
+ owner: typeof source.owner === "string" ? source.owner : undefined,
102
+ repo: typeof source.repo === "string" ? source.repo : undefined,
103
+ ref: typeof source.ref === "string" ? source.ref : undefined,
104
+ subpath: typeof source.subpath === "string" ? source.subpath : undefined,
105
+ commit: typeof source.commit === "string" ? source.commit : undefined,
106
+ },
107
+ last_update_check_at: typeof entry.last_update_check_at === "string" ? entry.last_update_check_at : undefined,
108
+ last_update_remote_commit: typeof entry.last_update_remote_commit === "string" ? entry.last_update_remote_commit : undefined,
109
+ update_available: typeof entry.update_available === "boolean" || entry.update_available === null
110
+ ? entry.update_available
111
+ : undefined,
112
+ update_error: typeof entry.update_error === "string" ? entry.update_error : undefined,
113
+ });
114
+ }
115
+ return {
116
+ version: MANAGED_EXTENSION_STATE_VERSION,
117
+ updated_at: typeof candidate.updated_at === "string" ? candidate.updated_at : nowIso(),
118
+ entries: sortManagedEntries(entries),
119
+ };
120
+ }
121
+ export async function readManagedExtensionState(extensionsRoot) {
122
+ const statePath = resolveManagedExtensionStatePath(extensionsRoot);
123
+ const fallback = createEmptyManagedExtensionState();
124
+ try {
125
+ const raw = await fs.readFile(statePath, "utf8");
126
+ const parsed = JSON.parse(raw);
127
+ const normalized = normalizeManagedState(parsed);
128
+ if (!normalized) {
129
+ throw new PmCliError(`Managed extension state file "${statePath}" has an invalid schema. Repair or remove it before mutating extension state.`, EXIT_CODE.GENERIC_FAILURE);
130
+ }
131
+ return {
132
+ path: statePath,
133
+ state: normalized,
134
+ warnings: [],
135
+ };
136
+ }
137
+ catch (error) {
138
+ if (error instanceof PmCliError) {
139
+ throw error;
140
+ }
141
+ if (typeof error === "object" && error !== null && "code" in error && error.code === "ENOENT") {
142
+ return {
143
+ path: statePath,
144
+ state: fallback,
145
+ warnings: [],
146
+ };
147
+ }
148
+ throw new PmCliError(`Managed extension state file "${statePath}" could not be read. Repair or remove it before mutating extension state.`, EXIT_CODE.GENERIC_FAILURE);
149
+ }
150
+ }
151
+ export async function writeManagedExtensionState(extensionsRoot, state) {
152
+ const statePath = resolveManagedExtensionStatePath(extensionsRoot);
153
+ const normalized = {
154
+ version: MANAGED_EXTENSION_STATE_VERSION,
155
+ updated_at: nowIso(),
156
+ entries: sortManagedEntries(state.entries),
157
+ };
158
+ await fs.mkdir(extensionsRoot, { recursive: true });
159
+ await fs.writeFile(statePath, `${JSON.stringify(normalized, null, 2)}\n`, "utf8");
160
+ }
161
+ export function upsertManagedEntry(state, entry) {
162
+ const updatedEntries = state.entries.filter((candidate) => normalizeExtensionNameForMatch(candidate.name) !== normalizeExtensionNameForMatch(entry.name) &&
163
+ normalizeExtensionNameForMatch(candidate.directory) !== normalizeExtensionNameForMatch(entry.directory));
164
+ updatedEntries.push(entry);
165
+ return {
166
+ ...state,
167
+ updated_at: nowIso(),
168
+ entries: sortManagedEntries(updatedEntries),
169
+ };
170
+ }
171
+ //# sourceMappingURL=managed-state.js.map
172
+ //# debugId=f7dc1712-641e-52f3-bbd9-7f20eafddd5f
@@ -0,0 +1 @@
1
+ {"version":3,"file":"managed-state.js","sources":["cli/commands/extension/managed-state.ts"],"sourceRoot":"/","sourcesContent":["import fs from \"node:fs/promises\";\nimport path from \"node:path\";\nimport { EXIT_CODE } from \"../../../core/shared/constants.js\";\nimport { PmCliError } from \"../../../core/shared/errors.js\";\nimport { nowIso } from \"../../../core/shared/time.js\";\nimport { normalizeExtensionNameForMatch, normalizeStringList } from \"./shared.js\";\nimport type { ExtensionScope } from \"../extension.js\";\n\nconst MANAGED_EXTENSION_STATE_FILENAME = \".managed-extensions.json\";\nconst MANAGED_EXTENSION_STATE_VERSION = 1;\n\nexport interface ManagedExtensionSource {\n kind: \"local\" | \"github\" | \"npm\" | \"builtin\";\n input: string;\n location: string;\n name?: string;\n package?: string;\n version?: string;\n repository?: string;\n owner?: string;\n repo?: string;\n ref?: string;\n subpath?: string;\n commit?: string;\n}\n\nexport interface ManagedExtensionRecord {\n name: string;\n directory: string;\n scope: ExtensionScope;\n manifest_version: string;\n manifest_entry: string;\n capabilities: string[];\n installed_at: string;\n updated_at: string;\n source: ManagedExtensionSource;\n last_update_check_at?: string;\n last_update_remote_commit?: string;\n update_available?: boolean | null;\n update_error?: string;\n}\n\nexport interface ManagedExtensionState {\n version: number;\n updated_at: string;\n entries: ManagedExtensionRecord[];\n}\n\nexport interface ManagedExtensionStateReadResult {\n path: string;\n state: ManagedExtensionState;\n warnings: string[];\n}\n\nexport function resolveManagedExtensionStatePath(extensionsRoot: string): string {\n return path.join(extensionsRoot, MANAGED_EXTENSION_STATE_FILENAME);\n}\n\nexport function createEmptyManagedExtensionState(): ManagedExtensionState {\n return {\n version: MANAGED_EXTENSION_STATE_VERSION,\n updated_at: nowIso(),\n entries: [],\n };\n}\n\nexport function sortManagedEntries(entries: ManagedExtensionRecord[]): ManagedExtensionRecord[] {\n return [...entries].sort((left, right) => {\n const byScope = left.scope.localeCompare(right.scope);\n if (byScope !== 0) {\n return byScope;\n }\n const byName = left.name.localeCompare(right.name);\n if (byName !== 0) {\n return byName;\n }\n return left.directory.localeCompare(right.directory);\n });\n}\n\nexport function managedExtensionSourcesEquivalent(left: ManagedExtensionSource, right: ManagedExtensionSource): boolean {\n if (left.kind !== right.kind || left.input !== right.input || left.location !== right.location) {\n return false;\n }\n if (left.kind === \"npm\" && right.kind === \"npm\") {\n return left.package === right.package && left.version === right.version;\n }\n if (left.kind === \"github\" && right.kind === \"github\") {\n return left.repository === right.repository && left.ref === right.ref && left.subpath === right.subpath && left.commit === right.commit;\n }\n if (left.kind === \"builtin\" && right.kind === \"builtin\") {\n return left.name === right.name;\n }\n return true;\n}\n\nexport function normalizeManagedState(raw: unknown): ManagedExtensionState | null {\n if (typeof raw !== \"object\" || raw === null) {\n return null;\n }\n const candidate = raw as Record<string, unknown>;\n if (candidate.version !== MANAGED_EXTENSION_STATE_VERSION || !Array.isArray(candidate.entries)) {\n return null;\n }\n\n const entries: ManagedExtensionRecord[] = [];\n for (const rawEntry of candidate.entries) {\n if (typeof rawEntry !== \"object\" || rawEntry === null) {\n continue;\n }\n const entry = rawEntry as Record<string, unknown>;\n if (\n typeof entry.name !== \"string\" ||\n entry.name.trim().length === 0 ||\n typeof entry.directory !== \"string\" ||\n entry.directory.trim().length === 0 ||\n (entry.scope !== \"project\" && entry.scope !== \"global\") ||\n typeof entry.manifest_version !== \"string\" ||\n typeof entry.manifest_entry !== \"string\" ||\n !Array.isArray(entry.capabilities) ||\n entry.capabilities.some((value) => typeof value !== \"string\") ||\n typeof entry.installed_at !== \"string\" ||\n typeof entry.updated_at !== \"string\" ||\n typeof entry.source !== \"object\" ||\n entry.source === null\n ) {\n continue;\n }\n const source = entry.source as Record<string, unknown>;\n if (\n (source.kind !== \"local\" && source.kind !== \"github\" && source.kind !== \"npm\" && source.kind !== \"builtin\") ||\n typeof source.input !== \"string\" ||\n typeof source.location !== \"string\"\n ) {\n continue;\n }\n entries.push({\n name: entry.name.trim(),\n directory: entry.directory.trim(),\n scope: entry.scope,\n manifest_version: entry.manifest_version,\n manifest_entry: entry.manifest_entry,\n capabilities: normalizeStringList(entry.capabilities as string[]),\n installed_at: entry.installed_at,\n updated_at: entry.updated_at,\n source: {\n kind: source.kind,\n input: source.input,\n location: source.location,\n name: typeof source.name === \"string\" ? source.name : undefined,\n package: typeof source.package === \"string\" ? source.package : undefined,\n version: typeof source.version === \"string\" ? source.version : undefined,\n repository: typeof source.repository === \"string\" ? source.repository : undefined,\n owner: typeof source.owner === \"string\" ? source.owner : undefined,\n repo: typeof source.repo === \"string\" ? source.repo : undefined,\n ref: typeof source.ref === \"string\" ? source.ref : undefined,\n subpath: typeof source.subpath === \"string\" ? source.subpath : undefined,\n commit: typeof source.commit === \"string\" ? source.commit : undefined,\n },\n last_update_check_at: typeof entry.last_update_check_at === \"string\" ? entry.last_update_check_at : undefined,\n last_update_remote_commit:\n typeof entry.last_update_remote_commit === \"string\" ? entry.last_update_remote_commit : undefined,\n update_available:\n typeof entry.update_available === \"boolean\" || entry.update_available === null\n ? entry.update_available\n : undefined,\n update_error: typeof entry.update_error === \"string\" ? entry.update_error : undefined,\n });\n }\n return {\n version: MANAGED_EXTENSION_STATE_VERSION,\n updated_at: typeof candidate.updated_at === \"string\" ? candidate.updated_at : nowIso(),\n entries: sortManagedEntries(entries),\n };\n}\n\nexport async function readManagedExtensionState(extensionsRoot: string): Promise<ManagedExtensionStateReadResult> {\n const statePath = resolveManagedExtensionStatePath(extensionsRoot);\n const fallback = createEmptyManagedExtensionState();\n try {\n const raw = await fs.readFile(statePath, \"utf8\");\n const parsed = JSON.parse(raw) as unknown;\n const normalized = normalizeManagedState(parsed);\n if (!normalized) {\n throw new PmCliError(\n `Managed extension state file \"${statePath}\" has an invalid schema. Repair or remove it before mutating extension state.`,\n EXIT_CODE.GENERIC_FAILURE,\n );\n }\n return {\n path: statePath,\n state: normalized,\n warnings: [],\n };\n } catch (error: unknown) {\n if (error instanceof PmCliError) {\n throw error;\n }\n if (typeof error === \"object\" && error !== null && \"code\" in error && (error as { code?: string }).code === \"ENOENT\") {\n return {\n path: statePath,\n state: fallback,\n warnings: [],\n };\n }\n throw new PmCliError(\n `Managed extension state file \"${statePath}\" could not be read. Repair or remove it before mutating extension state.`,\n EXIT_CODE.GENERIC_FAILURE,\n );\n }\n}\n\nexport async function writeManagedExtensionState(extensionsRoot: string, state: ManagedExtensionState): Promise<void> {\n const statePath = resolveManagedExtensionStatePath(extensionsRoot);\n const normalized: ManagedExtensionState = {\n version: MANAGED_EXTENSION_STATE_VERSION,\n updated_at: nowIso(),\n entries: sortManagedEntries(state.entries),\n };\n await fs.mkdir(extensionsRoot, { recursive: true });\n await fs.writeFile(statePath, `${JSON.stringify(normalized, null, 2)}\\n`, \"utf8\");\n}\n\n\nexport function upsertManagedEntry(state: ManagedExtensionState, entry: ManagedExtensionRecord): ManagedExtensionState {\n const updatedEntries = state.entries.filter(\n (candidate) =>\n normalizeExtensionNameForMatch(candidate.name) !== normalizeExtensionNameForMatch(entry.name) &&\n normalizeExtensionNameForMatch(candidate.directory) !== normalizeExtensionNameForMatch(entry.directory),\n );\n updatedEntries.push(entry);\n return {\n ...state,\n updated_at: nowIso(),\n entries: sortManagedEntries(updatedEntries),\n };\n}\n"],"names":[],"mappings":"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","debugId":"f7dc1712-641e-52f3-bbd9-7f20eafddd5f"}
@@ -0,0 +1,14 @@
1
+ interface ExtensionScaffoldFileResult {
2
+ path: string;
3
+ status: "created" | "unchanged";
4
+ }
5
+ interface ExtensionScaffoldResult {
6
+ extension_name: string;
7
+ command_name: string;
8
+ target_path: string;
9
+ created_directory: boolean;
10
+ files: ExtensionScaffoldFileResult[];
11
+ }
12
+ export declare function buildStarterExtensionScaffoldFiles(extensionName: string, commandName: string, vocabulary: "extension" | "package"): Record<string, string>;
13
+ export declare function scaffoldExtensionProject(target: string, vocabulary?: "extension" | "package"): Promise<ExtensionScaffoldResult>;
14
+ export {};
@@ -0,0 +1,202 @@
1
+
2
+ !function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="6ba6d57e-c873-51ad-b55d-3f35c6f9ef55")}catch(e){}}();
3
+ import fs from "node:fs/promises";
4
+ import path from "node:path";
5
+ import { pathExists } from "../../../core/fs/fs-utils.js";
6
+ import { EXIT_CODE } from "../../../core/shared/constants.js";
7
+ import { PmCliError } from "../../../core/shared/errors.js";
8
+ import { normalizeManagedDirectoryName } from "./shared.js";
9
+ export function buildStarterExtensionScaffoldFiles(extensionName, commandName, vocabulary) {
10
+ const packageName = `pm-${extensionName}`;
11
+ const manifest = `${JSON.stringify({
12
+ name: extensionName,
13
+ version: "0.1.0",
14
+ entry: "./index.js",
15
+ capabilities: ["commands"],
16
+ }, null, 2)}\n`;
17
+ // pm-fl0c B-1 (2026-05-28): emit the documented `defineExtension({...})`
18
+ // shape so the generated module gets full TypeScript narrowing on the
19
+ // `activate(api)` signature (matches docs/EXTENSIONS.md, docs/SDK.md and
20
+ // docs/examples/starter-extension/index.js). Codex P2 follow-up
21
+ // (PR #78 review): a bare `import "@unbrained/pm-cli/sdk"` cannot resolve
22
+ // in extension-only mode — there is no package.json with a dep on
23
+ // @unbrained/pm-cli, so the loader's `import(<file-url>)` would raise
24
+ // ERR_MODULE_NOT_FOUND. We therefore emit the typed shape ONLY in package
25
+ // mode (peerDependencies make the import resolvable from the package's
26
+ // own node_modules) and keep the import-free shape in extension-only
27
+ // mode, decorated with a JSDoc @param hint so editors that follow JSDoc
28
+ // still narrow the `api` parameter.
29
+ const entrypoint = vocabulary === "package"
30
+ ? [
31
+ 'import { defineExtension } from "@unbrained/pm-cli/sdk";',
32
+ "",
33
+ "export default defineExtension({",
34
+ " activate(api) {",
35
+ " api.registerCommand({",
36
+ ` name: ${JSON.stringify(commandName)},`,
37
+ ' description: "Starter scaffold command. Replace with your own behavior.",',
38
+ " run: async (context) => ({",
39
+ " ok: true,",
40
+ ` source: ${JSON.stringify(extensionName)},`,
41
+ " command: context.command,",
42
+ ' message: "Starter extension scaffold is active.",',
43
+ " }),",
44
+ " });",
45
+ " },",
46
+ "});",
47
+ "",
48
+ ].join("\n")
49
+ : [
50
+ '/** @param {import("@unbrained/pm-cli/sdk").ExtensionApi} api */',
51
+ "export function activate(api) {",
52
+ " api.registerCommand({",
53
+ ` name: ${JSON.stringify(commandName)},`,
54
+ ' description: "Starter scaffold command. Replace with your own behavior.",',
55
+ " run: async (context) => ({",
56
+ " ok: true,",
57
+ ` source: ${JSON.stringify(extensionName)},`,
58
+ " command: context.command,",
59
+ ' message: "Starter extension scaffold is active.",',
60
+ " }),",
61
+ " });",
62
+ "}",
63
+ "",
64
+ "export default {",
65
+ " activate,",
66
+ "};",
67
+ "",
68
+ ].join("\n");
69
+ if (vocabulary === "package") {
70
+ const packageJson = `${JSON.stringify({
71
+ name: packageName,
72
+ version: "0.1.0",
73
+ private: true,
74
+ type: "module",
75
+ keywords: ["pm-package"],
76
+ peerDependencies: {
77
+ "@unbrained/pm-cli": "*",
78
+ },
79
+ pm: {
80
+ aliases: [extensionName],
81
+ extensions: [`extensions/${extensionName}`],
82
+ docs: ["README.md"],
83
+ examples: ["README.md"],
84
+ catalog: {
85
+ display_name: extensionName,
86
+ category: "workflow",
87
+ summary: "Starter pm package scaffold.",
88
+ tags: ["starter"],
89
+ },
90
+ },
91
+ }, null, 2)}\n`;
92
+ const packageReadme = [
93
+ `# ${packageName}`,
94
+ "",
95
+ "Generated by `pm package init`.",
96
+ "",
97
+ "## Included Files",
98
+ "- `package.json`: package metadata and `pm` resource manifest.",
99
+ `- \`extensions/${extensionName}/manifest.json\`: extension metadata and capabilities.`,
100
+ `- \`extensions/${extensionName}/index.js\`: starter command registration using the \`commands\` capability.`,
101
+ "",
102
+ "## Quick Start",
103
+ "```bash",
104
+ "pm install --project <package-path>",
105
+ `pm ${commandName}`,
106
+ "pm package doctor --project --detail summary",
107
+ "```",
108
+ "",
109
+ "## Notes",
110
+ "- Keep package metadata in `package.json` and runtime behavior under `extensions/`.",
111
+ "- Add capabilities to the extension manifest only when the entrypoint uses the matching SDK API.",
112
+ "- Use `@unbrained/pm-cli/sdk` as the public SDK import for richer package runtimes.",
113
+ "",
114
+ ].join("\n");
115
+ return {
116
+ "package.json": packageJson,
117
+ [`extensions/${extensionName}/manifest.json`]: manifest,
118
+ [`extensions/${extensionName}/index.js`]: entrypoint,
119
+ "README.md": packageReadme,
120
+ };
121
+ }
122
+ const readme = [
123
+ `# ${extensionName}`,
124
+ "",
125
+ "Generated by `pm extension init`.",
126
+ "",
127
+ "## Included Files",
128
+ "- `manifest.json`: extension metadata and capabilities.",
129
+ "- `index.js`: starter command registration using the `commands` capability.",
130
+ "",
131
+ "## Quick Start",
132
+ "```bash",
133
+ "pm extension --install --project <scaffold-path>",
134
+ `pm ${commandName}`,
135
+ "pm extension --doctor --project --detail summary",
136
+ "```",
137
+ "",
138
+ "## Notes",
139
+ "- This scaffold uses ESM exports so it works in package scopes with `type: module`.",
140
+ "- Update `manifest.json` capabilities and `index.js` command behavior as your extension evolves.",
141
+ "",
142
+ ].join("\n");
143
+ return {
144
+ "manifest.json": manifest,
145
+ "index.js": entrypoint,
146
+ "README.md": readme,
147
+ };
148
+ }
149
+ export async function scaffoldExtensionProject(target, vocabulary = "extension") {
150
+ const normalizedTarget = target.trim();
151
+ const targetPath = path.resolve(process.cwd(), normalizedTarget);
152
+ const extensionName = normalizeManagedDirectoryName(path.basename(targetPath));
153
+ const commandName = `${extensionName} ping`;
154
+ const scaffoldFiles = buildStarterExtensionScaffoldFiles(extensionName, commandName, vocabulary);
155
+ let createdDirectory = false;
156
+ if (await pathExists(targetPath)) {
157
+ const existingTargetStats = await fs.stat(targetPath);
158
+ if (!existingTargetStats.isDirectory()) {
159
+ throw new PmCliError(`Scaffold target "${targetPath}" exists and is not a directory.`, EXIT_CODE.CONFLICT);
160
+ }
161
+ }
162
+ else {
163
+ await fs.mkdir(targetPath, { recursive: true });
164
+ createdDirectory = true;
165
+ }
166
+ for (const [relativePath, content] of Object.entries(scaffoldFiles)) {
167
+ const absolutePath = path.join(targetPath, relativePath);
168
+ if (!(await pathExists(absolutePath))) {
169
+ continue;
170
+ }
171
+ const existingContent = await fs.readFile(absolutePath, "utf8");
172
+ if (existingContent !== content) {
173
+ throw new PmCliError(`Scaffold file "${relativePath}" already exists with different content in "${targetPath}". Choose a new target path or remove conflicting files.`, EXIT_CODE.CONFLICT);
174
+ }
175
+ }
176
+ const files = [];
177
+ for (const [relativePath, content] of Object.entries(scaffoldFiles)) {
178
+ const absolutePath = path.join(targetPath, relativePath);
179
+ if (await pathExists(absolutePath)) {
180
+ files.push({
181
+ path: relativePath,
182
+ status: "unchanged",
183
+ });
184
+ continue;
185
+ }
186
+ await fs.mkdir(path.dirname(absolutePath), { recursive: true });
187
+ await fs.writeFile(absolutePath, content, "utf8");
188
+ files.push({
189
+ path: relativePath,
190
+ status: "created",
191
+ });
192
+ }
193
+ return {
194
+ extension_name: extensionName,
195
+ command_name: commandName,
196
+ target_path: targetPath,
197
+ created_directory: createdDirectory,
198
+ files,
199
+ };
200
+ }
201
+ //# sourceMappingURL=scaffold.js.map
202
+ //# debugId=6ba6d57e-c873-51ad-b55d-3f35c6f9ef55
@@ -0,0 +1 @@
1
+ {"version":3,"file":"scaffold.js","sources":["cli/commands/extension/scaffold.ts"],"sourceRoot":"/","sourcesContent":["import fs from \"node:fs/promises\";\nimport path from \"node:path\";\nimport { pathExists } from \"../../../core/fs/fs-utils.js\";\nimport { EXIT_CODE } from \"../../../core/shared/constants.js\";\nimport { PmCliError } from \"../../../core/shared/errors.js\";\nimport { normalizeManagedDirectoryName } from \"./shared.js\";\n\ninterface ExtensionScaffoldFileResult {\n path: string;\n status: \"created\" | \"unchanged\";\n}\n\ninterface ExtensionScaffoldResult {\n extension_name: string;\n command_name: string;\n target_path: string;\n created_directory: boolean;\n files: ExtensionScaffoldFileResult[];\n}\n\n\nexport function buildStarterExtensionScaffoldFiles(\n extensionName: string,\n commandName: string,\n vocabulary: \"extension\" | \"package\",\n): Record<string, string> {\n const packageName = `pm-${extensionName}`;\n const manifest = `${JSON.stringify(\n {\n name: extensionName,\n version: \"0.1.0\",\n entry: \"./index.js\",\n capabilities: [\"commands\"],\n },\n null,\n 2,\n )}\\n`;\n // pm-fl0c B-1 (2026-05-28): emit the documented `defineExtension({...})`\n // shape so the generated module gets full TypeScript narrowing on the\n // `activate(api)` signature (matches docs/EXTENSIONS.md, docs/SDK.md and\n // docs/examples/starter-extension/index.js). Codex P2 follow-up\n // (PR #78 review): a bare `import \"@unbrained/pm-cli/sdk\"` cannot resolve\n // in extension-only mode — there is no package.json with a dep on\n // @unbrained/pm-cli, so the loader's `import(<file-url>)` would raise\n // ERR_MODULE_NOT_FOUND. We therefore emit the typed shape ONLY in package\n // mode (peerDependencies make the import resolvable from the package's\n // own node_modules) and keep the import-free shape in extension-only\n // mode, decorated with a JSDoc @param hint so editors that follow JSDoc\n // still narrow the `api` parameter.\n const entrypoint =\n vocabulary === \"package\"\n ? [\n 'import { defineExtension } from \"@unbrained/pm-cli/sdk\";',\n \"\",\n \"export default defineExtension({\",\n \" activate(api) {\",\n \" api.registerCommand({\",\n ` name: ${JSON.stringify(commandName)},`,\n ' description: \"Starter scaffold command. Replace with your own behavior.\",',\n \" run: async (context) => ({\",\n \" ok: true,\",\n ` source: ${JSON.stringify(extensionName)},`,\n \" command: context.command,\",\n ' message: \"Starter extension scaffold is active.\",',\n \" }),\",\n \" });\",\n \" },\",\n \"});\",\n \"\",\n ].join(\"\\n\")\n : [\n '/** @param {import(\"@unbrained/pm-cli/sdk\").ExtensionApi} api */',\n \"export function activate(api) {\",\n \" api.registerCommand({\",\n ` name: ${JSON.stringify(commandName)},`,\n ' description: \"Starter scaffold command. Replace with your own behavior.\",',\n \" run: async (context) => ({\",\n \" ok: true,\",\n ` source: ${JSON.stringify(extensionName)},`,\n \" command: context.command,\",\n ' message: \"Starter extension scaffold is active.\",',\n \" }),\",\n \" });\",\n \"}\",\n \"\",\n \"export default {\",\n \" activate,\",\n \"};\",\n \"\",\n ].join(\"\\n\");\n if (vocabulary === \"package\") {\n const packageJson = `${JSON.stringify(\n {\n name: packageName,\n version: \"0.1.0\",\n private: true,\n type: \"module\",\n keywords: [\"pm-package\"],\n peerDependencies: {\n \"@unbrained/pm-cli\": \"*\",\n },\n pm: {\n aliases: [extensionName],\n extensions: [`extensions/${extensionName}`],\n docs: [\"README.md\"],\n examples: [\"README.md\"],\n catalog: {\n display_name: extensionName,\n category: \"workflow\",\n summary: \"Starter pm package scaffold.\",\n tags: [\"starter\"],\n },\n },\n },\n null,\n 2,\n )}\\n`;\n const packageReadme = [\n `# ${packageName}`,\n \"\",\n \"Generated by `pm package init`.\",\n \"\",\n \"## Included Files\",\n \"- `package.json`: package metadata and `pm` resource manifest.\",\n `- \\`extensions/${extensionName}/manifest.json\\`: extension metadata and capabilities.`,\n `- \\`extensions/${extensionName}/index.js\\`: starter command registration using the \\`commands\\` capability.`,\n \"\",\n \"## Quick Start\",\n \"```bash\",\n \"pm install --project <package-path>\",\n `pm ${commandName}`,\n \"pm package doctor --project --detail summary\",\n \"```\",\n \"\",\n \"## Notes\",\n \"- Keep package metadata in `package.json` and runtime behavior under `extensions/`.\",\n \"- Add capabilities to the extension manifest only when the entrypoint uses the matching SDK API.\",\n \"- Use `@unbrained/pm-cli/sdk` as the public SDK import for richer package runtimes.\",\n \"\",\n ].join(\"\\n\");\n return {\n \"package.json\": packageJson,\n [`extensions/${extensionName}/manifest.json`]: manifest,\n [`extensions/${extensionName}/index.js`]: entrypoint,\n \"README.md\": packageReadme,\n };\n }\n const readme = [\n `# ${extensionName}`,\n \"\",\n \"Generated by `pm extension init`.\",\n \"\",\n \"## Included Files\",\n \"- `manifest.json`: extension metadata and capabilities.\",\n \"- `index.js`: starter command registration using the `commands` capability.\",\n \"\",\n \"## Quick Start\",\n \"```bash\",\n \"pm extension --install --project <scaffold-path>\",\n `pm ${commandName}`,\n \"pm extension --doctor --project --detail summary\",\n \"```\",\n \"\",\n \"## Notes\",\n \"- This scaffold uses ESM exports so it works in package scopes with `type: module`.\",\n \"- Update `manifest.json` capabilities and `index.js` command behavior as your extension evolves.\",\n \"\",\n ].join(\"\\n\");\n return {\n \"manifest.json\": manifest,\n \"index.js\": entrypoint,\n \"README.md\": readme,\n };\n}\n\nexport async function scaffoldExtensionProject(\n target: string,\n vocabulary: \"extension\" | \"package\" = \"extension\",\n): Promise<ExtensionScaffoldResult> {\n const normalizedTarget = target.trim();\n const targetPath = path.resolve(process.cwd(), normalizedTarget);\n const extensionName = normalizeManagedDirectoryName(path.basename(targetPath));\n const commandName = `${extensionName} ping`;\n const scaffoldFiles = buildStarterExtensionScaffoldFiles(extensionName, commandName, vocabulary);\n\n let createdDirectory = false;\n if (await pathExists(targetPath)) {\n const existingTargetStats = await fs.stat(targetPath);\n if (!existingTargetStats.isDirectory()) {\n throw new PmCliError(\n `Scaffold target \"${targetPath}\" exists and is not a directory.`,\n EXIT_CODE.CONFLICT,\n );\n }\n } else {\n await fs.mkdir(targetPath, { recursive: true });\n createdDirectory = true;\n }\n\n for (const [relativePath, content] of Object.entries(scaffoldFiles)) {\n const absolutePath = path.join(targetPath, relativePath);\n if (!(await pathExists(absolutePath))) {\n continue;\n }\n const existingContent = await fs.readFile(absolutePath, \"utf8\");\n if (existingContent !== content) {\n throw new PmCliError(\n `Scaffold file \"${relativePath}\" already exists with different content in \"${targetPath}\". Choose a new target path or remove conflicting files.`,\n EXIT_CODE.CONFLICT,\n );\n }\n }\n\n const files: ExtensionScaffoldFileResult[] = [];\n for (const [relativePath, content] of Object.entries(scaffoldFiles)) {\n const absolutePath = path.join(targetPath, relativePath);\n if (await pathExists(absolutePath)) {\n files.push({\n path: relativePath,\n status: \"unchanged\",\n });\n continue;\n }\n await fs.mkdir(path.dirname(absolutePath), { recursive: true });\n await fs.writeFile(absolutePath, content, \"utf8\");\n files.push({\n path: relativePath,\n status: \"created\",\n });\n }\n\n return {\n extension_name: extensionName,\n command_name: commandName,\n target_path: targetPath,\n created_directory: createdDirectory,\n files,\n };\n}\n"],"names":[],"mappings":"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","debugId":"6ba6d57e-c873-51ad-b55d-3f35c6f9ef55"}
@@ -0,0 +1,14 @@
1
+ import type { ExtensionManifest } from "../../../core/extensions/loader.js";
2
+ export declare const DEFAULT_EXTENSION_PRIORITY = 100;
3
+ export interface ValidatedExtensionDirectory {
4
+ directory: string;
5
+ manifest_path: string;
6
+ entry_path: string;
7
+ manifest: ExtensionManifest;
8
+ }
9
+ export declare function normalizeStringList(values: readonly string[]): string[];
10
+ export declare function normalizeExtensionNameForMatch(value: string): string;
11
+ export declare function normalizeManagedDirectoryName(name: string): string;
12
+ export declare function parseExtensionManifest(raw: unknown): ExtensionManifest | null;
13
+ export declare function isCanonicalPathWithinDirectory(directory: string, targetPath: string): Promise<boolean>;
14
+ export declare function validateExtensionDirectory(directory: string): Promise<ValidatedExtensionDirectory>;
@@ -0,0 +1,106 @@
1
+
2
+ !function(){try{var e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof globalThis?globalThis:"undefined"!=typeof self?self:{},n=(new e.Error).stack;n&&(e._sentryDebugIds=e._sentryDebugIds||{},e._sentryDebugIds[n]="cb831def-7a8a-55f8-85e2-7749a5591bfe")}catch(e){}}();
3
+ import fs from "node:fs/promises";
4
+ import path from "node:path";
5
+ import { pathExists } from "../../../core/fs/fs-utils.js";
6
+ import { isPathWithinDirectory } from "../../../core/fs/path-utils.js";
7
+ import { EXIT_CODE } from "../../../core/shared/constants.js";
8
+ import { PmCliError } from "../../../core/shared/errors.js";
9
+ export const DEFAULT_EXTENSION_PRIORITY = 100;
10
+ export function normalizeStringList(values) {
11
+ return [...new Set(values.map((value) => value.trim()).filter((value) => value.length > 0))].sort((left, right) => left.localeCompare(right));
12
+ }
13
+ export function normalizeExtensionNameForMatch(value) {
14
+ return value.trim().toLowerCase();
15
+ }
16
+ export function normalizeManagedDirectoryName(name) {
17
+ const normalized = name
18
+ .trim()
19
+ .toLowerCase()
20
+ .replace(/[^a-z0-9._-]+/g, "-")
21
+ .replace(/^-+|-+$/g, "");
22
+ if (normalized.length === 0) {
23
+ throw new PmCliError("Extension manifest name must resolve to a non-empty directory name.", EXIT_CODE.USAGE);
24
+ }
25
+ if (normalized === "." || normalized === "..") {
26
+ // Manifest-controlled input must resolve to a dedicated child directory, never
27
+ // the extensions root itself or its parent (path-traversal guard).
28
+ throw new PmCliError("Extension manifest name must not resolve to \".\" or \"..\".", EXIT_CODE.USAGE);
29
+ }
30
+ return normalized;
31
+ }
32
+ export function parseExtensionManifest(raw) {
33
+ if (typeof raw !== "object" || raw === null) {
34
+ return null;
35
+ }
36
+ const candidate = raw;
37
+ if (typeof candidate.name !== "string" || candidate.name.trim().length === 0) {
38
+ return null;
39
+ }
40
+ if (typeof candidate.version !== "string" || candidate.version.trim().length === 0) {
41
+ return null;
42
+ }
43
+ if (typeof candidate.entry !== "string" || candidate.entry.trim().length === 0) {
44
+ return null;
45
+ }
46
+ let priority = DEFAULT_EXTENSION_PRIORITY;
47
+ if (candidate.priority !== undefined && candidate.priority !== null) {
48
+ if (typeof candidate.priority !== "number" || !Number.isInteger(candidate.priority)) {
49
+ return null;
50
+ }
51
+ priority = candidate.priority;
52
+ }
53
+ let capabilities = [];
54
+ if (candidate.capabilities !== undefined && candidate.capabilities !== null) {
55
+ if (!Array.isArray(candidate.capabilities) || candidate.capabilities.some((value) => typeof value !== "string")) {
56
+ return null;
57
+ }
58
+ capabilities = normalizeStringList(candidate.capabilities.map((value) => String(value).toLowerCase()));
59
+ }
60
+ return {
61
+ name: candidate.name.trim(),
62
+ version: candidate.version.trim(),
63
+ entry: candidate.entry.trim(),
64
+ priority,
65
+ capabilities,
66
+ };
67
+ }
68
+ export async function isCanonicalPathWithinDirectory(directory, targetPath) {
69
+ const [resolvedDirectory, resolvedTargetPath] = await Promise.all([fs.realpath(directory), fs.realpath(targetPath)]);
70
+ return isPathWithinDirectory(resolvedDirectory, resolvedTargetPath);
71
+ }
72
+ export async function validateExtensionDirectory(directory) {
73
+ const manifestPath = path.join(directory, "manifest.json");
74
+ if (!(await pathExists(manifestPath))) {
75
+ throw new PmCliError(`Extension manifest is missing at "${manifestPath}".`, EXIT_CODE.USAGE);
76
+ }
77
+ let parsedManifest;
78
+ try {
79
+ parsedManifest = JSON.parse(await fs.readFile(manifestPath, "utf8"));
80
+ }
81
+ catch (error) {
82
+ throw new PmCliError(`Failed to parse extension manifest at "${manifestPath}": ${error instanceof Error ? error.message : String(error)}`, EXIT_CODE.USAGE);
83
+ }
84
+ const manifest = parseExtensionManifest(parsedManifest);
85
+ if (!manifest) {
86
+ throw new PmCliError(`Extension manifest at "${manifestPath}" is invalid.`, EXIT_CODE.USAGE);
87
+ }
88
+ const entryPath = path.resolve(directory, manifest.entry);
89
+ if (!isPathWithinDirectory(directory, entryPath)) {
90
+ throw new PmCliError(`Extension entry "${manifest.entry}" resolves outside extension directory "${directory}".`, EXIT_CODE.USAGE);
91
+ }
92
+ if (!(await pathExists(entryPath))) {
93
+ throw new PmCliError(`Extension entry file is missing at "${entryPath}".`, EXIT_CODE.USAGE);
94
+ }
95
+ if (!(await isCanonicalPathWithinDirectory(directory, entryPath))) {
96
+ throw new PmCliError(`Extension entry "${manifest.entry}" resolves outside extension directory after symlink resolution.`, EXIT_CODE.USAGE);
97
+ }
98
+ return {
99
+ directory,
100
+ manifest_path: manifestPath,
101
+ entry_path: entryPath,
102
+ manifest,
103
+ };
104
+ }
105
+ //# sourceMappingURL=shared.js.map
106
+ //# debugId=cb831def-7a8a-55f8-85e2-7749a5591bfe
@@ -0,0 +1 @@
1
+ {"version":3,"file":"shared.js","sources":["cli/commands/extension/shared.ts"],"sourceRoot":"/","sourcesContent":["import fs from \"node:fs/promises\";\nimport path from \"node:path\";\nimport { pathExists } from \"../../../core/fs/fs-utils.js\";\nimport { isPathWithinDirectory } from \"../../../core/fs/path-utils.js\";\nimport { EXIT_CODE } from \"../../../core/shared/constants.js\";\nimport { PmCliError } from \"../../../core/shared/errors.js\";\nimport type { ExtensionManifest } from \"../../../core/extensions/loader.js\";\n\nexport const DEFAULT_EXTENSION_PRIORITY = 100;\n\nexport interface ValidatedExtensionDirectory {\n directory: string;\n manifest_path: string;\n entry_path: string;\n manifest: ExtensionManifest;\n}\n\nexport function normalizeStringList(values: readonly string[]): string[] {\n return [...new Set(values.map((value) => value.trim()).filter((value) => value.length > 0))].sort((left, right) =>\n left.localeCompare(right),\n );\n}\n\nexport function normalizeExtensionNameForMatch(value: string): string {\n return value.trim().toLowerCase();\n}\n\nexport function normalizeManagedDirectoryName(name: string): string {\n const normalized = name\n .trim()\n .toLowerCase()\n .replace(/[^a-z0-9._-]+/g, \"-\")\n .replace(/^-+|-+$/g, \"\");\n if (normalized.length === 0) {\n throw new PmCliError(\"Extension manifest name must resolve to a non-empty directory name.\", EXIT_CODE.USAGE);\n }\n if (normalized === \".\" || normalized === \"..\") {\n // Manifest-controlled input must resolve to a dedicated child directory, never\n // the extensions root itself or its parent (path-traversal guard).\n throw new PmCliError(\"Extension manifest name must not resolve to \\\".\\\" or \\\"..\\\".\", EXIT_CODE.USAGE);\n }\n return normalized;\n}\n\nexport function parseExtensionManifest(raw: unknown): ExtensionManifest | null {\n if (typeof raw !== \"object\" || raw === null) {\n return null;\n }\n const candidate = raw as Record<string, unknown>;\n if (typeof candidate.name !== \"string\" || candidate.name.trim().length === 0) {\n return null;\n }\n if (typeof candidate.version !== \"string\" || candidate.version.trim().length === 0) {\n return null;\n }\n if (typeof candidate.entry !== \"string\" || candidate.entry.trim().length === 0) {\n return null;\n }\n\n let priority = DEFAULT_EXTENSION_PRIORITY;\n if (candidate.priority !== undefined && candidate.priority !== null) {\n if (typeof candidate.priority !== \"number\" || !Number.isInteger(candidate.priority)) {\n return null;\n }\n priority = candidate.priority;\n }\n\n let capabilities: string[] = [];\n if (candidate.capabilities !== undefined && candidate.capabilities !== null) {\n if (!Array.isArray(candidate.capabilities) || candidate.capabilities.some((value) => typeof value !== \"string\")) {\n return null;\n }\n capabilities = normalizeStringList(candidate.capabilities.map((value) => String(value).toLowerCase()));\n }\n\n return {\n name: candidate.name.trim(),\n version: candidate.version.trim(),\n entry: candidate.entry.trim(),\n priority,\n capabilities,\n };\n}\n\nexport async function isCanonicalPathWithinDirectory(directory: string, targetPath: string): Promise<boolean> {\n const [resolvedDirectory, resolvedTargetPath] = await Promise.all([fs.realpath(directory), fs.realpath(targetPath)]);\n return isPathWithinDirectory(resolvedDirectory, resolvedTargetPath);\n}\n\nexport async function validateExtensionDirectory(directory: string): Promise<ValidatedExtensionDirectory> {\n const manifestPath = path.join(directory, \"manifest.json\");\n if (!(await pathExists(manifestPath))) {\n throw new PmCliError(`Extension manifest is missing at \"${manifestPath}\".`, EXIT_CODE.USAGE);\n }\n\n let parsedManifest: unknown;\n try {\n parsedManifest = JSON.parse(await fs.readFile(manifestPath, \"utf8\")) as unknown;\n } catch (error: unknown) {\n throw new PmCliError(\n `Failed to parse extension manifest at \"${manifestPath}\": ${error instanceof Error ? error.message : String(error)}`,\n EXIT_CODE.USAGE,\n );\n }\n\n const manifest = parseExtensionManifest(parsedManifest);\n if (!manifest) {\n throw new PmCliError(`Extension manifest at \"${manifestPath}\" is invalid.`, EXIT_CODE.USAGE);\n }\n\n const entryPath = path.resolve(directory, manifest.entry);\n if (!isPathWithinDirectory(directory, entryPath)) {\n throw new PmCliError(\n `Extension entry \"${manifest.entry}\" resolves outside extension directory \"${directory}\".`,\n EXIT_CODE.USAGE,\n );\n }\n if (!(await pathExists(entryPath))) {\n throw new PmCliError(`Extension entry file is missing at \"${entryPath}\".`, EXIT_CODE.USAGE);\n }\n if (!(await isCanonicalPathWithinDirectory(directory, entryPath))) {\n throw new PmCliError(\n `Extension entry \"${manifest.entry}\" resolves outside extension directory after symlink resolution.`,\n EXIT_CODE.USAGE,\n );\n }\n\n return {\n directory,\n manifest_path: manifestPath,\n entry_path: entryPath,\n manifest,\n };\n}\n"],"names":[],"mappings":"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","debugId":"cb831def-7a8a-55f8-85e2-7749a5591bfe"}