Another update
This commit is contained in:
parent
05491790df
commit
80711cd918
1
.gitignore
vendored
1
.gitignore
vendored
@ -7,3 +7,4 @@ doc/_build/
|
|||||||
doc/output/
|
doc/output/
|
||||||
api_test.py
|
api_test.py
|
||||||
log/*
|
log/*
|
||||||
|
synthea_output/
|
||||||
|
@ -4,4 +4,6 @@ django_js_reverse
|
|||||||
python-decouple
|
python-decouple
|
||||||
django-cryptography
|
django-cryptography
|
||||||
djangorestframework
|
djangorestframework
|
||||||
|
drf_yasg2
|
||||||
pandas
|
pandas
|
||||||
|
|
||||||
|
@ -9,58 +9,58 @@ class ApiConfig(AppConfig):
|
|||||||
verbose_name = _('API')
|
verbose_name = _('API')
|
||||||
verbose_name_plural = _('APIs')
|
verbose_name_plural = _('APIs')
|
||||||
|
|
||||||
try:
|
# try:
|
||||||
assert settings.SWAGGER_SETTINGS
|
# assert settings.SWAGGER_SETTINGS
|
||||||
except AttributeError:
|
# except AttributeError:
|
||||||
# We only load this setting, if it is not available in the overall settings.py file
|
# # We only load this setting, if it is not available in the overall settings.py file
|
||||||
settings.SWAGGER_SETTINGS = {
|
# settings.SWAGGER_SETTINGS = {
|
||||||
'SECURITY_DEFINITIONS': {
|
# 'SECURITY_DEFINITIONS': {
|
||||||
'Hawk': {
|
# 'Hawk': {
|
||||||
'type': 'apiKey',
|
# 'type': 'apiKey',
|
||||||
'description': 'HTTP Holder-Of-Key Authentication Scheme, https://github.com/hapijs/hawk, https://hawkrest.readthedocs.io/en/latest/<br /><strong>Ex header:</strong><br />\'Authorization\': \'Hawk mac="F4+S9cu7yZiZEgdtqzMpOOdudvqcV2V2Yzk2WcphECc=", hash="+7fKUX+djeQolvnLTxr0X47e//UHKbkRlajwMw3tx3w=", id="7FI5JET4", ts="1592905433", nonce="DlV-fL"\'',
|
# 'description': 'HTTP Holder-Of-Key Authentication Scheme, https://github.com/hapijs/hawk, https://hawkrest.readthedocs.io/en/latest/<br /><strong>Ex header:</strong><br />\'Authorization\': \'Hawk mac="F4+S9cu7yZiZEgdtqzMpOOdudvqcV2V2Yzk2WcphECc=", hash="+7fKUX+djeQolvnLTxr0X47e//UHKbkRlajwMw3tx3w=", id="7FI5JET4", ts="1592905433", nonce="DlV-fL"\'',
|
||||||
'name': 'Authorization',
|
# 'name': 'Authorization',
|
||||||
'in': 'header'
|
# 'in': 'header'
|
||||||
}
|
# }
|
||||||
}
|
# }
|
||||||
}
|
# }
|
||||||
|
|
||||||
try:
|
# try:
|
||||||
assert settings.REST_FRAMEWORK
|
# assert settings.REST_FRAMEWORK
|
||||||
except AttributeError:
|
# except AttributeError:
|
||||||
# We only load this setting, if it is not available in the overall settings.py file
|
# # We only load this setting, if it is not available in the overall settings.py file
|
||||||
# To protect all API views with Hawk by default, put this in your settings:
|
# # To protect all API views with Hawk by default, put this in your settings:
|
||||||
# https://hawkrest.readthedocs.io/en/latest/usage.html#protecting-api-views-with-hawk
|
# # https://hawkrest.readthedocs.io/en/latest/usage.html#protecting-api-views-with-hawk
|
||||||
settings.REST_FRAMEWORK = {
|
# settings.REST_FRAMEWORK = {
|
||||||
|
|
||||||
'DEFAULT_AUTHENTICATION_CLASSES': (
|
|
||||||
'apps.api.authentication.APIHawk',
|
|
||||||
),
|
|
||||||
|
|
||||||
'DEFAULT_PERMISSION_CLASSES': (
|
|
||||||
'rest_framework.permissions.IsAuthenticated',
|
|
||||||
),
|
|
||||||
|
|
||||||
# 'DEFAULT_AUTHENTICATION_CLASSES': (
|
# 'DEFAULT_AUTHENTICATION_CLASSES': (
|
||||||
# 'rest_framework.authentication.TokenAuthentication',
|
# 'apps.api.authentication.APIHawk',
|
||||||
# ),
|
# ),
|
||||||
|
|
||||||
# 'DEFAULT_PERMISSION_CLASSES': (
|
# 'DEFAULT_PERMISSION_CLASSES': (
|
||||||
# 'rest_framework.permissions.IsAuthenticated', ),
|
# 'rest_framework.permissions.IsAuthenticated',
|
||||||
|
# ),
|
||||||
|
|
||||||
# Use Django's standard `django.contrib.auth` permissions,
|
# # 'DEFAULT_AUTHENTICATION_CLASSES': (
|
||||||
# or allow read-only access for unauthenticated users.
|
# # 'rest_framework.authentication.TokenAuthentication',
|
||||||
#'DEFAULT_PERMISSION_CLASSES': [
|
# # ),
|
||||||
# 'rest_framework.permissions.DjangoModelPermissionsOrAnonReadOnly'
|
|
||||||
#],
|
|
||||||
'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination',
|
|
||||||
'PAGE_SIZE': 10
|
|
||||||
}
|
|
||||||
|
|
||||||
try:
|
# # 'DEFAULT_PERMISSION_CLASSES': (
|
||||||
assert settings.HAWK_MESSAGE_EXPIRATION
|
# # 'rest_framework.permissions.IsAuthenticated', ),
|
||||||
except AttributeError:
|
|
||||||
# We only load this setting, if it is not available in the overall settings.py file
|
# # Use Django's standard `django.contrib.auth` permissions,
|
||||||
settings.HAWK_MESSAGE_EXPIRATION = 60
|
# # or allow read-only access for unauthenticated users.
|
||||||
|
# #'DEFAULT_PERMISSION_CLASSES': [
|
||||||
|
# # 'rest_framework.permissions.DjangoModelPermissionsOrAnonReadOnly'
|
||||||
|
# #],
|
||||||
|
# 'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination',
|
||||||
|
# 'PAGE_SIZE': 10
|
||||||
|
# }
|
||||||
|
|
||||||
|
# try:
|
||||||
|
# assert settings.HAWK_MESSAGE_EXPIRATION
|
||||||
|
# except AttributeError:
|
||||||
|
# # We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
# settings.HAWK_MESSAGE_EXPIRATION = 60
|
||||||
|
|
||||||
def ready(self):
|
def ready(self):
|
||||||
from . import signals
|
from . import signals
|
@ -7,24 +7,24 @@ from drf_yasg2 import openapi
|
|||||||
|
|
||||||
from . import views
|
from . import views
|
||||||
|
|
||||||
from apps.dropoff.api.views import DatadropViewSet
|
# from apps.dropoff.api.views import DatadropViewSet
|
||||||
from apps.invitation.api.views import InvitationViewSet
|
# from apps.invitation.api.views import InvitationViewSet
|
||||||
from apps.researcher.api.views import ResearcherViewSet
|
# from apps.researcher.api.views import ResearcherViewSet
|
||||||
from apps.storage.api.views import StorageEngineViewSet, StorageLocationViewSet
|
# from apps.storage.api.views import StorageEngineViewSet, StorageLocationViewSet
|
||||||
from apps.study.api.views import StudyViewSet
|
# from apps.study.api.views import StudyViewSet
|
||||||
from apps.virtual_machine.api.views import (VirtualMachineViewSet,
|
# from apps.virtual_machine.api.views import (VirtualMachineViewSet,
|
||||||
VirtualMachineOperatingSystemViewSet,
|
# VirtualMachineOperatingSystemViewSet,
|
||||||
VirtualMachineProfileViewSet,
|
# VirtualMachineProfileViewSet,
|
||||||
VirtualMachineMemoryViewSet,
|
# VirtualMachineMemoryViewSet,
|
||||||
VirtualMachineNetworkViewSet,
|
# VirtualMachineNetworkViewSet,
|
||||||
VirtualMachineStorageViewSet,
|
# VirtualMachineStorageViewSet,
|
||||||
VirtualMachineGPUViewSet)
|
# VirtualMachineGPUViewSet)
|
||||||
|
|
||||||
schema_view = get_schema_view(
|
schema_view = get_schema_view(
|
||||||
openapi.Info(
|
openapi.Info(
|
||||||
title="Virtual Research Environment API",
|
title="Synthea WebService API",
|
||||||
default_version='v1',
|
default_version='v1',
|
||||||
description="Here you can see a list of API endpoints and actions that are available to communicate with the VRE API",
|
description="Info about Synthea WebServer API",
|
||||||
terms_of_service="https://www.rug.nl",
|
terms_of_service="https://www.rug.nl",
|
||||||
contact=openapi.Contact(email="vre_team@rug.nl"),
|
contact=openapi.Contact(email="vre_team@rug.nl"),
|
||||||
license=openapi.License(name="MIT License"),
|
license=openapi.License(name="MIT License"),
|
||||||
@ -33,27 +33,27 @@ schema_view = get_schema_view(
|
|||||||
permission_classes=(permissions.AllowAny,),
|
permission_classes=(permissions.AllowAny,),
|
||||||
)
|
)
|
||||||
|
|
||||||
api_router_v1 = routers.DefaultRouter()
|
#api_router_v1 = routers.DefaultRouter()
|
||||||
|
|
||||||
api_router_v1.register(r'researchers', ResearcherViewSet)
|
# api_router_v1.register(r'researchers', ResearcherViewSet)
|
||||||
|
|
||||||
api_router_v1.register(r'studies', StudyViewSet)
|
# api_router_v1.register(r'studies', StudyViewSet)
|
||||||
|
|
||||||
api_router_v1.register(r'dropoffs', DatadropViewSet)
|
# api_router_v1.register(r'dropoffs', DatadropViewSet)
|
||||||
|
|
||||||
api_router_v1.register(r'invitations', InvitationViewSet)
|
# api_router_v1.register(r'invitations', InvitationViewSet)
|
||||||
|
|
||||||
api_router_v1.register(r'storageengines', StorageEngineViewSet)
|
# api_router_v1.register(r'storageengines', StorageEngineViewSet)
|
||||||
api_router_v1.register(r'storagelocations', StorageLocationViewSet)
|
# api_router_v1.register(r'storagelocations', StorageLocationViewSet)
|
||||||
|
|
||||||
# Order is important for virtual machines. Longest match first
|
# # Order is important for virtual machines. Longest match first
|
||||||
api_router_v1.register(r'virtualmachines/profiles', VirtualMachineProfileViewSet)
|
# api_router_v1.register(r'virtualmachines/profiles', VirtualMachineProfileViewSet)
|
||||||
api_router_v1.register(r'virtualmachines/storage', VirtualMachineStorageViewSet)
|
# api_router_v1.register(r'virtualmachines/storage', VirtualMachineStorageViewSet)
|
||||||
api_router_v1.register(r'virtualmachines/memory', VirtualMachineMemoryViewSet)
|
# api_router_v1.register(r'virtualmachines/memory', VirtualMachineMemoryViewSet)
|
||||||
api_router_v1.register(r'virtualmachines/network', VirtualMachineNetworkViewSet)
|
# api_router_v1.register(r'virtualmachines/network', VirtualMachineNetworkViewSet)
|
||||||
api_router_v1.register(r'virtualmachines/gpu', VirtualMachineGPUViewSet)
|
# api_router_v1.register(r'virtualmachines/gpu', VirtualMachineGPUViewSet)
|
||||||
api_router_v1.register(r'virtualmachines/os', VirtualMachineOperatingSystemViewSet)
|
# api_router_v1.register(r'virtualmachines/os', VirtualMachineOperatingSystemViewSet)
|
||||||
api_router_v1.register(r'virtualmachines', VirtualMachineViewSet)
|
# api_router_v1.register(r'virtualmachines', VirtualMachineViewSet)
|
||||||
|
|
||||||
# Main namespace for the API urls
|
# Main namespace for the API urls
|
||||||
app_name = 'api'
|
app_name = 'api'
|
||||||
@ -66,6 +66,10 @@ urlpatterns = [
|
|||||||
# Also this will give the full url to the OpenAPI documentation
|
# Also this will give the full url to the OpenAPI documentation
|
||||||
path('info/', views.Info.as_view(), name='info'),
|
path('info/', views.Info.as_view(), name='info'),
|
||||||
|
|
||||||
|
path('states/', views.States.as_view(), name='states'),
|
||||||
|
path('modules/', views.Modules.as_view(), name='modules'),
|
||||||
|
path('generate/', views.Generate.as_view(), name='generate'),
|
||||||
|
|
||||||
# Add extra namespace for versioning the API
|
# Add extra namespace for versioning the API
|
||||||
path('v1/', include((api_router_v1.urls,'api'),namespace='v1')),
|
#path('v1/', include((api_router_v1.urls,'api'),namespace='v1')),
|
||||||
]
|
]
|
@ -1,11 +1,23 @@
|
|||||||
from rest_framework.views import APIView
|
from rest_framework.views import APIView
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
|
from django.http import FileResponse
|
||||||
|
|
||||||
|
from django.views.decorators.csrf import csrf_exempt
|
||||||
|
|
||||||
from rest_framework.decorators import schema
|
from rest_framework.decorators import schema
|
||||||
|
from rest_framework import status
|
||||||
|
|
||||||
from django.urls import reverse
|
from django.urls import reverse
|
||||||
|
|
||||||
from lib.utils.general import get_ip_address
|
from lib.utils.general import get_ip_address
|
||||||
|
|
||||||
|
from apps.synthea.api.serializers import SyntheaSerializer
|
||||||
|
|
||||||
|
from apps.synthea.lib.utils import available_states, available_modules,run_synthea
|
||||||
|
|
||||||
|
import mimetypes
|
||||||
|
|
||||||
|
|
||||||
@schema(None)
|
@schema(None)
|
||||||
class Info(APIView):
|
class Info(APIView):
|
||||||
"""
|
"""
|
||||||
@ -50,3 +62,36 @@ class Info(APIView):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
return Response(data)
|
return Response(data)
|
||||||
|
|
||||||
|
|
||||||
|
#@schema(None)
|
||||||
|
class States(APIView):
|
||||||
|
|
||||||
|
def get(self, request, format=None):
|
||||||
|
return Response(available_states())
|
||||||
|
|
||||||
|
|
||||||
|
class Modules(APIView):
|
||||||
|
|
||||||
|
def get(self, request, format=None):
|
||||||
|
return Response(available_modules())
|
||||||
|
|
||||||
|
class Generate(APIView):
|
||||||
|
|
||||||
|
def post(self, request, format=None):
|
||||||
|
print('Post data')
|
||||||
|
print(request.data)
|
||||||
|
|
||||||
|
serializer = SyntheaSerializer(data=request.data)
|
||||||
|
if serializer.is_valid():
|
||||||
|
synthea = serializer.save()
|
||||||
|
zipfile = synthea.generate()
|
||||||
|
print(zipfile)
|
||||||
|
response = FileResponse(zipfile.open('rb'), content_type='application/zip')
|
||||||
|
response['Content-Disposition'] = f'attachment; filename={zipfile.name}'
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
|
#return Response(serializer.data, status=status.HTTP_201_CREATED)
|
||||||
|
return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
|
8
webservice/apps/synthea/api/serializers.py
Normal file
8
webservice/apps/synthea/api/serializers.py
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from apps.synthea.models import Synthea
|
||||||
|
|
||||||
|
class SyntheaSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = Synthea
|
||||||
|
fields = ['state', 'population', 'gender', 'age', 'module']
|
@ -8,3 +8,42 @@ class SyntheaConfig(AppConfig):
|
|||||||
label = 'synthea'
|
label = 'synthea'
|
||||||
verbose_name = _('Synthea')
|
verbose_name = _('Synthea')
|
||||||
verbose_name_plural = _('Synthea')
|
verbose_name_plural = _('Synthea')
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_BASE_DIR
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_BASE_DIR = settings.BASE_DIR / '../synthea'
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_OUTPUT_DIR
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_OUTPUT_DIR = settings.BASE_DIR / '../synthea_output'
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_MODULE_DIR
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_MODULE_DIR = settings.SYNTHEA_BASE_DIR / 'src/main/resources/modules/'
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_RESOURCE_DIR
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_RESOURCE_DIR = settings.SYNTHEA_BASE_DIR / 'src/main/resources/'
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_STATES_DIR
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_STATES_DIR = settings.SYNTHEA_BASE_DIR / 'src/main/resources/geography/'
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
try:
|
||||||
|
assert settings.SYNTHEA_EXPORT_TYPE
|
||||||
|
except AttributeError:
|
||||||
|
# We only load this setting, if it is not available in the overall settings.py file
|
||||||
|
settings.SYNTHEA_EXPORT_TYPE = 'fhir_stu3'
|
@ -1,3 +1,5 @@
|
|||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
from django.forms import ModelForm
|
from django.forms import ModelForm
|
||||||
from django import forms
|
from django import forms
|
||||||
|
|
||||||
@ -12,16 +14,16 @@ class SyntheaForm(ModelForm):
|
|||||||
fields = ['state', 'population', 'gender', 'age', 'module']
|
fields = ['state', 'population', 'gender', 'age', 'module']
|
||||||
|
|
||||||
# This is loaded only once during startup. So changing the state data will not be picked up after a restart
|
# This is loaded only once during startup. So changing the state data will not be picked up after a restart
|
||||||
state_options = [('','Any')]
|
state_options = [('',_('Select'))]
|
||||||
for item in available_states():
|
for item in available_states():
|
||||||
state_options.append((item,item))
|
state_options.append((item['id'],item['name']))
|
||||||
|
|
||||||
module_options = [('','Any')]
|
module_options = [('',_('Any'))]
|
||||||
for item in available_modules():
|
for item in available_modules():
|
||||||
module_options.append((item['module'],item['name']))
|
module_options.append((item['id'],item['name']))
|
||||||
|
|
||||||
widgets = {
|
widgets = {
|
||||||
'state': forms.Select(choices=state_options),
|
'state': forms.Select(choices=state_options),
|
||||||
'gender': forms.Select(choices=[('','Any'),('m','Male'),('f','Female')]),
|
'gender': forms.Select(choices=[('',_('Any')),('m',_('Male')),('f',_('Female'))]),
|
||||||
'module': forms.Select(choices=module_options)
|
'module': forms.Select(choices=module_options)
|
||||||
}
|
}
|
@ -5,36 +5,41 @@ import subprocess
|
|||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
import json
|
import json
|
||||||
|
|
||||||
def available_states():
|
from django.conf import settings
|
||||||
#TODO: Make a setting for this path
|
from uuid import uuid4
|
||||||
location = Path('/opt/development/synthea_webservice/synthea/src/main/resources/geography/')
|
|
||||||
|
|
||||||
df = pd.read_csv(location / 'timezones.csv', index_col=False)
|
|
||||||
# The state information is expected in the first column
|
def available_states():
|
||||||
states = df[df.columns[0]].to_list()
|
states = []
|
||||||
states.sort()
|
# Read the timezones.csv file from the Synthea resources. This should give us all the 'state' on the first column
|
||||||
|
df = pd.read_csv(settings.SYNTHEA_STATES_DIR / 'timezones.csv', index_col=False)
|
||||||
|
for state in df[df.columns[0]].to_list():
|
||||||
|
states.append({'id' : state , 'name' : state})
|
||||||
|
|
||||||
|
#states = df[df.columns[0]].to_list()
|
||||||
|
# Sort on name
|
||||||
|
states = sorted(states, key=lambda k: k['name'].lower())
|
||||||
|
#states.sort()
|
||||||
return states
|
return states
|
||||||
|
|
||||||
def available_modules():
|
def available_modules():
|
||||||
#TODO: Make a setting for this path
|
# Assumption here: Only .json files in the main folder are modules. The rest are submodules...
|
||||||
location = Path('/opt/development/synthea_webservice/synthea/src/main/resources/modules/')
|
|
||||||
|
|
||||||
# Assumption here: A folder is a single module. And all .json in the main modules folder is a module.
|
|
||||||
modules = []
|
modules = []
|
||||||
for module in location.iterdir():
|
for module in settings.SYNTHEA_MODULE_DIR.iterdir():
|
||||||
if module.is_file() and module.suffix == '.json':
|
if module.is_file() and module.suffix == '.json':
|
||||||
data = json.loads(module.read_text())
|
data = json.loads(module.read_text())
|
||||||
modules.append({'module' : module.name.replace('.json',''), 'name' : data['name']})
|
modules.append({'id' : module.name.replace('.json',''), 'name' : data['name']})
|
||||||
|
|
||||||
modules = sorted(modules, key=lambda k: k['name'].lower())
|
modules = sorted(modules, key=lambda k: k['name'].lower())
|
||||||
return modules
|
return modules
|
||||||
|
|
||||||
def run_synthea(state = None, population = None, gender = None, age = None, module = None):
|
def run_synthea(state = None, population = None, gender = None, age = None, module = None):
|
||||||
# TODO: Make synthea setting(s)
|
# Add a unique dir to the output, so multiple Synthea processes can run parallel
|
||||||
location = '/opt/development/synthea_webservice/synthea/'
|
temp_id = uuid4().hex
|
||||||
synthea_cmd = ['/opt/development/synthea_webservice/synthea/run_synthea']
|
output_folder = settings.SYNTHEA_OUTPUT_DIR / temp_id
|
||||||
|
|
||||||
|
synthea_cmd = [settings.SYNTHEA_BASE_DIR / 'run_synthea','--exporter.baseDirectory',output_folder]
|
||||||
zip_file = 'Synthea_'
|
zip_file = 'Synthea_'
|
||||||
zip_export = location
|
|
||||||
|
|
||||||
if population:
|
if population:
|
||||||
synthea_cmd.append('-p')
|
synthea_cmd.append('-p')
|
||||||
@ -62,7 +67,7 @@ def run_synthea(state = None, population = None, gender = None, age = None, modu
|
|||||||
|
|
||||||
process_ok = False
|
process_ok = False
|
||||||
log = ''
|
log = ''
|
||||||
with subprocess.Popen(synthea_cmd,cwd=location, stdout=subprocess.PIPE,stderr=subprocess.PIPE) as process:
|
with subprocess.Popen(synthea_cmd,cwd=settings.SYNTHEA_BASE_DIR, stdout=subprocess.PIPE,stderr=subprocess.PIPE) as process:
|
||||||
for line in process.stdout:
|
for line in process.stdout:
|
||||||
line = line.decode('utf8')
|
line = line.decode('utf8')
|
||||||
log += line
|
log += line
|
||||||
@ -70,14 +75,10 @@ def run_synthea(state = None, population = None, gender = None, age = None, modu
|
|||||||
process_ok = line.find('BUILD SUCCESSFUL') >= 0
|
process_ok = line.find('BUILD SUCCESSFUL') >= 0
|
||||||
|
|
||||||
if process_ok:
|
if process_ok:
|
||||||
with ZipFile(f'{zip_export}/{zip_file}.zip', 'w') as export:
|
with ZipFile(f'{output_folder}/{zip_file}_{temp_id}.zip', 'w') as export:
|
||||||
for file in Path(location + 'output/fhir_stu3').iterdir():
|
for file in (output_folder / settings.SYNTHEA_EXPORT_TYPE).iterdir():
|
||||||
export.write(file,file.name)
|
export.write(file,file.name)
|
||||||
|
|
||||||
return Path(f'{zip_export}/{zip_file}.zip')
|
return (log,Path(f'{output_folder}/{zip_file}_{temp_id}.zip'))
|
||||||
else:
|
else:
|
||||||
raise Exception(log)
|
raise Exception(log)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -0,0 +1,38 @@
|
|||||||
|
# Generated by Django 3.1.3 on 2020-11-16 13:57
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('synthea', '0001_initial'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='synthea',
|
||||||
|
name='log',
|
||||||
|
field=models.TextField(blank=True, help_text='Synthea logfile output', verbose_name='Log'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='synthea',
|
||||||
|
name='age',
|
||||||
|
field=models.CharField(blank=True, default='18-100', help_text='Select the age range. Enter [min age]-[max age]', max_length=10, verbose_name='Age range'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='synthea',
|
||||||
|
name='gender',
|
||||||
|
field=models.CharField(blank=True, help_text='Select the gender type', max_length=1, verbose_name='Gender'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='synthea',
|
||||||
|
name='module',
|
||||||
|
field=models.CharField(blank=True, help_text='Select the module', max_length=50, verbose_name='Module'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='synthea',
|
||||||
|
name='state',
|
||||||
|
field=models.CharField(help_text='The state for which synthea generate data.', max_length=200, verbose_name='State'),
|
||||||
|
),
|
||||||
|
]
|
@ -7,6 +7,8 @@ from django_cryptography.fields import encrypt
|
|||||||
from lib.utils.general import get_random_string
|
from lib.utils.general import get_random_string
|
||||||
from lib.models.base import MetaDataModel
|
from lib.models.base import MetaDataModel
|
||||||
|
|
||||||
|
from .lib.utils import run_synthea
|
||||||
|
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
# Create your models here.
|
# Create your models here.
|
||||||
@ -18,7 +20,25 @@ class Synthea(MetaDataModel):
|
|||||||
|
|
||||||
id = models.UUIDField(_('ID'), primary_key=True, unique=True, default=uuid.uuid4, editable=False, help_text=_('A unique id'))
|
id = models.UUIDField(_('ID'), primary_key=True, unique=True, default=uuid.uuid4, editable=False, help_text=_('A unique id'))
|
||||||
state = models.CharField(_('State'), max_length=200, help_text=_('The state for which synthea generate data.'))
|
state = models.CharField(_('State'), max_length=200, help_text=_('The state for which synthea generate data.'))
|
||||||
population = models.PositiveSmallIntegerField(_('Population'), blank=True, default=50, help_text=_('The size of the population'))
|
population = models.PositiveSmallIntegerField(_('Population'), default=50, help_text=_('The size of the population'))
|
||||||
gender = models.CharField(_('Gender'), blank=True,max_length=1, help_text=_('Select the gender type'))
|
gender = models.CharField(_('Gender'), blank=True,max_length=1, help_text=_('Select the gender type'))
|
||||||
age = models.CharField(_('Age range'), blank=True,max_length=10, help_text=_('Select the age range'))
|
age = models.CharField(_('Age range'), blank=True,default='18-100', max_length=10, help_text=_('Select the age range. Enter [min age]-[max age]'))
|
||||||
module = models.CharField(_('Module'),blank=True, max_length=50, help_text=_('Select the module'))
|
module = models.CharField(_('Module'),blank=True, max_length=50, help_text=_('Select the module'))
|
||||||
|
log = models.TextField(_('Log'),blank=True, help_text=_('Synthea logfile output'))
|
||||||
|
|
||||||
|
|
||||||
|
def generate(self):
|
||||||
|
log,zip_file = run_synthea(
|
||||||
|
self.state,
|
||||||
|
self.population,
|
||||||
|
self.gender,
|
||||||
|
self.age,
|
||||||
|
self.module
|
||||||
|
)
|
||||||
|
|
||||||
|
self.log = log
|
||||||
|
|
||||||
|
self.save()
|
||||||
|
|
||||||
|
return zip_file
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@
|
|||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1>Synthea generartor</h1>
|
<h1>Synthea Generator</h1>
|
||||||
<p>Enter the form fields and press submit. <small>U vraagt, wij draaien ;)</small></p>
|
<p>Enter the form fields and press submit. <small>U vraagt, wij draaien ;)</small></p>
|
||||||
<form method="POST" class="post-form">
|
<form method="POST" class="post-form">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
|
@ -11,4 +11,5 @@
|
|||||||
{% block content %}
|
{% block content %}
|
||||||
<h1>dHealt Nederland</h1>
|
<h1>dHealt Nederland</h1>
|
||||||
<p>Onder leiding van UMCG</p>
|
<p>Onder leiding van UMCG</p>
|
||||||
|
<p>Github branches: https://github.com/dHealthNL </p>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
from django.shortcuts import render
|
from django.shortcuts import render
|
||||||
from django.http import HttpResponse
|
from django.http import HttpResponse
|
||||||
from apps.synthea.forms import SyntheaForm
|
from apps.synthea.forms import SyntheaForm
|
||||||
|
from apps.synthea.models import Synthea
|
||||||
|
|
||||||
from .lib.utils import run_synthea
|
from .lib.utils import run_synthea
|
||||||
|
|
||||||
@ -21,25 +22,14 @@ def show_synthea_form(request):
|
|||||||
# check whether it's valid:
|
# check whether it's valid:
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
# process the data in form.cleaned_data as required
|
# process the data in form.cleaned_data as required
|
||||||
|
synthea = form.save()
|
||||||
|
zipfile = synthea.generate()
|
||||||
|
|
||||||
try:
|
response = HttpResponse(zipfile.open('rb'), content_type='application/zip')
|
||||||
zipfile = run_synthea(
|
|
||||||
form.cleaned_data['state'],
|
|
||||||
form.cleaned_data['population'],
|
|
||||||
form.cleaned_data['gender'],
|
|
||||||
form.cleaned_data['age'],
|
|
||||||
form.cleaned_data['module']
|
|
||||||
)
|
|
||||||
|
|
||||||
mime_type, _ = mimetypes.guess_type(zipfile)
|
|
||||||
response = HttpResponse(zipfile.open('rb'), content_type=mime_type)
|
|
||||||
response['Content-Disposition'] = f'attachment; filename={zipfile.name}'
|
response['Content-Disposition'] = f'attachment; filename={zipfile.name}'
|
||||||
|
|
||||||
return response
|
return response
|
||||||
|
|
||||||
except Exception as ex:
|
|
||||||
print(ex)
|
|
||||||
|
|
||||||
# if a GET (or any other method) we'll create a blank form
|
# if a GET (or any other method) we'll create a blank form
|
||||||
else:
|
else:
|
||||||
form = SyntheaForm()
|
form = SyntheaForm()
|
||||||
|
@ -41,6 +41,8 @@ INSTALLED_APPS = [
|
|||||||
'django.contrib.staticfiles',
|
'django.contrib.staticfiles',
|
||||||
|
|
||||||
'django_js_reverse',
|
'django_js_reverse',
|
||||||
|
'rest_framework',
|
||||||
|
'drf_yasg2',
|
||||||
|
|
||||||
'apps.api',
|
'apps.api',
|
||||||
'apps.RUG_template',
|
'apps.RUG_template',
|
||||||
|
@ -21,6 +21,9 @@ from django_js_reverse.views import urls_js
|
|||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path('admin/', admin.site.urls),
|
path('admin/', admin.site.urls),
|
||||||
|
|
||||||
|
# Add API urls for all the known models
|
||||||
|
path('api/', include('apps.api.urls')),
|
||||||
|
|
||||||
path('', include('apps.synthea.urls')),
|
path('', include('apps.synthea.urls')),
|
||||||
|
|
||||||
# Add Default RUG HTML homepage
|
# Add Default RUG HTML homepage
|
||||||
|
Loading…
Reference in New Issue
Block a user