Skip to content

Commit

Permalink
Merge pull request open-webui#1209 from open-webui/dev
Browse files Browse the repository at this point in the history
0.1.114
  • Loading branch information
tjbck authored Mar 21, 2024
2 parents d865b9f + 8e52ba8 commit 2fa9495
Show file tree
Hide file tree
Showing 30 changed files with 2,318 additions and 257 deletions.
17 changes: 17 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,23 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).

## [0.1.114] - 2024-03-20

### Added

- **🔗 Webhook Integration**: Now you can subscribe to new user sign-up events via webhook. Simply navigate to the admin panel > admin settings > webhook URL.
- **🛡️ Enhanced Model Filtering**: Alongside Ollama, OpenAI proxy model whitelisting, we've added model filtering functionality for LiteLLM proxy.
- **🌍 Expanded Language Support**: Spanish, Catalan, and Vietnamese languages are now available, with improvements made to others.

### Fixed

- **🔧 Input Field Spelling**: Resolved issue with spelling mistakes in input fields.
- **🖊️ Light Mode Styling**: Fixed styling issue with light mode in document adding.

### Changed

- **🔄 Language Sorting**: Languages are now sorted alphabetically by their code for improved organization.

## [0.1.113] - 2024-03-18

### Added
Expand Down
68 changes: 61 additions & 7 deletions backend/apps/litellm/main.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,23 @@
from litellm.proxy.proxy_server import ProxyConfig, initialize
from litellm.proxy.proxy_server import app

from fastapi import FastAPI, Request, Depends, status
from fastapi import FastAPI, Request, Depends, status, Response
from fastapi.responses import JSONResponse

from starlette.middleware.base import BaseHTTPMiddleware, RequestResponseEndpoint
from starlette.responses import StreamingResponse
import json

from utils.utils import get_http_authorization_cred, get_current_user
from config import ENV


from config import (
MODEL_FILTER_ENABLED,
MODEL_FILTER_LIST,
)


proxy_config = ProxyConfig()


Expand All @@ -26,16 +38,58 @@ async def on_startup():
await startup()


app.state.MODEL_FILTER_ENABLED = MODEL_FILTER_ENABLED
app.state.MODEL_FILTER_LIST = MODEL_FILTER_LIST


@app.middleware("http")
async def auth_middleware(request: Request, call_next):
auth_header = request.headers.get("Authorization", "")
request.state.user = None

if ENV != "dev":
try:
user = get_current_user(get_http_authorization_cred(auth_header))
print(user)
except Exception as e:
return JSONResponse(status_code=400, content={"detail": str(e)})
try:
user = get_current_user(get_http_authorization_cred(auth_header))
print(user)
request.state.user = user
except Exception as e:
return JSONResponse(status_code=400, content={"detail": str(e)})

response = await call_next(request)
return response


class ModifyModelsResponseMiddleware(BaseHTTPMiddleware):
async def dispatch(
self, request: Request, call_next: RequestResponseEndpoint
) -> Response:

response = await call_next(request)
user = request.state.user

if "/models" in request.url.path:
if isinstance(response, StreamingResponse):
# Read the content of the streaming response
body = b""
async for chunk in response.body_iterator:
body += chunk

data = json.loads(body.decode("utf-8"))

if app.state.MODEL_FILTER_ENABLED:
if user and user.role == "user":
data["data"] = list(
filter(
lambda model: model["id"]
in app.state.MODEL_FILTER_LIST,
data["data"],
)
)

# Modified Flag
data["modified"] = True
return JSONResponse(content=data)

return response


app.add_middleware(ModifyModelsResponseMiddleware)
2 changes: 1 addition & 1 deletion backend/apps/ollama/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -709,7 +709,7 @@ class GenerateChatCompletionForm(BaseModel):
format: Optional[str] = None
options: Optional[dict] = None
template: Optional[str] = None
stream: Optional[bool] = True
stream: Optional[bool] = None
keep_alive: Optional[Union[int, str]] = None


Expand Down
2 changes: 2 additions & 0 deletions backend/apps/web/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
DEFAULT_USER_ROLE,
ENABLE_SIGNUP,
USER_PERMISSIONS,
WEBHOOK_URL,
)

app = FastAPI()
Expand All @@ -32,6 +33,7 @@
app.state.DEFAULT_PROMPT_SUGGESTIONS = DEFAULT_PROMPT_SUGGESTIONS
app.state.DEFAULT_USER_ROLE = DEFAULT_USER_ROLE
app.state.USER_PERMISSIONS = USER_PERMISSIONS
app.state.WEBHOOK_URL = WEBHOOK_URL


app.add_middleware(
Expand Down
14 changes: 13 additions & 1 deletion backend/apps/web/routers/auths.py
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,8 @@
create_token,
)
from utils.misc import parse_duration, validate_email_format
from constants import ERROR_MESSAGES
from utils.webhook import post_webhook
from constants import ERROR_MESSAGES, WEBHOOK_MESSAGES

router = APIRouter()

Expand Down Expand Up @@ -155,6 +156,17 @@ async def signup(request: Request, form_data: SignupForm):
)
# response.set_cookie(key='token', value=token, httponly=True)

if request.app.state.WEBHOOK_URL:
post_webhook(
request.app.state.WEBHOOK_URL,
WEBHOOK_MESSAGES.USER_SIGNUP(user.name),
{
"action": "signup",
"message": WEBHOOK_MESSAGES.USER_SIGNUP(user.name),
"user": user.model_dump_json(exclude_none=True),
},
)

