@lobu/gateway 3.0.8 → 3.0.12

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 (219) hide show
  1. package/dist/api/platform.d.ts.map +1 -1
  2. package/dist/api/platform.js +8 -26
  3. package/dist/api/platform.js.map +1 -1
  4. package/dist/auth/mcp/proxy.d.ts +14 -0
  5. package/dist/auth/mcp/proxy.d.ts.map +1 -1
  6. package/dist/auth/mcp/proxy.js +149 -13
  7. package/dist/auth/mcp/proxy.js.map +1 -1
  8. package/dist/cli/gateway.d.ts.map +1 -1
  9. package/dist/cli/gateway.js +29 -0
  10. package/dist/cli/gateway.js.map +1 -1
  11. package/dist/cli/index.js +2 -2
  12. package/dist/cli/index.js.map +1 -1
  13. package/dist/connections/chat-instance-manager.d.ts.map +1 -1
  14. package/dist/connections/chat-instance-manager.js +2 -1
  15. package/dist/connections/chat-instance-manager.js.map +1 -1
  16. package/dist/connections/interaction-bridge.d.ts +9 -2
  17. package/dist/connections/interaction-bridge.d.ts.map +1 -1
  18. package/dist/connections/interaction-bridge.js +132 -230
  19. package/dist/connections/interaction-bridge.js.map +1 -1
  20. package/dist/connections/message-handler-bridge.d.ts.map +1 -1
  21. package/dist/connections/message-handler-bridge.js +44 -26
  22. package/dist/connections/message-handler-bridge.js.map +1 -1
  23. package/dist/interactions.d.ts +9 -43
  24. package/dist/interactions.d.ts.map +1 -1
  25. package/dist/interactions.js +10 -52
  26. package/dist/interactions.js.map +1 -1
  27. package/dist/orchestration/base-deployment-manager.js +7 -7
  28. package/dist/orchestration/base-deployment-manager.js.map +1 -1
  29. package/dist/platform/unified-thread-consumer.d.ts.map +1 -1
  30. package/dist/platform/unified-thread-consumer.js +38 -34
  31. package/dist/platform/unified-thread-consumer.js.map +1 -1
  32. package/dist/routes/public/agent.d.ts +4 -0
  33. package/dist/routes/public/agent.d.ts.map +1 -1
  34. package/dist/routes/public/agent.js +21 -0
  35. package/dist/routes/public/agent.js.map +1 -1
  36. package/dist/services/core-services.d.ts.map +1 -1
  37. package/dist/services/core-services.js +4 -0
  38. package/dist/services/core-services.js.map +1 -1
  39. package/package.json +2 -2
  40. package/src/__tests__/agent-config-routes.test.ts +0 -254
  41. package/src/__tests__/agent-history-routes.test.ts +0 -72
  42. package/src/__tests__/agent-routes.test.ts +0 -68
  43. package/src/__tests__/agent-schedules-routes.test.ts +0 -59
  44. package/src/__tests__/agent-settings-store.test.ts +0 -323
  45. package/src/__tests__/bedrock-model-catalog.test.ts +0 -40
  46. package/src/__tests__/bedrock-openai-service.test.ts +0 -157
  47. package/src/__tests__/bedrock-provider-module.test.ts +0 -56
  48. package/src/__tests__/chat-instance-manager-slack.test.ts +0 -204
  49. package/src/__tests__/chat-response-bridge.test.ts +0 -131
  50. package/src/__tests__/config-memory-plugins.test.ts +0 -92
  51. package/src/__tests__/config-request-store.test.ts +0 -127
  52. package/src/__tests__/connection-routes.test.ts +0 -144
  53. package/src/__tests__/core-services-store-selection.test.ts +0 -92
  54. package/src/__tests__/docker-deployment.test.ts +0 -1211
  55. package/src/__tests__/embedded-deployment.test.ts +0 -342
  56. package/src/__tests__/grant-store.test.ts +0 -148
  57. package/src/__tests__/http-proxy.test.ts +0 -281
  58. package/src/__tests__/instruction-service.test.ts +0 -37
  59. package/src/__tests__/link-buttons.test.ts +0 -112
  60. package/src/__tests__/lobu.test.ts +0 -32
  61. package/src/__tests__/mcp-config-service.test.ts +0 -347
  62. package/src/__tests__/mcp-proxy.test.ts +0 -694
  63. package/src/__tests__/message-handler-bridge.test.ts +0 -17
  64. package/src/__tests__/model-selection.test.ts +0 -172
  65. package/src/__tests__/oauth-templates.test.ts +0 -39
  66. package/src/__tests__/platform-adapter-slack-send.test.ts +0 -114
  67. package/src/__tests__/platform-helpers-model-resolution.test.ts +0 -253
  68. package/src/__tests__/provider-inheritance.test.ts +0 -212
  69. package/src/__tests__/routes/cli-auth.test.ts +0 -337
  70. package/src/__tests__/routes/interactions.test.ts +0 -121
  71. package/src/__tests__/secret-proxy.test.ts +0 -85
  72. package/src/__tests__/session-manager.test.ts +0 -572
  73. package/src/__tests__/setup.ts +0 -133
  74. package/src/__tests__/skill-and-mcp-registry.test.ts +0 -203
  75. package/src/__tests__/slack-routes.test.ts +0 -161
  76. package/src/__tests__/system-config-resolver.test.ts +0 -75
  77. package/src/__tests__/system-message-limiter.test.ts +0 -89
  78. package/src/__tests__/system-skills-service.test.ts +0 -362
  79. package/src/__tests__/transcription-service.test.ts +0 -222
  80. package/src/__tests__/utils/rate-limiter.test.ts +0 -102
  81. package/src/__tests__/worker-connection-manager.test.ts +0 -497
  82. package/src/__tests__/worker-job-router.test.ts +0 -722
  83. package/src/api/index.ts +0 -1
  84. package/src/api/platform.ts +0 -292
  85. package/src/api/response-renderer.ts +0 -157
  86. package/src/auth/agent-metadata-store.ts +0 -168
  87. package/src/auth/api-auth-middleware.ts +0 -69
  88. package/src/auth/api-key-provider-module.ts +0 -213
  89. package/src/auth/base-provider-module.ts +0 -201
  90. package/src/auth/bedrock/provider-module.ts +0 -110
  91. package/src/auth/chatgpt/chatgpt-oauth-module.ts +0 -185
  92. package/src/auth/chatgpt/device-code-client.ts +0 -218
  93. package/src/auth/chatgpt/index.ts +0 -1
  94. package/src/auth/claude/oauth-module.ts +0 -280
  95. package/src/auth/cli/token-service.ts +0 -249
  96. package/src/auth/external/client.ts +0 -560
  97. package/src/auth/external/device-code-client.ts +0 -235
  98. package/src/auth/mcp/config-service.ts +0 -420
  99. package/src/auth/mcp/proxy.ts +0 -1086
  100. package/src/auth/mcp/string-substitution.ts +0 -17
  101. package/src/auth/mcp/tool-cache.ts +0 -90
  102. package/src/auth/oauth/base-client.ts +0 -267
  103. package/src/auth/oauth/client.ts +0 -153
  104. package/src/auth/oauth/credentials.ts +0 -7
  105. package/src/auth/oauth/providers.ts +0 -69
  106. package/src/auth/oauth/state-store.ts +0 -150
  107. package/src/auth/oauth-templates.ts +0 -179
  108. package/src/auth/provider-catalog.ts +0 -220
  109. package/src/auth/provider-model-options.ts +0 -41
  110. package/src/auth/settings/agent-settings-store.ts +0 -565
  111. package/src/auth/settings/auth-profiles-manager.ts +0 -216
  112. package/src/auth/settings/index.ts +0 -12
  113. package/src/auth/settings/model-preference-store.ts +0 -52
  114. package/src/auth/settings/model-selection.ts +0 -135
  115. package/src/auth/settings/resolved-settings-view.ts +0 -298
  116. package/src/auth/settings/template-utils.ts +0 -44
  117. package/src/auth/settings/token-service.ts +0 -88
  118. package/src/auth/system-env-store.ts +0 -98
  119. package/src/auth/user-agents-store.ts +0 -68
  120. package/src/channels/binding-service.ts +0 -214
  121. package/src/channels/index.ts +0 -4
  122. package/src/cli/gateway.ts +0 -1312
  123. package/src/cli/index.ts +0 -74
  124. package/src/commands/built-in-commands.ts +0 -80
  125. package/src/commands/command-dispatcher.ts +0 -94
  126. package/src/commands/command-reply-adapters.ts +0 -27
  127. package/src/config/file-loader.ts +0 -618
  128. package/src/config/index.ts +0 -588
  129. package/src/config/network-allowlist.ts +0 -71
  130. package/src/connections/chat-instance-manager.ts +0 -1284
  131. package/src/connections/chat-response-bridge.ts +0 -618
  132. package/src/connections/index.ts +0 -7
  133. package/src/connections/interaction-bridge.ts +0 -831
  134. package/src/connections/message-handler-bridge.ts +0 -415
  135. package/src/connections/platform-auth-methods.ts +0 -15
  136. package/src/connections/types.ts +0 -84
  137. package/src/gateway/connection-manager.ts +0 -291
  138. package/src/gateway/index.ts +0 -698
  139. package/src/gateway/job-router.ts +0 -201
  140. package/src/gateway-main.ts +0 -200
  141. package/src/index.ts +0 -41
  142. package/src/infrastructure/queue/index.ts +0 -12
  143. package/src/infrastructure/queue/queue-producer.ts +0 -148
  144. package/src/infrastructure/queue/redis-queue.ts +0 -361
  145. package/src/infrastructure/queue/types.ts +0 -133
  146. package/src/infrastructure/redis/system-message-limiter.ts +0 -94
  147. package/src/interactions/config-request-store.ts +0 -198
  148. package/src/interactions.ts +0 -363
  149. package/src/lobu.ts +0 -311
  150. package/src/metrics/prometheus.ts +0 -159
  151. package/src/modules/module-system.ts +0 -179
  152. package/src/orchestration/base-deployment-manager.ts +0 -900
  153. package/src/orchestration/deployment-utils.ts +0 -98
  154. package/src/orchestration/impl/docker-deployment.ts +0 -620
  155. package/src/orchestration/impl/embedded-deployment.ts +0 -268
  156. package/src/orchestration/impl/index.ts +0 -8
  157. package/src/orchestration/impl/k8s/deployment.ts +0 -1061
  158. package/src/orchestration/impl/k8s/helpers.ts +0 -610
  159. package/src/orchestration/impl/k8s/index.ts +0 -1
  160. package/src/orchestration/index.ts +0 -333
  161. package/src/orchestration/message-consumer.ts +0 -584
  162. package/src/orchestration/scheduled-wakeup.ts +0 -704
  163. package/src/permissions/approval-policy.ts +0 -36
  164. package/src/permissions/grant-store.ts +0 -219
  165. package/src/platform/file-handler.ts +0 -66
  166. package/src/platform/link-buttons.ts +0 -57
  167. package/src/platform/renderer-utils.ts +0 -44
  168. package/src/platform/response-renderer.ts +0 -84
  169. package/src/platform/unified-thread-consumer.ts +0 -187
  170. package/src/platform.ts +0 -318
  171. package/src/proxy/http-proxy.ts +0 -752
  172. package/src/proxy/proxy-manager.ts +0 -81
  173. package/src/proxy/secret-proxy.ts +0 -402
  174. package/src/proxy/token-refresh-job.ts +0 -143
  175. package/src/routes/internal/audio.ts +0 -141
  176. package/src/routes/internal/device-auth.ts +0 -652
  177. package/src/routes/internal/files.ts +0 -226
  178. package/src/routes/internal/history.ts +0 -69
  179. package/src/routes/internal/images.ts +0 -127
  180. package/src/routes/internal/interactions.ts +0 -84
  181. package/src/routes/internal/middleware.ts +0 -23
  182. package/src/routes/internal/schedule.ts +0 -226
  183. package/src/routes/internal/types.ts +0 -22
  184. package/src/routes/openapi-auto.ts +0 -239
  185. package/src/routes/public/agent-access.ts +0 -23
  186. package/src/routes/public/agent-config.ts +0 -675
  187. package/src/routes/public/agent-history.ts +0 -422
  188. package/src/routes/public/agent-schedules.ts +0 -296
  189. package/src/routes/public/agent.ts +0 -1086
  190. package/src/routes/public/agents.ts +0 -373
  191. package/src/routes/public/channels.ts +0 -191
  192. package/src/routes/public/cli-auth.ts +0 -896
  193. package/src/routes/public/connections.ts +0 -574
  194. package/src/routes/public/landing.ts +0 -16
  195. package/src/routes/public/oauth.ts +0 -147
  196. package/src/routes/public/settings-auth.ts +0 -104
  197. package/src/routes/public/slack.ts +0 -173
  198. package/src/routes/shared/agent-ownership.ts +0 -101
  199. package/src/routes/shared/token-verifier.ts +0 -34
  200. package/src/services/bedrock-model-catalog.ts +0 -217
  201. package/src/services/bedrock-openai-service.ts +0 -658
  202. package/src/services/core-services.ts +0 -1072
  203. package/src/services/image-generation-service.ts +0 -257
  204. package/src/services/instruction-service.ts +0 -318
  205. package/src/services/mcp-registry.ts +0 -94
  206. package/src/services/platform-helpers.ts +0 -287
  207. package/src/services/session-manager.ts +0 -262
  208. package/src/services/settings-resolver.ts +0 -74
  209. package/src/services/system-config-resolver.ts +0 -89
  210. package/src/services/system-skills-service.ts +0 -229
  211. package/src/services/transcription-service.ts +0 -684
  212. package/src/session.ts +0 -110
  213. package/src/spaces/index.ts +0 -1
  214. package/src/spaces/space-resolver.ts +0 -17
  215. package/src/stores/in-memory-agent-store.ts +0 -403
  216. package/src/stores/redis-agent-store.ts +0 -279
  217. package/src/utils/public-url.ts +0 -44
  218. package/src/utils/rate-limiter.ts +0 -94
  219. package/tsconfig.json +0 -33
