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.
Files changed (204) hide show
  1. endoreg_db/data/__init__.py +50 -1
  2. endoreg_db/data/case_template/rule/00_patient_lab_sample_add_default_value.yaml +167 -0
  3. endoreg_db/data/case_template/rule/01_patient-set-age.yaml +8 -0
  4. endoreg_db/data/case_template/rule/01_patient-set-gender.yaml +9 -0
  5. endoreg_db/data/case_template/rule/11_create_patient_lab_sample.yaml +23 -0
  6. endoreg_db/data/case_template/rule/12_create-patient_medication-anticoagulation.yaml +19 -0
  7. endoreg_db/data/case_template/rule/13_create-patient_medication_schedule-anticoagulation.yaml +19 -0
  8. endoreg_db/data/case_template/rule/19_create_patient.yaml +17 -0
  9. endoreg_db/data/case_template/rule_type/base_types.yaml +35 -0
  10. endoreg_db/data/case_template/rule_value_type/base_types.yaml +59 -0
  11. endoreg_db/data/case_template/template/base.yaml +8 -0
  12. endoreg_db/data/case_template/template_type/pre_endoscopy.yaml +3 -0
  13. endoreg_db/data/case_template/tmp/_rule_value +13 -0
  14. endoreg_db/data/case_template/tmp/rule/01_atrial_fibrillation.yaml +21 -0
  15. endoreg_db/data/case_template/tmp/rule/02_create_object.yaml +10 -0
  16. endoreg_db/data/case_template/tmp/template/atrial_fibrillation_low_risk.yaml +7 -0
  17. endoreg_db/data/center/data.yaml +8 -0
  18. endoreg_db/data/center_resource/green_endoscopy_dashboard_CenterResource.yaml +144 -0
  19. endoreg_db/data/center_waste/green_endoscopy_dashboard_CenterWaste.yaml +48 -0
  20. endoreg_db/data/disease/cardiovascular.yaml +37 -0
  21. endoreg_db/data/disease/hepatology.yaml +5 -0
  22. endoreg_db/data/disease/misc.yaml +6 -0
  23. endoreg_db/data/disease/renal.yaml +5 -0
  24. endoreg_db/data/disease_classification/chronic_kidney_disease.yaml +6 -0
  25. endoreg_db/data/disease_classification/coronary_vessel_disease.yaml +6 -0
  26. endoreg_db/data/disease_classification_choice/chronic_kidney_disease.yaml +41 -0
  27. endoreg_db/data/disease_classification_choice/coronary_vessel_disease.yaml +20 -0
  28. endoreg_db/data/distribution/date/patient.yaml +7 -0
  29. endoreg_db/data/distribution/single_categorical/patient.yaml +7 -0
  30. endoreg_db/data/emission_factor/green_endoscopy_dashboard_EmissionFactor.yaml +132 -0
  31. endoreg_db/data/event/cardiology.yaml +28 -0
  32. endoreg_db/data/event/neurology.yaml +14 -0
  33. endoreg_db/data/event/surgery.yaml +13 -0
  34. endoreg_db/data/event/thrombembolism.yaml +20 -0
  35. endoreg_db/data/examination/examinations/data.yaml +49 -0
  36. endoreg_db/data/gender/data.yaml +18 -0
  37. endoreg_db/data/information_source/medication.yaml +6 -0
  38. endoreg_db/data/lab_value/cardiac_enzymes.yaml +31 -0
  39. endoreg_db/data/lab_value/coagulation.yaml +49 -0
  40. endoreg_db/data/lab_value/electrolytes.yaml +190 -0
  41. endoreg_db/data/lab_value/gastrointestinal_function.yaml +121 -0
  42. endoreg_db/data/lab_value/hematology.yaml +169 -0
  43. endoreg_db/data/lab_value/hormones.yaml +53 -0
  44. endoreg_db/data/lab_value/lipids.yaml +44 -0
  45. endoreg_db/data/lab_value/misc.yaml +30 -0
  46. endoreg_db/data/lab_value/renal_function.yaml +11 -0
  47. endoreg_db/data/material/material.yaml +91 -0
  48. endoreg_db/data/medication/anticoagulation.yaml +65 -0
  49. endoreg_db/data/medication/tah.yaml +70 -0
  50. endoreg_db/data/medication_indication/anticoagulation.yaml +120 -0
  51. endoreg_db/data/medication_indication_type/data.yaml +11 -0
  52. endoreg_db/data/medication_indication_type/thrombembolism.yaml +41 -0
  53. endoreg_db/data/medication_intake_time/base.yaml +31 -0
  54. endoreg_db/data/medication_schedule/apixaban.yaml +95 -0
  55. endoreg_db/data/medication_schedule/ass.yaml +12 -0
  56. endoreg_db/data/medication_schedule/enoxaparin.yaml +26 -0
  57. endoreg_db/data/patient_lab_sample_type/generic.yaml +6 -0
  58. endoreg_db/data/product/green_endoscopy_dashboard_Product.yaml +66 -0
  59. endoreg_db/data/product_group/green_endoscopy_dashboard_ProductGroup.yaml +33 -0
  60. endoreg_db/data/product_material/green_endoscopy_dashboard_ProductMaterial.yaml +308 -0
  61. endoreg_db/data/product_weight/green_endoscopy_dashboard_ProductWeight.yaml +88 -0
  62. endoreg_db/data/reference_product/green_endoscopy_dashboard_ReferenceProduct.yaml +55 -0
  63. endoreg_db/data/resource/green_endoscopy_dashboard_Resource.yaml +15 -0
  64. endoreg_db/data/tmp/chronic_kidney_disease.yaml +0 -0
  65. endoreg_db/data/tmp/congestive_heart_failure.yaml +0 -0
  66. endoreg_db/data/transport_route/green_endoscopy_dashboard_TransportRoute.yaml +12 -0
  67. endoreg_db/data/unit/concentration.yaml +92 -0
  68. endoreg_db/data/unit/length.yaml +4 -4
  69. endoreg_db/data/unit/misc.yaml +20 -0
  70. endoreg_db/data/unit/rate.yaml +6 -0
  71. endoreg_db/data/unit/time.yaml +13 -0
  72. endoreg_db/data/unit/volume.yaml +13 -4
  73. endoreg_db/data/unit/weight.yaml +11 -4
  74. endoreg_db/data/waste/data.yaml +12 -0
  75. endoreg_db/forms/__init__.py +2 -1
  76. endoreg_db/forms/questionnaires/__init__.py +1 -0
  77. endoreg_db/forms/questionnaires/tto_questionnaire.py +23 -0
  78. endoreg_db/management/commands/load_base_db_data.py +58 -1
  79. endoreg_db/management/commands/load_disease_classification_choices_data.py +41 -0
  80. endoreg_db/management/commands/load_disease_classification_data.py +41 -0
  81. endoreg_db/management/commands/load_disease_data.py +40 -0
  82. endoreg_db/management/commands/load_distribution_data.py +66 -0
  83. endoreg_db/management/commands/load_event_data.py +41 -0
  84. endoreg_db/management/commands/load_g_play_data.py +113 -0
  85. endoreg_db/management/commands/load_gender_data.py +44 -0
  86. endoreg_db/management/commands/load_green_endoscopy_wuerzburg_data.py +133 -0
  87. endoreg_db/management/commands/load_lab_value_data.py +50 -0
  88. endoreg_db/management/commands/load_medication_data.py +41 -0
  89. endoreg_db/management/commands/load_medication_indication_data.py +63 -0
  90. endoreg_db/management/commands/load_medication_indication_type_data.py +41 -0
  91. endoreg_db/management/commands/load_medication_intake_time_data.py +41 -0
  92. endoreg_db/management/commands/load_medication_schedule_data.py +55 -0
  93. endoreg_db/migrations/0023_ttoquestionnaire_alter_pdftype_endoscope_info_line.py +59 -0
  94. endoreg_db/migrations/0024_remove_ttoquestionnaire_infections_and_more.py +27 -0
  95. endoreg_db/migrations/0025_event_alter_rawpdffile_file_patientevent.py +42 -0
  96. endoreg_db/migrations/0026_disease_diseaseclassification_and_more.py +166 -0
  97. endoreg_db/migrations/0027_labvalue_abbreviation_labvalue_default_normal_range_and_more.py +38 -0
  98. endoreg_db/migrations/0028_alter_unit_abbreviation.py +18 -0
  99. endoreg_db/migrations/0029_medicationintaketime_and_more.py +75 -0
  100. endoreg_db/migrations/0030_medicationindicationtype_and_more.py +101 -0
  101. endoreg_db/migrations/0031_rename_adapt_to_liver_function_medication_adapt_to_age_and_more.py +38 -0
  102. endoreg_db/migrations/0032_alter_medicationschedule_therapy_duration_d.py +18 -0
  103. endoreg_db/migrations/0033_medicationindication_sources.py +18 -0
  104. endoreg_db/migrations/0034_alter_rawpdffile_file.py +20 -0
  105. endoreg_db/migrations/0035_alter_medicationindication_sources.py +18 -0
  106. endoreg_db/migrations/0036_alter_rawpdffile_file.py +20 -0
  107. endoreg_db/migrations/0037_alter_medicationindication_sources.py +18 -0
  108. endoreg_db/migrations/0038_emissionfactor_material_product_productgroup_and_more.py +164 -0
  109. endoreg_db/migrations/0039_referenceproduct_name.py +19 -0
  110. endoreg_db/migrations/0040_quizanswertype_quizquestiontype_quizquestion_and_more.py +50 -0
  111. endoreg_db/migrations/0041_gender_patientmedication_medication_indication_and_more.py +40 -0
  112. endoreg_db/migrations/0042_casetemplateruletype_casetemplaterulevalue_and_more.py +74 -0
  113. endoreg_db/migrations/0043_casetemplatetype_name_de_casetemplatetype_name_en.py +23 -0
  114. endoreg_db/migrations/0044_casetemplateruletype_name_de_and_more.py +23 -0
  115. endoreg_db/migrations/0045_casetemplaterulevalue_value_type.py +19 -0
  116. endoreg_db/migrations/0046_casetemplaterulevalue_target_field.py +18 -0
  117. endoreg_db/migrations/0047_casetemplaterule_target_model.py +18 -0
  118. endoreg_db/migrations/0048_remove_casetemplaterule_chained_rules_and_more.py +22 -0
  119. endoreg_db/migrations/0049_remove_casetemplaterule_rule_values.py +17 -0
  120. endoreg_db/migrations/0050_casetemplaterule_rule_values.py +18 -0
  121. endoreg_db/migrations/0051_remove_casetemplaterule_calling_rules_and_more.py +27 -0
  122. endoreg_db/migrations/0052_rename_case_template_type_casetemplate_template_type.py +18 -0
  123. endoreg_db/migrations/0053_patientlabsampletype_patientlabsample_and_more.py +38 -0
  124. endoreg_db/migrations/0054_multiplecategoricalvaluedistribution_and_more.py +69 -0
  125. endoreg_db/migrations/0055_remove_casetemplaterule_rule_values_and_more.py +59 -0
  126. endoreg_db/migrations/0056_datevaluedistribution_and_more.py +32 -0
  127. endoreg_db/migrations/0057_remove_datevaluedistribution_max_date_and_more.py +72 -0
  128. endoreg_db/migrations/0058_datevaluedistribution_description_and_more.py +28 -0
  129. endoreg_db/migrations/0059_casetemplaterule_rule_values.py +18 -0
  130. endoreg_db/migrations/0060_labvalue__default_date_value_distribution_and_more.py +44 -0
  131. endoreg_db/migrations/0061_remove_patientlabvalue_date_patientlabvalue_datetime.py +24 -0
  132. endoreg_db/migrations/0062_labvalue_numeric_precision.py +18 -0
  133. endoreg_db/migrations/0063_alter_labvalue_numeric_precision.py +18 -0
  134. endoreg_db/migrations/0064_casetemplaterule_extra_parameters_and_more.py +23 -0
  135. endoreg_db/migrations/0065_rename__date_value_distribution_casetemplaterule_date_value_distribution_and_more.py +58 -0
  136. endoreg_db/migrations/0066_alter_patientlabvalue_patient_and_more.py +29 -0
  137. endoreg_db/migrations/0067_alter_medicationindication_indication_type.py +19 -0
  138. endoreg_db/models/__init__.py +30 -11
  139. endoreg_db/models/case_template/__init__.py +6 -0
  140. endoreg_db/models/case_template/case_template.py +81 -0
  141. endoreg_db/models/case_template/case_template_rule.py +276 -0
  142. endoreg_db/models/case_template/case_template_rule_value.py +73 -0
  143. endoreg_db/models/case_template/case_template_type.py +28 -0
  144. endoreg_db/models/center/__init__.py +4 -0
  145. endoreg_db/models/center/center_product.py +34 -0
  146. endoreg_db/models/center/center_resource.py +19 -0
  147. endoreg_db/models/center/center_waste.py +11 -0
  148. endoreg_db/models/data_file/import_classes/raw_pdf.py +4 -1
  149. endoreg_db/models/data_file/metadata/video_meta.py +4 -3
  150. endoreg_db/models/disease.py +56 -0
  151. endoreg_db/models/emission/__init__.py +1 -0
  152. endoreg_db/models/emission/emission_factor.py +20 -0
  153. endoreg_db/models/event.py +22 -0
  154. endoreg_db/models/information_source.py +7 -0
  155. endoreg_db/models/laboratory/__init__.py +1 -0
  156. endoreg_db/models/laboratory/lab_value.py +102 -0
  157. endoreg_db/models/medication/__init__.py +1 -0
  158. endoreg_db/models/medication/medication.py +148 -0
  159. endoreg_db/models/other/__init__.py +5 -0
  160. endoreg_db/models/other/distribution.py +215 -0
  161. endoreg_db/models/other/material.py +16 -0
  162. endoreg_db/models/other/resource.py +18 -0
  163. endoreg_db/models/other/transport_route.py +21 -0
  164. endoreg_db/models/other/waste.py +20 -0
  165. endoreg_db/models/persons/__init__.py +3 -2
  166. endoreg_db/models/persons/gender.py +22 -0
  167. endoreg_db/models/persons/patient/__init__.py +8 -0
  168. endoreg_db/models/persons/patient/case/__init__.py +0 -0
  169. endoreg_db/models/persons/patient/case/case.py +30 -0
  170. endoreg_db/models/persons/patient/patient.py +216 -0
  171. endoreg_db/models/persons/patient/patient_disease.py +16 -0
  172. endoreg_db/models/persons/patient/patient_event.py +22 -0
  173. endoreg_db/models/persons/patient/patient_lab_sample.py +106 -0
  174. endoreg_db/models/persons/patient/patient_lab_value.py +176 -0
  175. endoreg_db/models/persons/patient/patient_medication.py +44 -0
  176. endoreg_db/models/persons/patient/patient_medication_schedule.py +28 -0
  177. endoreg_db/models/persons/person.py +1 -4
  178. endoreg_db/models/persons/portal_user_information.py +0 -2
  179. endoreg_db/models/product/__init__.py +5 -0
  180. endoreg_db/models/product/product.py +97 -0
  181. endoreg_db/models/product/product_group.py +19 -0
  182. endoreg_db/models/product/product_material.py +24 -0
  183. endoreg_db/models/product/product_weight.py +26 -0
  184. endoreg_db/models/product/reference_product.py +99 -0
  185. endoreg_db/models/questionnaires/__init__.py +114 -0
  186. endoreg_db/models/quiz/__init__.py +2 -0
  187. endoreg_db/models/quiz/quiz_answer.py +41 -0
  188. endoreg_db/models/quiz/quiz_question.py +54 -0
  189. endoreg_db/models/rules/__init__.py +5 -0
  190. endoreg_db/models/rules/rule.py +24 -0
  191. endoreg_db/models/rules/rule_applicator.py +224 -0
  192. endoreg_db/models/rules/rule_attribute_dtype.py +19 -0
  193. endoreg_db/models/rules/rule_type.py +22 -0
  194. endoreg_db/models/rules/ruleset.py +19 -0
  195. endoreg_db/models/unit.py +6 -4
  196. endoreg_db/utils/dataloader.py +13 -106
  197. {endoreg_db-0.3.3.dist-info → endoreg_db-0.3.5.dist-info}/METADATA +1 -1
  198. endoreg_db-0.3.5.dist-info/RECORD +357 -0
  199. {endoreg_db-0.3.3.dist-info → endoreg_db-0.3.5.dist-info}/WHEEL +1 -1
  200. endoreg_db/models/persons/patient.py +0 -58
  201. endoreg_db/models.py +0 -3
  202. endoreg_db-0.3.3.dist-info/RECORD +0 -180
  203. /endoreg_db/models/{center.py → center/center.py} +0 -0
  204. {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.CharField(
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)
@@ -1,8 +1,6 @@
1
1
  from django.db import models
2
2
 
3
3
  # models.py in your main app
4
-
5
- from django.db import models
6
4
  from django.contrib.auth.models import User
7
5
 
8
6
  class ProfessionManager(models.Manager):
@@ -0,0 +1,5 @@
1
+ from .product import Product
2
+ from .product_material import ProductMaterial
3
+ from .product_group import ProductGroup
4
+ from .reference_product import ReferenceProduct
5
+ from .product_weight import ProductWeight
@@ -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
+