@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.
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,172 @@
1
+ import { describe, expect, test } from "bun:test";
2
+ import {
3
+ getModelSelectionState,
4
+ reconcileModelSelectionForInstalledProviders,
5
+ resolveEffectiveModelRef,
6
+ } from "../auth/settings/model-selection";
7
+
8
+ describe("model-selection", () => {
9
+ test("uses legacy model as pinned fallback", () => {
10
+ expect(getModelSelectionState({ model: "openai/gpt-5" })).toEqual({
11
+ mode: "pinned",
12
+ pinnedModel: "openai/gpt-5",
13
+ });
14
+ });
15
+
16
+ test("resolves pinned model when pinned provider is installed", () => {
17
+ const effective = resolveEffectiveModelRef({
18
+ modelSelection: { mode: "pinned", pinnedModel: "openai/gpt-5" },
19
+ installedProviders: [
20
+ { providerId: "openai", installedAt: 1 },
21
+ { providerId: "anthropic", installedAt: 2 },
22
+ ],
23
+ providerModelPreferences: {
24
+ openai: "openai/gpt-4.1",
25
+ },
26
+ } as any);
27
+
28
+ expect(effective).toBe("openai/gpt-5");
29
+ });
30
+
31
+ test("falls back to primary provider preference when pinned provider is removed", () => {
32
+ const effective = resolveEffectiveModelRef({
33
+ modelSelection: { mode: "pinned", pinnedModel: "anthropic/claude-3.7" },
34
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
35
+ providerModelPreferences: {
36
+ openai: "openai/gpt-5",
37
+ anthropic: "anthropic/claude-3.7",
38
+ },
39
+ } as any);
40
+
41
+ expect(effective).toBe("openai/gpt-5");
42
+ });
43
+
44
+ test("reconcile clears invalid pinned selection and removes uninstalled preferences", () => {
45
+ const reconciled = reconcileModelSelectionForInstalledProviders({
46
+ model: "anthropic/claude-3.7",
47
+ modelSelection: { mode: "pinned", pinnedModel: "anthropic/claude-3.7" },
48
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
49
+ providerModelPreferences: {
50
+ openai: "openai/gpt-5",
51
+ anthropic: "anthropic/claude-3.7",
52
+ },
53
+ } as any);
54
+
55
+ expect(reconciled).toEqual({
56
+ modelSelection: { mode: "auto" },
57
+ model: undefined,
58
+ providerModelPreferences: {
59
+ openai: "openai/gpt-5",
60
+ },
61
+ });
62
+ });
63
+
64
+ test("reconcile keeps valid pinned selection", () => {
65
+ const reconciled = reconcileModelSelectionForInstalledProviders({
66
+ model: "openai/gpt-5",
67
+ modelSelection: { mode: "pinned", pinnedModel: "openai/gpt-5" },
68
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
69
+ providerModelPreferences: {
70
+ openai: "openai/gpt-4.1",
71
+ },
72
+ } as any);
73
+
74
+ expect(reconciled.modelSelection).toEqual({
75
+ mode: "pinned",
76
+ pinnedModel: "openai/gpt-5",
77
+ });
78
+ expect(reconciled.model).toBe("openai/gpt-5");
79
+ });
80
+
81
+ test("auto mode follows primary provider order change", () => {
82
+ const before = resolveEffectiveModelRef({
83
+ modelSelection: { mode: "auto" },
84
+ installedProviders: [
85
+ { providerId: "openai", installedAt: 1 },
86
+ { providerId: "anthropic", installedAt: 2 },
87
+ ],
88
+ providerModelPreferences: {
89
+ openai: "openai/gpt-5",
90
+ anthropic: "anthropic/claude-sonnet-4",
91
+ },
92
+ } as any);
93
+ const after = resolveEffectiveModelRef({
94
+ modelSelection: { mode: "auto" },
95
+ installedProviders: [
96
+ { providerId: "anthropic", installedAt: 2 },
97
+ { providerId: "openai", installedAt: 1 },
98
+ ],
99
+ providerModelPreferences: {
100
+ openai: "openai/gpt-5",
101
+ anthropic: "anthropic/claude-sonnet-4",
102
+ },
103
+ } as any);
104
+
105
+ expect(before).toBe("openai/gpt-5");
106
+ expect(after).toBe("anthropic/claude-sonnet-4");
107
+ });
108
+
109
+ test("auto mode can keep non-primary provider preference without affecting effective model", () => {
110
+ const reconciled = reconcileModelSelectionForInstalledProviders({
111
+ modelSelection: { mode: "auto" },
112
+ installedProviders: [
113
+ { providerId: "openai", installedAt: 1 },
114
+ { providerId: "anthropic", installedAt: 2 },
115
+ ],
116
+ providerModelPreferences: {
117
+ openai: "openai/gpt-5",
118
+ anthropic: "anthropic/claude-opus-4",
119
+ },
120
+ } as any);
121
+
122
+ expect(reconciled.providerModelPreferences).toEqual({
123
+ openai: "openai/gpt-5",
124
+ anthropic: "anthropic/claude-opus-4",
125
+ });
126
+ expect(
127
+ resolveEffectiveModelRef({
128
+ modelSelection: reconciled.modelSelection,
129
+ installedProviders: [
130
+ { providerId: "openai", installedAt: 1 },
131
+ { providerId: "anthropic", installedAt: 2 },
132
+ ],
133
+ providerModelPreferences: reconciled.providerModelPreferences,
134
+ } as any)
135
+ ).toBe("openai/gpt-5");
136
+ });
137
+
138
+ test("reconcile removes stale preferences when provider is deleted", () => {
139
+ const reconciled = reconcileModelSelectionForInstalledProviders({
140
+ modelSelection: { mode: "auto" },
141
+ installedProviders: [{ providerId: "anthropic", installedAt: 2 }],
142
+ providerModelPreferences: {
143
+ openai: "openai/gpt-5",
144
+ anthropic: "anthropic/claude-sonnet-4",
145
+ },
146
+ } as any);
147
+
148
+ expect(reconciled.providerModelPreferences).toEqual({
149
+ anthropic: "anthropic/claude-sonnet-4",
150
+ });
151
+ });
152
+
153
+ test("falls back to auto when pinned model has no resolvable provider", () => {
154
+ const reconciled = reconcileModelSelectionForInstalledProviders({
155
+ modelSelection: { mode: "pinned", pinnedModel: "gpt-5" },
156
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
157
+ providerModelPreferences: {
158
+ openai: "openai/gpt-5",
159
+ },
160
+ } as any);
161
+
162
+ expect(reconciled.modelSelection).toEqual({ mode: "auto" });
163
+ expect(reconciled.model).toBeUndefined();
164
+ expect(
165
+ resolveEffectiveModelRef({
166
+ modelSelection: reconciled.modelSelection,
167
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
168
+ providerModelPreferences: reconciled.providerModelPreferences,
169
+ } as any)
170
+ ).toBe("openai/gpt-5");
171
+ });
172
+ });
@@ -0,0 +1,39 @@
1
+ import { describe, expect, test } from "bun:test";
2
+ import {
3
+ renderOAuthErrorPage,
4
+ renderOAuthSuccessPage,
5
+ } from "../auth/oauth-templates";
6
+
7
+ describe("OAuth template escaping", () => {
8
+ test("escapes reflected OAuth error params", () => {
9
+ const html = renderOAuthErrorPage(
10
+ '<script>alert("xss")</script>',
11
+ '<img src=x onerror=alert("xss")>'
12
+ );
13
+
14
+ expect(html).not.toContain('<script>alert("xss")</script>');
15
+ expect(html).not.toContain('<img src=x onerror=alert("xss")>');
16
+ expect(html).toContain(
17
+ "&lt;script&gt;alert(&quot;xss&quot;)&lt;/script&gt;"
18
+ );
19
+ expect(html).toContain("&lt;img src=x onerror=alert(&quot;xss&quot;)&gt;");
20
+ });
21
+
22
+ test("escapes provider name on success page", () => {
23
+ const html = renderOAuthSuccessPage('"><svg onload=alert(1)>');
24
+
25
+ expect(html).not.toContain('"><svg onload=alert(1)>');
26
+ expect(html).toContain("&quot;&gt;&lt;svg onload=alert(1)&gt;");
27
+ });
28
+
29
+ test("escapes settings URL on success page", () => {
30
+ const html = renderOAuthSuccessPage(
31
+ "Google",
32
+ '"><script>alert(1)</script>'
33
+ );
34
+
35
+ expect(html).not.toContain('"><script>alert(1)</script>');
36
+ expect(html).toContain("&quot;&gt;&lt;script&gt;alert(1)&lt;/script&gt;");
37
+ expect(html).toContain("Open Configuration");
38
+ });
39
+ });
@@ -0,0 +1,114 @@
1
+ import { describe, expect, mock, test } from "bun:test";
2
+ import { ChatInstanceManager } from "../connections/chat-instance-manager";
3
+
4
+ /**
5
+ * Helper: create a ChatInstanceManager with mocked internals for testing
6
+ * sendPlatformMessage without requiring Redis.
7
+ */
8
+ function createTestManager(overrides: {
9
+ listConnections: (...args: any[]) => Promise<any[]>;
10
+ has: (id: string) => boolean;
11
+ getInstance: (id: string) => any;
12
+ }): ChatInstanceManager {
13
+ const manager = new ChatInstanceManager();
14
+ // Patch internal methods used by sendPlatformMessage / selectConnectionForPlatform
15
+ (manager as any).listConnections = overrides.listConnections;
16
+ manager.has = overrides.has;
17
+ manager.getInstance = overrides.getInstance;
18
+ return manager;
19
+ }
20
+
21
+ describe("ChatInstanceManager Slack sendPlatformMessage", () => {
22
+ test("posts top-level messages through the channel API", async () => {
23
+ const post = mock(async () => ({ ts: "1700000000.000100" }));
24
+ const channel = mock(() => ({ post }));
25
+
26
+ const manager = createTestManager({
27
+ listConnections: async () => [
28
+ {
29
+ id: "conn-1",
30
+ platform: "slack",
31
+ agentId: "system:connection:slack",
32
+ config: {
33
+ platform: "slack",
34
+ botToken: "xoxb",
35
+ signingSecret: "sig",
36
+ },
37
+ settings: { allowGroups: true },
38
+ metadata: {},
39
+ status: "active",
40
+ createdAt: 0,
41
+ updatedAt: 0,
42
+ },
43
+ ],
44
+ has: () => true,
45
+ getInstance: () => ({
46
+ chat: {
47
+ channel,
48
+ },
49
+ }),
50
+ });
51
+
52
+ const result = await manager.sendPlatformMessage("slack", "@me hello", {
53
+ agentId: "agent-1",
54
+ channelId: "C123",
55
+ teamId: "T123",
56
+ });
57
+
58
+ expect(result.messageId).toBe("1700000000.000100");
59
+ expect(channel).toHaveBeenCalledTimes(1);
60
+ expect(channel.mock.calls[0]?.[0]).toBe("slack:C123");
61
+ expect(post).toHaveBeenCalledTimes(1);
62
+ expect(post.mock.calls[0]?.[0]).toBe("@me hello");
63
+ });
64
+
65
+ test("posts thread replies through a resolved thread", async () => {
66
+ const post = mock(async () => ({ ts: "1700000000.000200" }));
67
+ const createThread = mock(async () => ({ post }));
68
+ const getAdapter = mock(() => ({ name: "slack" }));
69
+
70
+ const manager = createTestManager({
71
+ listConnections: async () => [
72
+ {
73
+ id: "conn-1",
74
+ platform: "slack",
75
+ agentId: "system:connection:slack:T123",
76
+ config: {
77
+ platform: "slack",
78
+ botToken: "xoxb",
79
+ signingSecret: "sig",
80
+ },
81
+ settings: { allowGroups: true },
82
+ metadata: { teamId: "T123" },
83
+ status: "active",
84
+ createdAt: 0,
85
+ updatedAt: 0,
86
+ },
87
+ ],
88
+ has: () => true,
89
+ getInstance: () => ({
90
+ chat: {
91
+ getAdapter,
92
+ createThread,
93
+ },
94
+ }),
95
+ });
96
+
97
+ const result = await manager.sendPlatformMessage("slack", "@me follow up", {
98
+ agentId: "agent-1",
99
+ channelId: "C123",
100
+ conversationId: "1700000000.000100",
101
+ teamId: "T123",
102
+ });
103
+
104
+ expect(result.messageId).toBe("1700000000.000200");
105
+ expect(getAdapter).toHaveBeenCalledTimes(1);
106
+ expect(getAdapter.mock.calls[0]?.[0]).toBe("slack");
107
+ expect(createThread).toHaveBeenCalledTimes(1);
108
+ expect(createThread.mock.calls[0]?.[1]).toBe(
109
+ "slack:C123:1700000000.000100"
110
+ );
111
+ expect(post).toHaveBeenCalledTimes(1);
112
+ expect(post.mock.calls[0]?.[0]).toBe("@me follow up");
113
+ });
114
+ });
@@ -0,0 +1,253 @@
1
+ import { afterEach, describe, expect, test } from "bun:test";
2
+ import {
3
+ hasConfiguredProvider,
4
+ resolveAgentId,
5
+ resolveAgentOptions,
6
+ } from "../services/platform-helpers";
7
+
8
+ const originalDispatcherServiceName = process.env.DISPATCHER_SERVICE_NAME;
9
+ const originalKubernetesNamespace = process.env.KUBERNETES_NAMESPACE;
10
+
11
+ afterEach(() => {
12
+ if (originalDispatcherServiceName === undefined) {
13
+ delete process.env.DISPATCHER_SERVICE_NAME;
14
+ } else {
15
+ process.env.DISPATCHER_SERVICE_NAME = originalDispatcherServiceName;
16
+ }
17
+
18
+ if (originalKubernetesNamespace === undefined) {
19
+ delete process.env.KUBERNETES_NAMESPACE;
20
+ } else {
21
+ process.env.KUBERNETES_NAMESPACE = originalKubernetesNamespace;
22
+ }
23
+ });
24
+
25
+ describe("resolveAgentOptions model resolution", () => {
26
+ test("uses pinned model when pinned provider is installed", async () => {
27
+ const settingsStore = {
28
+ getEffectiveSettings: async () =>
29
+ ({
30
+ modelSelection: {
31
+ mode: "pinned",
32
+ pinnedModel: "openai/gpt-5",
33
+ },
34
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
35
+ }) as any,
36
+ getSettings: async () =>
37
+ ({
38
+ modelSelection: {
39
+ mode: "pinned",
40
+ pinnedModel: "openai/gpt-5",
41
+ },
42
+ installedProviders: [{ providerId: "openai", installedAt: 1 }],
43
+ }) as any,
44
+ };
45
+
46
+ const resolved = await resolveAgentOptions(
47
+ "agent-1",
48
+ { model: "fallback-model" },
49
+ settingsStore as any
50
+ );
51
+
52
+ expect(resolved.model).toBe("openai/gpt-5");
53
+ });
54
+
55
+ test("uses primary provider preference in auto mode", async () => {
56
+ const settingsStore = {
57
+ getEffectiveSettings: async () =>
58
+ ({
59
+ modelSelection: {
60
+ mode: "auto",
61
+ },
62
+ installedProviders: [
63
+ { providerId: "chatgpt", installedAt: 1 },
64
+ { providerId: "claude", installedAt: 2 },
65
+ ],
66
+ providerModelPreferences: {
67
+ chatgpt: "chatgpt/gpt-5",
68
+ claude: "claude/sonnet",
69
+ },
70
+ }) as any,
71
+ getSettings: async () =>
72
+ ({
73
+ modelSelection: {
74
+ mode: "auto",
75
+ },
76
+ installedProviders: [
77
+ { providerId: "chatgpt", installedAt: 1 },
78
+ { providerId: "claude", installedAt: 2 },
79
+ ],
80
+ providerModelPreferences: {
81
+ chatgpt: "chatgpt/gpt-5",
82
+ claude: "claude/sonnet",
83
+ },
84
+ }) as any,
85
+ };
86
+
87
+ const resolved = await resolveAgentOptions(
88
+ "agent-1",
89
+ { model: "fallback-model" },
90
+ settingsStore as any
91
+ );
92
+
93
+ expect(resolved.model).toBe("chatgpt/gpt-5");
94
+ });
95
+
96
+ test("clears model in auto mode when providers exist but no preference", async () => {
97
+ const settingsStore = {
98
+ getEffectiveSettings: async () =>
99
+ ({
100
+ modelSelection: {
101
+ mode: "auto",
102
+ },
103
+ installedProviders: [{ providerId: "chatgpt", installedAt: 1 }],
104
+ }) as any,
105
+ getSettings: async () =>
106
+ ({
107
+ modelSelection: {
108
+ mode: "auto",
109
+ },
110
+ installedProviders: [{ providerId: "chatgpt", installedAt: 1 }],
111
+ }) as any,
112
+ };
113
+
114
+ const resolved = await resolveAgentOptions(
115
+ "agent-1",
116
+ { model: "fallback-model" },
117
+ settingsStore as any
118
+ );
119
+
120
+ expect(resolved.model).toBeUndefined();
121
+ });
122
+
123
+ test("normalizes legacy Owletto gateway URLs to the runtime K8s service", async () => {
124
+ process.env.DISPATCHER_SERVICE_NAME = "lobu-gateway";
125
+ process.env.KUBERNETES_NAMESPACE = "lobu";
126
+
127
+ const settingsStore = {
128
+ getEffectiveSettings: async () =>
129
+ ({
130
+ pluginsConfig: {
131
+ plugins: [
132
+ {
133
+ source: "@lobu/owletto-openclaw",
134
+ slot: "memory",
135
+ enabled: true,
136
+ config: {
137
+ mcpUrl: "http://gateway:8080/mcp/owletto",
138
+ gatewayAuthUrl: "http://gateway:8080",
139
+ },
140
+ },
141
+ ],
142
+ },
143
+ }) as any,
144
+ };
145
+
146
+ const resolved = await resolveAgentOptions(
147
+ "agent-1",
148
+ {},
149
+ settingsStore as any
150
+ );
151
+
152
+ expect(resolved.pluginsConfig).toEqual({
153
+ plugins: [
154
+ {
155
+ source: "@lobu/owletto-openclaw",
156
+ slot: "memory",
157
+ enabled: true,
158
+ config: {
159
+ mcpUrl:
160
+ "http://lobu-gateway.lobu.svc.cluster.local:8080/mcp/owletto",
161
+ gatewayAuthUrl: "http://lobu-gateway.lobu.svc.cluster.local:8080",
162
+ },
163
+ },
164
+ ],
165
+ });
166
+ });
167
+
168
+ test("preserves custom Owletto endpoints", async () => {
169
+ process.env.DISPATCHER_SERVICE_NAME = "lobu-gateway";
170
+ process.env.KUBERNETES_NAMESPACE = "lobu";
171
+
172
+ const settingsStore = {
173
+ getEffectiveSettings: async () =>
174
+ ({
175
+ pluginsConfig: {
176
+ plugins: [
177
+ {
178
+ source: "@lobu/owletto-openclaw",
179
+ slot: "memory",
180
+ enabled: true,
181
+ config: {
182
+ mcpUrl: "https://owletto.example.com/mcp",
183
+ gatewayAuthUrl: "https://owletto.example.com",
184
+ },
185
+ },
186
+ ],
187
+ },
188
+ }) as any,
189
+ };
190
+
191
+ const resolved = await resolveAgentOptions(
192
+ "agent-1",
193
+ {},
194
+ settingsStore as any
195
+ );
196
+
197
+ expect(resolved.pluginsConfig).toEqual({
198
+ plugins: [
199
+ {
200
+ source: "@lobu/owletto-openclaw",
201
+ slot: "memory",
202
+ enabled: true,
203
+ config: {
204
+ mcpUrl: "https://owletto.example.com/mcp",
205
+ gatewayAuthUrl: "https://owletto.example.com",
206
+ },
207
+ },
208
+ ],
209
+ });
210
+ });
211
+ });
212
+
213
+ describe("hasConfiguredProvider", () => {
214
+ test("accepts inherited template credentials from effective settings", async () => {
215
+ const settingsStore = {
216
+ getEffectiveSettings: async () =>
217
+ ({
218
+ authProfiles: [
219
+ {
220
+ id: "profile-1",
221
+ provider: "z-ai",
222
+ credential: "secret",
223
+ authType: "api-key",
224
+ label: "z.ai",
225
+ model: "*",
226
+ createdAt: 1,
227
+ },
228
+ ],
229
+ installedProviders: [{ providerId: "z-ai", installedAt: 1 }],
230
+ }) as any,
231
+ };
232
+
233
+ await expect(
234
+ hasConfiguredProvider("telegram-6570514069", settingsStore as any)
235
+ ).resolves.toBe(true);
236
+ });
237
+ });
238
+
239
+ describe("resolveAgentId", () => {
240
+ test("uses deterministic id by default", async () => {
241
+ const resolved = await resolveAgentId({
242
+ platform: "telegram",
243
+ userId: "777",
244
+ channelId: "12345",
245
+ isGroup: false,
246
+ });
247
+
248
+ expect(resolved).toEqual({
249
+ agentId: "telegram-777",
250
+ promptSent: false,
251
+ });
252
+ });
253
+ });