2018-11-09 06:54:38 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
|
|
|
|
from django.db import transaction
|
|
|
|
from django.contrib.auth import get_user_model
|
|
|
|
from keycloak.realm import KeycloakRealm
|
|
|
|
from keycloak.keycloak_openid import KeycloakOpenID
|
2019-02-28 03:58:48 +00:00
|
|
|
|
2019-12-09 08:12:48 +00:00
|
|
|
from .signals import post_create_or_update_openid_user
|
2019-07-01 03:04:15 +00:00
|
|
|
from .decorator import ssl_verification
|
2018-11-09 06:54:38 +00:00
|
|
|
|
|
|
|
OIDT_ACCESS_TOKEN = 'oidt_access_token'
|
|
|
|
|
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
class Nonce(object):
|
|
|
|
"""
|
|
|
|
The openid-login is stored in cache as a temporary object, recording the
|
|
|
|
user's redirect_uri and next_pat
|
|
|
|
"""
|
|
|
|
def __init__(self, redirect_uri, next_path):
|
|
|
|
import uuid
|
|
|
|
self.state = uuid.uuid4()
|
|
|
|
self.redirect_uri = redirect_uri
|
|
|
|
self.next_path = next_path
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
|
|
|
|
class OpenIDTokenProfile(object):
|
2018-11-09 06:54:38 +00:00
|
|
|
def __init__(self, user, access_token, refresh_token):
|
|
|
|
"""
|
|
|
|
:param user: User object
|
|
|
|
:param access_token:
|
|
|
|
:param refresh_token:
|
|
|
|
"""
|
|
|
|
self.user = user
|
|
|
|
self.access_token = access_token
|
|
|
|
self.refresh_token = refresh_token
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return "{}'s OpenID token profile".format(self.user.username)
|
|
|
|
|
|
|
|
|
|
|
|
class Client(object):
|
|
|
|
def __init__(self, server_url, realm_name, client_id, client_secret):
|
|
|
|
self.server_url = server_url
|
|
|
|
self.realm_name = realm_name
|
|
|
|
self.client_id = client_id
|
|
|
|
self.client_secret = client_secret
|
2019-07-01 03:04:15 +00:00
|
|
|
self._openid_client = None
|
|
|
|
self._realm = None
|
|
|
|
self._openid_connect_client = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def realm(self):
|
|
|
|
if self._realm is None:
|
|
|
|
self._realm = KeycloakRealm(
|
|
|
|
server_url=self.server_url,
|
|
|
|
realm_name=self.realm_name,
|
|
|
|
headers={}
|
|
|
|
)
|
|
|
|
return self._realm
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
@property
|
|
|
|
def openid_connect_client(self):
|
2018-11-09 06:54:38 +00:00
|
|
|
"""
|
|
|
|
:rtype: keycloak.openid_connect.KeycloakOpenidConnect
|
|
|
|
"""
|
2019-07-01 03:04:15 +00:00
|
|
|
if self._openid_connect_client is None:
|
|
|
|
self._openid_connect_client = self.realm.open_id_connect(
|
|
|
|
client_id=self.client_id,
|
|
|
|
client_secret=self.client_secret
|
|
|
|
)
|
|
|
|
return self._openid_connect_client
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
@property
|
|
|
|
def openid_client(self):
|
2018-11-09 06:54:38 +00:00
|
|
|
"""
|
|
|
|
:rtype: keycloak.keycloak_openid.KeycloakOpenID
|
|
|
|
"""
|
2019-07-01 03:04:15 +00:00
|
|
|
if self._openid_client is None:
|
|
|
|
self._openid_client = KeycloakOpenID(
|
|
|
|
server_url='%sauth/' % self.server_url,
|
|
|
|
realm_name=self.realm_name,
|
|
|
|
client_id=self.client_id,
|
|
|
|
client_secret_key=self.client_secret,
|
|
|
|
)
|
|
|
|
return self._openid_client
|
|
|
|
|
|
|
|
@ssl_verification
|
|
|
|
def get_url(self, name):
|
|
|
|
return self.openid_connect_client.get_url(name=name)
|
|
|
|
|
|
|
|
def get_url_end_session_endpoint(self):
|
|
|
|
return self.get_url(name='end_session_endpoint')
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
@ssl_verification
|
|
|
|
def get_authorization_url(self, redirect_uri, scope, state):
|
|
|
|
url = self.openid_connect_client.authorization_url(
|
|
|
|
redirect_uri=redirect_uri, scope=scope, state=state
|
2018-11-09 06:54:38 +00:00
|
|
|
)
|
2019-07-01 03:04:15 +00:00
|
|
|
return url
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
@ssl_verification
|
|
|
|
def get_userinfo(self, token):
|
|
|
|
user_info = self.openid_connect_client.userinfo(token=token)
|
|
|
|
return user_info
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
@ssl_verification
|
|
|
|
def authorization_code(self, code, redirect_uri):
|
|
|
|
token_response = self.openid_connect_client.authorization_code(
|
|
|
|
code=code, redirect_uri=redirect_uri
|
|
|
|
)
|
|
|
|
return token_response
|
|
|
|
|
|
|
|
@ssl_verification
|
|
|
|
def authorization_password(self, username, password):
|
2018-11-09 06:54:38 +00:00
|
|
|
token_response = self.openid_client.token(
|
|
|
|
username=username, password=password
|
|
|
|
)
|
2019-07-01 03:04:15 +00:00
|
|
|
return token_response
|
2018-11-09 06:54:38 +00:00
|
|
|
|
|
|
|
def update_or_create_from_code(self, code, redirect_uri):
|
|
|
|
"""
|
|
|
|
Update or create an user based on an authentication code.
|
|
|
|
Response as specified in:
|
|
|
|
https://tools.ietf.org/html/rfc6749#section-4.1.4
|
|
|
|
:param str code: authentication code
|
|
|
|
:param str redirect_uri:
|
2019-02-28 06:41:33 +00:00
|
|
|
:rtype: OpenIDTokenProfile
|
2018-11-09 06:54:38 +00:00
|
|
|
"""
|
2019-07-01 03:04:15 +00:00
|
|
|
token_response = self.authorization_code(code, redirect_uri)
|
|
|
|
return self._update_or_create(token_response=token_response)
|
2018-11-09 06:54:38 +00:00
|
|
|
|
2019-07-01 03:04:15 +00:00
|
|
|
def update_or_create_from_password(self, username, password):
|
|
|
|
"""
|
|
|
|
Update or create an user based on an authentication username and password.
|
|
|
|
:param str username: authentication username
|
|
|
|
:param str password: authentication password
|
|
|
|
:return: OpenIDTokenProfile
|
|
|
|
"""
|
|
|
|
token_response = self.authorization_password(username, password)
|
2018-11-09 06:54:38 +00:00
|
|
|
return self._update_or_create(token_response=token_response)
|
|
|
|
|
|
|
|
def _update_or_create(self, token_response):
|
|
|
|
"""
|
|
|
|
Update or create an user based on a token response.
|
|
|
|
`token_response` contains the items returned by the OpenIDConnect Token API
|
|
|
|
end-point:
|
|
|
|
- id_token
|
|
|
|
- access_token
|
|
|
|
- expires_in
|
|
|
|
- refresh_token
|
|
|
|
- refresh_expires_in
|
|
|
|
:param dict token_response:
|
2019-02-28 06:41:33 +00:00
|
|
|
:rtype: OpenIDTokenProfile
|
2018-11-09 06:54:38 +00:00
|
|
|
"""
|
2019-07-01 03:04:15 +00:00
|
|
|
userinfo = self.get_userinfo(token=token_response['access_token'])
|
2018-11-09 06:54:38 +00:00
|
|
|
with transaction.atomic():
|
2019-12-09 08:12:48 +00:00
|
|
|
user, created = get_user_model().objects.update_or_create(
|
2018-11-09 06:54:38 +00:00
|
|
|
username=userinfo.get('preferred_username', ''),
|
|
|
|
defaults={
|
|
|
|
'email': userinfo.get('email', ''),
|
|
|
|
'first_name': userinfo.get('given_name', ''),
|
2020-01-27 02:15:06 +00:00
|
|
|
'last_name': userinfo.get('family_name', ''),
|
|
|
|
'name': userinfo.get('name', '')
|
2018-11-09 06:54:38 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
oidt_profile = OpenIDTokenProfile(
|
|
|
|
user=user,
|
|
|
|
access_token=token_response['access_token'],
|
|
|
|
refresh_token=token_response['refresh_token'],
|
|
|
|
)
|
|
|
|
if user:
|
2019-12-09 08:12:48 +00:00
|
|
|
post_create_or_update_openid_user.send(
|
|
|
|
sender=user.__class__, user=user, created=created
|
|
|
|
)
|
2018-11-09 06:54:38 +00:00
|
|
|
|
|
|
|
return oidt_profile
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
return self.client_id
|