@jmoyers/harness 0.1.11 → 0.1.20

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 (232) hide show
  1. package/README.md +31 -39
  2. package/package.json +31 -11
  3. package/packages/harness-ai/src/anthropic-protocol.ts +68 -68
  4. package/packages/harness-ai/src/stream-text.ts +13 -91
  5. package/packages/harness-ui/src/frame-primitives.ts +158 -0
  6. package/packages/harness-ui/src/index.ts +18 -0
  7. package/packages/harness-ui/src/interaction/conversation-input-forwarder.ts +221 -0
  8. package/packages/harness-ui/src/interaction/conversation-selection-input.ts +213 -0
  9. package/packages/harness-ui/src/interaction/global-shortcut-input.ts +172 -0
  10. package/{src/ui → packages/harness-ui/src/interaction}/input-preflight.ts +10 -12
  11. package/{src/ui → packages/harness-ui/src/interaction}/input-token-router.ts +120 -69
  12. package/packages/harness-ui/src/interaction/input.ts +420 -0
  13. package/packages/harness-ui/src/interaction/left-nav-input.ts +166 -0
  14. package/{src/ui → packages/harness-ui/src/interaction}/main-pane-pointer-input.ts +91 -23
  15. package/{src/ui → packages/harness-ui/src/interaction}/pointer-routing-input.ts +112 -48
  16. package/packages/harness-ui/src/interaction/rail-pointer-input.ts +62 -0
  17. package/packages/harness-ui/src/interaction/repository-fold-input.ts +118 -0
  18. package/packages/harness-ui/src/kit.ts +476 -0
  19. package/packages/harness-ui/src/layout.ts +238 -0
  20. package/packages/harness-ui/src/modal-manager.ts +222 -0
  21. package/{src/ui → packages/harness-ui/src}/screen.ts +53 -26
  22. package/packages/harness-ui/src/surface.ts +252 -0
  23. package/packages/harness-ui/src/text-layout.ts +210 -0
  24. package/packages/nim-core/src/contracts.ts +239 -0
  25. package/packages/nim-core/src/event-store.ts +299 -0
  26. package/packages/nim-core/src/events.ts +53 -0
  27. package/packages/nim-core/src/index.ts +9 -0
  28. package/packages/nim-core/src/provider-router.ts +129 -0
  29. package/packages/nim-core/src/providers/anthropic-driver.ts +291 -0
  30. package/packages/nim-core/src/runtime-factory.ts +49 -0
  31. package/packages/nim-core/src/runtime.ts +1797 -0
  32. package/packages/nim-core/src/session-store.ts +516 -0
  33. package/packages/nim-core/src/telemetry.ts +48 -0
  34. package/packages/nim-test-tui/src/index.ts +150 -0
  35. package/packages/nim-ui-core/src/index.ts +1 -0
  36. package/packages/nim-ui-core/src/projection.ts +87 -0
  37. package/scripts/codex-live-mux-runtime.ts +2 -3872
  38. package/scripts/control-plane-daemon.ts +11 -0
  39. package/scripts/harness-bin.js +5 -0
  40. package/scripts/harness-commands.ts +300 -0
  41. package/scripts/harness-runtime.ts +82 -0
  42. package/scripts/harness.ts +33 -3019
  43. package/scripts/nim-tui-smoke.ts +748 -0
  44. package/src/cli/auth/runtime.ts +948 -0
  45. package/src/cli/gateway/runtime.ts +1872 -0
  46. package/src/cli/parsing/flags.ts +23 -0
  47. package/src/cli/parsing/session.ts +42 -0
  48. package/src/cli/runtime/context.ts +193 -0
  49. package/src/cli/runtime-app/application.ts +392 -0
  50. package/src/cli/runtime-infra/gateway-control.ts +729 -0
  51. package/{scripts/harness-inspector.ts → src/cli/workflows/inspector.ts} +14 -11
  52. package/src/cli/workflows/runtime.ts +965 -0
  53. package/src/clients/tui/left-rail-interactions.ts +519 -0
  54. package/src/clients/tui/main-pane-interactions.ts +509 -0
  55. package/src/clients/tui/modal-input-routing.ts +71 -0
  56. package/src/clients/tui/render-snapshot-adapter.ts +88 -0
  57. package/src/clients/web/synced-selectors.ts +132 -0
  58. package/src/codex/live-session.ts +82 -29
  59. package/src/config/config-core.ts +348 -8
  60. package/src/config/harness.config.template.jsonc +33 -0
  61. package/src/control-plane/agent-realtime-api.ts +82 -427
  62. package/src/control-plane/session-summary.ts +10 -81
  63. package/src/control-plane/status/reducer-base.ts +12 -12
  64. package/src/control-plane/status/reducers/claude-status-reducer.ts +3 -3
  65. package/src/control-plane/status/reducers/codex-status-reducer.ts +4 -4
  66. package/src/control-plane/status/reducers/cursor-status-reducer.ts +3 -3
  67. package/src/control-plane/stream-client.ts +12 -2
  68. package/src/control-plane/stream-command-parser.ts +83 -143
  69. package/src/control-plane/stream-protocol.ts +53 -37
  70. package/src/control-plane/stream-server-command.ts +376 -69
  71. package/src/control-plane/stream-server-session-runtime.ts +3 -2
  72. package/src/control-plane/stream-server.ts +864 -70
  73. package/src/control-plane/stream-session-runtime-types.ts +41 -0
  74. package/src/{mux/live-mux/control-plane-records.ts → core/contracts/records.ts} +24 -97
  75. package/src/core/state/observed-stream-cursor.ts +43 -0
  76. package/src/core/state/synced-observed-state.ts +273 -0
  77. package/src/core/store/harness-synced-store.ts +81 -0
  78. package/src/diff/budget.ts +136 -0
  79. package/src/diff/build.ts +289 -0
  80. package/src/diff/chunker.ts +146 -0
  81. package/src/diff/git-invoke.ts +315 -0
  82. package/src/diff/git-parse.ts +472 -0
  83. package/src/diff/hash.ts +70 -0
  84. package/src/diff/index.ts +24 -0
  85. package/src/diff/normalize.ts +134 -0
  86. package/src/diff/types.ts +178 -0
  87. package/src/diff-ui/args.ts +346 -0
  88. package/src/diff-ui/commands.ts +123 -0
  89. package/src/diff-ui/finder.ts +94 -0
  90. package/src/diff-ui/highlight.ts +127 -0
  91. package/src/diff-ui/index.ts +2 -0
  92. package/src/diff-ui/model.ts +141 -0
  93. package/src/diff-ui/pager.ts +412 -0
  94. package/src/diff-ui/render.ts +337 -0
  95. package/src/diff-ui/runtime.ts +379 -0
  96. package/src/diff-ui/state.ts +224 -0
  97. package/src/diff-ui/types.ts +236 -0
  98. package/src/domain/workspace.ts +68 -5
  99. package/src/mux/control-plane-op-queue.ts +93 -7
  100. package/src/mux/conversation-rail.ts +28 -71
  101. package/src/mux/dual-pane-core.ts +13 -13
  102. package/src/mux/harness-core-ui.ts +313 -42
  103. package/src/mux/input-shortcuts.ts +13 -131
  104. package/src/mux/keybinding-catalog.ts +340 -0
  105. package/src/mux/keybinding-registry.ts +103 -0
  106. package/src/mux/live-mux/command-menu-open-in.ts +280 -0
  107. package/src/mux/live-mux/command-menu.ts +167 -4
  108. package/src/mux/live-mux/conversation-state.ts +13 -0
  109. package/src/mux/live-mux/directory-resolution.ts +1 -1
  110. package/src/mux/live-mux/git-snapshot.ts +33 -2
  111. package/src/mux/live-mux/global-shortcut-handlers.ts +6 -0
  112. package/src/mux/live-mux/home-pane-drop.ts +1 -1
  113. package/src/mux/live-mux/home-pane-pointer.ts +10 -0
  114. package/src/mux/live-mux/input-forwarding.ts +59 -2
  115. package/src/mux/live-mux/left-nav-activation.ts +124 -7
  116. package/src/mux/live-mux/left-nav.ts +35 -0
  117. package/src/mux/live-mux/link-click.ts +292 -0
  118. package/src/mux/live-mux/modal-command-menu-handler.ts +46 -9
  119. package/src/mux/live-mux/modal-conversation-handlers.ts +5 -1
  120. package/src/mux/live-mux/modal-input-reducers.ts +77 -12
  121. package/src/mux/live-mux/modal-overlays.ts +168 -34
  122. package/src/mux/live-mux/modal-pointer.ts +3 -7
  123. package/src/mux/live-mux/modal-prompt-handlers.ts +23 -2
  124. package/src/mux/live-mux/modal-release-notes-handler.ts +111 -0
  125. package/src/mux/live-mux/modal-task-editor-handler.ts +16 -11
  126. package/src/mux/live-mux/pointer-routing.ts +5 -2
  127. package/src/mux/live-mux/project-pane-pointer.ts +8 -0
  128. package/src/mux/live-mux/rail-layout.ts +33 -30
  129. package/src/mux/live-mux/release-notes.ts +383 -0
  130. package/src/mux/live-mux/render-trace-analysis.ts +52 -7
  131. package/src/mux/live-mux/repository-folding.ts +3 -0
  132. package/src/mux/live-mux/selection.ts +0 -4
  133. package/src/mux/live-mux/session-diagnostics-paths.ts +21 -0
  134. package/src/mux/project-pane-github-review.ts +271 -0
  135. package/src/mux/render-frame.ts +4 -0
  136. package/src/mux/runtime-app/codex-live-mux-runtime.ts +5191 -0
  137. package/src/mux/task-composer.ts +21 -14
  138. package/src/mux/task-focused-pane.ts +118 -117
  139. package/src/mux/task-screen-keybindings.ts +10 -101
  140. package/src/mux/workspace-rail-model.ts +270 -104
  141. package/src/mux/workspace-rail.ts +45 -22
  142. package/src/pty/session-broker.ts +1 -1
  143. package/{scripts → src/recording}/terminal-recording-gif-lib.ts +2 -2
  144. package/src/services/control-plane.ts +50 -32
  145. package/src/services/conversation-lifecycle.ts +118 -87
  146. package/src/services/conversation-startup-hydration.ts +20 -12
  147. package/src/services/directory-hydration.ts +21 -16
  148. package/src/services/event-persistence.ts +7 -0
  149. package/src/services/left-rail-pointer-handler.ts +329 -0
  150. package/src/services/mux-ui-state-persistence.ts +5 -1
  151. package/src/services/recording.ts +34 -26
  152. package/src/services/runtime-command-menu-agent-tools.ts +1 -1
  153. package/src/services/runtime-control-actions.ts +79 -61
  154. package/src/services/runtime-control-plane-ops.ts +122 -83
  155. package/src/services/runtime-conversation-actions.ts +40 -26
  156. package/src/services/runtime-conversation-activation.ts +73 -46
  157. package/src/services/runtime-conversation-starter.ts +53 -45
  158. package/src/services/runtime-conversation-title-edit.ts +91 -80
  159. package/src/services/runtime-envelope-handler.ts +107 -105
  160. package/src/services/runtime-git-state.ts +42 -29
  161. package/src/services/runtime-layout-resize.ts +3 -1
  162. package/src/services/runtime-left-rail-render.ts +99 -63
  163. package/src/services/runtime-nim-cli-session.ts +438 -0
  164. package/src/services/runtime-nim-session.ts +705 -0
  165. package/src/services/runtime-nim-tool-bridge.ts +141 -0
  166. package/src/services/runtime-observed-event-projection-pipeline.ts +45 -0
  167. package/src/services/runtime-process-wiring.ts +29 -36
  168. package/src/services/runtime-project-pane-github-review-cache.ts +164 -0
  169. package/src/services/runtime-render-flush.ts +63 -70
  170. package/src/services/runtime-render-lifecycle.ts +65 -64
  171. package/src/services/runtime-render-orchestrator.ts +55 -45
  172. package/src/services/runtime-render-pipeline.ts +106 -103
  173. package/src/services/runtime-render-state.ts +62 -49
  174. package/src/services/runtime-repository-actions.ts +97 -72
  175. package/src/services/runtime-right-pane-render.ts +80 -53
  176. package/src/services/runtime-shutdown.ts +38 -35
  177. package/src/services/runtime-stream-subscriptions.ts +35 -27
  178. package/src/services/runtime-task-composer-persistence.ts +71 -59
  179. package/src/services/runtime-task-composer-snapshot.ts +14 -0
  180. package/src/services/runtime-task-editor-actions.ts +46 -29
  181. package/src/services/runtime-task-pane-actions.ts +220 -134
  182. package/src/services/runtime-task-pane-shortcuts.ts +323 -123
  183. package/src/services/runtime-workspace-observed-effect-queue.ts +25 -0
  184. package/src/services/runtime-workspace-observed-events.ts +33 -184
  185. package/src/services/runtime-workspace-observed-transition-policy.ts +228 -0
  186. package/src/services/session-diagnostics-store.ts +217 -0
  187. package/src/services/startup-background-resume.ts +26 -21
  188. package/src/services/startup-orchestrator.ts +16 -13
  189. package/src/services/startup-paint-tracker.ts +29 -21
  190. package/src/services/startup-persisted-conversation-queue.ts +19 -13
  191. package/src/services/startup-settled-gate.ts +25 -15
  192. package/src/services/startup-shutdown.ts +18 -22
  193. package/src/services/startup-state-hydration.ts +44 -34
  194. package/src/services/startup-visibility.ts +12 -4
  195. package/src/services/task-pane-selection-actions.ts +89 -72
  196. package/src/services/task-planning-hydration.ts +24 -18
  197. package/src/services/task-planning-observed-events.ts +50 -52
  198. package/src/services/workspace-observed-events.ts +66 -63
  199. package/src/storage/storage-lifecycle-core.ts +438 -0
  200. package/src/store/control-plane-store-normalize.ts +33 -242
  201. package/src/store/control-plane-store-types.ts +1 -35
  202. package/src/store/control-plane-store.ts +360 -56
  203. package/src/store/event-store.ts +366 -8
  204. package/src/terminal/snapshot-oracle.ts +207 -94
  205. package/src/ui/mux-theme.ts +112 -8
  206. package/src/ui/panes/home-gridfire.ts +40 -31
  207. package/src/ui/panes/home.ts +10 -2
  208. package/src/ui/panes/nim.ts +315 -0
  209. package/src/mux/live-mux/actions-task.ts +0 -115
  210. package/src/mux/live-mux/left-rail-actions.ts +0 -118
  211. package/src/mux/live-mux/left-rail-conversation-click.ts +0 -85
  212. package/src/mux/live-mux/left-rail-pointer.ts +0 -74
  213. package/src/mux/live-mux/task-pane-shortcuts.ts +0 -206
  214. package/src/services/runtime-directory-actions.ts +0 -164
  215. package/src/services/runtime-input-pipeline.ts +0 -50
  216. package/src/services/runtime-input-router.ts +0 -195
  217. package/src/services/runtime-main-pane-input.ts +0 -230
  218. package/src/services/runtime-modal-input.ts +0 -137
  219. package/src/services/runtime-navigation-input.ts +0 -197
  220. package/src/services/runtime-rail-input.ts +0 -279
  221. package/src/services/runtime-task-pane.ts +0 -62
  222. package/src/services/runtime-workspace-actions.ts +0 -158
  223. package/src/ui/conversation-input-forwarder.ts +0 -114
  224. package/src/ui/conversation-selection-input.ts +0 -103
  225. package/src/ui/global-shortcut-input.ts +0 -89
  226. package/src/ui/input.ts +0 -269
  227. package/src/ui/kit.ts +0 -509
  228. package/src/ui/left-nav-input.ts +0 -80
  229. package/src/ui/left-rail-pointer-input.ts +0 -148
  230. package/src/ui/modals/manager.ts +0 -218
  231. package/src/ui/repository-fold-input.ts +0 -91
  232. package/src/ui/surface.ts +0 -224
