From 85072ac65a997152f7ef2d8f8d7e57791ef67188 Mon Sep 17 00:00:00 2001 From: Stavros Date: Tue, 14 May 2024 15:21:09 +0300 Subject: [PATCH] feat: openwebui app (#3491) * feat: openwebui app * refactor(openwebui): use text instead of fqdnip * refactor(openwebui): change logo to light mode * refactor(openwebui): remove wrong option from compose json --- apps/open-webui/config.json | 32 ++++++++ apps/open-webui/docker-compose.json | 17 +++++ apps/open-webui/docker-compose.yml | 40 ++++++++++ apps/open-webui/metadata/description.md | 97 ++++++++++++++++++++++++ apps/open-webui/metadata/logo.jpg | Bin 0 -> 10624 bytes 5 files changed, 186 insertions(+) create mode 100644 apps/open-webui/config.json create mode 100644 apps/open-webui/docker-compose.json create mode 100644 apps/open-webui/docker-compose.yml create mode 100644 apps/open-webui/metadata/description.md create mode 100644 apps/open-webui/metadata/logo.jpg diff --git a/apps/open-webui/config.json b/apps/open-webui/config.json new file mode 100644 index 00000000..5163353d --- /dev/null +++ b/apps/open-webui/config.json @@ -0,0 +1,32 @@ +{ + "$schema": "../schema.json", + "name": "Open WebUI", + "available": true, + "exposable": true, + "port": 8536, + "id": "open-webui", + "tipi_version": 1, + "version": "git-90503be", + "categories": ["ai"], + "description": "Open WebUI is an extensible, feature-rich, and user-friendly self-hosted WebUI designed to operate entirely offline.", + "short_desc": "User-friendly WebUI for LLMs", + "author": "open-webui", + "source": "https://github.com/open-webui/open-webui", + "website": "https://openwebui.com/", + "form_fields": [ + { + "type": "text", + "label": "Ollama Api Url", + "placeholder": "http://ollama-cpu:11434", + "required": false, + "env_variable": "OPENWEBUI_OLLAMA_URL" + }, + { + "type": "password", + "label": "Openai Api Key", + "required": false, + "env_variable": "OPENWEBUI_OPENAI_KEY" + } + ], + "supported_architectures": ["arm64", "amd64"] +} diff --git a/apps/open-webui/docker-compose.json b/apps/open-webui/docker-compose.json new file mode 100644 index 00000000..8a40005d --- /dev/null +++ b/apps/open-webui/docker-compose.json @@ -0,0 +1,17 @@ +{ + "openPort": true, + "image": "ghcr.io/open-webui/open-webui:git-90503be", + "name": "open-webui", + "internalPort": "8080", + "isMain": true, + "volumes": [ + { + "hostPath": "${APP_DATA_DIR}/data", + "containerPath": "/app/backend/data" + } + ], + "environment": { + "OLLAMA_BASE_URL": "${OPENWEBUI_OLLAMA_URL}", + "OPENAI_API_KEY": "${OPENWEBUI_OPENAI_KEY}" + } +} diff --git a/apps/open-webui/docker-compose.yml b/apps/open-webui/docker-compose.yml new file mode 100644 index 00000000..ea90f86e --- /dev/null +++ b/apps/open-webui/docker-compose.yml @@ -0,0 +1,40 @@ +version: "3.8" +services: + open-webui: + image: ghcr.io/open-webui/open-webui:git-90503be + container_name: open-webui + volumes: + - ${APP_DATA_DIR}/data:/app/backend/data + ports: + - ${APP_PORT}:8080 + environment: + - OLLAMA_BASE_URL=${OPENWEBUI_OLLAMA_URL} + - OPENAI_API_KEY=${OPENWEBUI_OPENAI_KEY} + restart: unless-stopped + networks: + - tipi_main_network + labels: + # Main + traefik.enable: true + traefik.http.middlewares.open-webui-web-redirect.redirectscheme.scheme: https + traefik.http.services.open-webui.loadbalancer.server.port: 8080 + # Web + traefik.http.routers.open-webui-insecure.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.open-webui-insecure.entrypoints: web + traefik.http.routers.open-webui-insecure.service: open-webui + traefik.http.routers.open-webui-insecure.middlewares: open-webui-web-redirect + # Websecure + traefik.http.routers.open-webui.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.open-webui.entrypoints: websecure + traefik.http.routers.open-webui.service: open-webui + traefik.http.routers.open-webui.tls.certresolver: myresolver + # Local domain + traefik.http.routers.open-webui-local-insecure.rule: Host(`open-webui.${LOCAL_DOMAIN}`) + traefik.http.routers.open-webui-local-insecure.entrypoints: web + traefik.http.routers.open-webui-local-insecure.service: open-webui + traefik.http.routers.open-webui-local-insecure.middlewares: open-webui-web-redirect + # Local domain secure + traefik.http.routers.open-webui-local.rule: Host(`open-webui.${LOCAL_DOMAIN}`) + traefik.http.routers.open-webui-local.entrypoints: websecure + traefik.http.routers.open-webui-local.service: open-webui + traefik.http.routers.open-webui-local.tls: true diff --git a/apps/open-webui/metadata/description.md b/apps/open-webui/metadata/description.md new file mode 100644 index 00000000..b1ea9fa5 --- /dev/null +++ b/apps/open-webui/metadata/description.md @@ -0,0 +1,97 @@ +## Open WebUI (Formerly Ollama WebUI) 👋 + +Open WebUI is an extensible, feature-rich, and user-friendly self-hosted WebUI designed to operate entirely offline. It supports various LLM runners, including Ollama and OpenAI-compatible APIs. For more information, be sure to check out our [Open WebUI Documentation](https://docs.openwebui.com/). + +![Open WebUI Demo](https://github.com/open-webui/open-webui/blob/main/demo.gif?raw=true) + +### Features ⭐ + +- 🖥️ **Intuitive Interface**: Our chat interface takes inspiration from ChatGPT, ensuring a user-friendly experience. + +- 📱 **Responsive Design**: Enjoy a seamless experience on both desktop and mobile devices. + +- ⚡ **Swift Responsiveness**: Enjoy fast and responsive performance. + +- 🚀 **Effortless Setup**: Install seamlessly using Docker or Kubernetes (kubectl, kustomize or helm) for a hassle-free experience. + +- 🌈 **Theme Customization**: Choose from a variety of themes to personalize your Open WebUI experience. + +- 💻 **Code Syntax Highlighting**: Enjoy enhanced code readability with our syntax highlighting feature. + +- ✒️🔢 **Full Markdown and LaTeX Support**: Elevate your LLM experience with comprehensive Markdown and LaTeX capabilities for enriched interaction. + +- 📚 **Local RAG Integration**: Dive into the future of chat interactions with the groundbreaking Retrieval Augmented Generation (RAG) support. This feature seamlessly integrates document interactions into your chat experience. You can load documents directly into the chat or add files to your document library, effortlessly accessing them using `#` command in the prompt. In its alpha phase, occasional issues may arise as we actively refine and enhance this feature to ensure optimal performance and reliability. + +- 🔍 **RAG Embedding Support**: Change the RAG embedding model directly in document settings, enhancing document processing. This feature supports Ollama and OpenAI models. + +- 🌐 **Web Browsing Capability**: Seamlessly integrate websites into your chat experience using the `#` command followed by the URL. This feature allows you to incorporate web content directly into your conversations, enhancing the richness and depth of your interactions. + +- 📜 **Prompt Preset Support**: Instantly access preset prompts using the `/` command in the chat input. Load predefined conversation starters effortlessly and expedite your interactions. Effortlessly import prompts through [Open WebUI Community](https://openwebui.com/) integration. + +- 👍👎 **RLHF Annotation**: Empower your messages by rating them with thumbs up and thumbs down, followed by the option to provide textual feedback, facilitating the creation of datasets for Reinforcement Learning from Human Feedback (RLHF). Utilize your messages to train or fine-tune models, all while ensuring the confidentiality of locally saved data. + +- 🏷️ **Conversation Tagging**: Effortlessly categorize and locate specific chats for quick reference and streamlined data collection. + +- 📥🗑️ **Download/Delete Models**: Easily download or remove models directly from the web UI. + +- 🔄 **Update All Ollama Models**: Easily update locally installed models all at once with a convenient button, streamlining model management. + +- ⬆️ **GGUF File Model Creation**: Effortlessly create Ollama models by uploading GGUF files directly from the web UI. Streamlined process with options to upload from your machine or download GGUF files from Hugging Face. + +- 🤖 **Multiple Model Support**: Seamlessly switch between different chat models for diverse interactions. + +- 🔄 **Multi-Modal Support**: Seamlessly engage with models that support multimodal interactions, including images (e.g., LLava). + +- 🧩 **Modelfile Builder**: Easily create Ollama modelfiles via the web UI. Create and add characters/agents, customize chat elements, and import modelfiles effortlessly through [Open WebUI Community](https://openwebui.com/) integration. + +- ⚙️ **Many Models Conversations**: Effortlessly engage with various models simultaneously, harnessing their unique strengths for optimal responses. Enhance your experience by leveraging a diverse set of models in parallel. + +- 💬 **Collaborative Chat**: Harness the collective intelligence of multiple models by seamlessly orchestrating group conversations. Use the `@` command to specify the model, enabling dynamic and diverse dialogues within your chat interface. Immerse yourself in the collective intelligence woven into your chat environment. + +- 🗨️ **Local Chat Sharing**: Generate and share chat links seamlessly between users, enhancing collaboration and communication. + +- 🔄 **Regeneration History Access**: Easily revisit and explore your entire regeneration history. + +- 📜 **Chat History**: Effortlessly access and manage your conversation history. + +- 📬 **Archive Chats**: Effortlessly store away completed conversations with LLMs for future reference, maintaining a tidy and clutter-free chat interface while allowing for easy retrieval and reference. + +- 📤📥 **Import/Export Chat History**: Seamlessly move your chat data in and out of the platform. + +- 🗣️ **Voice Input Support**: Engage with your model through voice interactions; enjoy the convenience of talking to your model directly. Additionally, explore the option for sending voice input automatically after 3 seconds of silence for a streamlined experience. + +- 🔊 **Configurable Text-to-Speech Endpoint**: Customize your Text-to-Speech experience with configurable OpenAI endpoints. + +- ⚙️ **Fine-Tuned Control with Advanced Parameters**: Gain a deeper level of control by adjusting parameters such as temperature and defining your system prompts to tailor the conversation to your specific preferences and needs. + +- 🎨🤖 **Image Generation Integration**: Seamlessly incorporate image generation capabilities using options such as AUTOMATIC1111 API (local), ComfyUI (local), and DALL-E, enriching your chat experience with dynamic visual content. + +- 🤝 **OpenAI API Integration**: Effortlessly integrate OpenAI-compatible API for versatile conversations alongside Ollama models. Customize the API Base URL to link with **LMStudio, Mistral, OpenRouter, and more**. + +- ✨ **Multiple OpenAI-Compatible API Support**: Seamlessly integrate and customize various OpenAI-compatible APIs, enhancing the versatility of your chat interactions. + +- 🔑 **API Key Generation Support**: Generate secret keys to leverage Open WebUI with OpenAI libraries, simplifying integration and development. + +- 🔗 **External Ollama Server Connection**: Seamlessly link to an external Ollama server hosted on a different address by configuring the environment variable. + +- 🔀 **Multiple Ollama Instance Load Balancing**: Effortlessly distribute chat requests across multiple Ollama instances for enhanced performance and reliability. + +- 👥 **Multi-User Management**: Easily oversee and administer users via our intuitive admin panel, streamlining user management processes. + +- 🔗 **Webhook Integration**: Subscribe to new user sign-up events via webhook (compatible with Google Chat and Microsoft Teams), providing real-time notifications and automation capabilities. + +- 🛡️ **Model Whitelisting**: Admins can whitelist models for users with the 'user' role, enhancing security and access control. + +- 📧 **Trusted Email Authentication**: Authenticate using a trusted email header, adding an additional layer of security and authentication. + +- 🔐 **Role-Based Access Control (RBAC)**: Ensure secure access with restricted permissions; only authorized individuals can access your Ollama, and exclusive model creation/pulling rights are reserved for administrators. + +- 🔒 **Backend Reverse Proxy Support**: Bolster security through direct communication between Open WebUI backend and Ollama. This key feature eliminates the need to expose Ollama over LAN. Requests made to the '/ollama/api' route from the web UI are seamlessly redirected to Ollama from the backend, enhancing overall system security. + +- 🌐🌍 **Multilingual Support**: Experience Open WebUI in your preferred language with our internationalization (i18n) support. Join us in expanding our supported languages! We're actively seeking contributors! + +- 🌟 **Continuous Updates**: We are committed to improving Open WebUI with regular updates and new features. + +## 🔗 Also Check Out Open WebUI Community! + +Don't forget to explore our sibling project, [Open WebUI Community](https://openwebui.com/), where you can discover, download, and explore customized Modelfiles. Open WebUI Community offers a wide range of exciting possibilities for enhancing your chat interactions with Open WebUI! 🚀 diff --git a/apps/open-webui/metadata/logo.jpg b/apps/open-webui/metadata/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..8df9a371652c07a4afc7b27328da6108f0f03879 GIT binary patch literal 10624 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 St%)t6%|%2hoRIQp{Qm)j4GpIN literal 0 HcmV?d00001