58 lines
2.2 KiB
JavaScript
58 lines
2.2 KiB
JavaScript
import { IterableReadableStream } from "@langchain/core/utils/stream";
|
|
async function* createOllamaStream(url, params, options) {
|
|
let formattedUrl = url;
|
|
if (formattedUrl.startsWith("http://localhost:")) {
|
|
// Node 18 has issues with resolving "localhost"
|
|
// See https://github.com/node-fetch/node-fetch/issues/1624
|
|
formattedUrl = formattedUrl.replace("http://localhost:", "http://127.0.0.1:");
|
|
}
|
|
const response = await fetch(formattedUrl, {
|
|
method: "POST",
|
|
body: JSON.stringify(params),
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
...options.headers,
|
|
},
|
|
signal: options.signal,
|
|
});
|
|
if (!response.ok) {
|
|
let error;
|
|
const responseText = await response.text();
|
|
try {
|
|
const json = JSON.parse(responseText);
|
|
error = new Error(`Ollama call failed with status code ${response.status}: ${json.error}`);
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
}
|
|
catch (e) {
|
|
error = new Error(`Ollama call failed with status code ${response.status}: ${responseText}`);
|
|
}
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
error.response = response;
|
|
throw error;
|
|
}
|
|
if (!response.body) {
|
|
throw new Error("Could not begin Ollama stream. Please check the given URL and try again.");
|
|
}
|
|
const stream = IterableReadableStream.fromReadableStream(response.body);
|
|
const decoder = new TextDecoder();
|
|
let extra = "";
|
|
for await (const chunk of stream) {
|
|
const decoded = extra + decoder.decode(chunk);
|
|
const lines = decoded.split("\n");
|
|
extra = lines.pop() || "";
|
|
for (const line of lines) {
|
|
try {
|
|
yield JSON.parse(line);
|
|
}
|
|
catch (e) {
|
|
console.warn(`Received a non-JSON parseable chunk: ${line}`);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
export async function* createOllamaGenerateStream(baseUrl, params, options) {
|
|
yield* createOllamaStream(`${baseUrl}/api/generate`, params, options);
|
|
}
|
|
export async function* createOllamaChatStream(baseUrl, params, options) {
|
|
yield* createOllamaStream(`${baseUrl}/api/chat`, params, options);
|
|
}
|