@lobu/gateway 3.0.5 → 3.0.7

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 (175) hide show
  1. package/package.json +2 -2
  2. package/src/__tests__/agent-config-routes.test.ts +254 -0
  3. package/src/__tests__/agent-history-routes.test.ts +72 -0
  4. package/src/__tests__/agent-routes.test.ts +68 -0
  5. package/src/__tests__/agent-schedules-routes.test.ts +59 -0
  6. package/src/__tests__/agent-settings-store.test.ts +323 -0
  7. package/src/__tests__/chat-instance-manager-slack.test.ts +204 -0
  8. package/src/__tests__/chat-response-bridge.test.ts +131 -0
  9. package/src/__tests__/config-memory-plugins.test.ts +92 -0
  10. package/src/__tests__/config-request-store.test.ts +127 -0
  11. package/src/__tests__/connection-routes.test.ts +144 -0
  12. package/src/__tests__/core-services-store-selection.test.ts +92 -0
  13. package/src/__tests__/docker-deployment.test.ts +1211 -0
  14. package/src/__tests__/embedded-deployment.test.ts +342 -0
  15. package/src/__tests__/grant-store.test.ts +148 -0
  16. package/src/__tests__/http-proxy.test.ts +281 -0
  17. package/src/__tests__/instruction-service.test.ts +37 -0
  18. package/src/__tests__/link-buttons.test.ts +112 -0
  19. package/src/__tests__/lobu.test.ts +32 -0
  20. package/src/__tests__/mcp-config-service.test.ts +347 -0
  21. package/src/__tests__/mcp-proxy.test.ts +696 -0
  22. package/src/__tests__/message-handler-bridge.test.ts +17 -0
  23. package/src/__tests__/model-selection.test.ts +172 -0
  24. package/src/__tests__/oauth-templates.test.ts +39 -0
  25. package/src/__tests__/platform-adapter-slack-send.test.ts +114 -0
  26. package/src/__tests__/platform-helpers-model-resolution.test.ts +253 -0
  27. package/src/__tests__/provider-inheritance.test.ts +212 -0
  28. package/src/__tests__/routes/cli-auth.test.ts +337 -0
  29. package/src/__tests__/routes/interactions.test.ts +121 -0
  30. package/src/__tests__/secret-proxy.test.ts +85 -0
  31. package/src/__tests__/session-manager.test.ts +572 -0
  32. package/src/__tests__/setup.ts +133 -0
  33. package/src/__tests__/skill-and-mcp-registry.test.ts +203 -0
  34. package/src/__tests__/slack-routes.test.ts +161 -0
  35. package/src/__tests__/system-config-resolver.test.ts +75 -0
  36. package/src/__tests__/system-message-limiter.test.ts +89 -0
  37. package/src/__tests__/system-skills-service.test.ts +362 -0
  38. package/src/__tests__/transcription-service.test.ts +222 -0
  39. package/src/__tests__/utils/rate-limiter.test.ts +102 -0
  40. package/src/__tests__/worker-connection-manager.test.ts +497 -0
  41. package/src/__tests__/worker-job-router.test.ts +722 -0
  42. package/src/api/index.ts +1 -0
  43. package/src/api/platform.ts +292 -0
  44. package/src/api/response-renderer.ts +157 -0
  45. package/src/auth/agent-metadata-store.ts +168 -0
  46. package/src/auth/api-auth-middleware.ts +69 -0
  47. package/src/auth/api-key-provider-module.ts +213 -0
  48. package/src/auth/base-provider-module.ts +201 -0
  49. package/src/auth/chatgpt/chatgpt-oauth-module.ts +185 -0
  50. package/src/auth/chatgpt/device-code-client.ts +218 -0
  51. package/src/auth/chatgpt/index.ts +1 -0
  52. package/src/auth/claude/oauth-module.ts +280 -0
  53. package/src/auth/cli/token-service.ts +249 -0
  54. package/src/auth/external/client.ts +560 -0
  55. package/src/auth/external/device-code-client.ts +225 -0
  56. package/src/auth/mcp/config-service.ts +392 -0
  57. package/src/auth/mcp/proxy.ts +1088 -0
  58. package/src/auth/mcp/string-substitution.ts +17 -0
  59. package/src/auth/mcp/tool-cache.ts +90 -0
  60. package/src/auth/oauth/base-client.ts +267 -0
  61. package/src/auth/oauth/client.ts +153 -0
  62. package/src/auth/oauth/credentials.ts +7 -0
  63. package/src/auth/oauth/providers.ts +69 -0
  64. package/src/auth/oauth/state-store.ts +150 -0
  65. package/src/auth/oauth-templates.ts +179 -0
  66. package/src/auth/provider-catalog.ts +220 -0
  67. package/src/auth/provider-model-options.ts +41 -0
  68. package/src/auth/settings/agent-settings-store.ts +565 -0
  69. package/src/auth/settings/auth-profiles-manager.ts +216 -0
  70. package/src/auth/settings/index.ts +12 -0
  71. package/src/auth/settings/model-preference-store.ts +52 -0
  72. package/src/auth/settings/model-selection.ts +135 -0
  73. package/src/auth/settings/resolved-settings-view.ts +298 -0
  74. package/src/auth/settings/template-utils.ts +44 -0
  75. package/src/auth/settings/token-service.ts +88 -0
  76. package/src/auth/system-env-store.ts +98 -0
  77. package/src/auth/user-agents-store.ts +68 -0
  78. package/src/channels/binding-service.ts +214 -0
  79. package/src/channels/index.ts +4 -0
  80. package/src/cli/gateway.ts +1304 -0
  81. package/src/cli/index.ts +74 -0
  82. package/src/commands/built-in-commands.ts +80 -0
  83. package/src/commands/command-dispatcher.ts +94 -0
  84. package/src/commands/command-reply-adapters.ts +27 -0
  85. package/src/config/file-loader.ts +618 -0
  86. package/src/config/index.ts +588 -0
  87. package/src/config/network-allowlist.ts +71 -0
  88. package/src/connections/chat-instance-manager.ts +1284 -0
  89. package/src/connections/chat-response-bridge.ts +618 -0
  90. package/src/connections/index.ts +7 -0
  91. package/src/connections/interaction-bridge.ts +831 -0
  92. package/src/connections/message-handler-bridge.ts +415 -0
  93. package/src/connections/platform-auth-methods.ts +15 -0
  94. package/src/connections/types.ts +84 -0
  95. package/src/gateway/connection-manager.ts +291 -0
  96. package/src/gateway/index.ts +700 -0
  97. package/src/gateway/job-router.ts +201 -0
  98. package/src/gateway-main.ts +200 -0
  99. package/src/index.ts +41 -0
  100. package/src/infrastructure/queue/index.ts +12 -0
  101. package/src/infrastructure/queue/queue-producer.ts +148 -0
  102. package/src/infrastructure/queue/redis-queue.ts +361 -0
  103. package/src/infrastructure/queue/types.ts +133 -0
  104. package/src/infrastructure/redis/system-message-limiter.ts +94 -0
  105. package/src/interactions/config-request-store.ts +198 -0
  106. package/src/interactions.ts +363 -0
  107. package/src/lobu.ts +311 -0
  108. package/src/metrics/prometheus.ts +159 -0
  109. package/src/modules/module-system.ts +179 -0
  110. package/src/orchestration/base-deployment-manager.ts +900 -0
  111. package/src/orchestration/deployment-utils.ts +98 -0
  112. package/src/orchestration/impl/docker-deployment.ts +620 -0
  113. package/src/orchestration/impl/embedded-deployment.ts +268 -0
  114. package/src/orchestration/impl/index.ts +8 -0
  115. package/src/orchestration/impl/k8s/deployment.ts +1061 -0
  116. package/src/orchestration/impl/k8s/helpers.ts +610 -0
  117. package/src/orchestration/impl/k8s/index.ts +1 -0
  118. package/src/orchestration/index.ts +333 -0
  119. package/src/orchestration/message-consumer.ts +584 -0
  120. package/src/orchestration/scheduled-wakeup.ts +704 -0
  121. package/src/permissions/approval-policy.ts +36 -0
  122. package/src/permissions/grant-store.ts +219 -0
  123. package/src/platform/file-handler.ts +66 -0
  124. package/src/platform/link-buttons.ts +57 -0
  125. package/src/platform/renderer-utils.ts +44 -0
  126. package/src/platform/response-renderer.ts +84 -0
  127. package/src/platform/unified-thread-consumer.ts +187 -0
  128. package/src/platform.ts +318 -0
  129. package/src/proxy/http-proxy.ts +752 -0
  130. package/src/proxy/proxy-manager.ts +81 -0
  131. package/src/proxy/secret-proxy.ts +402 -0
  132. package/src/proxy/token-refresh-job.ts +143 -0
  133. package/src/routes/internal/audio.ts +141 -0
  134. package/src/routes/internal/device-auth.ts +566 -0
  135. package/src/routes/internal/files.ts +226 -0
  136. package/src/routes/internal/history.ts +69 -0
  137. package/src/routes/internal/images.ts +127 -0
  138. package/src/routes/internal/interactions.ts +84 -0
  139. package/src/routes/internal/middleware.ts +23 -0
  140. package/src/routes/internal/schedule.ts +226 -0
  141. package/src/routes/internal/types.ts +22 -0
  142. package/src/routes/openapi-auto.ts +239 -0
  143. package/src/routes/public/agent-access.ts +23 -0
  144. package/src/routes/public/agent-config.ts +675 -0
  145. package/src/routes/public/agent-history.ts +422 -0
  146. package/src/routes/public/agent-schedules.ts +296 -0
  147. package/src/routes/public/agent.ts +1086 -0
  148. package/src/routes/public/agents.ts +373 -0
  149. package/src/routes/public/channels.ts +191 -0
  150. package/src/routes/public/cli-auth.ts +883 -0
  151. package/src/routes/public/connections.ts +574 -0
  152. package/src/routes/public/landing.ts +16 -0
  153. package/src/routes/public/oauth.ts +147 -0
  154. package/src/routes/public/settings-auth.ts +104 -0
  155. package/src/routes/public/slack.ts +173 -0
  156. package/src/routes/shared/agent-ownership.ts +101 -0
  157. package/src/routes/shared/token-verifier.ts +34 -0
  158. package/src/services/core-services.ts +1053 -0
  159. package/src/services/image-generation-service.ts +257 -0
  160. package/src/services/instruction-service.ts +318 -0
  161. package/src/services/mcp-registry.ts +94 -0
  162. package/src/services/platform-helpers.ts +287 -0
  163. package/src/services/session-manager.ts +262 -0
  164. package/src/services/settings-resolver.ts +74 -0
  165. package/src/services/system-config-resolver.ts +90 -0
  166. package/src/services/system-skills-service.ts +229 -0
  167. package/src/services/transcription-service.ts +684 -0
  168. package/src/session.ts +110 -0
  169. package/src/spaces/index.ts +1 -0
  170. package/src/spaces/space-resolver.ts +17 -0
  171. package/src/stores/in-memory-agent-store.ts +403 -0
  172. package/src/stores/redis-agent-store.ts +279 -0
  173. package/src/utils/public-url.ts +44 -0
  174. package/src/utils/rate-limiter.ts +94 -0
  175. package/tsconfig.json +33 -0
@@ -0,0 +1,216 @@
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
+ }
@@ -0,0 +1,12 @@
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";
@@ -0,0 +1,52 @@
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
+ }
@@ -0,0 +1,135 @@
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
+ }
@@ -0,0 +1,298 @@
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
+ }