Module: Kward::ConfigFiles
- Defined in:
- lib/kward/config_files.rb
Overview
Resolves Kward configuration, cache, memory, prompt, skill, and plugin paths, and reads/writes the JSON config file used by the CLI and RPC server.
This module is the configuration boundary, not a runtime settings cache.
Most methods read the filesystem each time so CLI commands and RPC reloads can
observe edits made outside the process. Callers that need caching should own
invalidation explicitly, as Client#reload_config does for provider state.
Keep path decisions here. Higher-level code should ask ConfigFiles for
config, prompt, skill, plugin, cache, memory, and session locations instead of
reconstructing ~/.kward paths independently.
Defined Under Namespace
Classes: PromptTemplate, Skill
Constant Summary collapse
- MAX_SKILL_FILE_BYTES =
100_000- MAX_PROMPT_FILE_BYTES =
32 * 1024
- DEFAULT_OVERLAY_SETTINGS =
{ "alignment" => "center", "width" => "maximum" }.freeze
- DEFAULT_PERSONAS =
{ "characters" => [ { "key" => "kward", "label" => "Kward", "instruction" => "Your name is Kward, the grim Andruid - robotic keeper of the Forrest of Code, protecting the nature of good engineering priciples. Speak like an old druid, be suspicous of everyone, but with a good intend." } ], "default" => "kward" }.freeze
- OVERLAY_ALIGNMENTS =
%w[left center right].freeze
- OVERLAY_WIDTHS =
%w[capped maximum].freeze
Class Method Summary collapse
-
.active_persona_label(workspace_root:, model: nil, config: read_config) ⇒ Object
Returns the label of the persona selected by default/workspace/model rules.
- .add_persona_entry(entries, layer, value, name: nil) ⇒ Object
-
.agents_prompt ⇒ String?
Reads global principle instructions from the config directory.
- .cache_dir ⇒ Object
- .canonical_workspace_root(path) ⇒ Object
- .character_entries(raw) ⇒ Object
- .character_entry(entry) ⇒ Object
- .code_search_cache_dir ⇒ Object
-
.composer_busy_help?(config = read_config) ⇒ Boolean
Returns whether the composer should show busy-state keyboard help.
-
.composer_tab_keybindings(config = read_config) ⇒ Object
Returns the configured tab keybinding family, or auto when unset/invalid.
- .config_agents_path ⇒ Object
-
.config_dir ⇒ String
Directory that contains Kward's user config and adjacent prompt/skill data.
-
.config_path ⇒ String
Expanded JSON config file path.
- .config_principles_path ⇒ Object
-
.config_value(config, *keys) ⇒ Object
Returns the first present non-empty string value among several config keys.
- .crew_character_labels(personas) ⇒ Object
- .crew_characters(personas) ⇒ Object
- .default_config ⇒ Object
-
.delete_config_key(key, path = config_path) ⇒ Object
Removes a top-level config key when it exists.
-
.editor_auto_close_pairs?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should auto-close typed pairs.
-
.editor_auto_indent?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should auto-indent new lines.
-
.editor_bar_cursor?(config = read_config) ⇒ Boolean
Returns whether editable built-in TUI editor buffers should use a bar cursor.
-
.editor_line_numbers(config = read_config) ⇒ Object
Returns the built-in TUI editor line-number display mode.
-
.editor_mode(config = read_config) ⇒ Object
Returns the built-in TUI editor keymap mode.
-
.editor_soft_wrap?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should soft-wrap long lines.
- .ekwsh_config_path ⇒ Object
-
.enforce_workspace_agents_file?(config = read_config) ⇒ Boolean
Returns whether workspace AGENTS.md contents should be injected directly instead of a compact read-when-relevant instruction.
-
.ensure_default_config!(path = config_path) ⇒ Object
Performs ensure default config for configuration file and path handling.
- .extract_character_instruction(definition) ⇒ Object
- .extract_character_label(definition) ⇒ Object
- .inside_directory?(path, base) ⇒ Boolean
- .markdown_parts(path) ⇒ Object
- .memory_core_path ⇒ Object
-
.memory_dir ⇒ String
Directory containing structured memory files.
- .memory_events_path ⇒ Object
- .memory_soft_path ⇒ Object
- .named_character_values(personas) ⇒ Object
- .normalize_ekwsh_aliases(values) ⇒ Object
- .normalize_ekwsh_config(data) ⇒ Object
- .normalize_ekwsh_env(values) ⇒ Object
- .normalize_ekwsh_shell(value) ⇒ Object
- .normalize_positive_integer(value, default) ⇒ Object
- .openrouter_models_cache_path ⇒ Object
-
.overlay_settings(config = read_config) ⇒ Hash
Returns validated overlay settings with defaults for missing or invalid values.
- .persona_entries(workspace_root:, model: nil, reasoning_effort: nil, now: Time.now, config: read_config, include_reasoning: true) ⇒ Object
- .persona_label_for_key(value, labels) ⇒ Object
-
.persona_prompt(workspace_root, model: nil, reasoning_effort: nil, now: Time.now, config: read_config) ⇒ String?
Builds persona prompt text from default, workspace, model, reasoning, time-of-day, weekday, and suffix config entries.
-
.plugin_dir ⇒ String
Trusted user plugin directory.
-
.plugin_paths ⇒ Array<String>
Finds trusted top-level plugin files.
- .presence(value) ⇒ Object
- .project_browser_state_path ⇒ Object
- .prompt_history_path(cwd, config_dir: self.config_dir, kind: "prompt") ⇒ Object
- .prompt_template_registry ⇒ Object
-
.prompt_templates(reserved_commands: []) ⇒ Array<PromptTemplate>
Lists prompt templates exposed as slash commands.
-
.read_config(path = config_path) ⇒ Hash
Reads the JSON config file.
- .read_ekwsh_config(path = ekwsh_config_path) ⇒ Object
- .read_prompt_file(path, label) ⇒ Object
-
.read_skill_file(name, relative_path = nil) ⇒ String
Reads a skill file by skill name and optional relative path.
- .resolved_persona_text(value, characters: {}) ⇒ Object
-
.session_auto_resume_enabled?(config = read_config) ⇒ Boolean
Returns whether new frontends should resume the last active session automatically.
-
.skills ⇒ Array<Skill>
Lists configured skills discovered under the config directory.
- .skills_registry ⇒ Object
- .time_of_day_bucket(now) ⇒ Object
-
.update_config(values, path = config_path) ⇒ Object
Merges top-level config values and writes the updated config privately.
-
.update_nested_config(section, values, path = config_path) ⇒ Object
Merges values into a one-level nested config section and writes privately.
-
.update_overlay_settings(values) ⇒ Object
Validates and persists terminal overlay settings.
-
.web_search_config(config = read_config) ⇒ Object
Returns the nested web-search config object, or an empty config when absent.
- .weekday_name(now) ⇒ Object
- .workspace_agents_file?(workspace_root) ⇒ Boolean
- .workspace_agents_path(workspace_root) ⇒ Object
- .workspace_agents_prompt(workspace_root) ⇒ Object
-
.workspace_guardrails_enabled?(config = read_config) ⇒ Boolean
Returns whether file tools must stay inside the active workspace root.
-
.write_config(config, path = config_path) ⇒ Object
Writes config JSON using private file permissions.
Class Method Details
.active_persona_label(workspace_root:, model: nil, config: read_config) ⇒ Object
Returns the label of the persona selected by default/workspace/model rules.
432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 |
# File 'lib/kward/config_files.rb', line 432 def active_persona_label(workspace_root:, model: nil, config: read_config) personas = config["personas"] return nil unless personas.is_a?(Hash) labels = crew_character_labels(personas) active_label = persona_label_for_key(personas["default"], labels) unless personas["default"].nil? workspaces = personas["workspaces"] if workspaces.is_a?(Hash) root = canonical_workspace_root(workspace_root) workspaces.each do |path, key| next unless canonical_workspace_root(path) == root active_label = persona_label_for_key(key, labels) break end end models = personas["models"] if models.is_a?(Hash) && !model.to_s.empty? && models.key?(model.to_s) active_label = persona_label_for_key(models[model.to_s], labels) end active_label end |
.add_persona_entry(entries, layer, value, name: nil) ⇒ Object
542 543 544 545 546 547 |
# File 'lib/kward/config_files.rb', line 542 def add_persona_entry(entries, layer, value, name: nil) text = presence(value) return unless text entries << { layer: layer.to_s, name: name.to_s, prompt: text } end |
.agents_prompt ⇒ String?
Reads global principle instructions from the config directory.
PRINCIPLES.md is preferred. AGENTS.md remains a backwards-compatible
alias for existing installations.
394 395 396 397 398 399 |
# File 'lib/kward/config_files.rb', line 394 def agents_prompt path = config_principles_path return read_prompt_file(path, "Kward principles file") if File.exist?(path) read_prompt_file(config_agents_path, "Kward AGENTS.md alias") end |
.cache_dir ⇒ Object
66 67 68 |
# File 'lib/kward/config_files.rb', line 66 def cache_dir File.join(config_dir, "cache") end |
.canonical_workspace_root(path) ⇒ Object
537 538 539 540 |
# File 'lib/kward/config_files.rb', line 537 def canonical_workspace_root(path) = File.(path.to_s.empty? ? Dir.pwd : path.to_s) File.directory?() ? File.realpath() : end |
.character_entries(raw) ⇒ Object
589 590 591 592 593 594 595 596 597 598 |
# File 'lib/kward/config_files.rb', line 589 def character_entries(raw) case raw when Hash raw.map { |key, definition| [key, definition] } when Array raw.filter_map { |entry| character_entry(entry) } else [] end end |
.character_entry(entry) ⇒ Object
600 601 602 603 604 605 606 607 608 609 |
# File 'lib/kward/config_files.rb', line 600 def character_entry(entry) return nil unless entry.is_a?(Hash) if entry.length == 1 && entry.keys.first.is_a?(String) [entry.keys.first, entry.values.first] else key = entry["key"] || entry[:key] || entry["id"] || entry[:id] || entry["name"] || entry[:name] key.to_s.empty? ? nil : [key, entry] end end |
.code_search_cache_dir ⇒ Object
112 113 114 |
# File 'lib/kward/config_files.rb', line 112 def code_search_cache_dir File.join(cache_dir, "code_search") end |
.composer_busy_help?(config = read_config) ⇒ Boolean
Returns whether the composer should show busy-state keyboard help.
292 293 294 295 |
# File 'lib/kward/config_files.rb', line 292 def composer_busy_help?(config = read_config) composer = config["composer"].is_a?(Hash) ? config["composer"] : {} composer["busy_help"] != false end |
.composer_tab_keybindings(config = read_config) ⇒ Object
Returns the configured tab keybinding family, or auto when unset/invalid.
298 299 300 301 302 |
# File 'lib/kward/config_files.rb', line 298 def composer_tab_keybindings(config = read_config) composer = config["composer"].is_a?(Hash) ? config["composer"] : {} value = composer["tab_keybindings"].to_s.downcase %w[auto ctrl alt].include?(value) ? value : "auto" end |
.config_agents_path ⇒ Object
405 406 407 |
# File 'lib/kward/config_files.rb', line 405 def config_agents_path File.join(config_dir, "AGENTS.md") end |
.config_dir ⇒ String
Directory that contains Kward's user config and adjacent prompt/skill
data. Defaults to ~/.kward, or the directory of KWARD_CONFIG_PATH.
54 55 56 57 58 59 |
# File 'lib/kward/config_files.rb', line 54 def config_dir config_path = ENV["KWARD_CONFIG_PATH"] return File.(File.dirname(config_path)) if config_path && !config_path.empty? File.("~/.kward") end |
.config_path ⇒ String
Returns expanded JSON config file path.
62 63 64 |
# File 'lib/kward/config_files.rb', line 62 def config_path File.(ENV["KWARD_CONFIG_PATH"] || File.join(config_dir, "config.json")) end |
.config_principles_path ⇒ Object
401 402 403 |
# File 'lib/kward/config_files.rb', line 401 def config_principles_path File.join(config_dir, "PRINCIPLES.md") end |
.config_value(config, *keys) ⇒ Object
Returns the first present non-empty string value among several config keys.
268 269 270 271 272 273 274 |
# File 'lib/kward/config_files.rb', line 268 def config_value(config, *keys) keys.each do |key| text = presence(config[key]) return text if text end nil end |
.crew_character_labels(personas) ⇒ Object
555 556 557 558 559 |
# File 'lib/kward/config_files.rb', line 555 def crew_character_labels(personas) named_character_values(personas) do |_key, definition| extract_character_label(definition) end end |
.crew_characters(personas) ⇒ Object
549 550 551 552 553 |
# File 'lib/kward/config_files.rb', line 549 def crew_characters(personas) named_character_values(personas) do |_key, definition| extract_character_instruction(definition) end end |
.default_config ⇒ Object
74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 |
# File 'lib/kward/config_files.rb', line 74 def default_config { "personas" => JSON.parse(JSON.generate(DEFAULT_PERSONAS)), "memory" => { "enabled" => false, "auto_summary" => false }, "composer" => { "busy_help" => true, "tab_keybindings" => "auto" }, "editor" => { "mode" => "modern", "auto_indent" => true, "auto_close_pairs" => true, "soft_wrap" => true, "bar_cursor" => true, "line_numbers" => "absolute" }, "sessions" => { "auto_resume" => false }, "enforce_workspace_agents_file" => false, "tools" => { "workspace_guardrails" => true } } end |
.delete_config_key(key, path = config_path) ⇒ Object
Removes a top-level config key when it exists.
259 260 261 262 263 264 265 |
# File 'lib/kward/config_files.rb', line 259 def delete_config_key(key, path = config_path) config = read_config(path) existed = config.key?(key.to_s) config.delete(key.to_s) write_config(config, path) if existed existed end |
.editor_auto_close_pairs?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should auto-close typed pairs.
317 318 319 320 |
# File 'lib/kward/config_files.rb', line 317 def editor_auto_close_pairs?(config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} editor["auto_close_pairs"] != false end |
.editor_auto_indent?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should auto-indent new lines.
311 312 313 314 |
# File 'lib/kward/config_files.rb', line 311 def editor_auto_indent?(config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} editor["auto_indent"] != false end |
.editor_bar_cursor?(config = read_config) ⇒ Boolean
Returns whether editable built-in TUI editor buffers should use a bar cursor.
329 330 331 332 |
# File 'lib/kward/config_files.rb', line 329 def (config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} editor["bar_cursor"] != false end |
.editor_line_numbers(config = read_config) ⇒ Object
Returns the built-in TUI editor line-number display mode.
335 336 337 338 |
# File 'lib/kward/config_files.rb', line 335 def editor_line_numbers(config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} EditorMode.normalize_line_numbers(editor["line_numbers"]) end |
.editor_mode(config = read_config) ⇒ Object
Returns the built-in TUI editor keymap mode.
305 306 307 308 |
# File 'lib/kward/config_files.rb', line 305 def editor_mode(config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} EditorMode.normalize(editor["mode"]) end |
.editor_soft_wrap?(config = read_config) ⇒ Boolean
Returns whether the built-in TUI editor should soft-wrap long lines.
323 324 325 326 |
# File 'lib/kward/config_files.rb', line 323 def editor_soft_wrap?(config = read_config) editor = config["editor"].is_a?(Hash) ? config["editor"] : {} editor["soft_wrap"] != false end |
.ekwsh_config_path ⇒ Object
70 71 72 |
# File 'lib/kward/config_files.rb', line 70 def ekwsh_config_path File.join(config_dir, "ekwsh.yml") end |
.enforce_workspace_agents_file?(config = read_config) ⇒ Boolean
Returns whether workspace AGENTS.md contents should be injected directly instead of a compact read-when-relevant instruction.
354 355 356 |
# File 'lib/kward/config_files.rb', line 354 def enforce_workspace_agents_file?(config = read_config) config["enforce_workspace_agents_file"] == true end |
.ensure_default_config!(path = config_path) ⇒ Object
Performs ensure default config for configuration file and path handling.
104 105 106 107 108 109 110 |
# File 'lib/kward/config_files.rb', line 104 def ensure_default_config!(path = config_path) path = File.(path) return false if File.exist?(path) write_config(default_config, path) true end |
.extract_character_instruction(definition) ⇒ Object
617 618 619 620 621 622 623 624 625 626 |
# File 'lib/kward/config_files.rb', line 617 def extract_character_instruction(definition) return nil if definition.nil? if definition.is_a?(Hash) value = definition["instruction"] || definition[:instruction] return presence(value) end presence(definition) end |
.extract_character_label(definition) ⇒ Object
611 612 613 614 615 |
# File 'lib/kward/config_files.rb', line 611 def extract_character_label(definition) return nil unless definition.is_a?(Hash) presence(definition["label"] || definition[:label]) end |
.inside_directory?(path, base) ⇒ Boolean
717 718 719 |
# File 'lib/kward/config_files.rb', line 717 def inside_directory?(path, base) path == base || path.start_with?(base + File::SEPARATOR) end |
.markdown_parts(path) ⇒ Object
704 705 706 707 708 709 710 711 712 713 714 715 |
# File 'lib/kward/config_files.rb', line 704 def markdown_parts(path) content = File.read(path) return [{}, content] unless content.start_with?("---\n", "---\r\n") _opening, rest = content.split(/\A---\r?\n/, 2) yaml_text, body = rest.to_s.split(/\r?\n---\r?\n/, 2) raise "missing frontmatter closing delimiter" if body.nil? data = yaml_text.to_s.empty? ? {} : YAML.safe_load(yaml_text, permitted_classes: [], aliases: false) frontmatter = data.is_a?(Hash) ? data.transform_keys(&:to_s) : {} [frontmatter, body] end |
.memory_core_path ⇒ Object
136 137 138 |
# File 'lib/kward/config_files.rb', line 136 def memory_core_path File.join(memory_dir, "core.json") end |
.memory_dir ⇒ String
Returns directory containing structured memory files.
132 133 134 |
# File 'lib/kward/config_files.rb', line 132 def memory_dir File.join(config_dir, "memory") end |
.memory_events_path ⇒ Object
144 145 146 |
# File 'lib/kward/config_files.rb', line 144 def memory_events_path File.join(memory_dir, "events.jsonl") end |
.memory_soft_path ⇒ Object
140 141 142 |
# File 'lib/kward/config_files.rb', line 140 def memory_soft_path File.join(memory_dir, "soft.jsonl") end |
.named_character_values(personas) ⇒ Object
580 581 582 583 584 585 586 587 |
# File 'lib/kward/config_files.rb', line 580 def named_character_values(personas) character_entries(personas["characters"] || personas["crew"]).each_with_object({}) do |(key, definition), mapping| value = yield(key, definition) next if value.to_s.empty? mapping[key.to_s] = value end end |
.normalize_ekwsh_aliases(values) ⇒ Object
224 225 226 227 228 229 230 231 232 233 234 235 |
# File 'lib/kward/config_files.rb', line 224 def normalize_ekwsh_aliases(values) return {} unless values.is_a?(Hash) values.each_with_object({}) do |(name, command), result| name = name.to_s command = command.to_s.strip next unless Ekwsh.valid_alias_name?(name) next if command.empty? result[name] = command end end |
.normalize_ekwsh_config(data) ⇒ Object
184 185 186 187 188 189 190 191 192 193 194 195 |
# File 'lib/kward/config_files.rb', line 184 def normalize_ekwsh_config(data) data = data.transform_keys(&:to_s) if data.is_a?(Hash) settings = data.is_a?(Hash) ? data : {} { shell: normalize_ekwsh_shell(settings["shell"]), timeout_seconds: normalize_positive_integer(settings["timeout_seconds"], Ekwsh::DEFAULT_TIMEOUT_SECONDS), max_output_bytes: normalize_positive_integer(settings["max_output_bytes"], Ekwsh::DEFAULT_MAX_OUTPUT_BYTES), history_limit: normalize_positive_integer(settings["history_limit"], Ekwsh::DEFAULT_HISTORY_LIMIT), env: normalize_ekwsh_env(settings["env"]), aliases: normalize_ekwsh_aliases(settings["aliases"]) } end |
.normalize_ekwsh_env(values) ⇒ Object
212 213 214 215 216 217 218 219 220 221 222 |
# File 'lib/kward/config_files.rb', line 212 def normalize_ekwsh_env(values) return {} unless values.is_a?(Hash) values.each_with_object({}) do |(key, value), result| key = key.to_s next unless key.match?(/\A[A-Za-z_][A-Za-z0-9_]*\z/) next if value.nil? result[key] = value.to_s end end |
.normalize_ekwsh_shell(value) ⇒ Object
197 198 199 200 201 202 203 |
# File 'lib/kward/config_files.rb', line 197 def normalize_ekwsh_shell(value) shell = value.to_s.strip return Ekwsh::DEFAULT_SHELL if shell.empty? return shell if shell.start_with?("/") && File.executable?(shell) Ekwsh::DEFAULT_SHELL end |
.normalize_positive_integer(value, default) ⇒ Object
205 206 207 208 209 210 |
# File 'lib/kward/config_files.rb', line 205 def normalize_positive_integer(value, default) integer = Integer(value) integer.positive? ? integer : default rescue ArgumentError, TypeError default end |
.openrouter_models_cache_path ⇒ Object
116 117 118 |
# File 'lib/kward/config_files.rb', line 116 def openrouter_models_cache_path File.join(cache_dir, "openrouter_models.json") end |
.overlay_settings(config = read_config) ⇒ Hash
Returns validated overlay settings with defaults for missing or invalid values.
281 282 283 284 285 286 287 288 289 |
# File 'lib/kward/config_files.rb', line 281 def (config = read_config) = config["overlay"].is_a?(Hash) ? config["overlay"] : {} settings = DEFAULT_OVERLAY_SETTINGS.dup alignment = ["alignment"].to_s width = ["width"].to_s settings["alignment"] = alignment if OVERLAY_ALIGNMENTS.include?(alignment) settings["width"] = width if OVERLAY_WIDTHS.include?(width) settings end |
.persona_entries(workspace_root:, model: nil, reasoning_effort: nil, now: Time.now, config: read_config, include_reasoning: true) ⇒ Object
458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 |
# File 'lib/kward/config_files.rb', line 458 def persona_entries(workspace_root:, model: nil, reasoning_effort: nil, now: Time.now, config: read_config, include_reasoning: true) personas = config["personas"] return [] unless personas.is_a?(Hash) characters = crew_characters(personas) entries = [] active_persona = { layer: "default", value: personas["default"], name: nil } workspaces = personas["workspaces"] if workspaces.is_a?(Hash) root = canonical_workspace_root(workspace_root) workspaces.each do |path, key| if canonical_workspace_root(path) == root active_persona = { layer: "workspace", value: key, name: path } break end end end models = personas["models"] if models.is_a?(Hash) && !model.to_s.empty? && models.key?(model.to_s) active_persona = { layer: "model", value: models[model.to_s], name: model.to_s } end add_persona_entry( entries, active_persona.fetch(:layer), resolved_persona_text(active_persona.fetch(:value), characters: characters), name: active_persona[:name] ) modifiers = personas["persona_modifiers"] if modifiers.is_a?(Hash) if include_reasoning reasoning = modifiers["reasoning"] add_persona_entry(entries, "reasoning", reasoning[reasoning_effort.to_s]) if reasoning.is_a?(Hash) && !reasoning_effort.to_s.empty? end time_of_day = modifiers["time_of_day"] bucket = time_of_day_bucket(now) add_persona_entry(entries, "time_of_day", time_of_day[bucket], name: bucket) if time_of_day.is_a?(Hash) weekday = modifiers["weekday"] day = weekday_name(now) add_persona_entry(entries, "weekday", weekday[day], name: day) if weekday.is_a?(Hash) add_persona_entry(entries, "suffix", modifiers["suffix"]) end entries end |
.persona_label_for_key(value, labels) ⇒ Object
573 574 575 576 577 578 |
# File 'lib/kward/config_files.rb', line 573 def persona_label_for_key(value, labels) key = value.to_s.strip return nil if key.empty? presence(labels[key]) end |
.persona_prompt(workspace_root, model: nil, reasoning_effort: nil, now: Time.now, config: read_config) ⇒ String?
Builds persona prompt text from default, workspace, model, reasoning, time-of-day, weekday, and suffix config entries.
Persona resolution is intentionally data-driven so users can edit config without plugin code. Keep new persona selectors additive and deterministic; prompt construction depends on stable ordering.
422 423 424 425 426 427 428 429 |
# File 'lib/kward/config_files.rb', line 422 def persona_prompt(workspace_root, model: nil, reasoning_effort: nil, now: Time.now, config: read_config) text = persona_entries(workspace_root: workspace_root, model: model, reasoning_effort: reasoning_effort, now: now, config: config).map do |entry| entry[:prompt] end.join("\n\n") return nil if text.empty? text end |
.plugin_dir ⇒ String
Returns trusted user plugin directory.
649 650 651 |
# File 'lib/kward/config_files.rb', line 649 def plugin_dir File.("~/.kward/plugins") end |
.plugin_paths ⇒ Array<String>
Finds trusted top-level plugin files.
Plugins are intentionally loaded only from ~/.kward/plugins, not from a
workspace or custom KWARD_CONFIG_PATH directory.
659 660 661 662 663 664 665 666 667 |
# File 'lib/kward/config_files.rb', line 659 def plugin_paths plugins_root = plugin_dir return [] unless Dir.exist?(plugins_root) Dir.glob(File.join(plugins_root, "*.rb")).sort rescue StandardError => e warn "Warning: skipping Kward plugins in #{plugins_root}: #{e.}" [] end |
.presence(value) ⇒ Object
721 722 723 724 |
# File 'lib/kward/config_files.rb', line 721 def presence(value) text = value.to_s text.empty? ? nil : text end |
.project_browser_state_path ⇒ Object
120 121 122 |
# File 'lib/kward/config_files.rb', line 120 def project_browser_state_path File.join(cache_dir, "project_browser_state.json") end |
.prompt_history_path(cwd, config_dir: self.config_dir, kind: "prompt") ⇒ Object
124 125 126 127 128 129 |
# File 'lib/kward/config_files.rb', line 124 def prompt_history_path(cwd, config_dir: self.config_dir, kind: "prompt") key = Digest::SHA256.hexdigest(canonical_workspace_root(cwd))[0, 24] return File.join(config_dir, "history", "#{key}.jsonl") if kind.to_s == "prompt" File.join(config_dir, "history", kind.to_s, "#{key}.jsonl") end |
.prompt_template_registry ⇒ Object
696 697 698 699 700 701 702 |
# File 'lib/kward/config_files.rb', line 696 def prompt_template_registry Prompts::Templates.new( config_dir: config_dir, template_class: PromptTemplate, markdown_parser: method(:markdown_parts) ) end |
.prompt_templates(reserved_commands: []) ⇒ Array<PromptTemplate>
Lists prompt templates exposed as slash commands.
673 674 675 |
# File 'lib/kward/config_files.rb', line 673 def prompt_templates(reserved_commands: []) prompt_template_registry.prompt_templates(reserved_commands: reserved_commands) end |
.read_config(path = config_path) ⇒ Hash
Reads the JSON config file.
Missing files are treated as an empty config. Invalid JSON raises a user-facing error that includes the file path. This method does not merge defaults; callers should apply feature-specific defaults at the point where behavior is decided.
157 158 159 160 161 162 163 164 |
# File 'lib/kward/config_files.rb', line 157 def read_config(path = config_path) path = File.(path) return {} unless File.exist?(path) JSON.parse(File.read(path)) rescue JSON::ParserError raise "Invalid Kward config JSON: #{path}" end |
.read_ekwsh_config(path = ekwsh_config_path) ⇒ Object
174 175 176 177 178 179 180 181 182 |
# File 'lib/kward/config_files.rb', line 174 def read_ekwsh_config(path = ekwsh_config_path) path = File.(path) return normalize_ekwsh_config(nil) unless File.exist?(path) data = YAML.safe_load(File.read(path), permitted_classes: [], aliases: false) normalize_ekwsh_config(data) rescue Psych::SyntaxError => e raise "Invalid ekwsh YAML config: #{path}: #{e.}" end |
.read_prompt_file(path, label) ⇒ Object
522 523 524 525 526 527 528 529 530 531 532 533 534 535 |
# File 'lib/kward/config_files.rb', line 522 def read_prompt_file(path, label) return nil unless File.exist?(path) size = File.size(path) if size > MAX_PROMPT_FILE_BYTES warn "Warning: skipping #{label} #{path}: file too large (#{size} bytes; limit is #{MAX_PROMPT_FILE_BYTES} bytes)" return nil end File.read(path) rescue StandardError => e warn "Warning: skipping #{label} #{path}: #{e.}" nil end |
.read_skill_file(name, relative_path = nil) ⇒ String
Reads a skill file by skill name and optional relative path.
682 683 684 |
# File 'lib/kward/config_files.rb', line 682 def read_skill_file(name, relative_path = nil) skills_registry.read_skill_file(name, relative_path) end |
.resolved_persona_text(value, characters: {}) ⇒ Object
561 562 563 564 565 566 567 568 569 570 571 |
# File 'lib/kward/config_files.rb', line 561 def resolved_persona_text(value, characters: {}) return nil if value.nil? key = value.to_s.strip return nil if key.empty? text = characters[key.to_s] return text unless text.to_s.empty? value end |
.session_auto_resume_enabled?(config = read_config) ⇒ Boolean
Returns whether new frontends should resume the last active session automatically.
347 348 349 350 |
# File 'lib/kward/config_files.rb', line 347 def session_auto_resume_enabled?(config = read_config) sessions = config["sessions"].is_a?(Hash) ? config["sessions"] : {} sessions["auto_resume"] == true end |
.skills ⇒ Array<Skill>
Lists configured skills discovered under the config directory.
644 645 646 |
# File 'lib/kward/config_files.rb', line 644 def skills skills_registry.skills end |
.skills_registry ⇒ Object
686 687 688 689 690 691 692 693 694 |
# File 'lib/kward/config_files.rb', line 686 def skills_registry Skills::Registry.new( config_dir: config_dir, skill_class: Skill, max_file_bytes: MAX_SKILL_FILE_BYTES, markdown_parser: method(:markdown_parts), inside_directory: method(:inside_directory?) ) end |
.time_of_day_bucket(now) ⇒ Object
628 629 630 631 632 633 634 635 |
# File 'lib/kward/config_files.rb', line 628 def time_of_day_bucket(now) hour = now.hour return "morning" if hour >= 5 && hour < 11 return "before_lunch" if hour == 11 return "late_evening" if hour >= 21 || hour < 5 nil end |
.update_config(values, path = config_path) ⇒ Object
Merges top-level config values and writes the updated config privately.
238 239 240 241 242 243 244 245 |
# File 'lib/kward/config_files.rb', line 238 def update_config(values, path = config_path) raise "Config values must be an object" unless values.is_a?(Hash) config = read_config(path) values.each { |key, value| config[key.to_s] = value } write_config(config, path) config end |
.update_nested_config(section, values, path = config_path) ⇒ Object
Merges values into a one-level nested config section and writes privately.
248 249 250 251 252 253 254 255 256 |
# File 'lib/kward/config_files.rb', line 248 def update_nested_config(section, values, path = config_path) raise "Config values must be an object" unless values.is_a?(Hash) config = read_config(path) current = config[section.to_s].is_a?(Hash) ? config[section.to_s].dup : {} config[section.to_s] = current.merge(values.transform_keys(&:to_s)) write_config(config, path) config end |
.update_overlay_settings(values) ⇒ Object
Validates and persists terminal overlay settings.
365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 |
# File 'lib/kward/config_files.rb', line 365 def (values) raise "Overlay settings must be an object" unless values.is_a?(Hash) config = read_config = config["overlay"].is_a?(Hash) ? config["overlay"].dup : {} values.each do |key, value| key = key.to_s value = value.to_s case key when "alignment" raise "Overlay alignment must be left, center, or right" unless OVERLAY_ALIGNMENTS.include?(value) when "width" raise "Overlay width must be capped or maximum" unless OVERLAY_WIDTHS.include?(value) else raise "Unknown overlay setting: #{key}" end [key] = value end config["overlay"] = write_config(config) (config) end |
.web_search_config(config = read_config) ⇒ Object
Returns the nested web-search config object, or an empty config when absent.
359 360 361 362 |
# File 'lib/kward/config_files.rb', line 359 def web_search_config(config = read_config) value = config["web_search"] value.is_a?(Hash) ? value : {} end |
.weekday_name(now) ⇒ Object
637 638 639 |
# File 'lib/kward/config_files.rb', line 637 def weekday_name(now) %w[sunday monday tuesday wednesday thursday friday saturday][now.wday] end |
.workspace_agents_file?(workspace_root) ⇒ Boolean
514 515 516 |
# File 'lib/kward/config_files.rb', line 514 def workspace_agents_file?(workspace_root) File.exist?(workspace_agents_path(workspace_root)) end |
.workspace_agents_path(workspace_root) ⇒ Object
510 511 512 |
# File 'lib/kward/config_files.rb', line 510 def workspace_agents_path(workspace_root) File.join(canonical_workspace_root(workspace_root), "AGENTS.md") end |
.workspace_agents_prompt(workspace_root) ⇒ Object
518 519 520 |
# File 'lib/kward/config_files.rb', line 518 def workspace_agents_prompt(workspace_root) read_prompt_file(workspace_agents_path(workspace_root), "workspace AGENTS.md") end |
.workspace_guardrails_enabled?(config = read_config) ⇒ Boolean
Returns whether file tools must stay inside the active workspace root.
341 342 343 344 |
# File 'lib/kward/config_files.rb', line 341 def workspace_guardrails_enabled?(config = read_config) tools = config["tools"].is_a?(Hash) ? config["tools"] : {} tools["workspace_guardrails"] != false end |
.write_config(config, path = config_path) ⇒ Object
Writes config JSON using private file permissions.
170 171 172 |
# File 'lib/kward/config_files.rb', line 170 def write_config(config, path = config_path) PrivateFile.write_json(path, config) end |