2014-08-05 23:23:55 -05:00
|
|
|
# -*- coding: utf-8 -*-
|
2013-05-07 19:41:58 -05:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# Rattail -- Retail Software Framework
|
2016-02-01 16:11:24 -06:00
|
|
|
# Copyright © 2010-2016 Lance Edgar
|
2013-05-07 19:41:58 -05:00
|
|
|
#
|
|
|
|
# This file is part of Rattail.
|
|
|
|
#
|
|
|
|
# Rattail is free software: you can redistribute it and/or modify it under the
|
|
|
|
# terms of the GNU Affero General Public License as published by the Free
|
|
|
|
# Software Foundation, either version 3 of the License, or (at your option)
|
|
|
|
# any later version.
|
|
|
|
#
|
|
|
|
# Rattail is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
|
|
# FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for
|
|
|
|
# more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
|
|
# along with Rattail. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
"""
|
2013-09-01 09:27:47 -05:00
|
|
|
User Views
|
2013-05-07 19:41:58 -05:00
|
|
|
"""
|
|
|
|
|
2016-02-01 16:11:24 -06:00
|
|
|
from __future__ import unicode_literals, absolute_import
|
2014-08-05 23:23:55 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
from sqlalchemy import orm
|
|
|
|
|
2015-02-09 15:33:16 -06:00
|
|
|
from rattail.db import model
|
2016-06-15 12:51:10 -05:00
|
|
|
from rattail.db.auth import guest_role, authenticated_role, set_user_password
|
2015-01-19 00:45:26 -06:00
|
|
|
|
|
|
|
import formalchemy
|
2013-09-01 17:31:50 -05:00
|
|
|
from formalchemy.fields import SelectFieldRenderer
|
2015-08-11 23:19:41 -05:00
|
|
|
from webhelpers.html import HTML, tags
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2016-08-12 01:58:07 -05:00
|
|
|
from tailbone import forms
|
2015-08-11 23:19:41 -05:00
|
|
|
from tailbone.db import Session
|
|
|
|
from tailbone.views import MasterView
|
|
|
|
from tailbone.views.continuum import VersionView, version_defaults
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2013-09-01 17:31:50 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def unique_username(value, field):
|
|
|
|
user = field.parent.model
|
|
|
|
query = Session.query(model.User).filter(model.User.username == value)
|
|
|
|
if user.uuid:
|
|
|
|
query = query.filter(model.User.uuid != user.uuid)
|
|
|
|
if query.count():
|
|
|
|
raise formalchemy.ValidationError("Username must be unique.")
|
2013-09-01 17:31:50 -05:00
|
|
|
|
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def passwords_match(value, field):
|
|
|
|
if field.parent.confirm_password.value != value:
|
|
|
|
raise formalchemy.ValidationError("Passwords do not match")
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
class PasswordFieldRenderer(formalchemy.PasswordFieldRenderer):
|
|
|
|
|
|
|
|
def render(self, **kwargs):
|
|
|
|
return tags.password(self.name, value='', maxlength=self.length, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
class PasswordField(formalchemy.Field):
|
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
kwargs.setdefault('value', lambda x: x.password)
|
|
|
|
kwargs.setdefault('renderer', PasswordFieldRenderer)
|
|
|
|
kwargs.setdefault('validate', passwords_match)
|
|
|
|
super(PasswordField, self).__init__(*args, **kwargs)
|
2013-09-01 17:31:50 -05:00
|
|
|
|
|
|
|
def sync(self):
|
|
|
|
if not self.is_readonly():
|
2015-08-11 23:19:41 -05:00
|
|
|
password = self.renderer.deserialize()
|
|
|
|
if password:
|
|
|
|
set_user_password(self.model, password)
|
|
|
|
|
2013-09-01 17:31:50 -05:00
|
|
|
|
|
|
|
def RolesFieldRenderer(request):
|
|
|
|
|
|
|
|
class RolesFieldRenderer(SelectFieldRenderer):
|
|
|
|
|
|
|
|
def render_readonly(self, **kwargs):
|
2015-08-11 23:19:41 -05:00
|
|
|
roles = Session.query(model.Role)
|
2013-09-01 17:31:50 -05:00
|
|
|
html = ''
|
|
|
|
for uuid in self.value:
|
|
|
|
role = roles.get(uuid)
|
|
|
|
link = tags.link_to(
|
2015-08-11 15:01:21 -05:00
|
|
|
role.name, request.route_url('roles.view', uuid=role.uuid))
|
2013-09-01 17:31:50 -05:00
|
|
|
html += HTML.tag('li', c=link)
|
|
|
|
html = HTML.tag('ul', c=html)
|
|
|
|
return html
|
|
|
|
|
|
|
|
return RolesFieldRenderer
|
|
|
|
|
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
class RolesField(formalchemy.Field):
|
2015-01-19 00:45:26 -06:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def __init__(self, name, **kwargs):
|
|
|
|
kwargs.setdefault('value', self.get_value)
|
|
|
|
kwargs.setdefault('options', self.get_options())
|
|
|
|
kwargs.setdefault('multiple', True)
|
|
|
|
super(RolesField, self).__init__(name, **kwargs)
|
2015-01-19 00:45:26 -06:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def get_value(self, user):
|
|
|
|
return [x.uuid for x in user.roles]
|
2015-01-19 00:45:26 -06:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def get_options(self):
|
|
|
|
return Session.query(model.Role.name, model.Role.uuid)\
|
|
|
|
.filter(model.Role.uuid != guest_role(Session()).uuid)\
|
2016-06-15 12:51:10 -05:00
|
|
|
.filter(model.Role.uuid != authenticated_role(Session()).uuid)\
|
2015-08-11 23:19:41 -05:00
|
|
|
.order_by(model.Role.name)\
|
|
|
|
.all()
|
2015-01-19 00:45:26 -06:00
|
|
|
|
|
|
|
def sync(self):
|
|
|
|
if not self.is_readonly():
|
2015-08-11 23:19:41 -05:00
|
|
|
user = self.model
|
|
|
|
roles = Session.query(model.Role)
|
|
|
|
data = self.renderer.deserialize()
|
|
|
|
user.roles = [roles.get(x) for x in data]
|
|
|
|
|
2015-01-19 00:45:26 -06:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
class UsersView(MasterView):
|
|
|
|
"""
|
|
|
|
Master view for the User model.
|
|
|
|
"""
|
|
|
|
model_class = model.User
|
2015-01-19 00:45:26 -06:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def query(self, session):
|
|
|
|
return session.query(model.User)\
|
|
|
|
.options(orm.joinedload(model.User.person))
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def configure_grid(self, g):
|
|
|
|
g.joiners['person'] = lambda q: q.outerjoin(model.Person)
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
del g.filters['password']
|
|
|
|
del g.filters['salt']
|
|
|
|
g.filters['username'].default_active = True
|
|
|
|
g.filters['username'].default_verb = 'contains'
|
|
|
|
g.filters['active'].default_active = True
|
|
|
|
g.filters['active'].default_verb = 'is_true'
|
|
|
|
g.filters['person'] = g.make_filter('person', model.Person.display_name, label="Person's Name",
|
|
|
|
default_active=True, default_verb='contains')
|
2016-02-01 16:11:24 -06:00
|
|
|
g.filters['password'] = g.make_filter('password', model.User.password,
|
|
|
|
verbs=['is_null', 'is_not_null'])
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
g.sorters['person'] = lambda q, d: q.order_by(getattr(model.Person.display_name, d)())
|
|
|
|
g.default_sortkey = 'username'
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
g.person.set(label="Person's Name")
|
|
|
|
g.configure(
|
|
|
|
include=[
|
|
|
|
g.username,
|
|
|
|
g.person,
|
|
|
|
],
|
|
|
|
readonly=True)
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2015-08-11 23:19:41 -05:00
|
|
|
def configure_fieldset(self, fs):
|
2016-10-10 10:58:49 -05:00
|
|
|
fs.username.set(renderer=forms.renderers.StrippedTextFieldRenderer, validate=unique_username)
|
2015-08-11 23:19:41 -05:00
|
|
|
fs.person.set(options=[])
|
2016-08-12 01:58:07 -05:00
|
|
|
fs.person.set(renderer=forms.renderers.PersonFieldLinkRenderer)
|
2015-08-11 23:19:41 -05:00
|
|
|
fs.append(PasswordField('password', label="Set Password"))
|
2016-05-06 11:40:25 -05:00
|
|
|
fs.password.attrs(autocomplete='off')
|
2015-08-11 23:19:41 -05:00
|
|
|
fs.append(formalchemy.Field('confirm_password', renderer=PasswordFieldRenderer))
|
2016-05-06 11:40:25 -05:00
|
|
|
fs.confirm_password.attrs(autocomplete='off')
|
2015-08-11 23:19:41 -05:00
|
|
|
fs.append(RolesField('roles', renderer=RolesFieldRenderer(self.request)))
|
2013-05-07 19:41:58 -05:00
|
|
|
fs.configure(
|
|
|
|
include=[
|
|
|
|
fs.username,
|
2015-08-11 23:19:41 -05:00
|
|
|
fs.person,
|
|
|
|
fs.active,
|
|
|
|
fs.password,
|
2013-05-07 19:41:58 -05:00
|
|
|
fs.confirm_password,
|
|
|
|
fs.roles,
|
2015-08-11 23:19:41 -05:00
|
|
|
])
|
|
|
|
if self.viewing:
|
2016-08-12 01:58:07 -05:00
|
|
|
permissions = self.request.registry.settings.get('tailbone_permissions', {})
|
|
|
|
renderer = forms.renderers.PermissionsFieldRenderer(permissions,
|
|
|
|
include_guest=True,
|
|
|
|
include_authenticated=True)
|
|
|
|
fs.append(formalchemy.Field('permissions', renderer=renderer))
|
2016-10-25 18:14:43 -05:00
|
|
|
if self.viewing or self.deleting:
|
|
|
|
del fs.password
|
|
|
|
del fs.confirm_password
|
2013-05-07 19:41:58 -05:00
|
|
|
|
|
|
|
|
2015-02-09 15:33:16 -06:00
|
|
|
class UserVersionView(VersionView):
|
|
|
|
"""
|
|
|
|
View which shows version history for a user.
|
|
|
|
"""
|
|
|
|
parent_class = model.User
|
2015-12-08 16:18:29 -06:00
|
|
|
route_model_view = 'users.view'
|
2015-02-09 15:33:16 -06:00
|
|
|
|
|
|
|
|
2013-05-07 19:41:58 -05:00
|
|
|
def includeme(config):
|
2015-08-11 23:19:41 -05:00
|
|
|
UsersView.defaults(config)
|
2015-02-09 15:33:16 -06:00
|
|
|
version_defaults(config, UserVersionView, 'user')
|