This commit is contained in:
Iliyan Angelov
2025-09-14 23:24:25 +03:00
commit c67067a2a4
71311 changed files with 6800714 additions and 0 deletions

View File

@@ -0,0 +1,67 @@
from allauth.socialaccount import app_settings
from allauth.socialaccount.providers.base import (
ProviderAccount,
ProviderException,
)
from allauth.socialaccount.providers.oauth2.provider import OAuth2Provider
class QuickBooksAccount(ProviderAccount):
def to_str(self):
dflt = super(QuickBooksAccount, self).to_str()
name = self.account.extra_data.get("name", dflt)
first_name = self.account.extra_data.get("givenName", None)
last_name = self.account.extra_data.get("familyName", None)
if first_name and last_name:
name = first_name + " " + last_name
return name
class QuickBooksOAuth2Provider(OAuth2Provider):
id = "quickbooks"
# Name is displayed to ordinary users -- don't include protocol
name = "QuickBooks"
account_class = QuickBooksAccount
def extract_uid(self, data):
if "sub" not in data:
raise ProviderException("QBO error", data)
return str(data["sub"])
def get_profile_fields(self):
default_fields = [
"address",
"sub",
"phoneNumber",
"givenName",
"familyName",
"email",
"emailVerified",
]
fields = self.get_settings().get("PROFILE_FIELDS", default_fields)
return fields
def get_default_scope(self):
scope = [
"openid",
"com.intuit.quickbooks.accounting",
"profile",
"phone",
]
if app_settings.QUERY_EMAIL:
scope.append("email")
return scope
def extract_common_fields(self, data):
return dict(
email=data.get("email"),
address=data.get("address"),
sub=data.get("sub"),
givenName=data.get("givenName"),
familynName=data.get("familyName"),
emailVerified=data.get("emailVerified"),
phoneNumber=data.get("phoneNumber"),
)
provider_classes = [QuickBooksOAuth2Provider]

View File

@@ -0,0 +1,22 @@
from allauth.socialaccount.tests import OAuth2TestsMixin
from allauth.tests import MockedResponse, TestCase
from .provider import QuickBooksOAuth2Provider
class QuickBooksOAuth2Tests(OAuth2TestsMixin, TestCase):
provider_id = QuickBooksOAuth2Provider.id
def get_mocked_response(self):
return MockedResponse(
200,
"""
{ "sub": "d8752092-0f2b-4b6e-86ef-6b72f2457a00",
"emailVerified": true,
"familyName": "Mckeeman",
"phoneNumber": "+1 4156694355",
"givenName": "Darren",
"phoneNumberVerified": true,
"email": "darren@blocklight.io"}
""",
)

View File

@@ -0,0 +1,6 @@
from allauth.socialaccount.providers.oauth2.urls import default_urlpatterns
from .provider import QuickBooksOAuth2Provider
urlpatterns = default_urlpatterns(QuickBooksOAuth2Provider)

View File

@@ -0,0 +1,43 @@
import requests
from allauth.socialaccount.providers.oauth2.views import (
OAuth2Adapter,
OAuth2CallbackView,
OAuth2LoginView,
)
from .provider import QuickBooksOAuth2Provider
class QuickBooksOAuth2Adapter(OAuth2Adapter):
provider_id = QuickBooksOAuth2Provider.id
access_token_url = "https://oauth.platform.intuit.com/oauth2/v1/tokens/bearer"
authorize_url = "https://appcenter.intuit.com/connect/oauth2"
profile_test = "https://sandbox-accounts.platform.intuit.com/v1/openid_connect/userinfo" # NOQA
profile_url = "https://accounts.platform.intuit.com/v1/openid_connect/userinfo"
profile_url_method = "GET"
access_token_method = "POST"
def complete_login(self, request, app, token, **kwargs):
realm_id = request.GET.get("realmId")
extra_data = self.get_user_info(token)
if realm_id:
extra_data["realmId"] = realm_id
return self.get_provider().sociallogin_from_response(request, extra_data)
def get_user_info(self, token):
auth_header = "Bearer " + token.token
headers = {
"Accept": "application/json",
"Authorization": auth_header,
"accept": "application/json",
}
is_sandbox = self.get_provider().get_settings().get("SANDBOX", False)
url = self.profile_test if is_sandbox else self.profile_url
resp = requests.get(url, headers=headers)
resp.raise_for_status()
return resp.json()
oauth2_login = OAuth2LoginView.adapter_view(QuickBooksOAuth2Adapter)
oauth2_callback = OAuth2CallbackView.adapter_view(QuickBooksOAuth2Adapter)