Spaces:
Paused
Paused
| import type { BackendModel } from "./server/models"; | |
| import type { Message } from "./types/Message"; | |
| import { format } from "date-fns"; | |
| import type { WebSearch } from "./types/WebSearch"; | |
| import { downloadFile } from "./server/files/downloadFile"; | |
| import type { Conversation } from "./types/Conversation"; | |
| interface buildPromptOptions { | |
| messages: Pick<Message, "from" | "content" | "files">[]; | |
| id?: Conversation["_id"]; | |
| model: BackendModel; | |
| locals?: App.Locals; | |
| webSearch?: WebSearch; | |
| preprompt?: string; | |
| files?: File[]; | |
| } | |
| export async function buildPrompt({ | |
| messages, | |
| model, | |
| webSearch, | |
| preprompt, | |
| id, | |
| }: buildPromptOptions): Promise<string> { | |
| if (webSearch && webSearch.context) { | |
| const lastMsg = messages.slice(-1)[0]; | |
| const messagesWithoutLastUsrMsg = messages.slice(0, -1); | |
| const previousUserMessages = messages.filter((el) => el.from === "user").slice(0, -1); | |
| const previousQuestions = | |
| previousUserMessages.length > 0 | |
| ? `Previous questions: \n${previousUserMessages | |
| .map(({ content }) => `- ${content}`) | |
| .join("\n")}` | |
| : ""; | |
| const currentDate = format(new Date(), "MMMM d, yyyy"); | |
| messages = [ | |
| ...messagesWithoutLastUsrMsg, | |
| { | |
| from: "user", | |
| content: `I searched the web using the query: ${webSearch.searchQuery}. Today is ${currentDate} and here are the results: | |
| ===================== | |
| ${webSearch.context} | |
| ===================== | |
| ${previousQuestions} | |
| Answer the question: ${lastMsg.content} | |
| `, | |
| }, | |
| ]; | |
| } | |
| // section to handle potential files input | |
| if (model.multimodal) { | |
| messages = await Promise.all( | |
| messages.map(async (el) => { | |
| let content = el.content; | |
| if (el.from === "user") { | |
| if (el?.files && el.files.length > 0 && id) { | |
| const markdowns = await Promise.all( | |
| el.files.map(async (hash) => { | |
| try { | |
| const { content: image, mime } = await downloadFile(hash, id); | |
| const b64 = image.toString("base64"); | |
| return `})`; | |
| } catch (e) { | |
| console.error(e); | |
| } | |
| }) | |
| ); | |
| content += markdowns.join("\n "); | |
| } else { | |
| // if no image, append an empty white image | |
| content += | |
| "\n"; | |
| } | |
| } | |
| return { ...el, content }; | |
| }) | |
| ); | |
| } | |
| return ( | |
| model | |
| .chatPromptRender({ messages, preprompt }) | |
| // Not super precise, but it's truncated in the model's backend anyway | |
| .split(" ") | |
| .slice(-(model.parameters?.truncate ?? 0)) | |
| .join(" ") | |
| ); | |
| } | |