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.
- checksums.yaml +4 -4
- data/.github/workflows/pages.yml +1 -1
- data/CHANGELOG.md +89 -3
- data/Gemfile +2 -0
- data/Gemfile.lock +90 -2
- data/README.md +34 -6
- data/Rakefile +96 -0
- data/doc/agent-tools.md +52 -0
- data/doc/api.md +92 -0
- data/doc/authentication.md +58 -23
- data/doc/code-search.md +42 -2
- data/doc/configuration.md +102 -13
- data/doc/context-budgeting.md +136 -0
- data/doc/context-tools.md +83 -0
- data/doc/editor.md +394 -0
- data/doc/extensibility.md +16 -7
- data/doc/files.md +100 -0
- data/doc/getting-started.md +25 -18
- data/doc/git.md +122 -0
- data/doc/memory.md +24 -4
- data/doc/personas.md +34 -5
- data/doc/plugins.md +74 -3
- data/doc/releasing.md +45 -8
- data/doc/rpc.md +77 -15
- data/doc/session-management.md +254 -0
- data/doc/shell.md +286 -0
- data/doc/tabs.md +122 -0
- data/doc/troubleshooting.md +77 -1
- data/doc/usage.md +60 -15
- data/doc/web-search.md +12 -4
- data/doc/workspace-tools.md +144 -0
- data/examples/plugins/space_invaders.rb +377 -0
- data/lib/kward/agent.rb +1 -1
- data/lib/kward/cli/commands.rb +41 -2
- data/lib/kward/cli/git.rb +150 -0
- data/lib/kward/cli/interactive_turn.rb +73 -9
- data/lib/kward/cli/openrouter_commands.rb +55 -0
- data/lib/kward/cli/plugins.rb +54 -4
- data/lib/kward/cli/prompt_interface.rb +111 -6
- data/lib/kward/cli/rendering.rb +11 -6
- data/lib/kward/cli/runtime_helpers.rb +133 -3
- data/lib/kward/cli/sessions.rb +262 -13
- data/lib/kward/cli/settings.rb +216 -37
- data/lib/kward/cli/slash_commands.rb +439 -8
- data/lib/kward/cli/tabs.rb +695 -0
- data/lib/kward/cli.rb +171 -26
- data/lib/kward/compactor.rb +4 -1
- data/lib/kward/config_files.rb +125 -5
- data/lib/kward/context_budget_meter.rb +44 -0
- data/lib/kward/conversation.rb +59 -22
- data/lib/kward/editor_mode.rb +25 -0
- data/lib/kward/ekwsh.rb +362 -0
- data/lib/kward/model/client.rb +37 -50
- data/lib/kward/model/context_usage.rb +13 -6
- data/lib/kward/model/model_info.rb +92 -16
- data/lib/kward/model/payloads.rb +2 -0
- data/lib/kward/openrouter_model_cache.rb +120 -0
- data/lib/kward/plugin_registry.rb +108 -1
- data/lib/kward/project_files.rb +52 -0
- data/lib/kward/prompt_history.rb +82 -0
- data/lib/kward/prompt_interface/banner.rb +16 -51
- data/lib/kward/prompt_interface/composer_controller.rb +124 -83
- data/lib/kward/prompt_interface/composer_renderer.rb +116 -14
- data/lib/kward/prompt_interface/composer_state.rb +96 -27
- data/lib/kward/prompt_interface/editor/auto_close_pairs.rb +123 -0
- data/lib/kward/prompt_interface/editor/auto_indent.rb +509 -0
- data/lib/kward/prompt_interface/editor/buffer.rb +109 -0
- data/lib/kward/prompt_interface/editor/controller.rb +1018 -0
- data/lib/kward/prompt_interface/editor/endwise.rb +321 -0
- data/lib/kward/prompt_interface/editor/file_marker.rb +40 -0
- data/lib/kward/prompt_interface/editor/indent_navigation.rb +61 -0
- data/lib/kward/prompt_interface/editor/kill_ring.rb +78 -0
- data/lib/kward/prompt_interface/editor/modes/emacs.rb +259 -0
- data/lib/kward/prompt_interface/editor/modes/modern.rb +353 -0
- data/lib/kward/prompt_interface/editor/modes/vibe.rb +1962 -0
- data/lib/kward/prompt_interface/editor/renderer.rb +243 -0
- data/lib/kward/prompt_interface/editor/search.rb +76 -0
- data/lib/kward/prompt_interface/editor/selections.rb +120 -0
- data/lib/kward/prompt_interface/editor/state.rb +1249 -0
- data/lib/kward/prompt_interface/editor/status_text.rb +23 -0
- data/lib/kward/prompt_interface/editor/syntax_highlighter.rb +420 -0
- data/lib/kward/prompt_interface/editor/undo_history.rb +46 -0
- data/lib/kward/prompt_interface/editor/vibe_state.rb +44 -0
- data/lib/kward/prompt_interface/file_overlay.rb +211 -0
- data/lib/kward/prompt_interface/git_prompt.rb +299 -0
- data/lib/kward/prompt_interface/interactive/controller.rb +186 -0
- data/lib/kward/prompt_interface/interactive/renderer.rb +71 -0
- data/lib/kward/prompt_interface/interactive/state.rb +62 -0
- data/lib/kward/prompt_interface/key_handler.rb +416 -43
- data/lib/kward/prompt_interface/layout.rb +2 -2
- data/lib/kward/prompt_interface/overlay_renderer.rb +21 -2
- data/lib/kward/prompt_interface/project_browser.rb +524 -0
- data/lib/kward/prompt_interface/prompt_renderer.rb +32 -13
- data/lib/kward/prompt_interface/question_prompt.rb +122 -82
- data/lib/kward/prompt_interface/runtime_state.rb +49 -1
- data/lib/kward/prompt_interface/screen.rb +17 -0
- data/lib/kward/prompt_interface/selection_prompt.rb +511 -58
- data/lib/kward/prompt_interface/stream_state.rb +7 -0
- data/lib/kward/prompt_interface/transcript_buffer.rb +13 -16
- data/lib/kward/prompt_interface/transcript_renderer.rb +3 -3
- data/lib/kward/prompt_interface.rb +307 -35
- data/lib/kward/prompts/commands.rb +7 -1
- data/lib/kward/prompts.rb +4 -2
- data/lib/kward/rpc/server.rb +45 -11
- data/lib/kward/rpc/session_manager.rb +52 -53
- data/lib/kward/rpc/session_tree_rows.rb +9 -115
- data/lib/kward/rpc/tool_event_normalizer.rb +1 -1
- data/lib/kward/session_store.rb +67 -4
- data/lib/kward/session_tree_nodes.rb +136 -0
- data/lib/kward/session_tree_renderer.rb +9 -131
- data/lib/kward/tab_store.rb +47 -0
- data/lib/kward/telemetry/logger.rb +5 -3
- data/lib/kward/text_boundary.rb +25 -0
- data/lib/kward/tool_output_compactor.rb +127 -0
- data/lib/kward/tools/base.rb +8 -2
- data/lib/kward/tools/context_budget_stats.rb +54 -0
- data/lib/kward/tools/context_for_task.rb +202 -0
- data/lib/kward/tools/read_file.rb +8 -4
- data/lib/kward/tools/registry.rb +92 -15
- data/lib/kward/tools/retrieve_tool_output.rb +71 -0
- data/lib/kward/tools/search/web.rb +2 -2
- data/lib/kward/tools/summarize_file_structure.rb +29 -0
- data/lib/kward/tools/tool_call.rb +12 -0
- data/lib/kward/version.rb +1 -1
- data/lib/kward/workers/git_guard.rb +68 -0
- data/lib/kward/workers/live_view.rb +49 -0
- data/lib/kward/workers/manager.rb +288 -0
- data/lib/kward/workers/store.rb +72 -0
- data/lib/kward/workers/tool_policy.rb +23 -0
- data/lib/kward/workers/worker.rb +82 -0
- data/lib/kward/workers/write_lock.rb +38 -0
- data/lib/kward/workers.rb +7 -0
- data/lib/kward/workspace.rb +154 -12
- data/templates/default/fulldoc/html/css/kward.css +362 -42
- data/templates/default/fulldoc/html/full_list.erb +107 -0
- data/templates/default/fulldoc/html/js/kward.js +161 -2
- data/templates/default/fulldoc/html/setup.rb +8 -0
- data/templates/default/kward_navigation.rb +102 -0
- data/templates/default/layout/html/layout.erb +43 -10
- data/templates/default/layout/html/setup.rb +39 -38
- metadata +65 -3
- data/lib/kward/resources/avatar_kward_logo.rb +0 -50
- 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
|