groove-dev 0.26.2 → 0.26.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -5,7 +5,7 @@
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
6
  <link rel="icon" type="image/png" href="/favicon.png" />
7
7
  <title>Groove GUI</title>
8
- <script type="module" crossorigin src="/assets/index-BC2Bhfv0.js"></script>
8
+ <script type="module" crossorigin src="/assets/index-CXBK_GNM.js"></script>
9
9
  <link rel="modulepreload" crossorigin href="/assets/vendor-C0HXlhrU.js">
10
10
  <link rel="modulepreload" crossorigin href="/assets/reactflow-BQPfi37R.js">
11
11
  <link rel="modulepreload" crossorigin href="/assets/codemirror-BBL3i_JW.js">
@@ -102,37 +102,60 @@ function ProviderCard({ provider, onKeyChange }) {
102
102
  }
103
103
  }
104
104
 
105
- // Ollama card
105
+ // Local models card
106
106
  if (isLocal) {
107
+ const installedCount = provider.models?.filter(m => !m.disabled)?.length || 0;
108
+ const goToModels = () => useGrooveStore.getState().setActiveView('models');
107
109
  return (
108
110
  <div className="flex flex-col rounded-lg border border-border-subtle bg-surface-1 overflow-hidden min-w-[220px]">
109
111
  <div className="flex items-center gap-2.5 px-4 py-3 border-b border-border-subtle">
110
- <StatusDot status={isReady ? 'running' : 'crashed'} size="sm" />
112
+ <StatusDot status={isReady && installedCount > 0 ? 'running' : 'crashed'} size="sm" />
111
113
  <span className="text-[13px] font-semibold text-text-0 font-sans">{provider.name}</span>
112
114
  <div className="flex-1" />
113
- {isReady ? (
114
- <Badge variant="success" className="text-2xs gap-1"><Check size={8} /> Ready</Badge>
115
+ {isReady && installedCount > 0 ? (
116
+ <Badge variant="success" className="text-2xs gap-1"><Check size={8} /> {installedCount} models</Badge>
117
+ ) : isReady ? (
118
+ <Badge variant="warning" className="text-2xs">No models pulled</Badge>
115
119
  ) : (
116
- <Badge variant="default" className="text-2xs">Not installed</Badge>
120
+ <Badge variant="default" className="text-2xs">Not set up</Badge>
117
121
  )}
118
122
  </div>
119
123
  <div className="flex-1">
120
124
  {ollamaOpen ? (
121
- <OllamaSetup isInstalled={isReady} onModelChange={onKeyChange} />
125
+ <>
126
+ <OllamaSetup isInstalled={isReady} onModelChange={onKeyChange} />
127
+ <div className="px-4 py-2 border-t border-border-subtle flex gap-2">
128
+ <Button variant="ghost" size="sm" onClick={() => setOllamaOpen(false)} className="flex-1 h-7 text-2xs">
129
+ Back
130
+ </Button>
131
+ <Button variant="secondary" size="sm" onClick={() => { setOllamaOpen(false); goToModels(); }} className="flex-1 h-7 text-2xs gap-1">
132
+ Models Tab
133
+ </Button>
134
+ </div>
135
+ </>
122
136
  ) : (
123
137
  <div className="px-4 py-3 flex flex-col h-full">
124
138
  <div className="text-xs text-text-3 font-sans flex-1">
125
- {isReady ? `${provider.models?.length || 0} models available` : 'Local AI models — free, private, no API key'}
139
+ {isReady && installedCount > 0
140
+ ? 'Full agentic runtime — tool calling, context rotation, zero cloud cost'
141
+ : isReady
142
+ ? 'Ollama is running. Pull a model to start using local agents.'
143
+ : 'Run any open-source model locally — free, private, fully offline. Requires Ollama.'}
144
+ </div>
145
+ <div className="flex gap-2 mt-3">
146
+ {!isReady ? (
147
+ <Button variant="primary" size="sm" onClick={() => setOllamaOpen(true)} className="flex-1 h-7 text-2xs gap-1.5">
148
+ <Cpu size={11} /> Set Up Ollama
149
+ </Button>
150
+ ) : (
151
+ <Button variant="primary" size="sm" onClick={() => setOllamaOpen(true)} className="flex-1 h-7 text-2xs gap-1.5">
152
+ <Cpu size={11} /> {installedCount > 0 ? 'Manage' : 'Pull Models'}
153
+ </Button>
154
+ )}
155
+ <Button variant="secondary" size="sm" onClick={goToModels} className="flex-1 h-7 text-2xs gap-1.5">
156
+ Models Tab
157
+ </Button>
126
158
  </div>
127
- <Button
128
- variant={isReady ? 'secondary' : 'primary'}
129
- size="sm"
130
- onClick={() => setOllamaOpen(true)}
131
- className="w-full h-7 text-2xs gap-1.5 mt-3"
132
- >
133
- <Cpu size={11} />
134
- {isReady ? 'Manage Models' : 'Set Up Ollama'}
135
- </Button>
136
159
  </div>
137
160
  )}
138
161
  </div>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "groove-dev",
3
- "version": "0.26.2",
3
+ "version": "0.26.4",
4
4
  "description": "Open-source agent orchestration layer — the AI company OS. Local model agent engine (GGUF/Ollama/llama-server), HuggingFace model browser, MCP integrations (Slack, Gmail, Stripe, 15+), agent scheduling (cron), business roles (CMO, CFO, EA). GUI dashboard, multi-agent coordination, zero cold-start, infinite sessions. Works with Claude Code, Codex, Gemini CLI, Ollama, any local model.",
5
5
  "license": "FSL-1.1-Apache-2.0",
6
6
  "author": "Groove Dev <hello@groovedev.ai> (https://groovedev.ai)",
@@ -173,8 +173,8 @@ export class ProcessManager {
173
173
  if (installed.length === 0) {
174
174
  throw new Error('No AI providers installed. Install Claude Code, Gemini CLI, Codex, or Ollama first.');
175
175
  }
176
- // Priority: claude-code > gemini > codex > ollama
177
- const priority = ['claude-code', 'gemini', 'codex', 'ollama'];
176
+ // Priority: claude-code > gemini > codex > local (local replaces ollama in UI)
177
+ const priority = ['claude-code', 'gemini', 'codex', 'local', 'ollama'];
178
178
  const best = priority.find((p) => installed.some((i) => i.id === p)) || installed[0].id;
179
179
  providerName = best;
180
180
  }
@@ -19,20 +19,26 @@ export function getProvider(name) {
19
19
  return providers[name] || null;
20
20
  }
21
21
 
22
+ // Providers hidden from UI but kept for backward compatibility
23
+ // (existing agents with provider='ollama' still resolve via getProvider)
24
+ const HIDDEN_PROVIDERS = new Set(['ollama']);
25
+
22
26
  export function listProviders() {
23
- return Object.entries(providers).map(([key, p]) => ({
24
- id: key,
25
- name: p.constructor.displayName,
26
- installed: p.constructor.isInstalled(),
27
- authType: p.constructor.authType,
28
- envKey: p.constructor.envKey || null,
29
- authHint: p.constructor.authHint || null,
30
- authStatus: p.constructor.isAuthenticated?.() || null,
31
- models: p.constructor.models,
32
- installCommand: p.constructor.installCommand(),
33
- canHotSwap: p.switchModel ? p.switchModel() : false,
34
- hardwareRequirements: p.constructor.hardwareRequirements?.() || null,
35
- }));
27
+ return Object.entries(providers)
28
+ .filter(([key]) => !HIDDEN_PROVIDERS.has(key))
29
+ .map(([key, p]) => ({
30
+ id: key,
31
+ name: p.constructor.displayName,
32
+ installed: p.constructor.isInstalled(),
33
+ authType: p.constructor.authType,
34
+ envKey: p.constructor.envKey || null,
35
+ authHint: p.constructor.authHint || null,
36
+ authStatus: p.constructor.isAuthenticated?.() || null,
37
+ models: p.constructor.models,
38
+ installCommand: p.constructor.installCommand(),
39
+ canHotSwap: p.switchModel ? p.switchModel() : false,
40
+ hardwareRequirements: p.constructor.hardwareRequirements?.() || null,
41
+ }));
36
42
  }
37
43
 
38
44
  export function getInstalledProviders() {
@@ -52,15 +52,37 @@ const TOOL_CALLING_MODELS = new Set([
52
52
 
53
53
  export class LocalProvider extends Provider {
54
54
  static name = 'local';
55
- static displayName = 'Local Models (Agent Loop)';
55
+ static displayName = 'Local Models';
56
56
  static command = 'ollama';
57
57
  static authType = 'local';
58
58
  static useAgentLoop = true;
59
59
 
60
+ // Only return models that are actually installed and ready to use
60
61
  static get models() {
61
- return OllamaProvider.models;
62
+ const installed = [];
63
+
64
+ // Ollama installed models
65
+ if (LocalProvider._hasOllama()) {
66
+ try {
67
+ const ollamaModels = OllamaProvider.getInstalledModels();
68
+ for (const m of ollamaModels) {
69
+ installed.push({
70
+ id: m.id, name: m.name || m.id,
71
+ tier: m.tier || 'medium', category: m.category || 'general',
72
+ });
73
+ }
74
+ } catch { /* Ollama not running */ }
75
+ }
76
+
77
+ // If nothing installed, show a hint instead of a blank list
78
+ if (installed.length === 0) {
79
+ return [{ id: '_none', name: 'No models installed — pull one with: ollama pull qwen2.5-coder:7b', tier: 'medium', disabled: true }];
80
+ }
81
+
82
+ return installed;
62
83
  }
63
84
 
85
+ // Full catalog for the Models browser (includes uninstalled)
64
86
  static get catalog() {
65
87
  return OllamaProvider.catalog;
66
88
  }
@@ -184,7 +184,7 @@ export class OllamaProvider extends Provider {
184
184
 
185
185
  static getInstalledModels() {
186
186
  try {
187
- const output = execSync('ollama list', { encoding: 'utf8', timeout: 10000 });
187
+ const output = execSync('ollama list', { encoding: 'utf8', timeout: 10000, stdio: ['pipe', 'pipe', 'pipe'] });
188
188
  const lines = output.split('\n').slice(1).filter(Boolean);
189
189
  return lines.map((line) => {
190
190
  const parts = line.split(/\s+/);