oauth.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. import base64
  2. import logging
  3. import mimetypes
  4. import uuid
  5. import aiohttp
  6. from authlib.integrations.starlette_client import OAuth
  7. from authlib.oidc.core import UserInfo
  8. from fastapi import (
  9. HTTPException,
  10. status,
  11. )
  12. from starlette.responses import RedirectResponse
  13. from open_webui.models.auths import Auths
  14. from open_webui.models.users import Users
  15. from open_webui.models.groups import Groups, GroupModel, GroupUpdateForm
  16. from open_webui.config import (
  17. DEFAULT_USER_ROLE,
  18. ENABLE_OAUTH_SIGNUP,
  19. OAUTH_MERGE_ACCOUNTS_BY_EMAIL,
  20. OAUTH_PROVIDERS,
  21. ENABLE_OAUTH_ROLE_MANAGEMENT,
  22. ENABLE_OAUTH_GROUP_MANAGEMENT,
  23. OAUTH_ROLES_CLAIM,
  24. OAUTH_GROUPS_CLAIM,
  25. OAUTH_EMAIL_CLAIM,
  26. OAUTH_PICTURE_CLAIM,
  27. OAUTH_USERNAME_CLAIM,
  28. OAUTH_ALLOWED_ROLES,
  29. OAUTH_ADMIN_ROLES,
  30. OAUTH_ALLOWED_DOMAINS,
  31. WEBHOOK_URL,
  32. JWT_EXPIRES_IN,
  33. AppConfig,
  34. )
  35. from open_webui.constants import ERROR_MESSAGES, WEBHOOK_MESSAGES
  36. from open_webui.env import WEBUI_SESSION_COOKIE_SAME_SITE, WEBUI_SESSION_COOKIE_SECURE
  37. from open_webui.utils.misc import parse_duration
  38. from open_webui.utils.auth import get_password_hash, create_token
  39. from open_webui.utils.webhook import post_webhook
  40. log = logging.getLogger(__name__)
  41. auth_manager_config = AppConfig()
  42. auth_manager_config.DEFAULT_USER_ROLE = DEFAULT_USER_ROLE
  43. auth_manager_config.ENABLE_OAUTH_SIGNUP = ENABLE_OAUTH_SIGNUP
  44. auth_manager_config.OAUTH_MERGE_ACCOUNTS_BY_EMAIL = OAUTH_MERGE_ACCOUNTS_BY_EMAIL
  45. auth_manager_config.ENABLE_OAUTH_ROLE_MANAGEMENT = ENABLE_OAUTH_ROLE_MANAGEMENT
  46. auth_manager_config.ENABLE_OAUTH_GROUP_MANAGEMENT = ENABLE_OAUTH_GROUP_MANAGEMENT
  47. auth_manager_config.OAUTH_ROLES_CLAIM = OAUTH_ROLES_CLAIM
  48. auth_manager_config.OAUTH_GROUPS_CLAIM = OAUTH_GROUPS_CLAIM
  49. auth_manager_config.OAUTH_EMAIL_CLAIM = OAUTH_EMAIL_CLAIM
  50. auth_manager_config.OAUTH_PICTURE_CLAIM = OAUTH_PICTURE_CLAIM
  51. auth_manager_config.OAUTH_USERNAME_CLAIM = OAUTH_USERNAME_CLAIM
  52. auth_manager_config.OAUTH_ALLOWED_ROLES = OAUTH_ALLOWED_ROLES
  53. auth_manager_config.OAUTH_ADMIN_ROLES = OAUTH_ADMIN_ROLES
  54. auth_manager_config.OAUTH_ALLOWED_DOMAINS = OAUTH_ALLOWED_DOMAINS
  55. auth_manager_config.WEBHOOK_URL = WEBHOOK_URL
  56. auth_manager_config.WEBHOOK_MESSAGES = WEBHOOK_MESSAGES
  57. auth_manager_config.JWT_EXPIRES_IN = JWT_EXPIRES_IN
  58. class OAuthManager:
  59. def __init__(self):
  60. self.oauth = OAuth()
  61. for provider_name, provider_config in OAUTH_PROVIDERS.items():
  62. self.oauth.register(
  63. name=provider_name,
  64. client_id=provider_config["client_id"],
  65. client_secret=provider_config["client_secret"],
  66. server_metadata_url=provider_config["server_metadata_url"],
  67. client_kwargs={
  68. "scope": provider_config["scope"],
  69. },
  70. redirect_uri=provider_config["redirect_uri"],
  71. )
  72. def get_client(self, provider_name):
  73. return self.oauth.create_client(provider_name)
  74. def get_user_role(self, user, user_data):
  75. if user and Users.get_num_users() == 1:
  76. # If the user is the only user, assign the role "admin" - actually repairs role for single user on login
  77. return "admin"
  78. if not user and Users.get_num_users() == 0:
  79. # If there are no users, assign the role "admin", as the first user will be an admin
  80. return "admin"
  81. if auth_manager_config.ENABLE_OAUTH_ROLE_MANAGEMENT:
  82. oauth_claim = auth_manager_config.OAUTH_ROLES_CLAIM
  83. oauth_allowed_roles = auth_manager_config.OAUTH_ALLOWED_ROLES
  84. oauth_admin_roles = auth_manager_config.OAUTH_ADMIN_ROLES
  85. oauth_roles = None
  86. role = "pending" # Default/fallback role if no matching roles are found
  87. # Next block extracts the roles from the user data, accepting nested claims of any depth
  88. if oauth_claim and oauth_allowed_roles and oauth_admin_roles:
  89. claim_data = user_data
  90. nested_claims = oauth_claim.split(".")
  91. for nested_claim in nested_claims:
  92. claim_data = claim_data.get(nested_claim, {})
  93. oauth_roles = claim_data if isinstance(claim_data, list) else None
  94. # If any roles are found, check if they match the allowed or admin roles
  95. if oauth_roles:
  96. # If role management is enabled, and matching roles are provided, use the roles
  97. for allowed_role in oauth_allowed_roles:
  98. # If the user has any of the allowed roles, assign the role "user"
  99. if allowed_role in oauth_roles:
  100. role = "user"
  101. break
  102. for admin_role in oauth_admin_roles:
  103. # If the user has any of the admin roles, assign the role "admin"
  104. if admin_role in oauth_roles:
  105. role = "admin"
  106. break
  107. else:
  108. if not user:
  109. # If role management is disabled, use the default role for new users
  110. role = auth_manager_config.DEFAULT_USER_ROLE
  111. else:
  112. # If role management is disabled, use the existing role for existing users
  113. role = user.role
  114. return role
  115. def update_user_groups(self, user, user_data, default_permissions):
  116. oauth_claim = auth_manager_config.OAUTH_GROUPS_CLAIM
  117. user_oauth_groups: list[str] = user_data.get(oauth_claim, list())
  118. user_current_groups: list[GroupModel] = Groups.get_groups_by_member_id(user.id)
  119. all_available_groups: list[GroupModel] = Groups.get_groups()
  120. # Remove groups that user is no longer a part of
  121. for group_model in user_current_groups:
  122. if group_model.name not in user_oauth_groups:
  123. # Remove group from user
  124. user_ids = group_model.user_ids
  125. user_ids = [i for i in user_ids if i != user.id]
  126. # In case a group is created, but perms are never assigned to the group by hitting "save"
  127. group_permissions = group_model.permissions
  128. if not group_permissions:
  129. group_permissions = default_permissions
  130. update_form = GroupUpdateForm(
  131. name=group_model.name,
  132. description=group_model.description,
  133. permissions=group_permissions,
  134. user_ids=user_ids,
  135. )
  136. Groups.update_group_by_id(
  137. id=group_model.id, form_data=update_form, overwrite=False
  138. )
  139. # Add user to new groups
  140. for group_model in all_available_groups:
  141. if group_model.name in user_oauth_groups and not any(
  142. gm.name == group_model.name for gm in user_current_groups
  143. ):
  144. # Add user to group
  145. user_ids = group_model.user_ids
  146. user_ids.append(user.id)
  147. # In case a group is created, but perms are never assigned to the group by hitting "save"
  148. group_permissions = group_model.permissions
  149. if not group_permissions:
  150. group_permissions = default_permissions
  151. update_form = GroupUpdateForm(
  152. name=group_model.name,
  153. description=group_model.description,
  154. permissions=group_permissions,
  155. user_ids=user_ids,
  156. )
  157. Groups.update_group_by_id(
  158. id=group_model.id, form_data=update_form, overwrite=False
  159. )
  160. async def handle_login(self, provider, request):
  161. if provider not in OAUTH_PROVIDERS:
  162. raise HTTPException(404)
  163. # If the provider has a custom redirect URL, use that, otherwise automatically generate one
  164. redirect_uri = OAUTH_PROVIDERS[provider].get("redirect_uri") or request.url_for(
  165. "oauth_callback", provider=provider
  166. )
  167. client = self.get_client(provider)
  168. if client is None:
  169. raise HTTPException(404)
  170. return await client.authorize_redirect(request, redirect_uri)
  171. async def handle_callback(self, provider, request, response):
  172. if provider not in OAUTH_PROVIDERS:
  173. raise HTTPException(404)
  174. client = self.get_client(provider)
  175. try:
  176. token = await client.authorize_access_token(request)
  177. except Exception as e:
  178. log.warning(f"OAuth callback error: {e}")
  179. raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED)
  180. user_data: UserInfo = token["userinfo"]
  181. if not user_data:
  182. user_data: UserInfo = await client.userinfo(token=token)
  183. if not user_data:
  184. log.warning(f"OAuth callback failed, user data is missing: {token}")
  185. raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED)
  186. sub = user_data.get("sub")
  187. if not sub:
  188. log.warning(f"OAuth callback failed, sub is missing: {user_data}")
  189. raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED)
  190. provider_sub = f"{provider}@{sub}"
  191. email_claim = auth_manager_config.OAUTH_EMAIL_CLAIM
  192. email = user_data.get(email_claim, "").lower()
  193. # We currently mandate that email addresses are provided
  194. if not email:
  195. log.warning(f"OAuth callback failed, email is missing: {user_data}")
  196. raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED)
  197. if (
  198. "*" not in auth_manager_config.OAUTH_ALLOWED_DOMAINS
  199. and email.split("@")[-1] not in auth_manager_config.OAUTH_ALLOWED_DOMAINS
  200. ):
  201. log.warning(
  202. f"OAuth callback failed, e-mail domain is not in the list of allowed domains: {user_data}"
  203. )
  204. raise HTTPException(400, detail=ERROR_MESSAGES.INVALID_CRED)
  205. # Check if the user exists
  206. user = Users.get_user_by_oauth_sub(provider_sub)
  207. if not user:
  208. # If the user does not exist, check if merging is enabled
  209. if auth_manager_config.OAUTH_MERGE_ACCOUNTS_BY_EMAIL:
  210. # Check if the user exists by email
  211. user = Users.get_user_by_email(email)
  212. if user:
  213. # Update the user with the new oauth sub
  214. Users.update_user_oauth_sub_by_id(user.id, provider_sub)
  215. if user:
  216. determined_role = self.get_user_role(user, user_data)
  217. if user.role != determined_role:
  218. Users.update_user_role_by_id(user.id, determined_role)
  219. if not user:
  220. # If the user does not exist, check if signups are enabled
  221. if auth_manager_config.ENABLE_OAUTH_SIGNUP:
  222. # Check if an existing user with the same email already exists
  223. existing_user = Users.get_user_by_email(
  224. user_data.get("email", "").lower()
  225. )
  226. if existing_user:
  227. raise HTTPException(400, detail=ERROR_MESSAGES.EMAIL_TAKEN)
  228. picture_claim = auth_manager_config.OAUTH_PICTURE_CLAIM
  229. picture_url = user_data.get(picture_claim, "")
  230. if picture_url:
  231. # Download the profile image into a base64 string
  232. try:
  233. async with aiohttp.ClientSession() as session:
  234. async with session.get(picture_url) as resp:
  235. picture = await resp.read()
  236. base64_encoded_picture = base64.b64encode(
  237. picture
  238. ).decode("utf-8")
  239. guessed_mime_type = mimetypes.guess_type(picture_url)[0]
  240. if guessed_mime_type is None:
  241. # assume JPG, browsers are tolerant enough of image formats
  242. guessed_mime_type = "image/jpeg"
  243. picture_url = f"data:{guessed_mime_type};base64,{base64_encoded_picture}"
  244. except Exception as e:
  245. log.error(
  246. f"Error downloading profile image '{picture_url}': {e}"
  247. )
  248. picture_url = ""
  249. if not picture_url:
  250. picture_url = "/user.png"
  251. username_claim = auth_manager_config.OAUTH_USERNAME_CLAIM
  252. role = self.get_user_role(None, user_data)
  253. user = Auths.insert_new_auth(
  254. email=email,
  255. password=get_password_hash(
  256. str(uuid.uuid4())
  257. ), # Random password, not used
  258. name=user_data.get(username_claim, "User"),
  259. profile_image_url=picture_url,
  260. role=role,
  261. oauth_sub=provider_sub,
  262. )
  263. if auth_manager_config.WEBHOOK_URL:
  264. post_webhook(
  265. auth_manager_config.WEBHOOK_URL,
  266. auth_manager_config.WEBHOOK_MESSAGES.USER_SIGNUP(user.name),
  267. {
  268. "action": "signup",
  269. "message": auth_manager_config.WEBHOOK_MESSAGES.USER_SIGNUP(
  270. user.name
  271. ),
  272. "user": user.model_dump_json(exclude_none=True),
  273. },
  274. )
  275. else:
  276. raise HTTPException(
  277. status.HTTP_403_FORBIDDEN, detail=ERROR_MESSAGES.ACCESS_PROHIBITED
  278. )
  279. jwt_token = create_token(
  280. data={"id": user.id},
  281. expires_delta=parse_duration(auth_manager_config.JWT_EXPIRES_IN),
  282. )
  283. if auth_manager_config.ENABLE_OAUTH_GROUP_MANAGEMENT and user.role != "admin":
  284. self.update_user_groups(
  285. user=user,
  286. user_data=user_data,
  287. default_permissions=request.app.state.config.USER_PERMISSIONS,
  288. )
  289. # Set the cookie token
  290. response.set_cookie(
  291. key="token",
  292. value=jwt_token,
  293. httponly=True, # Ensures the cookie is not accessible via JavaScript
  294. samesite=WEBUI_SESSION_COOKIE_SAME_SITE,
  295. secure=WEBUI_SESSION_COOKIE_SECURE,
  296. )
  297. if ENABLE_OAUTH_SIGNUP.value:
  298. oauth_id_token = token.get("id_token")
  299. response.set_cookie(
  300. key="oauth_id_token",
  301. value=oauth_id_token,
  302. httponly=True,
  303. samesite=WEBUI_SESSION_COOKIE_SAME_SITE,
  304. secure=WEBUI_SESSION_COOKIE_SECURE,
  305. )
  306. # Redirect back to the frontend with the JWT token
  307. redirect_url = f"{request.base_url}auth#token={jwt_token}"
  308. return RedirectResponse(url=redirect_url, headers=response.headers)
  309. oauth_manager = OAuthManager()