@lumenflow/cli 4.23.0 → 5.0.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 (296) hide show
  1. package/README.md +54 -52
  2. package/dist/agent-issues-query.js +10 -2
  3. package/dist/agent-issues-query.js.map +1 -1
  4. package/dist/agent-runtime-enrollment-events.js +44 -0
  5. package/dist/agent-runtime-enrollment-events.js.map +1 -0
  6. package/dist/agent-session-end.js +47 -0
  7. package/dist/agent-session-end.js.map +1 -1
  8. package/dist/agent-session-heartbeat.js +250 -0
  9. package/dist/agent-session-heartbeat.js.map +1 -0
  10. package/dist/agent-session.js +299 -5
  11. package/dist/agent-session.js.map +1 -1
  12. package/dist/capacity-snapshot-emitter.js +73 -0
  13. package/dist/capacity-snapshot-emitter.js.map +1 -0
  14. package/dist/claim-queue.js +276 -0
  15. package/dist/claim-queue.js.map +1 -0
  16. package/dist/config-set.js +22 -3
  17. package/dist/config-set.js.map +1 -1
  18. package/dist/control-plane-sidecar-runner.js +145 -0
  19. package/dist/control-plane-sidecar-runner.js.map +1 -0
  20. package/dist/delegation-list.js +160 -1
  21. package/dist/delegation-list.js.map +1 -1
  22. package/dist/delegation-role-resolver.js +69 -0
  23. package/dist/delegation-role-resolver.js.map +1 -0
  24. package/dist/docs-generate-pack-reference.js +500 -0
  25. package/dist/docs-generate-pack-reference.js.map +1 -0
  26. package/dist/docs-sync.js +116 -1
  27. package/dist/docs-sync.js.map +1 -1
  28. package/dist/file-edit.js +28 -8
  29. package/dist/file-edit.js.map +1 -1
  30. package/dist/file-write.js +29 -5
  31. package/dist/file-write.js.map +1 -1
  32. package/dist/gate-co-change.js +25 -7
  33. package/dist/gate-co-change.js.map +1 -1
  34. package/dist/gate-conditional.js +19 -7
  35. package/dist/gate-conditional.js.map +1 -1
  36. package/dist/gates-runners.js +42 -33
  37. package/dist/gates-runners.js.map +1 -1
  38. package/dist/gates-utils.js +34 -20
  39. package/dist/gates-utils.js.map +1 -1
  40. package/dist/gates.js +79 -7
  41. package/dist/gates.js.map +1 -1
  42. package/dist/hooks/config-resolver.js +10 -1
  43. package/dist/hooks/config-resolver.js.map +1 -1
  44. package/dist/init-package-config.js +1 -1
  45. package/dist/init-package-config.js.map +1 -1
  46. package/dist/init-scaffolding.js +5 -1
  47. package/dist/init-scaffolding.js.map +1 -1
  48. package/dist/init-templates.js +10 -0
  49. package/dist/init-templates.js.map +1 -1
  50. package/dist/init.js +1 -1
  51. package/dist/init.js.map +1 -1
  52. package/dist/initiative-create.js +17 -0
  53. package/dist/initiative-create.js.map +1 -1
  54. package/dist/initiative-remove-wu.js +17 -3
  55. package/dist/initiative-remove-wu.js.map +1 -1
  56. package/dist/kernel-event-sync/emitters.js +104 -0
  57. package/dist/kernel-event-sync/emitters.js.map +1 -0
  58. package/dist/kernel-event-sync/index.js +13 -0
  59. package/dist/kernel-event-sync/index.js.map +1 -0
  60. package/dist/kernel-event-sync/lifecycle-emitters.js +160 -0
  61. package/dist/kernel-event-sync/lifecycle-emitters.js.map +1 -0
  62. package/dist/kernel-event-sync/narrow-emissions.js +89 -0
  63. package/dist/kernel-event-sync/narrow-emissions.js.map +1 -0
  64. package/dist/kernel-event-sync/software-delivery-emitters.js +297 -0
  65. package/dist/kernel-event-sync/software-delivery-emitters.js.map +1 -0
  66. package/dist/lane-lock.js +14 -1
  67. package/dist/lane-lock.js.map +1 -1
  68. package/dist/lane-suggest.js +21 -0
  69. package/dist/lane-suggest.js.map +1 -1
  70. package/dist/lumenflow-upgrade.js +7 -5
  71. package/dist/lumenflow-upgrade.js.map +1 -1
  72. package/dist/mem-context.js +145 -0
  73. package/dist/mem-context.js.map +1 -1
  74. package/dist/mem-create.js +39 -6
  75. package/dist/mem-create.js.map +1 -1
  76. package/dist/mem-inbox.js +16 -0
  77. package/dist/mem-inbox.js.map +1 -1
  78. package/dist/mem-roster.js +95 -0
  79. package/dist/mem-roster.js.map +1 -0
  80. package/dist/mem-signal.js +97 -2
  81. package/dist/mem-signal.js.map +1 -1
  82. package/dist/metrics-cli.js +3 -2
  83. package/dist/metrics-cli.js.map +1 -1
  84. package/dist/metrics-snapshot.js +271 -13
  85. package/dist/metrics-snapshot.js.map +1 -1
  86. package/dist/orchestrate-init-status.js +117 -2
  87. package/dist/orchestrate-init-status.js.map +1 -1
  88. package/dist/orchestrate-initiative.js +83 -10
  89. package/dist/orchestrate-initiative.js.map +1 -1
  90. package/dist/orchestrate-monitor-quality.js +289 -0
  91. package/dist/orchestrate-monitor-quality.js.map +1 -0
  92. package/dist/orchestrate-monitor.js +85 -0
  93. package/dist/orchestrate-monitor.js.map +1 -1
  94. package/dist/pack-validate.js +127 -2
  95. package/dist/pack-validate.js.map +1 -1
  96. package/dist/plan-create.js +18 -0
  97. package/dist/plan-create.js.map +1 -1
  98. package/dist/plan-link.js +13 -0
  99. package/dist/plan-link.js.map +1 -1
  100. package/dist/plan-promote.js +14 -0
  101. package/dist/plan-promote.js.map +1 -1
  102. package/dist/pre-commit-check.js +4 -3
  103. package/dist/pre-commit-check.js.map +1 -1
  104. package/dist/public-manifest.js +17 -3
  105. package/dist/public-manifest.js.map +1 -1
  106. package/dist/release.js +10 -10
  107. package/dist/release.js.map +1 -1
  108. package/dist/session-cross-link.js +139 -0
  109. package/dist/session-cross-link.js.map +1 -0
  110. package/dist/sidecar-manager.js +208 -0
  111. package/dist/sidecar-manager.js.map +1 -0
  112. package/dist/state-path-resolvers.js +18 -0
  113. package/dist/state-path-resolvers.js.map +1 -1
  114. package/dist/stream-heartbeat.js +151 -0
  115. package/dist/stream-heartbeat.js.map +1 -0
  116. package/dist/sync-templates.js +56 -2
  117. package/dist/sync-templates.js.map +1 -1
  118. package/dist/wu-block.js +47 -5
  119. package/dist/wu-block.js.map +1 -1
  120. package/dist/wu-claim-branch.js +8 -4
  121. package/dist/wu-claim-branch.js.map +1 -1
  122. package/dist/wu-claim-state.js +5 -3
  123. package/dist/wu-claim-state.js.map +1 -1
  124. package/dist/wu-claim-worktree.js +5 -3
  125. package/dist/wu-claim-worktree.js.map +1 -1
  126. package/dist/wu-claim.js +261 -9
  127. package/dist/wu-claim.js.map +1 -1
  128. package/dist/wu-done-auto-cleanup.js +3 -2
  129. package/dist/wu-done-auto-cleanup.js.map +1 -1
  130. package/dist/wu-done-git-ops.js +12 -8
  131. package/dist/wu-done-git-ops.js.map +1 -1
  132. package/dist/wu-done-preflight.js +3 -3
  133. package/dist/wu-done-preflight.js.map +1 -1
  134. package/dist/wu-done.js +46 -10
  135. package/dist/wu-done.js.map +1 -1
  136. package/dist/wu-lifecycle-sync/gate-scope-resolver.js +16 -0
  137. package/dist/wu-lifecycle-sync/gate-scope-resolver.js.map +1 -0
  138. package/dist/wu-lifecycle-sync/kernel-event-sync-shim.js +10 -0
  139. package/dist/wu-lifecycle-sync/kernel-event-sync-shim.js.map +1 -0
  140. package/dist/wu-prep.js +363 -22
  141. package/dist/wu-prep.js.map +1 -1
  142. package/dist/wu-prune.js +68 -27
  143. package/dist/wu-prune.js.map +1 -1
  144. package/dist/wu-release.js +34 -3
  145. package/dist/wu-release.js.map +1 -1
  146. package/dist/wu-review.js +167 -0
  147. package/dist/wu-review.js.map +1 -0
  148. package/dist/wu-spawn-prompt-builders.js +296 -40
  149. package/dist/wu-spawn-prompt-builders.js.map +1 -1
  150. package/dist/wu-spawn-strategy-resolver.js +126 -14
  151. package/dist/wu-spawn-strategy-resolver.js.map +1 -1
  152. package/dist/wu-unblock.js +52 -22
  153. package/dist/wu-unblock.js.map +1 -1
  154. package/package.json +13 -8
  155. package/packs/agent-runtime/.turbo/turbo-build.log +1 -1
  156. package/packs/agent-runtime/.turbo/turbo-test.log +25 -0
  157. package/packs/agent-runtime/.turbo/turbo-typecheck.log +4 -0
  158. package/packs/agent-runtime/agent-heartbeat.ts +163 -0
  159. package/packs/agent-runtime/auto-session-integration.ts +874 -0
  160. package/packs/agent-runtime/delegation-registry-schema.ts +220 -0
  161. package/packs/agent-runtime/delegation-registry-store.ts +269 -0
  162. package/packs/agent-runtime/delegation-tree.ts +328 -0
  163. package/packs/agent-runtime/index.ts +9 -0
  164. package/packs/agent-runtime/manifest.ts +103 -19
  165. package/packs/agent-runtime/manifest.yaml +132 -0
  166. package/packs/agent-runtime/memory-coordination-contract.ts +86 -0
  167. package/packs/agent-runtime/memory.d.ts +19 -0
  168. package/packs/agent-runtime/orchestration.ts +238 -23
  169. package/packs/agent-runtime/package.json +11 -2
  170. package/packs/agent-runtime/remote-controls/index.ts +7 -0
  171. package/packs/agent-runtime/remote-controls/operations.ts +399 -0
  172. package/packs/agent-runtime/remote-controls/port.ts +48 -0
  173. package/packs/agent-runtime/remote-controls/state-store.ts +258 -0
  174. package/packs/agent-runtime/remote-controls/types.ts +105 -0
  175. package/packs/agent-runtime/session-schema.ts +423 -0
  176. package/packs/agent-runtime/tool-impl/index.ts +1 -0
  177. package/packs/agent-runtime/tool-impl/remote-controls.mock.ts +252 -0
  178. package/packs/agent-runtime/tool-impl/remote-controls.ts +273 -0
  179. package/packs/agent-runtime/tsconfig.json +1 -1
  180. package/packs/agent-runtime/turn-lifecycle-events.ts +501 -0
  181. package/packs/sidekick/.lumenflow/state/conductor/outbox/sidekick-events.jsonl +213 -0
  182. package/packs/sidekick/.turbo/turbo-build.log +1 -1
  183. package/packs/sidekick/.turbo/turbo-test.log +25 -0
  184. package/packs/sidekick/.turbo/turbo-typecheck.log +4 -0
  185. package/packs/sidekick/channel-ingress.ts +137 -0
  186. package/packs/sidekick/manifest.ts +74 -0
  187. package/packs/sidekick/manifest.yaml +88 -0
  188. package/packs/sidekick/package.json +3 -1
  189. package/packs/sidekick/sidekick-events.ts +517 -0
  190. package/packs/sidekick/src/adapters/cloud-queue.ts +101 -0
  191. package/packs/sidekick/src/adapters/control-plane-bridge.adapter.ts +378 -0
  192. package/packs/sidekick/src/adapters/filesystem-bridge.adapter.ts +224 -0
  193. package/packs/sidekick/src/domain/channel.types.ts +84 -0
  194. package/packs/sidekick/src/ports/channel-bridge.port.ts +75 -0
  195. package/packs/sidekick/src/routines/commit.ts +74 -0
  196. package/packs/sidekick/tool-impl/channel-tools.ts +47 -0
  197. package/packs/sidekick/tool-impl/memory-tools.ts +17 -0
  198. package/packs/sidekick/tool-impl/routine-commit.ts +102 -0
  199. package/packs/sidekick/tool-impl/routine-tools.ts +67 -7
  200. package/packs/sidekick/tool-impl/runtime-context.ts +4 -0
  201. package/packs/sidekick/tool-impl/storage.ts +3 -0
  202. package/packs/sidekick/tool-impl/system-tools.ts +7 -0
  203. package/packs/sidekick/tool-impl/task-tools.ts +46 -0
  204. package/packs/sidekick/tsconfig.json +1 -1
  205. package/packs/software-delivery/.turbo/turbo-build.log +1 -1
  206. package/packs/software-delivery/.turbo/turbo-test.log +63 -0
  207. package/packs/software-delivery/.turbo/turbo-typecheck.log +4 -0
  208. package/packs/software-delivery/manifest-schema.ts +30 -0
  209. package/packs/software-delivery/manifest.ts +99 -1
  210. package/packs/software-delivery/manifest.yaml +46 -0
  211. package/packs/software-delivery/package.json +88 -3
  212. package/packs/software-delivery/src/commands/index.ts +5 -0
  213. package/packs/software-delivery/src/config/delivery-review-contract.ts +20 -0
  214. package/packs/software-delivery/src/config/env-accessors.ts +19 -0
  215. package/packs/software-delivery/src/config/index.ts +8 -0
  216. package/packs/software-delivery/src/config/normalize-config-keys.ts +19 -0
  217. package/packs/software-delivery/src/config/schemas/lumenflow-config-schema-types.ts +436 -0
  218. package/packs/software-delivery/src/config/workspace-reader.ts +310 -0
  219. package/packs/software-delivery/src/constants/backlog-patterns.ts +31 -0
  220. package/packs/software-delivery/src/constants/client-ids.ts +19 -0
  221. package/packs/software-delivery/src/constants/config-contract.ts +7 -0
  222. package/packs/software-delivery/src/constants/docs-layout-presets.ts +50 -0
  223. package/packs/software-delivery/src/constants/duration-constants.ts +20 -0
  224. package/packs/software-delivery/src/constants/gate-constants.ts +32 -0
  225. package/packs/software-delivery/src/constants/index.ts +29 -0
  226. package/packs/software-delivery/src/constants/lock-constants.ts +35 -0
  227. package/packs/software-delivery/src/constants/object-guards.ts +12 -0
  228. package/packs/software-delivery/src/constants/section-headings.ts +107 -0
  229. package/packs/software-delivery/src/constants/wu-cli-constants.ts +485 -0
  230. package/packs/software-delivery/src/constants/wu-domain-constants.ts +466 -0
  231. package/packs/software-delivery/src/constants/wu-git-constants.ts +7 -0
  232. package/packs/software-delivery/src/constants/wu-id-format.ts +327 -0
  233. package/packs/software-delivery/src/constants/wu-paths-constants.ts +358 -0
  234. package/packs/software-delivery/src/constants/wu-statuses.ts +287 -0
  235. package/packs/software-delivery/src/constants/wu-type-helpers.ts +67 -0
  236. package/packs/software-delivery/src/constants/wu-ui-constants.ts +267 -0
  237. package/packs/software-delivery/src/constants/wu-validation-constants.ts +73 -0
  238. package/packs/software-delivery/src/domain/index.ts +5 -0
  239. package/packs/software-delivery/src/domain/orchestration.constants.ts +168 -0
  240. package/packs/software-delivery/src/domain/orchestration.schemas.ts +239 -0
  241. package/packs/software-delivery/src/domain/orchestration.types.ts +178 -0
  242. package/packs/software-delivery/src/methodology/incremental-test.ts +90 -0
  243. package/packs/software-delivery/src/methodology/index.ts +6 -0
  244. package/packs/software-delivery/src/methodology/manual-test-validator.ts +292 -0
  245. package/packs/software-delivery/src/policy/coverage-gate.ts +270 -0
  246. package/packs/software-delivery/src/policy/gates-agent-mode.ts +223 -0
  247. package/packs/software-delivery/src/policy/gates-config-internal.ts +121 -0
  248. package/packs/software-delivery/src/policy/gates-config.ts +293 -0
  249. package/packs/software-delivery/src/policy/gates-coverage.ts +247 -0
  250. package/packs/software-delivery/src/policy/gates-presets.ts +134 -0
  251. package/packs/software-delivery/src/policy/gates-schemas.ts +173 -0
  252. package/packs/software-delivery/src/policy/index.ts +22 -0
  253. package/packs/software-delivery/src/policy/package-manager-resolver.ts +319 -0
  254. package/packs/software-delivery/src/policy/resolve-policy.ts +518 -0
  255. package/packs/software-delivery/src/ports/config.ports.ts +90 -0
  256. package/packs/software-delivery/src/ports/dashboard-renderer.port.ts +125 -0
  257. package/packs/software-delivery/src/ports/index.ts +10 -0
  258. package/packs/software-delivery/src/ports/sync-validator.ports.ts +59 -0
  259. package/packs/software-delivery/src/ports/wu-helpers.ports.ts +168 -0
  260. package/packs/software-delivery/src/ports/wu-state.ports.ts +241 -0
  261. package/packs/software-delivery/src/primitives/index.ts +5 -0
  262. package/packs/software-delivery/src/runtime/index.ts +6 -0
  263. package/packs/software-delivery/src/runtime/work-classifier.ts +561 -0
  264. package/packs/software-delivery/src/sandbox/index.ts +10 -0
  265. package/packs/software-delivery/src/sandbox/sandbox-allowlist.ts +118 -0
  266. package/packs/software-delivery/src/sandbox/sandbox-backend-linux.ts +88 -0
  267. package/packs/software-delivery/src/sandbox/sandbox-backend-macos.ts +154 -0
  268. package/packs/software-delivery/src/sandbox/sandbox-backend-windows.ts +47 -0
  269. package/packs/software-delivery/src/sandbox/sandbox-profile.ts +153 -0
  270. package/packs/software-delivery/src/schemas/index.ts +5 -0
  271. package/packs/software-delivery/src/state/date-utils.ts +158 -0
  272. package/packs/software-delivery/src/state/index.ts +15 -0
  273. package/packs/software-delivery/src/state/state-machine.ts +119 -0
  274. package/packs/software-delivery/src/state/wu-doc-types.ts +51 -0
  275. package/packs/software-delivery/src/state/wu-paths.ts +381 -0
  276. package/packs/software-delivery/src/state/wu-schema.ts +1139 -0
  277. package/packs/software-delivery/src/state/wu-state-schema.ts +255 -0
  278. package/packs/software-delivery/src/state/wu-yaml.ts +338 -0
  279. package/packs/software-delivery/src/types.d.ts +16 -0
  280. package/packs/software-delivery/tool-impl/wu-lifecycle-tools.ts +18 -0
  281. package/packs/software-delivery/tsconfig.json +28 -2
  282. package/templates/core/AGENTS.md.template +76 -17
  283. package/templates/core/LUMENFLOW.md.template +265 -66
  284. package/templates/core/_frameworks/lumenflow/wu-sizing-guide.md.template +180 -116
  285. package/templates/core/ai/onboarding/agent-invocation-guide.md.template +26 -8
  286. package/templates/core/ai/onboarding/existing-project-bootstrap.md.template +171 -0
  287. package/templates/core/ai/onboarding/first-15-mins.md.template +3 -1
  288. package/templates/core/ai/onboarding/first-wu-mistakes.md.template +1 -1
  289. package/templates/core/ai/onboarding/initiative-orchestration.md.template +46 -30
  290. package/templates/core/ai/onboarding/quick-ref-commands.md.template +36 -33
  291. package/templates/core/ai/onboarding/release-process.md.template +8 -7
  292. package/templates/core/ai/onboarding/starting-prompt.md.template +2 -0
  293. package/templates/core/ai/onboarding/troubleshooting-wu-done.md.template +62 -0
  294. package/templates/vendors/claude/.claude/CLAUDE.md.template +29 -54
  295. package/templates/vendors/cursor/.cursor/rules/lumenflow.md.template +24 -52
  296. package/templates/vendors/windsurf/.windsurf/rules/lumenflow.md.template +24 -52
