mirror of
https://github.com/BerriAI/litellm.git
synced 2025-04-26 11:14:04 +00:00
[UI Polish] - Polish login screen (#9778)
* fix admin ui utils login screen * ui - add layer of polish on login screen * ui fix design of login page * ui fix color scheme on login page
This commit is contained in:
parent
f2d6e4bee9
commit
4c12083b1b
4 changed files with 219 additions and 73 deletions
|
@ -1,6 +1,3 @@
|
||||||
import os
|
|
||||||
|
|
||||||
|
|
||||||
def show_missing_vars_in_env():
|
def show_missing_vars_in_env():
|
||||||
from fastapi.responses import HTMLResponse
|
from fastapi.responses import HTMLResponse
|
||||||
|
|
||||||
|
@ -26,74 +23,6 @@ def show_missing_vars_in_env():
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
# LiteLLM Admin UI - Non SSO Login
|
|
||||||
url_to_redirect_to = os.getenv("PROXY_BASE_URL", "")
|
|
||||||
url_to_redirect_to += "/login"
|
|
||||||
html_form = f"""
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<title>LiteLLM Login</title>
|
|
||||||
<style>
|
|
||||||
body {{
|
|
||||||
font-family: Arial, sans-serif;
|
|
||||||
background-color: #f4f4f4;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
height: 100vh;
|
|
||||||
}}
|
|
||||||
|
|
||||||
form {{
|
|
||||||
background-color: #fff;
|
|
||||||
padding: 20px;
|
|
||||||
border-radius: 8px;
|
|
||||||
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
|
|
||||||
}}
|
|
||||||
|
|
||||||
label {{
|
|
||||||
display: block;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}}
|
|
||||||
|
|
||||||
input {{
|
|
||||||
width: 100%;
|
|
||||||
padding: 8px;
|
|
||||||
margin-bottom: 16px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
border: 1px solid #ccc;
|
|
||||||
border-radius: 4px;
|
|
||||||
}}
|
|
||||||
|
|
||||||
input[type="submit"] {{
|
|
||||||
background-color: #4caf50;
|
|
||||||
color: #fff;
|
|
||||||
cursor: pointer;
|
|
||||||
}}
|
|
||||||
|
|
||||||
input[type="submit"]:hover {{
|
|
||||||
background-color: #45a049;
|
|
||||||
}}
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<form action="{url_to_redirect_to}" method="post">
|
|
||||||
<h2>LiteLLM Login</h2>
|
|
||||||
|
|
||||||
<p>By default Username is "admin" and Password is your set LiteLLM Proxy `MASTER_KEY`</p>
|
|
||||||
<p>If you need to set UI credentials / SSO docs here: <a href="https://docs.litellm.ai/docs/proxy/ui" target="_blank">https://docs.litellm.ai/docs/proxy/ui</a></p>
|
|
||||||
<br>
|
|
||||||
<label for="username">Username:</label>
|
|
||||||
<input type="text" id="username" name="username" required>
|
|
||||||
<label for="password">Password:</label>
|
|
||||||
<input type="password" id="password" name="password" required>
|
|
||||||
<input type="submit" value="Submit">
|
|
||||||
</form>
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def missing_keys_form(missing_key_names: str):
|
def missing_keys_form(missing_key_names: str):
|
||||||
missing_keys_html_form = """
|
missing_keys_html_form = """
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
|
|
217
litellm/proxy/common_utils/html_forms/ui_login.py
Normal file
217
litellm/proxy/common_utils/html_forms/ui_login.py
Normal file
|
@ -0,0 +1,217 @@
|
||||||
|
import os
|
||||||
|
|
||||||
|
url_to_redirect_to = os.getenv("PROXY_BASE_URL", "")
|
||||||
|
url_to_redirect_to += "/login"
|
||||||
|
html_form = f"""
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>LiteLLM Login</title>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<style>
|
||||||
|
body {{
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, sans-serif;
|
||||||
|
background-color: #f8fafc;
|
||||||
|
margin: 0;
|
||||||
|
padding: 20px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
min-height: 100vh;
|
||||||
|
color: #333;
|
||||||
|
}}
|
||||||
|
|
||||||
|
form {{
|
||||||
|
background-color: #fff;
|
||||||
|
padding: 40px;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
||||||
|
width: 450px;
|
||||||
|
max-width: 100%;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.logo-container {{
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.logo {{
|
||||||
|
font-size: 24px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #1e293b;
|
||||||
|
}}
|
||||||
|
|
||||||
|
h2 {{
|
||||||
|
margin: 0 0 10px;
|
||||||
|
color: #1e293b;
|
||||||
|
font-size: 28px;
|
||||||
|
font-weight: 600;
|
||||||
|
text-align: center;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.subtitle {{
|
||||||
|
color: #64748b;
|
||||||
|
margin: 0 0 20px;
|
||||||
|
font-size: 16px;
|
||||||
|
text-align: center;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.info-box {{
|
||||||
|
background-color: #f1f5f9;
|
||||||
|
border-radius: 6px;
|
||||||
|
padding: 20px;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
border-left: 4px solid #2563eb;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.info-header {{
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 12px;
|
||||||
|
color: #1e40af;
|
||||||
|
font-weight: 600;
|
||||||
|
font-size: 16px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.info-header svg {{
|
||||||
|
margin-right: 8px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.info-box p {{
|
||||||
|
color: #475569;
|
||||||
|
margin: 8px 0;
|
||||||
|
line-height: 1.5;
|
||||||
|
font-size: 14px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
label {{
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #334155;
|
||||||
|
font-size: 14px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.required {{
|
||||||
|
color: #dc2626;
|
||||||
|
margin-left: 2px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
input[type="text"],
|
||||||
|
input[type="password"] {{
|
||||||
|
width: 100%;
|
||||||
|
padding: 10px 14px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
box-sizing: border-box;
|
||||||
|
border: 1px solid #e2e8f0;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 15px;
|
||||||
|
color: #1e293b;
|
||||||
|
background-color: #fff;
|
||||||
|
transition: border-color 0.2s, box-shadow 0.2s;
|
||||||
|
}}
|
||||||
|
|
||||||
|
input[type="text"]:focus,
|
||||||
|
input[type="password"]:focus {{
|
||||||
|
outline: none;
|
||||||
|
border-color: #3b82f6;
|
||||||
|
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.2);
|
||||||
|
}}
|
||||||
|
|
||||||
|
.toggle-password {{
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-top: -15px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.toggle-password input {{
|
||||||
|
margin-right: 6px;
|
||||||
|
}}
|
||||||
|
|
||||||
|
input[type="submit"] {{
|
||||||
|
background-color: #6466E9;
|
||||||
|
color: #fff;
|
||||||
|
cursor: pointer;
|
||||||
|
font-weight: 500;
|
||||||
|
border: none;
|
||||||
|
padding: 10px 16px;
|
||||||
|
transition: background-color 0.2s;
|
||||||
|
border-radius: 6px;
|
||||||
|
margin-top: 10px;
|
||||||
|
font-size: 14px;
|
||||||
|
width: 100%;
|
||||||
|
}}
|
||||||
|
|
||||||
|
input[type="submit"]:hover {{
|
||||||
|
background-color: #4138C2;
|
||||||
|
}}
|
||||||
|
|
||||||
|
a {{
|
||||||
|
color: #3b82f6;
|
||||||
|
text-decoration: none;
|
||||||
|
}}
|
||||||
|
|
||||||
|
a:hover {{
|
||||||
|
text-decoration: underline;
|
||||||
|
}}
|
||||||
|
|
||||||
|
code {{
|
||||||
|
background-color: #f1f5f9;
|
||||||
|
padding: 2px 4px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #334155;
|
||||||
|
}}
|
||||||
|
|
||||||
|
.help-text {{
|
||||||
|
color: #64748b;
|
||||||
|
font-size: 14px;
|
||||||
|
margin-top: -12px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<form action="{url_to_redirect_to}" method="post">
|
||||||
|
<div class="logo-container">
|
||||||
|
<div class="logo">
|
||||||
|
🚅 LiteLLM
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<h2>Login</h2>
|
||||||
|
<p class="subtitle">Access your LiteLLM Admin UI.</p>
|
||||||
|
<div class="info-box">
|
||||||
|
<div class="info-header">
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
|
||||||
|
<circle cx="12" cy="12" r="10"></circle>
|
||||||
|
<line x1="12" y1="16" x2="12" y2="12"></line>
|
||||||
|
<line x1="12" y1="8" x2="12.01" y2="8"></line>
|
||||||
|
</svg>
|
||||||
|
Default Credentials
|
||||||
|
</div>
|
||||||
|
<p>By default, Username is <code>admin</code> and Password is your set LiteLLM Proxy <code>MASTER_KEY</code>.</p>
|
||||||
|
<p>Need to set UI credentials or SSO? <a href="https://docs.litellm.ai/docs/proxy/ui" target="_blank">Check the documentation</a>.</p>
|
||||||
|
</div>
|
||||||
|
<label for="username">Username<span class="required">*</span></label>
|
||||||
|
<input type="text" id="username" name="username" required placeholder="Enter your username" autocomplete="username">
|
||||||
|
|
||||||
|
<label for="password">Password<span class="required">*</span></label>
|
||||||
|
<input type="password" id="password" name="password" required placeholder="Enter your password" autocomplete="current-password">
|
||||||
|
<div class="toggle-password">
|
||||||
|
<input type="checkbox" id="show-password" onclick="togglePasswordVisibility()">
|
||||||
|
<label for="show-password">Show password</label>
|
||||||
|
</div>
|
||||||
|
<input type="submit" value="Login">
|
||||||
|
</form>
|
||||||
|
<script>
|
||||||
|
function togglePasswordVisibility() {{
|
||||||
|
var passwordField = document.getElementById("password");
|
||||||
|
passwordField.type = passwordField.type === "password" ? "text" : "password";
|
||||||
|
}}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
"""
|
|
@ -34,9 +34,9 @@ from litellm.proxy.auth.handle_jwt import JWTHandler
|
||||||
from litellm.proxy.auth.user_api_key_auth import user_api_key_auth
|
from litellm.proxy.auth.user_api_key_auth import user_api_key_auth
|
||||||
from litellm.proxy.common_utils.admin_ui_utils import (
|
from litellm.proxy.common_utils.admin_ui_utils import (
|
||||||
admin_ui_disabled,
|
admin_ui_disabled,
|
||||||
html_form,
|
|
||||||
show_missing_vars_in_env,
|
show_missing_vars_in_env,
|
||||||
)
|
)
|
||||||
|
from litellm.proxy.common_utils.html_forms.ui_login import html_form
|
||||||
from litellm.proxy.management_endpoints.internal_user_endpoints import new_user
|
from litellm.proxy.management_endpoints.internal_user_endpoints import new_user
|
||||||
from litellm.proxy.management_endpoints.sso_helper_utils import (
|
from litellm.proxy.management_endpoints.sso_helper_utils import (
|
||||||
check_is_admin_only_access,
|
check_is_admin_only_access,
|
||||||
|
|
|
@ -167,7 +167,6 @@ from litellm.proxy.batches_endpoints.endpoints import router as batches_router
|
||||||
## Import All Misc routes here ##
|
## Import All Misc routes here ##
|
||||||
from litellm.proxy.caching_routes import router as caching_router
|
from litellm.proxy.caching_routes import router as caching_router
|
||||||
from litellm.proxy.common_request_processing import ProxyBaseLLMRequestProcessing
|
from litellm.proxy.common_request_processing import ProxyBaseLLMRequestProcessing
|
||||||
from litellm.proxy.common_utils.admin_ui_utils import html_form
|
|
||||||
from litellm.proxy.common_utils.callback_utils import initialize_callbacks_on_proxy
|
from litellm.proxy.common_utils.callback_utils import initialize_callbacks_on_proxy
|
||||||
from litellm.proxy.common_utils.debug_utils import init_verbose_loggers
|
from litellm.proxy.common_utils.debug_utils import init_verbose_loggers
|
||||||
from litellm.proxy.common_utils.debug_utils import router as debugging_endpoints_router
|
from litellm.proxy.common_utils.debug_utils import router as debugging_endpoints_router
|
||||||
|
@ -175,6 +174,7 @@ from litellm.proxy.common_utils.encrypt_decrypt_utils import (
|
||||||
decrypt_value_helper,
|
decrypt_value_helper,
|
||||||
encrypt_value_helper,
|
encrypt_value_helper,
|
||||||
)
|
)
|
||||||
|
from litellm.proxy.common_utils.html_forms.ui_login import html_form
|
||||||
from litellm.proxy.common_utils.http_parsing_utils import (
|
from litellm.proxy.common_utils.http_parsing_utils import (
|
||||||
_read_request_body,
|
_read_request_body,
|
||||||
check_file_size_under_limit,
|
check_file_size_under_limit,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue