2017-02-23 16:43:09 +00:00
|
|
|
from flask_login import UserMixin, AnonymousUserMixin
|
2017-02-17 14:06:09 +00:00
|
|
|
from flask import session
|
2016-03-01 10:45:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class User(UserMixin):
|
|
|
|
|
def __init__(self, fields, max_failed_login_count=3):
|
|
|
|
|
self._id = fields.get('id')
|
|
|
|
|
self._name = fields.get('name')
|
|
|
|
|
self._email_address = fields.get('email_address')
|
|
|
|
|
self._mobile_number = fields.get('mobile_number')
|
|
|
|
|
self._password_changed_at = fields.get('password_changed_at')
|
|
|
|
|
self._permissions = fields.get('permissions')
|
2017-11-01 15:36:27 +00:00
|
|
|
self._auth_type = fields.get('auth_type')
|
2016-03-11 11:47:21 +00:00
|
|
|
self._failed_login_count = fields.get('failed_login_count')
|
2016-03-01 10:45:13 +00:00
|
|
|
self._state = fields.get('state')
|
|
|
|
|
self.max_failed_login_count = max_failed_login_count
|
2016-03-17 10:46:47 +00:00
|
|
|
self.platform_admin = fields.get('platform_admin')
|
2017-02-17 14:06:09 +00:00
|
|
|
self.current_session_id = fields.get('current_session_id')
|
2016-03-01 10:45:13 +00:00
|
|
|
|
|
|
|
|
def get_id(self):
|
|
|
|
|
return self.id
|
|
|
|
|
|
2017-02-17 14:06:09 +00:00
|
|
|
def logged_in_elsewhere(self):
|
2017-02-23 16:38:18 +00:00
|
|
|
# if the current user (ie: db object) has no session, they've never logged in before
|
|
|
|
|
return self.current_session_id is not None and session.get('current_session_id') != self.current_session_id
|
2017-02-17 14:06:09 +00:00
|
|
|
|
2016-05-04 13:01:55 +01:00
|
|
|
@property
|
2016-03-01 10:45:13 +00:00
|
|
|
def is_active(self):
|
|
|
|
|
return self.state == 'active'
|
|
|
|
|
|
2016-05-04 13:01:55 +01:00
|
|
|
@property
|
2016-03-01 10:45:13 +00:00
|
|
|
def is_authenticated(self):
|
2017-02-17 14:06:09 +00:00
|
|
|
return (
|
|
|
|
|
not self.logged_in_elsewhere() and
|
|
|
|
|
super(User, self).is_authenticated
|
|
|
|
|
)
|
2016-03-01 10:45:13 +00:00
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def id(self):
|
|
|
|
|
return self._id
|
|
|
|
|
|
|
|
|
|
@id.setter
|
|
|
|
|
def id(self, id):
|
|
|
|
|
self._id = id
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def name(self):
|
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
|
|
@name.setter
|
|
|
|
|
def name(self, name):
|
|
|
|
|
self._name = name
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def email_address(self):
|
|
|
|
|
return self._email_address
|
|
|
|
|
|
|
|
|
|
@email_address.setter
|
|
|
|
|
def email_address(self, email_address):
|
|
|
|
|
self._email_address = email_address
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def mobile_number(self):
|
|
|
|
|
return self._mobile_number
|
|
|
|
|
|
|
|
|
|
@mobile_number.setter
|
|
|
|
|
def mobile_number(self, mobile_number):
|
|
|
|
|
self._mobile_number = mobile_number
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def password_changed_at(self):
|
|
|
|
|
return self._password_changed_at
|
|
|
|
|
|
|
|
|
|
@password_changed_at.setter
|
|
|
|
|
def password_changed_at(self, password_changed_at):
|
|
|
|
|
self._password_changed_at = password_changed_at
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def state(self):
|
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
|
|
@state.setter
|
|
|
|
|
def state(self, state):
|
|
|
|
|
self._state = state
|
|
|
|
|
|
|
|
|
|
@property
|
|
|
|
|
def permissions(self):
|
|
|
|
|
return self._permissions
|
|
|
|
|
|
|
|
|
|
@permissions.setter
|
|
|
|
|
def permissions(self, permissions):
|
|
|
|
|
raise AttributeError("Read only property")
|
|
|
|
|
|
2017-10-15 15:02:01 +01:00
|
|
|
def has_permissions(self, *permissions, any_=False, admin_override=False):
|
|
|
|
|
|
2016-03-18 16:20:37 +00:00
|
|
|
# Only available to the platform admin user
|
2016-03-17 10:46:47 +00:00
|
|
|
if admin_override and self.platform_admin:
|
|
|
|
|
return True
|
2016-03-18 16:20:37 +00:00
|
|
|
# Not available to the non platform admin users.
|
|
|
|
|
# For example the list all-services page is only available to platform admin users and is not service specific
|
|
|
|
|
if admin_override and not permissions:
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
from flask import request
|
|
|
|
|
# Service id is always set on the request for service specific views.
|
|
|
|
|
service_id = request.view_args.get('service_id', None)
|
2016-03-01 15:23:00 +00:00
|
|
|
if service_id in self._permissions:
|
2016-03-30 11:30:18 +01:00
|
|
|
if any_:
|
2016-03-01 15:23:00 +00:00
|
|
|
return any([x in self._permissions[service_id] for x in permissions])
|
2016-03-08 16:54:07 +00:00
|
|
|
return set(self._permissions[service_id]) >= set(permissions)
|
2016-03-01 15:23:00 +00:00
|
|
|
return False
|
2016-03-01 10:45:13 +00:00
|
|
|
|
2017-11-01 15:36:27 +00:00
|
|
|
@property
|
|
|
|
|
def auth_type(self):
|
|
|
|
|
return self._auth_type
|
|
|
|
|
|
|
|
|
|
@auth_type.setter
|
|
|
|
|
def auth_type(self, auth_type):
|
|
|
|
|
self._auth_type = auth_type
|
|
|
|
|
|
2016-03-01 10:45:13 +00:00
|
|
|
@property
|
|
|
|
|
def failed_login_count(self):
|
|
|
|
|
return self._failed_login_count
|
|
|
|
|
|
|
|
|
|
@failed_login_count.setter
|
|
|
|
|
def failed_login_count(self, num):
|
|
|
|
|
self._failed_login_count += num
|
|
|
|
|
|
|
|
|
|
def is_locked(self):
|
|
|
|
|
return self.failed_login_count >= self.max_failed_login_count
|
|
|
|
|
|
|
|
|
|
def serialize(self):
|
2017-02-17 14:06:09 +00:00
|
|
|
dct = {
|
|
|
|
|
"id": self.id,
|
|
|
|
|
"name": self.name,
|
|
|
|
|
"email_address": self.email_address,
|
|
|
|
|
"mobile_number": self.mobile_number,
|
|
|
|
|
"password_changed_at": self.password_changed_at,
|
|
|
|
|
"state": self.state,
|
|
|
|
|
"failed_login_count": self.failed_login_count,
|
|
|
|
|
"permissions": [x for x in self._permissions],
|
|
|
|
|
"current_session_id": self.current_session_id
|
|
|
|
|
}
|
|
|
|
|
if hasattr(self, '_password'):
|
2016-03-01 10:45:13 +00:00
|
|
|
dct['password'] = self._password
|
|
|
|
|
return dct
|
|
|
|
|
|
|
|
|
|
def set_password(self, pwd):
|
|
|
|
|
self._password = pwd
|
2016-02-29 17:35:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class InvitedUser(object):
|
|
|
|
|
|
2017-11-13 13:39:31 +00:00
|
|
|
def __init__(self, id, service, from_user, email_address, permissions, status, created_at, auth_type):
|
2016-02-29 17:35:21 +00:00
|
|
|
self.id = id
|
|
|
|
|
self.service = str(service)
|
|
|
|
|
self.from_user = from_user
|
|
|
|
|
self.email_address = email_address
|
2016-03-03 15:26:18 +00:00
|
|
|
if isinstance(permissions, list):
|
|
|
|
|
self.permissions = permissions
|
|
|
|
|
else:
|
2016-03-11 12:50:25 +00:00
|
|
|
if permissions:
|
|
|
|
|
self.permissions = permissions.split(',')
|
|
|
|
|
else:
|
|
|
|
|
self.permissions = []
|
2016-02-29 17:35:21 +00:00
|
|
|
self.status = status
|
|
|
|
|
self.created_at = created_at
|
2017-11-01 15:36:27 +00:00
|
|
|
self.auth_type = auth_type
|
2016-02-29 17:35:21 +00:00
|
|
|
|
2017-10-15 15:02:01 +01:00
|
|
|
def has_permissions(self, *permissions):
|
2016-03-03 13:00:12 +00:00
|
|
|
return set(self.permissions) > set(permissions)
|
2016-03-02 15:25:04 +00:00
|
|
|
|
2016-03-03 15:26:18 +00:00
|
|
|
def __eq__(self, other):
|
|
|
|
|
return ((self.id,
|
|
|
|
|
self.service,
|
|
|
|
|
self.from_user,
|
|
|
|
|
self.email_address,
|
2017-11-01 15:36:27 +00:00
|
|
|
self.auth_type,
|
2016-03-03 15:26:18 +00:00
|
|
|
self.status) == (other.id,
|
|
|
|
|
other.service,
|
|
|
|
|
other.from_user,
|
|
|
|
|
other.email_address,
|
2017-11-01 15:36:27 +00:00
|
|
|
other.auth_type,
|
2016-03-03 15:26:18 +00:00
|
|
|
other.status))
|
|
|
|
|
|
|
|
|
|
def serialize(self, permissions_as_string=False):
|
|
|
|
|
data = {'id': self.id,
|
2016-03-02 15:25:04 +00:00
|
|
|
'service': self.service,
|
|
|
|
|
'from_user': self.from_user,
|
|
|
|
|
'email_address': self.email_address,
|
2016-03-03 15:26:18 +00:00
|
|
|
'status': self.status,
|
2017-11-01 15:36:27 +00:00
|
|
|
'created_at': str(self.created_at),
|
|
|
|
|
'auth_type': self.auth_type
|
2016-03-02 15:25:04 +00:00
|
|
|
}
|
2016-03-03 15:26:18 +00:00
|
|
|
if permissions_as_string:
|
|
|
|
|
data['permissions'] = ','.join(self.permissions)
|
|
|
|
|
else:
|
|
|
|
|
data['permissions'] = self.permissions
|
|
|
|
|
return data
|
2017-02-17 14:06:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class AnonymousUser(AnonymousUserMixin):
|
|
|
|
|
# set the anonymous user so that if a new browser hits us we don't error http://stackoverflow.com/a/19275188
|
|
|
|
|
def logged_in_elsewhere(self):
|
|
|
|
|
return False
|