1use std::path::{Path, PathBuf};
27
28use crate::error::Result;
29
30#[derive(Debug, Clone)]
32pub struct ToolHookConfig {
33 pub tool_name: String,
35 pub config_path: PathBuf,
37 pub config_content: String,
39 pub scope: HookScope,
41}
42
43#[derive(Debug, Clone, Copy, PartialEq, Eq)]
44pub enum HookScope {
45 Project,
47 User,
49}
50
51#[derive(Debug, Clone, Copy, PartialEq, Eq)]
54pub enum HookPlatform {
55 ClaudeCode,
57 Cursor,
59 GeminiCli,
61 Windsurf,
63}
64
65pub fn process_hook(input: &str) -> Result<String> {
84 process_hook_for_platform(input, HookPlatform::ClaudeCode)
85}
86
87pub fn process_hook_cursor(input: &str) -> Result<String> {
92 process_hook_for_platform(input, HookPlatform::Cursor)
93}
94
95pub fn process_hook_gemini(input: &str) -> Result<String> {
99 process_hook_for_platform(input, HookPlatform::GeminiCli)
100}
101
102pub fn process_hook_windsurf(input: &str) -> Result<String> {
107 process_hook_for_platform(input, HookPlatform::Windsurf)
108}
109
110fn process_hook_for_platform(input: &str, platform: HookPlatform) -> Result<String> {
114 let parsed: serde_json::Value = serde_json::from_str(input)
115 .map_err(|e| crate::error::SqzError::Other(format!("hook: invalid JSON input: {e}")))?;
116
117 let tool_name = parsed
121 .get("tool_name")
122 .or_else(|| parsed.get("toolName"))
123 .and_then(|v| v.as_str())
124 .unwrap_or("");
125
126 let hook_event = parsed
127 .get("hook_event_name")
128 .or_else(|| parsed.get("agent_action_name"))
129 .and_then(|v| v.as_str())
130 .unwrap_or("");
131
132 let is_shell = matches!(tool_name, "Bash" | "bash" | "Shell" | "shell" | "terminal"
141 | "run_terminal_command" | "run_shell_command")
142 || matches!(hook_event, "beforeShellExecution" | "pre_run_command");
143
144 if !is_shell {
145 return Ok(match platform {
148 HookPlatform::Cursor => "{}".to_string(),
149 _ => input.to_string(),
150 });
151 }
152
153 let command = parsed
158 .get("tool_input")
159 .and_then(|v| v.get("command"))
160 .and_then(|v| v.as_str())
161 .or_else(|| parsed.get("command").and_then(|v| v.as_str()))
162 .or_else(|| {
163 parsed
164 .get("tool_info")
165 .and_then(|v| v.get("command_line"))
166 .and_then(|v| v.as_str())
167 })
168 .or_else(|| {
169 parsed
170 .get("toolCall")
171 .and_then(|v| v.get("command"))
172 .and_then(|v| v.as_str())
173 })
174 .unwrap_or("");
175
176 if command.is_empty() {
177 return Ok(match platform {
178 HookPlatform::Cursor => "{}".to_string(),
179 _ => input.to_string(),
180 });
181 }
182
183 let base_cmd = extract_base_command(command);
195 if base_cmd == "sqz"
196 || command.starts_with("SQZ_CMD=")
197 || command.contains("sqz compress --cmd ")
198 || command.contains("sqz.exe compress --cmd ")
199 {
200 return Ok(match platform {
201 HookPlatform::Cursor => "{}".to_string(),
202 _ => input.to_string(),
203 });
204 }
205
206 if is_interactive_command(command) {
208 return Ok(match platform {
209 HookPlatform::Cursor => "{}".to_string(),
210 _ => input.to_string(),
211 });
212 }
213
214 if has_shell_operators(command) {
219 return Ok(match platform {
220 HookPlatform::Cursor => "{}".to_string(),
221 _ => input.to_string(),
222 });
223 }
224
225 let rewritten = format!(
235 "{} 2>&1 | sqz compress --cmd {}",
236 command,
237 shell_escape(extract_base_command(command)),
238 );
239
240 let output = match platform {
266 HookPlatform::ClaudeCode => serde_json::json!({
267 "hookSpecificOutput": {
268 "hookEventName": "PreToolUse",
269 "permissionDecision": "allow",
270 "permissionDecisionReason": "sqz: command output will be compressed for token savings",
271 "updatedInput": {
272 "command": rewritten
273 }
274 }
275 }),
276 HookPlatform::Cursor => serde_json::json!({
277 "permission": "allow",
278 "updated_input": {
279 "command": rewritten
280 }
281 }),
282 HookPlatform::GeminiCli => serde_json::json!({
283 "decision": "allow",
284 "hookSpecificOutput": {
285 "tool_input": {
286 "command": rewritten
287 }
288 }
289 }),
290 HookPlatform::Windsurf => {
291 serde_json::json!({
295 "hookSpecificOutput": {
296 "hookEventName": "PreToolUse",
297 "permissionDecision": "allow",
298 "permissionDecisionReason": "sqz: command output will be compressed for token savings",
299 "updatedInput": {
300 "command": rewritten
301 }
302 }
303 })
304 }
305 };
306
307 serde_json::to_string(&output)
308 .map_err(|e| crate::error::SqzError::Other(format!("hook: JSON serialize error: {e}")))
309}
310
311pub fn generate_hook_configs(sqz_path: &str) -> Vec<ToolHookConfig> {
313 let sqz_path_raw = sqz_path;
324 let sqz_path_json = json_escape_string_value(sqz_path);
325 let sqz_path = &sqz_path_json;
326
327 let escape_hatch = format!(
335 r#"## Escape hatch — when sqz output confuses the agent
336
337If you see a `§ref:HASH§` token in tool output and can't parse it, or if
338compressed output is making it harder to work (loops, repeated tiny
339requests), you have three ways out:
340
341- **`{sqz_path_raw} expand <prefix>`** — resolve a ref back to the
342 original bytes. Accepts the bare hex prefix (`sqz expand a1b2c3d4`)
343 or the full token pasted verbatim (`sqz expand §ref:a1b2c3d4§`).
344- **`SQZ_NO_DEDUP=1`** — set this env var (e.g. `SQZ_NO_DEDUP=1 git
345 status 2>&1 | sqz compress`) to disable dedup for that one command.
346 Strictly more tokens, strictly less ambiguous.
347- **`--no-cache`** — the same opt-out as a CLI flag:
348 `git status 2>&1 | sqz compress --no-cache`.
349
350Use these when you genuinely need raw data. The default compressed
351path is still what you want for most output."#,
352 );
353 let escape_hatch = &escape_hatch;
354
355 vec![
356 ToolHookConfig {
370 tool_name: "Claude Code".to_string(),
371 config_path: PathBuf::from(".claude/settings.local.json"),
372 config_content: format!(
373 r#"{{
374 "hooks": {{
375 "PreToolUse": [
376 {{
377 "matcher": "Bash",
378 "hooks": [
379 {{
380 "type": "command",
381 "command": "{sqz_path} hook claude"
382 }}
383 ]
384 }}
385 ],
386 "PreCompact": [
387 {{
388 "hooks": [
389 {{
390 "type": "command",
391 "command": "{sqz_path} hook precompact"
392 }}
393 ]
394 }}
395 ],
396 "SessionStart": [
397 {{
398 "matcher": "compact",
399 "hooks": [
400 {{
401 "type": "command",
402 "command": "{sqz_path} resume"
403 }}
404 ]
405 }}
406 ]
407 }}
408}}"#
409 ),
410 scope: HookScope::Project,
411 },
412 ToolHookConfig {
421 tool_name: "Cursor".to_string(),
422 config_path: PathBuf::from(".cursor/rules/sqz.mdc"),
423 config_content: format!(
424 r#"---
425description: sqz — Token-Optimized CLI Output
426alwaysApply: true
427---
428
429# sqz — Token-Optimized CLI Output
430
431When running shell commands whose output may be long (directory listings,
432git log/diff, test runners, build logs, `docker ps`, `kubectl get`, etc.),
433pipe the output through `sqz compress` to reduce token consumption.
434
435`sqz` is a stdin-to-stdout compressor, not a command wrapper. The correct
436usage is to pipe the command's output into `sqz compress`:
437
438```bash
439# Instead of: Use:
440git status git status 2>&1 | {sqz_path_raw} compress
441cargo test cargo test 2>&1 | {sqz_path_raw} compress
442git log -10 git log -10 2>&1 | {sqz_path_raw} compress
443docker ps docker ps 2>&1 | {sqz_path_raw} compress
444ls -la ls -la 2>&1 | {sqz_path_raw} compress
445```
446
447The `2>&1` captures stderr too, which is useful for commands like `cargo
448test` where diagnostics go to stderr. `sqz compress` filters and compresses
449the combined output while preserving filenames, paths, and identifiers.
450It typically saves 60-90% tokens on verbose commands.
451
452Do NOT pipe output for:
453- Interactive commands (`vim`, `ssh`, `python`, REPLs)
454- Compound commands with operators (`cmd && other`, `cmd > file.txt`,
455 `cmd; other`) — run those directly
456- Short commands whose output is already a few lines
457
458If `sqz` is not on PATH, run commands normally.
459
460{escape_hatch}
461"#
462 ),
463 scope: HookScope::Project,
464 },
465 ToolHookConfig {
469 tool_name: "Windsurf".to_string(),
470 config_path: PathBuf::from(".windsurfrules"),
471 config_content: format!(
472 r#"# sqz — Token-Optimized CLI Output
473
474Pipe verbose shell command output through `sqz compress` to save tokens.
475`sqz` reads from stdin and writes the compressed output to stdout — it is
476NOT a command wrapper, so `{sqz_path_raw} git status` is not valid.
477
478```bash
479# Instead of: Use:
480git status git status 2>&1 | {sqz_path_raw} compress
481cargo test cargo test 2>&1 | {sqz_path_raw} compress
482git log -10 git log -10 2>&1 | {sqz_path_raw} compress
483docker ps docker ps 2>&1 | {sqz_path_raw} compress
484```
485
486sqz filters and compresses command outputs while preserving filenames,
487paths, and identifiers (typically 60-90% token reduction on verbose
488commands). Skip short commands, interactive commands (vim, ssh, python),
489and commands with shell operators (`&&`, `||`, `;`, `>`, `<`). If sqz is
490not on PATH, run commands normally.
491
492{escape_hatch}
493"#
494 ),
495 scope: HookScope::Project,
496 },
497 ToolHookConfig {
501 tool_name: "Cline".to_string(),
502 config_path: PathBuf::from(".clinerules"),
503 config_content: format!(
504 r#"# sqz — Token-Optimized CLI Output
505
506Pipe verbose shell command output through `sqz compress` to save tokens.
507`sqz` reads from stdin and writes the compressed output to stdout — it is
508NOT a command wrapper, so `{sqz_path_raw} git status` is not valid.
509
510```bash
511# Instead of: Use:
512git status git status 2>&1 | {sqz_path_raw} compress
513cargo test cargo test 2>&1 | {sqz_path_raw} compress
514git log -10 git log -10 2>&1 | {sqz_path_raw} compress
515docker ps docker ps 2>&1 | {sqz_path_raw} compress
516```
517
518sqz filters and compresses command outputs while preserving filenames,
519paths, and identifiers (typically 60-90% token reduction on verbose
520commands). Skip short commands, interactive commands (vim, ssh, python),
521and commands with shell operators (`&&`, `||`, `;`, `>`, `<`). If sqz is
522not on PATH, run commands normally.
523
524{escape_hatch}
525"#
526 ),
527 scope: HookScope::Project,
528 },
529 ToolHookConfig {
531 tool_name: "Gemini CLI".to_string(),
532 config_path: PathBuf::from(".gemini/settings.json"),
533 config_content: format!(
534 r#"{{
535 "hooks": {{
536 "BeforeTool": [
537 {{
538 "matcher": "run_shell_command",
539 "hooks": [
540 {{
541 "type": "command",
542 "command": "{sqz_path} hook gemini"
543 }}
544 ]
545 }}
546 ]
547 }}
548}}"#
549 ),
550 scope: HookScope::Project,
551 },
552 ToolHookConfig {
561 tool_name: "OpenCode".to_string(),
562 config_path: PathBuf::from("opencode.json"),
563 config_content: format!(
564 r#"{{
565 "$schema": "https://opencode.ai/config.json",
566 "mcp": {{
567 "sqz": {{
568 "type": "local",
569 "command": ["sqz-mcp", "--transport", "stdio"]
570 }}
571 }},
572 "plugin": ["sqz"]
573}}"#
574 ),
575 scope: HookScope::Project,
576 },
577 ToolHookConfig {
596 tool_name: "Codex".to_string(),
597 config_path: PathBuf::from("AGENTS.md"),
598 config_content: crate::codex_integration::agents_md_guidance_block(
599 sqz_path_raw,
600 ),
601 scope: HookScope::Project,
602 },
603 ]
604}
605
606pub fn install_tool_hooks(project_dir: &Path, sqz_path: &str) -> Vec<String> {
612 install_tool_hooks_scoped(project_dir, sqz_path, InstallScope::Project)
613}
614
615#[derive(Debug, Clone, Copy, PartialEq, Eq)]
639pub enum InstallScope {
640 Project,
643 Global,
646}
647
648#[derive(Debug, Clone, PartialEq, Eq)]
661pub enum ToolFilter {
662 All,
665 Only(Vec<String>),
669 Skip(Vec<String>),
674}
675
676impl Default for ToolFilter {
677 fn default() -> Self {
678 ToolFilter::All
679 }
680}
681
682impl ToolFilter {
683 pub fn includes(&self, tool_name: &str) -> bool {
694 let canon = canonicalize_tool_name(tool_name);
695 match self {
696 ToolFilter::All => true,
697 ToolFilter::Only(allow) => allow.iter().any(|n| {
698 n == &canon
701 }),
702 ToolFilter::Skip(deny) => !deny.iter().any(|n| n == &canon),
703 }
704 }
705}
706
707pub const SUPPORTED_TOOL_NAMES: &[&str] = &[
712 "Claude Code",
713 "Cursor",
714 "Windsurf",
715 "Cline",
716 "Gemini CLI",
717 "OpenCode",
718 "Codex",
719];
720
721pub fn canonicalize_tool_name(name: &str) -> String {
740 let lowered: String = name
741 .chars()
742 .filter(|c| !c.is_whitespace())
743 .flat_map(|c| c.to_lowercase())
744 .filter(|c| *c != '-' && *c != '_')
745 .collect();
746 match lowered.as_str() {
747 "claude" | "claudecode" => "claudecode".to_string(),
748 "cursor" => "cursor".to_string(),
749 "windsurf" => "windsurf".to_string(),
750 "cline" | "roo" | "roocode" => "cline".to_string(),
754 "gemini" | "geminicli" => "gemini".to_string(),
755 "opencode" => "opencode".to_string(),
756 "codex" => "codex".to_string(),
757 other => other.to_string(),
758 }
759}
760
761pub fn parse_tool_list(raw: &str) -> Result<Vec<String>> {
771 let mut out = Vec::new();
772 let known: std::collections::HashSet<String> = SUPPORTED_TOOL_NAMES
773 .iter()
774 .map(|n| canonicalize_tool_name(n))
775 .collect();
776 for part in raw.split(',') {
777 let trimmed = part.trim();
778 if trimmed.is_empty() {
779 continue;
780 }
781 let canon = canonicalize_tool_name(trimmed);
782 if !known.contains(&canon) {
783 let valid: Vec<String> = SUPPORTED_TOOL_NAMES
784 .iter()
785 .map(|n| canonicalize_tool_name(n))
786 .collect();
787 return Err(crate::error::SqzError::Other(format!(
788 "unknown agent name '{}'. Valid options: {}",
789 trimmed,
790 valid.join(", ")
791 )));
792 }
793 if !out.contains(&canon) {
794 out.push(canon);
795 }
796 }
797 Ok(out)
798}
799
800pub fn install_tool_hooks_scoped(
826 project_dir: &Path,
827 sqz_path: &str,
828 scope: InstallScope,
829) -> Vec<String> {
830 install_tool_hooks_scoped_filtered(project_dir, sqz_path, scope, &ToolFilter::All)
831}
832
833pub fn install_tool_hooks_scoped_filtered(
847 project_dir: &Path,
848 sqz_path: &str,
849 scope: InstallScope,
850 filter: &ToolFilter,
851) -> Vec<String> {
852 let configs = generate_hook_configs(sqz_path);
853 let mut installed = Vec::new();
854
855 for config in &configs {
856 if !filter.includes(&config.tool_name) {
860 continue;
861 }
862
863 if config.tool_name == "OpenCode" {
872 match crate::opencode_plugin::update_opencode_config_detailed(project_dir) {
873 Ok((updated, _comments_lost)) => {
874 if updated && !installed.iter().any(|n| n == "OpenCode") {
875 installed.push("OpenCode".to_string());
876 }
877 }
878 Err(_e) => {
879 }
882 }
883 continue;
884 }
885
886 if config.tool_name == "Codex" {
891 let agents_changed = crate::codex_integration::install_agents_md_guidance(
892 project_dir, sqz_path,
893 )
894 .unwrap_or(false);
895 let mcp_changed = crate::codex_integration::install_codex_mcp_config()
896 .unwrap_or(false);
897 if (agents_changed || mcp_changed)
898 && !installed.iter().any(|n| n == "Codex")
899 {
900 installed.push("Codex".to_string());
901 }
902 continue;
903 }
904
905 if config.tool_name == "Claude Code" && scope == InstallScope::Global {
915 let hook_installed = match install_claude_global(sqz_path) {
916 Ok(v) => v,
917 Err(_) => false,
918 };
919 let md_changed = crate::claude_md_integration::install_claude_md_guidance(
920 project_dir, sqz_path,
921 )
922 .unwrap_or(false);
923 let mcp_changed =
924 crate::claude_md_integration::install_claude_mcp_config()
925 .unwrap_or(false);
926 if (hook_installed || md_changed || mcp_changed)
927 && !installed.iter().any(|n| n == "Claude Code")
928 {
929 installed.push("Claude Code".to_string());
930 }
931 continue;
932 }
933
934 let full_path = project_dir.join(&config.config_path);
935
936 if full_path.exists() {
938 if config.tool_name == "Claude Code" {
943 let md_changed =
944 crate::claude_md_integration::install_claude_md_guidance(
945 project_dir, sqz_path,
946 )
947 .unwrap_or(false);
948 let mcp_changed =
949 crate::claude_md_integration::install_claude_mcp_config()
950 .unwrap_or(false);
951 if (md_changed || mcp_changed)
952 && !installed.iter().any(|n| n == "Claude Code")
953 {
954 installed.push("Claude Code".to_string());
955 }
956 }
957 continue;
958 }
959
960 if let Some(parent) = full_path.parent() {
962 if std::fs::create_dir_all(parent).is_err() {
963 continue;
964 }
965 }
966
967 if std::fs::write(&full_path, &config.config_content).is_ok() {
968 installed.push(config.tool_name.clone());
969 if config.tool_name == "Claude Code" {
976 let _ = crate::claude_md_integration::install_claude_md_guidance(
977 project_dir, sqz_path,
978 );
979 let _ = crate::claude_md_integration::install_claude_mcp_config();
980 }
981 }
982 }
983
984 if filter.includes("OpenCode") {
995 if let Ok(true) = crate::opencode_plugin::install_opencode_plugin(sqz_path) {
996 if !installed.iter().any(|n| n == "OpenCode") {
997 installed.push("OpenCode".to_string());
998 }
999 }
1000 }
1001
1002 installed
1003}
1004
1005pub fn claude_user_settings_path() -> Option<PathBuf> {
1018 dirs_next::home_dir().map(|h| h.join(".claude").join("settings.json"))
1019}
1020
1021fn install_claude_global(sqz_path: &str) -> Result<bool> {
1036 install_claude_global_at(sqz_path, None)
1037}
1038
1039fn install_claude_global_at(sqz_path: &str, home_override: Option<&Path>) -> Result<bool> {
1044 let path = match home_override {
1045 Some(h) => h.join(".claude").join("settings.json"),
1046 None => claude_user_settings_path().ok_or_else(|| {
1047 crate::error::SqzError::Other(
1048 "Could not resolve home directory for ~/.claude/settings.json".to_string(),
1049 )
1050 })?,
1051 };
1052
1053 let mut root: serde_json::Value = if path.exists() {
1055 let content = std::fs::read_to_string(&path).map_err(|e| {
1056 crate::error::SqzError::Other(format!(
1057 "read {}: {e}",
1058 path.display()
1059 ))
1060 })?;
1061 if content.trim().is_empty() {
1062 serde_json::Value::Object(serde_json::Map::new())
1063 } else {
1064 serde_json::from_str(&content).map_err(|e| {
1065 crate::error::SqzError::Other(format!(
1066 "parse {}: {e} — please fix or move the file before re-running sqz init",
1067 path.display()
1068 ))
1069 })?
1070 }
1071 } else {
1072 serde_json::Value::Object(serde_json::Map::new())
1073 };
1074
1075 let root_obj = root.as_object_mut().ok_or_else(|| {
1078 crate::error::SqzError::Other(format!(
1079 "{} is not a JSON object — refusing to overwrite",
1080 path.display()
1081 ))
1082 })?;
1083
1084 let pre_tool_use = serde_json::json!({
1086 "matcher": "Bash",
1087 "hooks": [{ "type": "command", "command": format!("{sqz_path} hook claude") }]
1088 });
1089 let pre_compact = serde_json::json!({
1090 "hooks": [{ "type": "command", "command": format!("{sqz_path} hook precompact") }]
1091 });
1092 let session_start = serde_json::json!({
1093 "matcher": "compact",
1094 "hooks": [{ "type": "command", "command": format!("{sqz_path} resume") }]
1095 });
1096
1097 let before = serde_json::to_string(&root_obj).unwrap_or_default();
1099
1100 let hooks = root_obj
1102 .entry("hooks".to_string())
1103 .or_insert_with(|| serde_json::Value::Object(serde_json::Map::new()));
1104 let hooks_obj = hooks.as_object_mut().ok_or_else(|| {
1105 crate::error::SqzError::Other(format!(
1106 "{}: `hooks` is not an object — refusing to overwrite",
1107 path.display()
1108 ))
1109 })?;
1110
1111 upsert_sqz_hook_entry(hooks_obj, "PreToolUse", pre_tool_use, "sqz hook claude");
1112 upsert_sqz_hook_entry(hooks_obj, "PreCompact", pre_compact, "sqz hook precompact");
1113 upsert_sqz_hook_entry(hooks_obj, "SessionStart", session_start, "sqz resume");
1114
1115 let after = serde_json::to_string(&root_obj).unwrap_or_default();
1116 if before == after && path.exists() {
1117 return Ok(false);
1119 }
1120
1121 if let Some(parent) = path.parent() {
1123 std::fs::create_dir_all(parent).map_err(|e| {
1124 crate::error::SqzError::Other(format!(
1125 "create {}: {e}",
1126 parent.display()
1127 ))
1128 })?;
1129 }
1130
1131 let parent = path.parent().ok_or_else(|| {
1135 crate::error::SqzError::Other(format!(
1136 "path {} has no parent directory",
1137 path.display()
1138 ))
1139 })?;
1140 let tmp = tempfile::NamedTempFile::new_in(parent).map_err(|e| {
1141 crate::error::SqzError::Other(format!(
1142 "create temp file in {}: {e}",
1143 parent.display()
1144 ))
1145 })?;
1146 let serialized = serde_json::to_string_pretty(&serde_json::Value::Object(root_obj.clone()))
1147 .map_err(|e| crate::error::SqzError::Other(format!("serialize settings.json: {e}")))?;
1148 std::fs::write(tmp.path(), serialized).map_err(|e| {
1149 crate::error::SqzError::Other(format!(
1150 "write to temp file {}: {e}",
1151 tmp.path().display()
1152 ))
1153 })?;
1154 tmp.persist(&path).map_err(|e| {
1155 crate::error::SqzError::Other(format!(
1156 "rename temp file into place at {}: {e}",
1157 path.display()
1158 ))
1159 })?;
1160
1161 Ok(true)
1162}
1163
1164pub fn remove_claude_global_hook() -> Result<Option<(PathBuf, bool)>> {
1178 remove_claude_global_hook_at(None)
1179}
1180
1181fn remove_claude_global_hook_at(
1184 home_override: Option<&Path>,
1185) -> Result<Option<(PathBuf, bool)>> {
1186 let path = match home_override {
1187 Some(h) => h.join(".claude").join("settings.json"),
1188 None => match claude_user_settings_path() {
1189 Some(p) => p,
1190 None => return Ok(None),
1191 },
1192 };
1193 if !path.exists() {
1194 return Ok(None);
1195 }
1196
1197 let content = std::fs::read_to_string(&path).map_err(|e| {
1198 crate::error::SqzError::Other(format!("read {}: {e}", path.display()))
1199 })?;
1200 if content.trim().is_empty() {
1201 return Ok(Some((path, false)));
1202 }
1203
1204 let mut root: serde_json::Value = serde_json::from_str(&content).map_err(|e| {
1205 crate::error::SqzError::Other(format!(
1206 "parse {}: {e} — refusing to rewrite an unparseable file",
1207 path.display()
1208 ))
1209 })?;
1210 let Some(root_obj) = root.as_object_mut() else {
1211 return Ok(Some((path, false)));
1212 };
1213
1214 let mut changed = false;
1215 if let Some(hooks) = root_obj.get_mut("hooks").and_then(|h| h.as_object_mut()) {
1216 for (event, sentinel) in &[
1217 ("PreToolUse", "sqz hook claude"),
1218 ("PreCompact", "sqz hook precompact"),
1219 ("SessionStart", "sqz resume"),
1220 ] {
1221 if let Some(arr) = hooks.get_mut(*event).and_then(|v| v.as_array_mut()) {
1222 let before = arr.len();
1223 arr.retain(|entry| !hook_entry_command_contains(entry, sentinel));
1224 if arr.len() != before {
1225 changed = true;
1226 }
1227 }
1228 }
1229
1230 hooks.retain(|_, v| match v {
1233 serde_json::Value::Array(a) => !a.is_empty(),
1234 _ => true,
1235 });
1236
1237 let hooks_empty = hooks.is_empty();
1240 if hooks_empty {
1241 root_obj.remove("hooks");
1242 changed = true;
1243 }
1244 }
1245
1246 if !changed {
1247 return Ok(Some((path, false)));
1248 }
1249
1250 if root_obj.is_empty() {
1254 std::fs::remove_file(&path).map_err(|e| {
1255 crate::error::SqzError::Other(format!(
1256 "remove {}: {e}",
1257 path.display()
1258 ))
1259 })?;
1260 return Ok(Some((path, true)));
1261 }
1262
1263 let parent = path.parent().ok_or_else(|| {
1265 crate::error::SqzError::Other(format!(
1266 "path {} has no parent directory",
1267 path.display()
1268 ))
1269 })?;
1270 let tmp = tempfile::NamedTempFile::new_in(parent).map_err(|e| {
1271 crate::error::SqzError::Other(format!(
1272 "create temp file in {}: {e}",
1273 parent.display()
1274 ))
1275 })?;
1276 let serialized = serde_json::to_string_pretty(&serde_json::Value::Object(root_obj.clone()))
1277 .map_err(|e| {
1278 crate::error::SqzError::Other(format!("serialize settings.json: {e}"))
1279 })?;
1280 std::fs::write(tmp.path(), serialized).map_err(|e| {
1281 crate::error::SqzError::Other(format!(
1282 "write to temp file {}: {e}",
1283 tmp.path().display()
1284 ))
1285 })?;
1286 tmp.persist(&path).map_err(|e| {
1287 crate::error::SqzError::Other(format!(
1288 "rename temp file into place at {}: {e}",
1289 path.display()
1290 ))
1291 })?;
1292
1293 Ok(Some((path, true)))
1294}
1295
1296fn upsert_sqz_hook_entry(
1303 hooks_obj: &mut serde_json::Map<String, serde_json::Value>,
1304 event_name: &str,
1305 new_entry: serde_json::Value,
1306 sentinel: &str,
1307) {
1308 let arr = hooks_obj
1309 .entry(event_name.to_string())
1310 .or_insert_with(|| serde_json::Value::Array(Vec::new()));
1311 let Some(arr) = arr.as_array_mut() else {
1312 hooks_obj.insert(
1316 event_name.to_string(),
1317 serde_json::Value::Array(vec![new_entry]),
1318 );
1319 return;
1320 };
1321
1322 arr.retain(|entry| !hook_entry_command_contains(entry, sentinel));
1324
1325 arr.push(new_entry);
1326}
1327
1328fn hook_entry_command_contains(entry: &serde_json::Value, needle: &str) -> bool {
1332 entry
1333 .get("hooks")
1334 .and_then(|h| h.as_array())
1335 .map(|hooks_arr| {
1336 hooks_arr.iter().any(|h| {
1337 h.get("command")
1338 .and_then(|c| c.as_str())
1339 .map(|c| c.contains(needle))
1340 .unwrap_or(false)
1341 })
1342 })
1343 .unwrap_or(false)
1344}
1345
1346fn extract_base_command(cmd: &str) -> &str {
1350 cmd.split_whitespace()
1351 .next()
1352 .unwrap_or("unknown")
1353 .rsplit('/')
1354 .next()
1355 .unwrap_or("unknown")
1356}
1357
1358pub(crate) fn json_escape_string_value(s: &str) -> String {
1369 let mut out = String::with_capacity(s.len() + 2);
1370 for ch in s.chars() {
1371 match ch {
1372 '\\' => out.push_str("\\\\"),
1373 '"' => out.push_str("\\\""),
1374 '\n' => out.push_str("\\n"),
1375 '\r' => out.push_str("\\r"),
1376 '\t' => out.push_str("\\t"),
1377 '\x08' => out.push_str("\\b"),
1378 '\x0c' => out.push_str("\\f"),
1379 c if (c as u32) < 0x20 => {
1380 out.push_str(&format!("\\u{:04x}", c as u32));
1382 }
1383 c => out.push(c),
1384 }
1385 }
1386 out
1387}
1388
1389fn shell_escape(s: &str) -> String {
1391 if s.chars().all(|c| c.is_ascii_alphanumeric() || c == '_' || c == '-' || c == '.') {
1392 s.to_string()
1393 } else {
1394 format!("'{}'", s.replace('\'', "'\\''"))
1395 }
1396}
1397
1398fn has_shell_operators(cmd: &str) -> bool {
1402 cmd.contains("&&")
1405 || cmd.contains("||")
1406 || cmd.contains(';')
1407 || cmd.contains('>')
1408 || cmd.contains('<')
1409 || cmd.contains('|') || cmd.contains('&') && !cmd.contains("&&") || cmd.contains("<<") || cmd.contains("$(") || cmd.contains('`') }
1415
1416fn is_interactive_command(cmd: &str) -> bool {
1418 let base = extract_base_command(cmd);
1419 matches!(
1420 base,
1421 "vim" | "vi" | "nano" | "emacs" | "less" | "more" | "top" | "htop"
1422 | "ssh" | "python" | "python3" | "node" | "irb" | "ghci"
1423 | "psql" | "mysql" | "sqlite3" | "mongo" | "redis-cli"
1424 ) || cmd.contains("--watch")
1425 || cmd.contains("-w ")
1426 || cmd.ends_with(" -w")
1427 || cmd.contains("run dev")
1428 || cmd.contains("run start")
1429 || cmd.contains("run serve")
1430}
1431
1432#[cfg(test)]
1435mod tests {
1436 use super::*;
1437
1438 #[test]
1439 fn test_process_hook_rewrites_bash_command() {
1440 let input = r#"{"tool_name":"Bash","tool_input":{"command":"git status"}}"#;
1442 let result = process_hook(input).unwrap();
1443 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1444 let hook_output = &parsed["hookSpecificOutput"];
1446 assert_eq!(hook_output["hookEventName"].as_str().unwrap(), "PreToolUse");
1447 assert_eq!(hook_output["permissionDecision"].as_str().unwrap(), "allow");
1448 let cmd = hook_output["updatedInput"]["command"].as_str().unwrap();
1450 assert!(cmd.contains("sqz compress"), "should pipe through sqz: {cmd}");
1451 assert!(cmd.contains("git status"), "should preserve original command: {cmd}");
1452 assert!(cmd.contains("--cmd git"), "should pass base command as --cmd: {cmd}");
1455 assert!(
1456 !cmd.contains("SQZ_CMD="),
1457 "new rewrites must not emit the legacy sh-style env prefix: {cmd}"
1458 );
1459 assert!(parsed.get("decision").is_none(), "Claude Code format should not have top-level decision");
1461 assert!(parsed.get("permission").is_none(), "Claude Code format should not have top-level permission");
1462 assert!(parsed.get("continue").is_none(), "Claude Code format should not have top-level continue");
1463 }
1464
1465 #[test]
1466 fn test_process_hook_passes_through_non_bash() {
1467 let input = r#"{"tool_name":"Read","tool_input":{"file_path":"file.txt"}}"#;
1468 let result = process_hook(input).unwrap();
1469 assert_eq!(result, input, "non-bash tools should pass through unchanged");
1470 }
1471
1472 #[test]
1473 fn test_process_hook_skips_sqz_commands() {
1474 let input = r#"{"tool_name":"Bash","tool_input":{"command":"sqz stats"}}"#;
1475 let result = process_hook(input).unwrap();
1476 assert_eq!(result, input, "sqz commands should not be double-wrapped");
1477 }
1478
1479 #[test]
1480 fn test_process_hook_skips_interactive() {
1481 let input = r#"{"tool_name":"Bash","tool_input":{"command":"vim file.txt"}}"#;
1482 let result = process_hook(input).unwrap();
1483 assert_eq!(result, input, "interactive commands should pass through");
1484 }
1485
1486 #[test]
1487 fn test_process_hook_skips_watch_mode() {
1488 let input = r#"{"tool_name":"Bash","tool_input":{"command":"npm run dev --watch"}}"#;
1489 let result = process_hook(input).unwrap();
1490 assert_eq!(result, input, "watch mode should pass through");
1491 }
1492
1493 #[test]
1494 fn test_process_hook_empty_command() {
1495 let input = r#"{"tool_name":"Bash","tool_input":{"command":""}}"#;
1496 let result = process_hook(input).unwrap();
1497 assert_eq!(result, input);
1498 }
1499
1500 #[test]
1501 fn test_process_hook_gemini_format() {
1502 let input = r#"{"tool_name":"run_shell_command","tool_input":{"command":"git log"}}"#;
1504 let result = process_hook_gemini(input).unwrap();
1505 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1506 assert_eq!(parsed["decision"].as_str().unwrap(), "allow");
1508 let cmd = parsed["hookSpecificOutput"]["tool_input"]["command"].as_str().unwrap();
1510 assert!(cmd.contains("sqz compress"), "should pipe through sqz: {cmd}");
1511 assert!(parsed.get("hookSpecificOutput").unwrap().get("updatedInput").is_none(),
1513 "Gemini format should not have updatedInput");
1514 assert!(parsed.get("hookSpecificOutput").unwrap().get("permissionDecision").is_none(),
1515 "Gemini format should not have permissionDecision");
1516 }
1517
1518 #[test]
1519 fn test_process_hook_legacy_format() {
1520 let input = r#"{"toolName":"Bash","toolCall":{"command":"git status"}}"#;
1522 let result = process_hook(input).unwrap();
1523 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1524 let cmd = parsed["hookSpecificOutput"]["updatedInput"]["command"].as_str().unwrap();
1525 assert!(cmd.contains("sqz compress"), "legacy format should still work: {cmd}");
1526 }
1527
1528 #[test]
1529 fn test_process_hook_cursor_format() {
1530 let input = r#"{"tool_name":"Shell","tool_input":{"command":"git status"},"conversation_id":"abc"}"#;
1532 let result = process_hook_cursor(input).unwrap();
1533 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1534 assert_eq!(parsed["permission"].as_str().unwrap(), "allow");
1536 let cmd = parsed["updated_input"]["command"].as_str().unwrap();
1537 assert!(cmd.contains("sqz compress"), "cursor format should work: {cmd}");
1538 assert!(cmd.contains("git status"));
1539 assert!(parsed.get("hookSpecificOutput").is_none(),
1541 "Cursor format should not have hookSpecificOutput");
1542 }
1543
1544 #[test]
1545 fn test_process_hook_cursor_passthrough_returns_empty_json() {
1546 let input = r#"{"tool_name":"Read","tool_input":{"file_path":"file.txt"}}"#;
1548 let result = process_hook_cursor(input).unwrap();
1549 assert_eq!(result, "{}", "Cursor passthrough must return empty JSON object");
1550 }
1551
1552 #[test]
1553 fn test_process_hook_cursor_no_rewrite_returns_empty_json() {
1554 let input = r#"{"tool_name":"Shell","tool_input":{"command":"sqz stats"}}"#;
1556 let result = process_hook_cursor(input).unwrap();
1557 assert_eq!(result, "{}", "Cursor no-rewrite must return empty JSON object");
1558 }
1559
1560 #[test]
1561 fn test_process_hook_windsurf_format() {
1562 let input = r#"{"agent_action_name":"pre_run_command","tool_info":{"command_line":"cargo test","cwd":"/project"}}"#;
1564 let result = process_hook_windsurf(input).unwrap();
1565 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1566 let cmd = parsed["hookSpecificOutput"]["updatedInput"]["command"].as_str().unwrap();
1568 assert!(cmd.contains("sqz compress"), "windsurf format should work: {cmd}");
1569 assert!(cmd.contains("cargo test"));
1570 assert!(cmd.contains("--cmd cargo"), "label must be passed via --cmd flag");
1572 assert!(!cmd.contains("SQZ_CMD="), "must not emit legacy env prefix: {cmd}");
1573 }
1574
1575 #[test]
1576 fn test_process_hook_invalid_json() {
1577 let result = process_hook("not json");
1578 assert!(result.is_err());
1579 }
1580
1581 #[test]
1582 fn test_extract_base_command() {
1583 assert_eq!(extract_base_command("git status"), "git");
1584 assert_eq!(extract_base_command("/usr/bin/git log"), "git");
1585 assert_eq!(extract_base_command("cargo test --release"), "cargo");
1586 }
1587
1588 #[test]
1589 fn test_is_interactive_command() {
1590 assert!(is_interactive_command("vim file.txt"));
1591 assert!(is_interactive_command("npm run dev --watch"));
1592 assert!(is_interactive_command("python3"));
1593 assert!(!is_interactive_command("git status"));
1594 assert!(!is_interactive_command("cargo test"));
1595 }
1596
1597 #[test]
1612 fn issue_10_rewrite_is_shell_neutral() {
1613 let input = r#"{"tool_name":"Bash","tool_input":{"command":"dotnet build NewNeonCheckers3.sln"}}"#;
1614 let result = process_hook(input).unwrap();
1615 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1616 let cmd = parsed["hookSpecificOutput"]["updatedInput"]["command"]
1617 .as_str()
1618 .unwrap();
1619
1620 assert!(
1622 cmd.contains("--cmd dotnet"),
1623 "issue #10: rewrite must pass label via --cmd, got: {cmd}"
1624 );
1625 assert!(
1627 !cmd.contains("SQZ_CMD="),
1628 "issue #10: rewrite must NOT emit `SQZ_CMD=` prefix \
1629 (broken in PowerShell and cmd.exe), got: {cmd}"
1630 );
1631 assert!(
1633 cmd.contains("dotnet build NewNeonCheckers3.sln"),
1634 "original command must be preserved verbatim: {cmd}"
1635 );
1636 assert!(cmd.contains("| sqz compress"), "must pipe through sqz: {cmd}");
1638 }
1639
1640 #[test]
1648 fn issue_10_already_wrapped_command_passes_through() {
1649 let input = r#"{"tool_name":"Bash","tool_input":{"command":"git status 2>&1 | sqz compress --cmd git"}}"#;
1650 let result = process_hook(input).unwrap();
1651 let parsed: serde_json::Value = serde_json::from_str(&result).unwrap();
1652 assert_eq!(
1655 result, input,
1656 "already-wrapped command must pass through unchanged; \
1657 otherwise each pass accumulates another `| sqz compress` tail"
1658 );
1659 let _ = parsed; }
1664
1665 #[test]
1666 fn test_generate_hook_configs() {
1667 let configs = generate_hook_configs("sqz");
1668 assert!(configs.len() >= 5, "should generate configs for multiple tools (including OpenCode)");
1669 assert!(configs.iter().any(|c| c.tool_name == "Claude Code"));
1670 assert!(configs.iter().any(|c| c.tool_name == "Cursor"));
1671 assert!(configs.iter().any(|c| c.tool_name == "OpenCode"));
1672 let windsurf = configs.iter().find(|c| c.tool_name == "Windsurf").unwrap();
1675 assert_eq!(windsurf.config_path, PathBuf::from(".windsurfrules"),
1676 "Windsurf should use .windsurfrules, not .windsurf/hooks.json");
1677 let cline = configs.iter().find(|c| c.tool_name == "Cline").unwrap();
1678 assert_eq!(cline.config_path, PathBuf::from(".clinerules"),
1679 "Cline should use .clinerules, not .clinerules/hooks/PreToolUse");
1680 let cursor = configs.iter().find(|c| c.tool_name == "Cursor").unwrap();
1684 assert_eq!(cursor.config_path, PathBuf::from(".cursor/rules/sqz.mdc"),
1685 "Cursor should use .cursor/rules/sqz.mdc (modern rules), not \
1686 .cursor/hooks.json (non-functional) or .cursorrules (legacy)");
1687 assert!(cursor.config_content.starts_with("---"),
1688 "Cursor rule should start with YAML frontmatter");
1689 assert!(cursor.config_content.contains("alwaysApply: true"),
1690 "Cursor rule should use alwaysApply: true so the guidance loads \
1691 for every agent interaction");
1692 assert!(cursor.config_content.contains("sqz"),
1693 "Cursor rule body should mention sqz");
1694 }
1695
1696 #[test]
1697 fn test_claude_config_includes_precompact_hook() {
1698 let configs = generate_hook_configs("sqz");
1703 let claude = configs.iter().find(|c| c.tool_name == "Claude Code").unwrap();
1704 let parsed: serde_json::Value = serde_json::from_str(&claude.config_content)
1705 .expect("Claude Code config must be valid JSON");
1706
1707 let precompact = parsed["hooks"]["PreCompact"]
1708 .as_array()
1709 .expect("PreCompact hook array must be present");
1710 assert!(
1711 !precompact.is_empty(),
1712 "PreCompact must have at least one registered hook"
1713 );
1714
1715 let cmd = precompact[0]["hooks"][0]["command"]
1716 .as_str()
1717 .expect("command field must be a string");
1718 assert!(
1719 cmd.ends_with(" hook precompact"),
1720 "PreCompact hook should invoke `sqz hook precompact`; got: {cmd}"
1721 );
1722 }
1723
1724 #[test]
1727 fn test_json_escape_string_value() {
1728 assert_eq!(json_escape_string_value("sqz"), "sqz");
1730 assert_eq!(json_escape_string_value("/usr/local/bin/sqz"), "/usr/local/bin/sqz");
1731 assert_eq!(json_escape_string_value(r"C:\Users\Alice\sqz.exe"),
1733 r"C:\\Users\\Alice\\sqz.exe");
1734 assert_eq!(json_escape_string_value(r#"path with "quotes""#),
1736 r#"path with \"quotes\""#);
1737 assert_eq!(json_escape_string_value("a\nb\tc"), r"a\nb\tc");
1739 }
1740
1741 #[test]
1742 fn test_windows_path_produces_valid_json_for_claude() {
1743 let windows_path = r"C:\Users\SqzUser\.cargo\bin\sqz.exe";
1746 let configs = generate_hook_configs(windows_path);
1747
1748 let claude = configs.iter().find(|c| c.tool_name == "Claude Code")
1749 .expect("Claude config should be generated");
1750 let parsed: serde_json::Value = serde_json::from_str(&claude.config_content)
1751 .expect("Claude hook config must be valid JSON on Windows paths");
1752
1753 let cmd = parsed["hooks"]["PreToolUse"][0]["hooks"][0]["command"]
1755 .as_str()
1756 .expect("command field must be a string");
1757 assert!(cmd.contains(windows_path),
1758 "command '{cmd}' must contain the original Windows path '{windows_path}'");
1759 }
1760
1761 #[test]
1762 fn test_windows_path_in_cursor_rules_file() {
1763 let windows_path = r"C:\Users\SqzUser\.cargo\bin\sqz.exe";
1769 let configs = generate_hook_configs(windows_path);
1770
1771 let cursor = configs.iter().find(|c| c.tool_name == "Cursor").unwrap();
1772 assert_eq!(cursor.config_path, PathBuf::from(".cursor/rules/sqz.mdc"));
1773 assert!(cursor.config_content.contains(windows_path),
1774 "Cursor rule must contain the raw (unescaped) path so users can \
1775 copy-paste the shown commands — got:\n{}", cursor.config_content);
1776 assert!(!cursor.config_content.contains(r"C:\\Users"),
1777 "Cursor rule must NOT double-escape backslashes in markdown — \
1778 got:\n{}", cursor.config_content);
1779 }
1780
1781 #[test]
1782 fn test_windows_path_produces_valid_json_for_gemini() {
1783 let windows_path = r"C:\Users\SqzUser\.cargo\bin\sqz.exe";
1784 let configs = generate_hook_configs(windows_path);
1785
1786 let gemini = configs.iter().find(|c| c.tool_name == "Gemini CLI").unwrap();
1787 let parsed: serde_json::Value = serde_json::from_str(&gemini.config_content)
1788 .expect("Gemini hook config must be valid JSON on Windows paths");
1789 let cmd = parsed["hooks"]["BeforeTool"][0]["hooks"][0]["command"].as_str().unwrap();
1790 assert!(cmd.contains(windows_path));
1791 }
1792
1793 #[test]
1794 fn test_rules_files_use_raw_path_for_readability() {
1795 let windows_path = r"C:\Users\SqzUser\.cargo\bin\sqz.exe";
1799 let configs = generate_hook_configs(windows_path);
1800
1801 for tool in &["Windsurf", "Cline", "Cursor"] {
1802 let cfg = configs.iter().find(|c| &c.tool_name == tool).unwrap();
1803 assert!(cfg.config_content.contains(windows_path),
1804 "{tool} rules file must contain the raw (unescaped) path — got:\n{}",
1805 cfg.config_content);
1806 assert!(!cfg.config_content.contains(r"C:\\Users"),
1807 "{tool} rules file must NOT double-escape backslashes — got:\n{}",
1808 cfg.config_content);
1809 }
1810 }
1811
1812 #[test]
1813 fn test_unix_path_still_works() {
1814 let unix_path = "/usr/local/bin/sqz";
1817 let configs = generate_hook_configs(unix_path);
1818
1819 let claude = configs.iter().find(|c| c.tool_name == "Claude Code").unwrap();
1820 let parsed: serde_json::Value = serde_json::from_str(&claude.config_content)
1821 .expect("Unix path should produce valid JSON");
1822 let cmd = parsed["hooks"]["PreToolUse"][0]["hooks"][0]["command"].as_str().unwrap();
1823 assert_eq!(cmd, "/usr/local/bin/sqz hook claude");
1824 }
1825
1826 #[test]
1827 fn test_shell_escape_simple() {
1828 assert_eq!(shell_escape("git"), "git");
1829 assert_eq!(shell_escape("cargo-test"), "cargo-test");
1830 }
1831
1832 #[test]
1833 fn test_shell_escape_special_chars() {
1834 assert_eq!(shell_escape("git log --oneline"), "'git log --oneline'");
1835 }
1836
1837 #[test]
1838 fn test_install_tool_hooks_creates_files() {
1839 let dir = tempfile::tempdir().unwrap();
1840 let installed = install_tool_hooks(dir.path(), "sqz");
1841 assert!(!installed.is_empty(), "should install at least one hook config");
1843 for name in &installed {
1845 let configs = generate_hook_configs("sqz");
1846 let config = configs.iter().find(|c| &c.tool_name == name).unwrap();
1847 let path = dir.path().join(&config.config_path);
1848 assert!(path.exists(), "hook config should exist: {}", path.display());
1849 }
1850 }
1851
1852 #[test]
1853 fn test_install_tool_hooks_does_not_overwrite() {
1854 let dir = tempfile::tempdir().unwrap();
1855 install_tool_hooks(dir.path(), "sqz");
1857 let custom_path = dir.path().join(".claude/settings.local.json");
1859 std::fs::write(&custom_path, "custom content").unwrap();
1860 install_tool_hooks(dir.path(), "sqz");
1862 let content = std::fs::read_to_string(&custom_path).unwrap();
1863 assert_eq!(content, "custom content", "should not overwrite existing config");
1864 }
1865}
1866
1867#[cfg(test)]
1868mod global_install_tests {
1869 use super::*;
1870
1871 #[test]
1872 fn global_install_creates_fresh_settings_json() {
1873 let tmp = tempfile::tempdir().unwrap();
1874 let changed = install_claude_global_at("/usr/local/bin/sqz", Some(tmp.path())).unwrap();
1875 assert!(changed, "first install should report a change");
1876
1877 let path = tmp.path().join(".claude").join("settings.json");
1878 assert!(path.exists(), "user settings.json should be created");
1879
1880 let content = std::fs::read_to_string(&path).unwrap();
1881 let parsed: serde_json::Value = serde_json::from_str(&content).unwrap();
1882
1883 let pre = &parsed["hooks"]["PreToolUse"];
1885 assert!(pre.is_array(), "PreToolUse should be an array");
1886 assert_eq!(pre.as_array().unwrap().len(), 1);
1887 let cmd = pre[0]["hooks"][0]["command"].as_str().unwrap();
1888 assert!(
1889 cmd.contains("/usr/local/bin/sqz"),
1890 "hook command should use the passed sqz_path, got: {cmd}"
1891 );
1892 assert!(cmd.contains("hook claude"));
1893
1894 let precompact = &parsed["hooks"]["PreCompact"];
1895 assert!(precompact.is_array());
1896 let precompact_cmd = precompact[0]["hooks"][0]["command"].as_str().unwrap();
1897 assert!(precompact_cmd.contains("hook precompact"));
1898
1899 let session = &parsed["hooks"]["SessionStart"];
1900 assert!(session.is_array());
1901 assert_eq!(
1902 session[0]["matcher"].as_str().unwrap(),
1903 "compact",
1904 "SessionStart should only match /compact resume"
1905 );
1906 }
1907
1908 #[test]
1909 fn global_install_preserves_existing_user_config() {
1910 let tmp = tempfile::tempdir().unwrap();
1911 let settings = tmp.path().join(".claude").join("settings.json");
1912 std::fs::create_dir_all(settings.parent().unwrap()).unwrap();
1913
1914 let existing = serde_json::json!({
1915 "permissions": {
1916 "allow": ["Bash(npm test *)"],
1917 "deny": ["Read(./.env)"]
1918 },
1919 "env": { "FOO": "bar" },
1920 "statusLine": {
1921 "type": "command",
1922 "command": "~/.claude/statusline.sh"
1923 },
1924 "hooks": {
1925 "PreToolUse": [
1926 {
1927 "matcher": "Edit",
1928 "hooks": [
1929 {
1930 "type": "command",
1931 "command": "~/.claude/hooks/format-on-edit.sh"
1932 }
1933 ]
1934 }
1935 ]
1936 }
1937 });
1938 std::fs::write(&settings, serde_json::to_string_pretty(&existing).unwrap()).unwrap();
1939
1940 let changed = install_claude_global_at("/usr/local/bin/sqz", Some(tmp.path())).unwrap();
1941 assert!(changed, "install should report a change on new hook");
1942
1943 let content = std::fs::read_to_string(&settings).unwrap();
1944 let parsed: serde_json::Value = serde_json::from_str(&content).unwrap();
1945
1946 assert_eq!(
1948 parsed["permissions"]["allow"][0].as_str().unwrap(),
1949 "Bash(npm test *)"
1950 );
1951 assert_eq!(
1952 parsed["permissions"]["deny"][0].as_str().unwrap(),
1953 "Read(./.env)"
1954 );
1955 assert_eq!(parsed["env"]["FOO"].as_str().unwrap(), "bar");
1957 assert_eq!(
1959 parsed["statusLine"]["command"].as_str().unwrap(),
1960 "~/.claude/statusline.sh"
1961 );
1962
1963 let pre = parsed["hooks"]["PreToolUse"].as_array().unwrap();
1966 assert_eq!(pre.len(), 2, "expected user's hook + sqz's hook, got: {pre:?}");
1967 let matchers: Vec<&str> = pre
1968 .iter()
1969 .map(|e| e["matcher"].as_str().unwrap_or(""))
1970 .collect();
1971 assert!(matchers.contains(&"Edit"), "user's Edit hook must survive");
1972 assert!(matchers.contains(&"Bash"), "sqz Bash hook must be present");
1973 }
1974
1975 #[test]
1976 fn global_install_is_idempotent() {
1977 let tmp = tempfile::tempdir().unwrap();
1978 assert!(install_claude_global_at("sqz", Some(tmp.path())).unwrap());
1979 assert!(
1980 !install_claude_global_at("sqz", Some(tmp.path())).unwrap(),
1981 "second install with identical args should report no change"
1982 );
1983
1984 let path = tmp.path().join(".claude").join("settings.json");
1985 let parsed: serde_json::Value =
1986 serde_json::from_str(&std::fs::read_to_string(&path).unwrap()).unwrap();
1987 for event in &["PreToolUse", "PreCompact", "SessionStart"] {
1988 let arr = parsed["hooks"][event].as_array().unwrap();
1989 assert_eq!(
1990 arr.len(),
1991 1,
1992 "{event} must have exactly one sqz entry after 2 installs, got {arr:?}"
1993 );
1994 }
1995 }
1996
1997 #[test]
1998 fn global_install_upgrades_stale_sqz_hook_in_place() {
1999 let tmp = tempfile::tempdir().unwrap();
2000 install_claude_global_at("/old/path/sqz", Some(tmp.path())).unwrap();
2001 let changed = install_claude_global_at("/new/path/sqz", Some(tmp.path())).unwrap();
2002 assert!(changed, "different sqz_path must be seen as a change");
2003
2004 let path = tmp.path().join(".claude").join("settings.json");
2005 let parsed: serde_json::Value =
2006 serde_json::from_str(&std::fs::read_to_string(&path).unwrap()).unwrap();
2007 let pre = parsed["hooks"]["PreToolUse"].as_array().unwrap();
2008 assert_eq!(pre.len(), 1, "stale sqz entry must be replaced, not duplicated");
2009 let cmd = pre[0]["hooks"][0]["command"].as_str().unwrap();
2010 assert!(cmd.contains("/new/path/sqz"));
2011 assert!(!cmd.contains("/old/path/sqz"));
2012 }
2013
2014 #[test]
2015 fn global_uninstall_removes_sqz_and_preserves_the_rest() {
2016 let tmp = tempfile::tempdir().unwrap();
2017 let settings = tmp.path().join(".claude").join("settings.json");
2018 std::fs::create_dir_all(settings.parent().unwrap()).unwrap();
2019 std::fs::write(
2020 &settings,
2021 serde_json::json!({
2022 "permissions": { "allow": ["Bash(git status)"] },
2023 "hooks": {
2024 "PreToolUse": [
2025 {
2026 "matcher": "Edit",
2027 "hooks": [
2028 { "type": "command", "command": "~/format.sh" }
2029 ]
2030 }
2031 ]
2032 }
2033 })
2034 .to_string(),
2035 )
2036 .unwrap();
2037
2038 install_claude_global_at("/usr/local/bin/sqz", Some(tmp.path())).unwrap();
2039 let result = remove_claude_global_hook_at(Some(tmp.path())).unwrap().unwrap();
2040 assert_eq!(result.0, settings);
2041 assert!(result.1, "should report that the file was modified");
2042
2043 assert!(settings.exists(), "settings.json should be preserved");
2044 let parsed: serde_json::Value =
2045 serde_json::from_str(&std::fs::read_to_string(&settings).unwrap()).unwrap();
2046
2047 assert_eq!(
2048 parsed["permissions"]["allow"][0].as_str().unwrap(),
2049 "Bash(git status)"
2050 );
2051
2052 let pre = parsed["hooks"]["PreToolUse"].as_array().unwrap();
2053 assert_eq!(pre.len(), 1, "only the user's Edit hook should remain");
2054 assert_eq!(pre[0]["matcher"].as_str().unwrap(), "Edit");
2055
2056 assert!(parsed["hooks"].get("PreCompact").is_none());
2057 assert!(parsed["hooks"].get("SessionStart").is_none());
2058 }
2059
2060 #[test]
2061 fn global_uninstall_deletes_settings_json_if_it_was_sqz_only() {
2062 let tmp = tempfile::tempdir().unwrap();
2063 install_claude_global_at("sqz", Some(tmp.path())).unwrap();
2064 let path = tmp.path().join(".claude").join("settings.json");
2065 assert!(path.exists(), "precondition: install created the file");
2066
2067 let result = remove_claude_global_hook_at(Some(tmp.path())).unwrap().unwrap();
2068 assert!(result.1);
2069 assert!(!path.exists(), "sqz-only settings.json should be removed on uninstall");
2070 }
2071
2072 #[test]
2073 fn global_uninstall_on_missing_file_is_noop() {
2074 let tmp = tempfile::tempdir().unwrap();
2075 assert!(
2076 remove_claude_global_hook_at(Some(tmp.path())).unwrap().is_none(),
2077 "missing file should return None, not error"
2078 );
2079 }
2080
2081 #[test]
2082 fn global_uninstall_refuses_to_touch_unparseable_file() {
2083 let tmp = tempfile::tempdir().unwrap();
2084 let settings = tmp.path().join(".claude").join("settings.json");
2085 std::fs::create_dir_all(settings.parent().unwrap()).unwrap();
2086 std::fs::write(&settings, "{ invalid json because").unwrap();
2087
2088 assert!(
2089 remove_claude_global_hook_at(Some(tmp.path())).is_err(),
2090 "bad JSON must surface as an error"
2091 );
2092
2093 let after = std::fs::read_to_string(&settings).unwrap();
2094 assert_eq!(after, "{ invalid json because");
2095 }
2096}
2097
2098#[cfg(test)]
2099mod issue_11_tool_filter_tests {
2100 use super::*;
2109
2110 #[test]
2111 fn canonicalize_collapses_common_aliases() {
2112 for aliases in &[
2115 (vec!["Claude Code", "claude-code", "claude", "CLAUDE", "ClaudeCode"], "claudecode"),
2116 (vec!["Cursor", "cursor", "CURSOR"], "cursor"),
2117 (vec!["Windsurf", "WINDSURF"], "windsurf"),
2118 (vec!["Cline", "cline", "Roo", "roo-code", "RooCode"], "cline"),
2122 (vec!["Gemini CLI", "gemini-cli", "gemini", "GEMINI"], "gemini"),
2123 (vec!["OpenCode", "open-code", "opencode", "OPENCODE"], "opencode"),
2124 (vec!["Codex", "codex"], "codex"),
2125 ] {
2126 for alias in &aliases.0 {
2127 assert_eq!(
2128 canonicalize_tool_name(alias),
2129 aliases.1,
2130 "alias '{}' must canonicalise to '{}'",
2131 alias,
2132 aliases.1
2133 );
2134 }
2135 }
2136 }
2137
2138 #[test]
2139 fn canonicalize_leaves_unknown_names_unchanged_but_normalised() {
2140 assert_eq!(canonicalize_tool_name("unknown-tool"), "unknowntool");
2145 assert_eq!(canonicalize_tool_name("Some Thing"), "something");
2146 }
2147
2148 #[test]
2149 fn parse_tool_list_accepts_comma_separated_with_whitespace() {
2150 let names = parse_tool_list("opencode,codex").unwrap();
2153 assert_eq!(names, vec!["opencode", "codex"]);
2154
2155 let names = parse_tool_list(" opencode , codex ").unwrap();
2156 assert_eq!(names, vec!["opencode", "codex"]);
2157
2158 let names = parse_tool_list("opencode").unwrap();
2160 assert_eq!(names, vec!["opencode"]);
2161
2162 let names = parse_tool_list("claude-code").unwrap();
2164 assert_eq!(names, vec!["claudecode"]);
2165 }
2166
2167 #[test]
2168 fn parse_tool_list_dedupes_repeated_entries() {
2169 let names = parse_tool_list("opencode,opencode").unwrap();
2174 assert_eq!(names, vec!["opencode"]);
2175
2176 let names = parse_tool_list("Claude Code, claude, claude-code").unwrap();
2179 assert_eq!(names, vec!["claudecode"]);
2180 }
2181
2182 #[test]
2183 fn parse_tool_list_rejects_unknown_names_with_helpful_error() {
2184 let err = parse_tool_list("opncode").unwrap_err();
2191 let msg = err.to_string();
2192 assert!(
2193 msg.contains("unknown agent name 'opncode'"),
2194 "error must quote the bad input: {msg}"
2195 );
2196 assert!(msg.contains("opencode"), "error must list valid options: {msg}");
2197 assert!(msg.contains("cursor"), "error must list valid options: {msg}");
2198 }
2199
2200 #[test]
2201 fn parse_tool_list_rejects_one_bad_entry_in_a_list() {
2202 let err = parse_tool_list("opencode,xyz").unwrap_err();
2207 assert!(err.to_string().contains("xyz"));
2208 }
2209
2210 #[test]
2211 fn parse_tool_list_empty_and_whitespace_return_empty_vec() {
2212 assert_eq!(parse_tool_list("").unwrap(), Vec::<String>::new());
2218 assert_eq!(parse_tool_list(" ").unwrap(), Vec::<String>::new());
2219 assert_eq!(parse_tool_list(" , , ").unwrap(), Vec::<String>::new());
2220 }
2221
2222 #[test]
2223 fn tool_filter_all_includes_every_supported_tool() {
2224 let filter = ToolFilter::All;
2225 for tool in SUPPORTED_TOOL_NAMES {
2226 assert!(
2227 filter.includes(tool),
2228 "default filter must include {tool}"
2229 );
2230 }
2231 }
2232
2233 #[test]
2234 fn tool_filter_only_opencode_excludes_everything_else() {
2235 let filter = ToolFilter::Only(vec!["opencode".to_string()]);
2237 assert!(filter.includes("OpenCode"));
2238 for tool in SUPPORTED_TOOL_NAMES {
2240 if *tool == "OpenCode" {
2241 continue;
2242 }
2243 assert!(
2244 !filter.includes(tool),
2245 "--only opencode must not include {tool}"
2246 );
2247 }
2248 }
2249
2250 #[test]
2251 fn tool_filter_only_multi_tool_includes_exactly_those() {
2252 let filter = ToolFilter::Only(vec!["opencode".to_string(), "codex".to_string()]);
2253 assert!(filter.includes("OpenCode"));
2254 assert!(filter.includes("Codex"));
2255 assert!(!filter.includes("Claude Code"));
2257 assert!(!filter.includes("Cursor"));
2258 assert!(!filter.includes("Windsurf"));
2259 assert!(!filter.includes("Cline"));
2260 assert!(!filter.includes("Gemini CLI"));
2261 }
2262
2263 #[test]
2264 fn tool_filter_skip_inverts_the_set() {
2265 let filter = ToolFilter::Skip(vec!["cursor".to_string(), "windsurf".to_string()]);
2268 assert!(!filter.includes("Cursor"));
2269 assert!(!filter.includes("Windsurf"));
2270 assert!(filter.includes("Claude Code"));
2272 assert!(filter.includes("Cline"));
2273 assert!(filter.includes("Gemini CLI"));
2274 assert!(filter.includes("OpenCode"));
2275 assert!(filter.includes("Codex"));
2276 }
2277
2278 #[test]
2279 fn tool_filter_only_empty_excludes_everything() {
2280 let filter = ToolFilter::Only(vec![]);
2286 for tool in SUPPORTED_TOOL_NAMES {
2287 assert!(
2288 !filter.includes(tool),
2289 "empty --only must exclude every tool, got {tool}"
2290 );
2291 }
2292 }
2293
2294 #[test]
2295 fn tool_filter_only_accepts_display_name_or_canonical() {
2296 let filter = ToolFilter::Only(vec!["claudecode".to_string()]);
2302 assert!(filter.includes("Claude Code"));
2303 assert!(!filter.includes("Cursor"));
2304
2305 let filter = ToolFilter::Only(vec!["gemini".to_string()]);
2306 assert!(filter.includes("Gemini CLI"));
2307 }
2308
2309 #[test]
2310 fn supported_tool_names_matches_generate_hook_configs_exactly() {
2311 let configs = generate_hook_configs("sqz");
2317 let emitted: std::collections::HashSet<&str> =
2318 configs.iter().map(|c| c.tool_name.as_str()).collect();
2319 let declared: std::collections::HashSet<&str> =
2320 SUPPORTED_TOOL_NAMES.iter().copied().collect();
2321 assert_eq!(
2322 emitted, declared,
2323 "SUPPORTED_TOOL_NAMES must equal the set of tool_name values \
2324 from generate_hook_configs. emitted={:?}, declared={:?}",
2325 emitted, declared
2326 );
2327 }
2328
2329 #[test]
2330 fn filtered_install_only_opencode_writes_only_opencode_files() {
2331 let dir = tempfile::tempdir().unwrap();
2337 let filter = ToolFilter::Only(vec!["opencode".to_string()]);
2338 let _installed = install_tool_hooks_scoped_filtered(
2339 dir.path(),
2340 "sqz",
2341 InstallScope::Project,
2342 &filter,
2343 );
2344
2345 assert!(
2347 dir.path().join("opencode.json").exists(),
2348 "OpenCode config must be written when --only opencode is used"
2349 );
2350
2351 for (path, tool) in &[
2353 (".claude/settings.local.json", "Claude Code"),
2354 (".cursor/rules/sqz.mdc", "Cursor"),
2355 (".windsurfrules", "Windsurf"),
2356 (".clinerules", "Cline"),
2357 (".gemini/settings.json", "Gemini CLI"),
2358 ("AGENTS.md", "Codex"),
2359 ] {
2360 assert!(
2361 !dir.path().join(path).exists(),
2362 "filter rejected {tool} but the installer still wrote {path}"
2363 );
2364 }
2365 }
2366
2367 #[test]
2368 fn filtered_install_skip_cursor_omits_only_cursor() {
2369 let dir = tempfile::tempdir().unwrap();
2371 let filter = ToolFilter::Skip(vec!["cursor".to_string()]);
2372 let _installed = install_tool_hooks_scoped_filtered(
2373 dir.path(),
2374 "sqz",
2375 InstallScope::Project,
2376 &filter,
2377 );
2378
2379 assert!(
2381 !dir.path().join(".cursor/rules/sqz.mdc").exists(),
2382 "skip cursor: .cursor/rules/sqz.mdc must not be written"
2383 );
2384 assert!(
2386 dir.path().join(".windsurfrules").exists(),
2387 "skip cursor should not skip windsurf"
2388 );
2389 assert!(
2390 dir.path().join(".clinerules").exists(),
2391 "skip cursor should not skip cline"
2392 );
2393 }
2394}
2395