Files
mistral-vibe/vibe/core/agents/models.py
Mathias Gesbert 6a50d1d521 v2.7.0 (#534)
Co-authored-by: Quentin Torroba <quentin.torroba@mistral.ai>
Co-authored-by: Clément Drouin <clement.drouin@mistral.ai>
Co-authored-by: Clément Sirieix <clement.sirieix@mistral.ai>
Co-authored-by: Mistral Vibe <vibe@mistral.ai>
2026-03-26 10:11:34 +01:00

199 lines
5.6 KiB
Python

from __future__ import annotations
from dataclasses import dataclass, field
from enum import StrEnum, auto
from pathlib import Path
import tomllib
from typing import TYPE_CHECKING, Any
from vibe.core.paths import PLANS_DIR
if TYPE_CHECKING:
from vibe.core.config import VibeConfig
def _deep_merge(base: dict[str, Any], override: dict[str, Any]) -> dict[str, Any]:
result = base.copy()
for key, value in override.items():
if key in result and isinstance(result[key], dict) and isinstance(value, dict):
result[key] = _deep_merge(result[key], value)
else:
result[key] = value
return result
class AgentSafety(StrEnum):
SAFE = auto()
NEUTRAL = auto()
DESTRUCTIVE = auto()
YOLO = auto()
class AgentType(StrEnum):
AGENT = auto()
SUBAGENT = auto()
class BuiltinAgentName(StrEnum):
DEFAULT = "default"
CHAT = "chat"
PLAN = "plan"
ACCEPT_EDITS = "accept-edits"
AUTO_APPROVE = "auto-approve"
EXPLORE = "explore"
LEAN = "lean"
@dataclass(frozen=True)
class AgentProfile:
name: str
display_name: str
description: str
safety: AgentSafety
agent_type: AgentType = AgentType.AGENT
overrides: dict[str, Any] = field(default_factory=dict)
install_required: bool = False
def apply_to_config(self, base: VibeConfig) -> VibeConfig:
from vibe.core.config import VibeConfig as VC
merged = _deep_merge(
base.model_dump(),
{k: v for k, v in self.overrides.items() if k != "base_disabled"},
)
base_disabled = self.overrides.get("base_disabled")
if isinstance(base_disabled, list):
merged["disabled_tools"] = list({
*base_disabled,
*merged.get("disabled_tools", []),
})
return VC.model_validate(merged)
@classmethod
def from_toml(cls, path: Path) -> AgentProfile:
with path.open("rb") as f:
data = tomllib.load(f)
return cls(
name=path.stem,
display_name=data.pop("display_name", path.stem.replace("-", " ").title()),
description=data.pop("description", ""),
safety=AgentSafety(data.pop("safety", AgentSafety.NEUTRAL)),
agent_type=AgentType(data.pop("agent_type", AgentType.AGENT)),
overrides=data,
)
CHAT_AGENT_TOOLS = ["grep", "read_file", "ask_user_question", "task"]
def _plan_overrides() -> dict[str, Any]:
plans_pattern = str(PLANS_DIR.path / "*")
return {
"tools": {
"write_file": {"permission": "never", "allowlist": [plans_pattern]},
"search_replace": {"permission": "never", "allowlist": [plans_pattern]},
}
}
DEFAULT = AgentProfile(
BuiltinAgentName.DEFAULT,
"Default",
"Requires approval for tool executions",
AgentSafety.NEUTRAL,
overrides={"base_disabled": ["exit_plan_mode"]},
)
PLAN = AgentProfile(
BuiltinAgentName.PLAN,
"Plan",
"Read-only agent for exploration and planning",
AgentSafety.SAFE,
overrides=_plan_overrides(),
)
CHAT = AgentProfile(
BuiltinAgentName.CHAT,
"Chat",
"Read-only conversational mode for questions and discussions",
AgentSafety.SAFE,
overrides={"auto_approve": True, "enabled_tools": CHAT_AGENT_TOOLS},
)
ACCEPT_EDITS = AgentProfile(
BuiltinAgentName.ACCEPT_EDITS,
"Accept Edits",
"Auto-approves file edits only",
AgentSafety.DESTRUCTIVE,
overrides={
"base_disabled": ["exit_plan_mode"],
"tools": {
"write_file": {"permission": "always"},
"search_replace": {"permission": "always"},
},
},
)
AUTO_APPROVE = AgentProfile(
BuiltinAgentName.AUTO_APPROVE,
"Auto Approve",
"Auto-approves all tool executions",
AgentSafety.YOLO,
overrides={"auto_approve": True, "base_disabled": ["exit_plan_mode"]},
)
EXPLORE = AgentProfile(
name=BuiltinAgentName.EXPLORE,
display_name="Explore",
description="Read-only subagent for codebase exploration",
safety=AgentSafety.SAFE,
agent_type=AgentType.SUBAGENT,
overrides={"enabled_tools": ["grep", "read_file"], "system_prompt_id": "explore"},
)
LEAN = AgentProfile(
name=BuiltinAgentName.LEAN,
display_name="Lean",
description="Specialized mode for Lean 4 code analysis, proof assistance, and theorem proving",
safety=AgentSafety.NEUTRAL,
agent_type=AgentType.AGENT,
install_required=True,
overrides={
"system_prompt_id": "lean",
"active_model": "leanstral",
"providers": [
{
"name": "mistral-testing",
"api_base": "https://api.mistral.ai/v1",
"api_key_env_var": "MISTRAL_API_KEY",
"backend": "mistral",
}
],
"models": [
{
"name": "labs-leanstral-2603",
"provider": "mistral-testing",
"alias": "leanstral",
"thinking": "high",
"temperature": 1.0,
"auto_compact_threshold": 168_000,
}
],
"compaction_model": {
"name": "mistral-vibe-cli-latest",
"provider": "mistral-testing",
"alias": "devstral-compact",
"temperature": 0.2,
"thinking": "off",
},
"tools": {"bash": {"default_timeout": 1200}},
"base_disabled": ["exit_plan_mode"],
},
)
BUILTIN_AGENTS: dict[str, AgentProfile] = {
BuiltinAgentName.DEFAULT: DEFAULT,
BuiltinAgentName.PLAN: PLAN,
BuiltinAgentName.ACCEPT_EDITS: ACCEPT_EDITS,
BuiltinAgentName.AUTO_APPROVE: AUTO_APPROVE,
BuiltinAgentName.EXPLORE: EXPLORE,
BuiltinAgentName.LEAN: LEAN,
}