spidersan 0.6.0 → 0.9.0

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.
Files changed (214) hide show
  1. package/CHANGELOG.md +42 -52
  2. package/README.md +238 -3
  3. package/dist/bin/spidersan.d.ts.map +1 -1
  4. package/dist/bin/spidersan.js +13 -1
  5. package/dist/bin/spidersan.js.map +1 -1
  6. package/dist/commands/abandon.d.ts.map +1 -1
  7. package/dist/commands/abandon.js +1 -9
  8. package/dist/commands/abandon.js.map +1 -1
  9. package/dist/commands/ai.d.ts +15 -0
  10. package/dist/commands/ai.d.ts.map +1 -0
  11. package/dist/commands/ai.js +498 -0
  12. package/dist/commands/ai.js.map +1 -0
  13. package/dist/commands/auto.d.ts.map +1 -1
  14. package/dist/commands/auto.js +2 -1
  15. package/dist/commands/auto.js.map +1 -1
  16. package/dist/commands/bot.d.ts +16 -0
  17. package/dist/commands/bot.d.ts.map +1 -0
  18. package/dist/commands/bot.js +398 -0
  19. package/dist/commands/bot.js.map +1 -0
  20. package/dist/commands/conflicts.d.ts.map +1 -1
  21. package/dist/commands/conflicts.js +260 -277
  22. package/dist/commands/conflicts.js.map +1 -1
  23. package/dist/commands/context.d.ts +8 -0
  24. package/dist/commands/context.d.ts.map +1 -0
  25. package/dist/commands/context.js +104 -0
  26. package/dist/commands/context.js.map +1 -0
  27. package/dist/commands/cross-conflicts.d.ts.map +1 -1
  28. package/dist/commands/cross-conflicts.js +1 -41
  29. package/dist/commands/cross-conflicts.js.map +1 -1
  30. package/dist/commands/depends.d.ts.map +1 -1
  31. package/dist/commands/depends.js +1 -9
  32. package/dist/commands/depends.js.map +1 -1
  33. package/dist/commands/fleet-status.d.ts +14 -0
  34. package/dist/commands/fleet-status.d.ts.map +1 -0
  35. package/dist/commands/fleet-status.js +127 -0
  36. package/dist/commands/fleet-status.js.map +1 -0
  37. package/dist/commands/git-watch.d.ts +24 -0
  38. package/dist/commands/git-watch.d.ts.map +1 -0
  39. package/dist/commands/git-watch.js +84 -0
  40. package/dist/commands/git-watch.js.map +1 -0
  41. package/dist/commands/index.d.ts +5 -0
  42. package/dist/commands/index.d.ts.map +1 -1
  43. package/dist/commands/index.js +7 -0
  44. package/dist/commands/index.js.map +1 -1
  45. package/dist/commands/merge-order.d.ts.map +1 -1
  46. package/dist/commands/merge-order.js +18 -67
  47. package/dist/commands/merge-order.js.map +1 -1
  48. package/dist/commands/merged.d.ts.map +1 -1
  49. package/dist/commands/merged.js +1 -9
  50. package/dist/commands/merged.js.map +1 -1
  51. package/dist/commands/pulse.d.ts.map +1 -1
  52. package/dist/commands/pulse.js +134 -63
  53. package/dist/commands/pulse.js.map +1 -1
  54. package/dist/commands/queen.d.ts.map +1 -1
  55. package/dist/commands/queen.js +11 -7
  56. package/dist/commands/queen.js.map +1 -1
  57. package/dist/commands/ready-check.d.ts +2 -1
  58. package/dist/commands/ready-check.d.ts.map +1 -1
  59. package/dist/commands/ready-check.js +6 -30
  60. package/dist/commands/ready-check.js.map +1 -1
  61. package/dist/commands/register.d.ts.map +1 -1
  62. package/dist/commands/register.js +7 -29
  63. package/dist/commands/register.js.map +1 -1
  64. package/dist/commands/torrent.d.ts.map +1 -1
  65. package/dist/commands/torrent.js +29 -18
  66. package/dist/commands/torrent.js.map +1 -1
  67. package/dist/commands/watch.d.ts.map +1 -1
  68. package/dist/commands/watch.js +13 -34
  69. package/dist/commands/watch.js.map +1 -1
  70. package/dist/lib/ai/context-builder.d.ts +16 -0
  71. package/dist/lib/ai/context-builder.d.ts.map +1 -0
  72. package/dist/lib/ai/context-builder.js +216 -0
  73. package/dist/lib/ai/context-builder.js.map +1 -0
  74. package/dist/lib/ai/event-handler.d.ts +21 -0
  75. package/dist/lib/ai/event-handler.d.ts.map +1 -0
  76. package/dist/lib/ai/event-handler.js +98 -0
  77. package/dist/lib/ai/event-handler.js.map +1 -0
  78. package/dist/lib/ai/index.d.ts +13 -0
  79. package/dist/lib/ai/index.d.ts.map +1 -0
  80. package/dist/lib/ai/index.js +11 -0
  81. package/dist/lib/ai/index.js.map +1 -0
  82. package/dist/lib/ai/llm-client.d.ts +37 -0
  83. package/dist/lib/ai/llm-client.d.ts.map +1 -0
  84. package/dist/lib/ai/llm-client.js +225 -0
  85. package/dist/lib/ai/llm-client.js.map +1 -0
  86. package/dist/lib/ai/reasoner.d.ts +11 -0
  87. package/dist/lib/ai/reasoner.d.ts.map +1 -0
  88. package/dist/lib/ai/reasoner.js +246 -0
  89. package/dist/lib/ai/reasoner.js.map +1 -0
  90. package/dist/lib/ai/setup.d.ts +40 -0
  91. package/dist/lib/ai/setup.d.ts.map +1 -0
  92. package/dist/lib/ai/setup.js +154 -0
  93. package/dist/lib/ai/setup.js.map +1 -0
  94. package/dist/lib/ai/types.d.ts +135 -0
  95. package/dist/lib/ai/types.d.ts.map +1 -0
  96. package/dist/lib/ai/types.js +39 -0
  97. package/dist/lib/ai/types.js.map +1 -0
  98. package/dist/lib/colony-subscriber.d.ts +15 -12
  99. package/dist/lib/colony-subscriber.d.ts.map +1 -1
  100. package/dist/lib/colony-subscriber.js +146 -65
  101. package/dist/lib/colony-subscriber.js.map +1 -1
  102. package/dist/lib/config.d.ts.map +1 -1
  103. package/dist/lib/config.js +18 -4
  104. package/dist/lib/config.js.map +1 -1
  105. package/dist/lib/conflict-analyzer.d.ts +33 -0
  106. package/dist/lib/conflict-analyzer.d.ts.map +1 -0
  107. package/dist/lib/conflict-analyzer.js +114 -0
  108. package/dist/lib/conflict-analyzer.js.map +1 -0
  109. package/dist/lib/conflict-renderer.d.ts +7 -0
  110. package/dist/lib/conflict-renderer.d.ts.map +1 -0
  111. package/dist/lib/conflict-renderer.js +162 -0
  112. package/dist/lib/conflict-renderer.js.map +1 -0
  113. package/dist/lib/conflict-tier.d.ts +20 -0
  114. package/dist/lib/conflict-tier.d.ts.map +1 -0
  115. package/dist/lib/conflict-tier.js +49 -0
  116. package/dist/lib/conflict-tier.js.map +1 -0
  117. package/dist/lib/crypto.js +1 -1
  118. package/dist/lib/crypto.js.map +1 -1
  119. package/dist/lib/git-events-subscriber.d.ts +59 -0
  120. package/dist/lib/git-events-subscriber.d.ts.map +1 -0
  121. package/dist/lib/git-events-subscriber.js +779 -0
  122. package/dist/lib/git-events-subscriber.js.map +1 -0
  123. package/dist/lib/git.d.ts +15 -0
  124. package/dist/lib/git.d.ts.map +1 -0
  125. package/dist/lib/git.js +180 -0
  126. package/dist/lib/git.js.map +1 -0
  127. package/dist/lib/github.d.ts.map +1 -1
  128. package/dist/lib/github.js +14 -9
  129. package/dist/lib/github.js.map +1 -1
  130. package/dist/lib/graph.d.ts +23 -0
  131. package/dist/lib/graph.d.ts.map +1 -0
  132. package/dist/lib/graph.js +134 -0
  133. package/dist/lib/graph.js.map +1 -0
  134. package/dist/lib/hub.d.ts +31 -0
  135. package/dist/lib/hub.d.ts.map +1 -0
  136. package/dist/lib/hub.js +92 -0
  137. package/dist/lib/hub.js.map +1 -0
  138. package/dist/lib/remote-drift.d.ts +60 -0
  139. package/dist/lib/remote-drift.d.ts.map +1 -0
  140. package/dist/lib/remote-drift.js +225 -0
  141. package/dist/lib/remote-drift.js.map +1 -0
  142. package/dist/lib/salvage-analyzer.d.ts.map +1 -1
  143. package/dist/lib/salvage-analyzer.js +2 -3
  144. package/dist/lib/salvage-analyzer.js.map +1 -1
  145. package/dist/lib/security.d.ts +11 -0
  146. package/dist/lib/security.d.ts.map +1 -1
  147. package/dist/lib/security.js +24 -1
  148. package/dist/lib/security.js.map +1 -1
  149. package/dist/lib/session-logger.d.ts +54 -0
  150. package/dist/lib/session-logger.d.ts.map +1 -0
  151. package/dist/lib/session-logger.js +136 -0
  152. package/dist/lib/session-logger.js.map +1 -0
  153. package/dist/storage/adapter.d.ts +4 -0
  154. package/dist/storage/adapter.d.ts.map +1 -1
  155. package/dist/storage/branch-registry-store.d.ts +13 -0
  156. package/dist/storage/branch-registry-store.d.ts.map +1 -0
  157. package/dist/storage/branch-registry-store.js +2 -0
  158. package/dist/storage/branch-registry-store.js.map +1 -0
  159. package/dist/storage/factory.d.ts +4 -0
  160. package/dist/storage/factory.d.ts.map +1 -1
  161. package/dist/storage/factory.js +25 -9
  162. package/dist/storage/factory.js.map +1 -1
  163. package/dist/storage/git-messages.d.ts +53 -0
  164. package/dist/storage/git-messages.d.ts.map +1 -0
  165. package/dist/storage/git-messages.js +376 -0
  166. package/dist/storage/git-messages.js.map +1 -0
  167. package/dist/storage/index.d.ts +5 -0
  168. package/dist/storage/index.d.ts.map +1 -1
  169. package/dist/storage/index.js +5 -0
  170. package/dist/storage/index.js.map +1 -1
  171. package/dist/storage/json-branch-registry-store.d.ts +19 -0
  172. package/dist/storage/json-branch-registry-store.d.ts.map +1 -0
  173. package/dist/storage/json-branch-registry-store.js +112 -0
  174. package/dist/storage/json-branch-registry-store.js.map +1 -0
  175. package/dist/storage/local-messages.d.ts +37 -0
  176. package/dist/storage/local-messages.d.ts.map +1 -0
  177. package/dist/storage/local-messages.js +151 -0
  178. package/dist/storage/local-messages.js.map +1 -0
  179. package/dist/storage/local.d.ts +2 -6
  180. package/dist/storage/local.d.ts.map +1 -1
  181. package/dist/storage/local.js +13 -76
  182. package/dist/storage/local.js.map +1 -1
  183. package/dist/storage/memory-branch-registry-store.d.ts +16 -0
  184. package/dist/storage/memory-branch-registry-store.d.ts.map +1 -0
  185. package/dist/storage/memory-branch-registry-store.js +65 -0
  186. package/dist/storage/memory-branch-registry-store.js.map +1 -0
  187. package/dist/storage/message-adapter.d.ts +86 -0
  188. package/dist/storage/message-adapter.d.ts.map +1 -0
  189. package/dist/storage/message-adapter.js +28 -0
  190. package/dist/storage/message-adapter.js.map +1 -0
  191. package/dist/storage/message-factory.d.ts +36 -0
  192. package/dist/storage/message-factory.d.ts.map +1 -0
  193. package/dist/storage/message-factory.js +99 -0
  194. package/dist/storage/message-factory.js.map +1 -0
  195. package/dist/storage/mycmail-adapter.d.ts +38 -0
  196. package/dist/storage/mycmail-adapter.d.ts.map +1 -0
  197. package/dist/storage/mycmail-adapter.js +300 -0
  198. package/dist/storage/mycmail-adapter.js.map +1 -0
  199. package/dist/storage/supabase-registry-sync-client-impl.d.ts +46 -0
  200. package/dist/storage/supabase-registry-sync-client-impl.d.ts.map +1 -0
  201. package/dist/storage/supabase-registry-sync-client-impl.js +322 -0
  202. package/dist/storage/supabase-registry-sync-client-impl.js.map +1 -0
  203. package/dist/storage/supabase-registry-sync-client.d.ts +9 -0
  204. package/dist/storage/supabase-registry-sync-client.d.ts.map +1 -0
  205. package/dist/storage/supabase-registry-sync-client.js +2 -0
  206. package/dist/storage/supabase-registry-sync-client.js.map +1 -0
  207. package/dist/storage/supabase.d.ts +8 -46
  208. package/dist/storage/supabase.d.ts.map +1 -1
  209. package/dist/storage/supabase.js +30 -342
  210. package/dist/storage/supabase.js.map +1 -1
  211. package/dist/tui/screen.d.ts.map +1 -1
  212. package/dist/tui/screen.js +5 -3
  213. package/dist/tui/screen.js.map +1 -1
  214. package/package.json +92 -90
