Unify provider/model config for Tier 1 and Tier 2 agents
- Add provider and model fields to both GroupAgentConfig and ProjectConfig - Wire model override through AgentSession → AgentPane → queryAgent → sidecar - Add model preset dropdown per provider (Opus/Sonnet/Haiku, GPT-5.4/o3, etc.) with custom model ID input at the bottom - Add provider dropdown to Tier 1 agents (was Tier 2 only) - Add "Apply & Restart" button on both tiers to restart agent with new settings - Changing provider auto-resets model selection - Admin bypasses stale heartbeat check in btmsg so DMs always deliver Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
parent
2c710fa0db
commit
35963be686
8 changed files with 243 additions and 8 deletions
|
|
@ -58,6 +58,8 @@
|
||||||
useWorktrees?: boolean;
|
useWorktrees?: boolean;
|
||||||
/** Prepended to system_prompt for agent role instructions */
|
/** Prepended to system_prompt for agent role instructions */
|
||||||
agentSystemPrompt?: string;
|
agentSystemPrompt?: string;
|
||||||
|
/** Model override (e.g. 'claude-sonnet-4-5-20250514'). Passed to sidecar. */
|
||||||
|
model?: string;
|
||||||
/** Extra env vars injected into agent process (e.g. BTMSG_AGENT_ID) */
|
/** Extra env vars injected into agent process (e.g. BTMSG_AGENT_ID) */
|
||||||
extraEnv?: Record<string, string>;
|
extraEnv?: Record<string, string>;
|
||||||
/** Auto-triggered prompt (e.g. periodic context refresh). Picked up when agent is idle. */
|
/** Auto-triggered prompt (e.g. periodic context refresh). Picked up when agent is idle. */
|
||||||
|
|
@ -67,7 +69,7 @@
|
||||||
onExit?: () => void;
|
onExit?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
let { sessionId, projectId, prompt: initialPrompt = '', cwd: initialCwd, profile: profileName, provider: providerId = 'claude', capabilities = DEFAULT_CAPABILITIES, useWorktrees = false, agentSystemPrompt, extraEnv, autoPrompt, onautopromptconsumed, onExit }: Props = $props();
|
let { sessionId, projectId, prompt: initialPrompt = '', cwd: initialCwd, profile: profileName, provider: providerId = 'claude', capabilities = DEFAULT_CAPABILITIES, useWorktrees = false, agentSystemPrompt, model: modelOverride, extraEnv, autoPrompt, onautopromptconsumed, onExit }: Props = $props();
|
||||||
|
|
||||||
let session = $derived(getAgentSession(sessionId));
|
let session = $derived(getAgentSession(sessionId));
|
||||||
let inputPrompt = $state(initialPrompt);
|
let inputPrompt = $state(initialPrompt);
|
||||||
|
|
@ -209,6 +211,7 @@
|
||||||
setting_sources: ['user', 'project'],
|
setting_sources: ['user', 'project'],
|
||||||
claude_config_dir: profile?.config_dir,
|
claude_config_dir: profile?.config_dir,
|
||||||
system_prompt: systemPrompt,
|
system_prompt: systemPrompt,
|
||||||
|
model: modelOverride || undefined,
|
||||||
worktree_name: useWorktrees ? sessionId : undefined,
|
worktree_name: useWorktrees ? sessionId : undefined,
|
||||||
extra_env: extraEnv,
|
extra_env: extraEnv,
|
||||||
});
|
});
|
||||||
|
|
|
||||||
|
|
@ -269,6 +269,7 @@
|
||||||
capabilities={providerMeta?.capabilities}
|
capabilities={providerMeta?.capabilities}
|
||||||
useWorktrees={project.useWorktrees ?? false}
|
useWorktrees={project.useWorktrees ?? false}
|
||||||
agentSystemPrompt={agentPrompt}
|
agentSystemPrompt={agentPrompt}
|
||||||
|
model={project.model}
|
||||||
extraEnv={agentEnv}
|
extraEnv={agentEnv}
|
||||||
autoPrompt={contextRefreshPrompt}
|
autoPrompt={contextRefreshPrompt}
|
||||||
onautopromptconsumed={handleAutoPromptConsumed}
|
onautopromptconsumed={handleAutoPromptConsumed}
|
||||||
|
|
|
||||||
|
|
@ -12,6 +12,8 @@
|
||||||
addGroup,
|
addGroup,
|
||||||
removeGroup,
|
removeGroup,
|
||||||
switchGroup,
|
switchGroup,
|
||||||
|
emitAgentStop,
|
||||||
|
emitAgentStart,
|
||||||
} from '../../stores/workspace.svelte';
|
} from '../../stores/workspace.svelte';
|
||||||
import { deriveIdentifier, type GroupAgentRole, AGENT_ROLE_ICONS } from '../../types/groups';
|
import { deriveIdentifier, type GroupAgentRole, AGENT_ROLE_ICONS } from '../../types/groups';
|
||||||
import { ProjectId, GroupId } from '../../types/ids';
|
import { ProjectId, GroupId } from '../../types/ids';
|
||||||
|
|
@ -56,6 +58,7 @@
|
||||||
let expandedProvider = $state<string | null>(null);
|
let expandedProvider = $state<string | null>(null);
|
||||||
let registeredProviders = $derived(getProviders());
|
let registeredProviders = $derived(getProviders());
|
||||||
let providerDropdownOpenFor = $state<string | null>(null);
|
let providerDropdownOpenFor = $state<string | null>(null);
|
||||||
|
let modelDropdownOpenFor = $state<string | null>(null);
|
||||||
|
|
||||||
let activeGroupId = $derived(getActiveGroupId());
|
let activeGroupId = $derived(getActiveGroupId());
|
||||||
let activeGroup = $derived(getActiveGroup());
|
let activeGroup = $derived(getActiveGroup());
|
||||||
|
|
@ -372,6 +375,7 @@
|
||||||
uiFontDropdownOpen = false;
|
uiFontDropdownOpen = false;
|
||||||
termFontDropdownOpen = false;
|
termFontDropdownOpen = false;
|
||||||
providerDropdownOpenFor = null;
|
providerDropdownOpenFor = null;
|
||||||
|
modelDropdownOpenFor = null;
|
||||||
secretsKeyDropdownOpen = false;
|
secretsKeyDropdownOpen = false;
|
||||||
}
|
}
|
||||||
if (!target.closest('.icon-field')) {
|
if (!target.closest('.icon-field')) {
|
||||||
|
|
@ -389,6 +393,7 @@
|
||||||
termFontDropdownOpen = false;
|
termFontDropdownOpen = false;
|
||||||
iconPickerOpenFor = null;
|
iconPickerOpenFor = null;
|
||||||
profileDropdownOpenFor = null;
|
profileDropdownOpenFor = null;
|
||||||
|
modelDropdownOpenFor = null;
|
||||||
secretsKeyDropdownOpen = false;
|
secretsKeyDropdownOpen = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -961,6 +966,7 @@
|
||||||
|
|
||||||
<div class="agent-cards">
|
<div class="agent-cards">
|
||||||
{#each activeGroup.agents ?? [] as agent (agent.id)}
|
{#each activeGroup.agents ?? [] as agent (agent.id)}
|
||||||
|
{@const agentProvider = registeredProviders.find(p => p.id === (agent.provider ?? 'claude'))}
|
||||||
<div class="agent-config-card">
|
<div class="agent-config-card">
|
||||||
<div class="card-top-row">
|
<div class="card-top-row">
|
||||||
<span class="agent-config-icon">{AGENT_ROLE_ICONS[agent.role] ?? '🤖'}</span>
|
<span class="agent-config-icon">{AGENT_ROLE_ICONS[agent.role] ?? '🤖'}</span>
|
||||||
|
|
@ -1000,17 +1006,89 @@
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{#if registeredProviders.length > 1}
|
||||||
|
<div class="card-field">
|
||||||
|
<span class="card-field-label">
|
||||||
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 2L2 7l10 5 10-5-10-5z"/><path d="M2 17l10 5 10-5"/><path d="M2 12l10 5 10-5"/></svg>
|
||||||
|
Provider
|
||||||
|
</span>
|
||||||
|
<div class="custom-dropdown">
|
||||||
|
<button
|
||||||
|
class="dropdown-trigger provider-trigger"
|
||||||
|
onclick={() => { providerDropdownOpenFor = providerDropdownOpenFor === agent.id ? null : agent.id; }}
|
||||||
|
aria-haspopup="listbox"
|
||||||
|
aria-expanded={providerDropdownOpenFor === agent.id}
|
||||||
|
>
|
||||||
|
<span class="dropdown-label">{registeredProviders.find(p => p.id === (agent.provider ?? 'claude'))?.name ?? 'Claude Code'}</span>
|
||||||
|
<span class="dropdown-arrow">{providerDropdownOpenFor === agent.id ? '\u25B4' : '\u25BE'}</span>
|
||||||
|
</button>
|
||||||
|
{#if providerDropdownOpenFor === agent.id}
|
||||||
|
<div class="dropdown-menu" role="listbox">
|
||||||
|
{#each registeredProviders.filter(p => isProviderEnabled(p.id)) as prov}
|
||||||
|
<button
|
||||||
|
class="dropdown-option"
|
||||||
|
class:active={(agent.provider ?? 'claude') === prov.id}
|
||||||
|
role="option"
|
||||||
|
aria-selected={(agent.provider ?? 'claude') === prov.id}
|
||||||
|
onclick={() => {
|
||||||
|
updateAgent(activeGroupId, agent.id, { provider: prov.id, model: undefined });
|
||||||
|
providerDropdownOpenFor = null;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<span class="dropdown-option-label">{prov.name}</span>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
<div class="card-field">
|
<div class="card-field">
|
||||||
<span class="card-field-label">
|
<span class="card-field-label">
|
||||||
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="2" y="3" width="20" height="14" rx="2" ry="2"/><line x1="8" y1="21" x2="16" y2="21"/><line x1="12" y1="17" x2="12" y2="21"/></svg>
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="2" y="3" width="20" height="14" rx="2" ry="2"/><line x1="8" y1="21" x2="16" y2="21"/><line x1="12" y1="17" x2="12" y2="21"/></svg>
|
||||||
Model
|
Model
|
||||||
</span>
|
</span>
|
||||||
<input
|
<div class="custom-dropdown">
|
||||||
class="card-field-input"
|
<button
|
||||||
value={agent.model ?? ''}
|
class="dropdown-trigger"
|
||||||
placeholder="Default (provider default)"
|
onclick={() => { modelDropdownOpenFor = modelDropdownOpenFor === agent.id ? null : agent.id; }}
|
||||||
onchange={e => updateAgent(activeGroupId, agent.id, { model: (e.target as HTMLInputElement).value || undefined })}
|
>
|
||||||
/>
|
<span class="dropdown-label">{agent.model || `Default (${agentProvider?.defaultModel ?? '?'})`}</span>
|
||||||
|
<span class="dropdown-arrow">{modelDropdownOpenFor === agent.id ? '\u25B4' : '\u25BE'}</span>
|
||||||
|
</button>
|
||||||
|
{#if modelDropdownOpenFor === agent.id}
|
||||||
|
<div class="dropdown-menu" role="listbox">
|
||||||
|
<button
|
||||||
|
class="dropdown-option"
|
||||||
|
class:active={!agent.model}
|
||||||
|
onclick={() => { updateAgent(activeGroupId, agent.id, { model: undefined }); modelDropdownOpenFor = null; }}
|
||||||
|
>
|
||||||
|
<span class="dropdown-option-label">Default ({agentProvider?.defaultModel ?? '?'})</span>
|
||||||
|
</button>
|
||||||
|
{#each agentProvider?.models ?? [] as m}
|
||||||
|
<button
|
||||||
|
class="dropdown-option"
|
||||||
|
class:active={agent.model === m.id}
|
||||||
|
onclick={() => { updateAgent(activeGroupId, agent.id, { model: m.id }); modelDropdownOpenFor = null; }}
|
||||||
|
>
|
||||||
|
<span class="dropdown-option-label">{m.label}</span>
|
||||||
|
<span class="dropdown-option-hint">{m.id}</span>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
<div class="dropdown-custom">
|
||||||
|
<input
|
||||||
|
class="dropdown-custom-input"
|
||||||
|
value={agent.model ?? ''}
|
||||||
|
placeholder="Custom model ID..."
|
||||||
|
onclick={e => e.stopPropagation()}
|
||||||
|
onkeydown={e => { if (e.key === 'Enter') { updateAgent(activeGroupId, agent.id, { model: (e.target as HTMLInputElement).value || undefined }); modelDropdownOpenFor = null; } }}
|
||||||
|
onchange={e => updateAgent(activeGroupId, agent.id, { model: (e.target as HTMLInputElement).value || undefined })}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if agent.role === 'manager'}
|
{#if agent.role === 'manager'}
|
||||||
|
|
@ -1086,6 +1164,17 @@
|
||||||
></textarea>
|
></textarea>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="card-actions-row">
|
||||||
|
<button
|
||||||
|
class="apply-btn"
|
||||||
|
title="Restart agent with current settings"
|
||||||
|
onclick={() => { emitAgentStop(agent.id); setTimeout(() => emitAgentStart(agent.id), 300); }}
|
||||||
|
>
|
||||||
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M23 4v6h-6"/><path d="M1 20v-6h6"/><path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15"/></svg>
|
||||||
|
Apply & Restart
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<details class="prompt-preview">
|
<details class="prompt-preview">
|
||||||
<summary class="prompt-preview-toggle">
|
<summary class="prompt-preview-toggle">
|
||||||
<svg width="10" height="10" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M1 12s4-8 11-8 11 8 11 8-4 8-11 8-11-8-11-8z"/><circle cx="12" cy="12" r="3"/></svg>
|
<svg width="10" height="10" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M1 12s4-8 11-8 11 8 11 8-4 8-11 8-11-8-11-8z"/><circle cx="12" cy="12" r="3"/></svg>
|
||||||
|
|
@ -1111,6 +1200,7 @@
|
||||||
|
|
||||||
<div class="project-cards">
|
<div class="project-cards">
|
||||||
{#each activeGroup.projects as project}
|
{#each activeGroup.projects as project}
|
||||||
|
{@const projProvider = registeredProviders.find(p => p.id === (project.provider ?? 'claude'))}
|
||||||
<div class="project-card">
|
<div class="project-card">
|
||||||
<div class="card-top-row">
|
<div class="card-top-row">
|
||||||
<div class="icon-field">
|
<div class="icon-field">
|
||||||
|
|
@ -1239,7 +1329,7 @@
|
||||||
role="option"
|
role="option"
|
||||||
aria-selected={(project.provider ?? 'claude') === prov.id}
|
aria-selected={(project.provider ?? 'claude') === prov.id}
|
||||||
onclick={() => {
|
onclick={() => {
|
||||||
updateProject(activeGroupId, project.id, { provider: prov.id });
|
updateProject(activeGroupId, project.id, { provider: prov.id, model: undefined });
|
||||||
providerDropdownOpenFor = null;
|
providerDropdownOpenFor = null;
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
|
|
@ -1252,6 +1342,53 @@
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
|
<div class="card-field">
|
||||||
|
<span class="card-field-label">
|
||||||
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="2" y="3" width="20" height="14" rx="2" ry="2"/><line x1="8" y1="21" x2="16" y2="21"/><line x1="12" y1="17" x2="12" y2="21"/></svg>
|
||||||
|
Model
|
||||||
|
</span>
|
||||||
|
<div class="custom-dropdown">
|
||||||
|
<button
|
||||||
|
class="dropdown-trigger"
|
||||||
|
onclick={() => { modelDropdownOpenFor = modelDropdownOpenFor === project.id ? null : project.id; }}
|
||||||
|
>
|
||||||
|
<span class="dropdown-label">{project.model || `Default (${projProvider?.defaultModel ?? '?'})`}</span>
|
||||||
|
<span class="dropdown-arrow">{modelDropdownOpenFor === project.id ? '\u25B4' : '\u25BE'}</span>
|
||||||
|
</button>
|
||||||
|
{#if modelDropdownOpenFor === project.id}
|
||||||
|
<div class="dropdown-menu" role="listbox">
|
||||||
|
<button
|
||||||
|
class="dropdown-option"
|
||||||
|
class:active={!project.model}
|
||||||
|
onclick={() => { updateProject(activeGroupId, project.id, { model: undefined }); modelDropdownOpenFor = null; }}
|
||||||
|
>
|
||||||
|
<span class="dropdown-option-label">Default ({projProvider?.defaultModel ?? '?'})</span>
|
||||||
|
</button>
|
||||||
|
{#each projProvider?.models ?? [] as m}
|
||||||
|
<button
|
||||||
|
class="dropdown-option"
|
||||||
|
class:active={project.model === m.id}
|
||||||
|
onclick={() => { updateProject(activeGroupId, project.id, { model: m.id }); modelDropdownOpenFor = null; }}
|
||||||
|
>
|
||||||
|
<span class="dropdown-option-label">{m.label}</span>
|
||||||
|
<span class="dropdown-option-hint">{m.id}</span>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
<div class="dropdown-custom">
|
||||||
|
<input
|
||||||
|
class="dropdown-custom-input"
|
||||||
|
value={project.model ?? ''}
|
||||||
|
placeholder="Custom model ID..."
|
||||||
|
onclick={e => e.stopPropagation()}
|
||||||
|
onkeydown={e => { if (e.key === 'Enter') { updateProject(activeGroupId, project.id, { model: (e.target as HTMLInputElement).value || undefined }); modelDropdownOpenFor = null; } }}
|
||||||
|
onchange={e => updateProject(activeGroupId, project.id, { model: (e.target as HTMLInputElement).value || undefined })}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="card-field">
|
<div class="card-field">
|
||||||
<span class="card-field-label">
|
<span class="card-field-label">
|
||||||
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 20V10"/><path d="M18 20V4"/><path d="M6 20v-4"/></svg>
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M12 20V10"/><path d="M18 20V4"/><path d="M6 20v-4"/></svg>
|
||||||
|
|
@ -1337,6 +1474,17 @@
|
||||||
></textarea>
|
></textarea>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="card-actions-row">
|
||||||
|
<button
|
||||||
|
class="apply-btn"
|
||||||
|
title="Restart agent with current settings"
|
||||||
|
onclick={() => { emitAgentStop(project.id); setTimeout(() => emitAgentStart(project.id), 300); }}
|
||||||
|
>
|
||||||
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M23 4v6h-6"/><path d="M1 20v-6h6"/><path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15"/></svg>
|
||||||
|
Apply & Restart
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="card-footer">
|
<div class="card-footer">
|
||||||
<button class="btn-remove" onclick={() => removeProject(activeGroupId, project.id)}>
|
<button class="btn-remove" onclick={() => removeProject(activeGroupId, project.id)}>
|
||||||
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 6h18"/><path d="M8 6V4h8v2"/><path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6"/></svg>
|
<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 6h18"/><path d="M8 6V4h8v2"/><path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6"/></svg>
|
||||||
|
|
@ -1539,6 +1687,34 @@
|
||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.dropdown-option-hint {
|
||||||
|
font-size: 0.55rem;
|
||||||
|
color: var(--ctp-overlay0);
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-custom {
|
||||||
|
padding: 0.25rem;
|
||||||
|
border-top: 1px solid var(--ctp-surface1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-custom-input {
|
||||||
|
width: 100%;
|
||||||
|
background: var(--ctp-surface0);
|
||||||
|
border: 1px solid var(--ctp-surface1);
|
||||||
|
border-radius: 0.2rem;
|
||||||
|
color: var(--ctp-text);
|
||||||
|
font-size: 0.7rem;
|
||||||
|
font-family: monospace;
|
||||||
|
padding: 0.25rem 0.4rem;
|
||||||
|
outline: none;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dropdown-custom-input:focus {
|
||||||
|
border-color: var(--ctp-blue);
|
||||||
|
}
|
||||||
|
|
||||||
/* Theme-specific dropdown extras */
|
/* Theme-specific dropdown extras */
|
||||||
.theme-swatch {
|
.theme-swatch {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
|
@ -1911,6 +2087,35 @@
|
||||||
margin-left: auto;
|
margin-left: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Apply & Restart button */
|
||||||
|
.card-actions-row {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
padding-top: 0.35rem;
|
||||||
|
margin-top: 0.25rem;
|
||||||
|
border-top: 1px solid var(--ctp-surface0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.apply-btn {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 0.3rem;
|
||||||
|
padding: 0.3rem 0.6rem;
|
||||||
|
background: color-mix(in srgb, var(--ctp-blue) 15%, transparent);
|
||||||
|
color: var(--ctp-blue);
|
||||||
|
border: 1px solid var(--ctp-blue);
|
||||||
|
border-radius: 0.25rem;
|
||||||
|
font-size: 0.65rem;
|
||||||
|
font-weight: 600;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background 0.1s, color 0.1s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.apply-btn:hover {
|
||||||
|
background: var(--ctp-blue);
|
||||||
|
color: var(--ctp-base);
|
||||||
|
}
|
||||||
|
|
||||||
/* Card footer */
|
/* Card footer */
|
||||||
.card-footer {
|
.card-footer {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|
|
||||||
|
|
@ -17,4 +17,9 @@ export const CLAUDE_PROVIDER: ProviderMeta = {
|
||||||
},
|
},
|
||||||
sidecarRunner: 'claude-runner.mjs',
|
sidecarRunner: 'claude-runner.mjs',
|
||||||
defaultModel: 'claude-opus-4-6',
|
defaultModel: 'claude-opus-4-6',
|
||||||
|
models: [
|
||||||
|
{ id: 'claude-opus-4-6', label: 'Opus 4.6' },
|
||||||
|
{ id: 'claude-sonnet-4-6', label: 'Sonnet 4.6' },
|
||||||
|
{ id: 'claude-haiku-4-5-20251001', label: 'Haiku 4.5' },
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -17,4 +17,9 @@ export const CODEX_PROVIDER: ProviderMeta = {
|
||||||
},
|
},
|
||||||
sidecarRunner: 'codex-runner.mjs',
|
sidecarRunner: 'codex-runner.mjs',
|
||||||
defaultModel: 'gpt-5.4',
|
defaultModel: 'gpt-5.4',
|
||||||
|
models: [
|
||||||
|
{ id: 'gpt-5.4', label: 'GPT-5.4' },
|
||||||
|
{ id: 'o3', label: 'o3' },
|
||||||
|
{ id: 'o4-mini', label: 'o4-mini' },
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -17,4 +17,11 @@ export const OLLAMA_PROVIDER: ProviderMeta = {
|
||||||
},
|
},
|
||||||
sidecarRunner: 'ollama-runner.mjs',
|
sidecarRunner: 'ollama-runner.mjs',
|
||||||
defaultModel: 'qwen3:8b',
|
defaultModel: 'qwen3:8b',
|
||||||
|
models: [
|
||||||
|
{ id: 'qwen3:8b', label: 'Qwen3 8B' },
|
||||||
|
{ id: 'qwen3:32b', label: 'Qwen3 32B' },
|
||||||
|
{ id: 'llama3.3:70b', label: 'Llama 3.3 70B' },
|
||||||
|
{ id: 'deepseek-r1:14b', label: 'DeepSeek R1 14B' },
|
||||||
|
{ id: 'codellama:13b', label: 'Code Llama 13B' },
|
||||||
|
],
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -23,6 +23,8 @@ export interface ProviderMeta {
|
||||||
sidecarRunner: string;
|
sidecarRunner: string;
|
||||||
/** Default model identifier, if applicable */
|
/** Default model identifier, if applicable */
|
||||||
defaultModel?: string;
|
defaultModel?: string;
|
||||||
|
/** Available model presets for dropdown selection */
|
||||||
|
models?: { id: string; label: string }[];
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Per-provider configuration (stored in settings) */
|
/** Per-provider configuration (stored in settings) */
|
||||||
|
|
|
||||||
|
|
@ -14,6 +14,8 @@ export interface ProjectConfig {
|
||||||
enabled: boolean;
|
enabled: boolean;
|
||||||
/** Agent provider for this project (defaults to 'claude') */
|
/** Agent provider for this project (defaults to 'claude') */
|
||||||
provider?: ProviderId;
|
provider?: ProviderId;
|
||||||
|
/** Model override (e.g. 'claude-sonnet-4-5-20250514'). Falls back to provider default. */
|
||||||
|
model?: string;
|
||||||
/** When true, agents for this project use git worktrees for isolation */
|
/** When true, agents for this project use git worktrees for isolation */
|
||||||
useWorktrees?: boolean;
|
useWorktrees?: boolean;
|
||||||
/** When true, sidecar process is sandboxed via Landlock (Linux 5.13+, restricts filesystem access) */
|
/** When true, sidecar process is sandboxed via Landlock (Linux 5.13+, restricts filesystem access) */
|
||||||
|
|
@ -49,6 +51,8 @@ export function agentToProject(agent: GroupAgentConfig, groupCwd: string): Proje
|
||||||
cwd: agent.cwd ?? groupCwd,
|
cwd: agent.cwd ?? groupCwd,
|
||||||
profile: 'default',
|
profile: 'default',
|
||||||
enabled: agent.enabled,
|
enabled: agent.enabled,
|
||||||
|
provider: agent.provider,
|
||||||
|
model: agent.model,
|
||||||
isAgent: true,
|
isAgent: true,
|
||||||
agentRole: agent.role,
|
agentRole: agent.role,
|
||||||
systemPrompt: agent.systemPrompt,
|
systemPrompt: agent.systemPrompt,
|
||||||
|
|
@ -66,6 +70,9 @@ export interface GroupAgentConfig {
|
||||||
id: AgentId;
|
id: AgentId;
|
||||||
name: string;
|
name: string;
|
||||||
role: GroupAgentRole;
|
role: GroupAgentRole;
|
||||||
|
/** Agent provider (defaults to 'claude') */
|
||||||
|
provider?: ProviderId;
|
||||||
|
/** Model override (e.g. 'claude-sonnet-4-5-20250514'). Falls back to provider default. */
|
||||||
model?: string;
|
model?: string;
|
||||||
cwd?: string;
|
cwd?: string;
|
||||||
systemPrompt?: string;
|
systemPrompt?: string;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue