Add basic support for performing / tracking app upgrades
also add `MasterView.executable` and friends
This commit is contained in:
parent
f476c696fd
commit
f5688f1f90
11 changed files with 386 additions and 26 deletions
|
@ -26,21 +26,25 @@ Model Master View
|
|||
|
||||
from __future__ import unicode_literals, absolute_import
|
||||
|
||||
import os
|
||||
import logging
|
||||
|
||||
import six
|
||||
import sqlalchemy as sa
|
||||
from sqlalchemy import orm
|
||||
|
||||
import sqlalchemy_continuum as continuum
|
||||
|
||||
from rattail.db import Session as RattailSession
|
||||
from rattail.db import model, Session as RattailSession
|
||||
from rattail.db.continuum import model_transaction_query
|
||||
from rattail.util import prettify
|
||||
from rattail.time import localtime
|
||||
from rattail.time import localtime #, make_utc
|
||||
from rattail.threads import Thread
|
||||
|
||||
import formalchemy as fa
|
||||
from pyramid import httpexceptions
|
||||
from pyramid.renderers import get_renderer, render_to_response, render
|
||||
from pyramid.response import FileResponse
|
||||
from webhelpers2.html import HTML, tags
|
||||
|
||||
from tailbone import forms, grids
|
||||
|
@ -48,6 +52,9 @@ from tailbone.views import View
|
|||
from tailbone.progress import SessionProgress
|
||||
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class MasterView(View):
|
||||
"""
|
||||
Base "master" view class. All model master views should derive from this.
|
||||
|
@ -64,6 +71,7 @@ class MasterView(View):
|
|||
bulk_deletable = False
|
||||
mergeable = False
|
||||
downloadable = False
|
||||
executable = False
|
||||
|
||||
supports_mobile = False
|
||||
mobile_creatable = False
|
||||
|
@ -236,7 +244,10 @@ class MasterView(View):
|
|||
self.after_create(obj)
|
||||
self.flash_after_create(obj)
|
||||
return self.redirect_after_create(obj)
|
||||
return self.render_to_response('create', {'form': form})
|
||||
context = {'form': form}
|
||||
if hasattr(form, 'make_deform_form'):
|
||||
context['dform'] = form.make_deform_form()
|
||||
return self.render_to_response('create', context)
|
||||
|
||||
def mobile_create(self):
|
||||
"""
|
||||
|
@ -624,6 +635,27 @@ class MasterView(View):
|
|||
self.grid_count = len(data)
|
||||
return self.view(instance)
|
||||
|
||||
def download(self):
|
||||
"""
|
||||
View for downloading a data file.
|
||||
"""
|
||||
obj = self.get_instance()
|
||||
filename = self.request.GET.get('filename', None)
|
||||
path = self.download_path(obj, filename)
|
||||
response = FileResponse(path, request=self.request)
|
||||
response.content_length = os.path.getsize(path)
|
||||
content_type = self.download_content_type(path, filename)
|
||||
if content_type:
|
||||
response.content_type = six.binary_type(content_type)
|
||||
filename = os.path.basename(path).encode('ascii', 'replace')
|
||||
response.content_disposition = b'attachment; filename={}'.format(filename)
|
||||
return response
|
||||
|
||||
def download_content_type(self, path, filename):
|
||||
"""
|
||||
Return a content type for a file download, if known.
|
||||
"""
|
||||
|
||||
def edit(self):
|
||||
"""
|
||||
View for editing an existing model record.
|
||||
|
@ -647,11 +679,15 @@ class MasterView(View):
|
|||
self.get_model_title(), self.get_instance_title(instance)))
|
||||
return self.redirect_after_edit(instance)
|
||||
|
||||
return self.render_to_response('edit', {
|
||||
context = {
|
||||
'instance': instance,
|
||||
'instance_title': instance_title,
|
||||
'instance_deletable': self.deletable_instance(instance),
|
||||
'form': form})
|
||||
'form': form,
|
||||
}
|
||||
if hasattr(form, 'make_deform_form'):
|
||||
context['dform'] = form.make_deform_form()
|
||||
return self.render_to_response('edit', context)
|
||||
|
||||
def validate_form(self, form):
|
||||
return form.validate()
|
||||
|
@ -760,6 +796,64 @@ class MasterView(View):
|
|||
progress.session['success_url'] = self.get_index_url()
|
||||
progress.session.save()
|
||||
|
||||
def execute(self):
|
||||
"""
|
||||
Execute an object.
|
||||
"""
|
||||
obj = self.get_instance()
|
||||
model_title = self.get_model_title()
|
||||
if self.request.method == 'POST':
|
||||
|
||||
key = '{}.execute'.format(self.get_grid_key())
|
||||
kwargs = {'progress': SessionProgress(self.request, key)}
|
||||
thread = Thread(target=self.execute_thread, args=(obj.uuid, self.request.user.uuid), kwargs=kwargs)
|
||||
thread.start()
|
||||
|
||||
return self.render_progress({
|
||||
'key': key,
|
||||
'cancel_url': self.get_action_url('view', obj),
|
||||
'cancel_msg': "{} execution was canceled".format(model_title),
|
||||
})
|
||||
|
||||
self.request.session.flash("Sorry, you must POST to execute a {}.".format(model_title), 'error')
|
||||
return self.redirect(self.get_action_url('view', obj))
|
||||
|
||||
def execute_thread(self, uuid, user_uuid, progress=None, **kwargs):
|
||||
"""
|
||||
Thread target for executing an object.
|
||||
"""
|
||||
session = RattailSession()
|
||||
obj = session.query(self.model_class).get(uuid)
|
||||
user = session.query(model.User).get(user_uuid)
|
||||
try:
|
||||
self.execute_instance(obj, user, progress=progress, **kwargs)
|
||||
|
||||
# If anything goes wrong, rollback and log the error etc.
|
||||
except Exception as error:
|
||||
session.rollback()
|
||||
log.exception("execution failed for object: {}".format(obj))
|
||||
session.close()
|
||||
if progress:
|
||||
progress.session.load()
|
||||
progress.session['error'] = True
|
||||
progress.session['error_msg'] = self.execute_error_message(error)
|
||||
progress.session.save()
|
||||
|
||||
# If no error, check result flag (false means user canceled).
|
||||
else:
|
||||
session.commit()
|
||||
session.refresh(obj)
|
||||
success_url = self.get_execute_success_url(obj)
|
||||
session.close()
|
||||
if progress:
|
||||
progress.session.load()
|
||||
progress.session['complete'] = True
|
||||
progress.session['success_url'] = success_url
|
||||
progress.session.save()
|
||||
|
||||
def get_execute_success_url(self, obj, **kwargs):
|
||||
return self.get_action_url('view', obj, **kwargs)
|
||||
|
||||
def get_merge_fields(self):
|
||||
if hasattr(self, 'merge_fields'):
|
||||
return self.merge_fields
|
||||
|
@ -1709,6 +1803,14 @@ class MasterView(View):
|
|||
config.add_tailbone_permission(permission_prefix, '{0}.edit'.format(permission_prefix),
|
||||
"Edit {0}".format(model_title))
|
||||
|
||||
# execute
|
||||
if cls.executable:
|
||||
config.add_tailbone_permission(permission_prefix, '{}.execute'.format(permission_prefix),
|
||||
"Execute {}".format(model_title))
|
||||
config.add_route('{}.execute'.format(route_prefix), '{}/{{{}}}/execute'.format(url_prefix, model_key))
|
||||
config.add_view(cls, attr='execute', route_name='{}.execute'.format(route_prefix),
|
||||
permission='{}.execute'.format(permission_prefix))
|
||||
|
||||
# delete
|
||||
if cls.deletable:
|
||||
config.add_route('{0}.delete'.format(route_prefix), '{0}/{{{1}}}/delete'.format(url_prefix, model_key))
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue