@lobu/gateway 3.0.5 → 3.0.6
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.
- package/package.json +2 -2
- package/src/__tests__/agent-config-routes.test.ts +254 -0
- package/src/__tests__/agent-history-routes.test.ts +72 -0
- package/src/__tests__/agent-routes.test.ts +68 -0
- package/src/__tests__/agent-schedules-routes.test.ts +59 -0
- package/src/__tests__/agent-settings-store.test.ts +323 -0
- package/src/__tests__/chat-instance-manager-slack.test.ts +204 -0
- package/src/__tests__/chat-response-bridge.test.ts +131 -0
- package/src/__tests__/config-memory-plugins.test.ts +92 -0
- package/src/__tests__/config-request-store.test.ts +127 -0
- package/src/__tests__/connection-routes.test.ts +144 -0
- package/src/__tests__/core-services-store-selection.test.ts +92 -0
- package/src/__tests__/docker-deployment.test.ts +1211 -0
- package/src/__tests__/embedded-deployment.test.ts +342 -0
- package/src/__tests__/grant-store.test.ts +148 -0
- package/src/__tests__/http-proxy.test.ts +281 -0
- package/src/__tests__/instruction-service.test.ts +37 -0
- package/src/__tests__/link-buttons.test.ts +112 -0
- package/src/__tests__/lobu.test.ts +32 -0
- package/src/__tests__/mcp-config-service.test.ts +347 -0
- package/src/__tests__/mcp-proxy.test.ts +696 -0
- package/src/__tests__/message-handler-bridge.test.ts +17 -0
- package/src/__tests__/model-selection.test.ts +172 -0
- package/src/__tests__/oauth-templates.test.ts +39 -0
- package/src/__tests__/platform-adapter-slack-send.test.ts +114 -0
- package/src/__tests__/platform-helpers-model-resolution.test.ts +253 -0
- package/src/__tests__/provider-inheritance.test.ts +212 -0
- package/src/__tests__/routes/cli-auth.test.ts +337 -0
- package/src/__tests__/routes/interactions.test.ts +121 -0
- package/src/__tests__/secret-proxy.test.ts +85 -0
- package/src/__tests__/session-manager.test.ts +572 -0
- package/src/__tests__/setup.ts +133 -0
- package/src/__tests__/skill-and-mcp-registry.test.ts +203 -0
- package/src/__tests__/slack-routes.test.ts +161 -0
- package/src/__tests__/system-config-resolver.test.ts +75 -0
- package/src/__tests__/system-message-limiter.test.ts +89 -0
- package/src/__tests__/system-skills-service.test.ts +362 -0
- package/src/__tests__/transcription-service.test.ts +222 -0
- package/src/__tests__/utils/rate-limiter.test.ts +102 -0
- package/src/__tests__/worker-connection-manager.test.ts +497 -0
- package/src/__tests__/worker-job-router.test.ts +722 -0
- package/src/api/index.ts +1 -0
- package/src/api/platform.ts +292 -0
- package/src/api/response-renderer.ts +157 -0
- package/src/auth/agent-metadata-store.ts +168 -0
- package/src/auth/api-auth-middleware.ts +69 -0
- package/src/auth/api-key-provider-module.ts +213 -0
- package/src/auth/base-provider-module.ts +201 -0
- package/src/auth/chatgpt/chatgpt-oauth-module.ts +185 -0
- package/src/auth/chatgpt/device-code-client.ts +218 -0
- package/src/auth/chatgpt/index.ts +1 -0
- package/src/auth/claude/oauth-module.ts +280 -0
- package/src/auth/cli/token-service.ts +249 -0
- package/src/auth/external/client.ts +560 -0
- package/src/auth/external/device-code-client.ts +225 -0
- package/src/auth/mcp/config-service.ts +392 -0
- package/src/auth/mcp/proxy.ts +1088 -0
- package/src/auth/mcp/string-substitution.ts +17 -0
- package/src/auth/mcp/tool-cache.ts +90 -0
- package/src/auth/oauth/base-client.ts +267 -0
- package/src/auth/oauth/client.ts +153 -0
- package/src/auth/oauth/credentials.ts +7 -0
- package/src/auth/oauth/providers.ts +69 -0
- package/src/auth/oauth/state-store.ts +150 -0
- package/src/auth/oauth-templates.ts +179 -0
- package/src/auth/provider-catalog.ts +220 -0
- package/src/auth/provider-model-options.ts +41 -0
- package/src/auth/settings/agent-settings-store.ts +565 -0
- package/src/auth/settings/auth-profiles-manager.ts +216 -0
- package/src/auth/settings/index.ts +12 -0
- package/src/auth/settings/model-preference-store.ts +52 -0
- package/src/auth/settings/model-selection.ts +135 -0
- package/src/auth/settings/resolved-settings-view.ts +298 -0
- package/src/auth/settings/template-utils.ts +44 -0
- package/src/auth/settings/token-service.ts +88 -0
- package/src/auth/system-env-store.ts +98 -0
- package/src/auth/user-agents-store.ts +68 -0
- package/src/channels/binding-service.ts +214 -0
- package/src/channels/index.ts +4 -0
- package/src/cli/gateway.ts +1304 -0
- package/src/cli/index.ts +74 -0
- package/src/commands/built-in-commands.ts +80 -0
- package/src/commands/command-dispatcher.ts +94 -0
- package/src/commands/command-reply-adapters.ts +27 -0
- package/src/config/file-loader.ts +618 -0
- package/src/config/index.ts +588 -0
- package/src/config/network-allowlist.ts +71 -0
- package/src/connections/chat-instance-manager.ts +1284 -0
- package/src/connections/chat-response-bridge.ts +618 -0
- package/src/connections/index.ts +7 -0
- package/src/connections/interaction-bridge.ts +831 -0
- package/src/connections/message-handler-bridge.ts +415 -0
- package/src/connections/platform-auth-methods.ts +15 -0
- package/src/connections/types.ts +84 -0
- package/src/gateway/connection-manager.ts +291 -0
- package/src/gateway/index.ts +700 -0
- package/src/gateway/job-router.ts +201 -0
- package/src/gateway-main.ts +200 -0
- package/src/index.ts +41 -0
- package/src/infrastructure/queue/index.ts +12 -0
- package/src/infrastructure/queue/queue-producer.ts +148 -0
- package/src/infrastructure/queue/redis-queue.ts +361 -0
- package/src/infrastructure/queue/types.ts +133 -0
- package/src/infrastructure/redis/system-message-limiter.ts +94 -0
- package/src/interactions/config-request-store.ts +198 -0
- package/src/interactions.ts +363 -0
- package/src/lobu.ts +311 -0
- package/src/metrics/prometheus.ts +159 -0
- package/src/modules/module-system.ts +179 -0
- package/src/orchestration/base-deployment-manager.ts +900 -0
- package/src/orchestration/deployment-utils.ts +98 -0
- package/src/orchestration/impl/docker-deployment.ts +620 -0
- package/src/orchestration/impl/embedded-deployment.ts +268 -0
- package/src/orchestration/impl/index.ts +8 -0
- package/src/orchestration/impl/k8s/deployment.ts +1061 -0
- package/src/orchestration/impl/k8s/helpers.ts +610 -0
- package/src/orchestration/impl/k8s/index.ts +1 -0
- package/src/orchestration/index.ts +333 -0
- package/src/orchestration/message-consumer.ts +584 -0
- package/src/orchestration/scheduled-wakeup.ts +704 -0
- package/src/permissions/approval-policy.ts +36 -0
- package/src/permissions/grant-store.ts +219 -0
- package/src/platform/file-handler.ts +66 -0
- package/src/platform/link-buttons.ts +57 -0
- package/src/platform/renderer-utils.ts +44 -0
- package/src/platform/response-renderer.ts +84 -0
- package/src/platform/unified-thread-consumer.ts +187 -0
- package/src/platform.ts +318 -0
- package/src/proxy/http-proxy.ts +752 -0
- package/src/proxy/proxy-manager.ts +81 -0
- package/src/proxy/secret-proxy.ts +402 -0
- package/src/proxy/token-refresh-job.ts +143 -0
- package/src/routes/internal/audio.ts +141 -0
- package/src/routes/internal/device-auth.ts +566 -0
- package/src/routes/internal/files.ts +226 -0
- package/src/routes/internal/history.ts +69 -0
- package/src/routes/internal/images.ts +127 -0
- package/src/routes/internal/interactions.ts +84 -0
- package/src/routes/internal/middleware.ts +23 -0
- package/src/routes/internal/schedule.ts +226 -0
- package/src/routes/internal/types.ts +22 -0
- package/src/routes/openapi-auto.ts +239 -0
- package/src/routes/public/agent-access.ts +23 -0
- package/src/routes/public/agent-config.ts +675 -0
- package/src/routes/public/agent-history.ts +422 -0
- package/src/routes/public/agent-schedules.ts +296 -0
- package/src/routes/public/agent.ts +1086 -0
- package/src/routes/public/agents.ts +373 -0
- package/src/routes/public/channels.ts +191 -0
- package/src/routes/public/cli-auth.ts +883 -0
- package/src/routes/public/connections.ts +574 -0
- package/src/routes/public/landing.ts +16 -0
- package/src/routes/public/oauth.ts +147 -0
- package/src/routes/public/settings-auth.ts +104 -0
- package/src/routes/public/slack.ts +173 -0
- package/src/routes/shared/agent-ownership.ts +101 -0
- package/src/routes/shared/token-verifier.ts +34 -0
- package/src/services/core-services.ts +1053 -0
- package/src/services/image-generation-service.ts +257 -0
- package/src/services/instruction-service.ts +318 -0
- package/src/services/mcp-registry.ts +94 -0
- package/src/services/platform-helpers.ts +287 -0
- package/src/services/session-manager.ts +262 -0
- package/src/services/settings-resolver.ts +74 -0
- package/src/services/system-config-resolver.ts +90 -0
- package/src/services/system-skills-service.ts +229 -0
- package/src/services/transcription-service.ts +684 -0
- package/src/session.ts +110 -0
- package/src/spaces/index.ts +1 -0
- package/src/spaces/space-resolver.ts +17 -0
- package/src/stores/in-memory-agent-store.ts +403 -0
- package/src/stores/redis-agent-store.ts +279 -0
- package/src/utils/public-url.ts +44 -0
- package/src/utils/rate-limiter.ts +94 -0
- 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
|
+
}
|