|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106 |
- #
- # Copyright 2025 The InfiniFlow Authors. All Rights Reserved.
- #
- # Licensed under the Apache License, Version 2.0 (the "License");
- # you may not use this file except in compliance with the License.
- # You may obtain a copy of the License at
- #
- # http://www.apache.org/licenses/LICENSE-2.0
- #
- # Unless required by applicable law or agreed to in writing, software
- # distributed under the License is distributed on an "AS IS" BASIS,
- # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- # See the License for the specific language governing permissions and
- # limitations under the License.
- #
-
- import requests
- import urllib.parse
-
-
- class UserInfo:
- def __init__(self, email, username, nickname, avatar_url):
- self.email = email
- self.username = username
- self.nickname = nickname
- self.avatar_url = avatar_url
-
- def to_dict(self):
- return {key: value for key, value in self.__dict__.items()}
-
-
- class OAuthClient:
- def __init__(self, config):
- """
- Initialize the OAuthClient with the provider's configuration.
- """
- self.client_id = config["client_id"]
- self.client_secret = config["client_secret"]
- self.authorization_url = config["authorization_url"]
- self.token_url = config["token_url"]
- self.userinfo_url = config["userinfo_url"]
- self.redirect_uri = config["redirect_uri"]
- self.scope = config.get("scope", None)
-
- self.http_request_timeout = 7
-
-
- def get_authorization_url(self):
- """
- Generate the authorization URL for user login.
- """
- params = {
- "client_id": self.client_id,
- "redirect_uri": self.redirect_uri,
- "response_type": "code",
- }
- if self.scope:
- params["scope"] = self.scope
- authorization_url = f"{self.authorization_url}?{urllib.parse.urlencode(params)}"
- return authorization_url
-
-
- def exchange_code_for_token(self, code):
- """
- Exchange authorization code for access token.
- """
- try:
- payload = {
- "client_id": self.client_id,
- "client_secret": self.client_secret,
- "code": code,
- "redirect_uri": self.redirect_uri,
- "grant_type": "authorization_code"
- }
- response = requests.post(
- self.token_url,
- data=payload,
- headers={"Accept": "application/json"},
- timeout=self.http_request_timeout
- )
- response.raise_for_status()
- return response.json()
- except requests.exceptions.RequestException as e:
- raise ValueError(f"Failed to exchange authorization code for token: {e}")
-
-
- def fetch_user_info(self, access_token, **kwargs):
- """
- Fetch user information using access token.
- """
- try:
- headers = {"Authorization": f"Bearer {access_token}"}
- response = requests.get(self.userinfo_url, headers=headers, timeout=self.http_request_timeout)
- response.raise_for_status()
- user_info = response.json()
- return self.normalize_user_info(user_info)
- except requests.exceptions.RequestException as e:
- raise ValueError(f"Failed to fetch user info: {e}")
-
-
- def normalize_user_info(self, user_info):
- email = user_info.get("email")
- username = user_info.get("username", str(email).split("@")[0])
- nickname = user_info.get("nickname", username)
- avatar_url = user_info.get("picture", "")
- return UserInfo(email=email, username=username, nickname=nickname, avatar_url=avatar_url)
|