@objectstack/embedder-openai 7.8.0 → 8.0.0

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.
@@ -1,5 +1,5 @@
1
1
 
2
- > @objectstack/embedder-openai@7.8.0 build /home/runner/work/framework/framework/packages/plugins/embedder-openai
2
+ > @objectstack/embedder-openai@8.0.0 build /home/runner/work/framework/framework/packages/plugins/embedder-openai
3
3
  > tsup --config ../../../tsup.config.ts
4
4
 
5
5
  CLI Building entry: src/index.ts
@@ -10,13 +10,13 @@
10
10
  CLI Cleaning output folder
11
11
  ESM Build start
12
12
  CJS Build start
13
- CJS dist/index.js 4.27 KB
14
- CJS dist/index.js.map 9.70 KB
15
- CJS ⚡️ Build success in 52ms
16
- ESM dist/index.mjs 3.16 KB
17
- ESM dist/index.mjs.map 9.64 KB
18
- ESM ⚡️ Build success in 53ms
13
+ ESM dist/index.mjs 3.25 KB
14
+ ESM dist/index.mjs.map 9.77 KB
15
+ ESM ⚡️ Build success in 48ms
16
+ CJS dist/index.js 4.38 KB
17
+ CJS dist/index.js.map 9.82 KB
18
+ CJS ⚡️ Build success in 48ms
19
19
  DTS Build start
20
- DTS ⚡️ Build success in 4410ms
20
+ DTS ⚡️ Build success in 5103ms
21
21
  DTS dist/index.d.mts 4.86 KB
22
22
  DTS dist/index.d.ts 4.86 KB
package/CHANGELOG.md CHANGED
@@ -1,5 +1,46 @@
1
1
  # @objectstack/embedder-openai
2
2
 
3
+ ## 8.0.0
4
+
5
+ ### Patch Changes
6
+
7
+ - d5a8161: feat(spec): resilientFetch — timeout + backoff for outbound HTTP (P1-1)
8
+
9
+ Outbound calls in the connectors/embedder were naked `fetch` with no timeout or
10
+ retry, so a slow or rate-limited external API could hang an agent turn with no
11
+ recovery.
12
+
13
+ New shared `resilientFetch` (`@objectstack/spec/shared`):
14
+
15
+ - per-attempt timeout via `AbortController` (default 30s);
16
+ - exponential backoff with jitter, up to 3 attempts, on network errors / 429 / 5xx;
17
+ - honours a `Retry-After` header on 429;
18
+ - never retries a caller-initiated abort (intentional cancellation).
19
+
20
+ Wired into `connector-rest`, `connector-slack`, and `embedder-openai`.
21
+ `connector-mcp` talks through the MCP SDK transport, so it gets a 30s per-request
22
+ `timeout` on `callTool` / `listTools` instead.
23
+
24
+ A stateful per-host **circuit breaker** is deliberately left as a follow-up:
25
+ timeout + backoff already removes the hang/no-recovery risk.
26
+
27
+ - Updated dependencies [a46c017]
28
+ - Updated dependencies [b990b89]
29
+ - Updated dependencies [99111ec]
30
+ - Updated dependencies [d5a8161]
31
+ - Updated dependencies [5cf1f1b]
32
+ - Updated dependencies [9ef89d4]
33
+ - Updated dependencies [3306d2f]
34
+ - Updated dependencies [bc44195]
35
+ - Updated dependencies [9e2e229]
36
+ - @objectstack/spec@8.0.0
37
+
38
+ ## 7.9.0
39
+
40
+ ### Patch Changes
41
+
42
+ - @objectstack/spec@7.9.0
43
+
3
44
  ## 7.8.0
4
45
 
5
46
  ### Patch Changes
package/dist/index.js CHANGED
@@ -26,6 +26,7 @@ __export(index_exports, {
26
26
  default: () => index_default
27
27
  });
28
28
  module.exports = __toCommonJS(index_exports);
