mirror of
https://github.com/nottinghamtec/PyRIGS.git
synced 2026-01-17 05:22:16 +00:00
* Upgrade to heroku-20 stack * Move some gulp deps to dev rather than prod * npm upgrade * Fix audit time check in asset audit test * Attempt at parallelising tests where possible * Add basic calendar button test Mainly to pickup on FullCalendar loading errors * Upgrade python deps * Tends to help if I push valid yaml * You valid now? * Fix whoops in requirements.txt * Change python ver * Define service in coveralls task * Run parallelised RIGS tests as one matrix job * Update python version in tests * Cache python dependencies Should majorly speedup parallelillelelised testing * Purge old vagrant config * No Ruby compass bodge, no need for rubocop! * Purge old .idea config * Switch to gh-a artifact uploading instead of imgur 'hack' For test failure screenshots. Happy now @mattysmith22? ;p * Oops, remove unused import * Exclude tests from the coverage stats Seems to be artifically deflating our stats * Refactor asset audit tests with better selectors Also fixed a silly title error with the modal * Add title checking to the slightly insane assets test * Fix unauth test to not just immediately pass out * Upload failure screenshots as individual artifacts not a zip Turns out I can't unzip things from my phone, which is a pain * Should fix asset test on CI * What about this? * What about this? Swear I spend my life jiggerypokerying the damn test suite... * Does this help the coverage be less weird? * Revert "Does this help the coverage be less weird?" This reverts commit39ab9df836. * Use pytest as our test runner for better parallelism Also rewrote some asset tests to be in the pytest style. May do some more. Some warnings cleaned up in the process. * Bah, codestyle * Oops, remove obsolete if check * Fix screenshot uploading on CI (again) * Try this way of parallel coverage * Add codeclimate maintainability badge * Remove some unused gulp dependencies * Run asset building serverside * Still helps if I commit valid YAML * See below * Different approach to CI dependencies * Exclude node_modules from codestyle * Does this work? * Parallel parallel builds were giving me a headache, try this * Update codeclimate settings, purge some config files * Well the YAML was *syntactically* valid.... * Switch back to old coveralls method * Fix codeclimate config, mark 2 * Attempt to bodge asset test * Oops, again Probably bedtime.. * Might fix heroku building * Attempt #2 at fixing heroku * Belt and braces approach to coverage * Github, you need a Actions YAML validator! * Might fix actions? * Try ignoring some third party deprecation warnings * Another go at making coverage show up * Some template cleanup * Minor python cleanup * Import optimisation * Revert "Minor python cleanup" This reverts commit6a4620a2e5. * Add format arg to coverage command * Ignore test directories from Heroku slug * Maybe this works to purge deps postbuild * Bunch of test refactoring * Restore signals import, screw you import optimisation * Further template refactoring * Add support for running tests with geckodriver, do this on CI * Screw you codestyle * Disable firefox tests for now That was way more errors than I expected * Run cleanup script from the right location * Plausibly fix tests * Helps if I don't delete the pipeline folder prior to collectstatic * Enable whitenoise * Can I delete pipeline here? * Allow seconds difference in assert_times_equal * Disable codeclimate * Remove not working rm command * Maybe this fixes coverage? * Try different coverage reporter * Fix search_help to need login * Made versioning magic a bit less expansive We have more apps than I thought... * Fix IDI0T error in Assets URLS * Refactor 'no access to unauthed' test to cover all of PyRIGS * Add RAs/Checklists to sample data generator * Fix some HTML errors in templates Which apparently only Django's HTML parser cares about, browsers DGAF... * Port title test to project level * Fix more HTML * Fix cable type detail
431 lines
16 KiB
Python
431 lines
16 KiB
Python
import copy
|
|
import datetime
|
|
import re
|
|
import urllib.error
|
|
import urllib.parse
|
|
import urllib.request
|
|
from io import BytesIO
|
|
|
|
import premailer
|
|
import simplejson
|
|
from PyPDF2 import PdfFileMerger, PdfFileReader
|
|
from django.conf import settings
|
|
from django.contrib import messages
|
|
from django.contrib.staticfiles.storage import staticfiles_storage
|
|
from django.core import signing
|
|
from django.core.exceptions import SuspiciousOperation
|
|
from django.core.mail import EmailMultiAlternatives
|
|
from django.db.models import Q
|
|
from django.http import HttpResponse
|
|
from django.shortcuts import get_object_or_404
|
|
from django.template.loader import get_template
|
|
from django.urls import reverse
|
|
from django.urls import reverse_lazy
|
|
from django.utils import timezone
|
|
from django.utils.decorators import method_decorator
|
|
from django.views import generic
|
|
from z3c.rml import rml2pdf
|
|
|
|
from PyRIGS import decorators
|
|
from RIGS import models, forms
|
|
|
|
__author__ = 'ghost'
|
|
|
|
|
|
class RigboardIndex(generic.TemplateView):
|
|
template_name = 'rigboard.html'
|
|
|
|
def get_context_data(self, **kwargs):
|
|
# get super context
|
|
context = super(RigboardIndex, self).get_context_data(**kwargs)
|
|
|
|
# call out method to get current events
|
|
context['events'] = models.Event.objects.current_events()
|
|
context['page_title'] = "Rigboard"
|
|
return context
|
|
|
|
|
|
class WebCalendar(generic.TemplateView):
|
|
template_name = 'calendar.html'
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(WebCalendar, self).get_context_data(**kwargs)
|
|
context['view'] = kwargs.get('view', '')
|
|
context['date'] = kwargs.get('date', '')
|
|
return context
|
|
|
|
|
|
class EventDetail(generic.DetailView):
|
|
template_name = 'event_detail.html'
|
|
model = models.Event
|
|
|
|
|
|
class EventOembed(generic.View):
|
|
model = models.Event
|
|
|
|
def get(self, request, pk=None):
|
|
embed_url = reverse('event_embed', args=[pk])
|
|
full_url = "{0}://{1}{2}".format(request.scheme, request.META['HTTP_HOST'], embed_url)
|
|
|
|
data = {
|
|
'html': '<iframe src="{0}" frameborder="0" width="100%" height="250"></iframe>'.format(full_url),
|
|
'version': '1.0',
|
|
'type': 'rich',
|
|
'height': '250'
|
|
}
|
|
|
|
json = simplejson.JSONEncoderForHTML().encode(data)
|
|
return HttpResponse(json, content_type="application/json")
|
|
|
|
|
|
class EventEmbed(EventDetail):
|
|
template_name = 'event_embed.html'
|
|
|
|
|
|
class EventCreate(generic.CreateView):
|
|
model = models.Event
|
|
form_class = forms.EventForm
|
|
template_name = 'event_form.html'
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(EventCreate, self).get_context_data(**kwargs)
|
|
context['page_title'] = "New Event"
|
|
context['edit'] = True
|
|
context['currentVAT'] = models.VatRate.objects.current_rate()
|
|
|
|
form = context['form']
|
|
if hasattr(form, 'items_json') and re.search(r'"-\d+"', form['items_json'].value()):
|
|
messages.info(self.request, "Your item changes have been saved. Please fix the errors and save the event.")
|
|
|
|
# Get some other objects to include in the form. Used when there are errors but also nice and quick.
|
|
for field, model in form.related_models.items():
|
|
value = form[field].value()
|
|
if value is not None and value != '':
|
|
context[field] = model.objects.get(pk=value)
|
|
return context
|
|
|
|
def get_success_url(self):
|
|
return reverse_lazy('event_detail', kwargs={'pk': self.object.pk})
|
|
|
|
|
|
class EventUpdate(generic.UpdateView):
|
|
model = models.Event
|
|
form_class = forms.EventForm
|
|
template_name = 'event_form.html'
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(EventUpdate, self).get_context_data(**kwargs)
|
|
context['page_title'] = "Event {}".format(self.object.display_id)
|
|
context['edit'] = True
|
|
|
|
form = context['form']
|
|
|
|
# Get some other objects to include in the form. Used when there are errors but also nice and quick.
|
|
for field, model in form.related_models.items():
|
|
value = form[field].value()
|
|
if value is not None and value != '':
|
|
context[field] = model.objects.get(pk=value)
|
|
|
|
return context
|
|
|
|
def render_to_response(self, context, **response_kwargs):
|
|
if hasattr(context, 'duplicate') and not context['duplicate']:
|
|
# If this event has already been emailed to a client, show a warning
|
|
if self.object.auth_request_at is not None:
|
|
messages.info(self.request,
|
|
'This event has already been sent to the client for authorisation, any changes you make will be visible to them immediately.')
|
|
|
|
if hasattr(self.object, 'authorised'):
|
|
messages.warning(self.request,
|
|
'This event has already been authorised by the client, any changes to the price will require reauthorisation.')
|
|
return super(EventUpdate, self).render_to_response(context, **response_kwargs)
|
|
|
|
def get_success_url(self):
|
|
return reverse_lazy('event_detail', kwargs={'pk': self.object.pk})
|
|
|
|
|
|
class EventDuplicate(EventUpdate):
|
|
def get_object(self, queryset=None):
|
|
old = super(EventDuplicate, self).get_object(queryset) # Get the object (the event you're duplicating)
|
|
new = copy.copy(old) # Make a copy of the object in memory
|
|
new.based_on = old # Make the new event based on the old event
|
|
new.purchase_order = None # Remove old PO
|
|
new.status = new.PROVISIONAL # Return status to provisional
|
|
|
|
# Clear checked in by if it's a dry hire
|
|
if new.dry_hire is True:
|
|
new.checked_in_by = None
|
|
|
|
# Remove all the authorisation information from the new event
|
|
new.auth_request_to = None
|
|
new.auth_request_by = None
|
|
new.auth_request_at = None
|
|
|
|
if self.request.method in (
|
|
'POST', 'PUT'): # This only happens on save (otherwise items won't display in editor)
|
|
new.pk = None # This means a new event will be created on save, and all items will be re-created
|
|
else:
|
|
messages.info(self.request, 'Event data duplicated but not yet saved. Click save to complete operation.')
|
|
|
|
return new
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(EventDuplicate, self).get_context_data(**kwargs)
|
|
context['page_title'] = "Duplicate of Event {}".format(self.object.display_id)
|
|
context["duplicate"] = True
|
|
return context
|
|
|
|
|
|
class EventPrint(generic.View):
|
|
def get(self, request, pk):
|
|
object = get_object_or_404(models.Event, pk=pk)
|
|
template = get_template('event_print.xml')
|
|
|
|
merger = PdfFileMerger()
|
|
|
|
context = {
|
|
'object': object,
|
|
'fonts': {
|
|
'opensans': {
|
|
'regular': 'static/fonts/OPENSANS-REGULAR.TTF',
|
|
'bold': 'static/fonts/OPENSANS-BOLD.TTF',
|
|
}
|
|
},
|
|
'quote': True,
|
|
'current_user': request.user,
|
|
'filename': 'Event {} {} {}.pdf'.format(object.display_id, re.sub(r'[^a-zA-Z0-9 \n\.]', '', object.name), object.start_date)
|
|
}
|
|
|
|
rml = template.render(context)
|
|
buffer = rml2pdf.parseString(rml)
|
|
merger.append(PdfFileReader(buffer))
|
|
buffer.close()
|
|
|
|
terms = urllib.request.urlopen(settings.TERMS_OF_HIRE_URL)
|
|
merger.append(BytesIO(terms.read()))
|
|
|
|
merged = BytesIO()
|
|
merger.write(merged)
|
|
|
|
response = HttpResponse(content_type='application/pdf')
|
|
response['Content-Disposition'] = 'filename="{}"'.format(context['filename'])
|
|
response.write(merged.getvalue())
|
|
return response
|
|
|
|
|
|
class EventArchive(generic.ListView):
|
|
template_name = "event_archive.html"
|
|
model = models.Event
|
|
paginate_by = 25
|
|
|
|
def get_context_data(self, **kwargs):
|
|
# get super context
|
|
context = super(EventArchive, self).get_context_data(**kwargs)
|
|
|
|
context['start'] = self.request.GET.get('start', None)
|
|
context['end'] = self.request.GET.get('end', datetime.date.today().strftime('%Y-%m-%d'))
|
|
context['statuses'] = models.Event.EVENT_STATUS_CHOICES
|
|
context['page_title'] = 'Event Archive'
|
|
return context
|
|
|
|
def get_queryset(self):
|
|
start = self.request.GET.get('start', None)
|
|
end = self.request.GET.get('end', datetime.date.today())
|
|
|
|
# Assume idiots, always check
|
|
if start and start > end:
|
|
messages.add_message(self.request, messages.INFO,
|
|
"Muppet! Check the dates, it has been fixed for you.")
|
|
start, end = end, start # Stop the impending fail
|
|
|
|
filter = Q()
|
|
if end != "":
|
|
filter &= Q(start_date__lte=end)
|
|
if start:
|
|
filter &= Q(start_date__gte=start)
|
|
|
|
q = self.request.GET.get('q', "")
|
|
|
|
if q != "":
|
|
qfilter = Q(name__icontains=q) | Q(description__icontains=q) | Q(notes__icontains=q)
|
|
|
|
# try and parse an int
|
|
try:
|
|
val = int(q)
|
|
qfilter = qfilter | Q(pk=val)
|
|
except: # noqa not an integer
|
|
pass
|
|
|
|
try:
|
|
if q[0] == "N":
|
|
val = int(q[1:])
|
|
qfilter = Q(pk=val) # If string is N###### then do a simple PK filter
|
|
except: # noqa
|
|
pass
|
|
|
|
filter &= qfilter
|
|
|
|
status = self.request.GET.getlist('status', "")
|
|
|
|
if len(status) > 0:
|
|
filter &= Q(status__in=status)
|
|
|
|
qs = self.model.objects.filter(filter).order_by('-start_date')
|
|
|
|
# Preselect related for efficiency
|
|
qs.select_related('person', 'organisation', 'venue', 'mic')
|
|
|
|
if len(qs) == 0:
|
|
messages.add_message(self.request, messages.WARNING, "No events have been found matching those criteria.")
|
|
|
|
return qs
|
|
|
|
|
|
class EventAuthorise(generic.UpdateView):
|
|
template_name = 'eventauthorisation_form.html'
|
|
success_template = 'eventauthorisation_success.html'
|
|
|
|
def form_valid(self, form):
|
|
self.object = form.save()
|
|
|
|
self.template_name = self.success_template
|
|
messages.add_message(self.request, messages.SUCCESS,
|
|
'Success! Your event has been authorised. ' +
|
|
'You will also receive email confirmation to %s.' % self.object.email)
|
|
return self.render_to_response(self.get_context_data())
|
|
|
|
@property
|
|
def event(self):
|
|
return models.Event.objects.select_related('organisation', 'person', 'venue').get(pk=self.kwargs['pk'])
|
|
|
|
def get_object(self, queryset=None):
|
|
return getattr(self.event, 'authorisation', None)
|
|
|
|
def get_form_class(self):
|
|
return forms.InternalClientEventAuthorisationForm
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(EventAuthorise, self).get_context_data(**kwargs)
|
|
context['event'] = self.event
|
|
context['tos_url'] = settings.TERMS_OF_HIRE_URL
|
|
context['page_title'] = "{}: {}".format(self.event.display_id, self.event.name)
|
|
if self.event.dry_hire:
|
|
context['page_title'] += ' <span class="badge badge-secondary align-top">Dry Hire</span>'
|
|
return context
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
if self.get_object() is not None and self.get_object().pk is not None:
|
|
if self.event.authorised:
|
|
messages.add_message(self.request, messages.WARNING,
|
|
"This event has already been authorised. "
|
|
"Reauthorising is not necessary at this time.")
|
|
else:
|
|
messages.add_message(self.request, messages.WARNING,
|
|
"This event has already been authorised, but the amount has changed. " +
|
|
"Please check the amount and reauthorise.")
|
|
return super(EventAuthorise, self).get(request, *args, **kwargs)
|
|
|
|
def get_form(self, **kwargs):
|
|
form = super(EventAuthorise, self).get_form(**kwargs)
|
|
form.instance.event = self.event
|
|
form.instance.email = self.request.email
|
|
form.instance.sent_by = self.request.sent_by
|
|
return form
|
|
|
|
def dispatch(self, request, *args, **kwargs):
|
|
# Verify our signature matches up and all is well with the integrity of the URL
|
|
try:
|
|
data = signing.loads(kwargs.get('hmac'))
|
|
assert int(kwargs.get('pk')) == int(data.get('pk'))
|
|
request.email = data['email']
|
|
request.sent_by = models.Profile.objects.get(pk=data['sent_by'])
|
|
except (signing.BadSignature, AssertionError, KeyError, models.Profile.DoesNotExist):
|
|
raise SuspiciousOperation(
|
|
"This URL is invalid. Please ask your TEC contact for a new URL")
|
|
return super(EventAuthorise, self).dispatch(request, *args, **kwargs)
|
|
|
|
|
|
class EventAuthorisationRequest(generic.FormView, generic.detail.SingleObjectMixin):
|
|
model = models.Event
|
|
form_class = forms.EventAuthorisationRequestForm
|
|
template_name = 'eventauthorisation_request.html'
|
|
|
|
@method_decorator(decorators.nottinghamtec_address_required)
|
|
def dispatch(self, *args, **kwargs):
|
|
return super(EventAuthorisationRequest, self).dispatch(*args, **kwargs)
|
|
|
|
@property
|
|
def object(self):
|
|
return self.get_object()
|
|
|
|
def get_success_url(self):
|
|
if self.request.is_ajax():
|
|
url = reverse_lazy('closemodal')
|
|
messages.info(self.request, "location.reload()")
|
|
else:
|
|
url = reverse_lazy('event_detail', kwargs={
|
|
'pk': self.object.pk,
|
|
})
|
|
messages.add_message(self.request, messages.SUCCESS, "Authorisation request successfully sent.")
|
|
return url
|
|
|
|
def form_valid(self, form):
|
|
email = form.cleaned_data['email']
|
|
event = self.object
|
|
event.auth_request_by = self.request.user
|
|
event.auth_request_at = timezone.now()
|
|
event.auth_request_to = email
|
|
event.save()
|
|
|
|
context = {
|
|
'object': self.object,
|
|
'request': self.request,
|
|
'hmac': signing.dumps({
|
|
'pk': self.object.pk,
|
|
'email': email,
|
|
'sent_by': self.request.user.pk,
|
|
}),
|
|
}
|
|
if event.person is not None and email == event.person.email:
|
|
context['to_name'] = event.person.name
|
|
elif event.organisation is not None and email == event.organisation.email:
|
|
context['to_name'] = event.organisation.name
|
|
|
|
msg = EmailMultiAlternatives(
|
|
"N%05d | %s - Event Authorisation Request" % (self.object.pk, self.object.name),
|
|
get_template("eventauthorisation_client_request.txt").render(context),
|
|
to=[email],
|
|
reply_to=[self.request.user.email],
|
|
)
|
|
css = staticfiles_storage.path('css/email.css')
|
|
html = premailer.Premailer(get_template("eventauthorisation_client_request.html").render(context),
|
|
external_styles=css).transform()
|
|
msg.attach_alternative(html, 'text/html')
|
|
|
|
msg.send()
|
|
|
|
return super(EventAuthorisationRequest, self).form_valid(form)
|
|
|
|
|
|
class EventAuthoriseRequestEmailPreview(generic.DetailView):
|
|
template_name = "eventauthorisation_client_request.html"
|
|
model = models.Event
|
|
|
|
def render_to_response(self, context, **response_kwargs):
|
|
from django.contrib.staticfiles.storage import staticfiles_storage
|
|
css = staticfiles_storage.path('css/email.css')
|
|
response = super(EventAuthoriseRequestEmailPreview, self).render_to_response(context, **response_kwargs)
|
|
assert isinstance(response, HttpResponse)
|
|
response.content = premailer.Premailer(response.rendered_content, external_styles=css).transform()
|
|
return response
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super(EventAuthoriseRequestEmailPreview, self).get_context_data(**kwargs)
|
|
context['hmac'] = signing.dumps({
|
|
'pk': self.object.pk,
|
|
'email': self.request.GET.get('email', 'hello@world.test'),
|
|
'sent_by': self.request.user.pk,
|
|
})
|
|
context['to_name'] = self.request.GET.get('to_name', None)
|
|
return context
|