@@ -1,216 +0,0 @@
1
- import { type AuthProfile, createLogger } from "@lobu/core";
2
- import type { AgentSettingsStore } from "./agent-settings-store";
3
-
4
- const logger = createLogger("auth-profiles-manager");
5
-
6
- const ANY_MODEL_SCOPE = "*";
7
-
8
- export interface UpsertAuthProfileInput {
9
- agentId: string;
10
- provider: string;
11
- credential: string;
12
- authType: AuthProfile["authType"];
13
- label: string;
14
- model?: string;
15
- metadata?: AuthProfile["metadata"];
16
- makePrimary?: boolean;
17
- id?: string;
18
- }
19
-
20
- export class AuthProfilesManager {
21
- constructor(private readonly agentSettingsStore: AgentSettingsStore) {}
22
-
23
- async listProfiles(agentId: string): Promise<AuthProfile[]> {
24
- const settings =
25
- await this.agentSettingsStore.getEffectiveSettings(agentId);
26
- return this.normalizeProfiles(settings?.authProfiles);
27
- }
28
-
29
- async hasProviderProfiles(
30
- agentId: string,
31
- provider: string
32
- ): Promise<boolean> {
33
- const profiles = await this.listProfiles(agentId);
34
- return profiles.some((profile) => profile.provider === provider);
35
- }
36
-
37
- async getProviderProfiles(
38
- agentId: string,
39
- provider: string
40
- ): Promise<AuthProfile[]> {
41
- const profiles = await this.listProfiles(agentId);
42
- return profiles.filter((profile) => profile.provider === provider);
43
- }
44
-
45
- async getBestProfile(
46
- agentId: string,
47
- provider: string,
48
- model?: string
49
- ): Promise<AuthProfile | null> {
50
- const providerProfiles = await this.getProviderProfiles(agentId, provider);
51
- if (providerProfiles.length === 0) {
52
- return null;
53
- }
54
-
55
- const now = Date.now();
56
- const validProfiles = providerProfiles.filter((profile) => {
57
- const expiresAt = profile.metadata?.expiresAt;
58
- return !expiresAt || expiresAt > now;
59
- });
60
-
61
- if (validProfiles.length === 0) {
62
- logger.warn(
63
- { agentId, provider, profileCount: providerProfiles.length },
64
- "All auth profiles for provider are expired"
65
- );
66
- return null;
67
- }
68
-
69
- const candidates = validProfiles;
70
- if (!model) {
71
- return candidates[0] || null;
72
- }
73
-
74
- const exact = candidates.find((profile) => profile.model === model);
75
- if (exact) return exact;
76
-
77
- const wildcard = candidates.find(
78
- (profile) => profile.model === ANY_MODEL_SCOPE
79
- );
80
- return wildcard || candidates[0] || null;
81
- }
82
-
83
- async upsertProfile(input: UpsertAuthProfileInput): Promise<AuthProfile> {
84
- const settings = await this.agentSettingsStore.getSettings(input.agentId);
85
- const current = this.normalizeProfiles(settings?.authProfiles);
86
- const modelScope = input.model?.trim() || ANY_MODEL_SCOPE;
87
-
88
- const nextProfile: AuthProfile = {
89
- id: input.id || crypto.randomUUID(),
90
- provider: input.provider,
91
- credential: input.credential,
92
- authType: input.authType,
93
- label: input.label,
94
- model: modelScope,
95
- metadata: input.metadata,
96
- createdAt: Date.now(),
97
- };
98
-
99
- let replaced = false;
100
- const withoutExisting = current.filter((profile) => {
101
- if (input.id && profile.id === input.id) {
102
- replaced = true;
103
- nextProfile.createdAt = profile.createdAt;
104
- return false;
105
- }
106
- return true;
107
- });
108
-
109
- if (!input.id) {
110
- const existingPrimary = withoutExisting.find(
111
- (profile) =>
112
- profile.provider === input.provider && profile.model === modelScope
113
- );
114
- if (existingPrimary) {
115
- replaced = true;
116
- nextProfile.createdAt = existingPrimary.createdAt;
117
- }
118
- }
119
-
120
- const withoutSameScope = withoutExisting.filter(
121
- (profile) =>
122
- !(
123
- profile.provider === input.provider &&
124
- profile.model === modelScope &&
125
- (!input.id || profile.id !== input.id)
126
- )
127
- );
128
-
129
- const nextProfiles: AuthProfile[] = [];
130
- const providerProfiles: AuthProfile[] = [];
131
- const otherProfiles: AuthProfile[] = [];
132
-
133
- for (const profile of withoutSameScope) {
134
- if (profile.provider === input.provider) {
135
- providerProfiles.push(profile);
136
- } else {
137
- otherProfiles.push(profile);
138
- }
139
- }
140
-
141
- if (input.makePrimary !== false) {
142
- nextProfiles.push(nextProfile, ...providerProfiles, ...otherProfiles);
143
- } else {
144
- nextProfiles.push(...providerProfiles, nextProfile, ...otherProfiles);
145
- }
146
-
147
- await this.agentSettingsStore.updateSettings(input.agentId, {
148
- authProfiles: nextProfiles,
149
- });
150
-
151
- logger.info(
152
- {
153
- agentId: input.agentId,
154
- provider: input.provider,
155
- profileId: nextProfile.id,
156
- replaced,
157
- },
158
- "Saved auth profile"
159
- );
160
-
161
- return nextProfile;
162
- }
163
-
164
- async deleteProviderProfiles(
165
- agentId: string,
166
- provider: string,
167
- profileId?: string
168
- ): Promise<void> {
169
- const settings = await this.agentSettingsStore.getSettings(agentId);
170
- const current = this.normalizeProfiles(settings?.authProfiles);
171
- const filtered = current.filter((profile) => {
172
- if (profile.provider !== provider) return true;
173
- if (!profileId) return false;
174
- return profile.id !== profileId;
175
- });
176
-
177
- await this.agentSettingsStore.updateSettings(agentId, {
178
- authProfiles: filtered,
179
- });
180
-
181
- logger.info(
182
- { agentId, provider, profileId: profileId || "all" },
183
- "Deleted auth profiles"
184
- );
185
- }
186
-
187
- private normalizeProfiles(
188
- profiles: AuthProfile[] | undefined
189
- ): AuthProfile[] {
190
- if (!Array.isArray(profiles)) return [];
191
- return profiles.filter(
192
- (profile) =>
193
- typeof profile?.id === "string" &&
194
- typeof profile?.provider === "string" &&
195
- typeof profile?.credential === "string" &&
196
- typeof profile?.authType === "string"
197
- );
198
- }
199
- }
200
-
201
- export function createAuthProfileLabel(
202
- providerDisplayName: string,
203
- credential: string,
204
- accountHint?: string
205
- ): string {
206
- if (accountHint?.trim()) {
207
- return accountHint.trim();
208
- }
209
-
210
- const trimmed = credential.trim();
211
- if (trimmed.length <= 8) {
212
- return `${providerDisplayName} key`;
213
- }
214
-
215
- return `${providerDisplayName} ${trimmed.slice(0, 4)}...${trimmed.slice(-4)}`;
216
- }
@@ -1,12 +0,0 @@
1
- export { type AgentSettings, AgentSettingsStore } from "./agent-settings-store";
2
- export {
3
- AuthProfilesManager,
4
- createAuthProfileLabel,
5
- type UpsertAuthProfileInput,
6
- } from "./auth-profiles-manager";
7
- export type {
8
- PrefillMcpServer,
9
- PrefillSkill,
10
- SettingsSourceContext,
11
- SettingsTokenPayload,
12
- } from "./token-service";
@@ -1,52 +0,0 @@
1
- import { BaseRedisStore } from "@lobu/core";
2
- import type Redis from "ioredis";
3
-
4
- /**
5
- * Store and retrieve user's model preference from Redis
6
- * Pattern: {providerId}:model_preference:{userId}
7
- */
8
- export class ModelPreferenceStore extends BaseRedisStore<string> {
9
- constructor(redis: Redis, providerId: string) {
10
- super({
11
- redis,
12
- keyPrefix: `${providerId}:model_preference`,
13
- loggerName: `${providerId}-model-preference`,
14
- });
15
- }
16
-
17
- /**
18
- * Set user's model preference
19
- */
20
- async setModelPreference(userId: string, model: string): Promise<void> {
21
- const key = this.buildKey(userId);
22
- await this.set(key, model);
23
- this.logger.info(`Set model preference for user ${userId}: ${model}`);
24
- }
25
-
26
- /**
27
- * Get user's model preference
28
- * Returns null if no preference is set
29
- */
30
- async getModelPreference(userId: string): Promise<string | null> {
31
- const key = this.buildKey(userId);
32
- return this.get(key);
33
- }
34
-
35
- /**
36
- * Delete user's model preference
37
- */
38
- async deleteModelPreference(userId: string): Promise<void> {
39
- const key = this.buildKey(userId);
40
- await this.delete(key);
41
- this.logger.info(`Deleted model preference for user ${userId}`);
42
- }
43
-
44
- // Override serialize/deserialize for simple string values
45
- protected override serialize(value: string): string {
46
- return value; // Store as plain string, not JSON
47
- }
48
-
49
- protected override deserialize(data: string): string {
50
- return data; // Return as plain string
51
- }
52
- }
@@ -1,135 +0,0 @@
1
- import type { AgentSettings } from "./agent-settings-store";
2
-
3
- export type ModelSelectionMode = "auto" | "pinned";
4
-
5
- export interface ModelSelectionState {
6
- mode: ModelSelectionMode;
7
- pinnedModel?: string;
8
- }
9
-
10
- export type ProviderModelPreferences = Record<string, string>;
11
-
12
- function normalizePreferenceMap(
13
- map: ProviderModelPreferences | undefined
14
- ): ProviderModelPreferences {
15
- const normalized: ProviderModelPreferences = {};
16
- for (const [providerId, modelRef] of Object.entries(map || {})) {
17
- const cleanProviderId = providerId.trim();
18
- const cleanModelRef = modelRef.trim();
19
- if (!cleanProviderId || !cleanModelRef) continue;
20
- normalized[cleanProviderId] = cleanModelRef;
21
- }
22
- return normalized;
23
- }
24
-
25
- function extractProviderIdFromModelRef(
26
- modelRef: string | undefined
27
- ): string | undefined {
28
- const clean = modelRef?.trim();
29
- if (!clean) return undefined;
30
- const slashIndex = clean.indexOf("/");
31
- if (slashIndex <= 0) return undefined;
32
- return clean.slice(0, slashIndex);
33
- }
34
-
35
- export function getModelSelectionState(
36
- settings: Pick<AgentSettings, "model" | "modelSelection"> | null | undefined
37
- ): ModelSelectionState {
38
- const mode = settings?.modelSelection?.mode;
39
- const explicitPinnedModel = settings?.modelSelection?.pinnedModel?.trim();
40
- const legacyModel = settings?.model?.trim();
41
-
42
- if (mode === "auto") {
43
- return { mode: "auto" };
44
- }
45
-
46
- if (mode === "pinned" && explicitPinnedModel) {
47
- return { mode: "pinned", pinnedModel: explicitPinnedModel };
48
- }
49
-
50
- if (legacyModel) {
51
- return { mode: "pinned", pinnedModel: legacyModel };
52
- }
53
-
54
- return { mode: "auto" };
55
- }
56
-
57
- export function resolveEffectiveModelRef(
58
- settings:
59
- | Pick<
60
- AgentSettings,
61
- | "model"
62
- | "modelSelection"
63
- | "installedProviders"
64
- | "providerModelPreferences"
65
- >
66
- | null
67
- | undefined
68
- ): string | undefined {
69
- if (!settings) return undefined;
70
-
71
- const state = getModelSelectionState(settings);
72
- const installedProviderIds = new Set(
73
- (settings.installedProviders || []).map((p) => p.providerId)
74
- );
75
-
76
- if (state.mode === "pinned" && state.pinnedModel) {
77
- const pinnedProviderId = extractProviderIdFromModelRef(state.pinnedModel);
78
- if (pinnedProviderId && installedProviderIds.has(pinnedProviderId)) {
79
- return state.pinnedModel;
80
- }
81
- }
82
-
83
- const primaryProviderId = settings.installedProviders?.[0]?.providerId;
84
- if (!primaryProviderId) return undefined;
85
-
86
- const preferences = normalizePreferenceMap(settings.providerModelPreferences);
87
- return preferences[primaryProviderId];
88
- }
89
-
90
- export function reconcileModelSelectionForInstalledProviders(
91
- settings: Pick<
92
- AgentSettings,
93
- | "model"
94
- | "modelSelection"
95
- | "installedProviders"
96
- | "providerModelPreferences"
97
- >
98
- ): Pick<
99
- AgentSettings,
100
- "model" | "modelSelection" | "providerModelPreferences"
101
- > {
102
- const installedProviderIds = new Set(
103
- (settings.installedProviders || []).map((p) => p.providerId)
104
- );
105
- const currentState = getModelSelectionState(settings);
106
- const normalizedPrefs = normalizePreferenceMap(
107
- settings.providerModelPreferences
108
- );
109
- const filteredPrefs: ProviderModelPreferences = {};
110
-
111
- for (const [providerId, modelRef] of Object.entries(normalizedPrefs)) {
112
- if (installedProviderIds.has(providerId)) {
113
- filteredPrefs[providerId] = modelRef;
114
- }
115
- }
116
-
117
- const pinnedProviderId = extractProviderIdFromModelRef(
118
- currentState.pinnedModel
119
- );
120
- const hasPinnedProvider =
121
- currentState.mode === "pinned" &&
122
- !!pinnedProviderId &&
123
- installedProviderIds.has(pinnedProviderId);
124
-
125
- const nextState: ModelSelectionState = hasPinnedProvider
126
- ? { mode: "pinned", pinnedModel: currentState.pinnedModel }
127
- : { mode: "auto" };
128
-
129
- return {
130
- modelSelection: nextState,
131
- model: nextState.mode === "pinned" ? nextState.pinnedModel : undefined,
132
- providerModelPreferences:
133
- Object.keys(filteredPrefs).length > 0 ? filteredPrefs : undefined,
134
- };
135
- }
@@ -1,298 +0,0 @@
1
- import type { AgentMetadataStore } from "../agent-metadata-store";
2
- import type {
3
- AgentSettings,
4
- AgentSettingsContext,
5
- AgentSettingsStore,
6
- } from "./agent-settings-store";
7
-
8
- export const SETTINGS_SECTION_KEYS = [
9
- "model",
10
- "system-prompt",
11
- "skills",
12
- "packages",
13
- "permissions",
14
- "schedules",
15
- "logging",
16
- ] as const;
17
-
18
- export type SettingsSectionKey = (typeof SETTINGS_SECTION_KEYS)[number];
19
- export type SettingsScope = "agent" | "sandbox";
20
- export type SettingsSource = "local" | "inherited" | "mixed";
21
-
22
- export interface ResolvedSectionView {
23
- source: SettingsSource;
24
- editable: boolean;
25
- canReset: boolean;
26
- hasLocalOverride: boolean;
27
- }
28
-
29
- export interface ResolvedProviderView {
30
- id: string;
31
- source: SettingsSource;
32
- canEdit: boolean;
33
- canReset: boolean;
34
- hasLocalOverride: boolean;
35
- }
36
-
37
- export interface ResolvedSettingsView {
38
- agentId: string;
39
- scope: SettingsScope;
40
- isSandbox: boolean;
41
- templateAgentId?: string;
42
- templateAgentName?: string;
43
- localSettings: AgentSettings | null;
44
- effectiveSettings: AgentSettings | null;
45
- sections: Record<SettingsSectionKey, ResolvedSectionView>;
46
- providerSources: Record<string, ResolvedProviderView>;
47
- }
48
-
49
- export interface ResolvedSettingsViewer {
50
- settingsMode?: "admin" | "user";
51
- allowedScopes?: string[];
52
- isAdmin?: boolean;
53
- }
54
-
55
- export interface ResolvedSettingsViewInput {
56
- agentId: string;
57
- agentSettingsStore: AgentSettingsStore;
58
- agentMetadataStore?: AgentMetadataStore;
59
- viewer?: ResolvedSettingsViewer;
60
- }
61
-
62
- const SECTION_SETTING_KEYS: Record<
63
- Exclude<SettingsSectionKey, "permissions" | "schedules">,
64
- Array<keyof AgentSettings>
65
- > = {
66
- model: [
67
- "installedProviders",
68
- "authProfiles",
69
- "model",
70
- "modelSelection",
71
- "providerModelPreferences",
72
- ],
73
- "system-prompt": ["identityMd", "soulMd", "userMd"],
74
- skills: ["skillsConfig", "mcpServers", "pluginsConfig"],
75
- packages: ["nixConfig"],
76
- logging: ["verboseLogging"],
77
- };
78
-
79
- function hasOwnSetting(
80
- settings: AgentSettings | null | undefined,
81
- key: keyof AgentSettings
82
- ): boolean {
83
- return !!settings && Object.hasOwn(settings, key);
84
- }
85
-
86
- function sectionHasLocalOverride(
87
- section: SettingsSectionKey,
88
- localSettings: AgentSettings | null | undefined
89
- ): boolean {
90
- if (section === "permissions" || section === "schedules") {
91
- return false;
92
- }
93
- return SECTION_SETTING_KEYS[section].some((key) =>
94
- hasOwnSetting(localSettings, key)
95
- );
96
- }
97
-
98
- function sectionHasTemplateValue(
99
- section: SettingsSectionKey,
100
- templateSettings: AgentSettings | null | undefined
101
- ): boolean {
102
- if (section === "permissions" || section === "schedules") {
103
- return false;
104
- }
105
- return SECTION_SETTING_KEYS[section].some((key) =>
106
- hasOwnSetting(templateSettings, key)
107
- );
108
- }
109
-
110
- export function canViewSettingsSection(
111
- section: SettingsSectionKey,
112
- viewer?: ResolvedSettingsViewer
113
- ): boolean {
114
- if (!viewer || viewer.isAdmin || viewer.settingsMode === "admin") {
115
- return true;
116
- }
117
-
118
- const allowedScopes = viewer.allowedScopes || [];
119
- if (section === "model") {
120
- return (
121
- allowedScopes.includes("model") || allowedScopes.includes("view-model")
122
- );
123
- }
124
-
125
- if (allowedScopes.includes(section)) {
126
- return true;
127
- }
128
-
129
- if (section === "skills") {
130
- return (
131
- allowedScopes.includes("tools") || allowedScopes.includes("mcp-servers")
132
- );
133
- }
134
-
135
- if (section === "permissions" || section === "packages") {
136
- return allowedScopes.includes("tools");
137
- }
138
-
139
- return false;
140
- }
141
-
142
- export function canEditSettingsSection(
143
- section: SettingsSectionKey,
144
- viewer?: ResolvedSettingsViewer
145
- ): boolean {
146
- if (!viewer || viewer.isAdmin || viewer.settingsMode === "admin") {
147
- return true;
148
- }
149
-
150
- const allowedScopes = viewer.allowedScopes || [];
151
- if (section === "model") {
152
- return allowedScopes.includes("model");
153
- }
154
-
155
- if (allowedScopes.includes(section)) {
156
- return true;
157
- }
158
-
159
- if (section === "skills") {
160
- return (
161
- allowedScopes.includes("tools") || allowedScopes.includes("mcp-servers")
162
- );
163
- }
164
-
165
- if (section === "permissions" || section === "packages") {
166
- return allowedScopes.includes("tools");
167
- }
168
-
169
- return false;
170
- }
171
-
172
- function resolveSectionSource(
173
- isSandbox: boolean,
174
- hasLocalOverride: boolean,
175
- hasTemplateValue: boolean
176
- ): SettingsSource {
177
- if (!isSandbox) return "local";
178
- if (!hasLocalOverride && hasTemplateValue) return "inherited";
179
- if (hasLocalOverride && hasTemplateValue) return "mixed";
180
- return "local";
181
- }
182
-
183
- function resolveProviderSources(
184
- context: AgentSettingsContext,
185
- templateSettings: AgentSettings | null,
186
- viewer?: ResolvedSettingsViewer
187
- ): Record<string, ResolvedProviderView> {
188
- const effectiveSettings = context.effectiveSettings;
189
- const localSettings = context.localSettings;
190
- const isSandbox = !!context.templateAgentId;
191
-
192
- const effectiveProviderIds = (
193
- effectiveSettings?.installedProviders || []
194
- ).map((provider) => provider.providerId);
195
- const localProviderIds = new Set(
196
- (localSettings?.installedProviders || []).map(
197
- (provider) => provider.providerId
198
- )
199
- );
200
- const localProfileProviders = new Set(
201
- (localSettings?.authProfiles || []).map((profile) => profile.provider)
202
- );
203
- const localPreferenceProviders = new Set(
204
- Object.keys(localSettings?.providerModelPreferences || {})
205
- );
206
- const templateProviderIds = new Set(
207
- (templateSettings?.installedProviders || []).map(
208
- (provider) => provider.providerId
209
- )
210
- );
211
-
212
- return Object.fromEntries(
213
- effectiveProviderIds.map((providerId) => {
214
- const hasLocalOverride =
215
- localProviderIds.has(providerId) ||
216
- localProfileProviders.has(providerId) ||
217
- localPreferenceProviders.has(providerId);
218
-
219
- const source = resolveSectionSource(
220
- isSandbox,
221
- hasLocalOverride,
222
- templateProviderIds.has(providerId)
223
- );
224
-
225
- return [
226
- providerId,
227
- {
228
- id: providerId,
229
- source,
230
- canEdit: canEditSettingsSection("model", viewer),
231
- canReset: isSandbox && hasLocalOverride,
232
- hasLocalOverride,
233
- } satisfies ResolvedProviderView,
234
- ];
235
- })
236
- );
237
- }
238
-
239
- export async function resolveSettingsView(
240
- input: ResolvedSettingsViewInput
241
- ): Promise<ResolvedSettingsView> {
242
- const context = await input.agentSettingsStore.getSettingsContext(
243
- input.agentId
244
- );
245
- const templateAgentName =
246
- context.templateAgentId && input.agentMetadataStore
247
- ? (await input.agentMetadataStore.getMetadata(context.templateAgentId))
248
- ?.name
249
- : undefined;
250
-
251
- const templateSettings = context.templateAgentId
252
- ? await input.agentSettingsStore.getSettings(context.templateAgentId)
253
- : null;
254
- const isSandbox = !!context.templateAgentId;
255
-
256
- const sections = Object.fromEntries(
257
- SETTINGS_SECTION_KEYS.map((section) => {
258
- const hasLocalOverride = sectionHasLocalOverride(
259
- section,
260
- context.localSettings
261
- );
262
- const hasTemplateValue = sectionHasTemplateValue(
263
- section,
264
- templateSettings
265
- );
266
-
267
- return [
268
- section,
269
- {
270
- source: resolveSectionSource(
271
- isSandbox,
272
- hasLocalOverride,
273
- hasTemplateValue
274
- ),
275
- editable: canEditSettingsSection(section, input.viewer),
276
- canReset: isSandbox && hasLocalOverride,
277
- hasLocalOverride,
278
- } satisfies ResolvedSectionView,
279
- ];
280
- })
281
- ) as Record<SettingsSectionKey, ResolvedSectionView>;
282
-
283
- return {
284
- agentId: input.agentId,
285
- scope: isSandbox ? "sandbox" : "agent",
286
- isSandbox,
287
- templateAgentId: context.templateAgentId,
288
- templateAgentName,
289
- localSettings: context.localSettings,
290
- effectiveSettings: context.effectiveSettings,
291
- sections,
292
- providerSources: resolveProviderSources(
293
- context,
294
- templateSettings,
295
- input.viewer
296
- ),
297
- };
298
- }