@@ -0,0 +1,225 @@
1
+ /**
2
+ * Spidersan AI Core — LLM Client
3
+ *
4
+ * Unified interface for LLM providers with automatic fallback chain:
5
+ * LM Studio (Gemma 4) → Ollama → Hosted API (api.spidersan.net) → GitHub Copilot
6
+ *
7
+ * Tier routing (auto, no explicit provider):
8
+ * Tier 1: localhost:1234 / :8082 (lmstudio)
9
+ * Tier 2: localhost:11434 (ollama)
10
+ * Tier 3: api.spidersan.net (hosted) — requires API key
11
+ * Remote: copilot — only with allowRemoteFallback or explicit config
12
+ */
13
+ import { execFile } from 'child_process';
14
+ import { promisify } from 'util';
15
+ import { DEFAULT_LLM_CONFIGS } from './types.js';
16
+ import { loadAiConfig } from './setup.js';
17
+ const execFileAsync = promisify(execFile);
18
+ export function resolveConfig(overrides) {
19
+ const providerEnv = process.env['SPIDERSAN_LLM'];
20
+ const provider = overrides?.provider ?? providerEnv ?? 'lmstudio';
21
+ const base = DEFAULT_LLM_CONFIGS[provider] ?? DEFAULT_LLM_CONFIGS.lmstudio;
22
+ return {
23
+ ...base,
24
+ ...overrides,
25
+ provider,
26
+ model: overrides?.model ?? process.env['SPIDERSAN_LLM_MODEL'] ?? base.model,
27
+ baseUrl: overrides?.baseUrl ?? process.env['SPIDERSAN_LLM_URL'] ?? base.baseUrl,
28
+ };
29
+ }
30
+ async function getGitHubToken() {
31
+ try {
32
+ const { stdout } = await execFileAsync('gh', ['auth', 'token'], { timeout: 5000 });
33
+ return stdout.trim();
34
+ }
35
+ catch {
36
+ throw new Error('GitHub CLI not authenticated. Run `gh auth login` first.');
37
+ }
38
+ }
39
+ async function chatOpenAICompatible(config, messages, apiKey) {
40
+ // Normalize: strip trailing /v1 so we always build the full path ourselves.
41
+ // Handles both http://host:port (env var form) and http://host:port/v1 (default form).
42
+ const base = config.baseUrl.replace(/\/v1\/?$/, '');
43
+ const url = `${base}/v1/chat/completions`;
44
+ const headers = { 'Content-Type': 'application/json' };
45
+ if (apiKey)
46
+ headers['Authorization'] = `Bearer ${apiKey}`;
47
+ const resp = await fetch(url, {
48
+ method: 'POST',
49
+ headers,
50
+ body: JSON.stringify({
51
+ model: config.model,
52
+ messages,
53
+ temperature: config.temperature,
54
+ max_tokens: config.maxTokens,
55
+ }),
56
+ signal: AbortSignal.timeout(60_000),
57
+ });
58
+ if (!resp.ok) {
59
+ const body = await resp.text().catch(() => '');
60
+ throw new Error(`LLM ${config.provider} returned ${resp.status}: ${body.slice(0, 200)}`);
61
+ }
62
+ const data = await resp.json();
63
+ return {
64
+ content: data.choices[0]?.message?.content ?? '',
65
+ model: data.model ?? config.model,
66
+ provider: config.provider,
67
+ promptTokens: data.usage?.prompt_tokens ?? 0,
68
+ completionTokens: data.usage?.completion_tokens ?? 0,
69
+ };
70
+ }
71
+ async function chatOllama(config, messages) {
72
+ const url = `${config.baseUrl}/api/chat`;
73
+ const resp = await fetch(url, {
74
+ method: 'POST',
75
+ headers: { 'Content-Type': 'application/json' },
76
+ body: JSON.stringify({
77
+ model: config.model,
78
+ messages,
79
+ stream: false,
80
+ options: { temperature: config.temperature, num_predict: config.maxTokens },
81
+ }),
82
+ signal: AbortSignal.timeout(120_000),
83
+ });
84
+ if (!resp.ok) {
85
+ throw new Error(`Ollama returned ${resp.status}`);
86
+ }
87
+ const data = await resp.json();
88
+ return {
89
+ content: data.message?.content ?? '',
90
+ model: data.model ?? config.model,
91
+ provider: 'ollama',
92
+ promptTokens: data.prompt_eval_count ?? 0,
93
+ completionTokens: data.eval_count ?? 0,
94
+ };
95
+ }
96
+ async function chatWithProvider(config, messages, apiKey) {
97
+ switch (config.provider) {
98
+ case 'lmstudio':
99
+ return chatOpenAICompatible(config, messages);
100
+ case 'copilot': {
101
+ const token = await getGitHubToken();
102
+ return chatOpenAICompatible(config, messages, token);
103
+ }
104
+ case 'ollama':
105
+ return chatOllama(config, messages);
106
+ case 'hosted': {
107
+ const key = apiKey ?? config.apiKey;
108
+ if (!key)
109
+ throw new Error('No API key for hosted tier. Run: spidersan ai-setup');
110
+ return chatOpenAICompatible(config, messages, key);
111
+ }
112
+ default:
113
+ throw new Error(`Unknown LLM provider: ${config.provider}`);
114
+ }
115
+ }
116
+ // Local-first fallback: never send repo context to remote without opt-in
117
+ const LOCAL_FALLBACK = ['lmstudio', 'ollama'];
118
+ const REMOTE_PROVIDERS = ['hosted', 'copilot'];
119
+ /** Resolve the hosted API key from env var or persisted config. */
120
+ async function resolveHostedApiKey() {
121
+ const envKey = process.env['SPIDERSAN_API_KEY'];
122
+ if (envKey)
123
+ return envKey;
124
+ const config = await loadAiConfig();
125
+ return config?.apiKey ?? undefined;
126
+ }
127
+ /**
128
+ * Send a chat completion request with automatic fallback.
129
+ *
130
+ * Priority order (auto-routing when no explicit provider set):
131
+ * 1. lmstudio (local)
132
+ * 2. ollama (local)
133
+ * 3. hosted (api.spidersan.net) — only if API key available
134
+ * 4. copilot — only if allowRemoteFallback or explicitly configured
135
+ */
136
+ export async function chat(messages, configOverrides) {
137
+ const { allowRemoteFallback, ...llmOverrides } = configOverrides ?? {};
138
+ const config = resolveConfig(llmOverrides);
139
+ const isRemote = REMOTE_PROVIDERS.includes(config.provider);
140
+ const errors = [];
141
+ // Resolve hosted API key once (avoid repeated config reads in fallback loop)
142
+ const hostedApiKey = await resolveHostedApiKey();
143
+ // Try configured provider first
144
+ try {
145
+ return await chatWithProvider(config, messages, config.provider === 'hosted' ? hostedApiKey : undefined);
146
+ }
147
+ catch (err) {
148
+ errors.push({ provider: config.provider, error: err.message });
149
+ }
150
+ // Local fallback chain
151
+ for (const fallback of LOCAL_FALLBACK) {
152
+ if (fallback === config.provider)
153
+ continue;
154
+ try {
155
+ const result = await chatWithProvider(DEFAULT_LLM_CONFIGS[fallback], messages);
156
+ result.provider = fallback;
157
+ return result;
158
+ }
159
+ catch (err) {
160
+ errors.push({ provider: fallback, error: err.message });
161
+ }
162
+ }
163
+ // Remote fallback: hosted (if key available + tier allows), then copilot (if opted in)
164
+ //
165
+ // Tier routing: if the user configured free/contributor tier via ai-setup, hosted is
166
+ // always a valid fallback when local is unavailable — no explicit allowRemoteFallback needed.
167
+ const savedConfig = await loadAiConfig();
168
+ const tierAllowsHosted = savedConfig?.tier === 'free' || savedConfig?.tier === 'contributor';
169
+ if (allowRemoteFallback || isRemote || tierAllowsHosted) {
170
+ // hosted — only if we have a key
171
+ if (config.provider !== 'hosted' && hostedApiKey) {
172
+ try {
173
+ const result = await chatWithProvider(DEFAULT_LLM_CONFIGS.hosted, messages, hostedApiKey);
174
+ result.provider = 'hosted';
175
+ return result;
176
+ }
177
+ catch (err) {
178
+ errors.push({ provider: 'hosted', error: err.message });
179
+ }
180
+ }
181
+ // copilot — only with explicit allowRemoteFallback (never auto-routed)
182
+ if (allowRemoteFallback || isRemote) {
183
+ for (const fallback of ['copilot']) {
184
+ if (fallback === config.provider)
185
+ continue;
186
+ try {
187
+ const result = await chatWithProvider(DEFAULT_LLM_CONFIGS[fallback], messages);
188
+ result.provider = fallback;
189
+ return result;
190
+ }
191
+ catch (err) {
192
+ errors.push({ provider: fallback, error: err.message });
193
+ }
194
+ }
195
+ }
196
+ }
197
+ const summary = errors.map(e => ` ${e.provider}: ${e.error}`).join('\n');
198
+ throw new Error(`All LLM providers failed:\n${summary}`);
199
+ }
200
+ /**
201
+ * Quick health check — can we reach the configured provider?
202
+ */
203
+ export async function ping(configOverrides) {
204
+ const config = resolveConfig(configOverrides);
205
+ const start = Date.now();
206
+ try {
207
+ const result = await chat([{ role: 'user', content: 'Reply with just "ok".' }], { ...config, maxTokens: 10 });
208
+ return {
209
+ ok: true,
210
+ provider: result.provider,
211
+ model: result.model,
212
+ latencyMs: Date.now() - start,
213
+ };
214
+ }
215
+ catch (err) {
216
+ return {
217
+ ok: false,
218
+ provider: config.provider,
219
+ model: config.model,
220
+ latencyMs: Date.now() - start,
221
+ error: err.message,
222
+ };
223
+ }
224
+ }
225
+ //# sourceMappingURL=llm-client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"llm-client.js","sourceRoot":"","sources":["../../../src/lib/ai/llm-client.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Spidersan AI Core — Reasoner
3
+ *
4
+ * Sends SpiderContext + user question to LLM with spidersan-aware system prompt.
5
+ * The scenario playbook is embedded as a TS constant (no file I/O at runtime).
6
+ */
7
+ import type { LLMConfig, ReasoningRequest, ReasoningResult } from './types.js';
8
+ export declare function reason(request: ReasoningRequest, llmOverrides?: Partial<LLMConfig> & {
9
+ allowRemoteFallback?: boolean;
10
+ }): Promise<ReasoningResult>;
11
+ //# sourceMappingURL=reasoner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reasoner.d.ts","sourceRoot":"","sources":["../../../src/lib/ai/reasoner.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAGH,OAAO,KAAK,EACV,SAAS,EAGT,gBAAgB,EAChB,eAAe,EAEhB,MAAM,YAAY,CAAC;AAqMpB,wBAAsB,MAAM,CAC1B,OAAO,EAAE,gBAAgB,EACzB,YAAY,CAAC,EAAE,OAAO,CAAC,SAAS,CAAC,GAAG;IAAE,mBAAmB,CAAC,EAAE,OAAO,CAAA;CAAE,GACpE,OAAO,CAAC,eAAe,CAAC,CAkE1B"}
@@ -0,0 +1,246 @@
1
+ /**
2
+ * Spidersan AI Core — Reasoner
3
+ *
4
+ * Sends SpiderContext + user question to LLM with spidersan-aware system prompt.
5
+ * The scenario playbook is embedded as a TS constant (no file I/O at runtime).
6
+ */
7
+ import { chat } from './llm-client.js';
8
+ // ─── Known Command Registry ──────────────────────────────────
9
+ // Derived from src/commands/* + spidersan --help. Used to flag model hallucinations at output time.
10
+ // Update when new commands ship.
11
+ const KNOWN_SPIDERSAN_COMMANDS = new Set([
12
+ 'abandon', 'advise', 'ai-ping', 'ai-setup', 'ai-telemetry', 'ask', 'auto',
13
+ 'bot', 'check-opt-out', 'cleanup', 'collab', 'config', 'conflicts', 'context',
14
+ 'cross-conflicts', 'daily', 'dashboard', 'depends', 'doctor', 'explain',
15
+ 'fleet-status', 'git-watch', 'github-sync', 'init', 'list', 'log', 'merge-order', 'merged',
16
+ 'pulse', 'queen', 'ready-check', 'rebase-helper', 'register', 'registry-sync',
17
+ 'rescue', 'stale', 'sync', 'sync-advisor', 'torrent', 'watch', 'welcome',
18
+ ]);
19
+ // ─── Embedded Scenario Playbook ─────────────────────────────
20
+ // Optimized via PHC: Sonnet 4.6 proposer + Gemma 4 executor (score: 0.9718)
21
+ // Source: promptclimb/prompt_sonnet_spidersan.md
22
+ const SCENARIO_PLAYBOOK = `## Spidersan Command Reference & Scenarios
23
+
24
+ ### SCENARIO 1: New Repo Setup
25
+ Situation: Starting work in a new or uninitialized repository.
26
+ IMPORTANT: When setting up Spidersan for the first time, ALWAYS recommend \`welcome\` as the very first command — it provides guided onboarding before any other setup step.
27
+ Commands: welcome → spidersan init → register --files "f1,f2" --agent <id> → doctor → list
28
+ Reminder: Always verify repo health with \`doctor\` before proceeding.
29
+
30
+ ### SCENARIO 2: Ghost/Abandoned Branches
31
+ Situation: Unregistered branches drifting — some with salvageable code, others dead.
32
+ Commands: rescue --scan → rescue --symbols <branch> → sync
33
+ Decision: If \`rescue --symbols\` identifies salvageable code → cherry-pick. If empty/WIP → abandon.
34
+
35
+ ### SCENARIO 3: File Conflicts (Tier System)
36
+ - TIER 3 (BLOCK): Critical file (auth, secrets, env) — halt, coordinate immediately.
37
+ - TIER 2 (PAUSE): Important file (package.json, config, server) — coordinate before merging.
38
+ - TIER 1 (WARN): Adjacent/related files — proceed with caution.
39
+ Commands: conflicts → conflicts --tier 3 → conflicts --strict → conflicts --json → conflicts --wake
40
+ Reminder: Tier escalation is mandatory for TIER 3 conflicts. Always reference the TIER level when reporting conflicts.
41
+
42
+ ### SCENARIO 4: WIP / Not-Ready Code
43
+ Situation: Branch has TODO, FIXME, HACK, WIP, SPIDER_TRAP, UNFINISHED markers.
44
+ Commands: ready-check → ready-check --json
45
+ Rule: Never merge a branch that fails \`ready-check\`. All markers must be resolved first.
46
+
47
+ ### SCENARIO 5: Real-Time Monitoring
48
+ Commands: watch --agent <id> → auto start/stop/status
49
+ Warning: Watch CPU loop if ghost branches claim same files. Fix: sync first.
50
+ Recommendation: Use \`auto\` for periodic monitoring of agent activity.
51
+
52
+ ### SCENARIO 6: Task Decomposition (Torrent)
53
+ Situation: Breaking a single task into sequential child subtasks for a serial workflow.
54
+ IMPORTANT: \`torrent\` is for sequential subtask decomposition within a single workflow. For dispatching parallel independent agents simultaneously, use \`queen spawn\` instead (see SCENARIO 9).
55
+ Commands: torrent decompose <TASK> -c N → torrent create <CHILD> -a <agent> → torrent complete <CHILD> → torrent tree → torrent merge-order
56
+ Reminder: Ensure all child tasks are marked completed before executing \`torrent merge-order\`.
57
+
58
+ ### SCENARIO 7: Tangled Dependencies
59
+ Commands: merge-order → merge-order --json → merge-order --blocking-count → depends <branch> → conflicts --wake
60
+ Pattern: Merge foundation first → resolve blockers → re-check dependencies → merge dependents.
61
+
62
+ ### SCENARIO 8: Stale Branch Cleanup
63
+ Commands: stale → cleanup → abandon <branch> → merged <branch> → sync
64
+ Triage: sync → stale → cleanup → verify with \`list\`.
65
+
66
+ ### SCENARIO 9: Queen Dispatch (Parallel Agents)
67
+ Situation: Parallelizing work across multiple sub-agents or sub-spidersans simultaneously.
68
+ IMPORTANT: When asked to parallelize across multiple agents or sub-spidersans, ALWAYS recommend \`queen spawn\` FIRST — this is the dedicated parallel dispatch command. Do NOT suggest \`torrent decompose\` for parallel agent dispatch. Torrent = sequential subtasks. Queen = parallel agents.
69
+ Commands: queen spawn --task "desc" → queen status → queen dissolve <id>
70
+ Rule: Dissolve inactive queens to prevent resource leakage. Use \`queen status\` to monitor all active queens.
71
+
72
+ ### SCENARIO 10: Hive Lifecycle
73
+ Hive Operations (via envoak MCP, not spidersan CLI): hive enlist → hive signal --status in-progress → hive heartbeat → hive probe → hive accept → hive close → hive broadcast → hive broadcast-ack → hive broadcast-status → hive cns → hive gc
74
+ Reminder: Use \`probe\` to verify hive readiness before proceeding to \`accept\`.
75
+
76
+ ### SCENARIO 11: Cross-Machine Conflicts
77
+ Commands: pulse → cross-conflicts → cross-conflicts --local → doctor --remote
78
+ Pattern: Use \`pulse\` to identify cross-machine activity, followed by conflict resolution steps.
79
+
80
+ ### SCENARIO 12: Activity & Context
81
+ Situation: Investigating recent activity, mysterious branches, or understanding what happened over a time period.
82
+ IMPORTANT: When investigating recent or mysterious activity, ALWAYS start with \`log\` and \`daily\` commands first. Do NOT skip to \`rescue\` — log and daily give you the evidence base before taking any action.
83
+ Commands: log → log --since 7d → log --branch <name> → daily → daily --branch <name> --tldr → daily --context --branch <name>
84
+ Recommendation: Use \`daily --context\` for detailed summaries of recent activity.
85
+
86
+ ### SCENARIO 13: AI-Assisted Analysis & Proactive Advice
87
+ Situation: Complex repo state that needs intelligent analysis — stale branches, hidden conflicts, unknown branches.
88
+ IMPORTANT: When asked for proactive recommendations or advice about the repo state, ALWAYS recommend \`spidersan advise\` as your FIRST command. Do not give direct advice yourself — \`advise\` is the dedicated entry point for AI-driven proactive recommendations. Only follow with \`context\`, \`ask\`, or \`explain\` after \`advise\` has been run.
89
+ Commands: ai-ping → context → context --json → ask "<question>" → advise → explain <branch>
90
+ Decision: Use \`ai-ping\` to verify AI readiness. Use \`advise\` for proactive recommendations. Use \`ask\` for specific questions. Use \`explain\` for branch investigation.
91
+
92
+ ### SCENARIO 13B: Suspicious / Unknown Branch Investigation
93
+ Situation: A branch with an unknown owner, suspicious name (e.g. shadow/, unknown-agent), or no clear origin.
94
+ IMPORTANT: For any unknown or suspicious branch, ALWAYS use \`spidersan explain <branch>\` first. Do NOT jump to \`rescue\` or mention \`queen\` or \`hive\` in this context. \`explain\` is the dedicated command for branch investigation. After \`explain\`, use \`rescue\` only if code needs salvaging.
95
+ Commands: explain <branch> → rescue --symbols <branch> → abandon <branch>
96
+ Decision: explain first → if salvageable → rescue → if empty/dead → abandon.
97
+
98
+ ### SCENARIO 14: Branch Abandonment & Rebase Recovery
99
+ Situation: Dead branches (0 commits ahead) cluttering registry, or a branch stuck mid-rebase.
100
+ Commands: abandon <branch> → list --status abandoned → rebase-helper
101
+ Decision: If branch has 0 unique commits → abandon. If \`.git/rebase-merge\` exists → use \`rebase-helper\` to diagnose. After resolution → register updated files.
102
+
103
+ ### SCENARIO 15: Remote Sync & GitHub Integration
104
+ Situation: Local registry out of sync with remote. Need to see GitHub PR/CI state.
105
+ Commands: github-sync → sync-advisor → registry-sync --pull → registry-sync --push
106
+ Decision: Use \`github-sync\` for remote branch/PR overview. \`sync-advisor\` for recommendations. \`registry-sync\` to align local ↔ Supabase registry.
107
+
108
+ ### SCENARIO 16: Configuration & Onboarding
109
+ Situation: Setting up Spidersan for the first time or changing LLM/stale settings.
110
+ IMPORTANT: ALWAYS start with \`welcome\` for first-time setup — never skip it and go straight to \`init\`. \`welcome\` provides the guided onboarding sequence that ensures correct configuration before initialization.
111
+ Commands: welcome → config list → config set llm.provider <provider> → config set stale.days <n>
112
+ Decision: Use \`welcome\` for guided setup. Use \`config\` for tweaks. Set \`llm.provider\` to lmstudio (local), ollama (local), or copilot (remote, opt-in).
113
+
114
+ ### SCENARIO 17: Dependency Management
115
+ Situation: Multiple branches with merge-order dependencies that must be respected.
116
+ IMPORTANT: When dealing with a dependency chain across branches, ALWAYS use \`depends <branch>\` to declare dependencies first, then \`merge-order\` to determine the correct merge sequence. Missing \`depends\` leads to broken merge order.
117
+ Commands: depends <branch> --on <other> → depends --show → merge-order
118
+ Decision: Set dependencies before merging. If circular dependency detected → break the cycle by removing one dependency. \`merge-order\` respects the dependency graph.
119
+
120
+ ### SCENARIO 18: Fleet Dashboard & Monitoring
121
+ Situation: Need real-time visibility into branch state, conflicts, and agent activity.
122
+ Commands: dashboard → doctor --remote → pulse → cross-conflicts
123
+ Decision: Use \`dashboard\` for TUI overview. Use \`doctor --remote\` for repo status. Use \`pulse\` and \`cross-conflicts\` for fleet-wide coordination.
124
+
125
+ ### Decision Tree
126
+ Starting new work? → welcome → spidersan init → register + doctor
127
+ Found unknown/suspicious branch? → explain <branch> (NOT rescue, NOT queen, NOT hive)
128
+ About to merge? → conflicts → ready-check → depends --show → merge-order
129
+ Branches piling up? → stale → cleanup → sync
130
+ Parallelizing across agents? → queen spawn → queen status (NOT torrent decompose)
131
+ Multi-agent serial task? → torrent decompose → create → complete → tree
132
+ Blocked by another? → conflicts --wake → depends → merge-order
133
+ Cross-machine issues? → pulse → cross-conflicts → doctor --remote
134
+ Need a summary? → daily --branch <name> --tldr
135
+ Investigating recent activity? → log → log --since 7d → daily → daily --context
136
+ Want proactive advice? → advise (ALWAYS first for proactive recommendations)
137
+ Require AI insight? → ai-ping → context → ask/advise/explain
138
+ Dead branches? → abandon + list --status abandoned
139
+ Stuck rebase? → rebase-helper
140
+ Remote out of sync? → github-sync → sync-advisor → registry-sync
141
+ Setting up for first time? → welcome → config (ALWAYS welcome before init)
142
+ Dependency chain? → depends <branch> --on <other> → depends --show → merge-order`;
143
+ // ─── System Prompt Builder ──────────────────────────────────
144
+ function buildSystemPrompt(mode) {
145
+ const persona = `You are Spidersan — the gitops coordinator for a multi-agent AI development fleet called 'the flock'. You speak with authority: direct, structured, actionable. Use real spidersan commands. Reference tier levels. Give git commands where needed. Never speculate — if you don't have enough context, say so.`;
146
+ const modeInstructions = {
147
+ ask: 'Answer the user\'s question using the repository context provided. Recommend specific commands. Structure your response with clear action items per agent if applicable.',
148
+ advise: 'Analyze the current repository state and proactively recommend what should be done next. Prioritize: TIER 3 blocks first, then stale cleanup, then merge ordering. Be concise.',
149
+ explain: 'Explain the branch\'s purpose, owner, status, and any conflicts. Use activity log and daily collab entries if available. Be factual.',
150
+ };
151
+ return `${persona}\n\n${modeInstructions[mode]}\n\n${SCENARIO_PLAYBOOK}`;
152
+ }
153
+ // ─── Context Serializer ─────────────────────────────────────
154
+ function serializeContext(context, verbose) {
155
+ const sections = [
156
+ `## Repository State (${context.timestamp})`,
157
+ `Repo: ${context.repo} | Branch: ${context.branch}`,
158
+ `Git: ${context.gitStatus.ahead} ahead, ${context.gitStatus.behind} behind${context.gitStatus.dirty.length > 0 ? `, ${context.gitStatus.dirty.length} dirty files` : ''}`,
159
+ '',
160
+ `## Registry: ${context.registry.active} active, ${context.registry.stale} stale`,
161
+ ];
162
+ if (context.registry.branches.length > 0) {
163
+ for (const b of context.registry.branches.slice(0, 15)) {
164
+ sections.push(` - ${b.name} [${b.status}] agent=${b.agent ?? 'unknown'} files=${b.files.length}`);
165
+ }
166
+ }
167
+ sections.push('', `## Conflicts: ${context.conflicts.tier3} BLOCK, ${context.conflicts.tier2} PAUSE, ${context.conflicts.tier1} WARN`);
168
+ if (context.conflicts.details.length > 0) {
169
+ for (const c of context.conflicts.details) {
170
+ sections.push(` - TIER ${c.tier}: ${c.branch} ↔ ${c.otherBranch} on [${c.files.join(', ')}]`);
171
+ }
172
+ }
173
+ sections.push('', `## Colony: ${context.colony.online ? 'online' : 'offline'}, ${context.colony.activeAgents.length} agents`);
174
+ if (context.colony.inProgress.length > 0) {
175
+ for (const w of context.colony.inProgress) {
176
+ sections.push(` - ${w.agent}: ${w.task} (${w.status})`);
177
+ }
178
+ }
179
+ if (verbose && context.activityLog && context.activityLog.length > 0) {
180
+ sections.push('', '## Recent Activity (last 24h)');
181
+ for (const a of context.activityLog.slice(0, 10)) {
182
+ sections.push(` - [${a.event}] ${a.branch ?? ''} by ${a.agent ?? 'unknown'} at ${a.timestamp}`);
183
+ }
184
+ }
185
+ return sections.join('\n');
186
+ }
187
+ // ─── Main Reasoning Function ────────────────────────────────
188
+ export async function reason(request, llmOverrides) {
189
+ const systemPrompt = buildSystemPrompt(request.mode);
190
+ const contextText = serializeContext(request.context, request.verbose);
191
+ let userMessage;
192
+ switch (request.mode) {
193
+ case 'ask':
194
+ userMessage = `${contextText}\n\n## Question\n${request.question}`;
195
+ break;
196
+ case 'advise':
197
+ userMessage = `${contextText}\n\nWhat should I do next? Prioritize by urgency.`;
198
+ break;
199
+ case 'explain':
200
+ userMessage = `${contextText}\n\nExplain branch: ${request.branch}`;
201
+ break;
202
+ }
203
+ const messages = [
204
+ { role: 'system', content: systemPrompt },
205
+ { role: 'user', content: userMessage },
206
+ ];
207
+ const response = await chat(messages, llmOverrides);
208
+ // Extract commands from response (lines starting with spidersan/git/envoak/toak)
209
+ const commandPattern = /^\s*(?:spidersan|git|envoak|toak|npx spidersan)\s+.+/gm;
210
+ const commands = (response.content.match(commandPattern) ?? []).map(c => c.trim());
211
+ // Validate spidersan sub-commands against the known registry
212
+ const spidersanSubcommandPattern = /^(?:spidersan|npx spidersan)\s+(\S+)/;
213
+ const unknownCommands = commands.filter(cmd => {
214
+ const m = cmd.match(spidersanSubcommandPattern);
215
+ return m !== null && !m[1].startsWith('-') && !KNOWN_SPIDERSAN_COMMANDS.has(m[1]);
216
+ });
217
+ // Estimate confidence from context quality
218
+ const hasConflicts = request.context.conflicts.tier3 > 0 || request.context.conflicts.tier2 > 0;
219
+ const hasRegistry = request.context.registry.active > 0;
220
+ const confidence = hasRegistry
221
+ ? (hasConflicts ? 'high' : 'medium')
222
+ : 'low';
223
+ // Numeric confidence score: try to extract from model response (P2A-1 structured output),
224
+ // fall back to heuristic mapping.
225
+ const CONFIDENCE_SCORES = { high: 0.9, medium: 0.7, low: 0.3 };
226
+ let confidenceScore = CONFIDENCE_SCORES[confidence];
227
+ // Try to parse numeric confidence from structured JSON response (P2A-1 model format)
228
+ const jsonMatch = response.content.match(/^\s*\{[\s\S]*?"confidence"\s*:\s*(0(?:\.\d+)?|1(?:\.0+)?)/m);
229
+ if (jsonMatch?.[1]) {
230
+ const parsed = parseFloat(jsonMatch[1]);
231
+ if (!isNaN(parsed) && parsed >= 0 && parsed <= 1) {
232
+ confidenceScore = parsed;
233
+ }
234
+ }
235
+ return {
236
+ advice: response.content,
237
+ commands,
238
+ unknownCommands,
239
+ confidence: confidence,
240
+ confidenceScore,
241
+ mode: request.mode,
242
+ provider: response.provider,
243
+ tokensUsed: response.promptTokens + response.completionTokens,
244
+ };
245
+ }
246
+ //# sourceMappingURL=reasoner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reasoner.js","sourceRoot":"","sources":["../../../src/lib/ai/reasoner.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Spidersan AI Setup — tier config, server probe, BYOM URL validation
3
+ *
4
+ * Manages ~/.spidersan/config.json (AI-specific config, separate from .spidersanrc)
5
+ * Per spec: SPEC_spidersan_byom_telemetry.md §1.2, §1.3, §1.5, §2.1
6
+ */
7
+ export type AiTier = 'free' | 'contributor' | 'byom' | 'local';
8
+ export interface AiSetupConfig {
9
+ /** How spidersan ask routes LLM requests */
10
+ tier: AiTier;
11
+ /** Saved BYOM URL (only when tier=byom or explicit BYOM preference alongside hosted) */
12
+ byomUrl?: string;
13
+ /** User explicitly confirmed the BYOM URL is external — skip future prompts */
14
+ byomConfirmed?: boolean;
15
+ /** API key for hosted tier (free or contributor) */
16
+ apiKey?: string;
17
+ /** Whether the user has seen and responded to the consent wizard */
18
+ consentGiven?: boolean;
19
+ /** ISO timestamp of first setup */
20
+ setupAt?: string;
21
+ /** ISO timestamp of last update */
22
+ updatedAt?: string;
23
+ }
24
+ export declare function loadAiConfig(): Promise<AiSetupConfig | null>;
25
+ export declare function saveAiConfig(config: AiSetupConfig): Promise<void>;
26
+ export declare function appendByomAudit(source: 'local_probe' | 'env_var' | 'hosted_api' | 'byom_config', url: string, confirmed: 'implicit' | 'explicit'): Promise<void>;
27
+ export interface ProbeResult {
28
+ url: string;
29
+ port: number;
30
+ label: string;
31
+ responding: boolean;
32
+ models?: string[];
33
+ }
34
+ export declare function probeLocalServers(timeoutMs?: number): Promise<ProbeResult[]>;
35
+ /** Probe the hosted API tier. Returns a ProbeResult. */
36
+ export declare function probeHostedTier(apiKey: string, timeoutMs?: number): Promise<ProbeResult>;
37
+ export type UrlClass = 'local' | 'lan' | 'external';
38
+ export declare function classifyUrl(rawUrl: string): UrlClass;
39
+ export declare function isLocalOrLan(url: string): boolean;
40
+ //# sourceMappingURL=setup.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.d.ts","sourceRoot":"","sources":["../../../src/lib/ai/setup.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAUH,MAAM,MAAM,MAAM,GAAG,MAAM,GAAG,aAAa,GAAG,MAAM,GAAG,OAAO,CAAC;AAE/D,MAAM,WAAW,aAAa;IAC5B,4CAA4C;IAC5C,IAAI,EAAE,MAAM,CAAC;IACb,wFAAwF;IACxF,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,+EAA+E;IAC/E,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,oDAAoD;IACpD,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,oEAAoE;IACpE,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,mCAAmC;IACnC,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,mCAAmC;IACnC,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAUD,wBAAsB,YAAY,IAAI,OAAO,CAAC,aAAa,GAAG,IAAI,CAAC,CAQlE;AAED,wBAAsB,YAAY,CAAC,MAAM,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAQvE;AAID,wBAAsB,eAAe,CACnC,MAAM,EAAE,aAAa,GAAG,SAAS,GAAG,YAAY,GAAG,aAAa,EAChE,GAAG,EAAE,MAAM,EACX,SAAS,EAAE,UAAU,GAAG,UAAU,GACjC,OAAO,CAAC,IAAI,CAAC,CAkBf;AAID,MAAM,WAAW,WAAW;IAC1B,GAAG,EAAE,MAAM,CAAC;IACZ,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,OAAO,CAAC;IACpB,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC;CACnB;AASD,wBAAsB,iBAAiB,CAAC,SAAS,SAAM,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC,CA8B/E;AAED,wDAAwD;AACxD,wBAAsB,eAAe,CAAC,MAAM,EAAE,MAAM,EAAE,SAAS,SAAO,GAAG,OAAO,CAAC,WAAW,CAAC,CAiC5F;AAID,MAAM,MAAM,QAAQ,GAAG,OAAO,GAAG,KAAK,GAAG,UAAU,CAAC;AAMpD,wBAAgB,WAAW,CAAC,MAAM,EAAE,MAAM,GAAG,QAAQ,CAWpD;AAED,wBAAgB,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAEjD"}