Vercel AI SDK memudahkan untuk berinteraksi dengan API LLM seperti OpenAI, Anthropic dan sebagainya serta menstrim data supaya ia muncul dalam apl web anda dengan pantas semasa ia dimuatkan. Dalam artikel ini kita akan belajar cara menjalankan berbilang gesaan pada masa yang sama dan melihat keputusannya secara selari.
TL;DR: GitHub Repo ada di sini.
Adalah perkara biasa dalam apl web yang mahu menjalankan berbilang permintaan pengambilan data pada masa yang sama. Contohnya, dalam sistem blog hipotesis, apabila antara muka papan pemuka dimuatkan, kami mungkin ingin mengambil data profil pengguna, siaran yang mereka buat dan siaran pengguna lain yang mereka gemari semuanya pada masa yang sama.
Jika papan pemuka yang sama membuat permintaan kepada OpenAI pada masa yang sama, kami mungkin mahu meminta OpenAI secara serentak untuk mendapatkan petua tentang meningkatkan profil pengguna dan analisis siaran terbaharu mereka pada masa yang sama. Secara teori, kami boleh menggunakan berpuluh-puluh permintaan AI secara selari jika kami mahu (walaupun daripada platform dan model yang berbeza sama sekali), dan menganalisis maklumat, menjana kandungan dan melakukan semua jenis tugasan lain pada masa yang sama.
Anda boleh mengklon repo GitHub yang mengandungi hasil akhir di sini.
Untuk menyediakan dari awal:
Komponen utama yang melakukan semua kerja akan mengandungi borang dan beberapa bekas untuk output. Menggunakan beberapa komponen asas shadcn-ui, borang akan kelihatan seperti ini:
export function GenerationForm() { // State and other info will be defined here... return ( <form onSubmit={onSubmit} className="flex flex-col gap-3 w-full"> <div className="inline-block mb-4 w-full flex flex-row gap-1"> <Button type="submit">Generate News & Weather</Button> </div> {isGenerating ? ( <div className="flex flex-row w-full justify-center items-center p-4 transition-all"> <Spinner className="h-6 w-6 text-slate-900" /> </div> ) : null} <h3 className="font-bold">Historical Weather</h3> <div className="mt-4 mb-8 p-4 rounded-md shadow-md bg-blue-100"> {weather ? weather : null} </div> <h4 className="font-bold">Historical News</h4> <div className="mt-4 p-4 rounded-md shadow-md bg-green-100">{news ? news : null}</div> </form> ) }
Anda boleh melihat bahawa kami mempunyai beberapa perkara di sini:
Buat masa ini anda boleh mengekodkan nilai-nilai ini; mereka semua akan ditarik dari aliran kami.
Tindakan pelayan streamAnswer ialah perkara yang akan melakukan kerja mencipta dan mengemas kini strim kami.
Struktur tindakan adalah ini:
export async function streamAnswer(question: string) { // Booleans for indicating whether each stream is currently streaming const isGeneratingStream1 = createStreamableValue(true); const isGeneratingStream2 = createStreamableValue(true); // The current stream values const weatherStream = createStreamableValue(""); const newsStream = createStreamableValue(""); // Create the first stream. Notice that we don't use await here, so that we // don't block the rest of this function from running. streamText({ // ... params, including the LLM prompt }).then(async (result) => { // Read from the async iterator. Set the stream value to each new word // received. for await (const value of result.textStream) { weatherStream.update(value || ""); } } finally { // Set isGenerating to false, and close that stream. isGeneratingStream1.update(false); isGeneratingStream1.done(); // Close the given stream so the request doesn't hang. weatherStream.done(); } }); // Same thing for the second stream. streamText({ // ... params }).then(async (result) => { // ... }) // Return any streams we want to read on the client. return { isGeneratingStream1: isGeneratingStream1.value, isGeneratingStream2: isGeneratingStream2.value, weatherStream: weatherStream.value, newsStream: newsStream.value, }; }
Pengendali onSubmit borang akan melakukan semua kerja di sini. Berikut ialah pecahan cara ia berfungsi:
"use client"; import { SyntheticEvent, useState } from "react"; import { Button } from "./ui/button"; import { readStreamableValue, useUIState } from "ai/rsc"; import { streamAnswer } from "@/app/actions"; import { Spinner } from "./svgs/Spinner"; export function GenerationForm() { // State for loading flags const [isGeneratingStream1, setIsGeneratingStream1] = useState<boolean>(false); const [isGeneratingStream2, setIsGeneratingStream2] = useState<boolean>(false); // State for the LLM output streams const [weather, setWeather] = useState<string>(""); const [news, setNews] = useState<string>(""); // We'll hide the loader when both streams are done. const isGenerating = isGeneratingStream1 || isGeneratingStream2; async function onSubmit(e: SyntheticEvent) { e.preventDefault(); // Clear previous results. setNews(""); setWeather(""); // Call the server action. The returned object will have all the streams in it. const result = await streamAnswer(question); // Translate each stream into an async iterator so we can loop through // the values as they are generated. const isGeneratingStream1 = readStreamableValue(result.isGeneratingStream1); const isGeneratingStream2 = readStreamableValue(result.isGeneratingStream2); const weatherStream = readStreamableValue(result.weatherStream); const newsStream = readStreamableValue(result.newsStream); // Iterate through each stream, putting its values into state one by one. // Notice the IIFEs again! As on the server, these allow us to prevent blocking // the function, so that we can run these iterators in parallel. (async () => { for await (const value of isGeneratingStream1) { if (value != null) { setIsGeneratingStream1(value); } } })(); (async () => { for await (const value of isGeneratingStream2) { if (value != null) { setIsGeneratingStream2(value); } } })(); (async () => { for await (const value of weatherStream) { setWeather((existing) => (existing + value) as string); } })(); (async () => { for await (const value of newsStream) { setNews((existing) => (existing + value) as string); } })(); } return ( // ... The form code from before. ); }
Atas ialah kandungan terperinci Berbilang Strim AI Selari dengan Vercel AI SDK. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!