endoreg-db 0.3.4__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 (198) 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/management/commands/load_base_db_data.py +58 -1
  76. endoreg_db/management/commands/load_disease_classification_choices_data.py +41 -0
  77. endoreg_db/management/commands/load_disease_classification_data.py +41 -0
  78. endoreg_db/management/commands/load_disease_data.py +40 -0
  79. endoreg_db/management/commands/load_distribution_data.py +66 -0
  80. endoreg_db/management/commands/load_event_data.py +41 -0
  81. endoreg_db/management/commands/load_g_play_data.py +113 -0
  82. endoreg_db/management/commands/load_gender_data.py +44 -0
  83. endoreg_db/management/commands/load_green_endoscopy_wuerzburg_data.py +133 -0
  84. endoreg_db/management/commands/load_lab_value_data.py +50 -0
  85. endoreg_db/management/commands/load_medication_data.py +41 -0
  86. endoreg_db/management/commands/load_medication_indication_data.py +63 -0
  87. endoreg_db/management/commands/load_medication_indication_type_data.py +41 -0
  88. endoreg_db/management/commands/load_medication_intake_time_data.py +41 -0
  89. endoreg_db/management/commands/load_medication_schedule_data.py +55 -0
  90. endoreg_db/migrations/0025_event_alter_rawpdffile_file_patientevent.py +42 -0
  91. endoreg_db/migrations/0026_disease_diseaseclassification_and_more.py +166 -0
  92. endoreg_db/migrations/0027_labvalue_abbreviation_labvalue_default_normal_range_and_more.py +38 -0
  93. endoreg_db/migrations/0028_alter_unit_abbreviation.py +18 -0
  94. endoreg_db/migrations/0029_medicationintaketime_and_more.py +75 -0
  95. endoreg_db/migrations/0030_medicationindicationtype_and_more.py +101 -0
  96. endoreg_db/migrations/0031_rename_adapt_to_liver_function_medication_adapt_to_age_and_more.py +38 -0
  97. endoreg_db/migrations/0032_alter_medicationschedule_therapy_duration_d.py +18 -0
  98. endoreg_db/migrations/0033_medicationindication_sources.py +18 -0
  99. endoreg_db/migrations/0034_alter_rawpdffile_file.py +20 -0
  100. endoreg_db/migrations/0035_alter_medicationindication_sources.py +18 -0
  101. endoreg_db/migrations/0036_alter_rawpdffile_file.py +20 -0
  102. endoreg_db/migrations/0037_alter_medicationindication_sources.py +18 -0
  103. endoreg_db/migrations/0038_emissionfactor_material_product_productgroup_and_more.py +164 -0
  104. endoreg_db/migrations/0039_referenceproduct_name.py +19 -0
  105. endoreg_db/migrations/0040_quizanswertype_quizquestiontype_quizquestion_and_more.py +50 -0
  106. endoreg_db/migrations/0041_gender_patientmedication_medication_indication_and_more.py +40 -0
  107. endoreg_db/migrations/0042_casetemplateruletype_casetemplaterulevalue_and_more.py +74 -0
  108. endoreg_db/migrations/0043_casetemplatetype_name_de_casetemplatetype_name_en.py +23 -0
  109. endoreg_db/migrations/0044_casetemplateruletype_name_de_and_more.py +23 -0
  110. endoreg_db/migrations/0045_casetemplaterulevalue_value_type.py +19 -0
  111. endoreg_db/migrations/0046_casetemplaterulevalue_target_field.py +18 -0
  112. endoreg_db/migrations/0047_casetemplaterule_target_model.py +18 -0
  113. endoreg_db/migrations/0048_remove_casetemplaterule_chained_rules_and_more.py +22 -0
  114. endoreg_db/migrations/0049_remove_casetemplaterule_rule_values.py +17 -0
  115. endoreg_db/migrations/0050_casetemplaterule_rule_values.py +18 -0
  116. endoreg_db/migrations/0051_remove_casetemplaterule_calling_rules_and_more.py +27 -0
  117. endoreg_db/migrations/0052_rename_case_template_type_casetemplate_template_type.py +18 -0
  118. endoreg_db/migrations/0053_patientlabsampletype_patientlabsample_and_more.py +38 -0
  119. endoreg_db/migrations/0054_multiplecategoricalvaluedistribution_and_more.py +69 -0
  120. endoreg_db/migrations/0055_remove_casetemplaterule_rule_values_and_more.py +59 -0
  121. endoreg_db/migrations/0056_datevaluedistribution_and_more.py +32 -0
  122. endoreg_db/migrations/0057_remove_datevaluedistribution_max_date_and_more.py +72 -0
  123. endoreg_db/migrations/0058_datevaluedistribution_description_and_more.py +28 -0
  124. endoreg_db/migrations/0059_casetemplaterule_rule_values.py +18 -0
  125. endoreg_db/migrations/0060_labvalue__default_date_value_distribution_and_more.py +44 -0
  126. endoreg_db/migrations/0061_remove_patientlabvalue_date_patientlabvalue_datetime.py +24 -0
  127. endoreg_db/migrations/0062_labvalue_numeric_precision.py +18 -0
  128. endoreg_db/migrations/0063_alter_labvalue_numeric_precision.py +18 -0
  129. endoreg_db/migrations/0064_casetemplaterule_extra_parameters_and_more.py +23 -0
  130. endoreg_db/migrations/0065_rename__date_value_distribution_casetemplaterule_date_value_distribution_and_more.py +58 -0
  131. endoreg_db/migrations/0066_alter_patientlabvalue_patient_and_more.py +29 -0
  132. endoreg_db/migrations/0067_alter_medicationindication_indication_type.py +19 -0
  133. endoreg_db/models/__init__.py +28 -11
  134. endoreg_db/models/case_template/__init__.py +6 -0
  135. endoreg_db/models/case_template/case_template.py +81 -0
  136. endoreg_db/models/case_template/case_template_rule.py +276 -0
  137. endoreg_db/models/case_template/case_template_rule_value.py +73 -0
  138. endoreg_db/models/case_template/case_template_type.py +28 -0
  139. endoreg_db/models/center/__init__.py +4 -0
  140. endoreg_db/models/center/center_product.py +34 -0
  141. endoreg_db/models/center/center_resource.py +19 -0
  142. endoreg_db/models/center/center_waste.py +11 -0
  143. endoreg_db/models/data_file/import_classes/raw_pdf.py +4 -1
  144. endoreg_db/models/data_file/metadata/video_meta.py +4 -3
  145. endoreg_db/models/disease.py +56 -0
  146. endoreg_db/models/emission/__init__.py +1 -0
  147. endoreg_db/models/emission/emission_factor.py +20 -0
  148. endoreg_db/models/event.py +22 -0
  149. endoreg_db/models/information_source.py +7 -0
  150. endoreg_db/models/laboratory/__init__.py +1 -0
  151. endoreg_db/models/laboratory/lab_value.py +102 -0
  152. endoreg_db/models/medication/__init__.py +1 -0
  153. endoreg_db/models/medication/medication.py +148 -0
  154. endoreg_db/models/other/__init__.py +5 -0
  155. endoreg_db/models/other/distribution.py +215 -0
  156. endoreg_db/models/other/material.py +16 -0
  157. endoreg_db/models/other/resource.py +18 -0
  158. endoreg_db/models/other/transport_route.py +21 -0
  159. endoreg_db/models/other/waste.py +20 -0
  160. endoreg_db/models/persons/__init__.py +3 -2
  161. endoreg_db/models/persons/gender.py +22 -0
  162. endoreg_db/models/persons/patient/__init__.py +8 -0
  163. endoreg_db/models/persons/patient/case/__init__.py +0 -0
  164. endoreg_db/models/persons/patient/case/case.py +30 -0
  165. endoreg_db/models/persons/patient/patient.py +216 -0
  166. endoreg_db/models/persons/patient/patient_disease.py +16 -0
  167. endoreg_db/models/persons/patient/patient_event.py +22 -0
  168. endoreg_db/models/persons/patient/patient_lab_sample.py +106 -0
  169. endoreg_db/models/persons/patient/patient_lab_value.py +176 -0
  170. endoreg_db/models/persons/patient/patient_medication.py +44 -0
  171. endoreg_db/models/persons/patient/patient_medication_schedule.py +28 -0
  172. endoreg_db/models/persons/person.py +1 -4
  173. endoreg_db/models/persons/portal_user_information.py +0 -2
  174. endoreg_db/models/product/__init__.py +5 -0
  175. endoreg_db/models/product/product.py +97 -0
  176. endoreg_db/models/product/product_group.py +19 -0
  177. endoreg_db/models/product/product_material.py +24 -0
  178. endoreg_db/models/product/product_weight.py +26 -0
  179. endoreg_db/models/product/reference_product.py +99 -0
  180. endoreg_db/models/quiz/__init__.py +2 -0
  181. endoreg_db/models/quiz/quiz_answer.py +41 -0
  182. endoreg_db/models/quiz/quiz_question.py +54 -0
  183. endoreg_db/models/rules/__init__.py +5 -0
  184. endoreg_db/models/rules/rule.py +24 -0
  185. endoreg_db/models/rules/rule_applicator.py +224 -0
  186. endoreg_db/models/rules/rule_attribute_dtype.py +19 -0
  187. endoreg_db/models/rules/rule_type.py +22 -0
  188. endoreg_db/models/rules/ruleset.py +19 -0
  189. endoreg_db/models/unit.py +6 -4
  190. endoreg_db/utils/dataloader.py +13 -106
  191. {endoreg_db-0.3.4.dist-info → endoreg_db-0.3.5.dist-info}/METADATA +1 -1
  192. endoreg_db-0.3.5.dist-info/RECORD +357 -0
  193. endoreg_db/models/persons/patient.py +0 -58
  194. endoreg_db/models.py +0 -3
  195. endoreg_db-0.3.4.dist-info/RECORD +0 -185
  196. /endoreg_db/models/{center.py → center/center.py} +0 -0
  197. {endoreg_db-0.3.4.dist-info → endoreg_db-0.3.5.dist-info}/LICENSE +0 -0
  198. {endoreg_db-0.3.4.dist-info → endoreg_db-0.3.5.dist-info}/WHEEL +0 -0
