@lobu/gateway 3.0.7 → 3.0.8

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 (69) hide show
  1. package/dist/auth/bedrock/models.d.ts +13 -0
  2. package/dist/auth/bedrock/models.d.ts.map +1 -0
  3. package/dist/auth/bedrock/models.js +75 -0
  4. package/dist/auth/bedrock/models.js.map +1 -0
  5. package/dist/auth/bedrock/provider-module.d.ts +17 -0
  6. package/dist/auth/bedrock/provider-module.d.ts.map +1 -0
  7. package/dist/auth/bedrock/provider-module.js +80 -0
  8. package/dist/auth/bedrock/provider-module.js.map +1 -0
  9. package/dist/auth/external/device-code-client.d.ts +2 -0
  10. package/dist/auth/external/device-code-client.d.ts.map +1 -1
  11. package/dist/auth/external/device-code-client.js +12 -4
  12. package/dist/auth/external/device-code-client.js.map +1 -1
  13. package/dist/auth/mcp/config-service.d.ts +3 -4
  14. package/dist/auth/mcp/config-service.d.ts.map +1 -1
  15. package/dist/auth/mcp/config-service.js +40 -12
  16. package/dist/auth/mcp/config-service.js.map +1 -1
  17. package/dist/auth/mcp/proxy.d.ts +1 -3
  18. package/dist/auth/mcp/proxy.d.ts.map +1 -1
  19. package/dist/auth/mcp/proxy.js.map +1 -1
  20. package/dist/cli/gateway.d.ts.map +1 -1
  21. package/dist/cli/gateway.js +12 -5
  22. package/dist/cli/gateway.js.map +1 -1
  23. package/dist/gateway/index.d.ts.map +1 -1
  24. package/dist/gateway/index.js +1 -3
  25. package/dist/gateway/index.js.map +1 -1
  26. package/dist/routes/internal/device-auth.d.ts +7 -0
  27. package/dist/routes/internal/device-auth.d.ts.map +1 -1
  28. package/dist/routes/internal/device-auth.js +101 -48
  29. package/dist/routes/internal/device-auth.js.map +1 -1
  30. package/dist/routes/public/cli-auth.d.ts.map +1 -1
  31. package/dist/routes/public/cli-auth.js +10 -0
  32. package/dist/routes/public/cli-auth.js.map +1 -1
  33. package/dist/services/bedrock-anthropic-service.d.ts +87 -0
  34. package/dist/services/bedrock-anthropic-service.d.ts.map +1 -0
  35. package/dist/services/bedrock-anthropic-service.js +453 -0
  36. package/dist/services/bedrock-anthropic-service.js.map +1 -0
  37. package/dist/services/bedrock-model-catalog.d.ts +28 -0
  38. package/dist/services/bedrock-model-catalog.d.ts.map +1 -0
  39. package/dist/services/bedrock-model-catalog.js +160 -0
  40. package/dist/services/bedrock-model-catalog.js.map +1 -0
  41. package/dist/services/bedrock-openai-service.d.ts +119 -0
  42. package/dist/services/bedrock-openai-service.d.ts.map +1 -0
  43. package/dist/services/bedrock-openai-service.js +412 -0
  44. package/dist/services/bedrock-openai-service.js.map +1 -0
  45. package/dist/services/core-services.d.ts +3 -0
  46. package/dist/services/core-services.d.ts.map +1 -1
  47. package/dist/services/core-services.js +13 -0
  48. package/dist/services/core-services.js.map +1 -1
  49. package/dist/services/system-config-resolver.d.ts.map +1 -1
  50. package/dist/services/system-config-resolver.js +0 -2
  51. package/dist/services/system-config-resolver.js.map +1 -1
  52. package/package.json +3 -1
  53. package/src/__tests__/bedrock-model-catalog.test.ts +40 -0
  54. package/src/__tests__/bedrock-openai-service.test.ts +157 -0
  55. package/src/__tests__/bedrock-provider-module.test.ts +56 -0
  56. package/src/__tests__/mcp-config-service.test.ts +1 -1
  57. package/src/__tests__/mcp-proxy.test.ts +1 -3
  58. package/src/auth/bedrock/provider-module.ts +110 -0
  59. package/src/auth/external/device-code-client.ts +14 -4
  60. package/src/auth/mcp/config-service.ts +49 -21
  61. package/src/auth/mcp/proxy.ts +1 -3
  62. package/src/cli/gateway.ts +8 -0
  63. package/src/gateway/index.ts +1 -3
  64. package/src/routes/internal/device-auth.ts +137 -51
  65. package/src/routes/public/cli-auth.ts +13 -0
  66. package/src/services/bedrock-model-catalog.ts +217 -0
  67. package/src/services/bedrock-openai-service.ts +658 -0
  68. package/src/services/core-services.ts +19 -0
  69. package/src/services/system-config-resolver.ts +0 -1
