@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.
- package/.turbo/turbo-build.log +8 -8
- package/CHANGELOG.md +41 -0
- package/dist/index.js +3 -2
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +3 -2
- package/dist/index.mjs.map +1 -1
- package/package.json +2 -2
- package/src/index.ts +3 -2
- package/vitest.config.ts +1 -0
package/.turbo/turbo-build.log
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
|
|
2
|
-
> @objectstack/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
|
[34mCLI[39m Building entry: src/index.ts
|
|
@@ -10,13 +10,13 @@
|
|
|
10
10
|
[34mCLI[39m Cleaning output folder
|
|
11
11
|
[34mESM[39m Build start
|
|
12
12
|
[34mCJS[39m Build start
|
|
13
|
-
[
|
|
14
|
-
[
|
|
15
|
-
[
|
|
16
|
-
[
|
|
17
|
-
[
|
|
18
|
-
[
|
|
13
|
+
[32mESM[39m [1mdist/index.mjs [22m[32m3.25 KB[39m
|
|
14
|
+
[32mESM[39m [1mdist/index.mjs.map [22m[32m9.77 KB[39m
|
|
15
|
+
[32mESM[39m ⚡️ Build success in 48ms
|
|
16
|
+
[32mCJS[39m [1mdist/index.js [22m[32m4.38 KB[39m
|
|
17
|
+
[32mCJS[39m [1mdist/index.js.map [22m[32m9.82 KB[39m
|
|
18
|
+
[32mCJS[39m ⚡️ Build success in 48ms
|
|
19
19
|
[34mDTS[39m Build start
|
|
20
|
-
[32mDTS[39m ⚡️ Build success in
|
|
20
|
+
[32mDTS[39m ⚡️ Build success in 5103ms
|
|
21
21
|
[32mDTS[39m [1mdist/index.d.mts [22m[32m4.86 KB[39m
|
|
22
22
|
[32mDTS[39m [1mdist/index.d.ts [22m[32m4.86 KB[39m
|
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
|
|
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
|
|
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(
|
package/dist/index.mjs.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":";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": "
|
|
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": "
|
|
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
|
|
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
|
},
|