29
+ var import_shared = require("@objectstack/spec/shared");
29
30
  var KNOWN_DIMENSIONS = {
30
31
  // OpenAI
31
32
  "text-embedding-3-small": 1536,
@@ -74,7 +75,7 @@ var OpenAIEmbedder = class {
74
75
  if (texts.length === 0) return [];
75
76
  const body = { model: this.model, input: texts };
76
77
  if (this.requestedDims) body.dimensions = this.requestedDims;
77
- const res = await this.fetchImpl(`${this.baseUrl}/embeddings`, {
78
+ const res = await (0, import_shared.resilientFetch)(`${this.baseUrl}/embeddings`, {
78
79
  method: "POST",
79
80
  headers: {
80
81
  "content-type": "application/json",
@@ -82,7 +83,7 @@ var OpenAIEmbedder = class {
82
83
  ...this.extraHeaders
83
84
  },
84
85
  body: JSON.stringify(body)
85
- });
86
+ }, { fetchImpl: this.fetchImpl });
86
87
  if (!res.ok) {
87
88
  const text = await res.text().catch(() => "");
88
89
  throw new Error(
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2026 ObjectStack. Licensed under the Apache-2.0 license.\n\n/**\n * `@objectstack/embedder-openai`\n *\n * OpenAI-compatible embedder. Drop-in for any endpoint that speaks the\n * `POST /v1/embeddings` shape:\n *\n * - OpenAI https://api.openai.com/v1\n * - Azure OpenAI https://{resource}.openai.azure.com/openai/deployments/{deployment}\n * - 阿里通义 DashScope https://dashscope.aliyuncs.com/compatible-mode/v1\n * - 智谱 BigModel https://open.bigmodel.cn/api/paas/v4\n * - 硅基流动 SiliconFlow https://api.siliconflow.cn/v1\n * - 火山引擎 Doubao https://ark.cn-beijing.volces.com/api/v3\n * - MiniMax https://api.minimax.chat/v1\n * - Ollama (openai shim) http://localhost:11434/v1\n * - LiteLLM / vLLM / 任何兼容服务\n *\n * Implements the `IEmbedder` contract from `@objectstack/spec/contracts`.\n */\n\nimport type { IEmbedder } from '@objectstack/spec/contracts';\n\nexport interface OpenAIEmbedderOptions {\n /** Bearer token sent as `Authorization: Bearer <apiKey>`. Required. */\n apiKey: string;\n /**\n * Model id sent in the request body. Choose to match your provider:\n * - OpenAI: `'text-embedding-3-small'` (default), `'text-embedding-3-large'`\n * - 阿里通义: `'text-embedding-v3'`\n * - 智谱: `'embedding-3'`\n * - 硅基流动: `'BAAI/bge-m3'`, `'BAAI/bge-large-zh-v1.5'`\n * - 火山 Doubao: `'doubao-embedding-large-text-240915'`\n * - Ollama: `'bge-m3'`, `'nomic-embed-text'`\n *\n * @default 'text-embedding-3-small'\n */\n model?: string;\n /**\n * Override dimensions. Only Matryoshka-style models (OpenAI v3, 智谱 embedding-3,\n * BGE-m3 dense) support truncation. When set, also forwarded to the upstream\n * `dimensions` body field for providers that honour it.\n */\n dimensions?: number;\n /**\n * Endpoint base URL (without `/embeddings`). Defaults to OpenAI's. Set this\n * to point at any compatible provider.\n *\n * @default 'https://api.openai.com/v1'\n */\n baseUrl?: string;\n /** Stable id surfaced as `IEmbedder.id`. @default 'openai' */\n id?: string;\n /** Inject for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n /** Additional headers (e.g. provider-specific keys, tracing). */\n headers?: Record<string, string>;\n}\n\n/**\n * Known dimensions for popular models. Used as the default when the\n * caller doesn't pass `dimensions` explicitly.\n */\nconst KNOWN_DIMENSIONS: Record<string, number> = {\n // OpenAI\n 'text-embedding-3-small': 1536,\n 'text-embedding-3-large': 3072,\n 'text-embedding-ada-002': 1536,\n // 阿里通义\n 'text-embedding-v3': 1024,\n 'text-embedding-v2': 1536,\n 'text-embedding-v1': 1536,\n // 智谱\n 'embedding-3': 2048,\n 'embedding-2': 1024,\n // 硅基流动 / BGE 家族\n 'BAAI/bge-m3': 1024,\n 'BAAI/bge-large-zh-v1.5': 1024,\n 'BAAI/bge-large-en-v1.5': 1024,\n 'BAAI/bge-base-zh-v1.5': 768,\n 'BAAI/bge-small-zh-v1.5': 512,\n 'bge-m3': 1024,\n // 火山 Doubao\n 'doubao-embedding-large-text-240915': 4096,\n 'doubao-embedding-text-240715': 2048,\n // Nomic / Ollama defaults\n 'nomic-embed-text': 768,\n // MiniMax\n 'embo-01': 1536,\n};\n\n/**\n * `OpenAIEmbedder` — OpenAI-compatible embedder. One instance per\n * upstream provider + model combination. Pass into any knowledge\n * adapter that expects `IEmbedder`.\n *\n * @example\n * // OpenAI\n * new OpenAIEmbedder({ apiKey: process.env.OPENAI_API_KEY! });\n *\n * @example\n * // 阿里通义 DashScope\n * new OpenAIEmbedder({\n * apiKey: process.env.DASHSCOPE_API_KEY!,\n * baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n * model: 'text-embedding-v3',\n * });\n *\n * @example\n * // 硅基流动 SiliconFlow + BGE-m3\n * new OpenAIEmbedder({\n * apiKey: process.env.SILICONFLOW_API_KEY!,\n * baseUrl: 'https://api.siliconflow.cn/v1',\n * model: 'BAAI/bge-m3',\n * });\n *\n * @example\n * // Local Ollama\n * new OpenAIEmbedder({\n * apiKey: 'ollama',\n * baseUrl: 'http://localhost:11434/v1',\n * model: 'bge-m3',\n * });\n */\nexport class OpenAIEmbedder implements IEmbedder {\n readonly id: string;\n readonly dimensions: number;\n private readonly model: string;\n private readonly baseUrl: string;\n private readonly apiKey: string;\n private readonly fetchImpl: typeof fetch;\n private readonly requestedDims?: number;\n private readonly extraHeaders: Record<string, string>;\n\n constructor(opts: OpenAIEmbedderOptions) {\n if (!opts.apiKey) throw new Error('OpenAIEmbedder: apiKey required');\n this.apiKey = opts.apiKey;\n this.id = opts.id ?? 'openai';\n this.model = opts.model ?? 'text-embedding-3-small';\n this.baseUrl = (opts.baseUrl ?? 'https://api.openai.com/v1').replace(/\\/+$/, '');\n this.fetchImpl = opts.fetch ?? (globalThis.fetch as typeof fetch);\n this.requestedDims = opts.dimensions;\n this.extraHeaders = opts.headers ?? {};\n this.dimensions =\n opts.dimensions ?? KNOWN_DIMENSIONS[this.model] ?? 1536;\n if (!this.fetchImpl) {\n throw new Error(\n 'OpenAIEmbedder: no fetch available; pass options.fetch or run on Node 18+ / a fetch-capable runtime',\n );\n }\n }\n\n async embed(texts: string[]): Promise<number[][]> {\n if (texts.length === 0) return [];\n const body: Record<string, unknown> = { model: this.model, input: texts };\n if (this.requestedDims) body.dimensions = this.requestedDims;\n const res = await this.fetchImpl(`${this.baseUrl}/embeddings`, {\n method: 'POST',\n headers: {\n 'content-type': 'application/json',\n authorization: `Bearer ${this.apiKey}`,\n ...this.extraHeaders,\n },\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => '');\n throw new Error(\n `OpenAIEmbedder (${this.baseUrl}) → ${res.status} ${res.statusText}${text ? `: ${text.slice(0, 200)}` : ''}`,\n );\n }\n const json = (await res.json()) as { data?: Array<{ embedding: number[] }> };\n const data = json.data ?? [];\n if (data.length !== texts.length) {\n throw new Error(\n `OpenAIEmbedder: expected ${texts.length} vectors, got ${data.length}`,\n );\n }\n return data.map((d) => d.embedding);\n }\n}\n\n/**\n * Convenience presets for popular Chinese providers — saves callers\n * from memorising base URLs. Pass through `createXxxEmbedder({...})`.\n */\nexport const OPENAI_COMPATIBLE_PRESETS = {\n openai: 'https://api.openai.com/v1',\n azure: '', // user must provide full deployment URL via baseUrl\n dashscope: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n zhipu: 'https://open.bigmodel.cn/api/paas/v4',\n siliconflow: 'https://api.siliconflow.cn/v1',\n doubao: 'https://ark.cn-beijing.volces.com/api/v3',\n minimax: 'https://api.minimax.chat/v1',\n ollama: 'http://localhost:11434/v1',\n} as const;\n\nexport type OpenAICompatiblePreset = keyof typeof OPENAI_COMPATIBLE_PRESETS;\n\nexport interface PresetEmbedderOptions\n extends Omit<OpenAIEmbedderOptions, 'baseUrl'> {\n /** Pick a known provider; sets `baseUrl` automatically. */\n preset?: OpenAICompatiblePreset;\n /** Explicit override; takes precedence over `preset`. */\n baseUrl?: string;\n}\n\n/**\n * Helper: pick a provider by preset name. Equivalent to constructing\n * `OpenAIEmbedder` with the matching `baseUrl`.\n *\n * @example\n * createOpenAIEmbedder({ preset: 'dashscope', apiKey, model: 'text-embedding-v3' })\n */\nexport function createOpenAIEmbedder(opts: PresetEmbedderOptions): OpenAIEmbedder {\n const baseUrl =\n opts.baseUrl ??\n (opts.preset ? OPENAI_COMPATIBLE_PRESETS[opts.preset] : undefined);\n return new OpenAIEmbedder({ ...opts, baseUrl, id: opts.id ?? opts.preset ?? 'openai' });\n}\n\nexport default OpenAIEmbedder;\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AA+DA,IAAM,mBAA2C;AAAA;AAAA,EAE/C,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA;AAAA,EAE1B,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA;AAAA,EAErB,eAAe;AAAA,EACf,eAAe;AAAA;AAAA,EAEf,eAAe;AAAA,EACf,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,0BAA0B;AAAA,EAC1B,UAAU;AAAA;AAAA,EAEV,sCAAsC;AAAA,EACtC,gCAAgC;AAAA;AAAA,EAEhC,oBAAoB;AAAA;AAAA,EAEpB,WAAW;AACb;AAmCO,IAAM,iBAAN,MAA0C;AAAA,EAU/C,YAAY,MAA6B;AACvC,QAAI,CAAC,KAAK,OAAQ,OAAM,IAAI,MAAM,iCAAiC;AACnE,SAAK,SAAS,KAAK;AACnB,SAAK,KAAK,KAAK,MAAM;AACrB,SAAK,QAAQ,KAAK,SAAS;AAC3B,SAAK,WAAW,KAAK,WAAW,6BAA6B,QAAQ,QAAQ,EAAE;AAC/E,SAAK,YAAY,KAAK,SAAU,WAAW;AAC3C,SAAK,gBAAgB,KAAK;AAC1B,SAAK,eAAe,KAAK,WAAW,CAAC;AACrC,SAAK,aACH,KAAK,cAAc,iBAAiB,KAAK,KAAK,KAAK;AACrD,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,IAAI;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,MAAM,OAAsC;AAChD,QAAI,MAAM,WAAW,EAAG,QAAO,CAAC;AAChC,UAAM,OAAgC,EAAE,OAAO,KAAK,OAAO,OAAO,MAAM;AACxE,QAAI,KAAK,cAAe,MAAK,aAAa,KAAK;AAC/C,UAAM,MAAM,MAAM,KAAK,UAAU,GAAG,KAAK,OAAO,eAAe;AAAA,MAC7D,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,eAAe,UAAU,KAAK,MAAM;AAAA,QACpC,GAAG,KAAK;AAAA,MACV;AAAA,MACA,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B,CAAC;AACD,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,OAAO,MAAM,IAAI,KAAK,EAAE,MAAM,MAAM,EAAE;AAC5C,YAAM,IAAI;AAAA,QACR,mBAAmB,KAAK,OAAO,YAAO,IAAI,MAAM,IAAI,IAAI,UAAU,GAAG,OAAO,KAAK,KAAK,MAAM,GAAG,GAAG,CAAC,KAAK,EAAE;AAAA,MAC5G;AAAA,IACF;AACA,UAAM,OAAQ,MAAM,IAAI,KAAK;AAC7B,UAAM,OAAO,KAAK,QAAQ,CAAC;AAC3B,QAAI,KAAK,WAAW,MAAM,QAAQ;AAChC,YAAM,IAAI;AAAA,QACR,4BAA4B,MAAM,MAAM,iBAAiB,KAAK,MAAM;AAAA,MACtE;AAAA,IACF;AACA,WAAO,KAAK,IAAI,CAAC,MAAM,EAAE,SAAS;AAAA,EACpC;AACF;AAMO,IAAM,4BAA4B;AAAA,EACvC,QAAQ;AAAA,EACR,OAAO;AAAA;AAAA,EACP,WAAW;AAAA,EACX,OAAO;AAAA,EACP,aAAa;AAAA,EACb,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,QAAQ;AACV;AAmBO,SAAS,qBAAqB,MAA6C;AAChF,QAAM,UACJ,KAAK,YACJ,KAAK,SAAS,0BAA0B,KAAK,MAAM,IAAI;AAC1D,SAAO,IAAI,eAAe,EAAE,GAAG,MAAM,SAAS,IAAI,KAAK,MAAM,KAAK,UAAU,SAAS,CAAC;AACxF;AAEA,IAAO,gBAAQ;","names":[]}
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2026 ObjectStack. Licensed under the Apache-2.0 license.\n\n/**\n * `@objectstack/embedder-openai`\n *\n * OpenAI-compatible embedder. Drop-in for any endpoint that speaks the\n * `POST /v1/embeddings` shape:\n *\n * - OpenAI https://api.openai.com/v1\n * - Azure OpenAI https://{resource}.openai.azure.com/openai/deployments/{deployment}\n * - 阿里通义 DashScope https://dashscope.aliyuncs.com/compatible-mode/v1\n * - 智谱 BigModel https://open.bigmodel.cn/api/paas/v4\n * - 硅基流动 SiliconFlow https://api.siliconflow.cn/v1\n * - 火山引擎 Doubao https://ark.cn-beijing.volces.com/api/v3\n * - MiniMax https://api.minimax.chat/v1\n * - Ollama (openai shim) http://localhost:11434/v1\n * - LiteLLM / vLLM / 任何兼容服务\n *\n * Implements the `IEmbedder` contract from `@objectstack/spec/contracts`.\n */\n\nimport type { IEmbedder } from '@objectstack/spec/contracts';\nimport { resilientFetch } from '@objectstack/spec/shared';\n\nexport interface OpenAIEmbedderOptions {\n /** Bearer token sent as `Authorization: Bearer <apiKey>`. Required. */\n apiKey: string;\n /**\n * Model id sent in the request body. Choose to match your provider:\n * - OpenAI: `'text-embedding-3-small'` (default), `'text-embedding-3-large'`\n * - 阿里通义: `'text-embedding-v3'`\n * - 智谱: `'embedding-3'`\n * - 硅基流动: `'BAAI/bge-m3'`, `'BAAI/bge-large-zh-v1.5'`\n * - 火山 Doubao: `'doubao-embedding-large-text-240915'`\n * - Ollama: `'bge-m3'`, `'nomic-embed-text'`\n *\n * @default 'text-embedding-3-small'\n */\n model?: string;\n /**\n * Override dimensions. Only Matryoshka-style models (OpenAI v3, 智谱 embedding-3,\n * BGE-m3 dense) support truncation. When set, also forwarded to the upstream\n * `dimensions` body field for providers that honour it.\n */\n dimensions?: number;\n /**\n * Endpoint base URL (without `/embeddings`). Defaults to OpenAI's. Set this\n * to point at any compatible provider.\n *\n * @default 'https://api.openai.com/v1'\n */\n baseUrl?: string;\n /** Stable id surfaced as `IEmbedder.id`. @default 'openai' */\n id?: string;\n /** Inject for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n /** Additional headers (e.g. provider-specific keys, tracing). */\n headers?: Record<string, string>;\n}\n\n/**\n * Known dimensions for popular models. Used as the default when the\n * caller doesn't pass `dimensions` explicitly.\n */\nconst KNOWN_DIMENSIONS: Record<string, number> = {\n // OpenAI\n 'text-embedding-3-small': 1536,\n 'text-embedding-3-large': 3072,\n 'text-embedding-ada-002': 1536,\n // 阿里通义\n 'text-embedding-v3': 1024,\n 'text-embedding-v2': 1536,\n 'text-embedding-v1': 1536,\n // 智谱\n 'embedding-3': 2048,\n 'embedding-2': 1024,\n // 硅基流动 / BGE 家族\n 'BAAI/bge-m3': 1024,\n 'BAAI/bge-large-zh-v1.5': 1024,\n 'BAAI/bge-large-en-v1.5': 1024,\n 'BAAI/bge-base-zh-v1.5': 768,\n 'BAAI/bge-small-zh-v1.5': 512,\n 'bge-m3': 1024,\n // 火山 Doubao\n 'doubao-embedding-large-text-240915': 4096,\n 'doubao-embedding-text-240715': 2048,\n // Nomic / Ollama defaults\n 'nomic-embed-text': 768,\n // MiniMax\n 'embo-01': 1536,\n};\n\n/**\n * `OpenAIEmbedder` — OpenAI-compatible embedder. One instance per\n * upstream provider + model combination. Pass into any knowledge\n * adapter that expects `IEmbedder`.\n *\n * @example\n * // OpenAI\n * new OpenAIEmbedder({ apiKey: process.env.OPENAI_API_KEY! });\n *\n * @example\n * // 阿里通义 DashScope\n * new OpenAIEmbedder({\n * apiKey: process.env.DASHSCOPE_API_KEY!,\n * baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n * model: 'text-embedding-v3',\n * });\n *\n * @example\n * // 硅基流动 SiliconFlow + BGE-m3\n * new OpenAIEmbedder({\n * apiKey: process.env.SILICONFLOW_API_KEY!,\n * baseUrl: 'https://api.siliconflow.cn/v1',\n * model: 'BAAI/bge-m3',\n * });\n *\n * @example\n * // Local Ollama\n * new OpenAIEmbedder({\n * apiKey: 'ollama',\n * baseUrl: 'http://localhost:11434/v1',\n * model: 'bge-m3',\n * });\n */\nexport class OpenAIEmbedder implements IEmbedder {\n readonly id: string;\n readonly dimensions: number;\n private readonly model: string;\n private readonly baseUrl: string;\n private readonly apiKey: string;\n private readonly fetchImpl: typeof fetch;\n private readonly requestedDims?: number;\n private readonly extraHeaders: Record<string, string>;\n\n constructor(opts: OpenAIEmbedderOptions) {\n if (!opts.apiKey) throw new Error('OpenAIEmbedder: apiKey required');\n this.apiKey = opts.apiKey;\n this.id = opts.id ?? 'openai';\n this.model = opts.model ?? 'text-embedding-3-small';\n this.baseUrl = (opts.baseUrl ?? 'https://api.openai.com/v1').replace(/\\/+$/, '');\n this.fetchImpl = opts.fetch ?? (globalThis.fetch as typeof fetch);\n this.requestedDims = opts.dimensions;\n this.extraHeaders = opts.headers ?? {};\n this.dimensions =\n opts.dimensions ?? KNOWN_DIMENSIONS[this.model] ?? 1536;\n if (!this.fetchImpl) {\n throw new Error(\n 'OpenAIEmbedder: no fetch available; pass options.fetch or run on Node 18+ / a fetch-capable runtime',\n );\n }\n }\n\n async embed(texts: string[]): Promise<number[][]> {\n if (texts.length === 0) return [];\n const body: Record<string, unknown> = { model: this.model, input: texts };\n if (this.requestedDims) body.dimensions = this.requestedDims;\n const res = await resilientFetch(`${this.baseUrl}/embeddings`, {\n method: 'POST',\n headers: {\n 'content-type': 'application/json',\n authorization: `Bearer ${this.apiKey}`,\n ...this.extraHeaders,\n },\n body: JSON.stringify(body),\n }, { fetchImpl: this.fetchImpl });\n if (!res.ok) {\n const text = await res.text().catch(() => '');\n throw new Error(\n `OpenAIEmbedder (${this.baseUrl}) → ${res.status} ${res.statusText}${text ? `: ${text.slice(0, 200)}` : ''}`,\n );\n }\n const json = (await res.json()) as { data?: Array<{ embedding: number[] }> };\n const data = json.data ?? [];\n if (data.length !== texts.length) {\n throw new Error(\n `OpenAIEmbedder: expected ${texts.length} vectors, got ${data.length}`,\n );\n }\n return data.map((d) => d.embedding);\n }\n}\n\n/**\n * Convenience presets for popular Chinese providers — saves callers\n * from memorising base URLs. Pass through `createXxxEmbedder({...})`.\n */\nexport const OPENAI_COMPATIBLE_PRESETS = {\n openai: 'https://api.openai.com/v1',\n azure: '', // user must provide full deployment URL via baseUrl\n dashscope: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n zhipu: 'https://open.bigmodel.cn/api/paas/v4',\n siliconflow: 'https://api.siliconflow.cn/v1',\n doubao: 'https://ark.cn-beijing.volces.com/api/v3',\n minimax: 'https://api.minimax.chat/v1',\n ollama: 'http://localhost:11434/v1',\n} as const;\n\nexport type OpenAICompatiblePreset = keyof typeof OPENAI_COMPATIBLE_PRESETS;\n\nexport interface PresetEmbedderOptions\n extends Omit<OpenAIEmbedderOptions, 'baseUrl'> {\n /** Pick a known provider; sets `baseUrl` automatically. */\n preset?: OpenAICompatiblePreset;\n /** Explicit override; takes precedence over `preset`. */\n baseUrl?: string;\n}\n\n/**\n * Helper: pick a provider by preset name. Equivalent to constructing\n * `OpenAIEmbedder` with the matching `baseUrl`.\n *\n * @example\n * createOpenAIEmbedder({ preset: 'dashscope', apiKey, model: 'text-embedding-v3' })\n */\nexport function createOpenAIEmbedder(opts: PresetEmbedderOptions): OpenAIEmbedder {\n const baseUrl =\n opts.baseUrl ??\n (opts.preset ? OPENAI_COMPATIBLE_PRESETS[opts.preset] : undefined);\n return new OpenAIEmbedder({ ...opts, baseUrl, id: opts.id ?? opts.preset ?? 'openai' });\n}\n\nexport default OpenAIEmbedder;\n"],"mappings":"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","names":[]}
package/dist/index.mjs CHANGED
@@ -1,4 +1,5 @@
1
1
  // src/index.ts
2
+ import { resilientFetch } from "@objectstack/spec/shared";
2
3
  var KNOWN_DIMENSIONS = {
3
4
  // OpenAI
4
5
  "text-embedding-3-small": 1536,
@@ -47,7 +48,7 @@ var OpenAIEmbedder = class {
47
48
  if (texts.length === 0) return [];
48
49
  const body = { model: this.model, input: texts };
49
50
  if (this.requestedDims) body.dimensions = this.requestedDims;
50
- const res = await this.fetchImpl(`${this.baseUrl}/embeddings`, {
51
+ const res = await resilientFetch(`${this.baseUrl}/embeddings`, {
51
52
  method: "POST",
52
53
  headers: {
53
54
  "content-type": "application/json",
@@ -55,7 +56,7 @@ var OpenAIEmbedder = class {
55
56
  ...this.extraHeaders
56
57
  },
57
58
  body: JSON.stringify(body)
58
- });
59
+ }, { fetchImpl: this.fetchImpl });
59
60
  if (!res.ok) {
60
61
  const text = await res.text().catch(() => "");
61
62
  throw new Error(
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2026 ObjectStack. Licensed under the Apache-2.0 license.\n\n/**\n * `@objectstack/embedder-openai`\n *\n * OpenAI-compatible embedder. Drop-in for any endpoint that speaks the\n * `POST /v1/embeddings` shape:\n *\n * - OpenAI https://api.openai.com/v1\n * - Azure OpenAI https://{resource}.openai.azure.com/openai/deployments/{deployment}\n * - 阿里通义 DashScope https://dashscope.aliyuncs.com/compatible-mode/v1\n * - 智谱 BigModel https://open.bigmodel.cn/api/paas/v4\n * - 硅基流动 SiliconFlow https://api.siliconflow.cn/v1\n * - 火山引擎 Doubao https://ark.cn-beijing.volces.com/api/v3\n * - MiniMax https://api.minimax.chat/v1\n * - Ollama (openai shim) http://localhost:11434/v1\n * - LiteLLM / vLLM / 任何兼容服务\n *\n * Implements the `IEmbedder` contract from `@objectstack/spec/contracts`.\n */\n\nimport type { IEmbedder } from '@objectstack/spec/contracts';\n\nexport interface OpenAIEmbedderOptions {\n /** Bearer token sent as `Authorization: Bearer <apiKey>`. Required. */\n apiKey: string;\n /**\n * Model id sent in the request body. Choose to match your provider:\n * - OpenAI: `'text-embedding-3-small'` (default), `'text-embedding-3-large'`\n * - 阿里通义: `'text-embedding-v3'`\n * - 智谱: `'embedding-3'`\n * - 硅基流动: `'BAAI/bge-m3'`, `'BAAI/bge-large-zh-v1.5'`\n * - 火山 Doubao: `'doubao-embedding-large-text-240915'`\n * - Ollama: `'bge-m3'`, `'nomic-embed-text'`\n *\n * @default 'text-embedding-3-small'\n */\n model?: string;\n /**\n * Override dimensions. Only Matryoshka-style models (OpenAI v3, 智谱 embedding-3,\n * BGE-m3 dense) support truncation. When set, also forwarded to the upstream\n * `dimensions` body field for providers that honour it.\n */\n dimensions?: number;\n /**\n * Endpoint base URL (without `/embeddings`). Defaults to OpenAI's. Set this\n * to point at any compatible provider.\n *\n * @default 'https://api.openai.com/v1'\n */\n baseUrl?: string;\n /** Stable id surfaced as `IEmbedder.id`. @default 'openai' */\n id?: string;\n /** Inject for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n /** Additional headers (e.g. provider-specific keys, tracing). */\n headers?: Record<string, string>;\n}\n\n/**\n * Known dimensions for popular models. Used as the default when the\n * caller doesn't pass `dimensions` explicitly.\n */\nconst KNOWN_DIMENSIONS: Record<string, number> = {\n // OpenAI\n 'text-embedding-3-small': 1536,\n 'text-embedding-3-large': 3072,\n 'text-embedding-ada-002': 1536,\n // 阿里通义\n 'text-embedding-v3': 1024,\n 'text-embedding-v2': 1536,\n 'text-embedding-v1': 1536,\n // 智谱\n 'embedding-3': 2048,\n 'embedding-2': 1024,\n // 硅基流动 / BGE 家族\n 'BAAI/bge-m3': 1024,\n 'BAAI/bge-large-zh-v1.5': 1024,\n 'BAAI/bge-large-en-v1.5': 1024,\n 'BAAI/bge-base-zh-v1.5': 768,\n 'BAAI/bge-small-zh-v1.5': 512,\n 'bge-m3': 1024,\n // 火山 Doubao\n 'doubao-embedding-large-text-240915': 4096,\n 'doubao-embedding-text-240715': 2048,\n // Nomic / Ollama defaults\n 'nomic-embed-text': 768,\n // MiniMax\n 'embo-01': 1536,\n};\n\n/**\n * `OpenAIEmbedder` — OpenAI-compatible embedder. One instance per\n * upstream provider + model combination. Pass into any knowledge\n * adapter that expects `IEmbedder`.\n *\n * @example\n * // OpenAI\n * new OpenAIEmbedder({ apiKey: process.env.OPENAI_API_KEY! });\n *\n * @example\n * // 阿里通义 DashScope\n * new OpenAIEmbedder({\n * apiKey: process.env.DASHSCOPE_API_KEY!,\n * baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n * model: 'text-embedding-v3',\n * });\n *\n * @example\n * // 硅基流动 SiliconFlow + BGE-m3\n * new OpenAIEmbedder({\n * apiKey: process.env.SILICONFLOW_API_KEY!,\n * baseUrl: 'https://api.siliconflow.cn/v1',\n * model: 'BAAI/bge-m3',\n * });\n *\n * @example\n * // Local Ollama\n * new OpenAIEmbedder({\n * apiKey: 'ollama',\n * baseUrl: 'http://localhost:11434/v1',\n * model: 'bge-m3',\n * });\n */\nexport class OpenAIEmbedder implements IEmbedder {\n readonly id: string;\n readonly dimensions: number;\n private readonly model: string;\n private readonly baseUrl: string;\n private readonly apiKey: string;\n private readonly fetchImpl: typeof fetch;\n private readonly requestedDims?: number;\n private readonly extraHeaders: Record<string, string>;\n\n constructor(opts: OpenAIEmbedderOptions) {\n if (!opts.apiKey) throw new Error('OpenAIEmbedder: apiKey required');\n this.apiKey = opts.apiKey;\n this.id = opts.id ?? 'openai';\n this.model = opts.model ?? 'text-embedding-3-small';\n this.baseUrl = (opts.baseUrl ?? 'https://api.openai.com/v1').replace(/\\/+$/, '');\n this.fetchImpl = opts.fetch ?? (globalThis.fetch as typeof fetch);\n this.requestedDims = opts.dimensions;\n this.extraHeaders = opts.headers ?? {};\n this.dimensions =\n opts.dimensions ?? KNOWN_DIMENSIONS[this.model] ?? 1536;\n if (!this.fetchImpl) {\n throw new Error(\n 'OpenAIEmbedder: no fetch available; pass options.fetch or run on Node 18+ / a fetch-capable runtime',\n );\n }\n }\n\n async embed(texts: string[]): Promise<number[][]> {\n if (texts.length === 0) return [];\n const body: Record<string, unknown> = { model: this.model, input: texts };\n if (this.requestedDims) body.dimensions = this.requestedDims;\n const res = await this.fetchImpl(`${this.baseUrl}/embeddings`, {\n method: 'POST',\n headers: {\n 'content-type': 'application/json',\n authorization: `Bearer ${this.apiKey}`,\n ...this.extraHeaders,\n },\n body: JSON.stringify(body),\n });\n if (!res.ok) {\n const text = await res.text().catch(() => '');\n throw new Error(\n `OpenAIEmbedder (${this.baseUrl}) → ${res.status} ${res.statusText}${text ? `: ${text.slice(0, 200)}` : ''}`,\n );\n }\n const json = (await res.json()) as { data?: Array<{ embedding: number[] }> };\n const data = json.data ?? [];\n if (data.length !== texts.length) {\n throw new Error(\n `OpenAIEmbedder: expected ${texts.length} vectors, got ${data.length}`,\n );\n }\n return data.map((d) => d.embedding);\n }\n}\n\n/**\n * Convenience presets for popular Chinese providers — saves callers\n * from memorising base URLs. Pass through `createXxxEmbedder({...})`.\n */\nexport const OPENAI_COMPATIBLE_PRESETS = {\n openai: 'https://api.openai.com/v1',\n azure: '', // user must provide full deployment URL via baseUrl\n dashscope: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n zhipu: 'https://open.bigmodel.cn/api/paas/v4',\n siliconflow: 'https://api.siliconflow.cn/v1',\n doubao: 'https://ark.cn-beijing.volces.com/api/v3',\n minimax: 'https://api.minimax.chat/v1',\n ollama: 'http://localhost:11434/v1',\n} as const;\n\nexport type OpenAICompatiblePreset = keyof typeof OPENAI_COMPATIBLE_PRESETS;\n\nexport interface PresetEmbedderOptions\n extends Omit<OpenAIEmbedderOptions, 'baseUrl'> {\n /** Pick a known provider; sets `baseUrl` automatically. */\n preset?: OpenAICompatiblePreset;\n /** Explicit override; takes precedence over `preset`. */\n baseUrl?: string;\n}\n\n/**\n * Helper: pick a provider by preset name. Equivalent to constructing\n * `OpenAIEmbedder` with the matching `baseUrl`.\n *\n * @example\n * createOpenAIEmbedder({ preset: 'dashscope', apiKey, model: 'text-embedding-v3' })\n */\nexport function createOpenAIEmbedder(opts: PresetEmbedderOptions): OpenAIEmbedder {\n const baseUrl =\n opts.baseUrl ??\n (opts.preset ? OPENAI_COMPATIBLE_PRESETS[opts.preset] : undefined);\n return new OpenAIEmbedder({ ...opts, baseUrl, id: opts.id ?? opts.preset ?? 'openai' });\n}\n\nexport default OpenAIEmbedder;\n"],"mappings":";AA+DA,IAAM,mBAA2C;AAAA;AAAA,EAE/C,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA;AAAA,EAE1B,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA;AAAA,EAErB,eAAe;AAAA,EACf,eAAe;AAAA;AAAA,EAEf,eAAe;AAAA,EACf,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,0BAA0B;AAAA,EAC1B,UAAU;AAAA;AAAA,EAEV,sCAAsC;AAAA,EACtC,gCAAgC;AAAA;AAAA,EAEhC,oBAAoB;AAAA;AAAA,EAEpB,WAAW;AACb;AAmCO,IAAM,iBAAN,MAA0C;AAAA,EAU/C,YAAY,MAA6B;AACvC,QAAI,CAAC,KAAK,OAAQ,OAAM,IAAI,MAAM,iCAAiC;AACnE,SAAK,SAAS,KAAK;AACnB,SAAK,KAAK,KAAK,MAAM;AACrB,SAAK,QAAQ,KAAK,SAAS;AAC3B,SAAK,WAAW,KAAK,WAAW,6BAA6B,QAAQ,QAAQ,EAAE;AAC/E,SAAK,YAAY,KAAK,SAAU,WAAW;AAC3C,SAAK,gBAAgB,KAAK;AAC1B,SAAK,eAAe,KAAK,WAAW,CAAC;AACrC,SAAK,aACH,KAAK,cAAc,iBAAiB,KAAK,KAAK,KAAK;AACrD,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,IAAI;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,MAAM,OAAsC;AAChD,QAAI,MAAM,WAAW,EAAG,QAAO,CAAC;AAChC,UAAM,OAAgC,EAAE,OAAO,KAAK,OAAO,OAAO,MAAM;AACxE,QAAI,KAAK,cAAe,MAAK,aAAa,KAAK;AAC/C,UAAM,MAAM,MAAM,KAAK,UAAU,GAAG,KAAK,OAAO,eAAe;AAAA,MAC7D,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,eAAe,UAAU,KAAK,MAAM;AAAA,QACpC,GAAG,KAAK;AAAA,MACV;AAAA,MACA,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B,CAAC;AACD,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,OAAO,MAAM,IAAI,KAAK,EAAE,MAAM,MAAM,EAAE;AAC5C,YAAM,IAAI;AAAA,QACR,mBAAmB,KAAK,OAAO,YAAO,IAAI,MAAM,IAAI,IAAI,UAAU,GAAG,OAAO,KAAK,KAAK,MAAM,GAAG,GAAG,CAAC,KAAK,EAAE;AAAA,MAC5G;AAAA,IACF;AACA,UAAM,OAAQ,MAAM,IAAI,KAAK;AAC7B,UAAM,OAAO,KAAK,QAAQ,CAAC;AAC3B,QAAI,KAAK,WAAW,MAAM,QAAQ;AAChC,YAAM,IAAI;AAAA,QACR,4BAA4B,MAAM,MAAM,iBAAiB,KAAK,MAAM;AAAA,MACtE;AAAA,IACF;AACA,WAAO,KAAK,IAAI,CAAC,MAAM,EAAE,SAAS;AAAA,EACpC;AACF;AAMO,IAAM,4BAA4B;AAAA,EACvC,QAAQ;AAAA,EACR,OAAO;AAAA;AAAA,EACP,WAAW;AAAA,EACX,OAAO;AAAA,EACP,aAAa;AAAA,EACb,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,QAAQ;AACV;AAmBO,SAAS,qBAAqB,MAA6C;AAChF,QAAM,UACJ,KAAK,YACJ,KAAK,SAAS,0BAA0B,KAAK,MAAM,IAAI;AAC1D,SAAO,IAAI,eAAe,EAAE,GAAG,MAAM,SAAS,IAAI,KAAK,MAAM,KAAK,UAAU,SAAS,CAAC;AACxF;AAEA,IAAO,gBAAQ;","names":[]}
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["// Copyright (c) 2026 ObjectStack. Licensed under the Apache-2.0 license.\n\n/**\n * `@objectstack/embedder-openai`\n *\n * OpenAI-compatible embedder. Drop-in for any endpoint that speaks the\n * `POST /v1/embeddings` shape:\n *\n * - OpenAI https://api.openai.com/v1\n * - Azure OpenAI https://{resource}.openai.azure.com/openai/deployments/{deployment}\n * - 阿里通义 DashScope https://dashscope.aliyuncs.com/compatible-mode/v1\n * - 智谱 BigModel https://open.bigmodel.cn/api/paas/v4\n * - 硅基流动 SiliconFlow https://api.siliconflow.cn/v1\n * - 火山引擎 Doubao https://ark.cn-beijing.volces.com/api/v3\n * - MiniMax https://api.minimax.chat/v1\n * - Ollama (openai shim) http://localhost:11434/v1\n * - LiteLLM / vLLM / 任何兼容服务\n *\n * Implements the `IEmbedder` contract from `@objectstack/spec/contracts`.\n */\n\nimport type { IEmbedder } from '@objectstack/spec/contracts';\nimport { resilientFetch } from '@objectstack/spec/shared';\n\nexport interface OpenAIEmbedderOptions {\n /** Bearer token sent as `Authorization: Bearer <apiKey>`. Required. */\n apiKey: string;\n /**\n * Model id sent in the request body. Choose to match your provider:\n * - OpenAI: `'text-embedding-3-small'` (default), `'text-embedding-3-large'`\n * - 阿里通义: `'text-embedding-v3'`\n * - 智谱: `'embedding-3'`\n * - 硅基流动: `'BAAI/bge-m3'`, `'BAAI/bge-large-zh-v1.5'`\n * - 火山 Doubao: `'doubao-embedding-large-text-240915'`\n * - Ollama: `'bge-m3'`, `'nomic-embed-text'`\n *\n * @default 'text-embedding-3-small'\n */\n model?: string;\n /**\n * Override dimensions. Only Matryoshka-style models (OpenAI v3, 智谱 embedding-3,\n * BGE-m3 dense) support truncation. When set, also forwarded to the upstream\n * `dimensions` body field for providers that honour it.\n */\n dimensions?: number;\n /**\n * Endpoint base URL (without `/embeddings`). Defaults to OpenAI's. Set this\n * to point at any compatible provider.\n *\n * @default 'https://api.openai.com/v1'\n */\n baseUrl?: string;\n /** Stable id surfaced as `IEmbedder.id`. @default 'openai' */\n id?: string;\n /** Inject for tests. Defaults to `globalThis.fetch`. */\n fetch?: typeof fetch;\n /** Additional headers (e.g. provider-specific keys, tracing). */\n headers?: Record<string, string>;\n}\n\n/**\n * Known dimensions for popular models. Used as the default when the\n * caller doesn't pass `dimensions` explicitly.\n */\nconst KNOWN_DIMENSIONS: Record<string, number> = {\n // OpenAI\n 'text-embedding-3-small': 1536,\n 'text-embedding-3-large': 3072,\n 'text-embedding-ada-002': 1536,\n // 阿里通义\n 'text-embedding-v3': 1024,\n 'text-embedding-v2': 1536,\n 'text-embedding-v1': 1536,\n // 智谱\n 'embedding-3': 2048,\n 'embedding-2': 1024,\n // 硅基流动 / BGE 家族\n 'BAAI/bge-m3': 1024,\n 'BAAI/bge-large-zh-v1.5': 1024,\n 'BAAI/bge-large-en-v1.5': 1024,\n 'BAAI/bge-base-zh-v1.5': 768,\n 'BAAI/bge-small-zh-v1.5': 512,\n 'bge-m3': 1024,\n // 火山 Doubao\n 'doubao-embedding-large-text-240915': 4096,\n 'doubao-embedding-text-240715': 2048,\n // Nomic / Ollama defaults\n 'nomic-embed-text': 768,\n // MiniMax\n 'embo-01': 1536,\n};\n\n/**\n * `OpenAIEmbedder` — OpenAI-compatible embedder. One instance per\n * upstream provider + model combination. Pass into any knowledge\n * adapter that expects `IEmbedder`.\n *\n * @example\n * // OpenAI\n * new OpenAIEmbedder({ apiKey: process.env.OPENAI_API_KEY! });\n *\n * @example\n * // 阿里通义 DashScope\n * new OpenAIEmbedder({\n * apiKey: process.env.DASHSCOPE_API_KEY!,\n * baseUrl: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n * model: 'text-embedding-v3',\n * });\n *\n * @example\n * // 硅基流动 SiliconFlow + BGE-m3\n * new OpenAIEmbedder({\n * apiKey: process.env.SILICONFLOW_API_KEY!,\n * baseUrl: 'https://api.siliconflow.cn/v1',\n * model: 'BAAI/bge-m3',\n * });\n *\n * @example\n * // Local Ollama\n * new OpenAIEmbedder({\n * apiKey: 'ollama',\n * baseUrl: 'http://localhost:11434/v1',\n * model: 'bge-m3',\n * });\n */\nexport class OpenAIEmbedder implements IEmbedder {\n readonly id: string;\n readonly dimensions: number;\n private readonly model: string;\n private readonly baseUrl: string;\n private readonly apiKey: string;\n private readonly fetchImpl: typeof fetch;\n private readonly requestedDims?: number;\n private readonly extraHeaders: Record<string, string>;\n\n constructor(opts: OpenAIEmbedderOptions) {\n if (!opts.apiKey) throw new Error('OpenAIEmbedder: apiKey required');\n this.apiKey = opts.apiKey;\n this.id = opts.id ?? 'openai';\n this.model = opts.model ?? 'text-embedding-3-small';\n this.baseUrl = (opts.baseUrl ?? 'https://api.openai.com/v1').replace(/\\/+$/, '');\n this.fetchImpl = opts.fetch ?? (globalThis.fetch as typeof fetch);\n this.requestedDims = opts.dimensions;\n this.extraHeaders = opts.headers ?? {};\n this.dimensions =\n opts.dimensions ?? KNOWN_DIMENSIONS[this.model] ?? 1536;\n if (!this.fetchImpl) {\n throw new Error(\n 'OpenAIEmbedder: no fetch available; pass options.fetch or run on Node 18+ / a fetch-capable runtime',\n );\n }\n }\n\n async embed(texts: string[]): Promise<number[][]> {\n if (texts.length === 0) return [];\n const body: Record<string, unknown> = { model: this.model, input: texts };\n if (this.requestedDims) body.dimensions = this.requestedDims;\n const res = await resilientFetch(`${this.baseUrl}/embeddings`, {\n method: 'POST',\n headers: {\n 'content-type': 'application/json',\n authorization: `Bearer ${this.apiKey}`,\n ...this.extraHeaders,\n },\n body: JSON.stringify(body),\n }, { fetchImpl: this.fetchImpl });\n if (!res.ok) {\n const text = await res.text().catch(() => '');\n throw new Error(\n `OpenAIEmbedder (${this.baseUrl}) → ${res.status} ${res.statusText}${text ? `: ${text.slice(0, 200)}` : ''}`,\n );\n }\n const json = (await res.json()) as { data?: Array<{ embedding: number[] }> };\n const data = json.data ?? [];\n if (data.length !== texts.length) {\n throw new Error(\n `OpenAIEmbedder: expected ${texts.length} vectors, got ${data.length}`,\n );\n }\n return data.map((d) => d.embedding);\n }\n}\n\n/**\n * Convenience presets for popular Chinese providers — saves callers\n * from memorising base URLs. Pass through `createXxxEmbedder({...})`.\n */\nexport const OPENAI_COMPATIBLE_PRESETS = {\n openai: 'https://api.openai.com/v1',\n azure: '', // user must provide full deployment URL via baseUrl\n dashscope: 'https://dashscope.aliyuncs.com/compatible-mode/v1',\n zhipu: 'https://open.bigmodel.cn/api/paas/v4',\n siliconflow: 'https://api.siliconflow.cn/v1',\n doubao: 'https://ark.cn-beijing.volces.com/api/v3',\n minimax: 'https://api.minimax.chat/v1',\n ollama: 'http://localhost:11434/v1',\n} as const;\n\nexport type OpenAICompatiblePreset = keyof typeof OPENAI_COMPATIBLE_PRESETS;\n\nexport interface PresetEmbedderOptions\n extends Omit<OpenAIEmbedderOptions, 'baseUrl'> {\n /** Pick a known provider; sets `baseUrl` automatically. */\n preset?: OpenAICompatiblePreset;\n /** Explicit override; takes precedence over `preset`. */\n baseUrl?: string;\n}\n\n/**\n * Helper: pick a provider by preset name. Equivalent to constructing\n * `OpenAIEmbedder` with the matching `baseUrl`.\n *\n * @example\n * createOpenAIEmbedder({ preset: 'dashscope', apiKey, model: 'text-embedding-v3' })\n */\nexport function createOpenAIEmbedder(opts: PresetEmbedderOptions): OpenAIEmbedder {\n const baseUrl =\n opts.baseUrl ??\n (opts.preset ? OPENAI_COMPATIBLE_PRESETS[opts.preset] : undefined);\n return new OpenAIEmbedder({ ...opts, baseUrl, id: opts.id ?? opts.preset ?? 'openai' });\n}\n\nexport default OpenAIEmbedder;\n"],"mappings":";AAsBA,SAAS,sBAAsB;AA0C/B,IAAM,mBAA2C;AAAA;AAAA,EAE/C,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA;AAAA,EAE1B,qBAAqB;AAAA,EACrB,qBAAqB;AAAA,EACrB,qBAAqB;AAAA;AAAA,EAErB,eAAe;AAAA,EACf,eAAe;AAAA;AAAA,EAEf,eAAe;AAAA,EACf,0BAA0B;AAAA,EAC1B,0BAA0B;AAAA,EAC1B,yBAAyB;AAAA,EACzB,0BAA0B;AAAA,EAC1B,UAAU;AAAA;AAAA,EAEV,sCAAsC;AAAA,EACtC,gCAAgC;AAAA;AAAA,EAEhC,oBAAoB;AAAA;AAAA,EAEpB,WAAW;AACb;AAmCO,IAAM,iBAAN,MAA0C;AAAA,EAU/C,YAAY,MAA6B;AACvC,QAAI,CAAC,KAAK,OAAQ,OAAM,IAAI,MAAM,iCAAiC;AACnE,SAAK,SAAS,KAAK;AACnB,SAAK,KAAK,KAAK,MAAM;AACrB,SAAK,QAAQ,KAAK,SAAS;AAC3B,SAAK,WAAW,KAAK,WAAW,6BAA6B,QAAQ,QAAQ,EAAE;AAC/E,SAAK,YAAY,KAAK,SAAU,WAAW;AAC3C,SAAK,gBAAgB,KAAK;AAC1B,SAAK,eAAe,KAAK,WAAW,CAAC;AACrC,SAAK,aACH,KAAK,cAAc,iBAAiB,KAAK,KAAK,KAAK;AACrD,QAAI,CAAC,KAAK,WAAW;AACnB,YAAM,IAAI;AAAA,QACR;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAM,MAAM,OAAsC;AAChD,QAAI,MAAM,WAAW,EAAG,QAAO,CAAC;AAChC,UAAM,OAAgC,EAAE,OAAO,KAAK,OAAO,OAAO,MAAM;AACxE,QAAI,KAAK,cAAe,MAAK,aAAa,KAAK;AAC/C,UAAM,MAAM,MAAM,eAAe,GAAG,KAAK,OAAO,eAAe;AAAA,MAC7D,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,eAAe,UAAU,KAAK,MAAM;AAAA,QACpC,GAAG,KAAK;AAAA,MACV;AAAA,MACA,MAAM,KAAK,UAAU,IAAI;AAAA,IAC3B,GAAG,EAAE,WAAW,KAAK,UAAU,CAAC;AAChC,QAAI,CAAC,IAAI,IAAI;AACX,YAAM,OAAO,MAAM,IAAI,KAAK,EAAE,MAAM,MAAM,EAAE;AAC5C,YAAM,IAAI;AAAA,QACR,mBAAmB,KAAK,OAAO,YAAO,IAAI,MAAM,IAAI,IAAI,UAAU,GAAG,OAAO,KAAK,KAAK,MAAM,GAAG,GAAG,CAAC,KAAK,EAAE;AAAA,MAC5G;AAAA,IACF;AACA,UAAM,OAAQ,MAAM,IAAI,KAAK;AAC7B,UAAM,OAAO,KAAK,QAAQ,CAAC;AAC3B,QAAI,KAAK,WAAW,MAAM,QAAQ;AAChC,YAAM,IAAI;AAAA,QACR,4BAA4B,MAAM,MAAM,iBAAiB,KAAK,MAAM;AAAA,MACtE;AAAA,IACF;AACA,WAAO,KAAK,IAAI,CAAC,MAAM,EAAE,SAAS;AAAA,EACpC;AACF;AAMO,IAAM,4BAA4B;AAAA,EACvC,QAAQ;AAAA,EACR,OAAO;AAAA;AAAA,EACP,WAAW;AAAA,EACX,OAAO;AAAA,EACP,aAAa;AAAA,EACb,QAAQ;AAAA,EACR,SAAS;AAAA,EACT,QAAQ;AACV;AAmBO,SAAS,qBAAqB,MAA6C;AAChF,QAAM,UACJ,KAAK,YACJ,KAAK,SAAS,0BAA0B,KAAK,MAAM,IAAI;AAC1D,SAAO,IAAI,eAAe,EAAE,GAAG,MAAM,SAAS,IAAI,KAAK,MAAM,KAAK,UAAU,SAAS,CAAC;AACxF;AAEA,IAAO,gBAAQ;","names":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@objectstack/embedder-openai",
3
- "version": "7.8.0",
3
+ "version": "8.0.0",
4
4
  "license": "Apache-2.0",
5
5
  "description": "OpenAI-compatible embedder for ObjectStack — works against OpenAI, 阿里通义 DashScope, 智谱 BigModel, 硅基流动 SiliconFlow, 火山引擎 Doubao, MiniMax, Ollama, and any drop-in OpenAI-shape endpoint.",
6
6
  "main": "dist/index.js",
@@ -13,7 +13,7 @@
13
13
  }
14
14
  },
15
15
  "dependencies": {
16
- "@objectstack/spec": "7.8.0"
16
+ "@objectstack/spec": "8.0.0"
17
17
  },
18
18
  "devDependencies": {
19
19
  "@types/node": "^25.9.1",
package/src/index.ts CHANGED
@@ -20,6 +20,7 @@
20
20
  */
21
21
 
22
22
  import type { IEmbedder } from '@objectstack/spec/contracts';
23
+ import { resilientFetch } from '@objectstack/spec/shared';
23
24
 
24
25
  export interface OpenAIEmbedderOptions {
25
26
  /** Bearer token sent as `Authorization: Bearer <apiKey>`. Required. */
@@ -154,7 +155,7 @@ export class OpenAIEmbedder implements IEmbedder {
154
155
  if (texts.length === 0) return [];
155
156
  const body: Record<string, unknown> = { model: this.model, input: texts };
156
157
  if (this.requestedDims) body.dimensions = this.requestedDims;
157
- const res = await this.fetchImpl(`${this.baseUrl}/embeddings`, {
158
+ const res = await resilientFetch(`${this.baseUrl}/embeddings`, {
158
159
  method: 'POST',
159
160
  headers: {
160
161
  'content-type': 'application/json',
@@ -162,7 +163,7 @@ export class OpenAIEmbedder implements IEmbedder {
162
163
  ...this.extraHeaders,
163
164
  },
164
165
  body: JSON.stringify(body),
165
- });
166
+ }, { fetchImpl: this.fetchImpl });
166
167
  if (!res.ok) {
167
168
  const text = await res.text().catch(() => '');
168
169
  throw new Error(
package/vitest.config.ts CHANGED
@@ -11,6 +11,7 @@ export default defineConfig({
11
11
  resolve: {
12
12
  alias: {
13
13
  '@objectstack/spec/contracts': path.resolve(__dirname, '../../spec/src/contracts/index.ts'),
14
+ '@objectstack/spec/shared': path.resolve(__dirname, '../../spec/src/shared/index.ts'),
14
15
  '@objectstack/spec': path.resolve(__dirname, '../../spec/src/index.ts'),
15
16
  },
16
17
  },