@@ -0,0 +1,252 @@
1
+ // Copyright (c) 2026 Hellmai Ltd
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+ //
4
+ // WU-2732 (INIT-060 Phase 3, ADR-013 §1 abort + §6 governance):
5
+ // Pattern A tracer-bullet mock implementation for the six
6
+ // remote-control tools. Returns deterministic fixed responses so the
7
+ // wire contract (manifest entries, POST /tools/:name round-trip,
8
+ // tool_called emission) can be proven BEFORE the real kernel-side
9
+ // implementation lands in WU-2733.
10
+ //
11
+ // REAL IMPL WILL REPLACE THIS FILE: WU-2733 ships
12
+ // tool-impl/remote-controls.ts with the real pause/resume/abort logic.
13
+ // The mock's role is purely contract stabilisation.
14
+
15
+ import { randomUUID } from 'node:crypto';
16
+ import type { ExecutionContext, ToolOutput } from '@lumenflow/kernel';
17
+ import {
18
+ AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES,
19
+ MOCK_SESSION_ID,
20
+ type AbortTurnInvocationResult,
21
+ type AgentRuntimeRemoteControlToolName,
22
+ type AnyRemoteControlInvocationResult,
23
+ type RemoteControlInvocationResult,
24
+ } from '../remote-controls/types.js';
25
+ import type {
26
+ AbortTurnInvocationInput,
27
+ RemoteControlInvocationInput,
28
+ RemoteControlPort,
29
+ } from '../remote-controls/port.js';
30
+ import {
31
+ buildToolCalledEvent,
32
+ emitAgentRuntimeEvent,
33
+ type AgentRuntimeEventSink,
34
+ } from '../turn-lifecycle-events.js';
35
+
36
+ const MOCK_TURN_INDEX = 0;
37
+ const MOCK_CLEANUP_STATUS_CLEAN = 'clean' as const;
38
+ const MOCK_ABORT_RECOVERY_ACTION: string | null = null;
39
+
40
+ /**
41
+ * Mock adapter implementing RemoteControlPort. Every invocation:
42
+ * 1. Returns a deterministic response matching the declared contract.
43
+ * 2. Emits `agent-runtime:tool_called` via the supplied sink (ADR-013
44
+ * §6 governance — every tool call is auditable).
45
+ *
46
+ * Fixed session_id + turn_index in the emitted event mean tests can
47
+ * assert exact payloads without runtime coupling.
48
+ */
49
+ export class MockRemoteControlAdapter implements RemoteControlPort {
50
+ constructor(private readonly eventSink?: AgentRuntimeEventSink) {}
51
+
52
+ async resumeWorkflow(
53
+ input: RemoteControlInvocationInput,
54
+ ): Promise<RemoteControlInvocationResult> {
55
+ return this.handle(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.RESUME_WORKFLOW, input);
56
+ }
57
+
58
+ async pauseTurn(input: RemoteControlInvocationInput): Promise<RemoteControlInvocationResult> {
59
+ return this.handle(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.PAUSE_TURN, input);
60
+ }
61
+
62
+ async abortTurn(input: AbortTurnInvocationInput): Promise<AbortTurnInvocationResult> {
63
+ const sessionId = resolveSessionId(input.session_id);
64
+ this.emitToolCalled(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN, sessionId);
65
+ return {
66
+ status: 'ok',
67
+ tool_name: AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN,
68
+ session_id: sessionId,
69
+ cleanup_status: MOCK_CLEANUP_STATUS_CLEAN,
70
+ recovery_action: MOCK_ABORT_RECOVERY_ACTION,
71
+ };
72
+ }
73
+
74
+ async elevateAutonomy(
75
+ input: RemoteControlInvocationInput,
76
+ ): Promise<RemoteControlInvocationResult> {
77
+ return this.handle(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ELEVATE_AUTONOMY, input);
78
+ }
79
+
80
+ async lowerAutonomy(input: RemoteControlInvocationInput): Promise<RemoteControlInvocationResult> {
81
+ return this.handle(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.LOWER_AUTONOMY, input);
82
+ }
83
+
84
+ async approveInflight(
85
+ input: RemoteControlInvocationInput,
86
+ ): Promise<RemoteControlInvocationResult> {
87
+ return this.handle(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.APPROVE_INFLIGHT, input);
88
+ }
89
+
90
+ private handle(
91
+ toolName: AgentRuntimeRemoteControlToolName,
92
+ input: RemoteControlInvocationInput,
93
+ ): RemoteControlInvocationResult {
94
+ const sessionId = resolveSessionId(input.session_id);
95
+ this.emitToolCalled(toolName, sessionId);
96
+ return {
97
+ status: 'ok',
98
+ tool_name: toolName,
99
+ session_id: sessionId,
100
+ };
101
+ }
102
+
103
+ private emitToolCalled(toolName: AgentRuntimeRemoteControlToolName, sessionId: string): void {
104
+ if (!this.eventSink) {
105
+ return;
106
+ }
107
+ emitAgentRuntimeEvent(
108
+ this.eventSink,
109
+ buildToolCalledEvent({
110
+ session_id: sessionId,
111
+ turn_index: MOCK_TURN_INDEX,
112
+ tool_name: toolName,
113
+ tool_call_id: randomUUID(),
114
+ }),
115
+ );
116
+ }
117
+ }
118
+
119
+ function resolveSessionId(input: string | undefined): string {
120
+ if (input !== undefined && input.trim().length > 0) {
121
+ return input;
122
+ }
123
+ return MOCK_SESSION_ID;
124
+ }
125
+
126
+ /**
127
+ * Factory for the default mock adapter. Tests and the manifest-wired
128
+ * tool functions below both use this so the fixture surface has a
129
+ * single instantiation path.
130
+ */
131
+ export function createMockRemoteControlAdapter(
132
+ eventSink?: AgentRuntimeEventSink,
133
+ ): MockRemoteControlAdapter {
134
+ return new MockRemoteControlAdapter(eventSink);
135
+ }
136
+
137
+ /**
138
+ * Manifest-entrypoint adapter: each of the six exported functions
139
+ * matches the kernel's InProcessToolFn signature
140
+ * `(input: unknown, ctx: ExecutionContext) => Promise<ToolOutput>`.
141
+ *
142
+ * The module-scoped sink is rebound via `setMockRemoteControlEventSink`
143
+ * for tests that want to assert tool_called emission. Production
144
+ * wiring (HTTP surface → runtime → manifest-dispatch) will ignore the
145
+ * sink and rely on runtime-level tool_called events; the in-process
146
+ * sink is a mock-only convenience for the unit suite.
147
+ */
148
+ let moduleEventSink: AgentRuntimeEventSink | undefined;
149
+
150
+ export function setMockRemoteControlEventSink(sink: AgentRuntimeEventSink | undefined): void {
151
+ moduleEventSink = sink;
152
+ }
153
+
154
+ function buildToolOutput(data: AnyRemoteControlInvocationResult): ToolOutput {
155
+ return {
156
+ success: true,
157
+ data,
158
+ metadata: {
159
+ mock: true,
160
+ tracer_bullet: 'WU-2732',
161
+ },
162
+ };
163
+ }
164
+
165
+ async function invokeMock(
166
+ toolName: AgentRuntimeRemoteControlToolName,
167
+ input: unknown,
168
+ ): Promise<ToolOutput> {
169
+ const adapter = createMockRemoteControlAdapter(moduleEventSink);
170
+ const parsed = parseInput(input);
171
+
172
+ switch (toolName) {
173
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.RESUME_WORKFLOW:
174
+ return buildToolOutput(await adapter.resumeWorkflow(parsed));
175
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.PAUSE_TURN:
176
+ return buildToolOutput(await adapter.pauseTurn(parsed));
177
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN:
178
+ return buildToolOutput(await adapter.abortTurn(parsed));
179
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ELEVATE_AUTONOMY:
180
+ return buildToolOutput(await adapter.elevateAutonomy(parsed));
181
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.LOWER_AUTONOMY:
182
+ return buildToolOutput(await adapter.lowerAutonomy(parsed));
183
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.APPROVE_INFLIGHT:
184
+ return buildToolOutput(await adapter.approveInflight(parsed));
185
+ default:
186
+ return exhaustive(toolName);
187
+ }
188
+ }
189
+
190
+ function parseInput(input: unknown): AbortTurnInvocationInput {
191
+ if (input === null || input === undefined) {
192
+ return {};
193
+ }
194
+ if (typeof input !== 'object') {
195
+ return {};
196
+ }
197
+ const record = input as Record<string, unknown>;
198
+ const result: AbortTurnInvocationInput = {};
199
+ if (typeof record.session_id === 'string') {
200
+ result.session_id = record.session_id;
201
+ }
202
+ if (typeof record.reason === 'string') {
203
+ result.reason = record.reason;
204
+ }
205
+ return result;
206
+ }
207
+
208
+ function exhaustive(value: never): never {
209
+ throw new Error(`Unhandled remote-control tool: ${String(value)}`);
210
+ }
211
+
212
+ export async function resumeWorkflowMockTool(
213
+ input: unknown,
214
+ _ctx: ExecutionContext,
215
+ ): Promise<ToolOutput> {
216
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.RESUME_WORKFLOW, input);
217
+ }
218
+
219
+ export async function pauseTurnMockTool(
220
+ input: unknown,
221
+ _ctx: ExecutionContext,
222
+ ): Promise<ToolOutput> {
223
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.PAUSE_TURN, input);
224
+ }
225
+
226
+ export async function abortTurnMockTool(
227
+ input: unknown,
228
+ _ctx: ExecutionContext,
229
+ ): Promise<ToolOutput> {
230
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN, input);
231
+ }
232
+
233
+ export async function elevateAutonomyMockTool(
234
+ input: unknown,
235
+ _ctx: ExecutionContext,
236
+ ): Promise<ToolOutput> {
237
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ELEVATE_AUTONOMY, input);
238
+ }
239
+
240
+ export async function lowerAutonomyMockTool(
241
+ input: unknown,
242
+ _ctx: ExecutionContext,
243
+ ): Promise<ToolOutput> {
244
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.LOWER_AUTONOMY, input);
245
+ }
246
+
247
+ export async function approveInflightMockTool(
248
+ input: unknown,
249
+ _ctx: ExecutionContext,
250
+ ): Promise<ToolOutput> {
251
+ return invokeMock(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.APPROVE_INFLIGHT, input);
252
+ }
@@ -0,0 +1,273 @@
1
+ // Copyright (c) 2026 Hellmai Ltd
2
+ // SPDX-License-Identifier: AGPL-3.0-only
3
+ //
4
+ // WU-2733 (INIT-060 Phase 3, ADR-013 §1 + §6):
5
+ // Real kernel-side RemoteControlPort adapter + manifest-entrypoint tool
6
+ // functions. Replaces the mock tracer-bullet entries from WU-2732
7
+ // without changing tool names, permissions, scopes, approvals, input
8
+ // schemas, or result shapes — the wire contract survives unchanged
9
+ // because every shape is pinned by `remote-controls/port.ts` +
10
+ // `remote-controls/types.ts`.
11
+ //
12
+ // Side effects: this adapter persists remote-control state under
13
+ // `.agent-runtime/remote-controls/<session>.json` (sidecar, isolated
14
+ // from the core orchestration workflow state store so existing
15
+ // workflow schema validation is untouched) and emits events through
16
+ // the supplied sink.
17
+
18
+ import type { ExecutionContext, ToolOutput } from '@lumenflow/kernel';
19
+ import {
20
+ abortTurn,
21
+ approveInflight,
22
+ elevateAutonomy,
23
+ lowerAutonomy,
24
+ pauseTurn,
25
+ resumeWorkflow,
26
+ type RemoteControlOperationContext,
27
+ } from '../remote-controls/operations.js';
28
+ import type {
29
+ AbortTurnInvocationInput,
30
+ RemoteControlInvocationInput,
31
+ RemoteControlPort,
32
+ } from '../remote-controls/port.js';
33
+ import {
34
+ createRemoteControlStateStore,
35
+ type RemoteControlStateStore,
36
+ } from '../remote-controls/state-store.js';
37
+ import {
38
+ AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES,
39
+ type AbortTurnInvocationResult,
40
+ type AgentRuntimeRemoteControlToolName,
41
+ type AnyRemoteControlInvocationResult,
42
+ type RemoteControlInvocationResult,
43
+ } from '../remote-controls/types.js';
44
+ import type { AgentRuntimeEventSink } from '../turn-lifecycle-events.js';
45
+
46
+ export interface RealRemoteControlAdapterInput {
47
+ store: RemoteControlStateStore;
48
+ eventSink?: AgentRuntimeEventSink;
49
+ now?: () => string;
50
+ }
51
+
52
+ /**
53
+ * Real (non-mock) implementation of RemoteControlPort. Wires each port
54
+ * method to the corresponding operation in `operations.ts`. Every port
55
+ * method load-bears a `RemoteControlStateStore` so state survives
56
+ * restarts; emission goes through the `AgentRuntimeEventSink` supplied
57
+ * at construction (matches the mock's pattern so wiring parity holds).
58
+ */
59
+ export class RealRemoteControlAdapter implements RemoteControlPort {
60
+ private readonly context: RemoteControlOperationContext;
61
+
62
+ constructor(input: RealRemoteControlAdapterInput) {
63
+ this.context = {
64
+ store: input.store,
65
+ eventSink: input.eventSink,
66
+ now: input.now ?? (() => new Date().toISOString()),
67
+ };
68
+ }
69
+
70
+ async resumeWorkflow(
71
+ input: RemoteControlInvocationInput,
72
+ ): Promise<RemoteControlInvocationResult> {
73
+ return resumeWorkflow(input, this.context);
74
+ }
75
+
76
+ async pauseTurn(input: RemoteControlInvocationInput): Promise<RemoteControlInvocationResult> {
77
+ return pauseTurn(input, this.context);
78
+ }
79
+
80
+ async abortTurn(input: AbortTurnInvocationInput): Promise<AbortTurnInvocationResult> {
81
+ return abortTurn(input, this.context);
82
+ }
83
+
84
+ async elevateAutonomy(
85
+ input: RemoteControlInvocationInput,
86
+ ): Promise<RemoteControlInvocationResult> {
87
+ return elevateAutonomy(input, this.context);
88
+ }
89
+
90
+ async lowerAutonomy(input: RemoteControlInvocationInput): Promise<RemoteControlInvocationResult> {
91
+ return lowerAutonomy(input, this.context);
92
+ }
93
+
94
+ async approveInflight(
95
+ input: RemoteControlInvocationInput,
96
+ ): Promise<RemoteControlInvocationResult> {
97
+ return approveInflight(input, this.context);
98
+ }
99
+ }
100
+
101
+ export function createRealRemoteControlAdapter(
102
+ input: RealRemoteControlAdapterInput,
103
+ ): RealRemoteControlAdapter {
104
+ return new RealRemoteControlAdapter(input);
105
+ }
106
+
107
+ // ---------------------------------------------------------------------------
108
+ // Manifest-entrypoint wiring
109
+ // ---------------------------------------------------------------------------
110
+ //
111
+ // Each exported `*Tool` below matches the kernel's InProcessToolFn
112
+ // signature `(input, ctx) => Promise<ToolOutput>`. The `ctx` carries
113
+ // the run identifiers and allowed_scopes; we derive the storage root
114
+ // from the runtime's allowlist `.agent-runtime/**` path — or the
115
+ // module-scoped override when the host wires one directly (test
116
+ // harness + conductor runtime).
117
+
118
+ let moduleEventSink: AgentRuntimeEventSink | undefined;
119
+ let moduleStore: RemoteControlStateStore | undefined;
120
+ let moduleNow: (() => string) | undefined;
121
+
122
+ /**
123
+ * Test/host-wiring hooks so the manifest-entrypoint tool functions
124
+ * reuse a single adapter across invocations. Production wiring rebinds
125
+ * these during pack activation (capability factory registers the
126
+ * adapter once); tests clear them in beforeEach.
127
+ */
128
+ export function setRealRemoteControlEventSink(sink: AgentRuntimeEventSink | undefined): void {
129
+ moduleEventSink = sink;
130
+ }
131
+
132
+ export function setRealRemoteControlStateStore(store: RemoteControlStateStore | undefined): void {
133
+ moduleStore = store;
134
+ }
135
+
136
+ export function setRealRemoteControlNow(now: (() => string) | undefined): void {
137
+ moduleNow = now;
138
+ }
139
+
140
+ /** Resets all module-scoped wiring. Intended for unit-test teardown. */
141
+ export function resetRealRemoteControlModuleWiring(): void {
142
+ moduleEventSink = undefined;
143
+ moduleStore = undefined;
144
+ moduleNow = undefined;
145
+ }
146
+
147
+ function resolveStore(ctx: ExecutionContext): RemoteControlStateStore {
148
+ if (moduleStore) {
149
+ return moduleStore;
150
+ }
151
+ const workspaceRoot = resolveWorkspaceRoot(ctx);
152
+ return createRemoteControlStateStore({ workspaceRoot });
153
+ }
154
+
155
+ /**
156
+ * Pull the workspace root from the kernel ExecutionContext. Falls back
157
+ * to the process cwd when no explicit workspace_root metadata is
158
+ * present — matching how the rest of the agent-runtime pack resolves
159
+ * storage roots (keeps us on the kernel-provided path when available
160
+ * without hard-failing when a test harness omits it).
161
+ */
162
+ function resolveWorkspaceRoot(ctx: ExecutionContext): string {
163
+ const metadata = (ctx as unknown as { metadata?: Record<string, unknown> }).metadata;
164
+ const fromMetadata = metadata?.workspace_root;
165
+ if (typeof fromMetadata === 'string' && fromMetadata.length > 0) {
166
+ return fromMetadata;
167
+ }
168
+ const fromCtx = (ctx as unknown as { workspace_root?: unknown }).workspace_root;
169
+ if (typeof fromCtx === 'string' && fromCtx.length > 0) {
170
+ return fromCtx;
171
+ }
172
+ return process.cwd();
173
+ }
174
+
175
+ function buildToolOutput(data: AnyRemoteControlInvocationResult): ToolOutput {
176
+ return {
177
+ success: true,
178
+ data,
179
+ metadata: {
180
+ real: true,
181
+ source: 'WU-2733',
182
+ },
183
+ };
184
+ }
185
+
186
+ function parseInput(input: unknown): AbortTurnInvocationInput {
187
+ if (input === null || input === undefined) {
188
+ return {};
189
+ }
190
+ if (typeof input !== 'object') {
191
+ return {};
192
+ }
193
+ const record = input as Record<string, unknown>;
194
+ const result: AbortTurnInvocationInput = {};
195
+ if (typeof record.session_id === 'string') {
196
+ result.session_id = record.session_id;
197
+ }
198
+ if (typeof record.reason === 'string') {
199
+ result.reason = record.reason;
200
+ }
201
+ return result;
202
+ }
203
+
204
+ async function invoke(
205
+ toolName: AgentRuntimeRemoteControlToolName,
206
+ input: unknown,
207
+ ctx: ExecutionContext,
208
+ ): Promise<ToolOutput> {
209
+ const store = resolveStore(ctx);
210
+ const adapter = new RealRemoteControlAdapter({
211
+ store,
212
+ eventSink: moduleEventSink,
213
+ now: moduleNow,
214
+ });
215
+ const parsed = parseInput(input);
216
+
217
+ switch (toolName) {
218
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.RESUME_WORKFLOW:
219
+ return buildToolOutput(await adapter.resumeWorkflow(parsed));
220
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.PAUSE_TURN:
221
+ return buildToolOutput(await adapter.pauseTurn(parsed));
222
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN:
223
+ return buildToolOutput(await adapter.abortTurn(parsed));
224
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ELEVATE_AUTONOMY:
225
+ return buildToolOutput(await adapter.elevateAutonomy(parsed));
226
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.LOWER_AUTONOMY:
227
+ return buildToolOutput(await adapter.lowerAutonomy(parsed));
228
+ case AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.APPROVE_INFLIGHT:
229
+ return buildToolOutput(await adapter.approveInflight(parsed));
230
+ default:
231
+ return exhaustive(toolName);
232
+ }
233
+ }
234
+
235
+ function exhaustive(value: never): never {
236
+ throw new Error(`Unhandled remote-control tool: ${String(value)}`);
237
+ }
238
+
239
+ export async function resumeWorkflowTool(
240
+ input: unknown,
241
+ ctx: ExecutionContext,
242
+ ): Promise<ToolOutput> {
243
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.RESUME_WORKFLOW, input, ctx);
244
+ }
245
+
246
+ export async function pauseTurnTool(input: unknown, ctx: ExecutionContext): Promise<ToolOutput> {
247
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.PAUSE_TURN, input, ctx);
248
+ }
249
+
250
+ export async function abortTurnTool(input: unknown, ctx: ExecutionContext): Promise<ToolOutput> {
251
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ABORT_TURN, input, ctx);
252
+ }
253
+
254
+ export async function elevateAutonomyTool(
255
+ input: unknown,
256
+ ctx: ExecutionContext,
257
+ ): Promise<ToolOutput> {
258
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.ELEVATE_AUTONOMY, input, ctx);
259
+ }
260
+
261
+ export async function lowerAutonomyTool(
262
+ input: unknown,
263
+ ctx: ExecutionContext,
264
+ ): Promise<ToolOutput> {
265
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.LOWER_AUTONOMY, input, ctx);
266
+ }
267
+
268
+ export async function approveInflightTool(
269
+ input: unknown,
270
+ ctx: ExecutionContext,
271
+ ): Promise<ToolOutput> {
272
+ return invoke(AGENT_RUNTIME_REMOTE_CONTROL_TOOL_NAMES.APPROVE_INFLIGHT, input, ctx);
273
+ }
@@ -15,6 +15,6 @@
15
15
  "checkJs": false,
16
16
  "noEmitOnError": true
17
17
  },
18
- "include": ["*.ts", "tools/**/*.ts", "tool-impl/**/*.ts"],
18
+ "include": ["*.ts", "*.d.ts", "tools/**/*.ts", "tool-impl/**/*.ts"],
19
19
  "exclude": ["node_modules", "dist", "__tests__"]
20
20
  }