mirror of
				https://github.com/open-webui/open-webui
				synced 2025-06-26 18:26:48 +00:00 
			
		
		
		
	feat: chat completion notification
This commit is contained in:
		
							parent
							
								
									f133353734
								
							
						
					
					
						commit
						d9573befff
					
				@ -198,6 +198,13 @@ class ChatTable:
 | 
			
		||||
            self.add_chat_tag_by_id_and_user_id_and_tag_name(id, user.id, tag_name)
 | 
			
		||||
        return self.get_chat_by_id(id)
 | 
			
		||||
 | 
			
		||||
    def get_chat_title_by_id(self, id: str) -> Optional[str]:
 | 
			
		||||
        chat = self.get_chat_by_id(id)
 | 
			
		||||
        if chat is None:
 | 
			
		||||
            return None
 | 
			
		||||
 | 
			
		||||
        return chat.chat.get("title", "New Chat")
 | 
			
		||||
 | 
			
		||||
    def get_messages_by_chat_id(self, id: str) -> Optional[dict]:
 | 
			
		||||
        chat = self.get_chat_by_id(id)
 | 
			
		||||
        if chat is None:
 | 
			
		||||
 | 
			
		||||
@ -582,9 +582,10 @@ async def process_chat_response(request, response, user, events, metadata, tasks
 | 
			
		||||
 | 
			
		||||
                    except Exception as e:
 | 
			
		||||
                        done = "data: [DONE]" in line
 | 
			
		||||
                        title = Chats.get_chat_title_by_id(metadata["chat_id"])
 | 
			
		||||
 | 
			
		||||
                        if done:
 | 
			
		||||
                            data = {"done": True}
 | 
			
		||||
                            data = {"done": True, "content": content, "title": title}
 | 
			
		||||
                        else:
 | 
			
		||||
                            continue
 | 
			
		||||
 | 
			
		||||
@ -602,27 +603,41 @@ async def process_chat_response(request, response, user, events, metadata, tasks
 | 
			
		||||
                    messages = get_message_list(message_map, message.get("id"))
 | 
			
		||||
 | 
			
		||||
                    if tasks:
 | 
			
		||||
                        if (
 | 
			
		||||
                            TASKS.TITLE_GENERATION in tasks
 | 
			
		||||
                            and tasks[TASKS.TITLE_GENERATION]
 | 
			
		||||
                        ):
 | 
			
		||||
                            res = await generate_title(
 | 
			
		||||
                                request,
 | 
			
		||||
                                {
 | 
			
		||||
                                    "model": message["model"],
 | 
			
		||||
                                    "messages": messages,
 | 
			
		||||
                                    "chat_id": metadata["chat_id"],
 | 
			
		||||
                                },
 | 
			
		||||
                                user,
 | 
			
		||||
                            )
 | 
			
		||||
 | 
			
		||||
                            if res and isinstance(res, dict):
 | 
			
		||||
                                title = (
 | 
			
		||||
                                    res.get("choices", [])[0]
 | 
			
		||||
                                    .get("message", {})
 | 
			
		||||
                                    .get("content", message.get("content", "New Chat"))
 | 
			
		||||
                        if TASKS.TITLE_GENERATION in tasks:
 | 
			
		||||
                            if tasks[TASKS.TITLE_GENERATION]:
 | 
			
		||||
                                res = await generate_title(
 | 
			
		||||
                                    request,
 | 
			
		||||
                                    {
 | 
			
		||||
                                        "model": message["model"],
 | 
			
		||||
                                        "messages": messages,
 | 
			
		||||
                                        "chat_id": metadata["chat_id"],
 | 
			
		||||
                                    },
 | 
			
		||||
                                    user,
 | 
			
		||||
                                )
 | 
			
		||||
 | 
			
		||||
                                if res and isinstance(res, dict):
 | 
			
		||||
                                    title = (
 | 
			
		||||
                                        res.get("choices", [])[0]
 | 
			
		||||
                                        .get("message", {})
 | 
			
		||||
                                        .get(
 | 
			
		||||
                                            "content",
 | 
			
		||||
                                            message.get("content", "New Chat"),
 | 
			
		||||
                                        )
 | 
			
		||||
                                    )
 | 
			
		||||
 | 
			
		||||
                                    Chats.update_chat_title_by_id(
 | 
			
		||||
                                        metadata["chat_id"], title
 | 
			
		||||
                                    )
 | 
			
		||||
 | 
			
		||||
                                    await event_emitter(
 | 
			
		||||
                                        {
 | 
			
		||||
                                            "type": "chat:title",
 | 
			
		||||
                                            "data": title,
 | 
			
		||||
                                        }
 | 
			
		||||
                                    )
 | 
			
		||||
                            elif len(messages) == 2:
 | 
			
		||||
                                title = messages[0].get("content", "New Chat")
 | 
			
		||||
 | 
			
		||||
                                Chats.update_chat_title_by_id(
 | 
			
		||||
                                    metadata["chat_id"], title
 | 
			
		||||
                                )
 | 
			
		||||
@ -630,7 +645,7 @@ async def process_chat_response(request, response, user, events, metadata, tasks
 | 
			
		||||
                                await event_emitter(
 | 
			
		||||
                                    {
 | 
			
		||||
                                        "type": "chat:title",
 | 
			
		||||
                                        "data": title,
 | 
			
		||||
                                        "data": message.get("content", "New Chat"),
 | 
			
		||||
                                    }
 | 
			
		||||
                                )
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										34
									
								
								src/lib/components/NotificationToast.svelte
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								src/lib/components/NotificationToast.svelte
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,34 @@
 | 
			
		||||
<script lang="ts">
 | 
			
		||||
	import DOMPurify from 'dompurify';
 | 
			
		||||
 | 
			
		||||
	import { marked } from 'marked';
 | 
			
		||||
	import { createEventDispatcher } from 'svelte';
 | 
			
		||||
 | 
			
		||||
	const dispatch = createEventDispatcher();
 | 
			
		||||
 | 
			
		||||
	export let onClick: Function = () => {};
 | 
			
		||||
	export let title: string = 'HI';
 | 
			
		||||
	export let content: string;
 | 
			
		||||
</script>
 | 
			
		||||
 | 
			
		||||
<button
 | 
			
		||||
	class="flex gap-3 text-left w-[var(--width)] dark:bg-gray-850 dark:text-white bg-white text-black border border-gray-50 dark:border-gray-800 rounded-xl px-3 py-4"
 | 
			
		||||
	on:click={() => {
 | 
			
		||||
		onClick();
 | 
			
		||||
		dispatch('closeToast');
 | 
			
		||||
	}}
 | 
			
		||||
>
 | 
			
		||||
	<div class="flex-shrink-0 self-top -translate-y-1">
 | 
			
		||||
		<img src={'/static/favicon.png'} alt="favicon" class="size-8 rounded-full" />
 | 
			
		||||
	</div>
 | 
			
		||||
 | 
			
		||||
	<div>
 | 
			
		||||
		{#if title}
 | 
			
		||||
			<div class=" text-[13px] font-medium mb-0.5 line-clamp-1">{title}</div>
 | 
			
		||||
		{/if}
 | 
			
		||||
 | 
			
		||||
		<div class=" line-clamp-2 text-xs self-center dark:text-gray-300 font-normal">
 | 
			
		||||
			{@html DOMPurify.sanitize(marked(content))}
 | 
			
		||||
		</div>
 | 
			
		||||
	</div>
 | 
			
		||||
</button>
 | 
			
		||||
@ -79,6 +79,7 @@
 | 
			
		||||
	import EventConfirmDialog from '../common/ConfirmDialog.svelte';
 | 
			
		||||
	import Placeholder from './Placeholder.svelte';
 | 
			
		||||
	import { getTools } from '$lib/apis/tools';
 | 
			
		||||
	import NotificationToast from '../NotificationToast.svelte';
 | 
			
		||||
 | 
			
		||||
	export let chatIdProp = '';
 | 
			
		||||
 | 
			
		||||
@ -308,7 +309,23 @@
 | 
			
		||||
			const type = event?.data?.type ?? null;
 | 
			
		||||
			const data = event?.data?.data ?? null;
 | 
			
		||||
 | 
			
		||||
			if (type === 'chat:title') {
 | 
			
		||||
			if (type === 'chat:completion') {
 | 
			
		||||
				const { done, content, title } = data;
 | 
			
		||||
 | 
			
		||||
				if (done) {
 | 
			
		||||
					toast.custom(NotificationToast, {
 | 
			
		||||
						componentProps: {
 | 
			
		||||
							onClick: () => {
 | 
			
		||||
								goto(`/c/${event.chat_id}`);
 | 
			
		||||
							},
 | 
			
		||||
							content: content,
 | 
			
		||||
							title: title
 | 
			
		||||
						},
 | 
			
		||||
						duration: 15000,
 | 
			
		||||
						unstyled: true
 | 
			
		||||
					});
 | 
			
		||||
				}
 | 
			
		||||
			} else if (type === 'chat:title') {
 | 
			
		||||
				currentChatPage.set(1);
 | 
			
		||||
				await chats.set(await getChatList(localStorage.token, $currentChatPage));
 | 
			
		||||
			} else if (type === 'chat:tags') {
 | 
			
		||||
 | 
			
		||||
@ -219,5 +219,5 @@
 | 
			
		||||
				: 'light'
 | 
			
		||||
			: 'light'}
 | 
			
		||||
	richColors
 | 
			
		||||
	position="top-center"
 | 
			
		||||
	position="top-right"
 | 
			
		||||
/>
 | 
			
		||||
 | 
			
		||||
		Loading…
	
		Reference in New Issue
	
	Block a user