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,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
+
@@ -0,0 +1,2 @@
1
+ from .quiz_answer import QuizAnswer, QuizAnswerType
2
+ from .quiz_question import QuizQuestion, QuizQuestionType
@@ -0,0 +1,41 @@
1
+ from django.db import models
2
+
3
+ class QuizAnswerTypeManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class QuizAnswerType(models.Model):
8
+ objects = QuizAnswerTypeManager()
9
+
10
+ name = models.CharField(max_length=255)
11
+ description = models.TextField(blank=True, null=True)
12
+
13
+ def natural_key(self):
14
+ return (self.name,)
15
+
16
+ def __str__(self):
17
+ return self.name
18
+
19
+ class QuizAnswerManager(models.Manager):
20
+ def get_by_natural_key(self, text):
21
+ return self.get(text=text)
22
+
23
+ class QuizAnswer(models.Model):
24
+ objects = QuizAnswerManager()
25
+
26
+ text_value = models.CharField(max_length=255, null=True, blank=True)
27
+ number_value = models.FloatField(null=True, blank=True)
28
+ answer_type = models.ForeignKey("QuizAnswerType", on_delete=models.CASCADE)
29
+ question = models.ForeignKey("QuizQuestion", on_delete=models.CASCADE)
30
+
31
+ last_updated = models.DateTimeField(auto_now=True)
32
+
33
+ #TODO add user as foreign key
34
+
35
+ def natural_key(self):
36
+ return (self.text,)
37
+
38
+ def __str__(self):
39
+ return self.text
40
+
41
+
@@ -0,0 +1,54 @@
1
+ from django.db import models
2
+
3
+ class QuizQuestionTypeManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class QuizQuestionType(models.Model):
8
+ """
9
+ A class representing a quiz question type.
10
+
11
+ Attributes:
12
+ name (str): The name of the quiz question type.
13
+ description (str): A description of the quiz question type.
14
+
15
+ """
16
+ name = models.CharField(max_length=255)
17
+ description = models.TextField(blank=True, null=True)
18
+
19
+ objects = QuizQuestionTypeManager()
20
+
21
+ def natural_key(self):
22
+ return (self.name,)
23
+
24
+ def __str__(self):
25
+ return self.name
26
+
27
+ class QuizQuestionManager(models.Manager):
28
+ def get_by_natural_key(self, text):
29
+ return self.get(text=text)
30
+
31
+ class QuizQuestion(models.Model):
32
+ """
33
+ A class representing a quiz question.
34
+
35
+ Attributes:
36
+ text (str): The text of the quiz question.
37
+ question_type (QuizQuestionType): The type of the quiz question.
38
+ description (str): A description of the quiz question.
39
+
40
+ """
41
+ text = models.TextField()
42
+ description = models.TextField(blank=True, null=True)
43
+ question_type = models.ForeignKey("QuizQuestionType", on_delete=models.CASCADE)
44
+
45
+ objects = QuizQuestionManager()
46
+
47
+ def natural_key(self):
48
+ return (self.text,)
49
+
50
+ def __str__(self):
51
+ return self.text
52
+
53
+ def get_question_type(self):
54
+ return self.question_type.name
@@ -0,0 +1,5 @@
1
+ from .rule_type import RuleType
2
+ from .rule import Rule
3
+ from .ruleset import Ruleset
4
+ from .rule_attribute_dtype import RuleAttributeDType
5
+ from .rule_applicator import RuleApplicator
@@ -0,0 +1,24 @@
1
+ from django.db import models
2
+
3
+ class RuleManager(models.Manager):
4
+ def get_by_natural_key(self, name):
5
+ return self.get(name=name)
6
+
7
+ class Rule(models.Model):
8
+ name = models.CharField(max_length=255, unique=True)
9
+ name_de = models.CharField(max_length=255, blank=True, null=True)
10
+ name_en = models.CharField(max_length=255, blank=True, null=True)
11
+ description = models.TextField(blank=True, null=True)
12
+ attribute_key = models.CharField(max_length=255)
13
+ rule_type = models.ForeignKey("RuleType", on_delete=models.CASCADE)
14
+ attribute_dtype = models.ForeignKey("RuleAttributeDType", on_delete=models.CASCADE)
15
+
16
+ objects = RuleManager()
17
+
18
+ class Meta:
19
+ verbose_name = 'Rule'
20
+ verbose_name_plural = 'Rules'
21
+
22
+ def natural_key(self):
23
+ return (self.name,)
24
+