@@ -0,0 +1,217 @@
1
+ import {
2
+ BedrockClient,
3
+ InferenceType,
4
+ ListFoundationModelsCommand,
5
+ ModelModality,
6
+ type FoundationModelSummary,
7
+ } from "@aws-sdk/client-bedrock";
8
+ import { getModels, type Model } from "@mariozechner/pi-ai";
9
+ import { createLogger } from "@lobu/core";
10
+
11
+ const logger = createLogger("bedrock-model-catalog");
12
+
13
+ const CACHE_TTL_MS = 5 * 60 * 1000;
14
+
15
+ export interface BedrockCatalogModel {
16
+ id: string;
17
+ label: string;
18
+ providerName?: string;
19
+ modelName?: string;
20
+ inputModalities?: string[];
21
+ outputModalities?: string[];
22
+ }
23
+
24
+ export interface BedrockModelCatalogOptions {
25
+ cacheTtlMs?: number;
26
+ loadModels?: () => Promise<BedrockCatalogModel[]>;
27
+ }
28
+
29
+ export function resolveAwsRegion(): string | undefined {
30
+ return process.env.AWS_REGION || process.env.AWS_DEFAULT_REGION || undefined;
31
+ }
32
+
33
+ function hasTextOutput(summary: FoundationModelSummary): boolean {
34
+ return (summary.outputModalities || []).includes(ModelModality.TEXT);
35
+ }
36
+
37
+ function hasTextInput(summary: FoundationModelSummary): boolean {
38
+ return (summary.inputModalities || []).includes(ModelModality.TEXT);
39
+ }
40
+
41
+ function supportsStreaming(summary: FoundationModelSummary): boolean {
42
+ return summary.responseStreamingSupported === true;
43
+ }
44
+
45
+ function supportsOnDemand(summary: FoundationModelSummary): boolean {
46
+ const inferenceTypes = summary.inferenceTypesSupported || [];
47
+ return (
48
+ inferenceTypes.length === 0 ||
49
+ inferenceTypes.includes(InferenceType.ON_DEMAND)
50
+ );
51
+ }
52
+
53
+ function buildModelLabel(model: {
54
+ providerName?: string;
55
+ modelName?: string;
56
+ id: string;
57
+ }): string {
58
+ const provider = model.providerName?.trim();
59
+ const name = model.modelName?.trim();
60
+ if (provider && name) return `${provider} / ${name}`;
61
+ return name || model.id;
62
+ }
63
+
64
+ function normalizeFoundationModel(
65
+ summary: FoundationModelSummary
66
+ ): BedrockCatalogModel | null {
67
+ const id = summary.modelId?.trim();
68
+ if (!id) return null;
69
+ if (!hasTextOutput(summary) || !hasTextInput(summary)) return null;
70
+ if (!supportsStreaming(summary) || !supportsOnDemand(summary)) return null;
71
+
72
+ return {
73
+ id,
74
+ label: buildModelLabel({
75
+ providerName: summary.providerName,
76
+ modelName: summary.modelName,
77
+ id,
78
+ }),
79
+ providerName: summary.providerName,
80
+ modelName: summary.modelName,
81
+ inputModalities: summary.inputModalities,
82
+ outputModalities: summary.outputModalities,
83
+ };
84
+ }
85
+
86
+ function normalizeRegistryModel(model: Model<any>): BedrockCatalogModel {
87
+ return {
88
+ id: model.id,
89
+ label: model.name || model.id,
90
+ inputModalities: model.input.map((input) => input.toUpperCase()),
91
+ outputModalities: ["TEXT"],
92
+ };
93
+ }
94
+
95
+ function sortModels(a: BedrockCatalogModel, b: BedrockCatalogModel): number {
96
+ return a.label.localeCompare(b.label) || a.id.localeCompare(b.id);
97
+ }
98
+
99
+ export function buildDynamicBedrockModel(
100
+ modelId: string,
101
+ discovered?: Pick<
102
+ BedrockCatalogModel,
103
+ "inputModalities" | "modelName" | "providerName"
104
+ > | null
105
+ ): Model<"bedrock-converse-stream"> {
106
+ const staticModel = getModels("amazon-bedrock").find((m) => m.id === modelId);
107
+ if (staticModel) {
108
+ return staticModel as Model<"bedrock-converse-stream">;
109
+ }
110
+
111
+ const input = (discovered?.inputModalities || [])
112
+ .map((value) => value.toLowerCase())
113
+ .filter(
114
+ (value): value is "text" | "image" =>
115
+ value === "text" || value === "image"
116
+ );
117
+
118
+ return {
119
+ id: modelId,
120
+ name:
121
+ buildModelLabel({
122
+ providerName: discovered?.providerName,
123
+ modelName: discovered?.modelName,
124
+ id: modelId,
125
+ }) || modelId,
126
+ api: "bedrock-converse-stream",
127
+ provider: "amazon-bedrock",
128
+ baseUrl: "",
129
+ reasoning: false,
130
+ input: input.length > 0 ? input : ["text"],
131
+ contextWindow: 128000,
132
+ maxTokens: 8192,
133
+ cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
134
+ };
135
+ }
136
+
137
+ async function loadBedrockModelsFromAws(): Promise<BedrockCatalogModel[]> {
138
+ const region = resolveAwsRegion();
139
+ if (!region) {
140
+ throw new Error("AWS region is not configured");
141
+ }
142
+
143
+ const client = new BedrockClient({ region });
144
+ const response = await client.send(
145
+ new ListFoundationModelsCommand({
146
+ byOutputModality: ModelModality.TEXT,
147
+ })
148
+ );
149
+
150
+ return (response.modelSummaries || [])
151
+ .map(normalizeFoundationModel)
152
+ .filter((model): model is BedrockCatalogModel => Boolean(model))
153
+ .sort(sortModels);
154
+ }
155
+
156
+ function loadFallbackRegistryModels(): BedrockCatalogModel[] {
157
+ return getModels("amazon-bedrock")
158
+ .map(normalizeRegistryModel)
159
+ .sort(sortModels);
160
+ }
161
+
162
+ export class BedrockModelCatalog {
163
+ private readonly cacheTtlMs: number;
164
+ private readonly loadModelsImpl: () => Promise<BedrockCatalogModel[]>;
165
+ private cachedModels:
166
+ | { expiresAt: number; models: BedrockCatalogModel[] }
167
+ | undefined;
168
+
169
+ constructor(options: BedrockModelCatalogOptions = {}) {
170
+ this.cacheTtlMs = options.cacheTtlMs ?? CACHE_TTL_MS;
171
+ this.loadModelsImpl = options.loadModels || loadBedrockModelsFromAws;
172
+ }
173
+
174
+ async listModels(): Promise<BedrockCatalogModel[]> {
175
+ const now = Date.now();
176
+ if (this.cachedModels && this.cachedModels.expiresAt > now) {
177
+ return this.cachedModels.models;
178
+ }
179
+
180
+ try {
181
+ const models = await this.loadModelsImpl();
182
+ this.cachedModels = {
183
+ expiresAt: now + this.cacheTtlMs,
184
+ models,
185
+ };
186
+ return models;
187
+ } catch (error) {
188
+ logger.warn(
189
+ {
190
+ error: error instanceof Error ? error.message : String(error),
191
+ },
192
+ "Falling back to static Bedrock model registry"
193
+ );
194
+ const fallback = loadFallbackRegistryModels();
195
+ this.cachedModels = {
196
+ expiresAt: now + this.cacheTtlMs,
197
+ models: fallback,
198
+ };
199
+ return fallback;
200
+ }
201
+ }
202
+
203
+ async listModelOptions(): Promise<Array<{ id: string; label: string }>> {
204
+ const models = await this.listModels();
205
+ return models.map((model) => ({
206
+ id: model.id,
207
+ label: model.label,
208
+ }));
209
+ }
210
+
211
+ async getModel(modelId: string): Promise<BedrockCatalogModel | null> {
212
+ const normalized = modelId.trim();
213
+ if (!normalized) return null;
214
+ const models = await this.listModels();
215
+ return models.find((model) => model.id === normalized) || null;
216
+ }
217
+ }