tailbone/tailbone/views/core.py
Lance Edgar 18cec49a86 Add websockets progress, "multi-system" support for upgrades
and related things to better support that
2022-08-20 17:39:33 -05:00

192 lines
6.4 KiB
Python

# -*- coding: utf-8; -*-
################################################################################
#
# Rattail -- Retail Software Framework
# Copyright © 2010-2022 Lance Edgar
#
# This file is part of Rattail.
#
# Rattail is free software: you can redistribute it and/or modify it under the
# terms of the GNU 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 General Public License for more
# details.
#
# You should have received a copy of the GNU General Public License along with
# Rattail. If not, see <http://www.gnu.org/licenses/>.
#
################################################################################
"""
Base View Class
"""
from __future__ import unicode_literals, absolute_import
import os
import six
from rattail.db import model
from rattail.core import Object
from rattail.util import progress_loop
from pyramid import httpexceptions
from pyramid.renderers import render_to_response
from pyramid.response import FileResponse
from tailbone.db import Session
from tailbone.auth import logout_user
from tailbone.progress import SessionProgress
from tailbone.util import should_use_buefy
from tailbone.config import protected_usernames
class View(object):
"""
Base class for all class-based views.
"""
# quickie (search)
expose_quickie_search = False
def __init__(self, request, context=None):
self.request = request
# if user becomes inactive while logged in, log them out
if getattr(request, 'user', None):
# TODO: why is the user sometimes not attached to session?
# (this has only been seen on mobile, when creating a new ordering batch)
if request.user not in Session():
request.user = Session.merge(request.user)
if not request.user.active:
headers = logout_user(request)
raise self.redirect(request.route_url('home'))
config = self.rattail_config
if config:
self.enum = config.get_enum()
self.model = config.get_model()
@property
def rattail_config(self):
"""
Reference to the effective Rattail config object.
"""
return getattr(self.request, 'rattail_config', None)
def get_rattail_app(self):
"""
Returns the Rattail ``AppHandler`` instance, creating it if necessary.
"""
if not hasattr(self, 'rattail_app'):
self.rattail_app = self.rattail_config.get_app()
return self.rattail_app
def forbidden(self):
"""
Convenience method, to raise a HTTP 403 Forbidden exception.
"""
return httpexceptions.HTTPForbidden()
def notfound(self):
return httpexceptions.HTTPNotFound()
def get_use_buefy(self):
"""
Returns a flag indicating whether or not the current theme supports
(and therefore should use) the Buefy JS library.
"""
return should_use_buefy(self.request)
def late_login_user(self):
"""
Returns the :class:`rattail:rattail.db.model.User` instance
corresponding to the "late login" form data (if any), or ``None``.
"""
if self.request.method == 'POST':
uuid = self.request.POST.get('late-login-user')
if uuid:
return Session.query(model.User).get(uuid)
def user_is_protected(self, user):
"""
This logic will consult the settings for a list of "protected"
usernames, which should require root privileges to edit. If the given
``user`` object is represented in this list, it is considered to be
protected and this method will return ``True``; otherwise it returns
``False``.
"""
if not hasattr(self, 'protected_usernames'):
self.protected_usernames = protected_usernames(self.rattail_config)
if self.protected_usernames and user.username in self.protected_usernames:
return True
return False
def redirect(self, url, **kwargs):
"""
Convenience method to return a HTTP 302 response.
"""
return httpexceptions.HTTPFound(location=url, **kwargs)
def progress_loop(self, func, items, factory, *args, **kwargs):
return progress_loop(func, items, factory, *args, **kwargs)
def make_progress(self, key, **kwargs):
"""
Create and return a :class:`tailbone.progress.SessionProgress`
instance, with the given key.
"""
return SessionProgress(self.request, key, **kwargs)
# TODO: this signature seems wonky
def render_progress(self, progress, kwargs, template=None):
"""
Render the progress page, with given kwargs as context.
"""
if not template:
template = '/progress.mako'
kwargs['progress'] = progress
kwargs.setdefault('can_cancel', True)
return render_to_response(template, kwargs, request=self.request)
def json_response(self, data):
"""
Convenience method to return a JSON response.
"""
return render_to_response('json', data,
request=self.request)
def file_response(self, path, filename=None, attachment=True):
"""
Returns a generic FileResponse from the given path
"""
if not os.path.exists(path):
return self.notfound()
response = FileResponse(path, request=self.request)
response.content_length = os.path.getsize(path)
if attachment:
if not filename:
filename = os.path.basename(path)
if six.PY2:
filename = filename.encode('ascii', 'replace')
response.content_disposition = str('attachment; filename="{}"'.format(filename))
return response
def get_quickie_context(self):
return Object(
url=self.get_quickie_url(),
perm=self.get_quickie_perm(),
placeholder=self.get_quickie_placeholder())
def get_quickie_url(self):
raise NotImplementedError
def get_quickie_perm(self):
raise NotImplementedError
def get_quickie_placeholder(self):
pass