@@ -0,0 +1,18 @@
1
+ from django.db import models
2
+
3
+ class ResourceManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class Resource(models.Model):
8
+ objects = ResourceManager()
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
@@ -0,0 +1,21 @@
1
+ from django.db import models
2
+
3
+ class TransportRouteManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class TransportRoute(models.Model):
8
+ objects = TransportRouteManager()
9
+
10
+ distance = models.FloatField()
11
+ name = models.CharField(max_length=255)
12
+ name_de = models.CharField(max_length=255, null=True)
13
+ name_en = models.CharField(max_length=255, null=True)
14
+ emission_factor = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True)
15
+ unit = models.ForeignKey("Unit", on_delete=models.SET_NULL, null=True)
16
+
17
+ def natural_key(self):
18
+ return (self.name,)
19
+
20
+ def __str__(self):
21
+ return self.name
@@ -0,0 +1,20 @@
1
+ from django.db import models
2
+
3
+ class WasteManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class Waste(models.Model):
8
+ objects = WasteManager()
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
+ # emission_factor = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True)
14
+
15
+ def natural_key(self):
16
+ return (self.name,)
17
+
18
+ def __str__(self):
19
+ return self.name
20
+
@@ -1,6 +1,7 @@
1
+ from .gender import Gender
1
2
  from .person import Person
2
- from .patient import Patient, PatientForm
3
+ from .patient import *
3
4
  from .examiner import Examiner, ExaminerSerializer
4
5
  from .portal_user_information import PortalUserInfo, Profession
5
6
  from .first_name import FirstName
6
- from .last_name import LastName
7
+ from .last_name import LastName
@@ -0,0 +1,22 @@
1
+ from django.db import models
2
+
3
+ class GenderManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class Gender(models.Model):
8
+ """A class representing gender."""
9
+ objects = GenderManager()
10
+
11
+ name = models.CharField(max_length=255)
12
+ name_de = models.CharField(max_length=255, null=True)
13
+ name_en = models.CharField(max_length=255, null=True)
14
+ abbreviation = models.CharField(max_length=255, null=True)
15
+ description = models.TextField(blank=True, null=True)
16
+
17
+ def natural_key(self):
18
+ return (self.name,)
19
+
20
+ def __str__(self):
21
+ return self.name
22
+
@@ -0,0 +1,8 @@
1
+ from .patient import Patient, PatientForm
2
+ from .patient_event import PatientEvent
3
+ from .patient_disease import PatientDisease
4
+ from .patient_lab_sample import PatientLabSample, PatientLabSampleType
5
+ from .patient_lab_value import PatientLabValue
6
+ from .patient_medication import PatientMedication
7
+ from .patient_medication_schedule import PatientMedicationSchedule
8
+ from .case import *
File without changes
@@ -0,0 +1,30 @@
1
+ from django.db import models
2
+
3
+ class Case(models.Model):
4
+ """
5
+ A class representing a case.
6
+
7
+ Attributes:
8
+ name (str): The name of the case.
9
+ description (str): A description of the case.
10
+ case_template (CaseTemplate): The case template of the case.
11
+ patient (Patient): The patient of the case.
12
+ start_date (datetime): The start date of the case.
13
+ end_date (datetime): The end date of the case.
14
+ is_active (bool): A flag indicating whether the case is active.
15
+ is_closed (bool): A flag indicating whether the case is closed.
16
+ is_deleted (bool): A flag indicating whether the case is deleted.
17
+ created_at (datetime): The creation date of the case.
18
+ updated_at (datetime): The last update date of the case.
19
+
20
+ """
21
+ start_date = models.DateTimeField()
22
+ end_date = models.DateTimeField(null=True)
23
+ is_active = models.BooleanField(default=True)
24
+ is_closed = models.BooleanField(default=False)
25
+ is_deleted = models.BooleanField(default=False)
26
+ created_at = models.DateTimeField(auto_now_add=True)
27
+ updated_at = models.DateTimeField(auto_now=True)
28
+
29
+ def __str__(self):
30
+ return self.name
@@ -0,0 +1,216 @@
1
+ from ..person import Person
2
+ from django import forms
3
+ from django.forms import DateInput
4
+ from rest_framework import serializers
5
+ from ...patient_examination import PatientExamination
6
+ from ...data_file import ReportFile
7
+ from django.db import models
8
+ from faker import Faker
9
+ import random
10
+ from datetime import datetime
11
+
12
+ class Patient(Person):
13
+ """
14
+ A class representing a patient.
15
+
16
+ Attributes inhereted from Person:
17
+ first_name (str): The first name of the patient.
18
+ last_name (str): The last name of the patient.
19
+ dob (datetime.date): The date of birth of the patient.
20
+ gender (Foreign Key): The gender of the patient.
21
+ email (str): The email address of the patient.
22
+ phone (str): The phone number of the patient.
23
+
24
+ """
25
+ center = models.ForeignKey("Center", on_delete=models.CASCADE, blank=True, null=True)
26
+
27
+
28
+ def __str__(self):
29
+ return self.first_name + " " + self.last_name + " (" + str(self.dob) + ")"
30
+
31
+ def get_unmatched_report_files(self): #field: self.report_files; filter: report_file.patient_examination = None
32
+ '''Returns all report files for this patient that are not matched to a patient examination.'''
33
+
34
+ return self.reportfile_set.filter(patient_examination=None)
35
+
36
+ def get_unmatched_video_files(self): #field: self.videos; filter: video.patient_examination = None
37
+ '''Returns all video files for this patient that are not matched to a patient examination.'''
38
+ return self.videos.filter(patient_examination=None)
39
+
40
+ def get_patient_examinations(self): #field: self.patient_examinations
41
+ '''Returns all patient examinations for this patient ordered by date (most recent is first).'''
42
+ return self.patient_examinations.order_by('-date')
43
+
44
+ def create_examination_by_report_file(self, report_file:ReportFile):
45
+ '''Creates a patient examination for this patient based on the given report file.'''
46
+ patient_examination = PatientExamination(patient=self, report_file=report_file)
47
+ patient_examination.save()
48
+ return patient_examination
49
+
50
+ @classmethod
51
+ def get_random_gender(self, p_male=0.5, p_female=0.5):
52
+ """
53
+ Get a Gender object by name (male, female) from the database with given probability.
54
+
55
+ :param p_male: Probability of selecting 'male' gender.
56
+ :param p_female: Probability of selecting 'female' gender.
57
+ :return: Gender object selected based on given probabilities.
58
+ """
59
+ from endoreg_db.models import Gender
60
+
61
+ # Extract names and probabilities
62
+ gender_names = ["male", "female"]
63
+ probabilities = [0.5, 0.5]
64
+
65
+ # Debug: print the names and probabilities
66
+ print(f"Gender names: {gender_names}")
67
+ print(f"Probabilities: {probabilities}")
68
+
69
+ # Select a gender based on the given probabilities
70
+ selected_gender = random.choices(gender_names, probabilities)[0]
71
+ # Debug: print the selected gender
72
+ print(f"Selected gender: {selected_gender}")
73
+
74
+ # Fetch the corresponding Gender object from the database
75
+ gender_obj = Gender.objects.get(name=selected_gender)
76
+
77
+ return gender_obj
78
+
79
+
80
+ @classmethod
81
+ def get_random_age(self,
82
+ min_age = 55,
83
+ max_age = 90,
84
+ mean_age = 65,
85
+ std_age = 10,
86
+ distribution = "normal"
87
+ ):
88
+ """
89
+ Get a random age based on the given distribution.
90
+
91
+ :param min_age: Minimum age.
92
+ :param max_age: Maximum age.
93
+ :param mean_age: Mean age.
94
+ :param std_age: Standard deviation of the age.
95
+ :param distribution: Distribution of the age.
96
+ :return: Random age based on the given distribution.
97
+ """
98
+ if distribution == "normal":
99
+ age = int(random.normalvariate(mean_age, std_age))
100
+ else:
101
+ age = int(random.uniform(min_age, max_age))
102
+
103
+ return age
104
+
105
+ @classmethod
106
+ def get_dob_from_age(self, age, current_date=None):
107
+ """
108
+ Get a date of birth based on the given age and current date.
109
+
110
+ :param age: Age of the patient.
111
+ :param current_date: Current date.
112
+ :return: Date of birth based on the given age and current date.
113
+ """
114
+ if current_date is None:
115
+ current_date = datetime.now()
116
+ dob = current_date.replace(year=current_date.year - age).date()
117
+
118
+ # TODO
119
+ # randomize the day and month by adding a random number of days (0-364) to the date
120
+
121
+ return dob
122
+
123
+ @classmethod
124
+ def get_random_name_for_gender(self, gender_obj, locale="de_DE"):
125
+ gender = gender_obj.name
126
+ fake = Faker(locale)
127
+
128
+ if gender == "male":
129
+ first_name = fake.first_name_male()
130
+ last_name = fake.last_name_male()
131
+
132
+ else:
133
+ first_name = fake.first_name_female()
134
+ last_name = fake.last_name_female()
135
+
136
+ return last_name, first_name
137
+
138
+ @classmethod
139
+ def create_generic(self, center="gplay_case_generator"):
140
+ """
141
+ Create a generic patient with random attributes.
142
+
143
+ :param center: The center of the patient.
144
+ :return: The created patient.
145
+ """
146
+ from endoreg_db.models import Center
147
+ gender = Patient.get_random_gender()
148
+ last_name, first_name = Patient.get_random_name_for_gender(gender)
149
+
150
+ age = Patient.get_random_age()
151
+ dob = Patient.get_dob_from_age(age)
152
+
153
+ center = Center.objects.get(name=center)
154
+
155
+ patient = Patient.objects.create(
156
+ first_name=first_name,
157
+ last_name=last_name,
158
+ dob=dob
159
+ )
160
+ patient.save()
161
+ return patient
162
+
163
+ def age(self):
164
+ """
165
+ Get the age of the patient.
166
+
167
+ :return: The age of the patient.
168
+ """
169
+ # calculate correct age based on current date including day and month
170
+ current_date = datetime.now()
171
+ dob = self.dob
172
+ age = current_date.year - dob.year - ((current_date.month, current_date.day) < (dob.month, dob.day))
173
+ return age
174
+
175
+ def create_lab_sample(self, sample_type="generic", date=None, save=True):
176
+ """
177
+ Create a lab sample for this patient.
178
+
179
+ :param sample_type: The sample type. Should be either string of the sample types
180
+ name or the sample type object. If not set, the default sample type ("generic") is used.
181
+ :param date: The date of the lab sample.
182
+ :return: The created lab sample.
183
+ """
184
+ from endoreg_db.models import PatientLabSample, PatientLabSampleType
185
+ if date is None:
186
+ date = datetime.now()
187
+
188
+ if isinstance(sample_type, str):
189
+ sample_type = PatientLabSampleType.objects.get(name=sample_type)
190
+ assert sample_type is not None, f"Sample type with name '{sample_type}' not found."#
191
+ elif not isinstance(sample_type, PatientLabSampleType):
192
+ raise ValueError("Sample type must be either a string or a PatientLabSampleType object.")
193
+
194
+ patient_lab_sample = PatientLabSample.objects.create(
195
+ patient=self,
196
+ sample_type=sample_type,
197
+ date=date
198
+ )
199
+
200
+ if save:
201
+ patient_lab_sample.save()
202
+
203
+ return patient_lab_sample
204
+
205
+ class PatientForm(forms.ModelForm):
206
+ class Meta:
207
+ model = Patient
208
+ fields = '__all__'
209
+ widgets = {
210
+ 'dob': DateInput(attrs={'type': 'date'}),
211
+ }
212
+
213
+ def __init__(self, *args, **kwargs):
214
+ super().__init__(*args, **kwargs)
215
+ for field in self.fields.values():
216
+ field.widget.attrs['class'] = 'form-control'
@@ -0,0 +1,16 @@
1
+ from django.db import models
2
+
3
+
4
+
5
+ class PatientDisease(models.Model):
6
+ patient = models.ForeignKey("Patient", on_delete=models.CASCADE)
7
+ disease = models.ForeignKey("Disease", on_delete=models.CASCADE)
8
+ classification_choices = models.ManyToManyField("DiseaseClassificationChoice")
9
+ start_date = models.DateField(blank=True, null=True)
10
+ end_date = models.DateField(blank=True, null=True)
11
+
12
+ last_update = models.DateTimeField(auto_now=True)
13
+
14
+ def __str__(self):
15
+ return f"{self.patient} - {self.disease}"
16
+
@@ -0,0 +1,22 @@
1
+ from django.db import models
2
+ from .patient import Patient
3
+
4
+ class PatientEvent(models.Model):
5
+ """
6
+ A class representing an event for a patient.
7
+
8
+ Attributes:
9
+ patient (Patient): The patient associated with this event.
10
+ date (datetime.date): The date of the event.
11
+ description (str): A description of the event.
12
+ """
13
+ patient = models.ForeignKey(Patient, on_delete=models.CASCADE)
14
+ event = models.ForeignKey("Event", on_delete=models.CASCADE)
15
+ date_start = models.DateField()
16
+ date_end = models.DateField(blank=True, null=True)
17
+ description = models.TextField(blank=True, null=True)
18
+
19
+ last_update = models.DateTimeField(auto_now=True)
20
+
21
+ def __str__(self):
22
+ return str(self.date_start) + ": " + self.event
@@ -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
+