{"id":483314,"date":"2026-06-11T11:23:12","date_gmt":"2026-06-11T11:23:12","guid":{"rendered":"https:\/\/savepearlharbor.com\/?p=483314"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=483314","title":{"rendered":"Telegram-\u0431\u043e\u0442 \u0441 RAG \u043d\u0430 Cloudflare Workers: \u0431\u0430\u0437\u0430 \u0437\u043d\u0430\u043d\u0438\u0439 \u0431\u0435\u0437 \u0432\u0435\u043a\u0442\u043e\u0440\u043e\u0432 \u0438 \u0431\u0435\u0437 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445"},"content":{"rendered":"<div xmlns=\"http:\/\/www.w3.org\/1999\/xhtml\">\n<p>\u0412 <a href=\"https:\/\/habr.com\/ru\/articles\/1042264\/\" rel=\"noopener noreferrer nofollow\">\u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0435<\/a> \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u043b\u0438, \u043a\u0430\u043a \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u0443\u044e wiki \u0438\u0437 markdown-\u0444\u0430\u0439\u043b\u043e\u0432 \u043d\u0430 Astro\/Starlight \u2014 \u043d\u0430 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u043b\u0438\u0447\u043d\u043e\u0433\u043e \u043a\u0430\u0440\u044c\u0435\u0440\u043d\u043e\u0433\u043e \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u0430. \u0412 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445 \u043f\u043e\u044f\u0432\u0438\u043b\u0438\u0441\u044c \u0437\u0430\u043a\u043e\u043d\u043e\u043c\u0435\u0440\u043d\u044b\u0439 \u0432\u043e\u043f\u0440\u043e\u0441: \u00ab\u043f\u043e\u0447\u0435\u043c\u0443 \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0430\u043a?\u00bb, \u00ab\u0447\u0442\u043e \u0437\u0430 \u0441\u0442\u0440\u0430\u043d\u043d\u044b\u0439 \u0432\u044b\u0431\u043e\u0440 \u0441\u0442\u0435\u043a\u0430?\u00bb, \u00ab\u0430 \u0434\u043b\u044f \u0447\u0435\u0433\u043e \u0435\u0449\u0451 \u044d\u0442\u043e \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c, \u043a\u0440\u043e\u043c\u0435 \u043a\u0430\u043a \u0434\u043b\u044f \u0441\u0435\u0431\u044f?\u00bb.<\/p>\n<p>\u0425\u043e\u0440\u043e\u0448\u0438\u0439 \u0432\u043e\u043f\u0440\u043e\u0441. \u042d\u0442\u0430 \u0441\u0442\u0430\u0442\u044c\u044f \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u043d\u0430 \u043d\u0435\u0433\u043e \u0434\u0435\u043b\u043e\u043c.<\/p>\n<p>\u0422\u0430 \u0436\u0435 \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0430 \u2014 wiki \u0438\u0437 markdown \u2014 \u043d\u043e \u0442\u0435\u043f\u0435\u0440\u044c \u0441 Telegram-\u0431\u043e\u0442\u043e\u043c \u043f\u043e\u0432\u0435\u0440\u0445. \u0411\u043e\u0442 \u0443\u043c\u0435\u0435\u0442 \u0438\u0441\u043a\u0430\u0442\u044c \u043f\u043e \u0431\u0430\u0437\u0435 \u0437\u043d\u0430\u043d\u0438\u0439 \u0438 \u043e\u0442\u0432\u0435\u0447\u0430\u0442\u044c \u0441 \u0446\u0438\u0442\u0430\u0442\u0430\u043c\u0438 \u0438 \u0441\u0441\u044b\u043b\u043a\u0430\u043c\u0438 \u043d\u0430 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438. \u0412 \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0435 \u043f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u043e\u0439 \u043e\u0431\u043b\u0430\u0441\u0442\u0438 \u0432\u044b\u0431\u0440\u0430\u043d\u0430 \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u044f \u0438 \u0444\u0438\u043b\u043e\u0441\u043e\u0444\u0438\u044f: \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0441\u044f <a href=\"https:\/\/t.me\/pif_bbot\" rel=\"noopener noreferrer nofollow\">@pif_bbot<\/a> \u2014 \u044d\u043c\u043f\u0430\u0442\u0438\u0447\u043d\u044b\u0439 \u043f\u043e\u043c\u043e\u0449\u043d\u0438\u043a, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 <a href=\"https:\/\/anatolii-iumashev.github.io\/pifai\/\" rel=\"noopener noreferrer nofollow\">\u043e\u0442\u043a\u0440\u044b\u0442\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439<\/a> \u043f\u043e \u041d\u0412\u041e, \u042e\u043d\u0433\u0443, \u0424\u0440\u0430\u043d\u043a\u043b\u0443 \u0438 \u0434\u0440\u0443\u0433\u0438\u043c \u0430\u0432\u0442\u043e\u0440\u0430\u043c.<\/p>\n<p>\u0412\u0435\u0441\u044c \u043a\u043e\u0434 \u2014 \u0432 <a href=\"https:\/\/github.com\/anatolii-iumashev\/pifai\" rel=\"noopener noreferrer nofollow\">\u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438 \u043d\u0430 GitHub<\/a>, \u043f\u0430\u043f\u043a\u0430 <code>bot\/<\/code>.<\/p>\n<hr\/>\n<p>\u041a\u043e\u0433\u0434\u0430 \u0433\u043e\u0432\u043e\u0440\u044f\u0442 \u00abRAG\u00bb, \u0432 \u0433\u043e\u043b\u043e\u0432\u0435 \u0441\u0440\u0430\u0437\u0443 \u0432\u043e\u0437\u043d\u0438\u043a\u0430\u0435\u0442: \u0432\u0435\u043a\u0442\u043e\u0440\u043d\u0430\u044f \u0411\u0414, \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u0438, OpenAI API, Pinecone \u0438\u043b\u0438 pgvector. \u041a\u0430\u0436\u0435\u0442\u0441\u044f, \u0447\u0442\u043e \u0434\u043b\u044f \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0440\u0430\u0431\u043e\u0447\u0435\u0433\u043e \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u043d\u0443\u0436\u043d\u0430 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u0430\u044f \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u0438 \u043d\u0435\u043c\u0430\u043b\u0435\u043d\u044c\u043a\u0438\u0439 \u0431\u044e\u0434\u0436\u0435\u0442.<\/p>\n<p>\u041d\u043e \u0435\u0441\u0442\u044c \u0434\u0440\u0443\u0433\u043e\u0439 \u043f\u0443\u0442\u044c. \u0415\u0441\u043b\u0438 \u043f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u0430\u044f \u043e\u0431\u043b\u0430\u0441\u0442\u044c \u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e \u0441\u043f\u0435\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u043d\u0430 \u2014 \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u044f, \u044e\u0440\u0438\u0441\u043f\u0440\u0443\u0434\u0435\u043d\u0446\u0438\u044f, \u0442\u0435\u0445\u043d\u0438\u0447\u0435\u0441\u043a\u0430\u044f \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044f \u2014 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 \u043f\u043e\u0438\u0441\u043a \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c \u0434\u0430\u0451\u0442 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b, \u0441\u0440\u0430\u0432\u043d\u0438\u043c\u044b\u0435 \u0441 \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u043c. \u0411\u0435\u0437 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u043e\u0432. \u0411\u0435\u0437 \u0432\u043d\u0435\u0448\u043d\u0438\u0445 API. \u0411\u0435\u0437 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445.<\/p>\n<p>\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043c\u044b \u043f\u043e\u0441\u0442\u0440\u043e\u0438\u043c Telegram-\u0431\u043e\u0442\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439:<\/p>\n<ul>\n<li>\n<p><strong>\u0438\u0449\u0435\u0442 \u043f\u043e \u0431\u0430\u0437\u0435 \u0437\u043d\u0430\u043d\u0438\u0439<\/strong> \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c\u0430 Jaccard \u2014 \u0434\u0435\u0442\u0435\u0440\u043c\u0438\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e \u0438 \u0431\u044b\u0441\u0442\u0440\u043e<\/p>\n<\/li>\n<li>\n<p><strong>\u0446\u0438\u0442\u0438\u0440\u0443\u0435\u0442 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438<\/strong> \u0441\u043e \u0441\u0441\u044b\u043b\u043a\u0430\u043c\u0438 \u043d\u0430 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 wiki<\/p>\n<\/li>\n<li>\n<p><strong>\u043f\u043e\u043c\u043d\u0438\u0442 \u0438\u0441\u0442\u043e\u0440\u0438\u044e<\/strong> \u0434\u0438\u0430\u043b\u043e\u0433\u0430 \u043c\u0435\u0436\u0434\u0443 \u0441\u0435\u0441\u0441\u0438\u044f\u043c\u0438 \u0447\u0435\u0440\u0435\u0437 Cloudflare KV<\/p>\n<\/li>\n<li>\n<p><strong>\u0434\u0435\u043f\u043b\u043e\u0438\u0442\u0441\u044f \u043e\u0434\u043d\u043e\u0439 \u043a\u043e\u043c\u0430\u043d\u0434\u043e\u0439<\/strong> \u043d\u0430 Cloudflare Workers \u2014 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e \u043f\u0440\u0438 \u0443\u043c\u0435\u0440\u0435\u043d\u043d\u043e\u0439 \u043d\u0430\u0433\u0440\u0443\u0437\u043a\u0435<\/p>\n<\/li>\n<\/ul>\n<h3>\u0421\u0442\u0435\u043a<\/h3>\n<ul>\n<li>\n<p><strong>TypeScript<\/strong> \u2014 \u0435\u0434\u0438\u043d\u044b\u0439 \u044f\u0437\u044b\u043a \u0438 \u0434\u043b\u044f \u0431\u043e\u0442\u0430, \u0438 \u0434\u043b\u044f \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432 \u0441\u0431\u043e\u0440\u043a\u0438<\/p>\n<\/li>\n<li>\n<p><strong>Telegraf<\/strong> \u2014 \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a \u0434\u043b\u044f Telegram Bot API<\/p>\n<\/li>\n<li>\n<p><strong>Groq API<\/strong> \u2014 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 LLM (Llama-3.1-8b-instant, \u043e\u0447\u0435\u043d\u044c \u043d\u0438\u0437\u043a\u0430\u044f \u043b\u0430\u0442\u0435\u043d\u0442\u043d\u043e\u0441\u0442\u044c)<\/p>\n<\/li>\n<li>\n<p><strong>Cloudflare Workers<\/strong> \u2014 serverless edge, cold start &lt; 5ms, \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier<\/p>\n<\/li>\n<li>\n<p><strong>Cloudflare KV<\/strong> \u2014 \u0445\u0440\u0430\u043d\u0435\u043d\u0438\u0435 \u0438\u0441\u0442\u043e\u0440\u0438\u0438 \u0441\u0435\u0441\u0441\u0438\u0439<\/p>\n<\/li>\n<\/ul>\n<h3>\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430<\/h3>\n<pre><code>Wiki (Markdown) \u2500\u2500\u25ba build-knowledge.ts \u2500\u2500\u25ba knowledge.ts                                               \u2502                                         256 \u0447\u0430\u043d\u043a\u043e\u0432 \u0441                                         \u043f\u0440\u0435\u0434\u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u044b\u043c\u0438                                         \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c\u0438 \u0441\u043b\u043e\u0432\u0430\u043c\u0438                                               \u2502Telegram \u2500\u2500\u25ba CF Worker \u2500\u2500\u25ba Retriever \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518                               \u2502         (Jaccard)                               \u25bc                          Groq LLM \u2500\u2500\u25ba \u043e\u0442\u0432\u0435\u0442 \u0441 \u0446\u0438\u0442\u0430\u0442\u0430\u043c\u0438                               \u25b2                          KV (\u0438\u0441\u0442\u043e\u0440\u0438\u044f)<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:87px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0413\u043b\u0430\u0432\u043d\u0430\u044f \u0438\u0434\u0435\u044f: <strong>\u0431\u0430\u0437\u0430 \u0437\u043d\u0430\u043d\u0438\u0439 \u0432\u0441\u0442\u0440\u043e\u0435\u043d\u0430 \u043f\u0440\u044f\u043c\u043e \u0432 \u043a\u043e\u0434<\/strong>. \u041f\u0440\u0438 \u0434\u0435\u043f\u043b\u043e\u0435 <code>knowledge.ts<\/code> \u0441 256 \u0447\u0430\u043d\u043a\u0430\u043c\u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442\u0441\u044f \u0432 \u043f\u0430\u043c\u044f\u0442\u044c \u0432\u043e\u0440\u043a\u0435\u0440\u0430 \u2014 \u043d\u0438\u043a\u0430\u043a\u0438\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u043a \u0411\u0414, \u043d\u0443\u043b\u0435\u0432\u0430\u044f \u043b\u0430\u0442\u0435\u043d\u0442\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0438\u0441\u043a\u0430. \u0417\u0432\u0443\u0447\u0438\u0442 \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u0431\u0435\u0437\u0443\u043c\u043d\u043e, \u043d\u043e \u043d\u0430 \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043e\u0442\u043b\u0438\u0447\u043d\u043e: 29 \u0441\u0442\u0430\u0442\u0435\u0439, ~620KB, \u043f\u043e\u0438\u0441\u043a \u0437\u0430\u043d\u0438\u043c\u0430\u0435\u0442 \u0435\u0434\u0438\u043d\u0438\u0446\u044b \u043c\u0438\u043b\u043b\u0438\u0441\u0435\u043a\u0443\u043d\u0434.<\/p>\n<h3>\u041f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430<\/h3>\n<p>\u041d\u0443\u0436\u043d\u043e:<\/p>\n<ul>\n<li>\n<p>Node.js 20+<\/p>\n<\/li>\n<li>\n<p>\u0410\u043a\u043a\u0430\u0443\u043d\u0442 <a href=\"https:\/\/cloudflare.com\" rel=\"noopener noreferrer nofollow\">Cloudflare<\/a> (\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439)<\/p>\n<\/li>\n<li>\n<p>\u0422\u043e\u043a\u0435\u043d Telegram-\u0431\u043e\u0442\u0430 \u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0443 <a href=\"https:\/\/t.me\/BotFather\" rel=\"noopener noreferrer nofollow\">@BotFather<\/a><\/p>\n<\/li>\n<li>\n<p>API-\u043a\u043b\u044e\u0447 <a href=\"https:\/\/console.groq.com\" rel=\"noopener noreferrer nofollow\">Groq<\/a> (\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier)<\/p>\n<\/li>\n<\/ul>\n<p>\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 (wiki \u0443\u0436\u0435 \u0435\u0441\u0442\u044c \u0438\u0437 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u043f\u0430\u043f\u043a\u0443 <code>bot\/<\/code>):<\/p>\n<pre><code>pif\/\u251c\u2500\u2500 src\/content\/docs\/      # Wiki \u0438\u0437 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0438\u2502   \u251c\u2500\u2500 authors\/\u2502   \u2502   \u251c\u2500\u2500 jung\/\u2502   \u2502   \u2502   \u2514\u2500\u2500 shadow.md\u2502   \u2502   \u2514\u2500\u2500 frankl\/\u2502   \u2502       \u2514\u2500\u2500 logotherapy.md\u2502   \u2514\u2500\u2500 practices\/\u2502       \u2514\u2500\u2500 nvc.md\u2514\u2500\u2500 bot\/    \u251c\u2500\u2500 src\/    \u2502   \u251c\u2500\u2500 index.ts       # \u0442\u043e\u0447\u043a\u0430 \u0432\u0445\u043e\u0434\u0430 CF Workers    \u2502   \u251c\u2500\u2500 bot.ts         # Telegram-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438    \u2502   \u251c\u2500\u2500 knowledge.ts   # \u0430\u0432\u0442\u043e\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0438\u043d\u0434\u0435\u043a\u0441 (\u043d\u0435 \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c)    \u2502   \u251c\u2500\u2500 retriever.ts   # RAG-\u043f\u043e\u0438\u0441\u043a    \u2502   \u251c\u2500\u2500 llm.ts         # \u043a\u043b\u0438\u0435\u043d\u0442 Groq    \u2502   \u251c\u2500\u2500 session.ts     # \u0441\u0435\u0441\u0441\u0438\u0438 \u0447\u0435\u0440\u0435\u0437 KV    \u2502   \u2514\u2500\u2500 prompts.ts     # system prompt    \u251c\u2500\u2500 scripts\/    \u2502   \u2514\u2500\u2500 build-knowledge.ts    \u2514\u2500\u2500 wrangler.toml<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0423\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439:<\/p>\n<pre><code class=\"bash\">cd botnpm init -ynpm install telegrafnpm install -D wrangler tsx typescript @cloudflare\/workers-types<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>\u0428\u0430\u0433 1. \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044f \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439<\/h3>\n<p>\u041f\u0435\u0440\u0432\u044b\u0439 \u0448\u0430\u0433 \u2014 \u043f\u0440\u0435\u0432\u0440\u0430\u0442\u0438\u0442\u044c markdown-\u0444\u0430\u0439\u043b\u044b wiki \u0432 \u0438\u043d\u0434\u0435\u043a\u0441 \u0434\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430.<\/p>\n<p>\u0421\u043a\u0440\u0438\u043f\u0442 <code>scripts\/build-knowledge.ts<\/code> \u0434\u0435\u043b\u0430\u0435\u0442 \u0442\u0440\u0438 \u0432\u0435\u0449\u0438:<\/p>\n<ol>\n<li>\n<p>\u0421\u043a\u0430\u043d\u0438\u0440\u0443\u0435\u0442 <code>src\/content\/docs\/**\/*.md<\/code><\/p>\n<\/li>\n<li>\n<p>\u0420\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u0442 \u043a\u0430\u0436\u0434\u0443\u044e \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0443 \u043d\u0430 \u0441\u0435\u043a\u0446\u0438\u0438 \u043f\u043e <code>## \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430\u043c<\/code><\/p>\n<\/li>\n<li>\n<p>\u0414\u043b\u044f \u043a\u0430\u0436\u0434\u043e\u0439 \u0441\u0435\u043a\u0446\u0438\u0438 \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442 \u0441\u043f\u0438\u0441\u043e\u043a \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u0445 \u0441\u043b\u043e\u0432<\/p>\n<\/li>\n<\/ol>\n<pre><code class=\"typescript\">interface WikiChunk {  id: string;         \/\/ \"authors\/jung\/shadow#\u0422\u0435\u043d\u044c\"  title: string;      \/\/ \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043e\u043a \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b  sourcePath: string; \/\/ \"authors\/jung\/shadow.md\"  section: string;    \/\/ \"## \u0422\u0435\u043d\u044c\"  text: string;       \/\/ \u0442\u0435\u043a\u0441\u0442 \u0441\u0435\u043a\u0446\u0438\u0438  keywords: string[]; \/\/ \u043f\u0440\u0435\u0434\u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u044b\u0435 \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0441\u043b\u043e\u0432\u0430}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041a\u043b\u044e\u0447\u0435\u0432\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u2014 \u0440\u0430\u0437\u0431\u0438\u0432\u043a\u0430 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b \u043d\u0430 \u0447\u0430\u043d\u043a\u0438:<\/p>\n<pre><code class=\"typescript\">function chunkPage(page: WikiPage): WikiChunk[] {  const chunks: WikiChunk[] = [];  \/\/ \u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u0435\u043a\u0446\u0438\u044e \"\u041c\u0430\u0442\u0435\u0440\u0438\u0430\u043b\u044b \u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438\" \u2014 \u043d\u0435 \u043d\u0443\u0436\u043d\u0430 \u0434\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430  const body = page.content.replace(\/## \u041c\u0430\u0442\u0435\u0440\u0438\u0430\u043b\u044b \u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438[\\s\\S]*$\/, '').trim();  \/\/ \u0420\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u043c \u043f\u043e ## \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430\u043c  const sections = body.split(\/(?=^## )\/m);  for (const section of sections) {    const headerMatch = section.match(\/^## (.+)$\/m);    const sectionName = headerMatch ? headerMatch[1].trim() : '';    const text = section.replace(\/^## .+\\n*\/m, '').trim();    if (!text || text.length &lt; 20) continue;    \/\/ \u041a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0441\u043b\u043e\u0432\u0430: \u0442\u043e\u043a\u0435\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430 + \u0441\u0435\u043a\u0446\u0438\u0438 + \u043f\u0435\u0440\u0432\u044b\u0445 500 \u0441\u0438\u043c\u0432\u043e\u043b\u043e\u0432 \u0442\u0435\u043a\u0441\u0442\u0430    const keywords = tokenize(`${page.title} ${sectionName} ${text.slice(0, 500)}`);    chunks.push({      id: `${page.path}#${sectionName}`,      title: page.title,      sourcePath: page.path,      section: sectionName,      text,      keywords,    });  }  return chunks;}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u043e\u043a\u0435\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u043f\u0440\u043e\u0441\u0442\u0430\u044f: \u0440\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u043c \u043d\u0430 \u0441\u043b\u043e\u0432\u0430, \u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u043c \u0441\u0442\u043e\u043f-\u0441\u043b\u043e\u0432\u0430 (\u0440\u0443\u0441\u0441\u043a\u0438\u0435 + \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u0438\u0435), \u0443\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043b\u043e\u0432\u0430 \u043a\u043e\u0440\u043e\u0447\u0435 3 \u0441\u0438\u043c\u0432\u043e\u043b\u043e\u0432, \u0434\u0435\u0434\u0443\u043f\u043b\u0438\u0446\u0438\u0440\u0443\u0435\u043c:<\/p>\n<pre><code class=\"typescript\">const STOPWORDS = new Set([  '\u0438', '\u0432', '\u0432\u043e', '\u043d\u0435', '\u0447\u0442\u043e', '\u043e\u043d', '\u043d\u0430', '\u044f', '\u0441', '\u0441\u043e',  \/\/ ... \u043f\u043e\u043b\u043d\u044b\u0439 \u0441\u043f\u0438\u0441\u043e\u043a \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438  'the', 'a', 'an', 'and', 'or', 'but', 'in', 'on',]);function tokenize(text: string): string[] {  const words = text.toLowerCase().match(\/[\u0430-\u044f\u0451a-z]+\/gi) || [];  return [...new Set(words.filter(w =&gt; w.length &gt; 2 &amp;&amp; !STOPWORDS.has(w)))];}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u2014 \u0444\u0430\u0439\u043b <code>src\/knowledge.ts<\/code> \u0441 \u043c\u0430\u0441\u0441\u0438\u0432\u043e\u043c <code>KNOWLEDGE_CHUNKS<\/code>. \u041f\u0440\u0438 29 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0430\u0445 wiki \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f ~256 \u0447\u0430\u043d\u043a\u043e\u0432.<\/p>\n<pre><code class=\"bash\">npm run build  # \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u0442 build-knowledge.ts \u0447\u0435\u0440\u0435\u0437 tsx<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<blockquote>\n<p><strong>\u0412\u0430\u0436\u043d\u043e:<\/strong> <code>knowledge.ts<\/code> \u2014 \u0430\u0432\u0442\u043e\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0444\u0430\u0439\u043b, \u0435\u0433\u043e \u043d\u0435 \u043d\u0443\u0436\u043d\u043e \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432\u0440\u0443\u0447\u043d\u0443\u044e. \u041a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u043f\u0440\u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u0438 wiki \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435 <code>npm run build<\/code> \u043f\u0435\u0440\u0435\u0434 \u0434\u0435\u043f\u043b\u043e\u0435\u043c.<\/p>\n<\/blockquote>\n<h3>\u0428\u0430\u0433 2. Retriever: \u043f\u043e\u0438\u0441\u043a \u043f\u043e \u0447\u0430\u043d\u043a\u0430\u043c<\/h3>\n<p>\u0424\u0430\u0439\u043b <code>src\/retriever.ts<\/code> \u2014 \u0441\u0435\u0440\u0434\u0446\u0435 \u0432\u0441\u0435\u0439 RAG-\u0441\u0438\u0441\u0442\u0435\u043c\u044b.<\/p>\n<p>\u0414\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c <strong>Jaccard-\u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0435 \u0441\u0445\u043e\u0434\u0441\u0442\u0432\u043e \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c<\/strong>:<\/p>\n<pre><code>score = |queryTokens \u2229 chunkKeywords| \/ |queryTokens \u222a chunkKeywords|<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0427\u0435\u043c \u0431\u043e\u043b\u044c\u0448\u0435 \u043e\u0431\u0449\u0438\u0445 \u0441\u043b\u043e\u0432 \u043c\u0435\u0436\u0434\u0443 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u043c \u0438 \u0447\u0430\u043d\u043a\u043e\u043c \u2014 \u0442\u0435\u043c \u0432\u044b\u0448\u0435 score. \u0411\u0435\u0440\u0451\u043c top-K \u0447\u0430\u043d\u043a\u043e\u0432 \u0441 \u043d\u0435\u043d\u0443\u043b\u0435\u0432\u044b\u043c score.<\/p>\n<pre><code class=\"typescript\">export function createRetriever(chunks: WikiChunk[], baseUrl: string): Retriever {  return {    retrieve(query: string, topK: number = 3): RetrievedChunk[] {      const queryTokens = tokenize(query);      if (queryTokens.length === 0) return [];      const scored = chunks.map(chunk =&gt; {        const overlap = queryTokens.filter(t =&gt; chunk.keywords.includes(t)).length;        const union = new Set([...queryTokens, ...chunk.keywords]);        const score = union.size &gt; 0 ? overlap \/ union.size : 0;        return { chunk, score };      });      return scored        .sort((a, b) =&gt; b.score - a.score)        .slice(0, topK)        .filter(c =&gt; c.score &gt; 0)        .map(c =&gt; ({ ...c.chunk }));    },    \/\/ ...  };}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p><strong>\u041f\u043e\u0447\u0435\u043c\u0443 \u043d\u0435 \u0432\u0435\u043a\u0442\u043e\u0440\u044b?<\/strong><\/p>\n<p>\u0412\u043e\u043f\u0440\u043e\u0441 \u0438\u0437 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u0435\u0432 \u043a \u043f\u0435\u0440\u0432\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u2014 \u043e\u0442\u0432\u0435\u0447\u0430\u044e: \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u043e\u0438\u0441\u043a \u0447\u0435\u0440\u0435\u0437 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u0438 \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u043b\u0443\u0447\u0448\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u0442 \u0441\u0438\u043d\u043e\u043d\u0438\u043c\u044b \u0438 \u0441\u043c\u044b\u0441\u043b\u043e\u0432\u044b\u0435 \u0441\u0432\u044f\u0437\u0438. \u041d\u043e \u0437\u0430 \u044d\u0442\u043e \u043d\u0443\u0436\u043d\u043e \u043f\u043b\u0430\u0442\u0438\u0442\u044c: API \u0434\u043b\u044f \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u043e\u0432, \u0432\u0435\u043a\u0442\u043e\u0440\u043d\u043e\u0435 \u0445\u0440\u0430\u043d\u0438\u043b\u0438\u0449\u0435, \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u0441\u0435\u0442\u0435\u0432\u043e\u0439 \u0432\u044b\u0437\u043e\u0432 \u043d\u0430 \u043a\u0430\u0436\u0434\u044b\u0439 \u0437\u0430\u043f\u0440\u043e\u0441.<\/p>\n<p>Jaccard \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c \u043e\u043f\u0440\u0430\u0432\u0434\u0430\u043d, \u043a\u043e\u0433\u0434\u0430:<\/p>\n<ul>\n<li>\n<p>\u041f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u0430\u044f \u043e\u0431\u043b\u0430\u0441\u0442\u044c \u0443\u0437\u043a\u0430\u044f \u0438 \u0438\u043c\u0435\u0435\u0442 \u0447\u0451\u0442\u043a\u0443\u044e \u0442\u0435\u0440\u043c\u0438\u043d\u043e\u043b\u043e\u0433\u0438\u044e<\/p>\n<\/li>\n<li>\n<p>\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u044e\u0442 \u0442\u0435\u0440\u043c\u0438\u043d\u044b \u0438\u0437 \u0441\u0430\u043c\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439<\/p>\n<\/li>\n<li>\n<p>\u041d\u0443\u0436\u043d\u0430 \u0434\u0435\u0442\u0435\u0440\u043c\u0438\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0441\u0442\u044c \u2014 \u043e\u0434\u0438\u043d \u0438 \u0442\u043e\u0442 \u0436\u0435 \u0437\u0430\u043f\u0440\u043e\u0441 \u0432\u0441\u0435\u0433\u0434\u0430 \u0434\u0430\u0451\u0442 \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u044b\u0439 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442<\/p>\n<\/li>\n<li>\n<p>\u0412\u0430\u0436\u043d\u0430 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u0430\u044f \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u0438 \u043d\u0443\u043b\u0435\u0432\u044b\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435 \u0440\u0430\u0441\u0445\u043e\u0434\u044b<\/p>\n<\/li>\n<\/ul>\n<p>\u0414\u043b\u044f \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u0447\u0435\u0441\u043a\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439 \u044d\u0442\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442: \u0437\u0430\u043f\u0440\u043e\u0441 \u00ab\u0442\u0440\u0435\u0432\u043e\u0433\u0430 \u0438 \u0441\u0442\u0440\u0430\u0445\u00bb \u043d\u0430\u0439\u0434\u0451\u0442 \u0447\u0430\u043d\u043a \u043f\u0440\u043e \u0441\u0442\u0440\u0430\u0445 \u0432 \u043b\u043e\u0433\u043e\u0442\u0435\u0440\u0430\u043f\u0438\u0438 \u0424\u0440\u0430\u043d\u043a\u043b\u0430, \u00ab\u043a\u043e\u043d\u0444\u043b\u0438\u043a\u0442 \u0432 \u043e\u0442\u043d\u043e\u0448\u0435\u043d\u0438\u044f\u0445\u00bb \u2014 \u0447\u0430\u043d\u043a \u043f\u0440\u043e \u0430\u043c\u043e\u0440\u0442\u0438\u0437\u0430\u0446\u0438\u044e \u043f\u043e \u041b\u0438\u0442\u0432\u0430\u043a\u0443. <a href=\"https:\/\/t.me\/pif_bbot\" rel=\"noopener noreferrer nofollow\">\u041f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u0441\u0430\u043c\u0438<\/a>.<\/p>\n<p>Retriever \u0442\u0430\u043a\u0436\u0435 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u043d\u044b\u0445 \u0447\u0430\u043d\u043a\u043e\u0432 \u0434\u043b\u044f LLM:<\/p>\n<pre><code class=\"typescript\">formatContext(entries: RetrievedChunk[]): string {  if (entries.length === 0) return '';  return entries.map((e, i) =&gt; {    const url = `${baseUrl}${wikiPathToUrl(e.sourcePath)}`;    return `[\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a ${i + 1}]: ${e.title} \u2192 ${url}&gt; ${e.section ? `*${e.section}*` : ''}&gt;${e.text.split('\\n').map(line =&gt; `&gt; ${line}`).join('\\n')}`;  }).join('\\n\\n---\\n\\n');},<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0418 \u0437\u0430 \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e URL \u0438\u0437 \u043f\u0443\u0442\u0438 \u043a \u0444\u0430\u0439\u043b\u0443:<\/p>\n<pre><code class=\"typescript\">function wikiPathToUrl(sourcePath: string): string {  const withoutExt = sourcePath.replace(\/\\.md$\/, '');  if (withoutExt.endsWith('\/index')) {    return '\/' + withoutExt.replace('\/index', '') + '\/';  }  return '\/' + withoutExt + '\/';}\/\/ \"authors\/jung\/shadow.md\" \u2192 \"\/authors\/jung\/shadow\/\"<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>\u0428\u0430\u0433 3. LLM-\u043a\u043b\u0438\u0435\u043d\u0442<\/h3>\n<p>\u0424\u0430\u0439\u043b <code>src\/llm.ts<\/code> \u2014 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u0438\u0441\u0442\u0438\u0447\u043d\u044b\u0439 \u0432\u0440\u0430\u043f\u043f\u0435\u0440 \u043d\u0430\u0434 Groq API.<\/p>\n<p>\u041d\u0438\u043a\u0430\u043a\u0438\u0445 SDK \u2014 \u0442\u043e\u043b\u044c\u043a\u043e <code>fetch<\/code>. \u042d\u0442\u043e \u043f\u0440\u0438\u043d\u0446\u0438\u043f\u0438\u0430\u043b\u044c\u043d\u043e \u0434\u043b\u044f Cloudflare Workers: \u043a\u0440\u0443\u043f\u043d\u044b\u0435 SDK \u0432\u0440\u043e\u0434\u0435 \u043e\u0444\u0438\u0446\u0438\u0430\u043b\u044c\u043d\u043e\u0433\u043e OpenAI-\u043a\u043b\u0438\u0435\u043d\u0442\u0430 \u043c\u043e\u0433\u0443\u0442 \u043d\u0435 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0442\u044c Workers runtime \u0438\u043b\u0438 \u0442\u0430\u0449\u0438\u0442\u044c \u0437\u0430 \u0441\u043e\u0431\u043e\u0439 \u043f\u043e\u043b\u0442\u043e\u043d\u043d\u044b \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u041f\u0440\u043e\u0441\u0442\u043e\u0439 <code>fetch<\/code>-\u0432\u0440\u0430\u043f\u043f\u0435\u0440 \u043d\u0430\u0434\u0451\u0436\u043d\u0435\u0435.<\/p>\n<pre><code class=\"typescript\">export function initLLM(config: LLMConfig): LLMClient {  return {    async chat(messages) {      const response = await fetch('https:\/\/api.groq.com\/openai\/v1\/chat\/completions', {        method: 'POST',        headers: {          'Authorization': `Bearer ${config.apiKey}`,          'Content-Type': 'application\/json',        },        body: JSON.stringify({          model: config.model,          messages,          temperature: 0.7,          max_tokens: 2048,        }),      });      if (!response.ok) {        const err = await response.text();        throw new Error(`Groq API error ${response.status}: ${err}`);      }      const data = await response.json() as any;      return data.choices?.[0]?.message?.content || '';    },  };}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>Groq \u0432\u044b\u0431\u0440\u0430\u043d \u043f\u043e \u0442\u0440\u0451\u043c \u043f\u0440\u0438\u0447\u0438\u043d\u0430\u043c: \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier \u0441 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u044b\u043c\u0438 \u043b\u0438\u043c\u0438\u0442\u0430\u043c\u0438, Llama-3.1-8b-instant \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 200\u2013500ms, API \u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c \u0441 OpenAI \u2014 \u043f\u0440\u0438 \u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043c\u0435\u043d\u044f\u0442\u044c \u043f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440 \u043e\u0434\u043d\u043e\u0439 \u0441\u0442\u0440\u043e\u043a\u043e\u0439. \u041c\u043e\u0434\u0435\u043b\u044c \u0437\u0430\u0434\u0430\u0451\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 env <code>GROQ_MODEL<\/code>, \u0442\u0430\u043a \u0447\u0442\u043e \u0434\u043b\u044f \u0441\u043c\u0435\u043d\u044b \u043c\u043e\u0434\u0435\u043b\u0438 \u043d\u0435 \u043d\u0443\u0436\u0435\u043d \u0440\u0435\u0434\u0435\u043f\u043b\u043e\u0439.<\/p>\n<h3>\u0428\u0430\u0433 4. \u0421\u0435\u0441\u0441\u0438\u0438 \u0432 Cloudflare KV<\/h3>\n<p>\u0424\u0430\u0439\u043b <code>src\/session.ts<\/code> \u2014 \u0438\u0441\u0442\u043e\u0440\u0438\u044f \u0434\u0438\u0430\u043b\u043e\u0433\u0430.<\/p>\n<p>Cloudflare Workers stateless: \u043a\u0430\u0436\u0434\u044b\u0439 \u0432\u0445\u043e\u0434\u044f\u0449\u0438\u0439 \u0437\u0430\u043f\u0440\u043e\u0441 \u2014 \u0447\u0438\u0441\u0442\u044b\u0439 \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442. \u0418\u0441\u0442\u043e\u0440\u0438\u044e \u0434\u0438\u0430\u043b\u043e\u0433\u0430 \u043d\u0443\u0436\u043d\u043e \u0445\u0440\u0430\u043d\u0438\u0442\u044c \u0441\u043d\u0430\u0440\u0443\u0436\u0438. KV \u2014 \u0438\u0434\u0435\u0430\u043b\u044c\u043d\u044b\u0439 \u0432\u044b\u0431\u043e\u0440: \u0433\u043b\u043e\u0431\u0430\u043b\u044c\u043d\u043e \u0440\u0430\u0441\u043f\u0440\u0435\u0434\u0435\u043b\u0451\u043d\u043d\u044b\u0439 key-value store, \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier \u0432\u043a\u043b\u044e\u0447\u0430\u0435\u0442 100K \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u0439 \u0447\u0442\u0435\u043d\u0438\u044f \u0432 \u0434\u0435\u043d\u044c.<\/p>\n<pre><code class=\"typescript\">const MAX_HISTORY = 20;export function initSessionStore(env: { SESSIONS?: KVNamespace }): SessionStore {  const kv = env.SESSIONS;  return {    async get(userId: number): Promise&lt;SessionMessage[]&gt; {      if (!kv) return [];      const raw = await kv.get(`session:${userId}`, 'text');      return raw ? JSON.parse(raw) : [];    },    async add(userId: number, message: SessionMessage): Promise&lt;void&gt; {      if (!kv) return;      const history = await this.get(userId);      history.push(message);      \/\/ \u0425\u0440\u0430\u043d\u0438\u043c \u043d\u0435 \u0431\u043e\u043b\u044c\u0448\u0435 20 \u043f\u043e\u0441\u043b\u0435\u0434\u043d\u0438\u0445 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439      const trimmed = history.slice(-MAX_HISTORY);      await kv.put(`session:${userId}`, JSON.stringify(trimmed), {        expirationTtl: 86400 * 7, \/\/ TTL 7 \u0434\u043d\u0435\u0439      });    },    async clear(userId: number): Promise&lt;void&gt; {      if (!kv) return;      await kv.delete(`session:${userId}`);    },  };}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041a\u043b\u044e\u0447 \u0441\u0435\u0441\u0441\u0438\u0438: <code>session:{telegramUserId}<\/code>. TTL 7 \u0434\u043d\u0435\u0439 \u2014 \u0441\u0442\u0430\u0440\u044b\u0435 \u0441\u0435\u0441\u0441\u0438\u0438 \u0443\u0434\u0430\u043b\u044f\u044e\u0442\u0441\u044f \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438, \u0440\u0443\u0447\u043d\u0430\u044f \u043e\u0447\u0438\u0441\u0442\u043a\u0430 \u043d\u0435 \u043d\u0443\u0436\u043d\u0430.<\/p>\n<p>\u041b\u0438\u043c\u0438\u0442 \u0432 20 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439 \u2014 \u0437\u0430\u0449\u0438\u0442\u0430 \u043e\u0442 \u043f\u0435\u0440\u0435\u043f\u043e\u043b\u043d\u0435\u043d\u0438\u044f \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430 LLM. \u0415\u0441\u043b\u0438 \u0438\u0441\u0442\u043e\u0440\u0438\u044f \u0432\u0441\u0451 \u0440\u0430\u0432\u043d\u043e \u043e\u043a\u0430\u0437\u0430\u043b\u0430\u0441\u044c \u0441\u043b\u0438\u0448\u043a\u043e\u043c \u0431\u043e\u043b\u044c\u0448\u043e\u0439 (Groq \u0432\u0435\u0440\u043d\u0443\u043b <code>Request too large<\/code>), \u0431\u043e\u0442 \u0441\u043e\u043e\u0431\u0449\u0430\u0435\u0442 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044e \u0438 \u043f\u0440\u0435\u0434\u043b\u0430\u0433\u0430\u0435\u0442 \u043d\u0430\u043f\u0438\u0441\u0430\u0442\u044c <code>\/clear<\/code>.<\/p>\n<h3>\u0428\u0430\u0433 5. System prompt \u0438 \u0441\u0431\u043e\u0440\u043a\u0430 \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430<\/h3>\n<p>\u0424\u0430\u0439\u043b <code>src\/prompts.ts<\/code> \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u044f\u0435\u0442 \u043b\u0438\u0447\u043d\u043e\u0441\u0442\u044c \u0438 \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u0431\u043e\u0442\u0430.<\/p>\n<pre><code class=\"typescript\">export function SYSTEM_PROMPT(): string {  return `\u0422\u044b \u2014 \u041f\u0438\u0424, \u044d\u043c\u043f\u0430\u0442\u0438\u0447\u043d\u044b\u0439 \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u043e\u043c\u043e\u0449\u043d\u0438\u043a.\u0411\u0430\u0437\u0430 \u0437\u043d\u0430\u043d\u0438\u0439: \u041d\u041d\u041e (\u0420\u043e\u0437\u0435\u043d\u0431\u0435\u0440\u0433), \u042e\u043d\u0433, \u0424\u0440\u0430\u043d\u043a\u043b, \u0423\u0438\u043b\u0431\u0435\u0440, \u041c\u0438\u043d\u0434\u0435\u043b\u043b, \u0410\u0434\u0438\u0437\u0435\u0441, \u041b\u0438\u0442\u0432\u0430\u043a.## \u041f\u0440\u0430\u0432\u0438\u043b\u0430### \u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043e\u0442\u0432\u0435\u0442\u0430- \u0412\u0430\u043b\u0438\u0434\u0430\u0446\u0438\u044f \u2014 \u043e\u0442\u0440\u0430\u0437\u0438 \u0447\u0443\u0432\u0441\u0442\u0432\u0430- \u041d\u0430\u0431\u043b\u044e\u0434\u0435\u043d\u0438\u0435 \u2014 \u0444\u0430\u043a\u0442\u044b \u0431\u0435\u0437 \u043e\u0446\u0435\u043d\u043e\u043a- \u041a\u043e\u043d\u0446\u0435\u043f\u0446\u0438\u044f \u2014 1-2 \u043f\u0440\u0435\u0434\u043b\u043e\u0436\u0435\u043d\u0438\u044f \u0438\u0437 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439- \u0412\u043e\u043f\u0440\u043e\u0441 \u2014 \u043e\u0442\u043a\u0440\u044b\u0442\u044b\u0439 \u0432\u043e\u043f\u0440\u043e\u0441 \u0438\u043b\u0438 \u0442\u0435\u0445\u043d\u0438\u043a\u0430### \u0426\u0438\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435\u041a\u043e\u0433\u0434\u0430 \u0442\u0435\u0431\u0435 \u043f\u0435\u0440\u0435\u0434\u0430\u043d\u044b \u0441\u0442\u0430\u0442\u044c\u0438 \u0432 \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0435:- \u0418\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439 \u0422\u041e\u041b\u042c\u041a\u041e URL, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0434\u0430\u043d\u044b \u0432 \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0435 \u2014 \u043a\u043e\u043f\u0438\u0440\u0443\u0439 \u043a\u0430\u043a \u0435\u0441\u0442\u044c- \u0424\u043e\u0440\u043c\u0430\u0442 \u0446\u0438\u0442\u0430\u0442\u044b: &gt; \u0442\u0435\u043a\u0441\u0442\\n&gt; -- [\u041d\u0430\u0437\u0432\u0430\u043d\u0438\u0435](URL)- \u041d\u0418\u041a\u041e\u0413\u0414\u0410 \u043d\u0435 \u0432\u044b\u0434\u0443\u043c\u044b\u0432\u0430\u0439 \u0446\u0438\u0442\u0430\u0442\u044b### \u0411\u0435\u0437\u043e\u043f\u0430\u0441\u043d\u043e\u0441\u0442\u044c\u041f\u0440\u0438 \u0441\u0443\u0438\u0446\u0438\u0434\u0435\/\u0441\u0430\u043c\u043e\u043f\u043e\u0432\u0440\u0435\u0436\u0434\u0435\u043d\u0438\u0438: \u00ab\u041f\u043e\u0436\u0430\u043b\u0443\u0439\u0441\u0442\u0430, \u043f\u043e\u0437\u0432\u043e\u043d\u0438 112 \u0438\u043b\u0438 8-800-2000-122\u00bb.  `;}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u0441\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435 \u2014 \u043a\u0430\u043a \u0431\u043e\u0442 \u0441\u043e\u0431\u0438\u0440\u0430\u0435\u0442 \u0437\u0430\u043f\u0440\u043e\u0441 \u043a LLM \u0432 <code>src\/bot.ts<\/code>:<\/p>\n<pre><code class=\"typescript\">bot.on('text', async (ctx) =&gt; {  const userId = ctx.from.id;  const userMessage = ctx.message.text;  await ctx.sendChatAction('typing');  \/\/ 1. \u041f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0438\u0441\u0442\u043e\u0440\u0438\u044e \u0434\u0438\u0430\u043b\u043e\u0433\u0430 \u0438\u0437 KV  const history = await config.sessions.get(userId);  \/\/ 2. \u0418\u0449\u0435\u043c \u0440\u0435\u043b\u0435\u0432\u0430\u043d\u0442\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 \u0432 \u0431\u0430\u0437\u0435 \u0437\u043d\u0430\u043d\u0438\u0439  const relevant = config.retriever.retrieve(userMessage, 2);  const knowledgeContext = config.retriever.formatContext(relevant);  \/\/ 3. \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c messages \u0434\u043b\u044f LLM  const messages = [    { role: 'system', content: config.systemPrompt },    ...history.map(m =&gt; ({ role: m.role, content: m.content })),  ];  \/\/ 4. \u0418\u043d\u0436\u0435\u043a\u0442\u0438\u0440\u0443\u0435\u043c RAG-\u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442 \u0432 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f  const userContent = knowledgeContext    ? `\u041d\u0430\u0439\u0434\u0435\u043d\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 (\u0446\u0438\u0442\u0438\u0440\u0443\u0439 \u0438\u0445):\\n\\n${knowledgeContext}\\n\\n\u0412\u043e\u043f\u0440\u043e\u0441 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f: ${userMessage}`    : userMessage;  messages.push({ role: 'user', content: userContent });  \/\/ 5. \u0417\u0430\u043f\u0440\u043e\u0441 \u043a LLM  const response = await config.llm.chat(messages);  \/\/ 6. \u0421\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u043c \u0432 \u0438\u0441\u0442\u043e\u0440\u0438\u044e (\u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b\u044c\u043d\u043e\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435, \u0431\u0435\u0437 RAG-\u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430)  await config.sessions.add(userId, { role: 'user', content: userMessage, timestamp: Date.now() });  await config.sessions.add(userId, { role: 'assistant', content: response, timestamp: Date.now() });  await ctx.reply(response, { parse_mode: 'Markdown' });});<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041e\u0431\u0440\u0430\u0442\u0438\u0442\u0435 \u0432\u043d\u0438\u043c\u0430\u043d\u0438\u0435 \u043d\u0430 \u0448\u0430\u0433 6: \u0432 \u0438\u0441\u0442\u043e\u0440\u0438\u044e \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u0442\u0441\u044f <strong>\u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b\u044c\u043d\u043e\u0435<\/strong> \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f, \u0431\u0435\u0437 RAG-\u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442\u0430. \u042d\u0442\u043e \u0432\u0430\u0436\u043d\u043e \u2014 \u0438\u043d\u0430\u0447\u0435 \u0438\u0441\u0442\u043e\u0440\u0438\u044f \u0440\u0430\u0437\u0434\u0443\u0435\u0442\u0441\u044f \u043e\u0447\u0435\u043d\u044c \u0431\u044b\u0441\u0442\u0440\u043e. \u041a\u0430\u0436\u0434\u043e\u0435 \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0435\u0435 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u043f\u043e\u0442\u044f\u043d\u0443\u043b\u043e \u0431\u044b \u0437\u0430 \u0441\u043e\u0431\u043e\u0439 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u0441\u0442\u0430\u0442\u0435\u0439 \u0438\u0437 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439, \u0438 \u0447\u0435\u0440\u0435\u0437 \u043d\u0435\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043e\u0431\u043c\u0435\u043d\u043e\u0432 \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442 LLM \u043f\u0435\u0440\u0435\u043f\u043e\u043b\u043d\u0438\u043b\u0441\u044f \u0431\u044b.<\/p>\n<p>\u0412\u043e\u0442 \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442 LLM \u0432 <code>userContent<\/code>:<\/p>\n<pre><code>\u041d\u0430\u0439\u0434\u0435\u043d\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 (\u0446\u0438\u0442\u0438\u0440\u0443\u0439 \u0438\u0445):[\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a 1]: \u0422\u0435\u043d\u044c (\u042e\u043d\u0433) \u2192 https:\/\/anatolii-iumashev.github.io\/pifai\/authors\/jung\/shadow\/&gt; *## \u0427\u0442\u043e \u0442\u0430\u043a\u043e\u0435 \u0422\u0435\u043d\u044c*&gt;&gt; \u0422\u0435\u043d\u044c \u2014 \u044d\u0442\u043e \u0442\u0430 \u0447\u0430\u0441\u0442\u044c \u043d\u0430\u0448\u0435\u0439 \u043b\u0438\u0447\u043d\u043e\u0441\u0442\u0438, \u043a\u043e\u0442\u043e\u0440\u0443\u044e \u043c\u044b \u043e\u0442\u0432\u0435\u0440\u0433\u0430\u0435\u043c...---[\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a 2]: \u042d\u043c\u043e\u0446\u0438\u0438 \u0438 \u043f\u043e\u0442\u0440\u0435\u0431\u043d\u043e\u0441\u0442\u0438 \u2192 https:\/\/anatolii-iumashev.github.io\/pifai\/basics\/emotions\/&gt; *## \u0427\u0443\u0432\u0441\u0442\u0432\u0430 \u043a\u0430\u043a \u0441\u0438\u0433\u043d\u0430\u043b*&gt;&gt; \u0412 \u041d\u041d\u041e \u0447\u0443\u0432\u0441\u0442\u0432\u0430 \u2014 \u044d\u0442\u043e \u0438\u043d\u0434\u0438\u043a\u0430\u0442\u043e\u0440 \u0443\u0434\u043e\u0432\u043b\u0435\u0442\u0432\u043e\u0440\u0451\u043d\u043d\u043e\u0441\u0442\u0438 \u043f\u043e\u0442\u0440\u0435\u0431\u043d\u043e\u0441\u0442\u0435\u0439...\u0412\u043e\u043f\u0440\u043e\u0441 \u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u044f: \u043f\u043e\u0447\u0435\u043c\u0443 \u044f \u0437\u043b\u044e\u0441\u044c \u043d\u0430 \u0431\u043b\u0438\u0437\u043a\u0438\u0445 \u0431\u0435\u0437 \u043f\u0440\u0438\u0447\u0438\u043d\u044b?<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h3>\u0428\u0430\u0433 6. \u0422\u043e\u0447\u043a\u0430 \u0432\u0445\u043e\u0434\u0430: Cloudflare Workers<\/h3>\n<p>\u0424\u0430\u0439\u043b <code>src\/index.ts<\/code> \u2014 HTTP-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a \u0434\u043b\u044f Workers.<\/p>\n<pre><code class=\"typescript\">let botInstance: ReturnType&lt;typeof createBot&gt; | null = null;export default {  async fetch(request: Request, env: Env): Promise&lt;Response&gt; {    const url = new URL(request.url);    \/\/ Health check    if (request.method === 'GET' &amp;&amp; url.pathname === '\/health') {      return new Response(JSON.stringify({        status: 'ok',        knowledgeVersion: env.KNOWLEDGE_VERSION || '1.0.0',      }), { headers: { 'Content-Type': 'application\/json' } });    }    \/\/ Telegram webhook    if (request.method === 'POST' &amp;&amp; url.pathname === '\/webhook') {      \/\/ Lazy init \u2014 \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0431\u043e\u0442\u0430 \u043e\u0434\u0438\u043d \u0440\u0430\u0437      if (!botInstance) {        const llm = initLLM({ apiKey: env.GROQ_API_KEY, model: env.GROQ_MODEL || 'llama-3.1-8b-instant' });        const sessions = initSessionStore(env);        const retriever = createRetriever(KNOWLEDGE_CHUNKS, env.KNOWLEDGE_BASE_URL);        botInstance = createBot({ token: env.TELEGRAM_BOT_TOKEN, llm, sessions, systemPrompt: SYSTEM_PROMPT(), retriever });      }      const update = await request.json() as any;      await botInstance.handleUpdate(update);      return new Response('ok', { status: 200 });    }    return new Response('Not found', { status: 404 });  },};<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0414\u0432\u0430 \u043c\u043e\u043c\u0435\u043d\u0442\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0435 \u0432\u0430\u0436\u043d\u043e \u043f\u043e\u043d\u044f\u0442\u044c:<\/p>\n<p><strong>Lazy init.<\/strong> \u0412\u043e\u0440\u043a\u0435\u0440 \u0438\u043d\u0438\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u043f\u0440\u0438 \u043f\u0435\u0440\u0432\u043e\u043c \u0437\u0430\u043f\u0440\u043e\u0441\u0435 \u0438 \u043f\u0435\u0440\u0435\u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 \u044d\u043a\u0437\u0435\u043c\u043f\u043b\u044f\u0440 <code>botInstance<\/code>. Cloudflare Workers \u043d\u0435 \u0433\u0430\u0440\u0430\u043d\u0442\u0438\u0440\u0443\u0435\u0442, \u0447\u0442\u043e \u043e\u0434\u0438\u043d \u0438 \u0442\u043e\u0442 \u0436\u0435 \u0438\u043d\u0441\u0442\u0430\u043d\u0441 \u0431\u0443\u0434\u0435\u0442 \u0436\u0438\u0442\u044c \u0432\u0435\u0447\u043d\u043e, \u043d\u043e \u043d\u0430 \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435 \u043f\u0440\u0438 \u0440\u0435\u0433\u0443\u043b\u044f\u0440\u043d\u043e\u043c \u0442\u0440\u0430\u0444\u0438\u043a\u0435 \u043e\u043d \u0436\u0438\u0432\u0451\u0442 \u0434\u043e\u043b\u0433\u043e \u2014 cold start \u0441\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u0440\u0435\u0434\u043a\u043e.<\/p>\n<p><strong>\u0412\u0441\u0435\u0433\u0434\u0430 200 \u0434\u043b\u044f Telegram.<\/strong> \u0415\u0441\u043b\u0438 \u0432\u0435\u0440\u043d\u0443\u0442\u044c 4xx\/5xx, Telegram \u043d\u0430\u0447\u043d\u0451\u0442 \u043f\u043e\u0432\u0442\u043e\u0440\u044f\u0442\u044c \u0437\u0430\u043f\u0440\u043e\u0441 \u0441 \u043d\u0430\u0440\u0430\u0441\u0442\u0430\u044e\u0449\u0438\u043c\u0438 \u0438\u043d\u0442\u0435\u0440\u0432\u0430\u043b\u0430\u043c\u0438. \u041c\u044b \u0432\u043e\u0437\u0432\u0440\u0430\u0449\u0430\u0435\u043c 200 \u0434\u0430\u0436\u0435 \u043f\u0440\u0438 \u043e\u0448\u0438\u0431\u043a\u0435 \u2014 Telegram \u0441\u0447\u0438\u0442\u0430\u0435\u0442, \u0447\u0442\u043e \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0435 \u0434\u043e\u0441\u0442\u0430\u0432\u043b\u0435\u043d\u043e, \u0438 \u043d\u0435 \u0437\u0430\u0441\u044b\u043f\u0430\u0435\u0442 \u0431\u043e\u0442\u0430 \u0440\u0435\u0442\u0440\u0430\u044f\u043c\u0438.<\/p>\n<h3>\u041a\u043e\u043d\u0444\u0438\u0433\u0443\u0440\u0430\u0446\u0438\u044f: wrangler.toml<\/h3>\n<pre><code>name = \"pif-bot\"main = \"src\/index.ts\"compatibility_date = \"2026-05-01\"compatibility_flags = [\"nodejs_compat\"]# KV \u0434\u043b\u044f \u0441\u0435\u0441\u0441\u0438\u0439[[kv_namespaces]]binding = \"SESSIONS\"id = \"your-kv-namespace-id\"# \u041f\u0435\u0440\u0435\u043c\u0435\u043d\u043d\u044b\u0435 \u043e\u043a\u0440\u0443\u0436\u0435\u043d\u0438\u044f[vars]GROQ_MODEL = \"llama-3.1-8b-instant\"KNOWLEDGE_VERSION = \"1.0.0\"KNOWLEDGE_BASE_URL = \"https:\/\/anatolii-iumashev.github.io\/pifai\"<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0424\u043b\u0430\u0433 <code>nodejs_compat<\/code> \u043d\u0443\u0436\u0435\u043d, \u043f\u043e\u0442\u043e\u043c\u0443 \u0447\u0442\u043e Telegraf \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u0442 \u043d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0435 Node.js API. \u0411\u0435\u0437 \u043d\u0435\u0433\u043e \u043f\u0440\u0438 \u0434\u0435\u043f\u043b\u043e\u0435 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u0435 \u043e\u0448\u0438\u0431\u043a\u0438.<\/p>\n<h3>\u0414\u0435\u043f\u043b\u043e\u0439<\/h3>\n<h4>1. \u0421\u043e\u0437\u0434\u0430\u0451\u043c KV namespace<\/h4>\n<pre><code class=\"bash\">npx wrangler kv namespace create SESSIONS<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0411\u0435\u0440\u0451\u043c <code>id<\/code> \u0438\u0437 \u0432\u044b\u0432\u043e\u0434\u0430 \u0438 \u043f\u0440\u043e\u043f\u0438\u0441\u044b\u0432\u0430\u0435\u043c \u0432 <code>wrangler.toml<\/code>.<\/p>\n<h4>2. \u0414\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u0441\u0435\u043a\u0440\u0435\u0442\u044b<\/h4>\n<pre><code class=\"bash\">npx wrangler secret put TELEGRAM_BOT_TOKEN# \u0432\u0432\u043e\u0434\u0438\u043c \u0442\u043e\u043a\u0435\u043d \u0431\u043e\u0442\u0430npx wrangler secret put GROQ_API_KEY# \u0432\u0432\u043e\u0434\u0438\u043c \u043a\u043b\u044e\u0447 Groq<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<h4>3. \u0421\u043e\u0431\u0438\u0440\u0430\u0435\u043c \u0431\u0430\u0437\u0443 \u0437\u043d\u0430\u043d\u0438\u0439 \u0438 \u0434\u0435\u043f\u043b\u043e\u0438\u043c<\/h4>\n<pre><code class=\"bash\">npm run build    # \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442 src\/knowledge.ts \u0438\u0437 wikinpm run deploy   # wrangler deploy<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u043e\u0441\u043b\u0435 \u0434\u0435\u043f\u043b\u043e\u044f Wrangler \u0432\u044b\u0432\u0435\u0434\u0435\u0442 URL \u0432\u043e\u0440\u043a\u0435\u0440\u0430 \u0432\u0438\u0434\u0430 <code>https:\/\/pif-bot.username.workers.dev<\/code>.<\/p>\n<h4>4. \u0420\u0435\u0433\u0438\u0441\u0442\u0440\u0438\u0440\u0443\u0435\u043c webhook<\/h4>\n<pre><code class=\"bash\">curl \"https:\/\/api.telegram.org\/bot{TELEGRAM_BOT_TOKEN}\/setWebhook?url=https:\/\/pif-bot.username.workers.dev\/webhook\"<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c:<\/p>\n<pre><code class=\"bash\">curl \"https:\/\/api.telegram.org\/bot{TELEGRAM_BOT_TOKEN}\/getWebhookInfo\"<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0414\u043e\u043b\u0436\u043d\u044b \u0443\u0432\u0438\u0434\u0435\u0442\u044c <code>\"url\": \"https:\/\/pif-bot.username.workers.dev\/webhook\"<\/code> \u0438 <code>\"pending_update_count\": 0<\/code>.<\/p>\n<h4>5. \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c health<\/h4>\n<pre><code class=\"bash\">curl https:\/\/pif-bot.username.workers.dev\/health# {\"status\":\"ok\",\"knowledgeVersion\":\"1.0.0\"}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412\u0441\u0451 \u2014 \u0431\u043e\u0442 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442. \u041e\u0442\u043a\u0440\u044b\u0432\u0430\u0435\u043c Telegram \u0438 \u043f\u0438\u0448\u0435\u043c.<\/p>\n<h3>\u041b\u043e\u043a\u0430\u043b\u044c\u043d\u0430\u044f \u0440\u0430\u0437\u0440\u0430\u0431\u043e\u0442\u043a\u0430<\/h3>\n<p>\u0414\u043b\u044f \u0442\u0435\u0441\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f \u0431\u0435\u0437 \u0434\u0435\u043f\u043b\u043e\u044f \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0444\u0430\u0439\u043b <code>bot\/.dev.vars<\/code>:<\/p>\n<pre><code>TELEGRAM_BOT_TOKEN=your_token_hereGROQ_API_KEY=your_key_here<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0418 \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c:<\/p>\n<pre><code class=\"bash\">npm run dev# node --env-file=.dev.vars --import tsx src\/index.ts<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0412 \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u043e\u043c \u0440\u0435\u0436\u0438\u043c\u0435 Workers-\u0441\u0440\u0435\u0434\u044b \u043d\u0435\u0442, KV \u0442\u043e\u0436\u0435 \u043d\u0435\u0442 \u2014 \u0438\u0441\u0442\u043e\u0440\u0438\u044f \u043d\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u0442\u0441\u044f. \u041d\u043e LLM-\u043e\u0442\u0432\u0435\u0442\u044b \u0441 RAG \u0440\u0430\u0431\u043e\u0442\u0430\u044e\u0442. \u0414\u043b\u044f \u043e\u0442\u043b\u0430\u0434\u043a\u0438 webhook \u043b\u043e\u043a\u0430\u043b\u044c\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0439\u0442\u0435 <a href=\"https:\/\/ngrok.com\" rel=\"noopener noreferrer nofollow\">ngrok<\/a> \u0438\u043b\u0438 <code>wrangler dev<\/code> \u0441 \u0442\u0443\u043d\u043d\u0435\u043b\u0435\u043c.<\/p>\n<h3>\u041e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u0435 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439<\/h3>\n<p>\u041e\u0434\u0438\u043d \u0438\u0437 \u043d\u0435\u043e\u0447\u0435\u0432\u0438\u0434\u043d\u044b\u0445 \u043f\u043b\u044e\u0441\u043e\u0432 \u0442\u0430\u043a\u043e\u0433\u043e \u043f\u043e\u0434\u0445\u043e\u0434\u0430 \u2014 \u043d\u0430\u0441\u043a\u043e\u043b\u044c\u043a\u043e \u043f\u0440\u043e\u0441\u0442\u043e \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0442\u044c \u0437\u043d\u0430\u043d\u0438\u044f \u0431\u043e\u0442\u0430. \u0414\u043e\u0431\u0430\u0432\u0438\u043b\u0438 \u0441\u0442\u0430\u0442\u044c\u044e \u0432 wiki:<\/p>\n<pre><code class=\"bash\">npm run build   # \u043f\u0435\u0440\u0435\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442 knowledge.tsnpm run deploy  # \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442 \u043e\u0431\u043d\u043e\u0432\u043b\u0451\u043d\u043d\u044b\u0439 \u0432\u043e\u0440\u043a\u0435\u0440<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0414\u0432\u0430 \u0448\u0430\u0433\u0430. \u041d\u0438\u043a\u0430\u043a\u0438\u0445 \u043c\u0438\u0433\u0440\u0430\u0446\u0438\u0439, \u043d\u0438\u043a\u0430\u043a\u043e\u0433\u043e \u043f\u0435\u0440\u0435\u0438\u043d\u0434\u0435\u043a\u0441\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u044f, \u043d\u0438\u043a\u0430\u043a\u0438\u0445 embedding-\u0431\u0430\u0442\u0447\u0435\u0439.<\/p>\n<p>\u0415\u0441\u043b\u0438 wiki \u0436\u0438\u0432\u0451\u0442 \u0432 \u043e\u0442\u0434\u0435\u043b\u044c\u043d\u043e\u043c \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438 \u0438\u043b\u0438 \u043a\u0430\u043a submodule, \u044d\u0442\u043e \u043b\u0435\u0433\u043a\u043e \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0437\u0438\u0440\u0443\u0435\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 GitHub Actions:<\/p>\n<pre><code class=\"yaml\">on:  push:    paths:      - 'src\/content\/docs\/**'jobs:  deploy-bot:    runs-on: ubuntu-latest    steps:      - uses: actions\/checkout@v4      - run: npm ci      - run: npm run build      - run: npm run deploy        env:          CLOUDFLARE_API_TOKEN: ${{ secrets.CF_API_TOKEN }}<\/code><div class=\"code-explainer\"><a href=\"https:\/\/sourcecraft.dev\/\" class=\"tm-button code-explainer__link\" style=\"visibility: hidden;\"><img style=\"width:14px;height:14px;object-fit:cover;object-position:left;\"\/><\/a><\/div><\/pre>\n<p>\u0422\u0435\u043f\u0435\u0440\u044c \u043a\u0430\u0436\u0434\u044b\u0439 \u043a\u043e\u043c\u043c\u0438\u0442 \u0432 wiki \u0430\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u044f\u0435\u0442 \u0437\u043d\u0430\u043d\u0438\u044f \u0431\u043e\u0442\u0430.<\/p>\n<h3>\u0427\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u043e\u0441\u044c<\/h3>\n<p>\u0418\u0442\u043e\u0433\u043e\u0432\u044b\u0439 \u0431\u043e\u0442 <a href=\"https:\/\/t.me\/pif_bbot\" rel=\"noopener noreferrer nofollow\">@pif_bbot<\/a> \u0443\u043c\u0435\u0435\u0442:<\/p>\n<ul>\n<li>\n<p><strong>\u041d\u0430\u0445\u043e\u0434\u0438\u0442\u044c \u0440\u0435\u043b\u0435\u0432\u0430\u043d\u0442\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438<\/strong> \u043f\u043e \u0437\u0430\u043f\u0440\u043e\u0441\u0443 \u2014 \u0434\u0430\u0436\u0435 \u043f\u0440\u0438 \u043d\u0435\u0442\u043e\u0447\u043d\u044b\u0445 \u0444\u043e\u0440\u043c\u0443\u043b\u0438\u0440\u043e\u0432\u043a\u0430\u0445<\/p>\n<\/li>\n<li>\n<p><strong>\u0426\u0438\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438<\/strong> \u0441\u043e \u0441\u0441\u044b\u043b\u043a\u0430\u043c\u0438 \u043d\u0430 <a href=\"https:\/\/anatolii-iumashev.github.io\/pifai\/\" rel=\"noopener noreferrer nofollow\">\u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b wiki<\/a><\/p>\n<\/li>\n<li>\n<p><strong>\u041f\u043e\u043c\u043d\u0438\u0442\u044c \u043a\u043e\u043d\u0442\u0435\u043a\u0441\u0442 \u0434\u0438\u0430\u043b\u043e\u0433\u0430<\/strong> \u2014 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0441 \u0443\u0447\u0451\u0442\u043e\u043c \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0438\u0445 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439<\/p>\n<\/li>\n<li>\n<p><strong>\u0421\u0431\u0440\u0430\u0441\u044b\u0432\u0430\u0442\u044c \u0438\u0441\u0442\u043e\u0440\u0438\u044e<\/strong> \u043a\u043e\u043c\u0430\u043d\u0434\u043e\u0439 <code>\/clear<\/code><\/p>\n<\/li>\n<li>\n<p><strong>\u0420\u0435\u0430\u0433\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u043d\u0430 \u043a\u0440\u0438\u0437\u0438\u0441\u043d\u044b\u0435 \u0441\u0438\u0442\u0443\u0430\u0446\u0438\u0438<\/strong> \u2014 \u0441\u0440\u0430\u0437\u0443 \u0434\u0430\u0432\u0430\u0442\u044c \u043d\u043e\u043c\u0435\u0440\u0430 \u0442\u0435\u043b\u0435\u0444\u043e\u043d\u043e\u0432 \u0434\u043e\u0432\u0435\u0440\u0438\u044f<\/p>\n<\/li>\n<\/ul>\n<p>\u041f\u0440\u0438 \u044d\u0442\u043e\u043c \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u0430\u044f: Cloudflare Workers free tier + Groq free tier = <strong>~0\u20bd\/\u043c\u0435\u0441<\/strong> \u043f\u0440\u0438 \u0443\u043c\u0435\u0440\u0435\u043d\u043d\u043e\u0439 \u043d\u0430\u0433\u0440\u0443\u0437\u043a\u0435. \u0414\u043b\u044f \u043b\u0438\u0447\u043d\u043e\u0433\u043e \u043f\u0440\u043e\u0435\u043a\u0442\u0430 \u0438\u043b\u0438 \u043d\u0435\u0431\u043e\u043b\u044c\u0448\u043e\u0433\u043e \u0441\u043e\u043e\u0431\u0449\u0435\u0441\u0442\u0432\u0430 \u2014 \u0438\u0434\u0435\u0430\u043b\u044c\u043d\u043e.<\/p>\n<h3>\u0427\u0442\u043e \u043c\u043e\u0436\u043d\u043e \u0443\u043b\u0443\u0447\u0448\u0438\u0442\u044c<\/h3>\n<p><strong>\u0421\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u043e\u0438\u0441\u043a.<\/strong> Jaccard \u0445\u043e\u0440\u043e\u0448\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u0434\u043b\u044f \u043f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u044b\u0445 \u043e\u0431\u043b\u0430\u0441\u0442\u0435\u0439 \u0441 \u0443\u0441\u0442\u043e\u0439\u0447\u0438\u0432\u043e\u0439 \u0442\u0435\u0440\u043c\u0438\u043d\u043e\u043b\u043e\u0433\u0438\u0435\u0439. \u0414\u043b\u044f \u0431\u043e\u043b\u0435\u0435 \u0440\u0430\u0437\u043c\u044b\u0442\u044b\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u0441\u0442\u043e\u0438\u0442 \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0435\u0442\u044c \u043d\u0430 <a href=\"https:\/\/developers.cloudflare.com\/vectorize\/\" rel=\"noopener noreferrer nofollow\">Cloudflare Vectorize<\/a> \u0441 Workers AI \u0434\u043b\u044f \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u043e\u0432 \u2014 \u0432\u0441\u0451 \u0432 \u0440\u0430\u043c\u043a\u0430\u0445 \u0442\u043e\u0439 \u0436\u0435 \u043f\u043b\u0430\u0442\u0444\u043e\u0440\u043c\u044b, \u043d\u0438\u043a\u0430\u043a\u0438\u0445 \u0432\u043d\u0435\u0448\u043d\u0438\u0445 \u0441\u0435\u0440\u0432\u0438\u0441\u043e\u0432.<\/p>\n<p><strong>\u0410\u0432\u0442\u043e\u043c\u0430\u0442\u0438\u0447\u0435\u0441\u043a\u043e\u0435 \u043e\u043f\u0440\u0435\u0434\u0435\u043b\u0435\u043d\u0438\u0435 \u043a\u0440\u0438\u0437\u0438\u0441\u043d\u044b\u0445 \u0441\u043e\u0441\u0442\u043e\u044f\u043d\u0438\u0439.<\/strong> \u0412 <code>prompts.ts<\/code> \u0443\u0436\u0435 \u0435\u0441\u0442\u044c <code>CRISIS_DETECTION_PROMPT()<\/code> \u2014 \u043f\u0440\u043e\u043c\u043f\u0442 \u0434\u043b\u044f \u043a\u043b\u0430\u0441\u0441\u0438\u0444\u0438\u043a\u0430\u0446\u0438\u0438 \u0441\u043e\u043e\u0431\u0449\u0435\u043d\u0438\u0439. \u041c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u043f\u0440\u0435\u0434\u0432\u0430\u0440\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u0432\u044b\u0437\u043e\u0432 LLM \u043f\u0435\u0440\u0435\u0434 \u043e\u0441\u043d\u043e\u0432\u043d\u044b\u043c \u043e\u0442\u0432\u0435\u0442\u043e\u043c: \u0435\u0441\u043b\u0438 \u0431\u043e\u0442 \u0440\u0430\u0441\u043f\u043e\u0437\u043d\u0430\u043b \u043a\u0440\u0438\u0437\u0438\u0441 \u2014 \u0441\u0440\u0430\u0437\u0443 \u043f\u0435\u0440\u0435\u043a\u043b\u044e\u0447\u0430\u0435\u0442\u0441\u044f \u043d\u0430 \u043a\u0440\u0438\u0437\u0438\u0441\u043d\u044b\u0439 \u0441\u0446\u0435\u043d\u0430\u0440\u0438\u0439, \u043d\u0435 \u0434\u043e\u0436\u0438\u0434\u0430\u044f\u0441\u044c \u043a\u043e\u043d\u0446\u0430 \u0434\u0438\u0430\u043b\u043e\u0433\u0430.<\/p>\n<p><strong>Hybrid search.<\/strong> Jaccard + BM25 \u0443\u043b\u0443\u0447\u0448\u0430\u0442 \u043f\u043e\u0438\u0441\u043a \u043f\u043e \u0434\u043b\u0438\u043d\u043d\u044b\u043c \u0437\u0430\u043f\u0440\u043e\u0441\u0430\u043c \u0431\u0435\u0437 \u043f\u0435\u0440\u0435\u0445\u043e\u0434\u0430 \u043d\u0430 \u0432\u0435\u043a\u0442\u043e\u0440\u044b.<\/p>\n<p><strong>\u041c\u043e\u043d\u0438\u0442\u043e\u0440\u0438\u043d\u0433.<\/strong> Cloudflare Workers Analytics \u0438\u0437 \u043a\u043e\u0440\u043e\u0431\u043a\u0438 \u043f\u043e\u043a\u0430\u0437\u044b\u0432\u0430\u0435\u0442 \u0437\u0430\u043f\u0440\u043e\u0441\u044b, \u043e\u0448\u0438\u0431\u043a\u0438 \u0438 latency. \u041c\u043e\u0436\u043d\u043e \u0434\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0435 \u043b\u043e\u0433\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u0447\u0435\u0440\u0435\u0437 R2 \u0434\u043b\u044f \u0431\u043e\u043b\u0435\u0435 \u0434\u0435\u0442\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0430\u043d\u0430\u043b\u0438\u0437\u0430.<\/p>\n<hr\/>\n<p>\u041a\u043e\u0434 \u043f\u0440\u043e\u0435\u043a\u0442\u0430: <a href=\"https:\/\/github.com\/anatolii-iumashev\/pifai\" rel=\"noopener noreferrer nofollow\">github.com\/anatolii-iumashev\/pifai<\/a> (\u043f\u0430\u043f\u043a\u0430 <code>bot\/<\/code>)<\/p>\n<p>\u0411\u0430\u0437\u0430 \u0437\u043d\u0430\u043d\u0438\u0439: <a href=\"https:\/\/anatolii-iumashev.github.io\/pifai\/\" rel=\"noopener noreferrer nofollow\">anatolii-iumashev.github.io\/pifai<\/a><\/p>\n<p>\u0411\u043e\u0442: <a href=\"https:\/\/t.me\/pif_bbot\" rel=\"noopener noreferrer nofollow\">@pif_bbot<\/a><\/p>\n<p>\u041f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0430\u044f \u0441\u0442\u0430\u0442\u044c\u044f \u2014 <a href=\"https:\/\/habr.com\/ru\/articles\/1042264\/\" rel=\"noopener noreferrer nofollow\">\u0421\u043e\u0437\u0434\u0430\u043d\u0438\u0435 wiki \u043d\u0430 Astro\/Starlight<\/a><\/p>\n<\/div>\n<p>\u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"https:\/\/habr.com\/ru\/articles\/1046495\/\">https:\/\/habr.com\/ru\/articles\/1046495\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u0412 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u043b\u0438, \u043a\u0430\u043a \u0441\u043e\u0431\u0440\u0430\u0442\u044c \u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u0443\u044e wiki \u0438\u0437 markdown-\u0444\u0430\u0439\u043b\u043e\u0432 \u043d\u0430 Astro\/Starlight \u2014 \u043d\u0430 \u043f\u0440\u0438\u043c\u0435\u0440\u0435 \u043b\u0438\u0447\u043d\u043e\u0433\u043e \u043a\u0430\u0440\u044c\u0435\u0440\u043d\u043e\u0433\u043e \u043c\u0435\u043d\u0435\u0434\u0436\u0435\u0440\u0430. \u0412 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u0445 \u043f\u043e\u044f\u0432\u0438\u043b\u0438\u0441\u044c \u0437\u0430\u043a\u043e\u043d\u043e\u043c\u0435\u0440\u043d\u044b\u0439 \u0432\u043e\u043f\u0440\u043e\u0441: \u00ab\u043f\u043e\u0447\u0435\u043c\u0443 \u0438\u043c\u0435\u043d\u043d\u043e \u0442\u0430\u043a?\u00bb, \u00ab\u0447\u0442\u043e \u0437\u0430 \u0441\u0442\u0440\u0430\u043d\u043d\u044b\u0439 \u0432\u044b\u0431\u043e\u0440 \u0441\u0442\u0435\u043a\u0430?\u00bb, \u00ab\u0430 \u0434\u043b\u044f \u0447\u0435\u0433\u043e \u0435\u0449\u0451 \u044d\u0442\u043e \u043c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c, \u043a\u0440\u043e\u043c\u0435 \u043a\u0430\u043a \u0434\u043b\u044f \u0441\u0435\u0431\u044f?\u00bb.\u0425\u043e\u0440\u043e\u0448\u0438\u0439 \u0432\u043e\u043f\u0440\u043e\u0441. \u042d\u0442\u0430 \u0441\u0442\u0430\u0442\u044c\u044f \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u043d\u0430 \u043d\u0435\u0433\u043e \u0434\u0435\u043b\u043e\u043c.\u0422\u0430 \u0436\u0435 \u043c\u0435\u0445\u0430\u043d\u0438\u043a\u0430 \u2014 wiki \u0438\u0437 markdown \u2014 \u043d\u043e \u0442\u0435\u043f\u0435\u0440\u044c \u0441 Telegram-\u0431\u043e\u0442\u043e\u043c \u043f\u043e\u0432\u0435\u0440\u0445. \u0411\u043e\u0442 \u0443\u043c\u0435\u0435\u0442 \u0438\u0441\u043a\u0430\u0442\u044c \u043f\u043e \u0431\u0430\u0437\u0435 \u0437\u043d\u0430\u043d\u0438\u0439 \u0438 \u043e\u0442\u0432\u0435\u0447\u0430\u0442\u044c \u0441 \u0446\u0438\u0442\u0430\u0442\u0430\u043c\u0438 \u0438 \u0441\u0441\u044b\u043b\u043a\u0430\u043c\u0438 \u043d\u0430 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438. \u0412 \u043a\u0430\u0447\u0435\u0441\u0442\u0432\u0435 \u043f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u043e\u0439 \u043e\u0431\u043b\u0430\u0441\u0442\u0438 \u0432\u044b\u0431\u0440\u0430\u043d\u0430 \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u044f \u0438 \u0444\u0438\u043b\u043e\u0441\u043e\u0444\u0438\u044f: \u043f\u043e\u043b\u0443\u0447\u0438\u043b\u0441\u044f @pif_bbot \u2014 \u044d\u043c\u043f\u0430\u0442\u0438\u0447\u043d\u044b\u0439 \u043f\u043e\u043c\u043e\u0449\u043d\u0438\u043a, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043d\u0430 \u043e\u0441\u043d\u043e\u0432\u0435 \u043e\u0442\u043a\u0440\u044b\u0442\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439 \u043f\u043e \u041d\u0412\u041e, \u042e\u043d\u0433\u0443, \u0424\u0440\u0430\u043d\u043a\u043b\u0443 \u0438 \u0434\u0440\u0443\u0433\u0438\u043c \u0430\u0432\u0442\u043e\u0440\u0430\u043c.\u0412\u0435\u0441\u044c \u043a\u043e\u0434 \u2014 \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438 \u043d\u0430 GitHub, \u043f\u0430\u043f\u043a\u0430 bot\/.\u041a\u043e\u0433\u0434\u0430 \u0433\u043e\u0432\u043e\u0440\u044f\u0442 \u00abRAG\u00bb, \u0432 \u0433\u043e\u043b\u043e\u0432\u0435 \u0441\u0440\u0430\u0437\u0443 \u0432\u043e\u0437\u043d\u0438\u043a\u0430\u0435\u0442: \u0432\u0435\u043a\u0442\u043e\u0440\u043d\u0430\u044f \u0411\u0414, \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u0438, OpenAI API, Pinecone \u0438\u043b\u0438 pgvector. \u041a\u0430\u0436\u0435\u0442\u0441\u044f, \u0447\u0442\u043e \u0434\u043b\u044f \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u043e\u0433\u043e \u0440\u0430\u0431\u043e\u0447\u0435\u0433\u043e \u0440\u0435\u0448\u0435\u043d\u0438\u044f \u043d\u0443\u0436\u043d\u0430 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u0430\u044f \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u0438 \u043d\u0435\u043c\u0430\u043b\u0435\u043d\u044c\u043a\u0438\u0439 \u0431\u044e\u0434\u0436\u0435\u0442.\u041d\u043e \u0435\u0441\u0442\u044c \u0434\u0440\u0443\u0433\u043e\u0439 \u043f\u0443\u0442\u044c. \u0415\u0441\u043b\u0438 \u043f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u0430\u044f \u043e\u0431\u043b\u0430\u0441\u0442\u044c \u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e \u0441\u043f\u0435\u0446\u0438\u0430\u043b\u0438\u0437\u0438\u0440\u043e\u0432\u0430\u043d\u0430 \u2014 \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u044f, \u044e\u0440\u0438\u0441\u043f\u0440\u0443\u0434\u0435\u043d\u0446\u0438\u044f, \u0442\u0435\u0445\u043d\u0438\u0447\u0435\u0441\u043a\u0430\u044f \u0434\u043e\u043a\u0443\u043c\u0435\u043d\u0442\u0430\u0446\u0438\u044f \u2014 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 \u043f\u043e\u0438\u0441\u043a \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c \u0434\u0430\u0451\u0442 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u044b, \u0441\u0440\u0430\u0432\u043d\u0438\u043c\u044b\u0435 \u0441 \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u043c. \u0411\u0435\u0437 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u043e\u0432. \u0411\u0435\u0437 \u0432\u043d\u0435\u0448\u043d\u0438\u0445 API. \u0411\u0435\u0437 \u0431\u0430\u0437\u044b \u0434\u0430\u043d\u043d\u044b\u0445.\u0412 \u044d\u0442\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u043c\u044b \u043f\u043e\u0441\u0442\u0440\u043e\u0438\u043c Telegram-\u0431\u043e\u0442\u0430, \u043a\u043e\u0442\u043e\u0440\u044b\u0439:\u0438\u0449\u0435\u0442 \u043f\u043e \u0431\u0430\u0437\u0435 \u0437\u043d\u0430\u043d\u0438\u0439 \u0441 \u043f\u043e\u043c\u043e\u0449\u044c\u044e \u0430\u043b\u0433\u043e\u0440\u0438\u0442\u043c\u0430 Jaccard \u2014 \u0434\u0435\u0442\u0435\u0440\u043c\u0438\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e \u0438 \u0431\u044b\u0441\u0442\u0440\u043e\u0446\u0438\u0442\u0438\u0440\u0443\u0435\u0442 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438 \u0441\u043e \u0441\u0441\u044b\u043b\u043a\u0430\u043c\u0438 \u043d\u0430 \u043a\u043e\u043d\u043a\u0440\u0435\u0442\u043d\u044b\u0435 \u0441\u0442\u0430\u0442\u044c\u0438 wiki\u043f\u043e\u043c\u043d\u0438\u0442 \u0438\u0441\u0442\u043e\u0440\u0438\u044e \u0434\u0438\u0430\u043b\u043e\u0433\u0430 \u043c\u0435\u0436\u0434\u0443 \u0441\u0435\u0441\u0441\u0438\u044f\u043c\u0438 \u0447\u0435\u0440\u0435\u0437 Cloudflare KV\u0434\u0435\u043f\u043b\u043e\u0438\u0442\u0441\u044f \u043e\u0434\u043d\u043e\u0439 \u043a\u043e\u043c\u0430\u043d\u0434\u043e\u0439 \u043d\u0430 Cloudflare Workers \u2014 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u043e \u043f\u0440\u0438 \u0443\u043c\u0435\u0440\u0435\u043d\u043d\u043e\u0439 \u043d\u0430\u0433\u0440\u0443\u0437\u043a\u0435\u0421\u0442\u0435\u043aTypeScript \u2014 \u0435\u0434\u0438\u043d\u044b\u0439 \u044f\u0437\u044b\u043a \u0438 \u0434\u043b\u044f \u0431\u043e\u0442\u0430, \u0438 \u0434\u043b\u044f \u0441\u043a\u0440\u0438\u043f\u0442\u043e\u0432 \u0441\u0431\u043e\u0440\u043a\u0438Telegraf \u2014 \u0444\u0440\u0435\u0439\u043c\u0432\u043e\u0440\u043a \u0434\u043b\u044f Telegram Bot APIGroq API \u2014 \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 LLM (Llama-3.1-8b-instant, \u043e\u0447\u0435\u043d\u044c \u043d\u0438\u0437\u043a\u0430\u044f \u043b\u0430\u0442\u0435\u043d\u0442\u043d\u043e\u0441\u0442\u044c)Cloudflare Workers \u2014 serverless edge, cold start &lt; 5ms, \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tierCloudflare KV \u2014 \u0445\u0440\u0430\u043d\u0435\u043d\u0438\u0435 \u0438\u0441\u0442\u043e\u0440\u0438\u0438 \u0441\u0435\u0441\u0441\u0438\u0439\u0410\u0440\u0445\u0438\u0442\u0435\u043a\u0442\u0443\u0440\u0430Wiki (Markdown) \u2500\u2500\u25ba build-knowledge.ts \u2500\u2500\u25ba knowledge.ts                                               \u2502                                         256 \u0447\u0430\u043d\u043a\u043e\u0432 \u0441                                         \u043f\u0440\u0435\u0434\u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u044b\u043c\u0438                                         \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c\u0438 \u0441\u043b\u043e\u0432\u0430\u043c\u0438                                               \u2502Telegram \u2500\u2500\u25ba CF Worker \u2500\u2500\u25ba Retriever \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518                               \u2502         (Jaccard)                               \u25bc                          Groq LLM \u2500\u2500\u25ba \u043e\u0442\u0432\u0435\u0442 \u0441 \u0446\u0438\u0442\u0430\u0442\u0430\u043c\u0438                               \u25b2                          KV (\u0438\u0441\u0442\u043e\u0440\u0438\u044f)\u0413\u043b\u0430\u0432\u043d\u0430\u044f \u0438\u0434\u0435\u044f: \u0431\u0430\u0437\u0430 \u0437\u043d\u0430\u043d\u0438\u0439 \u0432\u0441\u0442\u0440\u043e\u0435\u043d\u0430 \u043f\u0440\u044f\u043c\u043e \u0432 \u043a\u043e\u0434. \u041f\u0440\u0438 \u0434\u0435\u043f\u043b\u043e\u0435 knowledge.ts \u0441 256 \u0447\u0430\u043d\u043a\u0430\u043c\u0438 \u0437\u0430\u0433\u0440\u0443\u0436\u0430\u0435\u0442\u0441\u044f \u0432 \u043f\u0430\u043c\u044f\u0442\u044c \u0432\u043e\u0440\u043a\u0435\u0440\u0430 \u2014 \u043d\u0438\u043a\u0430\u043a\u0438\u0445 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u0432 \u043a \u0411\u0414, \u043d\u0443\u043b\u0435\u0432\u0430\u044f \u043b\u0430\u0442\u0435\u043d\u0442\u043d\u043e\u0441\u0442\u044c \u043f\u043e\u0438\u0441\u043a\u0430. \u0417\u0432\u0443\u0447\u0438\u0442 \u043d\u0435\u043c\u043d\u043e\u0433\u043e \u0431\u0435\u0437\u0443\u043c\u043d\u043e, \u043d\u043e \u043d\u0430 \u043f\u0440\u0430\u043a\u0442\u0438\u043a\u0435 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442 \u043e\u0442\u043b\u0438\u0447\u043d\u043e: 29 \u0441\u0442\u0430\u0442\u0435\u0439, ~620KB, \u043f\u043e\u0438\u0441\u043a \u0437\u0430\u043d\u0438\u043c\u0430\u0435\u0442 \u0435\u0434\u0438\u043d\u0438\u0446\u044b \u043c\u0438\u043b\u043b\u0438\u0441\u0435\u043a\u0443\u043d\u0434.\u041f\u043e\u0434\u0433\u043e\u0442\u043e\u0432\u043a\u0430\u041d\u0443\u0436\u043d\u043e:Node.js 20+\u0410\u043a\u043a\u0430\u0443\u043d\u0442 Cloudflare (\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439)\u0422\u043e\u043a\u0435\u043d Telegram-\u0431\u043e\u0442\u0430 \u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u0443 @BotFatherAPI-\u043a\u043b\u044e\u0447 Groq (\u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier)\u0421\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u043f\u0440\u043e\u0435\u043a\u0442\u0430 (wiki \u0443\u0436\u0435 \u0435\u0441\u0442\u044c \u0438\u0437 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0438, \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u043c \u043f\u0430\u043f\u043a\u0443 bot\/):pif\/\u251c\u2500\u2500 src\/content\/docs\/      # Wiki \u0438\u0437 \u043f\u0440\u0435\u0434\u044b\u0434\u0443\u0449\u0435\u0439 \u0441\u0442\u0430\u0442\u044c\u0438\u2502   \u251c\u2500\u2500 authors\/\u2502   \u2502   \u251c\u2500\u2500 jung\/\u2502   \u2502   \u2502   \u2514\u2500\u2500 shadow.md\u2502   \u2502   \u2514\u2500\u2500 frankl\/\u2502   \u2502       \u2514\u2500\u2500 logotherapy.md\u2502   \u2514\u2500\u2500 practices\/\u2502       \u2514\u2500\u2500 nvc.md\u2514\u2500\u2500 bot\/    \u251c\u2500\u2500 src\/    \u2502   \u251c\u2500\u2500 index.ts       # \u0442\u043e\u0447\u043a\u0430 \u0432\u0445\u043e\u0434\u0430 CF Workers    \u2502   \u251c\u2500\u2500 bot.ts         # Telegram-\u043e\u0431\u0440\u0430\u0431\u043e\u0442\u0447\u0438\u043a\u0438    \u2502   \u251c\u2500\u2500 knowledge.ts   # \u0430\u0432\u0442\u043e\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0438\u043d\u0434\u0435\u043a\u0441 (\u043d\u0435 \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c)    \u2502   \u251c\u2500\u2500 retriever.ts   # RAG-\u043f\u043e\u0438\u0441\u043a    \u2502   \u251c\u2500\u2500 llm.ts         # \u043a\u043b\u0438\u0435\u043d\u0442 Groq    \u2502   \u251c\u2500\u2500 session.ts     # \u0441\u0435\u0441\u0441\u0438\u0438 \u0447\u0435\u0440\u0435\u0437 KV    \u2502   \u2514\u2500\u2500 prompts.ts     # system prompt    \u251c\u2500\u2500 scripts\/    \u2502   \u2514\u2500\u2500 build-knowledge.ts    \u2514\u2500\u2500 wrangler.toml\u0423\u0441\u0442\u0430\u043d\u043e\u0432\u043a\u0430 \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439:cd botnpm init -ynpm install telegrafnpm install -D wrangler tsx typescript @cloudflare\/workers-types\u0428\u0430\u0433 1. \u0413\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044f \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439\u041f\u0435\u0440\u0432\u044b\u0439 \u0448\u0430\u0433 \u2014 \u043f\u0440\u0435\u0432\u0440\u0430\u0442\u0438\u0442\u044c markdown-\u0444\u0430\u0439\u043b\u044b wiki \u0432 \u0438\u043d\u0434\u0435\u043a\u0441 \u0434\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430.\u0421\u043a\u0440\u0438\u043f\u0442 scripts\/build-knowledge.ts \u0434\u0435\u043b\u0430\u0435\u0442 \u0442\u0440\u0438 \u0432\u0435\u0449\u0438:\u0421\u043a\u0430\u043d\u0438\u0440\u0443\u0435\u0442 src\/content\/docs\/**\/*.md\u0420\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u0442 \u043a\u0430\u0436\u0434\u0443\u044e \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0443 \u043d\u0430 \u0441\u0435\u043a\u0446\u0438\u0438 \u043f\u043e ## \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430\u043c\u0414\u043b\u044f \u043a\u0430\u0436\u0434\u043e\u0439 \u0441\u0435\u043a\u0446\u0438\u0438 \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u0435\u0442 \u0441\u043f\u0438\u0441\u043e\u043a \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u0445 \u0441\u043b\u043e\u0432interface WikiChunk {  id: string;         \/\/ &#171;authors\/jung\/shadow#\u0422\u0435\u043d\u044c&#187;  title: string;      \/\/ \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043e\u043a \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b  sourcePath: string; \/\/ &#171;authors\/jung\/shadow.md&#187;  section: string;    \/\/ &#171;## \u0422\u0435\u043d\u044c&#187;  text: string;       \/\/ \u0442\u0435\u043a\u0441\u0442 \u0441\u0435\u043a\u0446\u0438\u0438  keywords: string[]; \/\/ \u043f\u0440\u0435\u0434\u0432\u044b\u0447\u0438\u0441\u043b\u0435\u043d\u043d\u044b\u0435 \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0441\u043b\u043e\u0432\u0430}\u041a\u043b\u044e\u0447\u0435\u0432\u0430\u044f \u0444\u0443\u043d\u043a\u0446\u0438\u044f \u2014 \u0440\u0430\u0437\u0431\u0438\u0432\u043a\u0430 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u044b \u043d\u0430 \u0447\u0430\u043d\u043a\u0438:function chunkPage(page: WikiPage): WikiChunk[] {  const chunks: WikiChunk[] = [];  \/\/ \u0423\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u0435\u043a\u0446\u0438\u044e &#171;\u041c\u0430\u0442\u0435\u0440\u0438\u0430\u043b\u044b \u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438&#187; \u2014 \u043d\u0435 \u043d\u0443\u0436\u043d\u0430 \u0434\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430  const body = page.content.replace(\/## \u041c\u0430\u0442\u0435\u0440\u0438\u0430\u043b\u044b \u0438 \u0438\u0441\u0442\u043e\u0447\u043d\u0438\u043a\u0438[\\s\\S]*$\/, &#187;).trim();  \/\/ \u0420\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u043c \u043f\u043e ## \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430\u043c  const sections = body.split(\/(?=^## )\/m);  for (const section of sections) {    const headerMatch = section.match(\/^## (.+)$\/m);    const sectionName = headerMatch ? headerMatch[1].trim() : &#187;;    const text = section.replace(\/^## .+\\n*\/m, &#187;).trim();    if (!text || text.length &lt; 20) continue;    \/\/ \u041a\u043b\u044e\u0447\u0435\u0432\u044b\u0435 \u0441\u043b\u043e\u0432\u0430: \u0442\u043e\u043a\u0435\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u0437\u0430\u0433\u043e\u043b\u043e\u0432\u043a\u0430 + \u0441\u0435\u043a\u0446\u0438\u0438 + \u043f\u0435\u0440\u0432\u044b\u0445 500 \u0441\u0438\u043c\u0432\u043e\u043b\u043e\u0432 \u0442\u0435\u043a\u0441\u0442\u0430    const keywords = tokenize(`${page.title} ${sectionName} ${text.slice(0, 500)}`);    chunks.push({      id: `${page.path}#${sectionName}`,      title: page.title,      sourcePath: page.path,      section: sectionName,      text,      keywords,    });  }  return chunks;}\u0422\u043e\u043a\u0435\u043d\u0438\u0437\u0430\u0446\u0438\u044f \u043f\u0440\u043e\u0441\u0442\u0430\u044f: \u0440\u0430\u0437\u0431\u0438\u0432\u0430\u0435\u043c \u043d\u0430 \u0441\u043b\u043e\u0432\u0430, \u0444\u0438\u043b\u044c\u0442\u0440\u0443\u0435\u043c \u0441\u0442\u043e\u043f-\u0441\u043b\u043e\u0432\u0430 (\u0440\u0443\u0441\u0441\u043a\u0438\u0435 + \u0430\u043d\u0433\u043b\u0438\u0439\u0441\u043a\u0438\u0435), \u0443\u0431\u0438\u0440\u0430\u0435\u043c \u0441\u043b\u043e\u0432\u0430 \u043a\u043e\u0440\u043e\u0447\u0435 3 \u0441\u0438\u043c\u0432\u043e\u043b\u043e\u0432, \u0434\u0435\u0434\u0443\u043f\u043b\u0438\u0446\u0438\u0440\u0443\u0435\u043c:const STOPWORDS = new Set([  &#8216;\u0438&#8217;, &#8216;\u0432&#8217;, &#8216;\u0432\u043e&#8217;, &#8216;\u043d\u0435&#8217;, &#8216;\u0447\u0442\u043e&#8217;, &#8216;\u043e\u043d&#8217;, &#8216;\u043d\u0430&#8217;, &#8216;\u044f&#8217;, &#8216;\u0441&#8217;, &#8216;\u0441\u043e&#8217;,  \/\/ &#8230; \u043f\u043e\u043b\u043d\u044b\u0439 \u0441\u043f\u0438\u0441\u043e\u043a \u0432 \u0440\u0435\u043f\u043e\u0437\u0438\u0442\u043e\u0440\u0438\u0438  &#8216;the&#8217;, &#8216;a&#8217;, &#8216;an&#8217;, &#8216;and&#8217;, &#8216;or&#8217;, &#8216;but&#8217;, &#8216;in&#8217;, &#8216;on&#8217;,]);function tokenize(text: string): string[] {  const words = text.toLowerCase().match(\/[\u0430-\u044f\u0451a-z]+\/gi) || [];  return [&#8230;new Set(words.filter(w =&gt; w.length &gt; 2 &amp;&amp; !STOPWORDS.has(w)))];}\u0420\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442 \u2014 \u0444\u0430\u0439\u043b src\/knowledge.ts \u0441 \u043c\u0430\u0441\u0441\u0438\u0432\u043e\u043c KNOWLEDGE_CHUNKS. \u041f\u0440\u0438 29 \u0441\u0442\u0440\u0430\u043d\u0438\u0446\u0430\u0445 wiki \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f ~256 \u0447\u0430\u043d\u043a\u043e\u0432.npm run build  # \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u0442 build-knowledge.ts \u0447\u0435\u0440\u0435\u0437 tsx\u0412\u0430\u0436\u043d\u043e: knowledge.ts \u2014 \u0430\u0432\u0442\u043e\u0433\u0435\u043d\u0435\u0440\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u044b\u0439 \u0444\u0430\u0439\u043b, \u0435\u0433\u043e \u043d\u0435 \u043d\u0443\u0436\u043d\u043e \u0440\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432\u0440\u0443\u0447\u043d\u0443\u044e. \u041a\u0430\u0436\u0434\u044b\u0439 \u0440\u0430\u0437 \u043f\u0440\u0438 \u043e\u0431\u043d\u043e\u0432\u043b\u0435\u043d\u0438\u0438 wiki \u0437\u0430\u043f\u0443\u0441\u043a\u0430\u0439\u0442\u0435 npm run build \u043f\u0435\u0440\u0435\u0434 \u0434\u0435\u043f\u043b\u043e\u0435\u043c.\u0428\u0430\u0433 2. Retriever: \u043f\u043e\u0438\u0441\u043a \u043f\u043e \u0447\u0430\u043d\u043a\u0430\u043c\u0424\u0430\u0439\u043b src\/retriever.ts \u2014 \u0441\u0435\u0440\u0434\u0446\u0435 \u0432\u0441\u0435\u0439 RAG-\u0441\u0438\u0441\u0442\u0435\u043c\u044b.\u0414\u043b\u044f \u043f\u043e\u0438\u0441\u043a\u0430 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u0435\u043c Jaccard-\u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0435 \u0441\u0445\u043e\u0434\u0441\u0442\u0432\u043e \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c:score = |queryTokens \u2229 chunkKeywords| \/ |queryTokens \u222a chunkKeywords|\u0427\u0435\u043c \u0431\u043e\u043b\u044c\u0448\u0435 \u043e\u0431\u0449\u0438\u0445 \u0441\u043b\u043e\u0432 \u043c\u0435\u0436\u0434\u0443 \u0437\u0430\u043f\u0440\u043e\u0441\u043e\u043c \u0438 \u0447\u0430\u043d\u043a\u043e\u043c \u2014 \u0442\u0435\u043c \u0432\u044b\u0448\u0435 score. \u0411\u0435\u0440\u0451\u043c top-K \u0447\u0430\u043d\u043a\u043e\u0432 \u0441 \u043d\u0435\u043d\u0443\u043b\u0435\u0432\u044b\u043c score.export function createRetriever(chunks: WikiChunk[], baseUrl: string): Retriever {  return {    retrieve(query: string, topK: number = 3): RetrievedChunk[] {      const queryTokens = tokenize(query);      if (queryTokens.length === 0) return [];      const scored = chunks.map(chunk =&gt; {        const overlap = queryTokens.filter(t =&gt; chunk.keywords.includes(t)).length;        const union = new Set([&#8230;queryTokens, &#8230;chunk.keywords]);        const score = union.size &gt; 0 ? overlap \/ union.size : 0;        return { chunk, score };      });      return scored        .sort((a, b) =&gt; b.score &#8212; a.score)        .slice(0, topK)        .filter(c =&gt; c.score &gt; 0)        .map(c =&gt; ({ &#8230;c.chunk }));    },    \/\/ &#8230;  };}\u041f\u043e\u0447\u0435\u043c\u0443 \u043d\u0435 \u0432\u0435\u043a\u0442\u043e\u0440\u044b?\u0412\u043e\u043f\u0440\u043e\u0441 \u0438\u0437 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u0435\u0432 \u043a \u043f\u0435\u0440\u0432\u043e\u0439 \u0441\u0442\u0430\u0442\u044c\u0435 \u2014 \u043e\u0442\u0432\u0435\u0447\u0430\u044e: \u0441\u0435\u043c\u0430\u043d\u0442\u0438\u0447\u0435\u0441\u043a\u0438\u0439 \u043f\u043e\u0438\u0441\u043a \u0447\u0435\u0440\u0435\u0437 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u0438 \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u043b\u0443\u0447\u0448\u0435 \u043f\u043e\u043d\u0438\u043c\u0430\u0435\u0442 \u0441\u0438\u043d\u043e\u043d\u0438\u043c\u044b \u0438 \u0441\u043c\u044b\u0441\u043b\u043e\u0432\u044b\u0435 \u0441\u0432\u044f\u0437\u0438. \u041d\u043e \u0437\u0430 \u044d\u0442\u043e \u043d\u0443\u0436\u043d\u043e \u043f\u043b\u0430\u0442\u0438\u0442\u044c: API \u0434\u043b\u044f \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u0438 \u044d\u043c\u0431\u0435\u0434\u0434\u0438\u043d\u0433\u043e\u0432, \u0432\u0435\u043a\u0442\u043e\u0440\u043d\u043e\u0435 \u0445\u0440\u0430\u043d\u0438\u043b\u0438\u0449\u0435, \u0434\u043e\u043f\u043e\u043b\u043d\u0438\u0442\u0435\u043b\u044c\u043d\u044b\u0439 \u0441\u0435\u0442\u0435\u0432\u043e\u0439 \u0432\u044b\u0437\u043e\u0432 \u043d\u0430 \u043a\u0430\u0436\u0434\u044b\u0439 \u0437\u0430\u043f\u0440\u043e\u0441.Jaccard \u043f\u043e \u043a\u043b\u044e\u0447\u0435\u0432\u044b\u043c \u0441\u043b\u043e\u0432\u0430\u043c \u043e\u043f\u0440\u0430\u0432\u0434\u0430\u043d, \u043a\u043e\u0433\u0434\u0430:\u041f\u0440\u0435\u0434\u043c\u0435\u0442\u043d\u0430\u044f \u043e\u0431\u043b\u0430\u0441\u0442\u044c \u0443\u0437\u043a\u0430\u044f \u0438 \u0438\u043c\u0435\u0435\u0442 \u0447\u0451\u0442\u043a\u0443\u044e \u0442\u0435\u0440\u043c\u0438\u043d\u043e\u043b\u043e\u0433\u0438\u044e\u041f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u0435\u043b\u0438 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u0443\u044e\u0442 \u0442\u0435\u0440\u043c\u0438\u043d\u044b \u0438\u0437 \u0441\u0430\u043c\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439\u041d\u0443\u0436\u043d\u0430 \u0434\u0435\u0442\u0435\u0440\u043c\u0438\u043d\u0438\u0440\u043e\u0432\u0430\u043d\u043d\u043e\u0441\u0442\u044c \u2014 \u043e\u0434\u0438\u043d \u0438 \u0442\u043e\u0442 \u0436\u0435 \u0437\u0430\u043f\u0440\u043e\u0441 \u0432\u0441\u0435\u0433\u0434\u0430 \u0434\u0430\u0451\u0442 \u043e\u0434\u0438\u043d\u0430\u043a\u043e\u0432\u044b\u0439 \u0440\u0435\u0437\u0443\u043b\u044c\u0442\u0430\u0442\u0412\u0430\u0436\u043d\u0430 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u044c\u043d\u0430\u044f \u0438\u043d\u0444\u0440\u0430\u0441\u0442\u0440\u0443\u043a\u0442\u0443\u0440\u0430 \u0438 \u043d\u0443\u043b\u0435\u0432\u044b\u0435 \u043e\u043f\u0435\u0440\u0430\u0446\u0438\u043e\u043d\u043d\u044b\u0435 \u0440\u0430\u0441\u0445\u043e\u0434\u044b\u0414\u043b\u044f \u043f\u0441\u0438\u0445\u043e\u043b\u043e\u0433\u0438\u0447\u0435\u0441\u043a\u043e\u0439 \u0431\u0430\u0437\u044b \u0437\u043d\u0430\u043d\u0438\u0439 \u044d\u0442\u043e \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442: \u0437\u0430\u043f\u0440\u043e\u0441 \u00ab\u0442\u0440\u0435\u0432\u043e\u0433\u0430 \u0438 \u0441\u0442\u0440\u0430\u0445\u00bb \u043d\u0430\u0439\u0434\u0451\u0442 \u0447\u0430\u043d\u043a \u043f\u0440\u043e \u0441\u0442\u0440\u0430\u0445 \u0432 \u043b\u043e\u0433\u043e\u0442\u0435\u0440\u0430\u043f\u0438\u0438 \u0424\u0440\u0430\u043d\u043a\u043b\u0430, \u00ab\u043a\u043e\u043d\u0444\u043b\u0438\u043a\u0442 \u0432 \u043e\u0442\u043d\u043e\u0448\u0435\u043d\u0438\u044f\u0445\u00bb \u2014 \u0447\u0430\u043d\u043a \u043f\u0440\u043e \u0430\u043c\u043e\u0440\u0442\u0438\u0437\u0430\u0446\u0438\u044e \u043f\u043e \u041b\u0438\u0442\u0432\u0430\u043a\u0443. \u041f\u0440\u043e\u0432\u0435\u0440\u044c\u0442\u0435 \u0441\u0430\u043c\u0438.Retriever \u0442\u0430\u043a\u0436\u0435 \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 \u0444\u043e\u0440\u043c\u0430\u0442\u0438\u0440\u043e\u0432\u0430\u043d\u0438\u0435 \u043d\u0430\u0439\u0434\u0435\u043d\u043d\u044b\u0445 \u0447\u0430\u043d\u043a\u043e\u0432 \u0434\u043b\u044f LLM:formatContext(entries: RetrievedChunk[]): string {  if (entries.length === 0) return &#187;;  return entries.map((e, i) =&gt; {    const url = `${baseUrl}${wikiPathToUrl(e.sourcePath)}`;    return `[\u0418\u0441\u0442\u043e\u0447\u043d\u0438\u043a ${i + 1}]: ${e.title} \u2192 ${url}&gt; ${e.section ? `*${e.section}*` : &#187;}&gt;${e.text.split(&#8216;\\n&#8217;).map(line =&gt; `&gt; ${line}`).join(&#8216;\\n&#8217;)}`;  }).join(&#8216;\\n\\n&#8212;\\n\\n&#8217;);},\u0418 \u0437\u0430 \u0433\u0435\u043d\u0435\u0440\u0430\u0446\u0438\u044e URL \u0438\u0437 \u043f\u0443\u0442\u0438 \u043a \u0444\u0430\u0439\u043b\u0443:function wikiPathToUrl(sourcePath: string): string {  const withoutExt = sourcePath.replace(\/\\.md$\/, &#187;);  if (withoutExt.endsWith(&#8216;\/index&#8217;)) {    return &#8216;\/&#8217; + withoutExt.replace(&#8216;\/index&#8217;, &#187;) + &#8216;\/&#8217;;  }  return &#8216;\/&#8217; + withoutExt + &#8216;\/&#8217;;}\/\/ &#171;authors\/jung\/shadow.md&#187; \u2192 &#171;\/authors\/jung\/shadow\/&#187;\u0428\u0430\u0433 3. LLM-\u043a\u043b\u0438\u0435\u043d\u0442\u0424\u0430\u0439\u043b src\/llm.ts \u2014 \u043c\u0438\u043d\u0438\u043c\u0430\u043b\u0438\u0441\u0442\u0438\u0447\u043d\u044b\u0439 \u0432\u0440\u0430\u043f\u043f\u0435\u0440 \u043d\u0430\u0434 Groq API.\u041d\u0438\u043a\u0430\u043a\u0438\u0445 SDK \u2014 \u0442\u043e\u043b\u044c\u043a\u043e fetch. \u042d\u0442\u043e \u043f\u0440\u0438\u043d\u0446\u0438\u043f\u0438\u0430\u043b\u044c\u043d\u043e \u0434\u043b\u044f Cloudflare Workers: \u043a\u0440\u0443\u043f\u043d\u044b\u0435 SDK \u0432\u0440\u043e\u0434\u0435 \u043e\u0444\u0438\u0446\u0438\u0430\u043b\u044c\u043d\u043e\u0433\u043e OpenAI-\u043a\u043b\u0438\u0435\u043d\u0442\u0430 \u043c\u043e\u0433\u0443\u0442 \u043d\u0435 \u043f\u043e\u0434\u0434\u0435\u0440\u0436\u0438\u0432\u0430\u0442\u044c Workers runtime \u0438\u043b\u0438 \u0442\u0430\u0449\u0438\u0442\u044c \u0437\u0430 \u0441\u043e\u0431\u043e\u0439 \u043f\u043e\u043b\u0442\u043e\u043d\u043d\u044b \u0437\u0430\u0432\u0438\u0441\u0438\u043c\u043e\u0441\u0442\u0435\u0439. \u041f\u0440\u043e\u0441\u0442\u043e\u0439 fetch-\u0432\u0440\u0430\u043f\u043f\u0435\u0440 \u043d\u0430\u0434\u0451\u0436\u043d\u0435\u0435.export function initLLM(config: LLMConfig): LLMClient {  return {    async chat(messages) {      const response = await fetch(&#8216;https:\/\/api.groq.com\/openai\/v1\/chat\/completions&#8217;, {        method: &#8216;POST&#8217;,        headers: {          &#8216;Authorization&#8217;: `Bearer ${config.apiKey}`,          &#8216;Content-Type&#8217;: &#8216;application\/json&#8217;,        },        body: JSON.stringify({          model: config.model,          messages,          temperature: 0.7,          max_tokens: 2048,        }),      });      if (!response.ok) {        const err = await response.text();        throw new Error(`Groq API error ${response.status}: ${err}`);      }      const data = await response.json() as any;      return data.choices?.[0]?.message?.content || &#187;;    },  };}Groq \u0432\u044b\u0431\u0440\u0430\u043d \u043f\u043e \u0442\u0440\u0451\u043c \u043f\u0440\u0438\u0447\u0438\u043d\u0430\u043c: \u0431\u0435\u0441\u043f\u043b\u0430\u0442\u043d\u044b\u0439 tier \u0441 \u043f\u0440\u0438\u043b\u0438\u0447\u043d\u044b\u043c\u0438 \u043b\u0438\u043c\u0438\u0442\u0430\u043c\u0438, Llama-3.1-8b-instant \u043e\u0442\u0432\u0435\u0447\u0430\u0435\u0442 \u0437\u0430 200\u2013500ms, API \u0441\u043e\u0432\u043c\u0435\u0441\u0442\u0438\u043c \u0441 OpenAI \u2014 \u043f\u0440\u0438 \u0436\u0435\u043b\u0430\u043d\u0438\u0438 \u043c\u043e\u0436\u043d\u043e \u043f\u043e\u043c\u0435\u043d\u044f\u0442\u044c \u043f\u0440\u043e\u0432\u0430\u0439\u0434\u0435\u0440 \u043e\u0434\u043d\u043e\u0439 \u0441\u0442\u0440\u043e\u043a\u043e\u0439. \u041c\u043e\u0434\u0435\u043b\u044c \u0437\u0430\u0434\u0430\u0451\u0442\u0441\u044f \u0447\u0435\u0440\u0435\u0437 env GROQ_MODEL, \u0442\u0430\u043a \u0447\u0442\u043e \u0434\u043b\u044f \u0441\u043c\u0435\u043d\u044b \u043c\u043e\u0434\u0435\u043b\u0438 \u043d\u0435 \u043d\u0443\u0436\u0435\u043d \u0440\u0435\u0434\u0435\u043f\u043b\u043e\u0439.\u0428\u0430\u0433 4. \u0421\u0435\u0441\u0441\u0438\u0438 \u0432 Cloudflare KV\u0424\u0430\u0439\u043b src\/session.ts \u2014 \u0438\u0441\u0442\u043e\u0440\u0438\u044f \u0434\u0438\u0430\u043b\u043e\u0433\u0430.Cloudflare Workers stateless: \u043a\u0430\u0436\u0434\u044b\u0439 \u0432\u0445\u043e\u0434\u044f\u0449\u0438\u0439&#8230;<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"closed","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-483314","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/483314","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=483314"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/483314\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=483314"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=483314"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=483314"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}