diff --git a/ragna/_cli/core.py b/ragna/_cli/core.py index 2e9030df..64d89e25 100644 --- a/ragna/_cli/core.py +++ b/ragna/_cli/core.py @@ -1,7 +1,6 @@ from pathlib import Path from typing import Annotated, Optional -import httpx import rich import typer import uvicorn @@ -74,13 +73,12 @@ def deploy( *, config: ConfigOption = "./ragna.toml", # type: ignore[assignment] api: Annotated[ - Optional[bool], + bool, typer.Option( "--api/--no-api", help="Deploy the Ragna REST API.", - show_default="True if UI is not deployed and otherwise check availability", ), - ] = None, + ] = True, ui: Annotated[ bool, typer.Option( @@ -98,22 +96,14 @@ def deploy( ] = False, open_browser: Annotated[ Optional[bool], - typer.Option(help="Open a browser when Ragna is deployed."), + typer.Option( + help="Open a browser when Ragna is deployed.", + show_default="value of ui / no-ui", + ), ] = None, ) -> None: - def api_available() -> bool: - try: - return httpx.get(f"{config._url}/health").is_success - except httpx.ConnectError: - return False - - if api is None: - api = not api_available() if ui else True - if not (api or ui): raise Exception - elif ui and not api and not api_available(): - raise Exception if open_browser is None: open_browser = ui diff --git a/ragna/deploy/_core.py b/ragna/deploy/_core.py index 6df4b71b..44c672a8 100644 --- a/ragna/deploy/_core.py +++ b/ragna/deploy/_core.py @@ -79,7 +79,7 @@ def server_available() -> bool: app.include_router(make_api_router(engine), prefix="/api") if ui: - panel_app = make_ui_app(config=config) + panel_app = make_ui_app(engine) panel_app.serve_with_fastapi(app, endpoint="/ui") @app.get("/", include_in_schema=False) diff --git a/ragna/deploy/_ui/api_wrapper.py b/ragna/deploy/_ui/api_wrapper.py index 5fb7b42d..170e8bbd 100644 --- a/ragna/deploy/_ui/api_wrapper.py +++ b/ragna/deploy/_ui/api_wrapper.py @@ -1,62 +1,53 @@ -import json +import uuid from datetime import datetime import emoji -import httpx import param +from ragna.core._utils import default_user +from ragna.deploy import _schemas as schemas +from ragna.deploy._engine import Engine -# The goal is this class is to provide ready-to-use functions to interact with the API -class ApiWrapper(param.Parameterized): - def __init__(self, api_url, **params): - self.api_url = api_url - self.client = httpx.AsyncClient(base_url=api_url, timeout=60) - super().__init__(**params) +class ApiWrapper(param.Parameterized): + def __init__(self, engine: Engine): + super().__init__() + self._user = default_user() + self._engine = engine async def get_chats(self): - json_data = (await self.client.get("/chats")).raise_for_status().json() + json_data = [ + chat.model_dump(mode="json") + for chat in self._engine.get_chats(user=self._user) + ] for chat in json_data: chat["messages"] = [self.improve_message(msg) for msg in chat["messages"]] return json_data async def answer(self, chat_id, prompt): - async with self.client.stream( - "POST", - f"/chats/{chat_id}/answer", - json={"prompt": prompt, "stream": True}, - ) as response: - async for data in response.aiter_lines(): - yield self.improve_message(json.loads(data)) + async for message in self._engine.answer_stream( + user=self._user, chat_id=uuid.UUID(chat_id), prompt=prompt + ): + yield self.improve_message(message.model_dump(mode="json")) async def get_components(self): - return (await self.client.get("/components")).raise_for_status().json() - - # Upload and related functions - def upload_endpoints(self): - return { - "informations_endpoint": f"{self.api_url}/document", - } + return self._engine.get_components().model_dump(mode="json") async def start_and_prepare( self, name, documents, source_storage, assistant, params ): - response = await self.client.post( - "/chats", - json={ - "name": name, - "documents": documents, - "source_storage": source_storage, - "assistant": assistant, - "params": params, - }, + chat = self._engine.create_chat( + user=self._user, + chat_creation=schemas.ChatCreation( + name=name, + document_ids=[document.id for document in documents], + source_storage=source_storage, + assistant=assistant, + params=params, + ), ) - chat = response.raise_for_status().json() - - response = await self.client.post(f"/chats/{chat['id']}/prepare", timeout=None) - response.raise_for_status() - - return chat["id"] + await self._engine.prepare_chat(user=self._user, id=chat.id) + return str(chat.id) def improve_message(self, msg): msg["timestamp"] = datetime.strptime(msg["timestamp"], "%Y-%m-%dT%H:%M:%S.%f") diff --git a/ragna/deploy/_ui/app.py b/ragna/deploy/_ui/app.py index 49b8d628..052ff36d 100644 --- a/ragna/deploy/_ui/app.py +++ b/ragna/deploy/_ui/app.py @@ -6,7 +6,7 @@ from fastapi import FastAPI from fastapi.staticfiles import StaticFiles -from ragna.deploy import Config +from ragna.deploy._engine import Engine from . import js from . import styles as ui @@ -22,13 +22,13 @@ class App(param.Parameterized): - def __init__(self, *, api_url): + def __init__(self, engine: Engine): super().__init__() # Apply the design modifiers to the panel components # It returns all the CSS files of the modifiers self.css_filepaths = ui.apply_design_modifiers() - self.api_url = api_url + self._engine = engine def get_template(self): # A bit hacky, but works. @@ -73,7 +73,7 @@ def get_template(self): return template def index_page(self): - api_wrapper = ApiWrapper(api_url=self.api_url) + api_wrapper = ApiWrapper(self._engine) template = self.get_template() main_page = MainPage(api_wrapper=api_wrapper, template=template) @@ -131,7 +131,7 @@ def get_component_resource(path: str): def serve_with_fastapi(self, app: FastAPI, endpoint: str): self.add_panel_app(app, self.index_page, endpoint) - for dir in ["css", "imgs", "resources"]: + for dir in ["css", "imgs"]: app.mount( f"/{dir}", StaticFiles(directory=str(Path(__file__).parent / dir)), @@ -139,5 +139,5 @@ def serve_with_fastapi(self, app: FastAPI, endpoint: str): ) -def app(*, config: Config) -> App: - return App(api_url=f"{config._url}/api") +def app(engine: Engine) -> App: + return App(engine) diff --git a/ragna/deploy/_ui/central_view.py b/ragna/deploy/_ui/central_view.py index ae0cefc3..00de3e8e 100644 --- a/ragna/deploy/_ui/central_view.py +++ b/ragna/deploy/_ui/central_view.py @@ -189,11 +189,11 @@ def on_click_chat_info_wrapper(self, event): pills = "".join( [ f"""
{d['name']}
""" - for d in self.current_chat["metadata"]["documents"] + for d in self.current_chat["documents"] ] ) - grid_height = len(self.current_chat["metadata"]["documents"]) // 3 + grid_height = len(self.current_chat["documents"]) // 3 markdown = "\n".join( [ @@ -202,14 +202,14 @@ def on_click_chat_info_wrapper(self, event): f"
{pills}

\n\n", "----", "**Source Storage**", - f"""{self.current_chat['metadata']['source_storage']}\n""", + f"""{self.current_chat['source_storage']}\n""", "----", "**Assistant**", - f"""{self.current_chat['metadata']['assistant']}\n""", + f"""{self.current_chat['assistant']}\n""", "**Advanced configuration**", *[ f"- **{key.replace('_', ' ').title()}**: {value}" - for key, value in self.current_chat["metadata"]["params"].items() + for key, value in self.current_chat["params"].items() ], ] ) @@ -275,7 +275,7 @@ def get_user_from_role(self, role: Literal["system", "user", "assistant"]) -> st elif role == "user": return cast(str, self.user) elif role == "assistant": - return cast(str, self.current_chat["metadata"]["assistant"]) + return cast(str, self.current_chat["assistant"]) else: raise RuntimeError @@ -301,12 +301,15 @@ async def chat_callback( message.clipboard_button.value = message.content_pane.object message.assistant_toolbar.visible = True - except Exception: + except Exception as exc: + import traceback + yield RagnaChatMessage( - ( - "Sorry, something went wrong. " - "If this problem persists, please contact your administrator." - ), + # ( + # "Sorry, something went wrong. " + # "If this problem persists, please contact your administrator." + # ), + "".join(traceback.format_exception(type(exc), exc, exc.__traceback__)), role="system", user=self.get_user_from_role("system"), ) @@ -358,7 +361,7 @@ def header(self): current_chat_name = "" if self.current_chat is not None: - current_chat_name = self.current_chat["metadata"]["name"] + current_chat_name = self.current_chat["name"] chat_name_header = pn.pane.HTML( f"

{current_chat_name}

", @@ -370,9 +373,9 @@ def header(self): if ( self.current_chat is not None and "metadata" in self.current_chat - and "documents" in self.current_chat["metadata"] + and "documents" in self.current_chat ): - doc_names = [d["name"] for d in self.current_chat["metadata"]["documents"]] + doc_names = [d["name"] for d in self.current_chat["documents"]] # FIXME: Instead of setting a hard limit of 20 documents here, this should # scale automatically with the width of page @@ -385,7 +388,9 @@ def header(self): chat_documents_pills.append(pill) - self.chat_info_button.name = f"{self.current_chat['metadata']['assistant']} | {self.current_chat['metadata']['source_storage']}" + self.chat_info_button.name = ( + f"{self.current_chat['assistant']} | {self.current_chat['source_storage']}" + ) return pn.Row( chat_name_header, diff --git a/ragna/deploy/_ui/left_sidebar.py b/ragna/deploy/_ui/left_sidebar.py index ab8bc1c0..21c747aa 100644 --- a/ragna/deploy/_ui/left_sidebar.py +++ b/ragna/deploy/_ui/left_sidebar.py @@ -62,7 +62,7 @@ def __panel__(self): self.chat_buttons = [] for chat in self.chats: button = pn.widgets.Button( - name=chat["metadata"]["name"], + name=chat["name"], css_classes=["chat_button"], ) button.on_click(lambda event, c=chat: self.on_click_chat_wrapper(event, c)) diff --git a/ragna/deploy/_ui/modal_configuration.py b/ragna/deploy/_ui/modal_configuration.py index 51ec02fb..a8bb44e7 100644 --- a/ragna/deploy/_ui/modal_configuration.py +++ b/ragna/deploy/_ui/modal_configuration.py @@ -1,11 +1,13 @@ from datetime import datetime, timedelta, timezone +from typing import AsyncIterator import panel as pn import param +from ragna.deploy import _schemas as schemas + from . import js from . import styles as ui -from .components.file_uploader import FileUploader def get_default_chat_name(timezone_offset=None): @@ -82,15 +84,11 @@ def __init__(self, api_wrapper, **params): self.api_wrapper = api_wrapper - upload_endpoints = self.api_wrapper.upload_endpoints() - self.chat_name_input = pn.widgets.TextInput.from_param( self.param.chat_name, ) - self.document_uploader = FileUploader( - [], # the allowed documents are set in the model_section function - upload_endpoints["informations_endpoint"], - ) + # FIXME: accept + self.document_uploader = pn.widgets.FileInput(multiple=True) # Most widgets (including those that use from_param) should be placed after the super init call self.cancel_button = pn.widgets.Button( @@ -114,12 +112,38 @@ def __init__(self, api_wrapper, **params): self.got_timezone = False - def did_click_on_start_chat_button(self, event): - if not self.document_uploader.can_proceed_to_upload(): + async def did_click_on_start_chat_button(self, event): + if not self.document_uploader.value: self.change_upload_files_label("missing_file") else: self.start_chat_button.disabled = True - self.document_uploader.perform_upload(event, self.did_finish_upload) + documents = self.api_wrapper._engine.register_documents( + user=self.api_wrapper._user, + document_registrations=[ + schemas.DocumentRegistration(name=name) + for name in self.document_uploader.filename + ], + ) + + if self.api_wrapper._engine.supports_store_documents: + + def make_content_stream(data: bytes) -> AsyncIterator[bytes]: + async def content_stream() -> AsyncIterator[bytes]: + yield data + + return content_stream() + + await self.api_wrapper._engine.store_documents( + user=self.api_wrapper._user, + ids_and_streams=[ + (document.id, make_content_stream(data)) + for document, data in zip( + documents, self.document_uploader.value + ) + ], + ) + + await self.did_finish_upload(documents) async def did_finish_upload(self, uploaded_documents): # at this point, the UI has uploaded the files to the API. diff --git a/tests/deploy/ui/test_ui.py b/tests/deploy/ui/test_ui.py index 85699278..e815011b 100644 --- a/tests/deploy/ui/test_ui.py +++ b/tests/deploy/ui/test_ui.py @@ -1,14 +1,13 @@ +import contextlib +import multiprocessing import socket -import subprocess -import sys import time import httpx -import panel as pn import pytest -from playwright.sync_api import Page, expect +from playwright.sync_api import expect -from ragna._utils import timeout_after +from ragna._cli.core import deploy as _deploy from ragna.deploy import Config from tests.deploy.utils import TestAssistant @@ -19,139 +18,118 @@ def get_available_port(): return s.getsockname()[1] +@contextlib.contextmanager +def deploy(config): + process = multiprocessing.Process( + target=_deploy, + kwargs=dict( + config=config, + api=False, + ui=True, + ignore_unavailable_components=False, + open_browser=False, + ), + ) + try: + process.start() + + client = httpx.Client(base_url=config._url) + + # FIXME: create a generic utility for this + def server_available() -> bool: + try: + return client.get("/health").is_success + except httpx.ConnectError: + return False + + while not server_available(): + time.sleep(0.1) + + yield process + finally: + process.terminate() + process.join() + process.close() + + @pytest.fixture -def config( - tmp_local_root, -): - config = Config( +def default_config(tmp_local_root): + return Config( local_root=tmp_local_root, assistants=[TestAssistant], - ui=dict(port=get_available_port()), - api=dict(port=get_available_port()), + port=get_available_port(), ) - path = tmp_local_root / "ragna.toml" - config.to_file(path) - return config - - -class Server: - def __init__(self, config): - self.config = config - self.base_url = f"http://{config.ui.hostname}:{config.ui.port}" - - def server_up(self): - try: - return httpx.get(self.base_url).is_success - except httpx.ConnectError: - return False - - @timeout_after(60) - def start(self): - self.proc = subprocess.Popen( - [ - sys.executable, - "-m", - "ragna", - "ui", - "--config", - self.config.local_root / "ragna.toml", - "--start-api", - "--ignore-unavailable-components", - "--no-open-browser", - ], - stdout=sys.stdout, - stderr=sys.stderr, - ) - - while not self.server_up(): - time.sleep(1) - - def stop(self): - self.proc.kill() - pn.state.kill_all_servers() - - def __enter__(self): - self.start() - return self - - def __exit__(self, *args): - self.stop() - - -def test_health(config, page: Page) -> None: - with Server(config) as server: - health_url = f"{server.base_url}/health" - response = page.goto(health_url) - assert response.ok - - -def test_start_chat(config, page: Page) -> None: - with Server(config) as server: - # Index page, no auth - index_url = server.base_url - page.goto(index_url) - expect(page.get_by_role("button", name="Sign In")).to_be_visible() - - # Authorize with no credentials - page.get_by_role("button", name="Sign In").click() - expect(page.get_by_role("button", name=" New Chat")).to_be_visible() - - # expect auth token to be set - cookies = page.context.cookies() - assert len(cookies) == 1 - cookie = cookies[0] - assert cookie.get("name") == "auth_token" - auth_token = cookie.get("value") - assert auth_token is not None - - # New page button - new_chat_button = page.get_by_role("button", name=" New Chat") - expect(new_chat_button).to_be_visible() - new_chat_button.click() - - document_root = config.local_root / "documents" - document_root.mkdir() - document_name = "test.txt" - document_path = document_root / document_name - with open(document_path, "w") as file: - file.write("!\n") - - # File upload selector - with page.expect_file_chooser() as fc_info: - page.locator(".fileUpload").click() - file_chooser = fc_info.value - file_chooser.set_files(document_path) - - # Upload document and expect to see it listed - file_list = page.locator(".fileListContainer") - expect(file_list.first).to_have_text(str(document_name)) - - chat_dialog = page.get_by_role("dialog") - expect(chat_dialog).to_be_visible() - start_chat_button = page.get_by_role("button", name="Start Conversation") - expect(start_chat_button).to_be_visible() - time.sleep(0.5) # hack while waiting for button to be fully clickable - start_chat_button.click(delay=5) - - chat_box_row = page.locator(".chat-interface-input-row") - expect(chat_box_row).to_be_visible() - - chat_box = chat_box_row.get_by_role("textbox") - expect(chat_box).to_be_visible() - - # Document should be in the database - chats_url = f"http://{config.api.hostname}:{config.api.port}/chats" - chats = httpx.get( - chats_url, headers={"Authorization": f"Bearer {auth_token}"} - ).json() - assert len(chats) == 1 - chat = chats[0] - chat_documents = chat["metadata"]["documents"] - assert len(chat_documents) == 1 - assert chat_documents[0]["name"] == document_name - - chat_box.fill("Tell me about the documents") - - chat_button = chat_box_row.get_by_role("button") - expect(chat_button).to_be_visible() - chat_button.click() + + +@pytest.fixture +def index_page(default_config, page): + config = default_config + with deploy(default_config): + page.goto(f"http://{config.hostname}:{config.port}/ui") + yield page + + +def test_start_chat(index_page, tmp_path) -> None: + # expect(page.get_by_role("button", name="Sign In")).to_be_visible() + + # # Authorize with no credentials + # page.get_by_role("button", name="Sign In").click() + # expect(page.get_by_role("button", name=" New Chat")).to_be_visible() + # + # # expect auth token to be set + # cookies = page.context.cookies() + # assert len(cookies) == 1 + # cookie = cookies[0] + # assert cookie.get("name") == "auth_token" + # auth_token = cookie.get("value") + # assert auth_token is not None + + # New page button + new_chat_button = index_page.get_by_role("button", name=" New Chat") + expect(new_chat_button).to_be_visible() + new_chat_button.click() + + # document_name = "test.txt" + # document_path = tmp_path / document_name + # with open(document_path, "w") as file: + # file.write("!\n") + + # # File upload selector + # with index_page.expect_file_chooser() as fc_info: + # index_page.locator(".fileUpload").click() + # file_chooser = fc_info.value + # file_chooser.set_files(document_path) + + # # Upload document and expect to see it listed + # file_list = page.locator(".fileListContainer") + # expect(file_list.first).to_have_text(str(document_name)) + # + # chat_dialog = page.get_by_role("dialog") + # expect(chat_dialog).to_be_visible() + # start_chat_button = page.get_by_role("button", name="Start Conversation") + # expect(start_chat_button).to_be_visible() + # time.sleep(0.5) # hack while waiting for button to be fully clickable + # start_chat_button.click(delay=5) + # + # chat_box_row = page.locator(".chat-interface-input-row") + # expect(chat_box_row).to_be_visible() + # + # chat_box = chat_box_row.get_by_role("textbox") + # expect(chat_box).to_be_visible() + # + # # Document should be in the database + # chats_url = f"http://{config.api.hostname}:{config.api.port}/chats" + # chats = httpx.get( + # chats_url, headers={"Authorization": f"Bearer {auth_token}"} + # ).json() + # assert len(chats) == 1 + # chat = chats[0] + # chat_documents = chat["metadata"]["documents"] + # assert len(chat_documents) == 1 + # assert chat_documents[0]["name"] == document_name + # + # chat_box.fill("Tell me about the documents") + # + # chat_button = chat_box_row.get_by_role("button") + # expect(chat_button).to_be_visible() + # chat_button.click()