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.
- endoreg_db/data/__init__.py +50 -1
- endoreg_db/data/case_template/rule/00_patient_lab_sample_add_default_value.yaml +167 -0
- endoreg_db/data/case_template/rule/01_patient-set-age.yaml +8 -0
- endoreg_db/data/case_template/rule/01_patient-set-gender.yaml +9 -0
- endoreg_db/data/case_template/rule/11_create_patient_lab_sample.yaml +23 -0
- endoreg_db/data/case_template/rule/12_create-patient_medication-anticoagulation.yaml +19 -0
- endoreg_db/data/case_template/rule/13_create-patient_medication_schedule-anticoagulation.yaml +19 -0
- endoreg_db/data/case_template/rule/19_create_patient.yaml +17 -0
- endoreg_db/data/case_template/rule_type/base_types.yaml +35 -0
- endoreg_db/data/case_template/rule_value_type/base_types.yaml +59 -0
- endoreg_db/data/case_template/template/base.yaml +8 -0
- endoreg_db/data/case_template/template_type/pre_endoscopy.yaml +3 -0
- endoreg_db/data/case_template/tmp/_rule_value +13 -0
- endoreg_db/data/case_template/tmp/rule/01_atrial_fibrillation.yaml +21 -0
- endoreg_db/data/case_template/tmp/rule/02_create_object.yaml +10 -0
- endoreg_db/data/case_template/tmp/template/atrial_fibrillation_low_risk.yaml +7 -0
- endoreg_db/data/center/data.yaml +8 -0
- endoreg_db/data/center_resource/green_endoscopy_dashboard_CenterResource.yaml +144 -0
- endoreg_db/data/center_waste/green_endoscopy_dashboard_CenterWaste.yaml +48 -0
- endoreg_db/data/disease/cardiovascular.yaml +37 -0
- endoreg_db/data/disease/hepatology.yaml +5 -0
- endoreg_db/data/disease/misc.yaml +6 -0
- endoreg_db/data/disease/renal.yaml +5 -0
- endoreg_db/data/disease_classification/chronic_kidney_disease.yaml +6 -0
- endoreg_db/data/disease_classification/coronary_vessel_disease.yaml +6 -0
- endoreg_db/data/disease_classification_choice/chronic_kidney_disease.yaml +41 -0
- endoreg_db/data/disease_classification_choice/coronary_vessel_disease.yaml +20 -0
- endoreg_db/data/distribution/date/patient.yaml +7 -0
- endoreg_db/data/distribution/single_categorical/patient.yaml +7 -0
- endoreg_db/data/emission_factor/green_endoscopy_dashboard_EmissionFactor.yaml +132 -0
- endoreg_db/data/event/cardiology.yaml +28 -0
- endoreg_db/data/event/neurology.yaml +14 -0
- endoreg_db/data/event/surgery.yaml +13 -0
- endoreg_db/data/event/thrombembolism.yaml +20 -0
- endoreg_db/data/examination/examinations/data.yaml +49 -0
- endoreg_db/data/gender/data.yaml +18 -0
- endoreg_db/data/information_source/medication.yaml +6 -0
- endoreg_db/data/lab_value/cardiac_enzymes.yaml +31 -0
- endoreg_db/data/lab_value/coagulation.yaml +49 -0
- endoreg_db/data/lab_value/electrolytes.yaml +190 -0
- endoreg_db/data/lab_value/gastrointestinal_function.yaml +121 -0
- endoreg_db/data/lab_value/hematology.yaml +169 -0
- endoreg_db/data/lab_value/hormones.yaml +53 -0
- endoreg_db/data/lab_value/lipids.yaml +44 -0
- endoreg_db/data/lab_value/misc.yaml +30 -0
- endoreg_db/data/lab_value/renal_function.yaml +11 -0
- endoreg_db/data/material/material.yaml +91 -0
- endoreg_db/data/medication/anticoagulation.yaml +65 -0
- endoreg_db/data/medication/tah.yaml +70 -0
- endoreg_db/data/medication_indication/anticoagulation.yaml +120 -0
- endoreg_db/data/medication_indication_type/data.yaml +11 -0
- endoreg_db/data/medication_indication_type/thrombembolism.yaml +41 -0
- endoreg_db/data/medication_intake_time/base.yaml +31 -0
- endoreg_db/data/medication_schedule/apixaban.yaml +95 -0
- endoreg_db/data/medication_schedule/ass.yaml +12 -0
- endoreg_db/data/medication_schedule/enoxaparin.yaml +26 -0
- endoreg_db/data/patient_lab_sample_type/generic.yaml +6 -0
- endoreg_db/data/product/green_endoscopy_dashboard_Product.yaml +66 -0
- endoreg_db/data/product_group/green_endoscopy_dashboard_ProductGroup.yaml +33 -0
- endoreg_db/data/product_material/green_endoscopy_dashboard_ProductMaterial.yaml +308 -0
- endoreg_db/data/product_weight/green_endoscopy_dashboard_ProductWeight.yaml +88 -0
- endoreg_db/data/reference_product/green_endoscopy_dashboard_ReferenceProduct.yaml +55 -0
- endoreg_db/data/resource/green_endoscopy_dashboard_Resource.yaml +15 -0
- endoreg_db/data/tmp/chronic_kidney_disease.yaml +0 -0
- endoreg_db/data/tmp/congestive_heart_failure.yaml +0 -0
- endoreg_db/data/transport_route/green_endoscopy_dashboard_TransportRoute.yaml +12 -0
- endoreg_db/data/unit/concentration.yaml +92 -0
- endoreg_db/data/unit/length.yaml +4 -4
- endoreg_db/data/unit/misc.yaml +20 -0
- endoreg_db/data/unit/rate.yaml +6 -0
- endoreg_db/data/unit/time.yaml +13 -0
- endoreg_db/data/unit/volume.yaml +13 -4
- endoreg_db/data/unit/weight.yaml +11 -4
- endoreg_db/data/waste/data.yaml +12 -0
- endoreg_db/management/commands/load_base_db_data.py +58 -1
- endoreg_db/management/commands/load_disease_classification_choices_data.py +41 -0
- endoreg_db/management/commands/load_disease_classification_data.py +41 -0
- endoreg_db/management/commands/load_disease_data.py +40 -0
- endoreg_db/management/commands/load_distribution_data.py +66 -0
- endoreg_db/management/commands/load_event_data.py +41 -0
- endoreg_db/management/commands/load_g_play_data.py +113 -0
- endoreg_db/management/commands/load_gender_data.py +44 -0
- endoreg_db/management/commands/load_green_endoscopy_wuerzburg_data.py +133 -0
- endoreg_db/management/commands/load_lab_value_data.py +50 -0
- endoreg_db/management/commands/load_medication_data.py +41 -0
- endoreg_db/management/commands/load_medication_indication_data.py +63 -0
- endoreg_db/management/commands/load_medication_indication_type_data.py +41 -0
- endoreg_db/management/commands/load_medication_intake_time_data.py +41 -0
- endoreg_db/management/commands/load_medication_schedule_data.py +55 -0
- endoreg_db/migrations/0025_event_alter_rawpdffile_file_patientevent.py +42 -0
- endoreg_db/migrations/0026_disease_diseaseclassification_and_more.py +166 -0
- endoreg_db/migrations/0027_labvalue_abbreviation_labvalue_default_normal_range_and_more.py +38 -0
- endoreg_db/migrations/0028_alter_unit_abbreviation.py +18 -0
- endoreg_db/migrations/0029_medicationintaketime_and_more.py +75 -0
- endoreg_db/migrations/0030_medicationindicationtype_and_more.py +101 -0
- endoreg_db/migrations/0031_rename_adapt_to_liver_function_medication_adapt_to_age_and_more.py +38 -0
- endoreg_db/migrations/0032_alter_medicationschedule_therapy_duration_d.py +18 -0
- endoreg_db/migrations/0033_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0034_alter_rawpdffile_file.py +20 -0
- endoreg_db/migrations/0035_alter_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0036_alter_rawpdffile_file.py +20 -0
- endoreg_db/migrations/0037_alter_medicationindication_sources.py +18 -0
- endoreg_db/migrations/0038_emissionfactor_material_product_productgroup_and_more.py +164 -0
- endoreg_db/migrations/0039_referenceproduct_name.py +19 -0
- endoreg_db/migrations/0040_quizanswertype_quizquestiontype_quizquestion_and_more.py +50 -0
- endoreg_db/migrations/0041_gender_patientmedication_medication_indication_and_more.py +40 -0
- endoreg_db/migrations/0042_casetemplateruletype_casetemplaterulevalue_and_more.py +74 -0
- endoreg_db/migrations/0043_casetemplatetype_name_de_casetemplatetype_name_en.py +23 -0
- endoreg_db/migrations/0044_casetemplateruletype_name_de_and_more.py +23 -0
- endoreg_db/migrations/0045_casetemplaterulevalue_value_type.py +19 -0
- endoreg_db/migrations/0046_casetemplaterulevalue_target_field.py +18 -0
- endoreg_db/migrations/0047_casetemplaterule_target_model.py +18 -0
- endoreg_db/migrations/0048_remove_casetemplaterule_chained_rules_and_more.py +22 -0
- endoreg_db/migrations/0049_remove_casetemplaterule_rule_values.py +17 -0
- endoreg_db/migrations/0050_casetemplaterule_rule_values.py +18 -0
- endoreg_db/migrations/0051_remove_casetemplaterule_calling_rules_and_more.py +27 -0
- endoreg_db/migrations/0052_rename_case_template_type_casetemplate_template_type.py +18 -0
- endoreg_db/migrations/0053_patientlabsampletype_patientlabsample_and_more.py +38 -0
- endoreg_db/migrations/0054_multiplecategoricalvaluedistribution_and_more.py +69 -0
- endoreg_db/migrations/0055_remove_casetemplaterule_rule_values_and_more.py +59 -0
- endoreg_db/migrations/0056_datevaluedistribution_and_more.py +32 -0
- endoreg_db/migrations/0057_remove_datevaluedistribution_max_date_and_more.py +72 -0
- endoreg_db/migrations/0058_datevaluedistribution_description_and_more.py +28 -0
- endoreg_db/migrations/0059_casetemplaterule_rule_values.py +18 -0
- endoreg_db/migrations/0060_labvalue__default_date_value_distribution_and_more.py +44 -0
- endoreg_db/migrations/0061_remove_patientlabvalue_date_patientlabvalue_datetime.py +24 -0
- endoreg_db/migrations/0062_labvalue_numeric_precision.py +18 -0
- endoreg_db/migrations/0063_alter_labvalue_numeric_precision.py +18 -0
- endoreg_db/migrations/0064_casetemplaterule_extra_parameters_and_more.py +23 -0
- endoreg_db/migrations/0065_rename__date_value_distribution_casetemplaterule_date_value_distribution_and_more.py +58 -0
- endoreg_db/migrations/0066_alter_patientlabvalue_patient_and_more.py +29 -0
- endoreg_db/migrations/0067_alter_medicationindication_indication_type.py +19 -0
- endoreg_db/models/__init__.py +28 -11
- endoreg_db/models/case_template/__init__.py +6 -0
- endoreg_db/models/case_template/case_template.py +81 -0
- endoreg_db/models/case_template/case_template_rule.py +276 -0
- endoreg_db/models/case_template/case_template_rule_value.py +73 -0
- endoreg_db/models/case_template/case_template_type.py +28 -0
- endoreg_db/models/center/__init__.py +4 -0
- endoreg_db/models/center/center_product.py +34 -0
- endoreg_db/models/center/center_resource.py +19 -0
- endoreg_db/models/center/center_waste.py +11 -0
- endoreg_db/models/data_file/import_classes/raw_pdf.py +4 -1
- endoreg_db/models/data_file/metadata/video_meta.py +4 -3
- endoreg_db/models/disease.py +56 -0
- endoreg_db/models/emission/__init__.py +1 -0
- endoreg_db/models/emission/emission_factor.py +20 -0
- endoreg_db/models/event.py +22 -0
- endoreg_db/models/information_source.py +7 -0
- endoreg_db/models/laboratory/__init__.py +1 -0
- endoreg_db/models/laboratory/lab_value.py +102 -0
- endoreg_db/models/medication/__init__.py +1 -0
- endoreg_db/models/medication/medication.py +148 -0
- endoreg_db/models/other/__init__.py +5 -0
- endoreg_db/models/other/distribution.py +215 -0
- endoreg_db/models/other/material.py +16 -0
- endoreg_db/models/other/resource.py +18 -0
- endoreg_db/models/other/transport_route.py +21 -0
- endoreg_db/models/other/waste.py +20 -0
- endoreg_db/models/persons/__init__.py +3 -2
- endoreg_db/models/persons/gender.py +22 -0
- endoreg_db/models/persons/patient/__init__.py +8 -0
- endoreg_db/models/persons/patient/case/__init__.py +0 -0
- endoreg_db/models/persons/patient/case/case.py +30 -0
- endoreg_db/models/persons/patient/patient.py +216 -0
- endoreg_db/models/persons/patient/patient_disease.py +16 -0
- endoreg_db/models/persons/patient/patient_event.py +22 -0
- endoreg_db/models/persons/patient/patient_lab_sample.py +106 -0
- endoreg_db/models/persons/patient/patient_lab_value.py +176 -0
- endoreg_db/models/persons/patient/patient_medication.py +44 -0
- endoreg_db/models/persons/patient/patient_medication_schedule.py +28 -0
- endoreg_db/models/persons/person.py +1 -4
- endoreg_db/models/persons/portal_user_information.py +0 -2
- endoreg_db/models/product/__init__.py +5 -0
- endoreg_db/models/product/product.py +97 -0
- endoreg_db/models/product/product_group.py +19 -0
- endoreg_db/models/product/product_material.py +24 -0
- endoreg_db/models/product/product_weight.py +26 -0
- endoreg_db/models/product/reference_product.py +99 -0
- endoreg_db/models/quiz/__init__.py +2 -0
- endoreg_db/models/quiz/quiz_answer.py +41 -0
- endoreg_db/models/quiz/quiz_question.py +54 -0
- endoreg_db/models/rules/__init__.py +5 -0
- endoreg_db/models/rules/rule.py +24 -0
- endoreg_db/models/rules/rule_applicator.py +224 -0
- endoreg_db/models/rules/rule_attribute_dtype.py +19 -0
- endoreg_db/models/rules/rule_type.py +22 -0
- endoreg_db/models/rules/ruleset.py +19 -0
- endoreg_db/models/unit.py +6 -4
- endoreg_db/utils/dataloader.py +13 -106
- {endoreg_db-0.3.4.dist-info → endoreg_db-0.3.5.dist-info}/METADATA +1 -1
- endoreg_db-0.3.5.dist-info/RECORD +357 -0
- endoreg_db/models/persons/patient.py +0 -58
- endoreg_db/models.py +0 -3
- endoreg_db-0.3.4.dist-info/RECORD +0 -185
- /endoreg_db/models/{center.py → center/center.py} +0 -0
- {endoreg_db-0.3.4.dist-info → endoreg_db-0.3.5.dist-info}/LICENSE +0 -0
- {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.
|
|
21
|
-
max_length=10, choices=[('Male', 'Male'), ('Female', 'Female'), ('Other', 'Other')],
|
|
22
|
-
blank=True, null=True
|
|
23
|
-
)
|
|
20
|
+
gender = models.ForeignKey("Gender", on_delete=models.SET_NULL, null=True)
|
|
24
21
|
email = models.EmailField(max_length=255, blank=True, null=True)
|
|
25
22
|
phone = models.CharField(max_length=255, blank=True, null=True)
|
|
26
23
|
is_real_person = models.BooleanField(default=True)
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
def sum_weights(product_materials):
|
|
8
|
+
# sum up the weights
|
|
9
|
+
weight = 0
|
|
10
|
+
reference_unit = None
|
|
11
|
+
for product_material in product_materials:
|
|
12
|
+
if not reference_unit:
|
|
13
|
+
reference_unit = product_material.unit
|
|
14
|
+
else:
|
|
15
|
+
assert reference_unit == product_material.unit, "ProductMaterial units do not match"
|
|
16
|
+
weight += product_material.quantity
|
|
17
|
+
|
|
18
|
+
return weight, reference_unit
|
|
19
|
+
|
|
20
|
+
def sum_emissions(product_materials):
|
|
21
|
+
# sum up the emissions
|
|
22
|
+
emission = 0
|
|
23
|
+
reference_unit = None
|
|
24
|
+
for product_material in product_materials:
|
|
25
|
+
if not reference_unit:
|
|
26
|
+
reference_unit = product_material.unit
|
|
27
|
+
else:
|
|
28
|
+
assert reference_unit == product_material.unit, "ProductMaterial units do not match"
|
|
29
|
+
emission, emission_unit = product_material.get_emission()
|
|
30
|
+
assert reference_unit == emission_unit, "ProductMaterial units do not match"
|
|
31
|
+
emission += emission
|
|
32
|
+
|
|
33
|
+
return emission, reference_unit
|
|
34
|
+
|
|
35
|
+
class Product(models.Model):
|
|
36
|
+
objects = ProductManager()
|
|
37
|
+
|
|
38
|
+
name = models.CharField(max_length=255)
|
|
39
|
+
name_de = models.CharField(max_length=255, null=True)
|
|
40
|
+
name_en = models.CharField(max_length=255, null=True)
|
|
41
|
+
|
|
42
|
+
transport_route = models.ForeignKey("TransportRoute", on_delete=models.SET_NULL, null=True)
|
|
43
|
+
product_group = models.ForeignKey("ProductGroup", on_delete=models.SET_NULL, null=True)
|
|
44
|
+
|
|
45
|
+
def natural_key(self):
|
|
46
|
+
return (self.name,)
|
|
47
|
+
|
|
48
|
+
def __str__(self):
|
|
49
|
+
return self.name
|
|
50
|
+
|
|
51
|
+
def get_product_weight(self):
|
|
52
|
+
# check if there is a product material weight
|
|
53
|
+
from .product_material import ProductMaterial
|
|
54
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
55
|
+
if product_materials:
|
|
56
|
+
return self.get_product_material_weight()
|
|
57
|
+
|
|
58
|
+
# check if there is a product weight
|
|
59
|
+
#TODO
|
|
60
|
+
|
|
61
|
+
def get_package_weight(self):
|
|
62
|
+
# check if there is a package material weight
|
|
63
|
+
from .product_material import ProductMaterial
|
|
64
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
65
|
+
if product_materials:
|
|
66
|
+
return self.get_package_material_weight()
|
|
67
|
+
|
|
68
|
+
# check if there is a package weight
|
|
69
|
+
#TODO
|
|
70
|
+
|
|
71
|
+
def get_product_material_weight(self):
|
|
72
|
+
# get all materials with component == "product"
|
|
73
|
+
from .product_material import ProductMaterial
|
|
74
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
75
|
+
|
|
76
|
+
return sum_weights(product_materials)
|
|
77
|
+
|
|
78
|
+
def get_package_material_weight(self):
|
|
79
|
+
# get all materials with component == "package"
|
|
80
|
+
from .product_material import ProductMaterial
|
|
81
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
82
|
+
|
|
83
|
+
return sum_weights(product_materials)
|
|
84
|
+
|
|
85
|
+
def get_product_material_emission(self):
|
|
86
|
+
# get all materials with component == "product"
|
|
87
|
+
from .product_material import ProductMaterial
|
|
88
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="product")
|
|
89
|
+
|
|
90
|
+
return sum_emissions(product_materials)
|
|
91
|
+
|
|
92
|
+
def get_package_material_emission(self):
|
|
93
|
+
# get all materials with component == "package"
|
|
94
|
+
from .product_material import ProductMaterial
|
|
95
|
+
product_materials = ProductMaterial.objects.filter(product=self, component="package")
|
|
96
|
+
|
|
97
|
+
return sum_emissions(product_materials)
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductGroupManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
class ProductGroup(models.Model):
|
|
8
|
+
objects = ProductGroupManager()
|
|
9
|
+
|
|
10
|
+
name = models.CharField(max_length=255)
|
|
11
|
+
name_de = models.CharField(max_length=255, null=True)
|
|
12
|
+
name_en = models.CharField(max_length=255, null=True)
|
|
13
|
+
|
|
14
|
+
def natural_key(self):
|
|
15
|
+
return (self.name,)
|
|
16
|
+
|
|
17
|
+
def __str__(self):
|
|
18
|
+
return self.name
|
|
19
|
+
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class ProductMaterial(models.Model):
|
|
4
|
+
component = models.CharField(max_length=255)
|
|
5
|
+
material = models.ForeignKey("Material", on_delete=models.CASCADE)
|
|
6
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE, related_name="product_materials")
|
|
7
|
+
unit = models.ForeignKey("Unit", on_delete=models.CASCADE)
|
|
8
|
+
quantity = models.FloatField()
|
|
9
|
+
|
|
10
|
+
def get_emission(self):
|
|
11
|
+
from ..emission import EmissionFactor
|
|
12
|
+
emission_factor:EmissionFactor = self.material.emission_factor
|
|
13
|
+
if emission_factor is None:
|
|
14
|
+
raise Exception("No emission factor for material " + self.material.name + " found.")
|
|
15
|
+
|
|
16
|
+
# make sure product_material.unit is the same as emission_factor.unit
|
|
17
|
+
if self.unit != emission_factor.unit:
|
|
18
|
+
raise Exception("Unit mismatch: " + self.unit.name + " != " + emission_factor.unit.name)
|
|
19
|
+
|
|
20
|
+
emmision_value = emission_factor.value * self.quantity
|
|
21
|
+
emission_unit = emission_factor.unit
|
|
22
|
+
return emmision_value, emission_unit
|
|
23
|
+
|
|
24
|
+
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
import pandas as pd
|
|
3
|
+
|
|
4
|
+
class ProductWeightManager(models.Manager):
|
|
5
|
+
def get_by_natural_key(self, product, product_group, weight):
|
|
6
|
+
return self.get(product=product, product_group=product_group, weight=weight)
|
|
7
|
+
|
|
8
|
+
class ProductWeight(models.Model):
|
|
9
|
+
objects = ProductWeightManager()
|
|
10
|
+
|
|
11
|
+
name = models.CharField(max_length=255, null = True)
|
|
12
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE)
|
|
13
|
+
measured = models.FloatField(null=True)
|
|
14
|
+
verified = models.FloatField(null=True)
|
|
15
|
+
manufacturer = models.FloatField(null=True)
|
|
16
|
+
unit = models.ForeignKey("Unit", on_delete=models.SET_NULL, null=True)
|
|
17
|
+
|
|
18
|
+
def get_weight(self):
|
|
19
|
+
if not pd.isnull(self.verified):
|
|
20
|
+
return self.verified
|
|
21
|
+
elif not pd.isnull(self.measured):
|
|
22
|
+
return self.measured
|
|
23
|
+
elif not pd.isnull(self.manufacturer):
|
|
24
|
+
return self.manufacturer
|
|
25
|
+
else:
|
|
26
|
+
return None
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
from typing import List
|
|
3
|
+
|
|
4
|
+
class ReferenceProductManager(models.Manager):
|
|
5
|
+
def get_by_natural_key(self, product_name:str, product_group_name:str):
|
|
6
|
+
return self.get(product__name=product_name, product_group__name=product_group_name)
|
|
7
|
+
|
|
8
|
+
class ReferenceProduct(models.Model):
|
|
9
|
+
name = models.CharField(max_length=255)
|
|
10
|
+
product = models.ForeignKey("Product", on_delete=models.CASCADE)
|
|
11
|
+
product_group = models.OneToOneField("ProductGroup", on_delete=models.CASCADE, related_name="reference_product")
|
|
12
|
+
emission_factor_total = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, blank = True)
|
|
13
|
+
emission_factor_package = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, related_name="reference_product_package")
|
|
14
|
+
emission_factor_product = models.ForeignKey("EmissionFactor", on_delete=models.SET_NULL, null=True, related_name="reference_product_product")
|
|
15
|
+
|
|
16
|
+
objects = ReferenceProductManager()
|
|
17
|
+
|
|
18
|
+
def __str__(self):
|
|
19
|
+
return self.product.name + " (" + self.product_group.name + ")"
|
|
20
|
+
|
|
21
|
+
def set_emission_factors(self):
|
|
22
|
+
from .product import Product
|
|
23
|
+
from .product_material import ProductMaterial
|
|
24
|
+
from ..emission import EmissionFactor
|
|
25
|
+
|
|
26
|
+
product:Product = self.product
|
|
27
|
+
materials = product.product_materials.all()
|
|
28
|
+
emission_factor_name = f"{self.product_group.name}_{product.name}_total_emission_factor"
|
|
29
|
+
emission_factor_package_name = f"{self.product_group.name}_{product.name}_package_emission_factor"
|
|
30
|
+
emission_factor_product_name = f"{self.product_group.name}_{product.name}_product_emission_factor"
|
|
31
|
+
|
|
32
|
+
product_emissions = 0
|
|
33
|
+
package_emissions = 0
|
|
34
|
+
|
|
35
|
+
product_weight, product_weight_unit = product.get_product_material_weight()
|
|
36
|
+
package_weight, package_weight_unit = product.get_package_material_weight()
|
|
37
|
+
product_emission, product_emission_unit = product.get_product_material_emission()
|
|
38
|
+
package_emission, package_emission_unit = product.get_package_material_emission()
|
|
39
|
+
|
|
40
|
+
total_weight = product_weight + package_weight
|
|
41
|
+
total_emission = product_emission + package_emission
|
|
42
|
+
|
|
43
|
+
reference_unit = product_weight_unit
|
|
44
|
+
assert reference_unit == package_weight_unit, "Package weight units do not match"
|
|
45
|
+
assert reference_unit == product_emission_unit, "Product emission units do not match"
|
|
46
|
+
assert reference_unit == package_emission_unit, "Package emission units do not match"
|
|
47
|
+
|
|
48
|
+
product_emission_factor_value = product_emission / product_weight
|
|
49
|
+
package_emission_factor_value = package_emission / package_weight
|
|
50
|
+
total_emission_factor_value = total_emission / total_weight
|
|
51
|
+
|
|
52
|
+
emission_factor, created = EmissionFactor.objects.get_or_create(
|
|
53
|
+
name=emission_factor_name,
|
|
54
|
+
defaults={
|
|
55
|
+
"name": emission_factor_name,
|
|
56
|
+
"value": total_emission_factor_value,
|
|
57
|
+
"unit": reference_unit
|
|
58
|
+
}
|
|
59
|
+
)
|
|
60
|
+
self.emission_factor_total = emission_factor
|
|
61
|
+
|
|
62
|
+
emission_factor_package, created = EmissionFactor.objects.get_or_create(
|
|
63
|
+
name=emission_factor_package_name,
|
|
64
|
+
defaults={
|
|
65
|
+
"name": emission_factor_package_name,
|
|
66
|
+
"value": package_emission_factor_value,
|
|
67
|
+
"unit": reference_unit
|
|
68
|
+
}
|
|
69
|
+
)
|
|
70
|
+
self.emission_factor_package = emission_factor_package
|
|
71
|
+
|
|
72
|
+
emission_factor_product, created = EmissionFactor.objects.get_or_create(
|
|
73
|
+
name=emission_factor_product_name,
|
|
74
|
+
defaults={
|
|
75
|
+
"name": emission_factor_product_name,
|
|
76
|
+
"value": product_emission_factor_value,
|
|
77
|
+
"unit": reference_unit
|
|
78
|
+
}
|
|
79
|
+
)
|
|
80
|
+
self.emission_factor_product = emission_factor_product
|
|
81
|
+
|
|
82
|
+
self.save()
|
|
83
|
+
|
|
84
|
+
def get_emission_factor(self, component:str):
|
|
85
|
+
# check if emission_factor_total exists:
|
|
86
|
+
if self.emission_factor_total is None:
|
|
87
|
+
self.set_emission_factors()
|
|
88
|
+
|
|
89
|
+
if component == "total":
|
|
90
|
+
return self.emission_factor_total
|
|
91
|
+
elif component == "package":
|
|
92
|
+
return self.emission_factor_package
|
|
93
|
+
elif component == "product":
|
|
94
|
+
return self.emission_factor_product
|
|
95
|
+
else:
|
|
96
|
+
raise Exception("Unknown component: " + component)
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
|
|
@@ -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,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
|
+
|