"""Consistency checks for agent configuration across runtime surfaces.""" from pathlib import Path from specify_cli import AGENT_CONFIG, AI_ASSISTANT_ALIASES, AI_ASSISTANT_HELP from specify_cli.extensions import CommandRegistrar REPO_ROOT = Path(__file__).resolve().parent.parent class TestAgentConfigConsistency: """Ensure kiro-cli migration stays synchronized across key surfaces.""" def test_runtime_config_uses_kiro_cli_and_removes_q(self): """AGENT_CONFIG should include kiro-cli and exclude legacy q.""" assert "kiro-cli" in AGENT_CONFIG assert AGENT_CONFIG["kiro-cli"]["folder"] == ".kiro/" assert AGENT_CONFIG["kiro-cli"]["commands_subdir"] == "prompts" assert "q" not in AGENT_CONFIG def test_extension_registrar_uses_kiro_cli_and_removes_q(self): """Extension command registrar should target .kiro/prompts.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "kiro-cli" in cfg assert cfg["kiro-cli"]["dir"] == ".kiro/prompts" assert "q" not in cfg def test_extension_registrar_includes_codex(self): """Extension command registrar should include codex targeting .agents/skills.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "codex" in cfg assert cfg["codex"]["dir"] == ".agents/skills" assert cfg["codex"]["extension"] == "/SKILL.md" def test_runtime_codex_uses_native_skills(self): """Codex runtime config should point at .agents/skills.""" assert AGENT_CONFIG["codex"]["folder"] == ".agents/" assert AGENT_CONFIG["codex"]["commands_subdir"] == "skills" def test_init_ai_help_includes_roo_and_kiro_alias(self): """CLI help text for --ai should stay in sync with agent config and alias guidance.""" assert "roo" in AI_ASSISTANT_HELP for alias, target in AI_ASSISTANT_ALIASES.items(): assert alias in AI_ASSISTANT_HELP assert target in AI_ASSISTANT_HELP def test_devcontainer_kiro_installer_uses_pinned_checksum(self): """Devcontainer installer should always verify Kiro installer via pinned SHA256.""" post_create_text = (REPO_ROOT / ".devcontainer" / "post-create.sh").read_text( encoding="utf-8" ) assert ( 'KIRO_INSTALLER_SHA256="7487a65cf310b7fb59b357c4b5e6e3f3259d383f4394ecedb39acf70f307cffb"' in post_create_text ) assert "sha256sum -c -" in post_create_text assert "KIRO_SKIP_KIRO_INSTALLER_VERIFY" not in post_create_text # --- Tabnine CLI consistency checks --- def test_runtime_config_includes_tabnine(self): """AGENT_CONFIG should include tabnine with correct folder and subdir.""" assert "tabnine" in AGENT_CONFIG assert AGENT_CONFIG["tabnine"]["folder"] == ".tabnine/agent/" assert AGENT_CONFIG["tabnine"]["commands_subdir"] == "commands" assert AGENT_CONFIG["tabnine"]["requires_cli"] is True assert AGENT_CONFIG["tabnine"]["install_url"] is not None def test_extension_registrar_includes_tabnine(self): """CommandRegistrar.AGENT_CONFIGS should include tabnine with correct TOML config.""" from specify_cli.extensions import CommandRegistrar assert "tabnine" in CommandRegistrar.AGENT_CONFIGS cfg = CommandRegistrar.AGENT_CONFIGS["tabnine"] assert cfg["dir"] == ".tabnine/agent/commands" assert cfg["format"] == "toml" assert cfg["args"] == "{{args}}" assert cfg["extension"] == ".toml" def test_ai_help_includes_tabnine(self): """CLI help text for --ai should include tabnine.""" assert "tabnine" in AI_ASSISTANT_HELP # --- Kimi Code CLI consistency checks --- def test_kimi_in_agent_config(self): """AGENT_CONFIG should include kimi with correct folder and commands_subdir.""" assert "kimi" in AGENT_CONFIG assert AGENT_CONFIG["kimi"]["folder"] == ".kimi/" assert AGENT_CONFIG["kimi"]["commands_subdir"] == "skills" assert AGENT_CONFIG["kimi"]["requires_cli"] is True def test_kimi_in_extension_registrar(self): """Extension command registrar should include kimi using .kimi/skills and SKILL.md.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "kimi" in cfg kimi_cfg = cfg["kimi"] assert kimi_cfg["dir"] == ".kimi/skills" assert kimi_cfg["extension"] == "/SKILL.md" def test_ai_help_includes_kimi(self): """CLI help text for --ai should include kimi.""" assert "kimi" in AI_ASSISTANT_HELP # --- Trae IDE consistency checks --- def test_trae_in_agent_config(self): """AGENT_CONFIG should include trae with correct folder and commands_subdir.""" assert "trae" in AGENT_CONFIG assert AGENT_CONFIG["trae"]["folder"] == ".trae/" assert AGENT_CONFIG["trae"]["commands_subdir"] == "skills" assert AGENT_CONFIG["trae"]["requires_cli"] is False assert AGENT_CONFIG["trae"]["install_url"] is None def test_trae_in_extension_registrar(self): """Extension command registrar should include trae using .trae/rules and markdown, if present.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "trae" in cfg trae_cfg = cfg["trae"] assert trae_cfg["format"] == "markdown" assert trae_cfg["args"] == "$ARGUMENTS" assert trae_cfg["extension"] == "/SKILL.md" def test_ai_help_includes_trae(self): """CLI help text for --ai should include trae.""" assert "trae" in AI_ASSISTANT_HELP # --- Pi Coding Agent consistency checks --- def test_pi_in_agent_config(self): """AGENT_CONFIG should include pi with correct folder and commands_subdir.""" assert "pi" in AGENT_CONFIG assert AGENT_CONFIG["pi"]["folder"] == ".pi/" assert AGENT_CONFIG["pi"]["commands_subdir"] == "prompts" assert AGENT_CONFIG["pi"]["requires_cli"] is True assert AGENT_CONFIG["pi"]["install_url"] is not None def test_pi_in_extension_registrar(self): """Extension command registrar should include pi using .pi/prompts.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "pi" in cfg pi_cfg = cfg["pi"] assert pi_cfg["dir"] == ".pi/prompts" assert pi_cfg["format"] == "markdown" assert pi_cfg["args"] == "$ARGUMENTS" assert pi_cfg["extension"] == ".md" def test_ai_help_includes_pi(self): """CLI help text for --ai should include pi.""" assert "pi" in AI_ASSISTANT_HELP # --- iFlow CLI consistency checks --- def test_iflow_in_agent_config(self): """AGENT_CONFIG should include iflow with correct folder and commands_subdir.""" assert "iflow" in AGENT_CONFIG assert AGENT_CONFIG["iflow"]["folder"] == ".iflow/" assert AGENT_CONFIG["iflow"]["commands_subdir"] == "commands" assert AGENT_CONFIG["iflow"]["requires_cli"] is True def test_iflow_in_extension_registrar(self): """Extension command registrar should include iflow targeting .iflow/commands.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "iflow" in cfg assert cfg["iflow"]["dir"] == ".iflow/commands" assert cfg["iflow"]["format"] == "markdown" assert cfg["iflow"]["args"] == "$ARGUMENTS" def test_ai_help_includes_iflow(self): """CLI help text for --ai should include iflow.""" assert "iflow" in AI_ASSISTANT_HELP # --- Goose consistency checks --- def test_goose_in_agent_config(self): """AGENT_CONFIG should include goose with correct folder and commands_subdir.""" assert "goose" in AGENT_CONFIG assert AGENT_CONFIG["goose"]["folder"] == ".goose/" assert AGENT_CONFIG["goose"]["commands_subdir"] == "recipes" assert AGENT_CONFIG["goose"]["requires_cli"] is True def test_goose_in_extension_registrar(self): """Extension command registrar should include goose targeting .goose/recipes.""" cfg = CommandRegistrar.AGENT_CONFIGS assert "goose" in cfg assert cfg["goose"]["dir"] == ".goose/recipes" assert cfg["goose"]["format"] == "yaml" assert cfg["goose"]["args"] == "{{args}}" def test_ai_help_includes_goose(self): """CLI help text for --ai should include goose.""" assert "goose" in AI_ASSISTANT_HELP # --- invoke_separator propagation checks --- def test_skills_agents_have_hyphen_invoke_separator_in_agent_configs(self): """Skills-based agents must expose invoke_separator='-' in AGENT_CONFIGS. SkillsIntegration sets ``invoke_separator = "-"`` as a class attribute, but individual skills integrations (claude, codex, …) do not repeat it in their ``registrar_config`` dicts. ``_build_agent_configs()`` must propagate the class attribute so that ``register_commands()`` resolves ``__SPECKIT_COMMAND_*__`` tokens with the correct hyphen separator. """ cfg = CommandRegistrar.AGENT_CONFIGS skills_agents = [ key for key, c in cfg.items() if c.get("extension") == "/SKILL.md" ] assert skills_agents, ( "Expected at least one skills-based agent in AGENT_CONFIGS" ) for agent in skills_agents: assert cfg[agent].get("invoke_separator") == "-", ( f"Skills agent '{agent}' has invoke_separator=" f"{cfg[agent].get('invoke_separator')!r} in AGENT_CONFIGS; " "expected '-' (propagated from SkillsIntegration.invoke_separator)" ) def test_skills_agent_command_token_resolves_with_hyphen(self, tmp_path): """__SPECKIT_COMMAND_*__ tokens in extension commands resolve to /speckit- when registered for a skills-based agent (e.g. claude). Regression guard: before the fix, _build_agent_configs() did not propagate invoke_separator from the integration class, so register_commands() fell back to '.' and emitted /speckit.specify instead of /speckit-specify for skills agents. """ import re from pathlib import Path from specify_cli.agents import CommandRegistrar repo_root = Path(__file__).resolve().parent.parent ext_dir = repo_root / "extensions" / "git" cmd_source = ext_dir / "commands" / "speckit.git.feature.md" assert cmd_source.exists(), ( f"Git extension command source not found at {cmd_source}" ) assert "__SPECKIT_COMMAND_SPECIFY__" in cmd_source.read_text( encoding="utf-8" ), ( "Expected __SPECKIT_COMMAND_SPECIFY__ token in speckit.git.feature.md; " "check that the file uses the token rather than a hard-coded ref." ) registrar = CommandRegistrar() commands = [ {"name": "speckit.git.feature", "file": "commands/speckit.git.feature.md"} ] registered = registrar.register_commands( "claude", commands, "git", ext_dir, tmp_path, ) assert "speckit.git.feature" in registered skill_file = ( tmp_path / ".claude" / "skills" / "speckit-git-feature" / "SKILL.md" ) assert skill_file.exists(), ( f"Expected Claude skill file not found at {skill_file}" ) content = skill_file.read_text(encoding="utf-8") assert "/speckit-specify" in content, ( "Expected '/speckit-specify' (hyphen) in generated Claude skill for git.feature; " "__SPECKIT_COMMAND_SPECIFY__ was not resolved with the correct separator." ) # Negative lookbehind (?) in generated Claude skill. " "Skills agents must use hyphen notation." )