@@ -9,7 +9,7 @@ interface RepositoryRecordLike {
9
9
  readonly archivedAt: string | null;
10
10
  }
11
11
 
12
- interface TaskPaneSelectionActionsOptions<TTaskRecord extends TaskRecordLike> {
12
+ export interface TaskPaneSelectionActionsOptions<TTaskRecord extends TaskRecordLike> {
13
13
  readonly workspace: WorkspaceModel;
14
14
  readonly taskRecordById: (taskId: string) => TTaskRecord | undefined;
15
15
  readonly hasTask: (taskId: string) => boolean;
@@ -21,133 +21,150 @@ interface TaskPaneSelectionActionsOptions<TTaskRecord extends TaskRecordLike> {
21
21
  readonly markDirty: () => void;
22
22
  }
23
23
 
24
- export class TaskPaneSelectionActions<TTaskRecord extends TaskRecordLike> {
25
- constructor(private readonly options: TaskPaneSelectionActionsOptions<TTaskRecord>) {}
24
+ export interface TaskPaneSelectionActions {
25
+ syncTaskPaneSelectionFocus(): void;
26
+ syncTaskPaneSelection(): void;
27
+ syncTaskPaneRepositorySelection(): void;
28
+ focusDraftComposer(): void;
29
+ focusTaskComposer(taskId: string): void;
30
+ selectTaskById(taskId: string): void;
31
+ selectRepositoryById(repositoryId: string): void;
32
+ }
26
33
 
27
- syncTaskPaneSelectionFocus(): void {
34
+ export function createTaskPaneSelectionActions<TTaskRecord extends TaskRecordLike>(
35
+ options: TaskPaneSelectionActionsOptions<TTaskRecord>,
36
+ ): TaskPaneSelectionActions {
37
+ function syncTaskPaneSelectionFocus(): void {
28
38
  const hasTaskSelection =
29
- this.options.workspace.taskPaneSelectedTaskId !== null &&
30
- this.options.hasTask(this.options.workspace.taskPaneSelectedTaskId);
39
+ options.workspace.taskPaneSelectedTaskId !== null &&
40
+ options.hasTask(options.workspace.taskPaneSelectedTaskId);
31
41
  const hasRepositorySelection =
32
- this.options.workspace.taskPaneSelectedRepositoryId !== null &&
33
- this.options.hasRepository(this.options.workspace.taskPaneSelectedRepositoryId);
34
- if (this.options.workspace.taskPaneSelectionFocus === 'task' && hasTaskSelection) {
42
+ options.workspace.taskPaneSelectedRepositoryId !== null &&
43
+ options.hasRepository(options.workspace.taskPaneSelectedRepositoryId);
44
+ if (options.workspace.taskPaneSelectionFocus === 'task' && hasTaskSelection) {
35
45
  return;
36
46
  }
37
- if (this.options.workspace.taskPaneSelectionFocus === 'repository' && hasRepositorySelection) {
47
+ if (options.workspace.taskPaneSelectionFocus === 'repository' && hasRepositorySelection) {
38
48
  return;
39
49
  }
40
50
  if (hasTaskSelection) {
41
- this.options.workspace.taskPaneSelectionFocus = 'task';
51
+ options.workspace.taskPaneSelectionFocus = 'task';
42
52
  return;
43
53
  }
44
54
  if (hasRepositorySelection) {
45
- this.options.workspace.taskPaneSelectionFocus = 'repository';
55
+ options.workspace.taskPaneSelectionFocus = 'repository';
46
56
  return;
47
57
  }
48
- this.options.workspace.taskPaneSelectionFocus = 'task';
58
+ options.workspace.taskPaneSelectionFocus = 'task';
49
59
  }
50
60
 
51
- syncTaskPaneSelection(): void {
52
- const scopedTaskIds = new Set(
53
- this.options.selectedRepositoryTasks().map((task) => task.taskId),
54
- );
61
+ function syncTaskPaneSelection(): void {
62
+ const scopedTaskIds = new Set(options.selectedRepositoryTasks().map((task) => task.taskId));
55
63
  if (
56
- this.options.workspace.taskPaneSelectedTaskId !== null &&
57
- !scopedTaskIds.has(this.options.workspace.taskPaneSelectedTaskId)
64
+ options.workspace.taskPaneSelectedTaskId !== null &&
65
+ !scopedTaskIds.has(options.workspace.taskPaneSelectedTaskId)
58
66
  ) {
59
- this.options.workspace.taskPaneSelectedTaskId = null;
67
+ options.workspace.taskPaneSelectedTaskId = null;
60
68
  }
61
- if (this.options.workspace.taskPaneSelectedTaskId === null) {
62
- const scopedTasks = this.options.selectedRepositoryTasks();
63
- this.options.workspace.taskPaneSelectedTaskId = scopedTasks[0]?.taskId ?? null;
69
+ if (options.workspace.taskPaneSelectedTaskId === null) {
70
+ const scopedTasks = options.selectedRepositoryTasks();
71
+ options.workspace.taskPaneSelectedTaskId = scopedTasks[0]?.taskId ?? null;
64
72
  }
65
- this.syncTaskPaneSelectionFocus();
73
+ syncTaskPaneSelectionFocus();
66
74
  if (
67
- this.options.workspace.taskEditorTarget.kind === 'task' &&
68
- !scopedTaskIds.has(this.options.workspace.taskEditorTarget.taskId)
75
+ options.workspace.taskEditorTarget.kind === 'task' &&
76
+ !scopedTaskIds.has(options.workspace.taskEditorTarget.taskId)
69
77
  ) {
70
- this.focusDraftComposer();
78
+ focusDraftComposer();
71
79
  }
72
80
  }
73
81
 
74
- syncTaskPaneRepositorySelection(): void {
75
- if (this.options.workspace.taskPaneSelectedRepositoryId !== null) {
76
- const selectedRepository = this.options.repositoryById(
77
- this.options.workspace.taskPaneSelectedRepositoryId,
82
+ function syncTaskPaneRepositorySelection(): void {
83
+ if (options.workspace.taskPaneSelectedRepositoryId !== null) {
84
+ const selectedRepository = options.repositoryById(
85
+ options.workspace.taskPaneSelectedRepositoryId,
78
86
  );
79
87
  if (selectedRepository === undefined || selectedRepository.archivedAt !== null) {
80
- this.options.workspace.taskPaneSelectedRepositoryId = null;
88
+ options.workspace.taskPaneSelectedRepositoryId = null;
81
89
  }
82
90
  }
83
- if (this.options.workspace.taskPaneSelectedRepositoryId === null) {
84
- this.options.workspace.taskPaneSelectedRepositoryId =
85
- this.options.activeRepositoryIds()[0] ?? null;
91
+ if (options.workspace.taskPaneSelectedRepositoryId === null) {
92
+ options.workspace.taskPaneSelectedRepositoryId = options.activeRepositoryIds()[0] ?? null;
86
93
  }
87
- this.options.workspace.taskRepositoryDropdownOpen = false;
88
- this.syncTaskPaneSelectionFocus();
89
- this.syncTaskPaneSelection();
94
+ options.workspace.taskRepositoryDropdownOpen = false;
95
+ syncTaskPaneSelectionFocus();
96
+ syncTaskPaneSelection();
90
97
  }
91
98
 
92
- focusDraftComposer(): void {
93
- if (this.options.workspace.taskEditorTarget.kind === 'task') {
94
- this.options.flushTaskComposerPersist(this.options.workspace.taskEditorTarget.taskId);
99
+ function focusDraftComposer(): void {
100
+ if (options.workspace.taskEditorTarget.kind === 'task') {
101
+ options.flushTaskComposerPersist(options.workspace.taskEditorTarget.taskId);
95
102
  }
96
- this.options.workspace.taskEditorTarget = {
103
+ options.workspace.taskEditorTarget = {
97
104
  kind: 'draft',
98
105
  };
99
- this.options.workspace.taskPaneSelectionFocus = 'task';
100
- this.options.markDirty();
106
+ options.workspace.taskPaneSelectionFocus = 'task';
107
+ options.markDirty();
101
108
  }
102
109
 
103
- focusTaskComposer(taskId: string): void {
104
- if (!this.options.hasTask(taskId)) {
110
+ function focusTaskComposer(taskId: string): void {
111
+ if (!options.hasTask(taskId)) {
105
112
  return;
106
113
  }
107
114
  if (
108
- this.options.workspace.taskEditorTarget.kind === 'task' &&
109
- this.options.workspace.taskEditorTarget.taskId !== taskId
115
+ options.workspace.taskEditorTarget.kind === 'task' &&
116
+ options.workspace.taskEditorTarget.taskId !== taskId
110
117
  ) {
111
- this.options.flushTaskComposerPersist(this.options.workspace.taskEditorTarget.taskId);
118
+ options.flushTaskComposerPersist(options.workspace.taskEditorTarget.taskId);
112
119
  }
113
- this.options.workspace.taskEditorTarget = {
120
+ options.workspace.taskEditorTarget = {
114
121
  kind: 'task',
115
122
  taskId,
116
123
  };
117
- this.options.workspace.taskPaneSelectedTaskId = taskId;
118
- this.options.workspace.taskPaneSelectionFocus = 'task';
119
- this.options.workspace.taskPaneNotice = null;
120
- this.options.markDirty();
124
+ options.workspace.taskPaneSelectedTaskId = taskId;
125
+ options.workspace.taskPaneSelectionFocus = 'task';
126
+ options.workspace.taskPaneNotice = null;
127
+ options.markDirty();
121
128
  }
122
129
 
123
- selectTaskById(taskId: string): void {
124
- const taskRecord = this.options.taskRecordById(taskId);
130
+ function selectTaskById(taskId: string): void {
131
+ const taskRecord = options.taskRecordById(taskId);
125
132
  if (taskRecord === undefined) {
126
133
  return;
127
134
  }
128
- this.options.workspace.taskPaneSelectedTaskId = taskId;
129
- this.options.workspace.taskPaneSelectionFocus = 'task';
130
- if (taskRecord.repositoryId !== null && this.options.hasRepository(taskRecord.repositoryId)) {
131
- this.options.workspace.taskPaneSelectedRepositoryId = taskRecord.repositoryId;
135
+ options.workspace.taskPaneSelectedTaskId = taskId;
136
+ options.workspace.taskPaneSelectionFocus = 'task';
137
+ if (taskRecord.repositoryId !== null && options.hasRepository(taskRecord.repositoryId)) {
138
+ options.workspace.taskPaneSelectedRepositoryId = taskRecord.repositoryId;
132
139
  }
133
- this.focusTaskComposer(taskId);
140
+ focusTaskComposer(taskId);
134
141
  }
135
142
 
136
- selectRepositoryById(repositoryId: string): void {
137
- if (!this.options.hasRepository(repositoryId)) {
143
+ function selectRepositoryById(repositoryId: string): void {
144
+ if (!options.hasRepository(repositoryId)) {
138
145
  return;
139
146
  }
140
- if (this.options.workspace.taskEditorTarget.kind === 'task') {
141
- this.options.flushTaskComposerPersist(this.options.workspace.taskEditorTarget.taskId);
147
+ if (options.workspace.taskEditorTarget.kind === 'task') {
148
+ options.flushTaskComposerPersist(options.workspace.taskEditorTarget.taskId);
142
149
  }
143
- this.options.workspace.taskPaneSelectedRepositoryId = repositoryId;
144
- this.options.workspace.taskRepositoryDropdownOpen = false;
145
- this.options.workspace.taskPaneSelectionFocus = 'repository';
146
- this.options.workspace.taskEditorTarget = {
150
+ options.workspace.taskPaneSelectedRepositoryId = repositoryId;
151
+ options.workspace.taskRepositoryDropdownOpen = false;
152
+ options.workspace.taskPaneSelectionFocus = 'repository';
153
+ options.workspace.taskEditorTarget = {
147
154
  kind: 'draft',
148
155
  };
149
- this.syncTaskPaneSelection();
150
- this.options.workspace.taskPaneNotice = null;
151
- this.options.markDirty();
156
+ syncTaskPaneSelection();
157
+ options.workspace.taskPaneNotice = null;
158
+ options.markDirty();
152
159
  }
160
+
161
+ return {
162
+ syncTaskPaneSelectionFocus,
163
+ syncTaskPaneSelection,
164
+ syncTaskPaneRepositorySelection,
165
+ focusDraftComposer,
166
+ focusTaskComposer,
167
+ selectTaskById,
168
+ selectRepositoryById,
169
+ };
153
170
  }
@@ -14,7 +14,7 @@ interface TaskPlanningHydrationServiceControlPlane<
14
14
  listTasks(limit: number): Promise<readonly TTaskRecord[]>;
15
15
  }
16
16
 
17
- interface TaskPlanningHydrationServiceOptions<
17
+ export interface TaskPlanningHydrationServiceOptions<
18
18
  TRepositoryRecord extends RepositoryRecordLike,
19
19
  TTaskRecord extends TaskRecordLike,
20
20
  > {
@@ -32,27 +32,33 @@ interface TaskPlanningHydrationServiceOptions<
32
32
  readonly taskLimit: number;
33
33
  }
34
34
 
35
- export class TaskPlanningHydrationService<
35
+ export interface TaskPlanningHydrationService {
36
+ hydrate(): Promise<void>;
37
+ }
38
+
39
+ export function createTaskPlanningHydrationService<
36
40
  TRepositoryRecord extends RepositoryRecordLike,
37
41
  TTaskRecord extends TaskRecordLike,
38
- > {
39
- constructor(
40
- private readonly options: TaskPlanningHydrationServiceOptions<TRepositoryRecord, TTaskRecord>,
41
- ) {}
42
-
43
- async hydrate(): Promise<void> {
44
- this.options.clearRepositories();
45
- for (const repository of await this.options.controlPlaneService.listRepositories()) {
46
- this.options.setRepository(repository);
42
+ >(
43
+ options: TaskPlanningHydrationServiceOptions<TRepositoryRecord, TTaskRecord>,
44
+ ): TaskPlanningHydrationService {
45
+ async function hydrate(): Promise<void> {
46
+ options.clearRepositories();
47
+ for (const repository of await options.controlPlaneService.listRepositories()) {
48
+ options.setRepository(repository);
47
49
  }
48
- this.options.syncTaskPaneRepositorySelection();
50
+ options.syncTaskPaneRepositorySelection();
49
51
 
50
- this.options.clearTasks();
51
- for (const task of await this.options.controlPlaneService.listTasks(this.options.taskLimit)) {
52
- this.options.setTask(task);
52
+ options.clearTasks();
53
+ for (const task of await options.controlPlaneService.listTasks(options.taskLimit)) {
54
+ options.setTask(task);
53
55
  }
54
- this.options.syncTaskPaneSelection();
55
- this.options.syncTaskPaneRepositorySelection();
56
- this.options.markDirty();
56
+ options.syncTaskPaneSelection();
57
+ options.syncTaskPaneRepositorySelection();
58
+ options.markDirty();
57
59
  }
60
+
61
+ return {
62
+ hydrate,
63
+ };
58
64
  }
@@ -1,5 +1,3 @@
1
- import type { StreamObservedEvent } from '../control-plane/stream-protocol.ts';
2
-
3
1
  interface RepositoryRecordLike {
4
2
  readonly repositoryId: string;
5
3
  readonly archivedAt: string | null;
@@ -9,13 +7,29 @@ interface TaskRecordLike {
9
7
  readonly taskId: string;
10
8
  }
11
9
 
12
- interface TaskPlanningObservedEventsOptions<
10
+ interface TaskPlanningSyncedProjectionState<
11
+ TRepositoryRecord extends RepositoryRecordLike,
12
+ TTaskRecord extends TaskRecordLike,
13
+ > {
14
+ readonly repositoriesById: Readonly<Record<string, TRepositoryRecord>>;
15
+ readonly tasksById: Readonly<Record<string, TTaskRecord>>;
16
+ }
17
+
18
+ interface TaskPlanningSyncedProjectionInput<
19
+ TRepositoryRecord extends RepositoryRecordLike,
20
+ TTaskRecord extends TaskRecordLike,
21
+ > {
22
+ readonly changed: boolean;
23
+ readonly state: TaskPlanningSyncedProjectionState<TRepositoryRecord, TTaskRecord>;
24
+ readonly removedTaskIds: readonly string[];
25
+ readonly upsertedRepositoryIds: readonly string[];
26
+ readonly upsertedTaskIds: readonly string[];
27
+ }
28
+
29
+ interface TaskPlanningSyncedProjectionOptions<
13
30
  TRepositoryRecord extends RepositoryRecordLike,
14
31
  TTaskRecord extends TaskRecordLike,
15
32
  > {
16
- readonly parseRepositoryRecord: (value: unknown) => TRepositoryRecord | null;
17
- readonly parseTaskRecord: (value: unknown) => TTaskRecord | null;
18
- readonly getRepository: (repositoryId: string) => TRepositoryRecord | undefined;
19
33
  readonly setRepository: (repositoryId: string, repository: TRepositoryRecord) => void;
20
34
  readonly setTask: (task: TTaskRecord) => void;
21
35
  readonly deleteTask: (taskId: string) => boolean;
@@ -24,66 +38,50 @@ interface TaskPlanningObservedEventsOptions<
24
38
  readonly markDirty: () => void;
25
39
  }
26
40
 
27
- export class TaskPlanningObservedEvents<
41
+ export class TaskPlanningSyncedProjection<
28
42
  TRepositoryRecord extends RepositoryRecordLike,
29
43
  TTaskRecord extends TaskRecordLike,
30
44
  > {
31
45
  constructor(
32
- private readonly options: TaskPlanningObservedEventsOptions<TRepositoryRecord, TTaskRecord>,
46
+ private readonly options: TaskPlanningSyncedProjectionOptions<TRepositoryRecord, TTaskRecord>,
33
47
  ) {}
34
48
 
35
- apply(observed: StreamObservedEvent): void {
36
- if (observed.type === 'repository-upserted' || observed.type === 'repository-updated') {
37
- const repository = this.options.parseRepositoryRecord(observed.repository);
38
- if (repository !== null) {
39
- this.options.setRepository(repository.repositoryId, repository);
40
- this.options.syncTaskPaneRepositorySelection();
41
- this.options.markDirty();
42
- }
49
+ apply(reduction: TaskPlanningSyncedProjectionInput<TRepositoryRecord, TTaskRecord>): void {
50
+ if (!reduction.changed) {
43
51
  return;
44
52
  }
45
- if (observed.type === 'repository-archived') {
46
- const repository = this.options.getRepository(observed.repositoryId);
47
- if (repository !== undefined) {
48
- this.options.setRepository(observed.repositoryId, {
49
- ...repository,
50
- archivedAt: observed.ts,
51
- });
52
- this.options.syncTaskPaneRepositorySelection();
53
- this.options.markDirty();
53
+
54
+ let repositoriesChanged = false;
55
+ for (const repositoryId of reduction.upsertedRepositoryIds) {
56
+ const repository = reduction.state.repositoriesById[repositoryId];
57
+ if (repository === undefined) {
58
+ continue;
54
59
  }
55
- return;
60
+ this.options.setRepository(repositoryId, repository);
61
+ repositoriesChanged = true;
56
62
  }
57
- if (observed.type === 'task-created' || observed.type === 'task-updated') {
58
- const task = this.options.parseTaskRecord(observed.task);
59
- if (task !== null) {
60
- this.options.setTask(task);
61
- this.options.syncTaskPaneSelection();
62
- this.options.markDirty();
63
- }
64
- return;
63
+ if (repositoriesChanged) {
64
+ this.options.syncTaskPaneRepositorySelection();
65
+ this.options.markDirty();
65
66
  }
66
- if (observed.type === 'task-deleted') {
67
- if (this.options.deleteTask(observed.taskId)) {
68
- this.options.syncTaskPaneSelection();
69
- this.options.markDirty();
67
+
68
+ let tasksChanged = false;
69
+ for (const taskId of reduction.removedTaskIds) {
70
+ if (this.options.deleteTask(taskId)) {
71
+ tasksChanged = true;
70
72
  }
71
- return;
72
73
  }
73
- if (observed.type === 'task-reordered') {
74
- let changed = false;
75
- for (const value of observed.tasks) {
76
- const task = this.options.parseTaskRecord(value);
77
- if (task === null) {
78
- continue;
79
- }
80
- this.options.setTask(task);
81
- changed = true;
82
- }
83
- if (changed) {
84
- this.options.syncTaskPaneSelection();
85
- this.options.markDirty();
74
+ for (const taskId of reduction.upsertedTaskIds) {
75
+ const task = reduction.state.tasksById[taskId];
76
+ if (task === undefined) {
77
+ continue;
86
78
  }
79
+ this.options.setTask(task);
80
+ tasksChanged = true;
81
+ }
82
+ if (tasksChanged) {
83
+ this.options.syncTaskPaneSelection();
84
+ this.options.markDirty();
87
85
  }
88
86
  }
89
87
  }
@@ -1,5 +1,3 @@
1
- import type { StreamObservedEvent } from '../control-plane/stream-protocol.ts';
2
-
3
1
  interface DirectoryRecordLike {
4
2
  readonly directoryId: string;
5
3
  }
@@ -15,99 +13,104 @@ interface WorkspaceObservedApplyResult {
15
13
  readonly removedDirectoryIds: readonly string[];
16
14
  }
17
15
 
18
- interface WorkspaceObservedEventsOptions<
16
+ interface WorkspaceSyncedProjectionState<
17
+ TDirectoryRecord extends DirectoryRecordLike,
18
+ TConversationRecord extends ConversationRecordLike,
19
+ > {
20
+ readonly directoriesById: Readonly<Record<string, TDirectoryRecord>>;
21
+ readonly conversationsById: Readonly<Record<string, TConversationRecord>>;
22
+ }
23
+
24
+ interface WorkspaceSyncedProjectionInput<
25
+ TDirectoryRecord extends DirectoryRecordLike,
26
+ TConversationRecord extends ConversationRecordLike,
27
+ > {
28
+ readonly changed: boolean;
29
+ readonly state: WorkspaceSyncedProjectionState<TDirectoryRecord, TConversationRecord>;
30
+ readonly removedConversationIds: readonly string[];
31
+ readonly removedDirectoryIds: readonly string[];
32
+ readonly upsertedDirectoryIds: readonly string[];
33
+ readonly upsertedConversationIds: readonly string[];
34
+ }
35
+
36
+ interface WorkspaceSyncedProjectionOptions<
19
37
  TDirectoryRecord extends DirectoryRecordLike,
20
38
  TConversationRecord extends ConversationRecordLike,
21
39
  > {
22
- readonly parseDirectoryRecord: (value: unknown) => TDirectoryRecord | null;
23
- readonly parseConversationRecord: (value: unknown) => TConversationRecord | null;
24
40
  readonly setDirectory: (directoryId: string, directory: TDirectoryRecord) => void;
25
41
  readonly deleteDirectory: (directoryId: string) => boolean;
26
42
  readonly deleteDirectoryGitState: (directoryId: string) => void;
27
43
  readonly syncGitStateWithDirectories: () => void;
28
44
  readonly upsertConversationFromPersistedRecord: (record: TConversationRecord) => void;
29
45
  readonly removeConversation: (sessionId: string) => boolean;
30
- readonly orderedConversationIds: () => readonly string[];
31
- readonly conversationDirectoryId: (sessionId: string) => string | null;
32
46
  }
33
47
 
34
- export class WorkspaceObservedEvents<
48
+ export class WorkspaceSyncedProjection<
35
49
  TDirectoryRecord extends DirectoryRecordLike,
36
50
  TConversationRecord extends ConversationRecordLike,
37
51
  > {
38
52
  constructor(
39
- private readonly options: WorkspaceObservedEventsOptions<TDirectoryRecord, TConversationRecord>,
53
+ private readonly options: WorkspaceSyncedProjectionOptions<
54
+ TDirectoryRecord,
55
+ TConversationRecord
56
+ >,
40
57
  ) {}
41
58
 
42
- apply(observed: StreamObservedEvent): WorkspaceObservedApplyResult {
43
- if (observed.type === 'directory-upserted') {
44
- const directory = this.options.parseDirectoryRecord(observed.directory);
45
- if (directory === null) {
46
- return {
47
- changed: false,
48
- removedConversationIds: [],
49
- removedDirectoryIds: [],
50
- };
51
- }
52
- this.options.setDirectory(directory.directoryId, directory);
53
- this.options.syncGitStateWithDirectories();
59
+ apply(
60
+ reduction: WorkspaceSyncedProjectionInput<TDirectoryRecord, TConversationRecord>,
61
+ ): WorkspaceObservedApplyResult {
62
+ if (!reduction.changed) {
54
63
  return {
55
- changed: true,
64
+ changed: false,
56
65
  removedConversationIds: [],
57
66
  removedDirectoryIds: [],
58
67
  };
59
68
  }
60
69
 
61
- if (observed.type === 'directory-archived') {
62
- const removedConversationIds: string[] = [];
63
- for (const sessionId of this.options.orderedConversationIds()) {
64
- if (this.options.conversationDirectoryId(sessionId) !== observed.directoryId) {
65
- continue;
66
- }
67
- if (this.options.removeConversation(sessionId)) {
68
- removedConversationIds.push(sessionId);
69
- }
70
+ let changed = false;
71
+ const removedConversationIds: string[] = [];
72
+ const removedDirectoryIds: string[] = [];
73
+
74
+ for (const directoryId of reduction.upsertedDirectoryIds) {
75
+ const directory = reduction.state.directoriesById[directoryId];
76
+ if (directory === undefined) {
77
+ continue;
70
78
  }
71
- const removedDirectory = this.options.deleteDirectory(observed.directoryId);
72
- this.options.deleteDirectoryGitState(observed.directoryId);
73
- this.options.syncGitStateWithDirectories();
74
- return {
75
- changed: removedDirectory || removedConversationIds.length > 0,
76
- removedConversationIds,
77
- removedDirectoryIds: removedDirectory ? [observed.directoryId] : [],
78
- };
79
+ this.options.setDirectory(directoryId, directory);
80
+ changed = true;
79
81
  }
80
-
81
- if (observed.type === 'conversation-created' || observed.type === 'conversation-updated') {
82
- const conversation = this.options.parseConversationRecord(observed.conversation);
83
- if (conversation === null) {
84
- return {
85
- changed: false,
86
- removedConversationIds: [],
87
- removedDirectoryIds: [],
88
- };
82
+ for (const conversationId of reduction.upsertedConversationIds) {
83
+ const conversation = reduction.state.conversationsById[conversationId];
84
+ if (conversation === undefined) {
85
+ continue;
89
86
  }
90
87
  this.options.upsertConversationFromPersistedRecord(conversation);
91
- return {
92
- changed: true,
93
- removedConversationIds: [],
94
- removedDirectoryIds: [],
95
- };
88
+ changed = true;
96
89
  }
97
90
 
98
- if (observed.type === 'conversation-archived' || observed.type === 'conversation-deleted') {
99
- const removed = this.options.removeConversation(observed.conversationId);
100
- return {
101
- changed: removed,
102
- removedConversationIds: removed ? [observed.conversationId] : [],
103
- removedDirectoryIds: [],
104
- };
91
+ for (const sessionId of reduction.removedConversationIds) {
92
+ if (!this.options.removeConversation(sessionId)) {
93
+ continue;
94
+ }
95
+ removedConversationIds.push(sessionId);
96
+ changed = true;
97
+ }
98
+
99
+ for (const directoryId of reduction.removedDirectoryIds) {
100
+ if (this.options.deleteDirectory(directoryId)) {
101
+ removedDirectoryIds.push(directoryId);
102
+ changed = true;
103
+ }
104
+ this.options.deleteDirectoryGitState(directoryId);
105
+ }
106
+ if (reduction.upsertedDirectoryIds.length > 0 || reduction.removedDirectoryIds.length > 0) {
107
+ this.options.syncGitStateWithDirectories();
105
108
  }
106
109
 
107
110
  return {
108
- changed: false,
109
- removedConversationIds: [],
110
- removedDirectoryIds: [],
111
+ changed,
112
+ removedConversationIds,
113
+ removedDirectoryIds,
111
114
  };
112
115
  }
113
116
  }