Spaces:
Sleeping
Sleeping
| <script lang="ts"> | |
| import { goto, replaceState } from "$app/navigation"; | |
| import { base } from "$app/paths"; | |
| import { page } from "$app/state"; | |
| import { usePublicConfig } from "$lib/utils/PublicConfig.svelte"; | |
| const publicConfig = usePublicConfig(); | |
| import ChatWindow from "$lib/components/chat/ChatWindow.svelte"; | |
| import { ERROR_MESSAGES, error } from "$lib/stores/errors"; | |
| import { pendingMessage } from "$lib/stores/pendingMessage"; | |
| import { useSettingsStore } from "$lib/stores/settings.js"; | |
| import { findCurrentModel } from "$lib/utils/models"; | |
| import { sanitizeUrlParam } from "$lib/utils/urlParams"; | |
| import { onMount, tick } from "svelte"; | |
| import { loading } from "$lib/stores/loading.js"; | |
| import { loadAttachmentsFromUrls } from "$lib/utils/loadAttachmentsFromUrls"; | |
| let { data } = $props(); | |
| let hasModels = $derived(Boolean(data.models?.length)); | |
| let files: File[] = $state([]); | |
| let draft = $state(""); | |
| const settings = useSettingsStore(); | |
| async function createConversation(message: string) { | |
| try { | |
| $loading = true; | |
| // check if $settings.activeModel is a valid model | |
| // else check if it's an assistant, and use that model | |
| // else use the first model | |
| const validModels = data.models.map((model) => model.id); | |
| let model; | |
| if (validModels.includes($settings.activeModel)) { | |
| model = $settings.activeModel; | |
| } else { | |
| model = data.models[0].id; | |
| } | |
| const res = await fetch(`${base}/conversation`, { | |
| method: "POST", | |
| headers: { | |
| "Content-Type": "application/json", | |
| }, | |
| body: JSON.stringify({ | |
| model, | |
| preprompt: $settings.customPrompts[$settings.activeModel], | |
| }), | |
| }); | |
| if (!res.ok) { | |
| const errorMessage = (await res.json()).message || ERROR_MESSAGES.default; | |
| error.set(errorMessage); | |
| console.error("Error while creating conversation: ", errorMessage); | |
| return; | |
| } | |
| const { conversationId } = await res.json(); | |
| // Ugly hack to use a store as temp storage, feel free to improve ^^ | |
| pendingMessage.set({ | |
| content: message, | |
| files, | |
| }); | |
| // invalidateAll to update list of conversations | |
| await goto(`${base}/conversation/${conversationId}`, { invalidateAll: true }); | |
| } catch (err) { | |
| error.set((err as Error).message || ERROR_MESSAGES.default); | |
| console.error(err); | |
| } finally { | |
| $loading = false; | |
| } | |
| } | |
| onMount(async () => { | |
| try { | |
| // Handle attachments parameter first | |
| if (page.url.searchParams.has("attachments")) { | |
| const result = await loadAttachmentsFromUrls(page.url.searchParams); | |
| files = result.files; | |
| // Show errors if any | |
| if (result.errors.length > 0) { | |
| console.error("Failed to load some attachments:", result.errors); | |
| error.set( | |
| `Failed to load ${result.errors.length} attachment(s). Check console for details.` | |
| ); | |
| } | |
| // Clean up URL | |
| const url = new URL(page.url); | |
| url.searchParams.delete("attachments"); | |
| history.replaceState({}, "", url); | |
| } | |
| const query = sanitizeUrlParam(page.url.searchParams.get("q")); | |
| if (query) { | |
| void createConversation(query); | |
| const url = new URL(page.url); | |
| url.searchParams.delete("q"); | |
| tick().then(() => { | |
| replaceState(url, page.state); | |
| }); | |
| return; | |
| } | |
| const promptQuery = sanitizeUrlParam(page.url.searchParams.get("prompt")); | |
| if (promptQuery && !draft) { | |
| draft = promptQuery; | |
| const url = new URL(page.url); | |
| url.searchParams.delete("prompt"); | |
| tick().then(() => { | |
| replaceState(url, page.state); | |
| }); | |
| } | |
| } catch (err) { | |
| console.error("Failed to process URL parameters:", err); | |
| } | |
| }); | |
| let currentModel = $derived(findCurrentModel(data.models, data.oldModels, $settings.activeModel)); | |
| </script> | |
| <svelte:head> | |
| <title>{publicConfig.PUBLIC_APP_NAME}</title> | |
| </svelte:head> | |
| {#if hasModels} | |
| <ChatWindow | |
| onmessage={(message) => createConversation(message)} | |
| loading={$loading} | |
| {currentModel} | |
| models={data.models} | |
| bind:files | |
| bind:draft | |
| /> | |
| {:else} | |
| <div class="mx-auto my-20 max-w-xl rounded-xl border p-6 text-center dark:border-gray-700"> | |
| <h2 class="mb-2 text-xl font-semibold">No models available</h2> | |
| <p class="text-gray-600 dark:text-gray-300"> | |
| No chat models are configured. Set `OPENAI_BASE_URL` and ensure the server can reach the | |
| endpoint, then reload. If unset, the app defaults to the Hugging Face router. | |
| </p> | |
| </div> | |
| {/if} | |