| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | import base64 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | import logging | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | import mimetypes | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | import uuid | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  | import json | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | import aiohttp | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | from authlib.integrations.starlette_client import OAuth | 
					
						
							|  |  |  | from authlib.oidc.core import UserInfo | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | from fastapi import ( | 
					
						
							|  |  |  |     HTTPException, | 
					
						
							|  |  |  |     status, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | from starlette.responses import RedirectResponse | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-10 16:54:13 +08:00
										 |  |  | from open_webui.models.auths import Auths | 
					
						
							|  |  |  | from open_webui.models.users import Users | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  | from open_webui.models.groups import Groups, GroupModel, GroupUpdateForm, GroupForm | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | from open_webui.config import ( | 
					
						
							|  |  |  |     DEFAULT_USER_ROLE, | 
					
						
							|  |  |  |     ENABLE_OAUTH_SIGNUP, | 
					
						
							|  |  |  |     OAUTH_MERGE_ACCOUNTS_BY_EMAIL, | 
					
						
							|  |  |  |     OAUTH_PROVIDERS, | 
					
						
							|  |  |  |     ENABLE_OAUTH_ROLE_MANAGEMENT, | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |     ENABLE_OAUTH_GROUP_MANAGEMENT, | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |     ENABLE_OAUTH_GROUP_CREATION, | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  |     OAUTH_BLOCKED_GROUPS, | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  |     OAUTH_ROLES_CLAIM, | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |     OAUTH_GROUPS_CLAIM, | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  |     OAUTH_EMAIL_CLAIM, | 
					
						
							|  |  |  |     OAUTH_PICTURE_CLAIM, | 
					
						
							|  |  |  |     OAUTH_USERNAME_CLAIM, | 
					
						
							|  |  |  |     OAUTH_ALLOWED_ROLES, | 
					
						
							| 
									
										
										
										
											2024-10-21 09:38:06 +08:00
										 |  |  |     OAUTH_ADMIN_ROLES, | 
					
						
							| 
									
										
										
										
											2024-12-02 16:36:56 +08:00
										 |  |  |     OAUTH_ALLOWED_DOMAINS, | 
					
						
							| 
									
										
										
										
											2024-10-21 09:38:06 +08:00
										 |  |  |     WEBHOOK_URL, | 
					
						
							|  |  |  |     JWT_EXPIRES_IN, | 
					
						
							|  |  |  |     AppConfig, | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2025-01-08 16:38:00 +08:00
										 |  |  | from open_webui.constants import ERROR_MESSAGES, WEBHOOK_MESSAGES | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  | from open_webui.env import ( | 
					
						
							|  |  |  |     WEBUI_NAME, | 
					
						
							|  |  |  |     WEBUI_AUTH_COOKIE_SAME_SITE, | 
					
						
							|  |  |  |     WEBUI_AUTH_COOKIE_SECURE, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | from open_webui.utils.misc import parse_duration | 
					
						
							| 
									
										
										
										
											2024-12-09 08:01:56 +08:00
										 |  |  | from open_webui.utils.auth import get_password_hash, create_token | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | from open_webui.utils.webhook import post_webhook | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  | from open_webui.env import SRC_LOG_LEVELS, GLOBAL_LOG_LEVEL | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | logging.basicConfig(stream=sys.stdout, level=GLOBAL_LOG_LEVEL) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | log = logging.getLogger(__name__) | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  | log.setLevel(SRC_LOG_LEVELS["OAUTH"]) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | auth_manager_config = AppConfig() | 
					
						
							|  |  |  | auth_manager_config.DEFAULT_USER_ROLE = DEFAULT_USER_ROLE | 
					
						
							|  |  |  | auth_manager_config.ENABLE_OAUTH_SIGNUP = ENABLE_OAUTH_SIGNUP | 
					
						
							|  |  |  | auth_manager_config.OAUTH_MERGE_ACCOUNTS_BY_EMAIL = OAUTH_MERGE_ACCOUNTS_BY_EMAIL | 
					
						
							|  |  |  | auth_manager_config.ENABLE_OAUTH_ROLE_MANAGEMENT = ENABLE_OAUTH_ROLE_MANAGEMENT | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | auth_manager_config.ENABLE_OAUTH_GROUP_MANAGEMENT = ENABLE_OAUTH_GROUP_MANAGEMENT | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  | auth_manager_config.ENABLE_OAUTH_GROUP_CREATION = ENABLE_OAUTH_GROUP_CREATION | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  | auth_manager_config.OAUTH_BLOCKED_GROUPS = OAUTH_BLOCKED_GROUPS | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | auth_manager_config.OAUTH_ROLES_CLAIM = OAUTH_ROLES_CLAIM | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | auth_manager_config.OAUTH_GROUPS_CLAIM = OAUTH_GROUPS_CLAIM | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | auth_manager_config.OAUTH_EMAIL_CLAIM = OAUTH_EMAIL_CLAIM | 
					
						
							|  |  |  | auth_manager_config.OAUTH_PICTURE_CLAIM = OAUTH_PICTURE_CLAIM | 
					
						
							|  |  |  | auth_manager_config.OAUTH_USERNAME_CLAIM = OAUTH_USERNAME_CLAIM | 
					
						
							|  |  |  | auth_manager_config.OAUTH_ALLOWED_ROLES = OAUTH_ALLOWED_ROLES | 
					
						
							|  |  |  | auth_manager_config.OAUTH_ADMIN_ROLES = OAUTH_ADMIN_ROLES | 
					
						
							| 
									
										
										
										
											2024-12-02 16:36:56 +08:00
										 |  |  | auth_manager_config.OAUTH_ALLOWED_DOMAINS = OAUTH_ALLOWED_DOMAINS | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | auth_manager_config.WEBHOOK_URL = WEBHOOK_URL | 
					
						
							|  |  |  | auth_manager_config.JWT_EXPIRES_IN = JWT_EXPIRES_IN | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class OAuthManager: | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |     def __init__(self, app): | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         self.oauth = OAuth() | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |         self.app = app | 
					
						
							| 
									
										
										
										
											2025-01-17 12:56:03 +08:00
										 |  |  |         for _, provider_config in OAUTH_PROVIDERS.items(): | 
					
						
							|  |  |  |             provider_config["register"](self.oauth) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |     def get_client(self, provider_name): | 
					
						
							|  |  |  |         return self.oauth.create_client(provider_name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_user_role(self, user, user_data): | 
					
						
							|  |  |  |         if user and Users.get_num_users() == 1: | 
					
						
							|  |  |  |             # If the user is the only user, assign the role "admin" - actually repairs role for single user on login | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |             log.debug("Assigning the only user the admin role") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             return "admin" | 
					
						
							|  |  |  |         if not user and Users.get_num_users() == 0: | 
					
						
							|  |  |  |             # If there are no users, assign the role "admin", as the first user will be an admin | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |             log.debug("Assigning the first user the admin role") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             return "admin" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if auth_manager_config.ENABLE_OAUTH_ROLE_MANAGEMENT: | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |             log.debug("Running OAUTH Role management") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             oauth_claim = auth_manager_config.OAUTH_ROLES_CLAIM | 
					
						
							|  |  |  |             oauth_allowed_roles = auth_manager_config.OAUTH_ALLOWED_ROLES | 
					
						
							|  |  |  |             oauth_admin_roles = auth_manager_config.OAUTH_ADMIN_ROLES | 
					
						
							| 
									
										
										
										
											2025-03-10 17:42:59 +08:00
										 |  |  |             oauth_roles = [] | 
					
						
							| 
									
										
										
										
											2025-01-31 22:05:33 +08:00
										 |  |  |             # Default/fallback role if no matching roles are found | 
					
						
							|  |  |  |             role = auth_manager_config.DEFAULT_USER_ROLE | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Next block extracts the roles from the user data, accepting nested claims of any depth | 
					
						
							|  |  |  |             if oauth_claim and oauth_allowed_roles and oauth_admin_roles: | 
					
						
							|  |  |  |                 claim_data = user_data | 
					
						
							|  |  |  |                 nested_claims = oauth_claim.split(".") | 
					
						
							|  |  |  |                 for nested_claim in nested_claims: | 
					
						
							|  |  |  |                     claim_data = claim_data.get(nested_claim, {}) | 
					
						
							| 
									
										
										
										
											2025-03-10 17:42:59 +08:00
										 |  |  |                 oauth_roles = claim_data if isinstance(claim_data, list) else [] | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |             log.debug(f"Oauth Roles claim: {oauth_claim}") | 
					
						
							|  |  |  |             log.debug(f"User roles from oauth: {oauth_roles}") | 
					
						
							|  |  |  |             log.debug(f"Accepted user roles: {oauth_allowed_roles}") | 
					
						
							|  |  |  |             log.debug(f"Accepted admin roles: {oauth_admin_roles}") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             # If any roles are found, check if they match the allowed or admin roles | 
					
						
							|  |  |  |             if oauth_roles: | 
					
						
							|  |  |  |                 # If role management is enabled, and matching roles are provided, use the roles | 
					
						
							|  |  |  |                 for allowed_role in oauth_allowed_roles: | 
					
						
							|  |  |  |                     # If the user has any of the allowed roles, assign the role "user" | 
					
						
							|  |  |  |                     if allowed_role in oauth_roles: | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |                         log.debug("Assigned user the user role") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                         role = "user" | 
					
						
							|  |  |  |                         break | 
					
						
							|  |  |  |                 for admin_role in oauth_admin_roles: | 
					
						
							|  |  |  |                     # If the user has any of the admin roles, assign the role "admin" | 
					
						
							|  |  |  |                     if admin_role in oauth_roles: | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |                         log.debug("Assigned user the admin role") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                         role = "admin" | 
					
						
							|  |  |  |                         break | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             if not user: | 
					
						
							|  |  |  |                 # If role management is disabled, use the default role for new users | 
					
						
							|  |  |  |                 role = auth_manager_config.DEFAULT_USER_ROLE | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 # If role management is disabled, use the existing role for existing users | 
					
						
							|  |  |  |                 role = user.role | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return role | 
					
						
							| 
									
										
										
										
											2024-12-18 05:51:29 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |     def update_user_groups(self, user, user_data, default_permissions): | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |         log.debug("Running OAUTH Group management") | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |         oauth_claim = auth_manager_config.OAUTH_GROUPS_CLAIM | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             blocked_groups = json.loads(auth_manager_config.OAUTH_BLOCKED_GROUPS) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             log.exception(f"Error loading OAUTH_BLOCKED_GROUPS: {e}") | 
					
						
							|  |  |  |             blocked_groups = [] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-10 17:42:59 +08:00
										 |  |  |         user_oauth_groups = [] | 
					
						
							| 
									
										
										
										
											2025-02-20 00:47:52 +08:00
										 |  |  |         # Nested claim search for groups claim | 
					
						
							|  |  |  |         if oauth_claim: | 
					
						
							|  |  |  |             claim_data = user_data | 
					
						
							|  |  |  |             nested_claims = oauth_claim.split(".") | 
					
						
							|  |  |  |             for nested_claim in nested_claims: | 
					
						
							|  |  |  |                 claim_data = claim_data.get(nested_claim, {}) | 
					
						
							| 
									
										
										
										
											2025-02-22 05:25:22 +08:00
										 |  |  |             user_oauth_groups = claim_data if isinstance(claim_data, list) else [] | 
					
						
							| 
									
										
										
										
											2025-02-20 00:47:52 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |         user_current_groups: list[GroupModel] = Groups.get_groups_by_member_id(user.id) | 
					
						
							|  |  |  |         all_available_groups: list[GroupModel] = Groups.get_groups() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |         # Create groups if they don't exist and creation is enabled | 
					
						
							|  |  |  |         if auth_manager_config.ENABLE_OAUTH_GROUP_CREATION: | 
					
						
							|  |  |  |             log.debug("Checking for missing groups to create...") | 
					
						
							|  |  |  |             all_group_names = {g.name for g in all_available_groups} | 
					
						
							|  |  |  |             groups_created = False | 
					
						
							|  |  |  |             # Determine creator ID: Prefer admin, fallback to current user if no admin exists | 
					
						
							|  |  |  |             admin_user = Users.get_admin_user() | 
					
						
							|  |  |  |             creator_id = admin_user.id if admin_user else user.id | 
					
						
							|  |  |  |             log.debug(f"Using creator ID {creator_id} for potential group creation.") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for group_name in user_oauth_groups: | 
					
						
							|  |  |  |                 if group_name not in all_group_names: | 
					
						
							| 
									
										
										
										
											2025-04-23 15:05:15 +08:00
										 |  |  |                     log.info( | 
					
						
							|  |  |  |                         f"Group '{group_name}' not found via OAuth claim. Creating group..." | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |                     try: | 
					
						
							|  |  |  |                         new_group_form = GroupForm( | 
					
						
							|  |  |  |                             name=group_name, | 
					
						
							|  |  |  |                             description=f"Group '{group_name}' created automatically via OAuth.", | 
					
						
							| 
									
										
										
										
											2025-04-23 15:05:15 +08:00
										 |  |  |                             permissions=default_permissions,  # Use default permissions from function args | 
					
						
							|  |  |  |                             user_ids=[],  # Start with no users, user will be added later by subsequent logic | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |                         ) | 
					
						
							|  |  |  |                         # Use determined creator ID (admin or fallback to current user) | 
					
						
							| 
									
										
										
										
											2025-04-23 15:05:15 +08:00
										 |  |  |                         created_group = Groups.insert_new_group( | 
					
						
							|  |  |  |                             creator_id, new_group_form | 
					
						
							|  |  |  |                         ) | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |                         if created_group: | 
					
						
							| 
									
										
										
										
											2025-04-23 15:05:15 +08:00
										 |  |  |                             log.info( | 
					
						
							|  |  |  |                                 f"Successfully created group '{group_name}' with ID {created_group.id} using creator ID {creator_id}" | 
					
						
							|  |  |  |                             ) | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |                             groups_created = True | 
					
						
							|  |  |  |                             # Add to local set to prevent duplicate creation attempts in this run | 
					
						
							|  |  |  |                             all_group_names.add(group_name) | 
					
						
							|  |  |  |                         else: | 
					
						
							| 
									
										
										
										
											2025-04-23 15:05:15 +08:00
										 |  |  |                             log.error( | 
					
						
							|  |  |  |                                 f"Failed to create group '{group_name}' via OAuth." | 
					
						
							|  |  |  |                             ) | 
					
						
							| 
									
										
										
										
											2025-04-19 01:17:08 +08:00
										 |  |  |                     except Exception as e: | 
					
						
							|  |  |  |                         log.error(f"Error creating group '{group_name}' via OAuth: {e}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Refresh the list of all available groups if any were created | 
					
						
							|  |  |  |             if groups_created: | 
					
						
							|  |  |  |                 all_available_groups = Groups.get_groups() | 
					
						
							|  |  |  |                 log.debug("Refreshed list of all available groups after creation.") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  |         log.debug(f"Oauth Groups claim: {oauth_claim}") | 
					
						
							|  |  |  |         log.debug(f"User oauth groups: {user_oauth_groups}") | 
					
						
							|  |  |  |         log.debug(f"User's current groups: {[g.name for g in user_current_groups]}") | 
					
						
							| 
									
										
										
										
											2025-02-10 14:20:47 +08:00
										 |  |  |         log.debug( | 
					
						
							|  |  |  |             f"All groups available in OpenWebUI: {[g.name for g in all_available_groups]}" | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2025-02-08 03:53:25 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |         # Remove groups that user is no longer a part of | 
					
						
							|  |  |  |         for group_model in user_current_groups: | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  |             if ( | 
					
						
							|  |  |  |                 user_oauth_groups | 
					
						
							|  |  |  |                 and group_model.name not in user_oauth_groups | 
					
						
							|  |  |  |                 and group_model.name not in blocked_groups | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |                 # Remove group from user | 
					
						
							| 
									
										
										
										
											2025-02-10 14:20:47 +08:00
										 |  |  |                 log.debug( | 
					
						
							|  |  |  |                     f"Removing user from group {group_model.name} as it is no longer in their oauth groups" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 user_ids = group_model.user_ids | 
					
						
							|  |  |  |                 user_ids = [i for i in user_ids if i != user.id] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # In case a group is created, but perms are never assigned to the group by hitting "save" | 
					
						
							|  |  |  |                 group_permissions = group_model.permissions | 
					
						
							|  |  |  |                 if not group_permissions: | 
					
						
							|  |  |  |                     group_permissions = default_permissions | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 05:51:29 +08:00
										 |  |  |                 update_form = GroupUpdateForm( | 
					
						
							|  |  |  |                     name=group_model.name, | 
					
						
							|  |  |  |                     description=group_model.description, | 
					
						
							|  |  |  |                     permissions=group_permissions, | 
					
						
							|  |  |  |                     user_ids=user_ids, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 Groups.update_group_by_id( | 
					
						
							|  |  |  |                     id=group_model.id, form_data=update_form, overwrite=False | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Add user to new groups | 
					
						
							|  |  |  |         for group_model in all_available_groups: | 
					
						
							| 
									
										
										
										
											2025-03-10 17:42:59 +08:00
										 |  |  |             if ( | 
					
						
							|  |  |  |                 user_oauth_groups | 
					
						
							|  |  |  |                 and group_model.name in user_oauth_groups | 
					
						
							|  |  |  |                 and not any(gm.name == group_model.name for gm in user_current_groups) | 
					
						
							| 
									
										
										
										
											2025-05-02 18:47:02 +08:00
										 |  |  |                 and group_model.name not in blocked_groups | 
					
						
							| 
									
										
										
										
											2024-12-18 05:51:29 +08:00
										 |  |  |             ): | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  |                 # Add user to group | 
					
						
							| 
									
										
										
										
											2025-02-10 14:20:47 +08:00
										 |  |  |                 log.debug( | 
					
						
							|  |  |  |                     f"Adding user to group {group_model.name} as it was found in their oauth groups" | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 user_ids = group_model.user_ids | 
					
						
							|  |  |  |                 user_ids.append(user.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # In case a group is created, but perms are never assigned to the group by hitting "save" | 
					
						
							|  |  |  |                 group_permissions = group_model.permissions | 
					
						
							|  |  |  |                 if not group_permissions: | 
					
						
							|  |  |  |                     group_permissions = default_permissions | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-18 05:51:29 +08:00
										 |  |  |                 update_form = GroupUpdateForm( | 
					
						
							|  |  |  |                     name=group_model.name, | 
					
						
							|  |  |  |                     description=group_model.description, | 
					
						
							|  |  |  |                     permissions=group_permissions, | 
					
						
							|  |  |  |                     user_ids=user_ids, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 Groups.update_group_by_id( | 
					
						
							|  |  |  |                     id=group_model.id, form_data=update_form, overwrite=False | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |     async def handle_login(self, request, provider): | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         if provider not in OAUTH_PROVIDERS: | 
					
						
							|  |  |  |             raise HTTPException(404) | 
					
						
							|  |  |  |         # If the provider has a custom redirect URL, use that, otherwise automatically generate one | 
					
						
							|  |  |  |         redirect_uri = OAUTH_PROVIDERS[provider].get("redirect_uri") or request.url_for( | 
					
						
							|  |  |  |             "oauth_callback", provider=provider | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         client = self.get_client(provider) | 
					
						
							|  |  |  |         if client is None: | 
					
						
							|  |  |  |             raise HTTPException(404) | 
					
						
							|  |  |  |         return await client.authorize_redirect(request, redirect_uri) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |     async def handle_callback(self, request, provider, response): | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         if provider not in OAUTH_PROVIDERS: | 
					
						
							|  |  |  |             raise HTTPException(404) | 
					
						
							|  |  |  |         client = self.get_client(provider) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             token = await client.authorize_access_token(request) | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             log.warning(f"OAuth callback error: {e}") | 
					
						
							|  |  |  |             raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							| 
									
										
										
										
											2025-01-17 00:33:20 +08:00
										 |  |  |         user_data: UserInfo = token.get("userinfo") | 
					
						
							| 
									
										
										
										
											2025-03-04 14:50:04 +08:00
										 |  |  |         if not user_data or auth_manager_config.OAUTH_EMAIL_CLAIM not in user_data: | 
					
						
							| 
									
										
										
										
											2024-10-23 03:55:12 +08:00
										 |  |  |             user_data: UserInfo = await client.userinfo(token=token) | 
					
						
							|  |  |  |         if not user_data: | 
					
						
							|  |  |  |             log.warning(f"OAuth callback failed, user data is missing: {token}") | 
					
						
							|  |  |  |             raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-17 12:56:03 +08:00
										 |  |  |         sub = user_data.get(OAUTH_PROVIDERS[provider].get("sub_claim", "sub")) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         if not sub: | 
					
						
							|  |  |  |             log.warning(f"OAuth callback failed, sub is missing: {user_data}") | 
					
						
							|  |  |  |             raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							|  |  |  |         provider_sub = f"{provider}@{sub}" | 
					
						
							|  |  |  |         email_claim = auth_manager_config.OAUTH_EMAIL_CLAIM | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |         email = user_data.get(email_claim, "") | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         # We currently mandate that email addresses are provided | 
					
						
							|  |  |  |         if not email: | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |             # If the provider is GitHub,and public email is not provided, we can use the access token to fetch the user's email | 
					
						
							|  |  |  |             if provider == "github": | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     access_token = token.get("access_token") | 
					
						
							| 
									
										
										
										
											2025-02-20 17:01:29 +08:00
										 |  |  |                     headers = {"Authorization": f"Bearer {access_token}"} | 
					
						
							| 
									
										
										
										
											2025-04-28 20:47:34 +08:00
										 |  |  |                     async with aiohttp.ClientSession(trust_env=True) as session: | 
					
						
							| 
									
										
										
										
											2025-02-20 17:01:29 +08:00
										 |  |  |                         async with session.get( | 
					
						
							|  |  |  |                             "https://api.github.com/user/emails", headers=headers | 
					
						
							|  |  |  |                         ) as resp: | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |                             if resp.ok: | 
					
						
							|  |  |  |                                 emails = await resp.json() | 
					
						
							|  |  |  |                                 # use the primary email as the user's email | 
					
						
							| 
									
										
										
										
											2025-02-20 17:01:29 +08:00
										 |  |  |                                 primary_email = next( | 
					
						
							|  |  |  |                                     (e["email"] for e in emails if e.get("primary")), | 
					
						
							|  |  |  |                                     None, | 
					
						
							|  |  |  |                                 ) | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |                                 if primary_email: | 
					
						
							|  |  |  |                                     email = primary_email | 
					
						
							|  |  |  |                                 else: | 
					
						
							| 
									
										
										
										
											2025-02-20 17:01:29 +08:00
										 |  |  |                                     log.warning( | 
					
						
							|  |  |  |                                         "No primary email found in GitHub response" | 
					
						
							|  |  |  |                                     ) | 
					
						
							|  |  |  |                                     raise HTTPException( | 
					
						
							|  |  |  |                                         400, detail=ERROR_MESSAGES.INVALID_CRED | 
					
						
							|  |  |  |                                     ) | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |                             else: | 
					
						
							|  |  |  |                                 log.warning("Failed to fetch GitHub email") | 
					
						
							| 
									
										
										
										
											2025-02-20 17:01:29 +08:00
										 |  |  |                                 raise HTTPException( | 
					
						
							|  |  |  |                                     400, detail=ERROR_MESSAGES.INVALID_CRED | 
					
						
							|  |  |  |                                 ) | 
					
						
							| 
									
										
										
										
											2025-02-20 15:06:07 +08:00
										 |  |  |                 except Exception as e: | 
					
						
							|  |  |  |                     log.warning(f"Error fetching GitHub email: {e}") | 
					
						
							|  |  |  |                     raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 log.warning(f"OAuth callback failed, email is missing: {user_data}") | 
					
						
							|  |  |  |                 raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							|  |  |  |         email = email.lower() | 
					
						
							| 
									
										
										
										
											2024-12-10 16:54:13 +08:00
										 |  |  |         if ( | 
					
						
							|  |  |  |             "*" not in auth_manager_config.OAUTH_ALLOWED_DOMAINS | 
					
						
							|  |  |  |             and email.split("@")[-1] not in auth_manager_config.OAUTH_ALLOWED_DOMAINS | 
					
						
							|  |  |  |         ): | 
					
						
							|  |  |  |             log.warning( | 
					
						
							|  |  |  |                 f"OAuth callback failed, e-mail domain is not in the list of allowed domains: {user_data}" | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-12-02 16:36:56 +08:00
										 |  |  |             raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # Check if the user exists | 
					
						
							|  |  |  |         user = Users.get_user_by_oauth_sub(provider_sub) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not user: | 
					
						
							|  |  |  |             # If the user does not exist, check if merging is enabled | 
					
						
							| 
									
										
										
										
											2024-10-25 15:44:03 +08:00
										 |  |  |             if auth_manager_config.OAUTH_MERGE_ACCOUNTS_BY_EMAIL: | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                 # Check if the user exists by email | 
					
						
							|  |  |  |                 user = Users.get_user_by_email(email) | 
					
						
							|  |  |  |                 if user: | 
					
						
							|  |  |  |                     # Update the user with the new oauth sub | 
					
						
							|  |  |  |                     Users.update_user_oauth_sub_by_id(user.id, provider_sub) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if user: | 
					
						
							|  |  |  |             determined_role = self.get_user_role(user, user_data) | 
					
						
							|  |  |  |             if user.role != determined_role: | 
					
						
							|  |  |  |                 Users.update_user_role_by_id(user.id, determined_role) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if not user: | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |             user_count = Users.get_num_users() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |             # If the user does not exist, check if signups are enabled | 
					
						
							| 
									
										
										
										
											2024-10-25 15:44:03 +08:00
										 |  |  |             if auth_manager_config.ENABLE_OAUTH_SIGNUP: | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                 # Check if an existing user with the same email already exists | 
					
						
							| 
									
										
										
										
											2025-02-20 11:32:48 +08:00
										 |  |  |                 existing_user = Users.get_user_by_email(email) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                 if existing_user: | 
					
						
							|  |  |  |                     raise HTTPException(400, detail=ERROR_MESSAGES.EMAIL_TAKEN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-04-03 10:23:24 +08:00
										 |  |  |                 picture_claim = auth_manager_config.OAUTH_PICTURE_CLAIM | 
					
						
							|  |  |  |                 if picture_claim: | 
					
						
							| 
									
										
										
										
											2025-04-03 08:24:14 +08:00
										 |  |  |                     picture_url = user_data.get( | 
					
						
							|  |  |  |                         picture_claim, OAUTH_PROVIDERS[provider].get("picture_url", "") | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |                     if picture_url: | 
					
						
							|  |  |  |                         # Download the profile image into a base64 string | 
					
						
							|  |  |  |                         try: | 
					
						
							|  |  |  |                             access_token = token.get("access_token") | 
					
						
							|  |  |  |                             get_kwargs = {} | 
					
						
							|  |  |  |                             if access_token: | 
					
						
							|  |  |  |                                 get_kwargs["headers"] = { | 
					
						
							|  |  |  |                                     "Authorization": f"Bearer {access_token}", | 
					
						
							|  |  |  |                                 } | 
					
						
							| 
									
										
										
										
											2025-04-28 20:47:34 +08:00
										 |  |  |                             async with aiohttp.ClientSession(trust_env=True) as session: | 
					
						
							| 
									
										
										
										
											2025-04-03 10:23:24 +08:00
										 |  |  |                                 async with session.get( | 
					
						
							|  |  |  |                                     picture_url, **get_kwargs | 
					
						
							|  |  |  |                                 ) as resp: | 
					
						
							| 
									
										
										
										
											2025-04-03 08:24:14 +08:00
										 |  |  |                                     if resp.ok: | 
					
						
							|  |  |  |                                         picture = await resp.read() | 
					
						
							|  |  |  |                                         base64_encoded_picture = base64.b64encode( | 
					
						
							|  |  |  |                                             picture | 
					
						
							|  |  |  |                                         ).decode("utf-8") | 
					
						
							|  |  |  |                                         guessed_mime_type = mimetypes.guess_type( | 
					
						
							|  |  |  |                                             picture_url | 
					
						
							|  |  |  |                                         )[0] | 
					
						
							|  |  |  |                                         if guessed_mime_type is None: | 
					
						
							|  |  |  |                                             # assume JPG, browsers are tolerant enough of image formats | 
					
						
							|  |  |  |                                             guessed_mime_type = "image/jpeg" | 
					
						
							|  |  |  |                                         picture_url = f"data:{guessed_mime_type};base64,{base64_encoded_picture}" | 
					
						
							|  |  |  |                                     else: | 
					
						
							|  |  |  |                                         picture_url = "/user.png" | 
					
						
							|  |  |  |                         except Exception as e: | 
					
						
							|  |  |  |                             log.error( | 
					
						
							|  |  |  |                                 f"Error downloading profile image '{picture_url}': {e}" | 
					
						
							|  |  |  |                             ) | 
					
						
							|  |  |  |                             picture_url = "/user.png" | 
					
						
							|  |  |  |                     if not picture_url: | 
					
						
							| 
									
										
										
										
											2025-02-01 07:32:04 +08:00
										 |  |  |                         picture_url = "/user.png" | 
					
						
							| 
									
										
										
										
											2025-04-03 08:24:14 +08:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                     picture_url = "/user.png" | 
					
						
							| 
									
										
										
										
											2025-01-12 19:03:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                 username_claim = auth_manager_config.OAUTH_USERNAME_CLAIM | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-27 20:09:44 +08:00
										 |  |  |                 name = user_data.get(username_claim) | 
					
						
							| 
									
										
										
										
											2025-02-10 20:36:01 +08:00
										 |  |  |                 if not name: | 
					
						
							| 
									
										
										
										
											2025-02-10 20:49:55 +08:00
										 |  |  |                     log.warning("Username claim is missing, using email as name") | 
					
						
							| 
									
										
										
										
											2025-01-27 20:09:44 +08:00
										 |  |  |                     name = email | 
					
						
							| 
									
										
										
										
											2025-01-12 19:03:21 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                 role = self.get_user_role(None, user_data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 user = Auths.insert_new_auth( | 
					
						
							|  |  |  |                     email=email, | 
					
						
							|  |  |  |                     password=get_password_hash( | 
					
						
							|  |  |  |                         str(uuid.uuid4()) | 
					
						
							|  |  |  |                     ),  # Random password, not used | 
					
						
							| 
									
										
										
										
											2025-01-27 20:09:44 +08:00
										 |  |  |                     name=name, | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                     profile_image_url=picture_url, | 
					
						
							|  |  |  |                     role=role, | 
					
						
							|  |  |  |                     oauth_sub=provider_sub, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if auth_manager_config.WEBHOOK_URL: | 
					
						
							|  |  |  |                     post_webhook( | 
					
						
							| 
									
										
										
										
											2025-02-16 16:11:18 +08:00
										 |  |  |                         WEBUI_NAME, | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                         auth_manager_config.WEBHOOK_URL, | 
					
						
							| 
									
										
										
										
											2025-01-08 16:57:52 +08:00
										 |  |  |                         WEBHOOK_MESSAGES.USER_SIGNUP(user.name), | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                         { | 
					
						
							|  |  |  |                             "action": "signup", | 
					
						
							| 
									
										
										
										
											2025-01-08 16:57:52 +08:00
										 |  |  |                             "message": WEBHOOK_MESSAGES.USER_SIGNUP(user.name), | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |                             "user": user.model_dump_json(exclude_none=True), | 
					
						
							|  |  |  |                         }, | 
					
						
							|  |  |  |                     ) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise HTTPException( | 
					
						
							|  |  |  |                     status.HTTP_403_FORBIDDEN, detail=ERROR_MESSAGES.ACCESS_PROHIBITED | 
					
						
							|  |  |  |                 ) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         jwt_token = create_token( | 
					
						
							|  |  |  |             data={"id": user.id}, | 
					
						
							|  |  |  |             expires_delta=parse_duration(auth_manager_config.JWT_EXPIRES_IN), | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-07 02:23:42 +08:00
										 |  |  |         if auth_manager_config.ENABLE_OAUTH_GROUP_MANAGEMENT and user.role != "admin": | 
					
						
							| 
									
										
										
										
											2024-12-18 05:51:29 +08:00
										 |  |  |             self.update_user_groups( | 
					
						
							|  |  |  |                 user=user, | 
					
						
							|  |  |  |                 user_data=user_data, | 
					
						
							|  |  |  |                 default_permissions=request.app.state.config.USER_PERMISSIONS, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-12-18 03:38:07 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         # Set the cookie token | 
					
						
							|  |  |  |         response.set_cookie( | 
					
						
							|  |  |  |             key="token", | 
					
						
							|  |  |  |             value=jwt_token, | 
					
						
							|  |  |  |             httponly=True,  # Ensures the cookie is not accessible via JavaScript | 
					
						
							| 
									
										
										
										
											2025-01-23 22:16:50 +08:00
										 |  |  |             samesite=WEBUI_AUTH_COOKIE_SAME_SITE, | 
					
						
							|  |  |  |             secure=WEBUI_AUTH_COOKIE_SECURE, | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-10-16 15:42:47 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-12-07 22:21:05 +08:00
										 |  |  |         if ENABLE_OAUTH_SIGNUP.value: | 
					
						
							| 
									
										
										
										
											2024-12-10 08:25:56 +08:00
										 |  |  |             oauth_id_token = token.get("id_token") | 
					
						
							| 
									
										
										
										
											2024-12-07 20:49:12 +08:00
										 |  |  |             response.set_cookie( | 
					
						
							| 
									
										
										
										
											2024-12-10 08:25:56 +08:00
										 |  |  |                 key="oauth_id_token", | 
					
						
							|  |  |  |                 value=oauth_id_token, | 
					
						
							| 
									
										
										
										
											2024-12-07 20:49:12 +08:00
										 |  |  |                 httponly=True, | 
					
						
							| 
									
										
										
										
											2025-01-23 22:16:50 +08:00
										 |  |  |                 samesite=WEBUI_AUTH_COOKIE_SAME_SITE, | 
					
						
							|  |  |  |                 secure=WEBUI_AUTH_COOKIE_SECURE, | 
					
						
							| 
									
										
										
										
											2024-12-07 20:49:12 +08:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2024-10-16 22:32:57 +08:00
										 |  |  |         # Redirect back to the frontend with the JWT token | 
					
						
							|  |  |  |         redirect_url = f"{request.base_url}auth#token={jwt_token}" | 
					
						
							| 
									
										
										
										
											2024-12-07 20:49:12 +08:00
										 |  |  |         return RedirectResponse(url=redirect_url, headers=response.headers) |