」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 使用 fetch 串流 HTTP 回應

使用 fetch 串流 HTTP 回應

發佈於2024-07-31
瀏覽:104

Streaming HTTP Responses using fetch

这篇文章将着眼于使用 JavaScript Streams API,它允许进行 fetch HTTP 调用并以块的形式接收流响应,这允许客户端开始更多地响应服务器响应快速构建像 ChatGPT 这样的 UI。

作为一个激励性的例子,我们将实现一个函数来处理来自 OpenAI(或任何使用相同 http 流 API 的服务器)的流式 LLM 响应,不使用 npm 依赖项,仅使用内置的 fetch。完整的代码在这里,包括指数退避重试、嵌入、非流式聊天以及用于与聊天完成和嵌入交互的更简单的 API。

如果您有兴趣了解如何将 HTTP 流返回给客户端,请查看这篇文章。

完整示例代码

这是完整的示例。我们将看看下面的每一个部分:

async function createChatCompletion(body: ChatCompletionCreateParams) {
  // Making the request
  const baseUrl = process.env.LLM_BASE_URL || "https://api.openai.com";
  const response = await fetch(baseUrl   "/v1/chat/completions", {
    method: "POST",
    headers: {
      "Content-Type": "application/json",
      "Authorization": "Bearer "   process.env.LLM_API_KEY,
    },
    body: JSON.stringify(body),
  });
  // Handling errors
  if (!response.ok) {
    const error = await response.text();
    throw new Error(`Failed (${response.status}): ${error}`,
  }
  if (!body.stream) { // the non-streaming case
    return response.json();
  }
  const stream = response.body;
  if (!stream) throw new Error("No body in response");
  // Returning an async iterator
  return {
    [Symbol.asyncIterator]: async function* () {
      for await (const data of splitStream(stream)) {
        // Handling the OpenAI HTTP streaming protocol
        if (data.startsWith("data:")) {
          const json = data.substring("data:".length).trimStart();
          if (json.startsWith("[DONE]")) {
            return;
          }
          yield JSON.parse(json);
        }
      }
    },
  };
}

// Reading the stream  
async function* splitStream(stream: ReadableStream) {
  const reader = stream.getReader();
  let lastFragment = "";
  try {
    while (true) {
      const { value, done } = await reader.read();
      if (done) {
        // Flush the last fragment now that we're done
        if (lastFragment !== "") {
          yield lastFragment;
        }
        break;
      }
      const data = new TextDecoder().decode(value);
      lastFragment  = data;
      const parts = lastFragment.split("\n\n");
      // Yield all except for the last part
      for (let i = 0; i 



请参阅此处的代码,了解具有流式和非流式参数变体的良好类型重载的版本,以及重试和其他改进。

这篇文章的其余部分是关于理解这段代码的作用。

提出请求

这部分其实很简单。流式 HTTP 响应来自普通 HTTP 请求:

const baseUrl = process.env.LLM_BASE_URL || "https://api.openai.com";
const response = await fetch(baseUrl   "/v1/chat/completions", {
  method: "POST",
  headers: {
    "Content-Type": "application/json",
    "Authorization": "Bearer "   process.env.LLM_API_KEY,
  },
  body: JSON.stringify(body),
});

HTTP 标头按平常方式发送,无需特别设置任何内容即可启用流式传输。您仍然可以利用常规缓存标头进行 HTTP 流式传输。

处理错误

关于客户端错误的故事对于 HTTP 流来说有点不幸。好处是,对于 HTTP 流式传输,客户端会在初始响应中立即获取状态代码,并可以检测到故障。 http 协议的缺点是,如果服务器返回成功,但随后在流中中断,则协议级别没有任何内容可以告诉客户端流已中断。我们将在下面看到 OpenAI 如何在最后编码“全部完成”哨兵来解决这个问题。

if (!response.ok) {
  const error = await response.text();
  throw new Error(`Failed (${response.status}): ${error}`,
}

读取流

为了读取 HTTP 流响应,客户端可以使用 response.body 属性,该属性是一个 ReadableStream,允许您使用 .getReader() 方法迭代从服务器传入的块。 1

const reader = request.body.getReader();
try {
    while (true) {
      const { value, done } = await reader.read();
      if (done) break;
      const text = TextDecoder().decode(value);
      //... do something with the chunk
    }
} finally {
  reader.releaseLock();
}

这会处理我们返回的每一位数据,但对于 OpenAI HTTP 协议,我们期望数据是由换行符分隔的 JSON,因此我们将拆分响应正文并“生成”每一行。重新完成。我们将进行中的行缓冲到lastFragment中,并且只返回由两个换行符分隔的完整行:

// stream here is request.body
async function* splitStream(stream: ReadableStream) {
  const reader = stream.getReader();
  let lastFragment = "";
  try {
    while (true) {
      const { value, done } = await reader.read();
      if (done) {
        // Flush the last fragment now that we're done
        if (lastFragment !== "") {
          yield lastFragment;
        }
        break;
      }
      const data = new TextDecoder().decode(value);
      lastFragment  = data;
      const parts = lastFragment.split("\n\n");
      // Yield all except for the last part
      for (let i = 0; i 



如果你不熟悉这个 function* 和yield 语法,只需将 function* 视为可以在循环中返回多个内容的函数,并将yield视为从函数中多次返回内容的方式。

然后您可以循环此 splitStream 函数,例如:

for await (const data of splitStream(response.body)) {
  // data here is a full line of text. For OpenAI, it might look like
  // "data: {...some json object...}" or "data: [DONE]" at the end
}

如果这个“for wait”语法让您感到困惑,那么它使用了所谓的“异步迭代器”——就像您在 for 循环中使用的常规迭代器一样,但每次它获取下一个值时,都会等待它。

对于我们的示例,当我们从 OpenAI 获取一些文本并且正在等待更多文本时,for 循环将等待,直到 splitStream 产生另一个值,这将在 wait reader.read() 返回一个完成的值时发生一行或多行文本。

接下来我们将研究另一种返回异步迭代器的方法,该迭代器不是 splitStream 等函数,因此调用者可以使用“for wait”循环来迭代此数据。

返回一个异步迭代器

现在我们有一个返回整行文本的异步迭代器,我们可以只返回 splitStream(response.body),但我们希望拦截每一行并转换它们,同时仍然让函数的调用者进行迭代。

该方法类似于上面的 async function* 语法。这里我们将直接返回一个异步迭代器,而不是调用时返回一个的异步函数。不同之处在于类型是 AsyncIterator 而不是需要首先调用的 AsyncGenerator。 AsyncIterator 可以通过某个命名函数来定义:Symbol.asyncIterator.2

      return {
        [Symbol.asyncIterator]: async function* () {
          for await (const data of splitStream(stream)) {
            //handle the data
            yield data;
          }
        },
      };

当您想要返回与来自 splitStream 的数据不同的内容时,这非常有用。每次从流式 HTTP 请求中传入新行时,sp​​litStream 都会生成它,该函数将在数据中接收它,并可以在将其生成给调用者之前执行一些操作。

接下来我们将看看如何在 OpenAI 的流式聊天完成 API 的情况下具体解释这些数据。

处理 OpenAI HTTP 流协议

OpenAI 响应协议是一系列以 data: 或 event: 开头的行,但我们只处理数据响应,因为这是完成聊天的有用部分。如果流完成,则有一个 [DONE] 标记,否则它只是 JSON。

for await (const data of splitStream(stream)) {
  if (data.startsWith("data:")) {
    const json = data.substring("data:".length).trimStart();
    if (json.startsWith("[DONE]")) {
      return;
    }
    yield JSON.parse(json);
  } else {
    console.debug("Unexpected data:", data);
  }
}

将所有内容整合在一起

既然您了解了 HTTP 流,您就可以放心地直接使用流 API,而无需依赖 sdk 或库。这使您可以隐藏延迟,因为您的 UI 可以立即开始更新,而不会因为多个请求而消耗更多带宽。您可以像使用官方 openai npm 包一样使用上述功能:

  const response = await createChatCompletion({
    model: "llama3",
    messages: [...your messages...],
    stream: true,
  });
  for await (const chunk of response) {
    if (chunk.choices[0].delta?.content) {
      console.log(chunk.choices[0].delta.content);
    }
  }

请参阅此处的代码,它还允许您创建一些实用函数,通过预先配置模型并提取 .choices[0].delta.content:
使这变得更加容易

const response = await chatStream(messages);
for await (const content of response) {
  console.log(content);
}

在复制代码之前,尝试自己实现它作为异步函数的练习。

更多资源

  • 有关从您自己的服务器端点返回 HTTP 流数据的信息,请查看关于 AI Chat with HTTP Streaming 的这篇文章,该文章既将数据从 OpenAI(或类似的)流式传输到您的服务器,又同时将其流式传输到客户端,同时执行自定义逻辑(例如将块保存到数据库)。
  • MDN 文档一如既往地很棒。除了上面的链接之外,这里还有关于可读流 API 的指南,它展示了如何将可读流连接到 使用 fetch 串流 HTTP 回應 标签以在图像请求中进行流式传输。注意:本指南使用 response.body 作为异步迭代器,但目前尚未广泛实现,并且不在 TypeScript 类型中。

  1. 注意:一次只能有一个流的读取器,因此您通常不会多次调用 .getReader() - 在这种情况下您可能需要 .tee() ,并且如果您想使用 .由于某种原因多次 getReader() ,请确保首先拥有第一个 .releaseLock() 。 ↩

  2. 或者,如果您不熟悉 Symbol,它的用途是在对象中包含非字符串或数字的键。这样,如果您添加了名为 asyncIterator 的键,它们就不会发生冲突。您可以使用 myIterator[Symbol.asyncIterator]() 访问该函数。 ↩

版本聲明 本文轉載於:https://dev.to/ianmacartney/streaming-http-responses-using-fetch-1fm2?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 在 Go 中使用 WebSocket 進行即時通信
    在 Go 中使用 WebSocket 進行即時通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要一种比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSoc...
    程式設計 發佈於2024-12-23
  • 如何在同一目錄中組織一個庫和 CLI 的 Go 專案?
    如何在同一目錄中組織一個庫和 CLI 的 Go 專案?
    在多包專案中組織程式碼在同時需要庫和命令列介面(CLI) 的Go 專案中,經常會遇到以下問題在同一目錄中有多個包。 這樣的專案架構:whatever.io/ myproject/ main.go myproject.go套件 main 和 func main ...
    程式設計 發佈於2024-12-23
  • 如何在 Android 中選擇後保持 ListView 項目突出顯示?
    如何在 Android 中選擇後保持 ListView 項目突出顯示?
    如何在Android 中選擇後保持ListView 項目突出顯示在Android 中,維護ListView 項目的選定狀態可以透過提供以下功能來增強使用者體驗:目前選擇的清晰視覺指示器。然而,有時開發人員會遇到這樣的問題:所選項目在某些事件(例如捲動或與 ListView 進一步互動)後失去突出顯示...
    程式設計 發佈於2024-12-23
  • 如何使用自訂 CSS 在 Bootstrap 3 中建立全高列?
    如何使用自訂 CSS 在 Bootstrap 3 中建立全高列?
    Bootstrap 3 全高列:自訂CSS 解決方案簡介:創建Twitter Bootstrap 3 的全高佈局可能具有挑戰性。雖然Bootstrap的原生類別不支援此功能,但可以使用自訂CSS來實現此效果。 自訂CSS方法:設定100% 高度:將body、container 和row 元素的高度設...
    程式設計 發佈於2024-12-23
  • 如何在不使用連結的情況下為 Span 元素添加工具提示?
    如何在不使用連結的情況下為 Span 元素添加工具提示?
    向不帶連結的Span 元素添加工具提示將滑鼠懸停在span 元素上時,通常需要向用戶提供附加資訊.這可以使用工具提示來實現,而不依賴連結。 解決方案:要使用內建HTML 屬性將工具提示新增至span 元素,只需如下使用title 屬性:<span title="My tip"...
    程式設計 發佈於2024-12-23
  • 為什麼我的 WebSocket 伺服器在 Docker 化後無法連線?
    為什麼我的 WebSocket 伺服器在 Docker 化後無法連線?
    Docker化 WebSocket 伺服器問題Docker化 WebSocket 伺服器問題開發人員在嘗試使用 Docker 容器化 WebSocket 伺服器時遇到問題。伺服器程式碼使用「connected」寫入新連接,並且在容器外運行良好,但當放置在Docker 容器內時,客戶端會因「連接重設」...
    程式設計 發佈於2024-12-23
  • Python中如何匯入同目錄或子目錄中的類別?
    Python中如何匯入同目錄或子目錄中的類別?
    在Python中從同一目錄或子目錄匯入類別在Python中,您可以透過下列方式從同一目錄或子目錄中的檔案導入類別利用__init__.py 檔案。該檔案是一個空佔位符,指示該目錄包含模組和套件。 從同一目錄匯入從與 main 相同的目錄中的檔案匯入類別.py,在該目錄中建立一個 __init__.p...
    程式設計 發佈於2024-12-23
  • 為什麼C90中函數名可以當函數指標?
    為什麼C90中函數名可以當函數指標?
    使用函數名稱作為函數指標C90 的基本原理文件深入了解了將函數名稱與函數指標等同的設計選擇。這種便利性簡化了在特定上下文中使用函數指標的過程。 函數宣告考慮宣告:int f(); int (*pf)();函數呼叫以下所有表示有效的函數呼叫:(&f)(); f(); (*f)(); (**f)...
    程式設計 發佈於2024-12-23
  • 如何在 Python 中使用多個單字邊界分隔符號將字串拆分為單字?
    如何在 Python 中使用多個單字邊界分隔符號將字串拆分為單字?
    使用多個單字邊界定界符將字串拆分為單字處理文字資料時,常見的任務是將字串拆分為單字。 Python 的 str.split() 方法提供了一個簡單的解決方案,但它僅支援單一分隔符號作為其參數。在處理包含多種類型的單字邊界(例如標點符號)的文字時,此限制可能會成為障礙。 Python re 模組提供了...
    程式設計 發佈於2024-12-23
  • 為什麼 Selenium 在 Chrome 中定位元素時會拋出“NoSuchElementException”?
    為什麼 Selenium 在 Chrome 中定位元素時會拋出“NoSuchElementException”?
    "NoSuchElementException" for Chrome with SeleniumIssue"NoSuchElementException" for Chrome with SeleniumIssueselenium.common.except...
    程式設計 發佈於2024-12-23
  • 為什麼Java中的靜態方法不能是抽象的?
    為什麼Java中的靜態方法不能是抽象的?
    為什麼Java中靜態方法不能是抽象的在Java程式設計中,靜態方法不能是抽象的。抽象方法意味著該方法有聲明但沒有實現,將其留給子類別來提供功能。但是,無論是否建立物件實例,靜態方法本質上都與類別本身相關聯。 以下範例示範了抽象靜態方法的問題:abstract class foo { abst...
    程式設計 發佈於2024-12-23
  • 如何將 Qt 偵錯輸出重定向到檔案?
    如何將 Qt 偵錯輸出重定向到檔案?
    重定向Qt 調試輸出調試Qt 應用程式時,大量的qDebug() 和相關語句可能會因過多的調試輸出而使控制台變得混亂。對此,開發人員經常尋求一種跨平台的方法來將此輸出重定向到檔案。 Qt方式:qInstallMessageHandlerQt提供了更方便的處理方法使用 qInstallMessageH...
    程式設計 發佈於2024-12-23
  • Lambda 與清單推導式:哪一種最適合 Python 清單中的元素差異?
    Lambda 與清單推導式:哪一種最適合 Python 清單中的元素差異?
    在列表中執行逐元素差異:Lambda 與列表理解查找列表中相鄰元素之間的差異是常見操作編程。在 Python 中,有多種方法可以實現此目的,包括使用 lambda 表達式或列表推導式。 Lambda 表達式:可以使用 lambda 表達式建立函數動態,然後可用於對清單中的每個元素進行操作。例如:t ...
    程式設計 發佈於2024-12-23
  • 為什麼我的 Java HttpClient 檔案上傳到 PHP 伺服器失敗,如何使用 MultipartEntity 修復它?
    為什麼我的 Java HttpClient 檔案上傳到 PHP 伺服器失敗,如何使用 MultipartEntity 修復它?
    使用PHP 透過Java 的HttpClient 上傳檔案在嘗試利用PHP 將檔案從Java 上傳到Apache 伺服器時,利用Jakarta 建立了一個JavaJava 應用程式HttpClient 函式庫版本4.0 beta2。然而,PHP 腳本無法識別上傳的文件,導致 $_FILES 數組為空...
    程式設計 發佈於2024-12-23
  • 如何使用 PIVOT 運算子在 SQL Server 中轉置資料?
    如何使用 PIVOT 運算子在 SQL Server 中轉置資料?
    如何使用PIVOT 轉置SQL Server 資料SQL Server 中的PIVOT 運算子允許您將行轉換為列,將資料從垂直方向的形式變成水平方向的形式。當尋求基於多個屬性匯總資料時,這非常有用。 範例資料和所需輸出考慮以下起始資料集:SELECT Name1, Name2, Value FROM...
    程式設計 發佈於2024-12-23

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3