endoreg-db 0.3.3__py3-none-any.whl → 0.3.5__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- endoreg_db/data/__init__.py +50 -1
- endoreg_db/data/case_template/rule/00_patient_lab_sample_add_default_value.yaml +167 -0
- endoreg_db/data/case_template/rule/01_patient-set-age.yaml +8 -0
- endoreg_db/data/case_template/rule/01_patient-set-gender.yaml +9 -0
- endoreg_db/data/case_template/rule/11_create_patient_lab_sample.yaml +23 -0
- endoreg_db/data/case_template/rule/12_create-patient_medication-anticoagulation.yaml +19 -0
- endoreg_db/data/case_template/rule/13_create-patient_medication_schedule-anticoagulation.yaml +19 -0
- endoreg_db/data/case_template/rule/19_create_patient.yaml +17 -0
- endoreg_db/data/case_template/rule_type/base_types.yaml +35 -0
- endoreg_db/data/case_template/rule_value_type/base_types.yaml +59 -0
- endoreg_db/data/case_template/template/base.yaml +8 -0
- endoreg_db/data/case_template/template_type/pre_endoscopy.yaml +3 -0
- endoreg_db/data/case_template/tmp/_rule_value +13 -0
- endoreg_db/data/case_template/tmp/rule/01_atrial_fibrillation.yaml +21 -0
- endoreg_db/data/case_template/tmp/rule/02_create_object.yaml +10 -0
- endoreg_db/data/case_template/tmp/template/atrial_fibrillation_low_risk.yaml +7 -0
- endoreg_db/data/center/data.yaml +8 -0
- endoreg_db/data/center_resource/green_endoscopy_dashboard_CenterResource.yaml +144 -0
- endoreg_db/data/center_waste/green_endoscopy_dashboard_CenterWaste.yaml +48 -0
- endoreg_db/data/disease/cardiovascular.yaml +37 -0
- endoreg_db/data/disease/hepatology.yaml +5 -0
- endoreg_db/data/disease/misc.yaml +6 -0
- endoreg_db/data/disease/renal.yaml +5 -0
- endoreg_db/data/disease_classification/chronic_kidney_disease.yaml +6 -0
- endoreg_db/data/disease_classification/coronary_vessel_disease.yaml +6 -0
- endoreg_db/data/disease_classification_choice/chronic_kidney_disease.yaml +41 -0
- endoreg_db/data/disease_classification_choice/coronary_vessel_disease.yaml +20 -0
- endoreg_db/data/distribution/date/patient.yaml +7 -0
- endoreg_db/data/distribution/single_categorical/patient.yaml +7 -0
- endoreg_db/data/emission_factor/green_endoscopy_dashboard_EmissionFactor.yaml +132 -0
- endoreg_db/data/event/cardiology.yaml +28 -0
- endoreg_db/data/event/neurology.yaml +14 -0
- endoreg_db/data/event/surgery.yaml +13 -0
- endoreg_db/data/event/thrombembolism.yaml +20 -0
- endoreg_db/data/examination/examinations/data.yaml +49 -0
- endoreg_db/data/gender/data.yaml +18 -0
- endoreg_db/data/information_source/medication.yaml +6 -0
- endoreg_db/data/lab_value/cardiac_enzymes.yaml +31 -0
- endoreg_db/data/lab_value/coagulation.yaml +49 -0
- endoreg_db/data/lab_value/electrolytes.yaml +190 -0
- endoreg_db/data/lab_value/gastrointestinal_function.yaml +121 -0
- endoreg_db/data/lab_value/hematology.yaml +169 -0
- endoreg_db/data/lab_value/hormones.yaml +53 -0
- endoreg_db/data/lab_value/lipids.yaml +44 -0
- endoreg_db/data/lab_value/misc.yaml +30 -0
- endoreg_db/data/lab_value/renal_function.yaml +11 -0
- endoreg_db/data/material/material.yaml +91 -0
- endoreg_db/data/medication/anticoagulation.yaml +65 -0
- endoreg_db/data/medication/tah.yaml +70 -0
- endoreg_db/data/medication_indication/anticoagulation.yaml +120 -0
- endoreg_db/data/medication_indication_type/data.yaml +11 -0
- endoreg_db/data/medication_indication_type/thrombembolism.yaml +41 -0
- endoreg_db/data/medication_intake_time/base.yaml +31 -0
- endoreg_db/data/medication_schedule/apixaban.yaml +95 -0
- endoreg_db/data/medication_schedule/ass.yaml +12 -0
- endoreg_db/data/medication_schedule/enoxaparin.yaml +26 -0
- endoreg_db/data/patient_lab_sample_type/generic.yaml +6 -0
- endoreg_db/data/product/green_endoscopy_dashboard_Product.yaml +66 -0
- endoreg_db/data/product_group/green_endoscopy_dashboard_ProductGroup.yaml +33 -0
- endoreg_db/data/product_material/green_endoscopy_dashboard_ProductMaterial.yaml +308 -0
- endoreg_db/data/product_weight/green_endoscopy_dashboard_ProductWeight.yaml +88 -0
- endoreg_db/data/reference_product/green_endoscopy_dashboard_ReferenceProduct.yaml +55 -0
- endoreg_db/data/resource/green_endoscopy_dashboard_Resource.yaml +15 -0
- endoreg_db/data/tmp/chronic_kidney_disease.yaml +0 -0
- endoreg_db/data/tmp/congestive_heart_failure.yaml +0 -0
- endoreg_db/data/transport_route/green_endoscopy_dashboard_TransportRoute.yaml +12 -0
- endoreg_db/data/unit/concentration.yaml +92 -0
- endoreg_db/data/unit/length.yaml +4 -4
- endoreg_db/data/unit/misc.yaml +20 -0
- endoreg_db/data/unit/rate.yaml +6 -0
- endoreg_db/data/unit/time.yaml +13 -0
- endoreg_db/data/unit/volume.yaml +13 -4
- endoreg_db/data/unit/weight.yaml +11 -4
- endoreg_db/data/waste/data.yaml +12 -0
- endoreg_db/forms/__init__.py +2 -1
- endoreg_db/forms/questionnaires/__init__.py +1 -0
- endoreg_db/forms/questionnaires/tto_questionnaire.py +23 -0
- endoreg_db/management/commands/load_base_db_data.py +58 -1
- endoreg_db/management/commands/load_disease_classification_choices_data.py +41 -0
- endoreg_db/management/commands/load_disease_classification_data.py +41 -0
- endoreg_db/management/commands/load_disease_data.py +40 -0
- endoreg_db/management/commands/load_distribution_data.py +66 -0
- endoreg_db/management/commands/load_event_data.py +41 -0
- endoreg_db/management/commands/load_g_play_data.py +113 -0
- endoreg_db/management/commands/load_gender_data.py +44 -0
- endoreg_db/management/commands/load_green_endoscopy_wuerzburg_data.py +133 -0
- endoreg_db/management/commands/load_lab_value_data.py +50 -0
- endoreg_db/management/commands/load_medication_data.py +41 -0
- endoreg_db/management/commands/load_medication_indication_data.py +63 -0
- endoreg_db/management/commands/load_medication_indication_type_data.py +41 -0
- endoreg_db/management/commands/load_medication_intake_time_data.py +41 -0
- endoreg_db/management/commands/load_medication_schedule_data.py +55 -0
- endoreg_db/migrations/0023_ttoquestionnaire_alter_pdftype_endoscope_info_line.py +59 -0
- endoreg_db/migrations/0024_remove_ttoquestionnaire_infections_and_more.py +27 -0
- endoreg_db/migrations/0025_event_alter_rawpdffile_file_patientevent.py +42 -0
- endoreg_db/migrations/0026_disease_diseaseclassification_and_more.py +166 -0
- endoreg_db/migrations/0027_labvalue_abbreviation_labvalue_default_normal_range_and_more.py +38 -0
- endoreg_db/migrations/0028_alter_unit_abbreviation.py +18 -0
- endoreg_db/migrations/0029_medicationintaketime_and_more.py +75 -0
- endoreg_db/migrations/0030_medicationindicationtype_and_more.py +101 -0
- endoreg_db/migrations/0031_rename_adapt_to_liver_function_medication_adapt_to_age_and_more.py +38 -0
- endoreg_db/migrations/0032_alter_medicationschedule_therapy_duration_d.py +18 -0
- endoreg_db/migrations/0033_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0034_alter_rawpdffile_file.py +20 -0
- endoreg_db/migrations/0035_alter_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0036_alter_rawpdffile_file.py +20 -0
- endoreg_db/migrations/0037_alter_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0038_emissionfactor_material_product_productgroup_and_more.py +164 -0
- endoreg_db/migrations/0039_referenceproduct_name.py +19 -0
- endoreg_db/migrations/0040_quizanswertype_quizquestiontype_quizquestion_and_more.py +50 -0
- endoreg_db/migrations/0041_gender_patientmedication_medication_indication_and_more.py +40 -0
- endoreg_db/migrations/0042_casetemplateruletype_casetemplaterulevalue_and_more.py +74 -0
- endoreg_db/migrations/0043_casetemplatetype_name_de_casetemplatetype_name_en.py +23 -0
- endoreg_db/migrations/0044_casetemplateruletype_name_de_and_more.py +23 -0
- endoreg_db/migrations/0045_casetemplaterulevalue_value_type.py +19 -0
- endoreg_db/migrations/0046_casetemplaterulevalue_target_field.py +18 -0
- endoreg_db/migrations/0047_casetemplaterule_target_model.py +18 -0
- endoreg_db/migrations/0048_remove_casetemplaterule_chained_rules_and_more.py +22 -0
- endoreg_db/migrations/0049_remove_casetemplaterule_rule_values.py +17 -0
- endoreg_db/migrations/0050_casetemplaterule_rule_values.py +18 -0
- endoreg_db/migrations/0051_remove_casetemplaterule_calling_rules_and_more.py +27 -0
- endoreg_db/migrations/0052_rename_case_template_type_casetemplate_template_type.py +18 -0
- endoreg_db/migrations/0053_patientlabsampletype_patientlabsample_and_more.py +38 -0
- endoreg_db/migrations/0054_multiplecategoricalvaluedistribution_and_more.py +69 -0
- endoreg_db/migrations/0055_remove_casetemplaterule_rule_values_and_more.py +59 -0
- endoreg_db/migrations/0056_datevaluedistribution_and_more.py +32 -0
- endoreg_db/migrations/0057_remove_datevaluedistribution_max_date_and_more.py +72 -0
- endoreg_db/migrations/0058_datevaluedistribution_description_and_more.py +28 -0
- endoreg_db/migrations/0059_casetemplaterule_rule_values.py +18 -0
- endoreg_db/migrations/0060_labvalue__default_date_value_distribution_and_more.py +44 -0
- endoreg_db/migrations/0061_remove_patientlabvalue_date_patientlabvalue_datetime.py +24 -0
- endoreg_db/migrations/0062_labvalue_numeric_precision.py +18 -0
- endoreg_db/migrations/0063_alter_labvalue_numeric_precision.py +18 -0
- endoreg_db/migrations/0064_casetemplaterule_extra_parameters_and_more.py +23 -0
- endoreg_db/migrations/0065_rename__date_value_distribution_casetemplaterule_date_value_distribution_and_more.py +58 -0
- endoreg_db/migrations/0066_alter_patientlabvalue_patient_and_more.py +29 -0
- endoreg_db/migrations/0067_alter_medicationindication_indication_type.py +19 -0
- endoreg_db/models/__init__.py +30 -11
- endoreg_db/models/case_template/__init__.py +6 -0
- endoreg_db/models/case_template/case_template.py +81 -0
- endoreg_db/models/case_template/case_template_rule.py +276 -0
- endoreg_db/models/case_template/case_template_rule_value.py +73 -0
- endoreg_db/models/case_template/case_template_type.py +28 -0
- endoreg_db/models/center/__init__.py +4 -0
- endoreg_db/models/center/center_product.py +34 -0
- endoreg_db/models/center/center_resource.py +19 -0
- endoreg_db/models/center/center_waste.py +11 -0
- endoreg_db/models/data_file/import_classes/raw_pdf.py +4 -1
- endoreg_db/models/data_file/metadata/video_meta.py +4 -3
- endoreg_db/models/disease.py +56 -0
- endoreg_db/models/emission/__init__.py +1 -0
- endoreg_db/models/emission/emission_factor.py +20 -0
- endoreg_db/models/event.py +22 -0
- endoreg_db/models/information_source.py +7 -0
- endoreg_db/models/laboratory/__init__.py +1 -0
- endoreg_db/models/laboratory/lab_value.py +102 -0
- endoreg_db/models/medication/__init__.py +1 -0
- endoreg_db/models/medication/medication.py +148 -0
- endoreg_db/models/other/__init__.py +5 -0
- endoreg_db/models/other/distribution.py +215 -0
- endoreg_db/models/other/material.py +16 -0
- endoreg_db/models/other/resource.py +18 -0
- endoreg_db/models/other/transport_route.py +21 -0
- endoreg_db/models/other/waste.py +20 -0
- endoreg_db/models/persons/__init__.py +3 -2
- endoreg_db/models/persons/gender.py +22 -0
- endoreg_db/models/persons/patient/__init__.py +8 -0
- endoreg_db/models/persons/patient/case/__init__.py +0 -0
- endoreg_db/models/persons/patient/case/case.py +30 -0
- endoreg_db/models/persons/patient/patient.py +216 -0
- endoreg_db/models/persons/patient/patient_disease.py +16 -0
- endoreg_db/models/persons/patient/patient_event.py +22 -0
- endoreg_db/models/persons/patient/patient_lab_sample.py +106 -0
- endoreg_db/models/persons/patient/patient_lab_value.py +176 -0
- endoreg_db/models/persons/patient/patient_medication.py +44 -0
- endoreg_db/models/persons/patient/patient_medication_schedule.py +28 -0
- endoreg_db/models/persons/person.py +1 -4
- endoreg_db/models/persons/portal_user_information.py +0 -2
- endoreg_db/models/product/__init__.py +5 -0
- endoreg_db/models/product/product.py +97 -0
- endoreg_db/models/product/product_group.py +19 -0
- endoreg_db/models/product/product_material.py +24 -0
- endoreg_db/models/product/product_weight.py +26 -0
- endoreg_db/models/product/reference_product.py +99 -0
- endoreg_db/models/questionnaires/__init__.py +114 -0
- endoreg_db/models/quiz/__init__.py +2 -0
- endoreg_db/models/quiz/quiz_answer.py +41 -0
- endoreg_db/models/quiz/quiz_question.py +54 -0
- endoreg_db/models/rules/__init__.py +5 -0
- endoreg_db/models/rules/rule.py +24 -0
- endoreg_db/models/rules/rule_applicator.py +224 -0
- endoreg_db/models/rules/rule_attribute_dtype.py +19 -0
- endoreg_db/models/rules/rule_type.py +22 -0
- endoreg_db/models/rules/ruleset.py +19 -0
- endoreg_db/models/unit.py +6 -4
- endoreg_db/utils/dataloader.py +13 -106
- {endoreg_db-0.3.3.dist-info → endoreg_db-0.3.5.dist-info}/METADATA +1 -1
- endoreg_db-0.3.5.dist-info/RECORD +357 -0
- {endoreg_db-0.3.3.dist-info → endoreg_db-0.3.5.dist-info}/WHEEL +1 -1
- endoreg_db/models/persons/patient.py +0 -58
- endoreg_db/models.py +0 -3
- endoreg_db-0.3.3.dist-info/RECORD +0 -180
- /endoreg_db/models/{center.py → center/center.py} +0 -0
- {endoreg_db-0.3.3.dist-info → endoreg_db-0.3.5.dist-info}/LICENSE +0 -0
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
DEFAULT_PATIENT_LAB_SAMPLE_TYPE_NAME = "generic"
|
|
4
|
+
|
|
5
|
+
class PatientLabSampleTypeManager(models.Manager):
|
|
6
|
+
def get_by_natural_key(self, name):
|
|
7
|
+
return self.get(name=name)
|
|
8
|
+
|
|
9
|
+
class PatientLabSampleType(models.Model):
|
|
10
|
+
"""
|
|
11
|
+
A class representing a patient lab sample type.
|
|
12
|
+
|
|
13
|
+
Attributes:
|
|
14
|
+
name (str): The name of the patient lab sample type.
|
|
15
|
+
description (str): A description of the patient lab sample type.
|
|
16
|
+
|
|
17
|
+
"""
|
|
18
|
+
name = models.CharField(max_length=255)
|
|
19
|
+
name_de = models.CharField(max_length=255, null=True)
|
|
20
|
+
name_en = models.CharField(max_length=255, null=True)
|
|
21
|
+
description = models.TextField(blank=True, null=True)
|
|
22
|
+
|
|
23
|
+
objects = PatientLabSampleTypeManager()
|
|
24
|
+
|
|
25
|
+
def natural_key(self):
|
|
26
|
+
return (self.name,)
|
|
27
|
+
|
|
28
|
+
def __str__(self):
|
|
29
|
+
return self.name
|
|
30
|
+
|
|
31
|
+
@classmethod
|
|
32
|
+
def get_default_sample_type(cls):
|
|
33
|
+
"""
|
|
34
|
+
Get the default patient lab sample type.
|
|
35
|
+
|
|
36
|
+
Returns:
|
|
37
|
+
PatientLabSampleType: The default patient lab sample type.
|
|
38
|
+
|
|
39
|
+
"""
|
|
40
|
+
return cls.objects.get_or_create(name="default")[0]
|
|
41
|
+
|
|
42
|
+
from datetime import datetime as dt
|
|
43
|
+
from datetime import timezone
|
|
44
|
+
class PatientLabSample(models.Model):
|
|
45
|
+
"""
|
|
46
|
+
A class representing a patient lab sample.
|
|
47
|
+
|
|
48
|
+
Attributes:
|
|
49
|
+
patient (Patient): The patient to which the lab sample belongs.
|
|
50
|
+
sample_type (PatientLabSampleType): The type of the lab sample.
|
|
51
|
+
date (datetime): The date of the lab sample.
|
|
52
|
+
values (PatientLabValue; One2Many): The value of the lab sample.
|
|
53
|
+
unit (Unit): The unit of the lab sample.
|
|
54
|
+
|
|
55
|
+
"""
|
|
56
|
+
patient = models.ForeignKey("Patient", on_delete=models.CASCADE, related_name="lab_samples")
|
|
57
|
+
sample_type = models.ForeignKey("PatientLabSampleType", on_delete=models.CASCADE)
|
|
58
|
+
date = models.DateTimeField()
|
|
59
|
+
|
|
60
|
+
def __str__(self):
|
|
61
|
+
return f"{self.patient} - {self.sample_type} - {self.date} ()"
|
|
62
|
+
|
|
63
|
+
def get_values(self):
|
|
64
|
+
return self.values
|
|
65
|
+
|
|
66
|
+
@classmethod
|
|
67
|
+
def create_by_patient(cls, patient=None, sample_type=None, date=None, save = True):
|
|
68
|
+
"""
|
|
69
|
+
Create a new patient lab sample by patient.
|
|
70
|
+
|
|
71
|
+
Args:
|
|
72
|
+
patient (Patient): The patient to which the lab sample belongs.
|
|
73
|
+
sample_type (PatientLabSampleType): The type of the lab sample.
|
|
74
|
+
date (datetime): The date of the lab sample.
|
|
75
|
+
|
|
76
|
+
Returns:
|
|
77
|
+
PatientLabSample: The new patient lab sample.
|
|
78
|
+
|
|
79
|
+
"""
|
|
80
|
+
from endoreg_db.models.persons.patient import Patient, PatientLabSampleType
|
|
81
|
+
from warnings import warn
|
|
82
|
+
|
|
83
|
+
if not patient:
|
|
84
|
+
warn("No patient given. Cannot create patient lab sample.")
|
|
85
|
+
return None
|
|
86
|
+
if not sample_type:
|
|
87
|
+
sample_type = PatientLabSampleType.get_default_sample_type()
|
|
88
|
+
else:
|
|
89
|
+
sample_type = PatientLabSampleType.objects.get(name=sample_type)
|
|
90
|
+
if not date:
|
|
91
|
+
date = dt.now(timezone.utc)
|
|
92
|
+
|
|
93
|
+
patient_lab_sample = cls.objects.create(
|
|
94
|
+
patient=patient,
|
|
95
|
+
sample_type=sample_type,
|
|
96
|
+
date=date
|
|
97
|
+
)
|
|
98
|
+
|
|
99
|
+
if save:
|
|
100
|
+
patient_lab_sample.save()
|
|
101
|
+
|
|
102
|
+
return patient_lab_sample
|
|
103
|
+
|
|
104
|
+
|
|
105
|
+
|
|
106
|
+
|
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class PatientLabValue(models.Model):
|
|
4
|
+
"""
|
|
5
|
+
A class representing a patient lab value.
|
|
6
|
+
|
|
7
|
+
Attributes:
|
|
8
|
+
patient (Patient): The patient.
|
|
9
|
+
lab_value (LabValue): The lab value.
|
|
10
|
+
value (float): The value of the lab value.
|
|
11
|
+
date (datetime): The date of the lab value.
|
|
12
|
+
"""
|
|
13
|
+
patient = models.ForeignKey('Patient', on_delete=models.CASCADE,
|
|
14
|
+
related_name="lab_values", blank=True, null=True
|
|
15
|
+
)
|
|
16
|
+
lab_value = models.ForeignKey('LabValue', on_delete=models.CASCADE)
|
|
17
|
+
value = models.FloatField(blank=True, null=True)
|
|
18
|
+
value_str = models.CharField(max_length=255, blank=True, null=True)
|
|
19
|
+
sample = models.ForeignKey(
|
|
20
|
+
'PatientLabSample', on_delete=models.CASCADE,
|
|
21
|
+
blank=True, null=True,
|
|
22
|
+
related_name='values'
|
|
23
|
+
)
|
|
24
|
+
datetime = models.DateTimeField(# if not set, use now
|
|
25
|
+
auto_now_add=True
|
|
26
|
+
)
|
|
27
|
+
normal_range = models.JSONField(
|
|
28
|
+
default = dict
|
|
29
|
+
)
|
|
30
|
+
unit = models.ForeignKey('Unit', on_delete=models.CASCADE, blank=True, null=True)
|
|
31
|
+
|
|
32
|
+
@classmethod
|
|
33
|
+
def create_lab_value_by_sample(cls, sample=None, lab_value_name=None, value=None, value_str=None, unit=None):
|
|
34
|
+
from endoreg_db.models import LabValue
|
|
35
|
+
patient = sample.patient
|
|
36
|
+
lab_value = LabValue.objects.get(name=lab_value_name)
|
|
37
|
+
value = value
|
|
38
|
+
value_str = value_str
|
|
39
|
+
|
|
40
|
+
pat_lab_val = cls.objects.create(
|
|
41
|
+
patient = patient,
|
|
42
|
+
lab_value = lab_value,
|
|
43
|
+
value = value,
|
|
44
|
+
value_str = value_str,
|
|
45
|
+
sample = sample,
|
|
46
|
+
unit = unit,
|
|
47
|
+
)
|
|
48
|
+
|
|
49
|
+
pat_lab_val.save()
|
|
50
|
+
|
|
51
|
+
return pat_lab_val
|
|
52
|
+
|
|
53
|
+
def __str__(self):
|
|
54
|
+
_str = f'{self.lab_value} - {self.value} {self.unit} ({self.datetime})'
|
|
55
|
+
print(_str)
|
|
56
|
+
return _str
|
|
57
|
+
|
|
58
|
+
def set_min_norm_value(self, value, save = True):
|
|
59
|
+
self.normal_range["min"] = value
|
|
60
|
+
if save:
|
|
61
|
+
self.save()
|
|
62
|
+
|
|
63
|
+
def set_max_norm_value(self, value, save = True):
|
|
64
|
+
self.normal_range["max"] = value
|
|
65
|
+
if save:
|
|
66
|
+
self.save()
|
|
67
|
+
|
|
68
|
+
def set_norm_values_from_default(self):
|
|
69
|
+
age = self.patient.age()
|
|
70
|
+
gender = self.patient.gender
|
|
71
|
+
min_value, max_value = self.lab_value.get_normal_range(age=age, gender=gender)
|
|
72
|
+
self.set_min_norm_value(min_value, save = False)
|
|
73
|
+
self.set_max_norm_value(max_value, save = False)
|
|
74
|
+
self.save()
|
|
75
|
+
|
|
76
|
+
|
|
77
|
+
def set_unit_from_default(self):
|
|
78
|
+
self.unit = self.lab_value.default_unit
|
|
79
|
+
self.save()
|
|
80
|
+
|
|
81
|
+
def get_value(self):
|
|
82
|
+
if self.value:
|
|
83
|
+
return self.value
|
|
84
|
+
else:
|
|
85
|
+
return self.value_str
|
|
86
|
+
|
|
87
|
+
def get_value_field_name(self):
|
|
88
|
+
if self.value:
|
|
89
|
+
return "value"
|
|
90
|
+
else:
|
|
91
|
+
return "value_str"
|
|
92
|
+
|
|
93
|
+
# customize save method so that if a numeric value exists, we round it to the precision of the lab value
|
|
94
|
+
def save(self, *args, **kwargs):
|
|
95
|
+
if self.value:
|
|
96
|
+
precision = self.lab_value.numeric_precision
|
|
97
|
+
self.value = round(self.value, precision)
|
|
98
|
+
super().save(*args, **kwargs)
|
|
99
|
+
|
|
100
|
+
def set_value_by_distribution(self, distribution=None, save = True):
|
|
101
|
+
from endoreg_db.models import (
|
|
102
|
+
Patient, LabValue, Gender,
|
|
103
|
+
DateValueDistribution,
|
|
104
|
+
SingleCategoricalValueDistribution,
|
|
105
|
+
NumericValueDistribution,
|
|
106
|
+
MultipleCategoricalValueDistribution,
|
|
107
|
+
)
|
|
108
|
+
import warnings
|
|
109
|
+
|
|
110
|
+
patient:Patient = self.patient
|
|
111
|
+
|
|
112
|
+
dob = patient.dob
|
|
113
|
+
gender:Gender = patient.gender
|
|
114
|
+
lab_value:LabValue = self.lab_value
|
|
115
|
+
|
|
116
|
+
assert self.lab_value, "Lab value must be set to set value by distribution"
|
|
117
|
+
self.unit = self.lab_value.default_unit
|
|
118
|
+
|
|
119
|
+
if not distribution:
|
|
120
|
+
distribution = lab_value.get_default_default_distribution()
|
|
121
|
+
|
|
122
|
+
if not distribution:
|
|
123
|
+
warnings.warn(
|
|
124
|
+
"No distribution set for lab value, assuming uniform numeric distribution based on normal values"
|
|
125
|
+
)
|
|
126
|
+
|
|
127
|
+
if not self.normal_range.get("min", None) or not self.normal_range.get("max", None):
|
|
128
|
+
self.set_norm_values_from_default()
|
|
129
|
+
|
|
130
|
+
self.normal_range:dict
|
|
131
|
+
_min = self.normal_range.get("min", 0.0001)
|
|
132
|
+
_max = self.normal_range.get("max", 100)
|
|
133
|
+
_name = "auto-" + self.lab_value.name + "-distribution-default-uniform"
|
|
134
|
+
distribution = NumericValueDistribution(
|
|
135
|
+
name = _name,
|
|
136
|
+
min_value = _min,
|
|
137
|
+
max_value = _max,
|
|
138
|
+
distribution_type = "uniform"
|
|
139
|
+
)
|
|
140
|
+
|
|
141
|
+
value = distribution.generate_value()
|
|
142
|
+
self.value = value
|
|
143
|
+
if save:
|
|
144
|
+
self.save()
|
|
145
|
+
|
|
146
|
+
return value
|
|
147
|
+
|
|
148
|
+
if isinstance(distribution, SingleCategoricalValueDistribution):
|
|
149
|
+
value = distribution.generate_value()
|
|
150
|
+
self.value_str = value
|
|
151
|
+
if save:
|
|
152
|
+
self.save()
|
|
153
|
+
return value
|
|
154
|
+
|
|
155
|
+
elif isinstance(distribution, NumericValueDistribution):
|
|
156
|
+
value = distribution.generate_value()
|
|
157
|
+
self.value = value
|
|
158
|
+
if save:
|
|
159
|
+
self.save()
|
|
160
|
+
return value
|
|
161
|
+
|
|
162
|
+
elif isinstance(distribution, MultipleCategoricalValueDistribution):
|
|
163
|
+
value = distribution.generate_value()
|
|
164
|
+
self.value_str = value
|
|
165
|
+
if save:
|
|
166
|
+
self.save()
|
|
167
|
+
return value
|
|
168
|
+
|
|
169
|
+
elif isinstance(distribution, DateValueDistribution):
|
|
170
|
+
# raise not implemented error
|
|
171
|
+
value = distribution.generate_value()
|
|
172
|
+
self.value = value
|
|
173
|
+
if save:
|
|
174
|
+
self.save()
|
|
175
|
+
|
|
176
|
+
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
from django.utils.translation import gettext_lazy as _
|
|
3
|
+
|
|
4
|
+
class PatientMedication(models.Model):
|
|
5
|
+
patient = models.ForeignKey("Patient", on_delete= models.CASCADE)
|
|
6
|
+
medication_indication = models.ForeignKey(
|
|
7
|
+
"MedicationIndication", on_delete=models.CASCADE,
|
|
8
|
+
related_name="patient_medications", null=True
|
|
9
|
+
)
|
|
10
|
+
|
|
11
|
+
medication_schedules = models.ManyToManyField(
|
|
12
|
+
'MedicationSchedule'
|
|
13
|
+
)
|
|
14
|
+
unit = models.ForeignKey('Unit', on_delete=models.CASCADE)
|
|
15
|
+
dosage = models.JSONField()
|
|
16
|
+
active = models.BooleanField(default=True)
|
|
17
|
+
|
|
18
|
+
objects = models.Manager()
|
|
19
|
+
|
|
20
|
+
class Meta:
|
|
21
|
+
verbose_name = _('Patient Medication')
|
|
22
|
+
verbose_name_plural = _('Patient Medications')
|
|
23
|
+
|
|
24
|
+
@classmethod
|
|
25
|
+
def create_by_patient_and_indication(cls, patient, medication_indication):
|
|
26
|
+
from endoreg_db.models import MedicationIndication
|
|
27
|
+
medication_indication: MedicationIndication
|
|
28
|
+
patient_medication = cls.objects.create(patient=patient, medication_indication=medication_indication)
|
|
29
|
+
patient_medication.save()
|
|
30
|
+
patient_medication.set_schedules_from_indication()
|
|
31
|
+
|
|
32
|
+
return patient_medication
|
|
33
|
+
|
|
34
|
+
def set_schedules_from_indication(self):
|
|
35
|
+
schedules = self.medication_indication.medication_schedules.all()
|
|
36
|
+
self.medication_schedules.set(schedules)
|
|
37
|
+
self.save()
|
|
38
|
+
|
|
39
|
+
def __str__(self):
|
|
40
|
+
indication = self.medication_indication
|
|
41
|
+
schedules = self.medication_schedules.all()
|
|
42
|
+
return f'{indication} - {schedules}'
|
|
43
|
+
|
|
44
|
+
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class PatientMedicationSchedule(models.Model):
|
|
4
|
+
patient = models.ForeignKey("Patient", on_delete= models.CASCADE)
|
|
5
|
+
medication = models.ManyToManyField(
|
|
6
|
+
'PatientMedication', related_name='patient_medication_schedules'
|
|
7
|
+
)
|
|
8
|
+
|
|
9
|
+
created_at = models.DateTimeField(auto_now_add=True)
|
|
10
|
+
updated_at = models.DateTimeField(auto_now=True)
|
|
11
|
+
|
|
12
|
+
def __str__(self):
|
|
13
|
+
return f'{self.patient} - {self.medication.all()}'
|
|
14
|
+
|
|
15
|
+
@classmethod
|
|
16
|
+
def create_by_patient_and_indication_type(cls, patient, indication_type):
|
|
17
|
+
from endoreg_db.models import MedicationIndicationType, PatientMedication
|
|
18
|
+
|
|
19
|
+
medication_indication = MedicationIndicationType.get_random_indication_by_type(name=indication_type)
|
|
20
|
+
|
|
21
|
+
patient_medication_schedule = cls.objects.create(patient=patient)
|
|
22
|
+
patient_medication_schedule.save()
|
|
23
|
+
|
|
24
|
+
patient_medication = PatientMedication.create_by_patient_and_indication(patient, medication_indication)
|
|
25
|
+
patient_medication_schedule.medication.add(patient_medication)
|
|
26
|
+
patient_medication_schedule.save()
|
|
27
|
+
|
|
28
|
+
return patient_medication_schedule
|
|
@@ -17,10 +17,7 @@ class Person(models.Model):
|
|
|
17
17
|
first_name = models.CharField(max_length=255)
|
|
18
18
|
last_name = models.CharField(max_length=255)
|
|
19
19
|
dob = models.DateField("Date of Birth", blank=True, null=True)
|
|
20
|
-
gender = models.
|
|
21
|
-
max_length=10, choices=[('Male', 'Male'), ('Female', 'Female'), ('Other', 'Other')],
|
|
22
|
-
blank=True, null=True
|
|
23
|
-
)
|
|
20
|
+
gender = models.ForeignKey("Gender", on_delete=models.SET_NULL, null=True)
|
|
24
21
|
email = models.EmailField(max_length=255, blank=True, null=True)
|
|
25
22
|
phone = models.CharField(max_length=255, blank=True, null=True)
|
|
26
23
|
is_real_person = models.BooleanField(default=True)
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
def sum_weights(product_materials):
|
|
8
|
+
# sum up the weights
|
|
9
|
+
weight = 0
|
|
10
|
+
reference_unit = None
|
|
11
|
+
for product_material in product_materials:
|
|
12
|
+
if not reference_unit:
|
|
13
|
+
reference_unit = product_material.unit
|
|
14
|
+
else:
|
|
15
|
+
assert reference_unit == product_material.unit, "ProductMaterial units do not match"
|
|
16
|
+
weight += product_material.quantity
|
|
17
|
+
|
|
18
|
+
return weight, reference_unit
|
|
19
|
+
|
|
20
|
+
def sum_emissions(product_materials):
|
|
21
|
+
# sum up the emissions
|
|
22
|
+
emission = 0
|
|
23
|
+
reference_unit = None
|
|
24
|
+
for product_material in product_materials:
|
|
25
|
+
if not reference_unit:
|
|
26
|
+
reference_unit = product_material.unit
|
|
27
|
+
else:
|
|
28
|
+
assert reference_unit == product_material.unit, "ProductMaterial units do not match"
|
|
29
|
+
emission, emission_unit = product_material.get_emission()
|
|
30
|
+
assert reference_unit == emission_unit, "ProductMaterial units do not match"
|
|
31
|
+
emission += emission
|
|
32
|
+
|
|
33
|
+
return emission, reference_unit
|
|
34
|
+
|
|
35
|
+
class Product(models.Model):
|
|
36
|
+
objects = ProductManager()
|
|
37
|
+
|
|
38
|
+
name = models.CharField(max_length=255)
|
|
39
|
+
name_de = models.CharField(max_length=255, null=True)
|
|
40
|
+
name_en = models.CharField(max_length=255, null=True)
|
|
41
|
+
|
|
42
|
+
transport_route = models.ForeignKey("TransportRoute", on_delete=models.SET_NULL, null=True)
|
|
43
|
+
product_group = models.ForeignKey("ProductGroup", on_delete=models.SET_NULL, null=True)
|
|
44
|
+
|
|
45
|
+
def natural_key(self):
|
|
46
|
+
return (self.name,)
|
|
47
|
+
|
|
48
|
+
def __str__(self):
|
|
49
|
+
return self.name
|
|
50
|
+
|
|
51
|
+
def get_product_weight(self):
|
|
52
|
+
# check if there is a product material weight
|
|
53
|
+
from .product_material import ProductMaterial
|
|
54
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
55
|
+
if product_materials:
|
|
56
|
+
return self.get_product_material_weight()
|
|
57
|
+
|
|
58
|
+
# check if there is a product weight
|
|
59
|
+
#TODO
|
|
60
|
+
|
|
61
|
+
def get_package_weight(self):
|
|
62
|
+
# check if there is a package material weight
|
|
63
|
+
from .product_material import ProductMaterial
|
|
64
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
65
|
+
if product_materials:
|
|
66
|
+
return self.get_package_material_weight()
|
|
67
|
+
|
|
68
|
+
# check if there is a package weight
|
|
69
|
+
#TODO
|
|
70
|
+
|
|
71
|
+
def get_product_material_weight(self):
|
|
72
|
+
# get all materials with component == "product"
|
|
73
|
+
from .product_material import ProductMaterial
|
|
74
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
75
|
+
|
|
76
|
+
return sum_weights(product_materials)
|
|
77
|
+
|
|
78
|
+
def get_package_material_weight(self):
|
|
79
|
+
# get all materials with component == "package"
|
|
80
|
+
from .product_material import ProductMaterial
|
|
81
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
82
|
+
|
|
83
|
+
return sum_weights(product_materials)
|
|
84
|
+
|
|
85
|
+
def get_product_material_emission(self):
|
|
86
|
+
# get all materials with component == "product"
|
|
87
|
+
from .product_material import ProductMaterial
|
|
88
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
89
|
+
|
|
90
|
+
return sum_emissions(product_materials)
|
|
91
|
+
|
|
92
|
+
def get_package_material_emission(self):
|
|
93
|
+
# get all materials with component == "package"
|
|
94
|
+
from .product_material import ProductMaterial
|
|
95
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
96
|
+
|
|
97
|
+
return sum_emissions(product_materials)
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductGroupManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
class ProductGroup(models.Model):
|
|
8
|
+
objects = ProductGroupManager()
|
|
9
|
+
|
|
10
|
+
name = models.CharField(max_length=255)
|
|
11
|
+
name_de = models.CharField(max_length=255, null=True)
|
|
12
|
+
name_en = models.CharField(max_length=255, null=True)
|
|
13
|
+
|
|
14
|
+
def natural_key(self):
|
|
15
|
+
return (self.name,)
|
|
16
|
+
|
|
17
|
+
def __str__(self):
|
|
18
|
+
return self.name
|
|
19
|
+
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductMaterial(models.Model):
|
|
4
|
+
component = models.CharField(max_length=255)
|
|
5
|
+
material = models.ForeignKey("Material", on_delete=models.CASCADE)
|
|
6
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE, related_name="product_materials")
|
|
7
|
+
unit = models.ForeignKey("Unit", on_delete=models.CASCADE)
|
|
8
|
+
quantity = models.FloatField()
|
|
9
|
+
|
|
10
|
+
def get_emission(self):
|
|
11
|
+
from ..emission import EmissionFactor
|
|
12
|
+
emission_factor:EmissionFactor = self.material.emission_factor
|
|
13
|
+
if emission_factor is None:
|
|
14
|
+
raise Exception("No emission factor for material " + self.material.name + " found.")
|
|
15
|
+
|
|
16
|
+
# make sure product_material.unit is the same as emission_factor.unit
|
|
17
|
+
if self.unit != emission_factor.unit:
|
|
18
|
+
raise Exception("Unit mismatch: " + self.unit.name + " != " + emission_factor.unit.name)
|
|
19
|
+
|
|
20
|
+
emmision_value = emission_factor.value * self.quantity
|
|
21
|
+
emission_unit = emission_factor.unit
|
|
22
|
+
return emmision_value, emission_unit
|
|
23
|
+
|
|
24
|
+
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
import pandas as pd
|
|
3
|
+
|
|
4
|
+
class ProductWeightManager(models.Manager):
|
|
5
|
+
def get_by_natural_key(self, product, product_group, weight):
|
|
6
|
+
return self.get(product=product, product_group=product_group, weight=weight)
|
|
7
|
+
|
|
8
|
+
class ProductWeight(models.Model):
|
|
9
|
+
objects = ProductWeightManager()
|
|
10
|
+
|
|
11
|
+
name = models.CharField(max_length=255, null = True)
|
|
12
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE)
|
|
13
|
+
measured = models.FloatField(null=True)
|
|
14
|
+
verified = models.FloatField(null=True)
|
|
15
|
+
manufacturer = models.FloatField(null=True)
|
|
16
|
+
unit = models.ForeignKey("Unit", on_delete=models.SET_NULL, null=True)
|
|
17
|
+
|
|
18
|
+
def get_weight(self):
|
|
19
|
+
if not pd.isnull(self.verified):
|
|
20
|
+
return self.verified
|
|
21
|
+
elif not pd.isnull(self.measured):
|
|
22
|
+
return self.measured
|
|
23
|
+
elif not pd.isnull(self.manufacturer):
|
|
24
|
+
return self.manufacturer
|
|
25
|
+
else:
|
|
26
|
+
return None
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
from typing import List
|
|
3
|
+
|
|
4
|
+
class ReferenceProductManager(models.Manager):
|
|
5
|
+
def get_by_natural_key(self, product_name:str, product_group_name:str):
|
|
6
|
+
return self.get(product__name=product_name, product_group__name=product_group_name)
|
|
7
|
+
|
|
8
|
+
class ReferenceProduct(models.Model):
|
|
9
|
+
name = models.CharField(max_length=255)
|
|
10
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE)
|
|
11
|
+
product_group = models.OneToOneField("ProductGroup", on_delete=models.CASCADE, related_name="reference_product")
|
|
12
|
+
emission_factor_total = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, blank = True)
|
|
13
|
+
emission_factor_package = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, related_name="reference_product_package")
|
|
14
|
+
emission_factor_product = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, related_name="reference_product_product")
|
|
15
|
+
|
|
16
|
+
objects = ReferenceProductManager()
|
|
17
|
+
|
|
18
|
+
def __str__(self):
|
|
19
|
+
return self.product.name + " (" + self.product_group.name + ")"
|
|
20
|
+
|
|
21
|
+
def set_emission_factors(self):
|
|
22
|
+
from .product import Product
|
|
23
|
+
from .product_material import ProductMaterial
|
|
24
|
+
from ..emission import EmissionFactor
|
|
25
|
+
|
|
26
|
+
product:Product = self.product
|
|
27
|
+
materials = product.product_materials.all()
|
|
28
|
+
emission_factor_name = f"{self.product_group.name}_{product.name}_total_emission_factor"
|
|
29
|
+
emission_factor_package_name = f"{self.product_group.name}_{product.name}_package_emission_factor"
|
|
30
|
+
emission_factor_product_name = f"{self.product_group.name}_{product.name}_product_emission_factor"
|
|
31
|
+
|
|
32
|
+
product_emissions = 0
|
|
33
|
+
package_emissions = 0
|
|
34
|
+
|
|
35
|
+
product_weight, product_weight_unit = product.get_product_material_weight()
|
|
36
|
+
package_weight, package_weight_unit = product.get_package_material_weight()
|
|
37
|
+
product_emission, product_emission_unit = product.get_product_material_emission()
|
|
38
|
+
package_emission, package_emission_unit = product.get_package_material_emission()
|
|
39
|
+
|
|
40
|
+
total_weight = product_weight + package_weight
|
|
41
|
+
total_emission = product_emission + package_emission
|
|
42
|
+
|
|
43
|
+
reference_unit = product_weight_unit
|
|
44
|
+
assert reference_unit == package_weight_unit, "Package weight units do not match"
|
|
45
|
+
assert reference_unit == product_emission_unit, "Product emission units do not match"
|
|
46
|
+
assert reference_unit == package_emission_unit, "Package emission units do not match"
|
|
47
|
+
|
|
48
|
+
product_emission_factor_value = product_emission / product_weight
|
|
49
|
+
package_emission_factor_value = package_emission / package_weight
|
|
50
|
+
total_emission_factor_value = total_emission / total_weight
|
|
51
|
+
|
|
52
|
+
emission_factor, created = EmissionFactor.objects.get_or_create(
|
|
53
|
+
name=emission_factor_name,
|
|
54
|
+
defaults={
|
|
55
|
+
"name": emission_factor_name,
|
|
56
|
+
"value": total_emission_factor_value,
|
|
57
|
+
"unit": reference_unit
|
|
58
|
+
}
|
|
59
|
+
)
|
|
60
|
+
self.emission_factor_total = emission_factor
|
|
61
|
+
|
|
62
|
+
emission_factor_package, created = EmissionFactor.objects.get_or_create(
|
|
63
|
+
name=emission_factor_package_name,
|
|
64
|
+
defaults={
|
|
65
|
+
"name": emission_factor_package_name,
|
|
66
|
+
"value": package_emission_factor_value,
|
|
67
|
+
"unit": reference_unit
|
|
68
|
+
}
|
|
69
|
+
)
|
|
70
|
+
self.emission_factor_package = emission_factor_package
|
|
71
|
+
|
|
72
|
+
emission_factor_product, created = EmissionFactor.objects.get_or_create(
|
|
73
|
+
name=emission_factor_product_name,
|
|
74
|
+
defaults={
|
|
75
|
+
"name": emission_factor_product_name,
|
|
76
|
+
"value": product_emission_factor_value,
|
|
77
|
+
"unit": reference_unit
|
|
78
|
+
}
|
|
79
|
+
)
|
|
80
|
+
self.emission_factor_product = emission_factor_product
|
|
81
|
+
|
|
82
|
+
self.save()
|
|
83
|
+
|
|
84
|
+
def get_emission_factor(self, component:str):
|
|
85
|
+
# check if emission_factor_total exists:
|
|
86
|
+
if self.emission_factor_total is None:
|
|
87
|
+
self.set_emission_factors()
|
|
88
|
+
|
|
89
|
+
if component == "total":
|
|
90
|
+
return self.emission_factor_total
|
|
91
|
+
elif component == "package":
|
|
92
|
+
return self.emission_factor_package
|
|
93
|
+
elif component == "product":
|
|
94
|
+
return self.emission_factor_product
|
|
95
|
+
else:
|
|
96
|
+
raise Exception("Unknown component: " + component)
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
|