return {
"token": token,
"token_type": "Bearer",
Expand Down
10 changes: 8 additions & 2 deletions backend/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -290,13 +290,19 @@ def create_config_file(file_path):


DEFAULT_USER_ROLE = os.getenv("DEFAULT_USER_ROLE", "pending")
USER_PERMISSIONS = {"chat": {"deletion": True}}

USER_PERMISSIONS_CHAT_DELETION = (
os.environ.get("USER_PERMISSIONS_CHAT_DELETION", "True").lower() == "true"
)

USER_PERMISSIONS = {"chat": {"deletion": USER_PERMISSIONS_CHAT_DELETION}}


MODEL_FILTER_ENABLED = os.environ.get("MODEL_FILTER_ENABLED", False)
MODEL_FILTER_ENABLED = os.environ.get("MODEL_FILTER_ENABLED", "False").lower() == "true"
MODEL_FILTER_LIST = os.environ.get("MODEL_FILTER_LIST", "")
MODEL_FILTER_LIST = [model.strip() for model in MODEL_FILTER_LIST.split(";")]

WEBHOOK_URL = os.environ.get("WEBHOOK_URL", "")

####################################
# WEBUI_VERSION
Expand Down
9 changes: 8 additions & 1 deletion backend/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,13 @@ class MESSAGES(str, Enum):
DEFAULT = lambda msg="": f"{msg if msg else ''}"


class WEBHOOK_MESSAGES(str, Enum):
DEFAULT = lambda msg="": f"{msg if msg else ''}"
USER_SIGNUP = lambda username="": (
f"New user signed up: {username}" if username else "New user signed up"
)


class ERROR_MESSAGES(str, Enum):
def __str__(self) -> str:
return super().__str__()
Expand Down Expand Up @@ -46,7 +53,7 @@ def __str__(self) -> str:

PANDOC_NOT_INSTALLED = "Pandoc is not installed on the server. Please contact your administrator for assistance."
INCORRECT_FORMAT = (
lambda err="": f"Invalid format. Please use the correct format{err if err else ''}"
lambda err="": f"Invalid format. Please use the correct format{err}"
)
RATE_LIMIT_EXCEEDED = "API rate limit exceeded"

Expand Down
2 changes: 1 addition & 1 deletion backend/data/config.json
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
{
"version": "0.0.1",
"version": 0,
"ui": {
"prompt_suggestions": [
{
Expand Down
28 changes: 27 additions & 1 deletion backend/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@
FRONTEND_BUILD_DIR,
MODEL_FILTER_ENABLED,
MODEL_FILTER_LIST,
WEBHOOK_URL,
)
from constants import ERROR_MESSAGES

Expand All @@ -58,6 +59,9 @@ async def get_response(self, path: str, scope):
app.state.MODEL_FILTER_ENABLED = MODEL_FILTER_ENABLED
app.state.MODEL_FILTER_LIST = MODEL_FILTER_LIST

app.state.WEBHOOK_URL = WEBHOOK_URL


origins = ["*"]


Expand Down Expand Up @@ -178,7 +182,7 @@ class ModelFilterConfigForm(BaseModel):


@app.post("/api/config/model/filter")
async def get_model_filter_config(
async def update_model_filter_config(
form_data: ModelFilterConfigForm, user=Depends(get_admin_user)
):

Expand All @@ -197,6 +201,28 @@ async def get_model_filter_config(
}


@app.get("/api/webhook")
async def get_webhook_url(user=Depends(get_admin_user)):
return {
"url": app.state.WEBHOOK_URL,
}


class UrlForm(BaseModel):
url: str


@app.post("/api/webhook")
async def update_webhook_url(form_data: UrlForm, user=Depends(get_admin_user)):
app.state.WEBHOOK_URL = form_data.url

webui_app.state.WEBHOOK_URL = app.state.WEBHOOK_URL

return {
"url": app.state.WEBHOOK_URL,
}


@app.get("/api/version")
async def get_app_config():

Expand Down
20 changes: 20 additions & 0 deletions backend/utils/webhook.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
import requests


def post_webhook(url: str, message: str, event_data: dict) -> bool:
try:
payload = {}

if "https://hooks.slack.com" in url:
payload["text"] = message
elif "https://discord.com/api/webhooks" in url:
payload["content"] = message
else:
payload = {**event_data}

r = requests.post(url, json=payload)
r.raise_for_status()
return True
except Exception as e:
print(e)
return False
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "open-webui",
"version": "0.1.113",
"version": "0.1.114",
"private": true,
"scripts": {
"dev": "vite dev --host",
Expand Down
57 changes: 57 additions & 0 deletions src/lib/apis/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -139,3 +139,60 @@ export const updateModelFilterConfig = async (

return res;
};

export const getWebhookUrl = async (token: string) => {
let error = null;

const res = await fetch(`${WEBUI_BASE_URL}/api/webhook`, {
method: 'GET',
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${token}`
}
})
.then(async (res) => {
if (!res.ok) throw await res.json();
return res.json();
})
.catch((err) => {
console.log(err);
error = err;
return null;
});

if (error) {
throw error;
}

return res.url;
};

export const updateWebhookUrl = async (token: string, url: string) => {
let error = null;

const res = await fetch(`${WEBUI_BASE_URL}/api/webhook`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
Authorization: `Bearer ${token}`
},
body: JSON.stringify({
url: url
})
})
.then(async (res) => {
if (!res.ok) throw await res.json();
return res.json();
})
.catch((err) => {
console.log(err);
error = err;
return null;
});

if (error) {
throw error;
}

return res.url;
};
Loading

0 comments on commit 2fa9495

Please sign in to comment.