kward 0.70.0 → 0.72.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 (143) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/pages.yml +1 -1
  3. data/CHANGELOG.md +89 -3
  4. data/Gemfile +2 -0
  5. data/Gemfile.lock +90 -2
  6. data/README.md +34 -6
  7. data/Rakefile +96 -0
  8. data/doc/agent-tools.md +52 -0
  9. data/doc/api.md +92 -0
  10. data/doc/authentication.md +58 -23
  11. data/doc/code-search.md +42 -2
  12. data/doc/configuration.md +102 -13
  13. data/doc/context-budgeting.md +136 -0
  14. data/doc/context-tools.md +83 -0
  15. data/doc/editor.md +394 -0
  16. data/doc/extensibility.md +16 -7
  17. data/doc/files.md +100 -0
  18. data/doc/getting-started.md +25 -18
  19. data/doc/git.md +122 -0
  20. data/doc/memory.md +24 -4
  21. data/doc/personas.md +34 -5
  22. data/doc/plugins.md +74 -3
  23. data/doc/releasing.md +45 -8
  24. data/doc/rpc.md +77 -15
  25. data/doc/session-management.md +254 -0
  26. data/doc/shell.md +286 -0
  27. data/doc/tabs.md +122 -0
  28. data/doc/troubleshooting.md +77 -1
  29. data/doc/usage.md +60 -15
  30. data/doc/web-search.md +12 -4
  31. data/doc/workspace-tools.md +144 -0
  32. data/examples/plugins/space_invaders.rb +377 -0
  33. data/lib/kward/agent.rb +1 -1
  34. data/lib/kward/cli/commands.rb +41 -2
  35. data/lib/kward/cli/git.rb +150 -0
  36. data/lib/kward/cli/interactive_turn.rb +73 -9
  37. data/lib/kward/cli/openrouter_commands.rb +55 -0
  38. data/lib/kward/cli/plugins.rb +54 -4
  39. data/lib/kward/cli/prompt_interface.rb +111 -6
  40. data/lib/kward/cli/rendering.rb +11 -6
  41. data/lib/kward/cli/runtime_helpers.rb +133 -3
  42. data/lib/kward/cli/sessions.rb +262 -13
  43. data/lib/kward/cli/settings.rb +216 -37
  44. data/lib/kward/cli/slash_commands.rb +439 -8
  45. data/lib/kward/cli/tabs.rb +695 -0
  46. data/lib/kward/cli.rb +171 -26
  47. data/lib/kward/compactor.rb +4 -1
  48. data/lib/kward/config_files.rb +125 -5
  49. data/lib/kward/context_budget_meter.rb +44 -0
  50. data/lib/kward/conversation.rb +59 -22
  51. data/lib/kward/editor_mode.rb +25 -0
  52. data/lib/kward/ekwsh.rb +362 -0
  53. data/lib/kward/model/client.rb +37 -50
  54. data/lib/kward/model/context_usage.rb +13 -6
  55. data/lib/kward/model/model_info.rb +92 -16
  56. data/lib/kward/model/payloads.rb +2 -0
  57. data/lib/kward/openrouter_model_cache.rb +120 -0
  58. data/lib/kward/plugin_registry.rb +108 -1
  59. data/lib/kward/project_files.rb +52 -0
  60. data/lib/kward/prompt_history.rb +82 -0
  61. data/lib/kward/prompt_interface/banner.rb +16 -51
  62. data/lib/kward/prompt_interface/composer_controller.rb +124 -83
  63. data/lib/kward/prompt_interface/composer_renderer.rb +116 -14
  64. data/lib/kward/prompt_interface/composer_state.rb +96 -27
  65. data/lib/kward/prompt_interface/editor/auto_close_pairs.rb +123 -0
  66. data/lib/kward/prompt_interface/editor/auto_indent.rb +509 -0
  67. data/lib/kward/prompt_interface/editor/buffer.rb +109 -0
  68. data/lib/kward/prompt_interface/editor/controller.rb +1018 -0
  69. data/lib/kward/prompt_interface/editor/endwise.rb +321 -0
  70. data/lib/kward/prompt_interface/editor/file_marker.rb +40 -0
  71. data/lib/kward/prompt_interface/editor/indent_navigation.rb +61 -0
  72. data/lib/kward/prompt_interface/editor/kill_ring.rb +78 -0
  73. data/lib/kward/prompt_interface/editor/modes/emacs.rb +259 -0
  74. data/lib/kward/prompt_interface/editor/modes/modern.rb +353 -0
  75. data/lib/kward/prompt_interface/editor/modes/vibe.rb +1962 -0
  76. data/lib/kward/prompt_interface/editor/renderer.rb +243 -0
  77. data/lib/kward/prompt_interface/editor/search.rb +76 -0
  78. data/lib/kward/prompt_interface/editor/selections.rb +120 -0
  79. data/lib/kward/prompt_interface/editor/state.rb +1249 -0
  80. data/lib/kward/prompt_interface/editor/status_text.rb +23 -0
  81. data/lib/kward/prompt_interface/editor/syntax_highlighter.rb +420 -0
  82. data/lib/kward/prompt_interface/editor/undo_history.rb +46 -0
  83. data/lib/kward/prompt_interface/editor/vibe_state.rb +44 -0
  84. data/lib/kward/prompt_interface/file_overlay.rb +211 -0
  85. data/lib/kward/prompt_interface/git_prompt.rb +299 -0
  86. data/lib/kward/prompt_interface/interactive/controller.rb +186 -0
  87. data/lib/kward/prompt_interface/interactive/renderer.rb +71 -0
  88. data/lib/kward/prompt_interface/interactive/state.rb +62 -0
  89. data/lib/kward/prompt_interface/key_handler.rb +416 -43
  90. data/lib/kward/prompt_interface/layout.rb +2 -2
  91. data/lib/kward/prompt_interface/overlay_renderer.rb +21 -2
  92. data/lib/kward/prompt_interface/project_browser.rb +524 -0
  93. data/lib/kward/prompt_interface/prompt_renderer.rb +32 -13
  94. data/lib/kward/prompt_interface/question_prompt.rb +122 -82
  95. data/lib/kward/prompt_interface/runtime_state.rb +49 -1
  96. data/lib/kward/prompt_interface/screen.rb +17 -0
  97. data/lib/kward/prompt_interface/selection_prompt.rb +511 -58
  98. data/lib/kward/prompt_interface/stream_state.rb +7 -0
  99. data/lib/kward/prompt_interface/transcript_buffer.rb +13 -16
  100. data/lib/kward/prompt_interface/transcript_renderer.rb +3 -3
  101. data/lib/kward/prompt_interface.rb +307 -35
  102. data/lib/kward/prompts/commands.rb +7 -1
  103. data/lib/kward/prompts.rb +4 -2
  104. data/lib/kward/rpc/server.rb +45 -11
  105. data/lib/kward/rpc/session_manager.rb +52 -53
  106. data/lib/kward/rpc/session_tree_rows.rb +9 -115
  107. data/lib/kward/rpc/tool_event_normalizer.rb +1 -1
  108. data/lib/kward/session_store.rb +67 -4
  109. data/lib/kward/session_tree_nodes.rb +136 -0
  110. data/lib/kward/session_tree_renderer.rb +9 -131
  111. data/lib/kward/tab_store.rb +47 -0
  112. data/lib/kward/telemetry/logger.rb +5 -3
  113. data/lib/kward/text_boundary.rb +25 -0
  114. data/lib/kward/tool_output_compactor.rb +127 -0
  115. data/lib/kward/tools/base.rb +8 -2
  116. data/lib/kward/tools/context_budget_stats.rb +54 -0
  117. data/lib/kward/tools/context_for_task.rb +202 -0
  118. data/lib/kward/tools/read_file.rb +8 -4
  119. data/lib/kward/tools/registry.rb +92 -15
  120. data/lib/kward/tools/retrieve_tool_output.rb +71 -0
  121. data/lib/kward/tools/search/web.rb +2 -2
  122. data/lib/kward/tools/summarize_file_structure.rb +29 -0
  123. data/lib/kward/tools/tool_call.rb +12 -0
  124. data/lib/kward/version.rb +1 -1
  125. data/lib/kward/workers/git_guard.rb +68 -0
  126. data/lib/kward/workers/live_view.rb +49 -0
  127. data/lib/kward/workers/manager.rb +288 -0
  128. data/lib/kward/workers/store.rb +72 -0
  129. data/lib/kward/workers/tool_policy.rb +23 -0
  130. data/lib/kward/workers/worker.rb +82 -0
  131. data/lib/kward/workers/write_lock.rb +38 -0
  132. data/lib/kward/workers.rb +7 -0
  133. data/lib/kward/workspace.rb +154 -12
  134. data/templates/default/fulldoc/html/css/kward.css +362 -42
  135. data/templates/default/fulldoc/html/full_list.erb +107 -0
  136. data/templates/default/fulldoc/html/js/kward.js +161 -2
  137. data/templates/default/fulldoc/html/setup.rb +8 -0
  138. data/templates/default/kward_navigation.rb +102 -0
  139. data/templates/default/layout/html/layout.erb +43 -10
  140. data/templates/default/layout/html/setup.rb +39 -38
  141. metadata +65 -3
  142. data/lib/kward/resources/avatar_kward_logo.rb +0 -50
  143. data/lib/kward/resources/pixel_logo.rb +0 -232
