2020-02-17 11:43:26 -05:00
|
|
|
import datetime
|
2020-03-23 13:57:19 -04:00
|
|
|
import os
|
2020-02-17 11:43:26 -05:00
|
|
|
from datetime import date
|
|
|
|
|
2020-01-08 15:57:00 -05:00
|
|
|
import connexion
|
2020-02-17 16:21:18 -05:00
|
|
|
import yaml
|
2020-02-14 15:19:30 -05:00
|
|
|
from flask import url_for, json, redirect, render_template, request, flash
|
2020-02-26 12:17:37 -05:00
|
|
|
from flask_assets import Environment, Bundle
|
2020-02-17 16:21:18 -05:00
|
|
|
from flask_sqlalchemy import SQLAlchemy
|
|
|
|
from flask_marshmallow import Marshmallow
|
2020-02-17 11:43:26 -05:00
|
|
|
from flask_migrate import Migrate
|
2020-02-14 15:19:30 -05:00
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
from wtforms.ext.appengine.db import model_form
|
2020-01-08 15:57:00 -05:00
|
|
|
|
|
|
|
PROTOCOLS = {}
|
|
|
|
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
def get_user_studies(uva_id):
|
|
|
|
studies = db.session.query(Study).filter(Study.NETBADGEID == uva_id).all()
|
|
|
|
return StudySchema(many=True).dump(studies)
|
2020-01-08 15:57:00 -05:00
|
|
|
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
def required_docs(studyid):
|
|
|
|
docs = db.session.query(RequiredDocument).filter(RequiredDocument.STUDYID == studyid).all()
|
|
|
|
return RequiredDocumentSchema(many=True).dump(docs)
|
2020-01-08 15:57:00 -05:00
|
|
|
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
def investigators(studyid):
|
|
|
|
inv = db.session.query(Investigator).filter(Investigator.STUDYID == studyid).all()
|
|
|
|
return InvestigatorSchema(many=True).dump(inv)
|
|
|
|
|
|
|
|
|
|
|
|
def get_study_details(studyid):
|
|
|
|
details = db.session.query(StudyDetails).filter(StudyDetails.STUDYID == studyid).first()
|
|
|
|
return StudyDetailsSchema().dump(details)
|
|
|
|
|
2020-01-08 15:57:00 -05:00
|
|
|
|
2020-02-11 14:50:32 -05:00
|
|
|
def get_form(id, requirement_code):
|
|
|
|
return
|
|
|
|
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
|
2020-02-14 15:19:30 -05:00
|
|
|
conn = connexion.App('Protocol Builder', specification_dir='./')
|
|
|
|
conn.add_api('api.yml')
|
2020-01-08 15:57:00 -05:00
|
|
|
|
2020-02-14 15:19:30 -05:00
|
|
|
app = conn.app
|
2020-01-08 15:57:00 -05:00
|
|
|
|
2020-03-23 13:57:19 -04:00
|
|
|
app.config.from_object('config.default')
|
|
|
|
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
|
|
|
|
|
|
|
|
if "TESTING" in os.environ and os.environ["TESTING"] == "true":
|
|
|
|
app.config.from_object('config.testing')
|
2020-03-23 17:05:34 -04:00
|
|
|
app.config.from_pyfile('config/testing.py')
|
2020-03-23 13:57:19 -04:00
|
|
|
else:
|
|
|
|
app.config.root_path = app.instance_path
|
|
|
|
app.config.from_pyfile('config.py', silent=True)
|
|
|
|
|
|
|
|
|
2020-02-14 15:19:30 -05:00
|
|
|
db = SQLAlchemy(app)
|
2020-02-17 11:43:26 -05:00
|
|
|
migrate = Migrate(app, db)
|
2020-02-17 16:21:18 -05:00
|
|
|
ma = Marshmallow(app)
|
2020-02-26 12:17:37 -05:00
|
|
|
assets = Environment(app)
|
|
|
|
assets.url = app.static_url_path
|
|
|
|
scss = Bundle('scss/app.scss', filters='pyscss', output='app.css')
|
|
|
|
assets.register('app_scss', scss)
|
2020-02-17 16:21:18 -05:00
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
# Loads all the descriptions from the API so we can display them in the editor.
|
2020-02-17 16:21:18 -05:00
|
|
|
description_map = {}
|
|
|
|
with open(r'api.yml') as file:
|
|
|
|
api_config = yaml.load(file, Loader=yaml.FullLoader)
|
|
|
|
study_detail_properties = api_config['components']['schemas']['StudyDetail']['properties']
|
|
|
|
for schema in api_config['components']['schemas']:
|
|
|
|
for field, values in api_config['components']['schemas'][schema]['properties'].items():
|
|
|
|
description_map[field] = values['description']
|
2020-01-08 15:57:00 -05:00
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
|
|
|
|
# **************************
|
|
|
|
# API ENDPOINTS
|
|
|
|
# **************************
|
2020-01-08 15:57:00 -05:00
|
|
|
def has_no_empty_params(rule):
|
|
|
|
defaults = rule.defaults if rule.defaults is not None else ()
|
|
|
|
arguments = rule.arguments if rule.arguments is not None else ()
|
|
|
|
return len(defaults) >= len(arguments)
|
|
|
|
|
2020-02-14 15:19:30 -05:00
|
|
|
@app.route("/site_map")
|
2020-01-08 15:57:00 -05:00
|
|
|
def site_map():
|
|
|
|
links = []
|
2020-02-14 15:19:30 -05:00
|
|
|
for rule in app.url_map.iter_rules():
|
2020-01-08 15:57:00 -05:00
|
|
|
# Filter out rules we can't navigate to in a browser
|
|
|
|
# and rules that require parameters
|
|
|
|
if "GET" in rule.methods and has_no_empty_params(rule):
|
|
|
|
url = url_for(rule.endpoint, **(rule.defaults or {}))
|
|
|
|
links.append((url, rule.endpoint))
|
|
|
|
return json.dumps({"links": links})
|
|
|
|
|
2020-01-08 16:05:16 -05:00
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# **************************
|
|
|
|
# WEB FORMS
|
|
|
|
# **************************
|
|
|
|
from forms import StudyForm, StudyTable, InvestigatorForm, StudyDetailsForm
|
|
|
|
from models import Study, RequiredDocument, Investigator, StudySchema, RequiredDocumentSchema, InvestigatorSchema, \
|
|
|
|
StudyDetails, StudyDetailsSchema
|
2020-02-17 11:43:26 -05:00
|
|
|
|
2020-02-14 15:19:30 -05:00
|
|
|
|
|
|
|
@app.route('/', methods=['GET', 'POST'])
|
|
|
|
def index():
|
2020-02-17 11:43:26 -05:00
|
|
|
# display results
|
2020-02-17 16:21:18 -05:00
|
|
|
studies = db.session.query(Study).order_by(Study.DATE_MODIFIED.desc()).all()
|
2020-02-17 11:43:26 -05:00
|
|
|
table = StudyTable(studies)
|
|
|
|
return render_template('index.html', table=table)
|
2020-02-14 15:19:30 -05:00
|
|
|
|
|
|
|
|
|
|
|
@app.route('/new_study', methods=['GET', 'POST'])
|
|
|
|
def new_study():
|
|
|
|
form = StudyForm(request.form)
|
2020-02-17 11:43:26 -05:00
|
|
|
action = "/new_study"
|
2020-02-17 13:25:14 -05:00
|
|
|
title = "New Study"
|
2020-02-14 15:19:30 -05:00
|
|
|
if request.method == 'POST':
|
|
|
|
study = Study()
|
2020-02-17 16:21:18 -05:00
|
|
|
study.study_details = StudyDetails()
|
2020-02-17 11:43:26 -05:00
|
|
|
_update_study(study, form)
|
|
|
|
flash('Study created successfully!')
|
2020-02-14 15:19:30 -05:00
|
|
|
return redirect('/')
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
return render_template('form.html', form=form,
|
|
|
|
action=action,
|
|
|
|
title=title,
|
|
|
|
description_map=description_map)
|
2020-02-14 15:19:30 -05:00
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
@app.route('/study/<study_id>', methods=['GET', 'POST'])
|
2020-02-17 11:43:26 -05:00
|
|
|
def edit_study(study_id):
|
2020-02-17 16:21:18 -05:00
|
|
|
study = db.session.query(Study).filter(Study.STUDYID == study_id).first()
|
2020-02-17 11:43:26 -05:00
|
|
|
form = StudyForm(request.form, obj=study)
|
|
|
|
if request.method == 'GET':
|
|
|
|
action = "/study/" + study_id
|
2020-02-17 13:25:14 -05:00
|
|
|
title = "Edit Study #" + study_id
|
2020-02-17 11:43:26 -05:00
|
|
|
if study.requirements:
|
2020-02-17 16:21:18 -05:00
|
|
|
form.requirements.data = list(map(lambda r: r.AUXDOCID, list(study.requirements)))
|
|
|
|
if study.Q_COMPLETE:
|
|
|
|
form.Q_COMPLETE.checked = True
|
2020-02-17 11:43:26 -05:00
|
|
|
if request.method == 'POST':
|
|
|
|
_update_study(study, form)
|
|
|
|
flash('Study updated successfully!')
|
|
|
|
return redirect('/')
|
2020-02-17 16:21:18 -05:00
|
|
|
return render_template('form.html', form=form,
|
|
|
|
action=action,
|
|
|
|
title=title,
|
|
|
|
description_map={})
|
2020-02-17 13:25:14 -05:00
|
|
|
|
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
@app.route('/investigator/<study_id>', methods=['GET', 'POST'])
|
2020-02-17 13:25:14 -05:00
|
|
|
def new_investigator(study_id):
|
|
|
|
form = InvestigatorForm(request.form)
|
|
|
|
action = "/investigator/" + study_id
|
|
|
|
title = "Add Investigator to Study " + study_id
|
|
|
|
if request.method == 'POST':
|
2020-02-17 16:21:18 -05:00
|
|
|
investigator = Investigator(STUDYID=study_id)
|
|
|
|
investigator.NETBADGEID = form.NETBADGEID.data
|
|
|
|
investigator.set_type(form.INVESTIGATORTYPE.data)
|
2020-02-17 13:25:14 -05:00
|
|
|
db.session.add(investigator)
|
|
|
|
db.session.commit()
|
|
|
|
flash('Investigator created successfully!')
|
|
|
|
return redirect('/')
|
|
|
|
|
2020-02-17 16:21:18 -05:00
|
|
|
return render_template('form.html', form=form,
|
|
|
|
action=action,
|
|
|
|
title=title,
|
|
|
|
description_map={})
|
2020-02-17 13:25:14 -05:00
|
|
|
|
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
@app.route('/del_investigator/<inv_id>', methods=['GET'])
|
2020-02-17 13:25:14 -05:00
|
|
|
def del_investigator(inv_id):
|
|
|
|
db.session.query(Investigator).filter(Investigator.id == inv_id).delete()
|
|
|
|
db.session.commit()
|
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
@app.route('/del_study/<study_id>', methods=['GET'])
|
2020-02-17 13:25:14 -05:00
|
|
|
def del_study(study_id):
|
2020-03-24 14:32:59 -04:00
|
|
|
db.session.query(RequiredDocument).filter(RequiredDocument.STUDYID == study_id).delete()
|
|
|
|
db.session.query(Investigator).filter(Investigator.STUDYID == study_id).delete()
|
2020-02-18 09:25:29 -05:00
|
|
|
db.session.query(StudyDetails).filter(StudyDetails.STUDYID == study_id).delete()
|
2020-03-24 14:32:59 -04:00
|
|
|
db.session.query(Study).filter(Study.STUDYID == study_id).delete()
|
2020-02-17 13:25:14 -05:00
|
|
|
db.session.commit()
|
|
|
|
return redirect('/')
|
2020-02-17 11:43:26 -05:00
|
|
|
|
|
|
|
|
|
|
|
def _update_study(study, form):
|
2020-02-17 16:21:18 -05:00
|
|
|
if study.STUDYID:
|
|
|
|
db.session.query(RequiredDocument).filter(RequiredDocument.STUDYID == study.STUDYID).delete()
|
2020-02-28 11:16:12 -05:00
|
|
|
|
|
|
|
study.STUDYID = form.STUDYID.data
|
2020-02-17 16:21:18 -05:00
|
|
|
study.TITLE = form.TITLE.data
|
|
|
|
study.NETBADGEID = form.NETBADGEID.data
|
|
|
|
study.DATE_MODIFIED = datetime.datetime.now()
|
|
|
|
study.Q_COMPLETE = form.Q_COMPLETE.data
|
|
|
|
study.HSRNUMBER = form.HSRNUMBER.data
|
2020-02-28 11:16:12 -05:00
|
|
|
|
|
|
|
for r in form.requirements:
|
|
|
|
if r.checked:
|
|
|
|
requirement = RequiredDocument(AUXDOCID=r.data, AUXDOC=r.label.text, study=study)
|
|
|
|
db.session.add(requirement)
|
|
|
|
|
2020-02-17 11:43:26 -05:00
|
|
|
db.session.add(study)
|
|
|
|
db.session.commit()
|
|
|
|
|
2020-02-18 09:25:29 -05:00
|
|
|
@app.route('/study_details/<study_id>', methods=['GET', 'POST'])
|
2020-02-17 16:21:18 -05:00
|
|
|
def study_details(study_id):
|
|
|
|
study_details = db.session.query(StudyDetails).filter(StudyDetails.STUDYID == study_id).first()
|
|
|
|
if not study_details:
|
|
|
|
study_details = StudyDetails(STUDYID=study_id)
|
|
|
|
form = StudyDetailsForm(request.form, obj=study_details)
|
|
|
|
if request.method == 'GET':
|
|
|
|
action = "/study_details/" + study_id
|
|
|
|
title = "Edit Study Details for Study #" + study_id
|
|
|
|
details = "Numeric fields can be 1 for true, 0 or false, or Null if not applicable."
|
|
|
|
if request.method == 'POST':
|
|
|
|
form.populate_obj(study_details)
|
|
|
|
db.session.add(study_details)
|
|
|
|
db.session.commit()
|
|
|
|
flash('Study updated successfully!')
|
|
|
|
return redirect('/')
|
|
|
|
return render_template('form.html', form=form,
|
|
|
|
action=action,
|
|
|
|
title=title,
|
|
|
|
details=details,
|
|
|
|
description_map=description_map)
|
2020-02-14 15:19:30 -05:00
|
|
|
|
2020-02-17 13:25:14 -05:00
|
|
|
|
2020-01-08 16:05:16 -05:00
|
|
|
if __name__ == '__main__':
|
|
|
|
# run our standalone gevent server
|
2020-02-26 12:17:37 -05:00
|
|
|
app.run(port=4200)
|