reimbursinator/back/backend/views.py

302 lines
9.9 KiB
Python
Raw Normal View History

from rest_framework.decorators import api_view, parser_classes
from django.http import JsonResponse
from .models import *
from .policy import pol
import os
from rest_framework.exceptions import ParseError
from rest_framework.parsers import FileUploadParser, MultiPartParser
# function that prints all the reports
def get_reports(report_pk):
queryset = Report.objects.filter(id=report_pk)
for i in queryset:
data = {
"report_pk": report_pk,
"title": i.title,
"date_created": i.date_created,
"submitted": i.submitted,
"date_submitted": i.date_submitted,
}
# append the sections for each report
data.update(get_sections(i.id))
# return JsonResponse(data)
return data
# function that gets all the sections
# takes report_id param
def get_sections(r_id):
# create a dict of arrays for section
section_set = {"sections": []}
queryset = Section.objects.filter(report_id=r_id)
for index in range(len(queryset)):
i = queryset[index]
data = {
"id": i.id,
"completed": i.completed,
"title": i.title,
"html_description": i.html_description,
"rule_violations": [],
}
# append the fields for corresponding section
data.update(get_fields(i.id))
# process rules from the policy file if the section is completed
if i.completed:
rules = pol.sections[index].rules
for rule in rules:
try:
named_fields = generate_named_fields_for_section(data['fields'])
if not rule['rule'](data, named_fields):
info = {
"label": rule['title'],
"rule_break_text": rule['rule_break_text'],
}
data['rule_violations'].append(info)
except Exception as e:
print('Rule "{}" encountered an error. {}'.format(rule['title'], e))
# append section to the array
section_set["sections"].append(data.copy())
return section_set
# function that gets all the fields
# takes section_id param
def get_fields(s_id):
# create dict of arrays for fields
field_set = {"fields": []}
queryset = Field.objects.filter(section_id=s_id).order_by('number')
for i in queryset:
# function that gets the corresponding datatype
value = Field.get_datatype(i)
data = {
"completed": i.completed,
"field_name": i.field_name,
"label": i.label,
"field_type": i.field_type,
"number": i.number,
"value": value,
"id": i.id,
}
# append the fields to array
# use copy() to avoid overwriting
field_set["fields"].append(data.copy())
return field_set
def generate_named_fields_for_section(fields):
'''
Converts a section's field data into key-value pairs
for use in policy rule lambda functions.
'''
result = {}
for field in fields:
key = field['field_name']
value = field['value']
result[key] = value
return result
# API Endpoints
@api_view(['POST'])
def report(request):
'''
Generate a new empty report and return it
'''
# Create the report
report = Report.objects.create(user_id=request.user, title=request.data['title'],
date_created=datetime.date.today())
report.save()
# Create the sections
for i in range(len(pol.sections)):
section = pol.sections[i]
s = Section.objects.create(report_id=report, auto_submit=section.auto_submit,
required=section.required, completed=False,
title=section.title, html_description=section.html_description,
number=i)
s.save()
# Create the fields
for key in section.fields:
field = section.fields[key]
f = Field.objects.create(section_id=s, field_name=key, label=field['label'],
number=field['number'], field_type=field['field_type'], completed=False)
f.save()
# Return the newly created report
data = get_reports(report.id)
return JsonResponse(data)
# View the list of reports
@api_view(['GET'])
def reports(request):
report_set = {"reports": []}
queryset = Report.objects.all().filter(user_id=request.user.id).order_by('date_created')
for i in queryset:
data = {
"user_id": request.user.id,
"report_pk": i.id,
"title": i.title,
"date_created": i.date_created,
"submitted": i.submitted,
"date_submitted": i.date_submitted,
}
# append the sections for each report
report_set["reports"].append(data.copy())
return JsonResponse(report_set)
def user_owns_report(user, report):
'''
Returns true if the specified user is owner of the report
'''
report_to_check = Report.objects.filter(id=report)
if len(report_to_check) < 1:
return False
return report_to_check[0].user_id == user
# actions for an individual report
@api_view(['GET', 'PUT', 'DELETE'])
def report_detail(request, report_pk):
# Check that the user owns the report
if not user_owns_report(user=request.user, report=report_pk):
return JsonResponse({"message": "Current user does not own the specified report."}, status=401)
# view the report
if request.method == 'GET':
data = get_reports(report_pk)
return JsonResponse(data)
# submit the report
elif request.method == 'PUT':
return JsonResponse({"message": "Report submitted."})
# Delete the report
elif request.method == 'DELETE':
# get corresponding sections
section_set = Section.objects.filter(report_id=report_pk)
for i in section_set:
2019-02-11 03:59:28 +00:00
# gets the fields that only have a data file in them
field_set = Field.objects.filter(section_id=i.id).exclude(data_file__exact='')
if field_set.exists():
for j in field_set:
# delete the file if exists
path_name = str(j.data_file)
os.remove(path_name)
2019-02-11 03:59:28 +00:00
# delete the full report and catch the title
r = Report.objects.get(id=report_pk)
title = r.title
r.delete()
return JsonResponse({"message": "Deleted report: {0}.".format(title)})
def user_owns_section(user, section):
'''
Returns true if the specified user is owner of the section
'''
section_to_check = Section.objects.filter(id=section)
if len(section_to_check) < 1:
return False
report_to_check = section_to_check[0].report_id
return report_to_check.user_id == user
# update a section with new data
@api_view(['PUT'])
def section(request, report_pk, section_pk):
# Check that the user owns the report
if not user_owns_section(user=request.user, section=section_pk):
return JsonResponse({"message": "Current user does not own the specified section."}, status=401)
for key in request.data:
2019-02-14 07:36:29 +00:00
# get the matching field object
update = Field.objects.get(section_id=section_pk, field_name=key)
2019-02-14 07:36:29 +00:00
if update.field_type == "boolean":
# flight check
2019-02-14 22:17:25 +00:00
if request.data[key] == "true":
2019-02-14 07:36:29 +00:00
update.data_bool = True
2019-02-14 22:17:25 +00:00
elif request.data[key] == "false":
2019-02-14 07:36:29 +00:00
update.data_bool = False
update.completed = True
2019-02-14 07:36:29 +00:00
if update.field_type == "decimal":
# initialize to 0
if (
request.data[key] == "" or
request.data[key] is None
):
2019-02-14 07:36:29 +00:00
update.data_decimal = 0.0
else:
update.completed = True
2019-02-14 07:36:29 +00:00
update.data_decimal = request.data[key]
2019-02-14 07:36:29 +00:00
if update.field_type == "date":
# initialize to today's date
if (
request.data[key] == "" or
request.data[key] is None
):
2019-02-14 07:36:29 +00:00
update.data_date = None
else:
update.completed = True
2019-02-14 07:36:29 +00:00
update.data_date = request.data[key]
2019-02-14 07:36:29 +00:00
if update.field_type == "file":
if not(
request.data[key] == "" or
request.data[key] is None
):
update.completed = True
update.data_file = request.data[key]
2019-02-14 07:36:29 +00:00
if update.field_type == "string":
if not(
request.data[key] == "" or
request.data[key] is None
):
update.completed = True
update.data_string = request.data[key]
2019-02-14 07:36:29 +00:00
if update.field_type == "integer":
# initialize to 0
if (
request.data[key] == "" or
request.data[key] is None
):
2019-02-14 07:36:29 +00:00
update.data_integer = 0
else:
update.completed = True
2019-02-14 07:36:29 +00:00
update.data_integer = request.data[key]
2019-02-14 07:36:29 +00:00
update.save()
2019-02-14 21:26:26 +00:00
# update section boolean to complete
complete = section_complete(section_pk)
s = Section.objects.get(id=section_pk)
2019-02-14 21:26:26 +00:00
if complete:
s.completed = True
s.save()
else:
s.completed = False
2019-02-14 21:26:26 +00:00
data = {
"message": "Updated report {0}, section {1}.".format(report_pk, section_pk),
"section completion": s.completed,
"request_data": "{}".format(request.data)
}
return JsonResponse(data)
# function checks if a field is complete
2019-02-14 21:26:26 +00:00
def section_complete(section_pk):
# grab field set
check_fields = Field.objects.filter(section_id=section_pk)
# return true if any field is complete
2019-02-14 21:26:26 +00:00
for i in check_fields:
if i.completed:
return True
return False