@@ -0,0 +1,288 @@
1
+ require "timeout"
2
+ require_relative "../agent"
3
+ require_relative "../cancellation"
4
+ require_relative "../conversation"
5
+ require_relative "../model/client"
6
+ require_relative "../session_store"
7
+ require_relative "../tools/registry"
8
+ require_relative "../workspace"
9
+ require_relative "git_guard"
10
+ require_relative "tool_policy"
11
+ require_relative "worker"
12
+
13
+ module Kward
14
+ module Workers
15
+ # Coordinates background worker execution and role-specific tool policy.
16
+ class Manager
17
+ DEFAULT_TIMEOUT_SECONDS = 180
18
+
19
+ def initialize(client_factory: -> { Client.new }, prompt: nil, workspace_root: Dir.pwd, timeout_seconds: DEFAULT_TIMEOUT_SECONDS, on_status_change: nil, session_store: nil, provider: nil, model: nil, reasoning_effort: nil, write_lock: nil, worker_store: nil, git_guard: nil, write_lane_available: -> { true })
20
+ @client_factory = client_factory
21
+ @prompt = prompt
22
+ @workspace_root = ConfigFiles.canonical_workspace_root(workspace_root)
23
+ @timeout_seconds = timeout_seconds
24
+ @on_status_change = on_status_change
25
+ @session_store = session_store
26
+ @provider = provider
27
+ @model = model
28
+ @reasoning_effort = reasoning_effort
29
+ @write_lock = write_lock
30
+ @worker_store = worker_store
31
+ @git_guard = git_guard || GitGuard.new(root: @workspace_root)
32
+ @write_lane_available = write_lane_available
33
+ @workers = {}
34
+ @mutex = Mutex.new
35
+ end
36
+
37
+ def start(role:, prompt:, title: nil, id: nil)
38
+ worker = build_worker(role: role, prompt: prompt, title: title, id: id)
39
+ enqueue(worker)
40
+ end
41
+
42
+ def continue(id, role:, prompt:, title: nil)
43
+ archived = nil
44
+ worker = build_worker(role: role, prompt: prompt, title: title, id: id)
45
+ @mutex.synchronize do
46
+ archived = @workers.delete(id.to_s)
47
+ @workers[worker.id] = worker
48
+ end
49
+ archived&.update_status("archived")
50
+ @worker_store&.archive(id) if archived || @worker_store&.find(id)
51
+ enqueue(worker, store: false)
52
+ end
53
+
54
+ def list
55
+ @mutex.synchronize { @workers.values.reject { |worker| worker.status == "archived" }.sort_by(&:created_at) }
56
+ end
57
+
58
+ def find(id)
59
+ @mutex.synchronize { @workers[id.to_s] }
60
+ end
61
+
62
+ def cancel(id)
63
+ worker = find(id) || raise(ArgumentError, "Unknown worker: #{id}")
64
+ worker.cancellation.cancel!
65
+ worker.thread.raise(Cancellation::CancelledError, "cancelled") if worker.thread&.alive?
66
+ update_status(worker, "cancelled")
67
+ end
68
+
69
+ def archive(id)
70
+ worker = find(id) || raise(ArgumentError, "Unknown worker: #{id}")
71
+ worker.cancellation.cancel! if %w[queued running].include?(worker.status)
72
+ worker.thread.raise(Cancellation::CancelledError, "cancelled") if worker.thread&.alive?
73
+ update_status(worker, "archived")
74
+ end
75
+
76
+ private
77
+
78
+ def build_worker(role:, prompt:, title: nil, id: nil)
79
+ Worker.new(
80
+ id: id || SecureRandom.hex(4),
81
+ title: title || title_for(prompt),
82
+ role: role,
83
+ prompt: prompt,
84
+ workspace_root: @workspace_root,
85
+ status: "queued"
86
+ )
87
+ end
88
+
89
+ def enqueue(worker, store: true)
90
+ @mutex.synchronize { @workers[worker.id] = worker }
91
+ @worker_store&.upsert(worker) if store
92
+ worker.thread = Thread.new { run_worker(worker) }
93
+ worker.thread.report_on_exception = false
94
+ worker
95
+ end
96
+
97
+ def run_worker(worker)
98
+ conversation = Conversation.new(
99
+ system_message: { role: "system", content: system_message(worker) },
100
+ workspace_root: worker.workspace_root,
101
+ provider: @provider,
102
+ model: @model,
103
+ reasoning_effort: @reasoning_effort
104
+ )
105
+ worker.conversation = conversation
106
+ attach_session(worker, conversation)
107
+ writer_id = wait_for_worker_writer(worker)
108
+ update_status(worker, "running")
109
+ registry = ToolRegistry.new(
110
+ workspace: Workspace.new(root: worker.workspace_root),
111
+ prompt: @prompt,
112
+ allowed_tool_names: ToolPolicy.allowed_tool_names(worker.role),
113
+ write_lock: @write_lock,
114
+ writer_id: writer_id
115
+ )
116
+ agent = Agent.new(client: @client_factory.call, tool_registry: registry, conversation: conversation)
117
+ report = Timeout.timeout(@timeout_seconds, WorkerTimeoutError) do
118
+ agent.ask(worker_prompt(worker), cancellation: worker.cancellation) do |event|
119
+ worker.record_event(event)
120
+ end
121
+ end
122
+ report = finalize_write_worker(worker, report)
123
+ update_status(worker, "ready", report: report, error: "")
124
+ rescue WorkerTimeoutError
125
+ update_status(worker, "failed", error: "Worker timed out after #{@timeout_seconds} seconds")
126
+ rescue Cancellation::CancelledError
127
+ update_status(worker, "cancelled")
128
+ rescue StandardError => e
129
+ update_status(worker, "failed", error: e.message)
130
+ ensure
131
+ release_worker_writer(worker)
132
+ end
133
+
134
+ def update_status(worker, status, **values)
135
+ return worker if worker.status == "archived" && status.to_s != "archived"
136
+
137
+ worker.update_status(status, **values)
138
+ @worker_store&.upsert(worker)
139
+ @on_status_change&.call(worker)
140
+ worker
141
+ end
142
+
143
+ def wait_for_worker_writer(worker)
144
+ return nil unless ToolPolicy.write_capable?(worker.role)
145
+
146
+ loop do
147
+ worker.cancellation.raise_if_cancelled!
148
+ wait_for_write_lane_available(worker)
149
+ wait_for_clean_workspace(worker)
150
+ return worker.id unless @write_lock
151
+
152
+ release_foreground_writer_if_clean
153
+ return worker.id if @write_lock.acquire(worker.id)
154
+
155
+ sleep 0.1
156
+ end
157
+ end
158
+
159
+ def wait_for_write_lane_available(worker)
160
+ until @write_lane_available.call
161
+ worker.cancellation.raise_if_cancelled!
162
+ sleep 0.1
163
+ end
164
+ end
165
+
166
+ def wait_for_clean_workspace(worker)
167
+ return unless @git_guard.repository?
168
+
169
+ until @git_guard.clean?
170
+ worker.cancellation.raise_if_cancelled!
171
+ sleep 0.5
172
+ end
173
+ end
174
+
175
+ def release_foreground_writer_if_clean
176
+ return unless @write_lock&.owned_by?("implementation")
177
+ return unless @git_guard.repository?
178
+ return unless @git_guard.clean?
179
+
180
+ @write_lock.release("implementation")
181
+ end
182
+
183
+ def finalize_write_worker(worker, report)
184
+ return report unless ToolPolicy.write_capable?(worker.role)
185
+ return report unless @git_guard.repository?
186
+ return report if @git_guard.clean?
187
+
188
+ commit = @git_guard.commit_all(commit_message(worker))
189
+ unless commit.success?
190
+ raise "Worker changed files but commit failed: #{commit.output}"
191
+ end
192
+
193
+ [report, "", "Committed workspace changes: #{commit.commit}"].join("\n")
194
+ end
195
+
196
+ def commit_message(worker)
197
+ "Kward worker #{worker.id}: #{worker.title}"
198
+ end
199
+
200
+ def release_worker_writer(worker)
201
+ return unless ToolPolicy.write_capable?(worker.role)
202
+
203
+ @write_lock&.release(worker.id)
204
+ end
205
+
206
+ def attach_session(worker, conversation)
207
+ return unless @session_store
208
+
209
+ session = @session_store.create(provider: @provider, model: @model, reasoning_effort: @reasoning_effort)
210
+ session.rename("#{worker.role}: #{worker.title}")
211
+ session.attach(conversation)
212
+ worker.session = session
213
+ @worker_store&.upsert(worker)
214
+ @on_status_change&.call(worker)
215
+ rescue StandardError
216
+ nil
217
+ end
218
+
219
+ def worker_prompt(worker)
220
+ return request_prompt(worker.prompt) if worker.role == "request"
221
+
222
+ worker.prompt
223
+ end
224
+
225
+ def system_message(worker)
226
+ return request_system_message if worker.role == "request"
227
+
228
+ "You are a Kward worker. Complete the user's task carefully."
229
+ end
230
+
231
+ def request_system_message
232
+ <<~SYSTEM
233
+ You are a Kward request worker running the read-only exploration phase.
234
+ Inspect the workspace, map relevant terrain, and produce a practical review for the user.
235
+ Do not edit files, write files, delete files, alter configuration, or claim implementation work was done.
236
+ SYSTEM
237
+ end
238
+
239
+ def request_prompt(prompt)
240
+ <<~PROMPT
241
+ #{prompt}
242
+
243
+ ---
244
+
245
+ You are handling this as a structured Kward background request.
246
+ First perform a read-only exploration phase. Inspect relevant files and documentation, reason about the request, and prepare a reviewable result for the user.
247
+ Do not modify files, write files, delete files, alter configuration, run destructive commands, or claim implementation work was done.
248
+
249
+ Return a concise request review with these sections:
250
+ # Request Review: <title>
251
+
252
+ ## Request
253
+ Restate the user's request.
254
+
255
+ ## Summary
256
+ The short answer.
257
+
258
+ ## Relevant files
259
+ Bullet list of likely files and why they matter.
260
+
261
+ ## Findings
262
+ What you discovered.
263
+
264
+ ## Recommended next step
265
+ A practical next step. If implementation appears useful, say so clearly, but do not implement it.
266
+
267
+ ## Risks
268
+ Important risks or unknowns.
269
+
270
+ ## Verification
271
+ Focused verification commands or checks.
272
+
273
+ ## Open questions
274
+ Decisions the user should make before proceeding.
275
+
276
+ End by asking: Should we proceed?
277
+ PROMPT
278
+ end
279
+
280
+ def title_for(prompt)
281
+ text = prompt.to_s.strip.gsub(/\s+/, " ")
282
+ text.empty? ? "Untitled worker" : text[0, 80]
283
+ end
284
+
285
+ class WorkerTimeoutError < StandardError; end
286
+ end
287
+ end
288
+ end
@@ -0,0 +1,72 @@
1
+ require "fileutils"
2
+ require "json"
3
+ require_relative "../config_files"
4
+
5
+ module Kward
6
+ module Workers
7
+ # JSON-backed metadata store for worker records.
8
+ class Store
9
+ def initialize(path: File.join(ConfigFiles.config_dir, "workers.json"))
10
+ @path = path
11
+ @mutex = Mutex.new
12
+ end
13
+
14
+ attr_reader :path
15
+
16
+ def upsert(worker)
17
+ record = worker.respond_to?(:to_h) ? worker.to_h : worker.to_h
18
+ update_records do |records|
19
+ index = records.index { |item| item["id"] == record["id"] }
20
+ index ? records[index] = record : records << record
21
+ end
22
+ record
23
+ end
24
+
25
+ def list(include_archived: false)
26
+ records = read_records
27
+ records = records.reject { |record| record["status"] == "archived" } unless include_archived
28
+ records.sort_by { |record| record["created_at"].to_s }
29
+ end
30
+
31
+ def find(id)
32
+ read_records.find { |record| record["id"] == id.to_s }
33
+ end
34
+
35
+ def archive(id)
36
+ record = nil
37
+ update_records do |records|
38
+ index = records.index { |item| item["id"] == id.to_s }
39
+ raise ArgumentError, "Unknown worker: #{id}" unless index
40
+
41
+ record = records[index].merge("status" => "archived")
42
+ records[index] = record
43
+ end
44
+ record
45
+ end
46
+
47
+ private
48
+
49
+ def read_records
50
+ @mutex.synchronize { read_records_unlocked }
51
+ end
52
+
53
+ def read_records_unlocked
54
+ return [] unless File.exist?(@path)
55
+
56
+ data = JSON.parse(File.read(@path))
57
+ data.is_a?(Array) ? data : []
58
+ rescue JSON::ParserError
59
+ raise "Invalid worker store JSON: #{@path}"
60
+ end
61
+
62
+ def update_records
63
+ @mutex.synchronize do
64
+ records = read_records_unlocked
65
+ yield records
66
+ FileUtils.mkdir_p(File.dirname(@path))
67
+ File.write(@path, JSON.pretty_generate(records) + "\n")
68
+ end
69
+ end
70
+ end
71
+ end
72
+ end
@@ -0,0 +1,23 @@
1
+ module Kward
2
+ module Workers
3
+ # Tool allowlists for worker roles.
4
+ module ToolPolicy
5
+ READ_ONLY_TOOLS = %w[list_directory read_file code_search summarize_file_structure retrieve_tool_output web_search fetch_content fetch_raw read_skill].freeze
6
+
7
+ module_function
8
+
9
+ def allowed_tool_names(role)
10
+ case role.to_s
11
+ when "request", "read_only"
12
+ READ_ONLY_TOOLS
13
+ else
14
+ nil
15
+ end
16
+ end
17
+
18
+ def write_capable?(role)
19
+ allowed_tool_names(role).nil?
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,82 @@
1
+ require "securerandom"
2
+ require "time"
3
+ require_relative "../config_files"
4
+ require_relative "../cancellation"
5
+
6
+ module Kward
7
+ module Workers
8
+ # Runtime record for one independent unit of agent work.
9
+ class Worker
10
+ STATUSES = %w[idle queued running ready failed cancelled archived].freeze
11
+
12
+ def initialize(id: SecureRandom.hex(4), title:, role:, workspace_root: Dir.pwd, status: "idle", prompt: nil, conversation: nil, session: nil, cancellation: Cancellation.new, created_at: Time.now.utc)
13
+ @id = id
14
+ @title = title.to_s
15
+ @role = role.to_s
16
+ @workspace_root = ConfigFiles.canonical_workspace_root(workspace_root)
17
+ @status = status.to_s
18
+ @prompt = prompt.to_s
19
+ @conversation = conversation
20
+ @session = session
21
+ @cancellation = cancellation
22
+ @created_at = created_at
23
+ @updated_at = created_at
24
+ @started_at = nil
25
+ @finished_at = nil
26
+ @report = nil
27
+ @error = nil
28
+ @thread = nil
29
+ @event_history = []
30
+ @event_queue = Queue.new
31
+ end
32
+
33
+ attr_reader :id, :title, :role, :workspace_root, :prompt, :conversation, :session, :cancellation, :created_at, :updated_at, :started_at, :finished_at, :report, :error, :thread, :event_history, :event_queue
34
+ attr_writer :conversation, :session, :thread
35
+
36
+ def status
37
+ @status
38
+ end
39
+
40
+ def update_status(status, error: nil, report: nil)
41
+ @status = status.to_s
42
+ @error = error unless error.nil?
43
+ @report = report unless report.nil?
44
+ now = Time.now.utc
45
+ @updated_at = now
46
+ @started_at ||= now if @status == "running"
47
+ @finished_at = now if %w[ready failed cancelled archived].include?(@status)
48
+ self
49
+ end
50
+
51
+ def record_event(event)
52
+ @event_history << event
53
+ @event_queue << event
54
+ end
55
+
56
+ def to_h
57
+ {
58
+ "id" => id,
59
+ "title" => title,
60
+ "role" => role,
61
+ "status" => status,
62
+ "prompt" => prompt,
63
+ "workspace_root" => workspace_root,
64
+ "session_id" => session&.id,
65
+ "session_path" => session&.path,
66
+ "created_at" => timestamp(created_at),
67
+ "updated_at" => timestamp(updated_at),
68
+ "started_at" => timestamp(started_at),
69
+ "finished_at" => timestamp(finished_at),
70
+ "report" => report,
71
+ "error" => error
72
+ }
73
+ end
74
+
75
+ private
76
+
77
+ def timestamp(value)
78
+ value&.utc&.iso8601(3)
79
+ end
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,38 @@
1
+ require "thread"
2
+
3
+ module Kward
4
+ module Workers
5
+ # Cooperative ownership guard for workspace-mutating worker tools.
6
+ class WriteLock
7
+ def initialize
8
+ @owner_id = nil
9
+ @mutex = Mutex.new
10
+ end
11
+
12
+ attr_reader :owner_id
13
+
14
+ def acquire(owner_id)
15
+ owner = owner_id.to_s
16
+ return false if owner.empty?
17
+
18
+ @mutex.synchronize do
19
+ return true if @owner_id == owner
20
+ return false if @owner_id
21
+
22
+ @owner_id = owner
23
+ true
24
+ end
25
+ end
26
+
27
+ def owned_by?(owner_id)
28
+ @mutex.synchronize { @owner_id == owner_id.to_s }
29
+ end
30
+
31
+ def release(owner_id)
32
+ @mutex.synchronize do
33
+ @owner_id = nil if @owner_id == owner_id.to_s
34
+ end
35
+ end
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,7 @@
1
+ require_relative "workers/git_guard"
2
+ require_relative "workers/live_view"
3
+ require_relative "workers/manager"
4
+ require_relative "workers/store"
5
+ require_relative "workers/tool_policy"
6
+ require_relative "workers/write_lock"
7
+ require_relative "workers/worker"