هسته اصلی ویژگیهای هوش مصنوعی برنامه شما درخواستهای مدل تولیدی است، اما به ندرت پیش میآید که بتوانید ورودی کاربر را دریافت کنید، آن را به مدل ارسال کنید و خروجی مدل را به کاربر نمایش دهید. معمولاً مراحل پیش و پس از پردازش وجود دارد که باید با فراخوانی مدل همراه باشد. به عنوان مثال:
- بازیابی اطلاعات متنی برای ارسال با تماس مدل
- بازیابی تاریخچه جلسه فعلی کاربر، به عنوان مثال در یک برنامه چت
- استفاده از یک مدل برای فرمت مجدد ورودی کاربر به روشی که برای انتقال به مدل دیگر مناسب باشد
- ارزیابی "ایمنی" خروجی یک مدل قبل از ارائه آن به کاربر
- ترکیب خروجی چند مدل
هر مرحله از این گردش کار باید برای موفقیت هر کار مرتبط با هوش مصنوعی با هم کار کند.
در Genkit، شما این منطق مرتبط را با استفاده از ساختاری به نام جریان نشان میدهید. جریانها دقیقاً مانند توابع با استفاده از کدهای TypeScript معمولی نوشته میشوند، اما قابلیتهای اضافی را برای سهولت توسعه ویژگیهای هوش مصنوعی اضافه میکنند:
- ایمنی نوع : طرحوارههای ورودی و خروجی که با استفاده از Zod تعریف شدهاند، که بررسی نوع استاتیک و زمان اجرا را فراهم میکند.
- یکپارچه سازی با رابط کاربری توسعه دهنده : اشکال زدایی مستقل از کد برنامه شما با استفاده از رابط کاربری توسعه دهنده جریان می یابد. در رابط کاربری توسعهدهنده، میتوانید جریانها را اجرا کنید و ردیابیها را برای هر مرحله از جریان مشاهده کنید.
- استقرار ساده : جریان ها را مستقیماً به عنوان نقاط پایانی API وب، با استفاده از توابع ابری برای Firebase یا هر پلتفرمی که می تواند یک برنامه وب را میزبانی کند، استقرار دهید.
برخلاف ویژگیهای مشابه در سایر فریمورکها، جریانهای Genkit سبک و محجوب هستند و برنامه شما را مجبور به انطباق با انتزاع خاصی نمیکنند. تمام منطق جریان در TypeScript استاندارد نوشته شده است و کد داخل یک جریان نیازی به اطلاع از جریان ندارد.
تعریف و فراخوانی جریان ها
در سادهترین شکل آن، یک جریان فقط یک تابع را میپیچد. مثال زیر تابعی را میپیچد که generate()
فراخوانی میکند:
export const menuSuggestionFlow = ai.defineFlow(
{
name: 'menuSuggestionFlow',
},
async (restaurantTheme) => {
const { text } = await ai.generate({
model: gemini15Flash,
prompt: `Invent a menu item for a ${restaurantTheme} themed restaurant.`,
});
return text;
}
);
فقط با قرار دادن فراخوانی های generate()
به این صورت، برخی قابلیت ها را اضافه می کنید: انجام این کار به شما امکان می دهد جریان را از Genkit CLI و از UI توسعه دهنده اجرا کنید و برای چندین ویژگی Genkit، از جمله استقرار و مشاهده پذیری (بخش های بعدی) لازم است. در مورد این موضوعات بحث کنید).
طرحواره های ورودی و خروجی
یکی از مهمترین مزایای جریان های Genkit نسبت به فراخوانی مستقیم API مدل، ایمنی نوع ورودی و خروجی است. هنگام تعریف جریان ها، می توانید طرحواره ها را با استفاده از Zod برای آنها تعریف کنید، تقریباً به همان روشی که طرح خروجی یک فراخوانی generate()
را تعریف می کنید. با این حال، بر خلاف generate()
، شما همچنین می توانید یک طرح ورودی را مشخص کنید.
در اینجا اصلاح آخرین مثال است، که جریانی را تعریف می کند که یک رشته را به عنوان ورودی می گیرد و یک شی را خروجی می دهد:
const MenuItemSchema = z.object({
dishname: z.string(),
description: z.string(),
});
export const menuSuggestionFlowWithSchema = ai.defineFlow(
{
name: 'menuSuggestionFlow',
inputSchema: z.string(),
outputSchema: MenuItemSchema,
},
async (restaurantTheme) => {
const { output } = await ai.generate({
model: gemini15Flash,
prompt: `Invent a menu item for a ${restaurantTheme} themed restaurant.`,
output: { schema: MenuItemSchema },
});
if (output == null) {
throw new Error("Response doesn't satisfy schema.");
}
return output;
}
);
توجه داشته باشید که طرح واره یک جریان لزوماً نباید با طرح فراخوانی های generate()
در داخل جریان همخوانی داشته باشد (در واقع، یک جریان ممکن است حتی شامل فراخوانی های generate()
نباشد). در اینجا یک نوع مثال است که یک طرحواره را برای generate()
ارسال میکند، اما از خروجی ساختاریافته برای قالببندی یک رشته ساده استفاده میکند که جریان آن را برمیگرداند.
export const menuSuggestionFlowMarkdown = ai.defineFlow(
{
name: 'menuSuggestionFlow',
inputSchema: z.string(),
outputSchema: z.string(),
},
async (restaurantTheme) => {
const { output } = await ai.generate({
model: gemini15Flash,
prompt: `Invent a menu item for a ${restaurantTheme} themed restaurant.`,
output: { schema: MenuItemSchema },
});
if (output == null) {
throw new Error("Response doesn't satisfy schema.");
}
return `**${output.dishname}**: ${output.description}`;
}
);
جریان های فراخوانی
هنگامی که یک جریان را تعریف کردید، می توانید آن را از کد Node.js خود فراخوانی کنید:
const { text } = await menuSuggestionFlow('bistro');
آرگومان جریان باید با طرح ورودی مطابقت داشته باشد، اگر شما یکی را تعریف کرده باشید.
اگر یک طرح خروجی تعریف کرده باشید، پاسخ جریان با آن مطابقت خواهد داشت. به عنوان مثال، اگر شما طرح خروجی را روی MenuItemSchema
تنظیم کنید، خروجی جریان حاوی ویژگی های آن خواهد بود:
const { dishname, description } =
await menuSuggestionFlowWithSchema('bistro');
جریان جریان دارد
Flow ها با استفاده از رابطی شبیه به رابط جریان generate()
از جریان پشتیبانی می کنند. پخش جریانی زمانی مفید است که جریان شما مقدار زیادی خروجی تولید می کند، زیرا می توانید خروجی را در حین تولید به کاربر ارائه دهید، که پاسخگویی درک شده برنامه شما را بهبود می بخشد. به عنوان یک مثال آشنا، رابط های LLM مبتنی بر چت اغلب پاسخ های خود را در حین تولید به کاربر ارسال می کنند.
در اینجا نمونه ای از جریانی است که از جریان پشتیبانی می کند:
export const menuSuggestionStreamingFlow = ai.defineFlow(
{
name: 'menuSuggestionFlow',
inputSchema: z.string(),
streamSchema: z.string(),
outputSchema: z.object({ theme: z.string(), menuItem: z.string() }),
},
async (restaurantTheme, streamingCallback) => {
const response = await ai.generateStream({
model: gemini15Flash,
prompt: `Invent a menu item for a ${restaurantTheme} themed restaurant.`,
});
if (streamingCallback) {
for await (const chunk of response.stream) {
// Here, you could process the chunk in some way before sending it to
// the output stream via streamingCallback(). In this example, we output
// the text of the chunk, unmodified.
streamingCallback(chunk.text);
}
}
return {
theme: restaurantTheme,
menuItem: (await response.response).text,
};
}
);
- گزینه
streamSchema
نوع مقادیر جریان های جریان شما را مشخص می کند. لزوماً لازم نیست که این همان نوعoutputSchema
باشد، که نوع خروجی کامل جریان است. -
streamingCallback
یک تابع پاسخ به تماس است که یک پارامتر واحد از نوع مشخص شده توسطstreamSchema
را می گیرد. هر زمان که داده در جریان شما در دسترس قرار گرفت، با فراخوانی این تابع، داده ها را به جریان خروجی ارسال کنید. توجه داشته باشید کهstreamingCallback
تنها در صورتی تعریف میشود که تماسگیرنده جریان شما خروجی جریان را درخواست کرده باشد، بنابراین باید قبل از فراخوانی آن را بررسی کنید.
در مثال بالا، مقادیر استریم شده توسط جریان مستقیماً با مقادیر استریم شده توسط generate()
در داخل جریان جفت می شوند. اگرچه اغلب اینطور است، اما لازم نیست اینطور باشد: میتوانید مقادیری را با استفاده از callback هر چند وقت یکبار برای جریان شما مفید است، به جریان خروجی بدهید.
فراخوانی جریانهای جریان
جریانهای جریانی نیز قابل فراخوانی هستند، اما آنها بلافاصله به جای یک وعده، یک شی پاسخ را برمیگردانند:
const response = menuSuggestionStreamingFlow.stream('Danube');
شی پاسخ دارای یک ویژگی جریان است که می توانید از آن برای تکرار بر روی خروجی جریان جریان هنگام تولید استفاده کنید:
for await (const chunk of response.stream) {
console.log('chunk', chunk);
}
شما همچنین می توانید خروجی کامل جریان را دریافت کنید، همانطور که می توانید با یک جریان غیر جریانی:
const output = await response.output;
توجه داشته باشید که خروجی جریان یک جریان ممکن است با خروجی کامل یکسان نباشد. خروجی جریان مطابق با streamSchema
است، در حالی که خروجی کامل مطابق با outputSchema
است.
در حال اجرا از خط فرمان جریان می یابد
میتوانید با استفاده از ابزار Genkit CLI، جریانها را از خط فرمان اجرا کنید:
genkit flow:run menuSuggestionFlow '"French"'
برای جریانهای جریان، میتوانید خروجی جریان را با افزودن پرچم -s
در کنسول چاپ کنید:
genkit flow:run menuSuggestionFlow '"French"' -s
اجرای یک جریان از خط فرمان برای آزمایش یک جریان یا برای اجرای جریان هایی که وظایف مورد نیاز را به طور موقت انجام می دهند مفید است - به عنوان مثال، برای اجرای جریانی که یک سند را در پایگاه داده برداری شما وارد می کند.
اشکال زدایی جریان دارد
یکی از مزایای کپسوله کردن منطق هوش مصنوعی در یک جریان این است که می توانید جریان را به طور مستقل از برنامه خود با استفاده از رابط کاربری توسعه دهنده Genkit آزمایش و اشکال زدایی کنید.
برای شروع UI توسعه دهنده، دستورات زیر را از دایرکتوری پروژه خود اجرا کنید:
genkit start -- tsx --watch src/your-code.ts
از تب Run در رابط توسعه دهنده، می توانید هر یک از جریان های تعریف شده در پروژه خود را اجرا کنید:
پس از اجرای یک جریان، میتوانید با کلیک روی View trace یا نگاه کردن به برگه Inspect ، ردی از فراخوان جریان را بررسی کنید.
در نمایشگر ردیابی، می توانید جزئیات مربوط به اجرای کل جریان و همچنین جزئیات هر یک از مراحل جداگانه در جریان را مشاهده کنید. به عنوان مثال، جریان زیر را در نظر بگیرید که شامل چندین درخواست تولید است:
const PrixFixeMenuSchema = z.object({
starter: z.string(),
soup: z.string(),
main: z.string(),
dessert: z.string(),
});
export const complexMenuSuggestionFlow = ai.defineFlow(
{
name: 'complexMenuSuggestionFlow',
inputSchema: z.string(),
outputSchema: PrixFixeMenuSchema,
},
async (theme: string): Promise<z.infer<typeof PrixFixeMenuSchema>> => {
const chat = ai.chat({ model: gemini15Flash });
await chat.send('What makes a good prix fixe menu?');
await chat.send(
'What are some ingredients, seasonings, and cooking techniques that ' +
`would work for a ${theme} themed menu?`
);
const { output } = await chat.send({
prompt:
`Based on our discussion, invent a prix fixe menu for a ${theme} ` +
'themed restaurant.',
output: {
schema: PrixFixeMenuSchema,
},
});
if (!output) {
throw new Error('No data generated.');
}
return output;
}
);
هنگامی که این جریان را اجرا می کنید، نمایشگر ردیابی جزئیات هر درخواست نسل از جمله خروجی آن را به شما نشان می دهد:
مراحل جریان
در مثال آخر، دیدید که هر فراخوانی generate()
به عنوان یک مرحله جداگانه در نمایشگر ردیابی نشان داده شد. هر یک از اقدامات اساسی Genkit به عنوان مراحل جداگانه یک جریان نشان داده می شود:
-
generate()
-
Chat.send()
-
embed()
-
index()
-
retrieve()
اگر میخواهید کدی غیر از موارد بالا را در ردیابیهای خود قرار دهید، میتوانید این کار را با قرار دادن کد در یک فراخوانی run()
انجام دهید. ممکن است این کار را برای تماس با کتابخانه های شخص ثالث که از Genkit آگاه نیستند یا برای هر بخش مهمی از کد انجام دهید.
برای مثال، در اینجا یک جریان با دو مرحله وجود دارد: مرحله اول یک منو را با استفاده از روشی نامشخص بازیابی می کند، و مرحله دوم شامل منو به عنوان زمینه برای یک فراخوانی generate()
است.
import { run } from 'genkit';
export const menuQuestionFlow = ai.defineFlow(
{
name: 'menuQuestionFlow',
inputSchema: z.string(),
outputSchema: z.string(),
},
async (input: string): Promise<string> => {
const menu = await ai.run(
'retrieve-daily-menu',
async (): Promise<string> => {
// Retrieve today's menu. (This could be a database access or simply
// fetching the menu from your website.)
// ...
return menu;
}
);
const { text } = await ai.generate({
model: gemini15Flash,
system: "Help the user answer questions about today's menu.",
prompt: input,
docs: [{ content: [{ text: menu }] }],
});
return text;
}
);
از آنجایی که مرحله بازیابی در یک فراخوانی run()
پیچیده شده است، به عنوان یک مرحله در نمایشگر ردیابی گنجانده شده است:
استقرار جریان ها
میتوانید جریانهای خود را مستقیماً بهعنوان نقاط پایانی API وب مستقر کنید، تا بتوانید از مشتریان برنامه خود تماس بگیرید. استقرار به طور مفصل در چندین صفحه دیگر مورد بحث قرار گرفته است، اما این بخش مروری کوتاه بر گزینه های استقرار شما ارائه می دهد.
توابع ابری برای Firebase
برای استقرار جریانها با توابع Cloud برای Firebase، از افزونه firebase
استفاده کنید. در تعاریف جریان خود، defineFlow
با onFlow
جایگزین کنید و یک authPolicy
اضافه کنید.
import { firebaseAuth } from '@genkit-ai/firebase/auth';
import { onFlow } from '@genkit-ai/firebase/functions';
export const menuSuggestion = onFlow(
ai,
{
name: 'menuSuggestionFlow',
authPolicy: firebaseAuth((user) => {
if (!user.email_verified) {
throw new Error('Verified email required to run flow');
}
}),
},
async (restaurantTheme) => {
// ...
}
);
برای اطلاعات بیشتر به صفحات زیر مراجعه کنید:
Express.js
برای استقرار جریانها با استفاده از هر پلتفرم میزبانی Node.js، مانند Cloud Run، جریانهای خود را با استفاده از defineFlow()
تعریف کنید و سپس startFlowServer()
فراخوانی کنید:
export const menuSuggestionFlow = ai.defineFlow(
{
name: 'menuSuggestionFlow',
},
async (restaurantTheme) => {
// ...
}
);
startFlowServer({
flows: [menuSuggestionFlow],
});
بهطور پیشفرض، startFlowServer
تمام جریانهای تعریفشده در پایگاه کد شما را بهعنوان نقاط پایانی HTTP ارائه میکند (به عنوان مثال، http://localhost:3400/menuSuggestionFlow
). می توانید یک جریان را با درخواست POST به صورت زیر فراخوانی کنید:
curl -X POST "http://localhost:3400/menuSuggestionFlow" \
-H "Content-Type: application/json" -d '{"data": "banana"}'
در صورت نیاز، میتوانید سرور flows را سفارشی کنید تا لیست خاصی از جریانها را ارائه کند، همانطور که در زیر نشان داده شده است. همچنین می توانید یک پورت سفارشی را مشخص کنید (اگر تنظیم شود از متغیر محیطی PORT استفاده می کند) یا تنظیمات CORS را مشخص کنید.
export const flowA = ai.defineFlow({ name: 'flowA' }, async (subject) => {
// ...
});
export const flowB = ai.defineFlow({ name: 'flowB' }, async (subject) => {
// ...
});
startFlowServer({
flows: [flowB],
port: 4567,
cors: {
origin: '*',
},
});
برای اطلاعات در مورد استقرار در پلتفرمهای خاص، به Deploy with Cloud Run و Deploy flows به هر پلتفرم Node.js مراجعه کنید.