endoreg-db 0.6.0__py3-none-any.whl → 0.6.2__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.
Potentially problematic release.
This version of endoreg-db might be problematic. Click here for more details.
- endoreg_db/case_generator/__init__.py +0 -0
- endoreg_db/case_generator/case_generator.py +159 -0
- endoreg_db/case_generator/lab_sample_factory.py +33 -0
- endoreg_db/case_generator/utils.py +30 -0
- endoreg_db/data/__init__.py +118 -0
- endoreg_db/data/agl_service/data.yaml +19 -0
- endoreg_db/data/ai_model/data.yaml +7 -0
- endoreg_db/data/ai_model_label/label/data.yaml +88 -0
- endoreg_db/data/ai_model_label/label-set/data.yaml +21 -0
- endoreg_db/data/ai_model_label/label-type/data.yaml +7 -0
- endoreg_db/data/ai_model_meta/default_multilabel_classification.yaml +5 -0
- endoreg_db/data/ai_model_type/data.yaml +7 -0
- endoreg_db/data/ai_model_video_segmentation_label/base_segmentation.yaml +176 -0
- endoreg_db/data/ai_model_video_segmentation_labelset/data.yaml +20 -0
- 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/.init +0 -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 +90 -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/contraindication/bleeding.yaml +11 -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/multiple_categorical/.init +0 -0
- endoreg_db/data/distribution/numeric/data.yaml +14 -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/endoscope/data.yaml +93 -0
- endoreg_db/data/endoscope_type/data.yaml +11 -0
- endoreg_db/data/endoscopy_processor/data.yaml +47 -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 +66 -0
- endoreg_db/data/examination/time/data.yaml +48 -0
- endoreg_db/data/examination/time-type/data.yaml +8 -0
- endoreg_db/data/examination/type/data.yaml +5 -0
- endoreg_db/data/examination_indication/endoscopy.yaml +8 -0
- endoreg_db/data/examination_indication_classification/endoscopy.yaml +8 -0
- endoreg_db/data/examination_indication_classification_choice/endoscopy.yaml +101 -0
- endoreg_db/data/finding/data.yaml +141 -0
- endoreg_db/data/finding_intervention/endoscopy.yaml +138 -0
- endoreg_db/data/finding_intervention_type/endoscopy.yaml +15 -0
- endoreg_db/data/finding_location_classification/colonoscopy.yaml +46 -0
- endoreg_db/data/finding_location_classification_choice/colonoscopy.yaml +240 -0
- endoreg_db/data/finding_morphology_classification/colonoscopy.yaml +48 -0
- endoreg_db/data/finding_morphology_classification_choice/colon_lesion_circularity_default.yaml +34 -0
- endoreg_db/data/finding_morphology_classification_choice/colon_lesion_nice.yaml +20 -0
- endoreg_db/data/finding_morphology_classification_choice/colon_lesion_paris.yaml +65 -0
- endoreg_db/data/finding_morphology_classification_choice/colon_lesion_planarity_default.yaml +56 -0
- endoreg_db/data/finding_morphology_classification_choice/colon_lesion_surface_intact_default.yaml +39 -0
- endoreg_db/data/finding_morphology_classification_choice/colonoscopy_size.yaml +57 -0
- endoreg_db/data/finding_morphology_classification_type/colonoscopy.yaml +79 -0
- endoreg_db/data/finding_type/data.yaml +30 -0
- endoreg_db/data/gender/data.yaml +35 -0
- endoreg_db/data/information_source/data.yaml +30 -0
- endoreg_db/data/information_source/medication.yaml +6 -0
- endoreg_db/data/lab_value/cardiac_enzymes.yaml +37 -0
- endoreg_db/data/lab_value/coagulation.yaml +54 -0
- endoreg_db/data/lab_value/electrolytes.yaml +228 -0
- endoreg_db/data/lab_value/gastrointestinal_function.yaml +133 -0
- endoreg_db/data/lab_value/hematology.yaml +184 -0
- endoreg_db/data/lab_value/hormones.yaml +59 -0
- endoreg_db/data/lab_value/lipids.yaml +53 -0
- endoreg_db/data/lab_value/misc.yaml +33 -0
- endoreg_db/data/lab_value/renal_function.yaml +12 -0
- endoreg_db/data/log_type/data.yaml +57 -0
- endoreg_db/data/lx_client_tag/base.yaml +54 -0
- endoreg_db/data/lx_client_type/base.yaml +30 -0
- endoreg_db/data/lx_permission/base.yaml +24 -0
- endoreg_db/data/lx_permission/endoreg.yaml +52 -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 +115 -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/names_first/first_names.yaml +51 -0
- endoreg_db/data/names_last/last_names.yaml +51 -0
- endoreg_db/data/network_device/data.yaml +59 -0
- endoreg_db/data/network_device_type/data.yaml +12 -0
- endoreg_db/data/organ/data.yaml +29 -0
- endoreg_db/data/patient_lab_sample_type/generic.yaml +6 -0
- endoreg_db/data/pdf_type/data.yaml +29 -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/profession/data.yaml +70 -0
- endoreg_db/data/reference_product/green_endoscopy_dashboard_ReferenceProduct.yaml +55 -0
- endoreg_db/data/report_reader_flag/ukw-examination-generic.yaml +30 -0
- endoreg_db/data/report_reader_flag/ukw-histology-generic.yaml +19 -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/data.yaml +17 -0
- endoreg_db/data/unit/length.yaml +31 -0
- 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 +35 -0
- endoreg_db/data/unit/weight.yaml +38 -0
- endoreg_db/data/waste/data.yaml +12 -0
- endoreg_db/forms/__init__.py +5 -0
- endoreg_db/forms/examination_form.py +11 -0
- endoreg_db/forms/patient_finding_intervention_form.py +19 -0
- endoreg_db/forms/patient_form.py +26 -0
- endoreg_db/forms/questionnaires/__init__.py +1 -0
- endoreg_db/forms/questionnaires/tto_questionnaire.py +23 -0
- endoreg_db/forms/settings/__init__.py +8 -0
- endoreg_db/forms/unit.py +6 -0
- endoreg_db/management/__init__.py +0 -0
- endoreg_db/management/commands/__init__.py +0 -0
- endoreg_db/management/commands/_load_model_template.py +41 -0
- endoreg_db/management/commands/delete_all.py +18 -0
- endoreg_db/management/commands/fetch_legacy_image_dataset.py +32 -0
- endoreg_db/management/commands/fix_auth_permission.py +20 -0
- endoreg_db/management/commands/load_active_model_data.py +45 -0
- endoreg_db/management/commands/load_ai_model_data.py +79 -0
- endoreg_db/management/commands/load_ai_model_label_data.py +59 -0
- endoreg_db/management/commands/load_base_db_data.py +178 -0
- endoreg_db/management/commands/load_center_data.py +43 -0
- endoreg_db/management/commands/load_contraindication_data.py +41 -0
- 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 +62 -0
- endoreg_db/management/commands/load_distribution_data.py +66 -0
- endoreg_db/management/commands/load_endoscope_data.py +68 -0
- endoreg_db/management/commands/load_event_data.py +41 -0
- endoreg_db/management/commands/load_examination_data.py +75 -0
- endoreg_db/management/commands/load_examination_indication_data.py +65 -0
- endoreg_db/management/commands/load_finding_data.py +171 -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_information_source.py +45 -0
- endoreg_db/management/commands/load_lab_value_data.py +50 -0
- endoreg_db/management/commands/load_logging_data.py +39 -0
- endoreg_db/management/commands/load_lx_data.py +64 -0
- endoreg_db/management/commands/load_medication_data.py +103 -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/management/commands/load_name_data.py +37 -0
- endoreg_db/management/commands/load_network_data.py +57 -0
- endoreg_db/management/commands/load_organ_data.py +43 -0
- endoreg_db/management/commands/load_pdf_type_data.py +61 -0
- endoreg_db/management/commands/load_profession_data.py +44 -0
- endoreg_db/management/commands/load_report_reader_flag_data.py +46 -0
- endoreg_db/management/commands/load_unit_data.py +46 -0
- endoreg_db/management/commands/load_user_groups.py +28 -0
- endoreg_db/management/commands/register_ai_model.py +64 -0
- endoreg_db/management/commands/reset_celery_schedule.py +9 -0
- endoreg_db/migrations/0001_initial.py +2045 -0
- endoreg_db/migrations/0002_alter_frame_image_alter_rawframe_image.py +23 -0
- endoreg_db/migrations/0003_alter_frame_image_alter_rawframe_image.py +23 -0
- endoreg_db/migrations/0004_alter_rawvideofile_file_alter_video_file.py +25 -0
- endoreg_db/migrations/0005_rawvideofile_frame_count_and_more.py +33 -0
- endoreg_db/migrations/0006_frame_extracted_rawframe_extracted.py +23 -0
- endoreg_db/migrations/0007_rename_pseudo_patient_video_patient_and_more.py +24 -0
- endoreg_db/migrations/0008_remove_reportfile_patient_examination_and_more.py +48 -0
- endoreg_db/migrations/__init__.py +0 -0
- endoreg_db/models/__init__.py +376 -0
- endoreg_db/models/ai_model/__init__.py +4 -0
- endoreg_db/models/ai_model/active_model.py +9 -0
- endoreg_db/models/ai_model/ai_model.py +103 -0
- endoreg_db/models/ai_model/lightning/__init__.py +3 -0
- endoreg_db/models/ai_model/lightning/inference_dataset.py +53 -0
- endoreg_db/models/ai_model/lightning/multilabel_classification_net.py +155 -0
- endoreg_db/models/ai_model/lightning/postprocess.py +53 -0
- endoreg_db/models/ai_model/lightning/predict.py +172 -0
- endoreg_db/models/ai_model/lightning/prediction_visualizer.py +55 -0
- endoreg_db/models/ai_model/lightning/preprocess.py +68 -0
- endoreg_db/models/ai_model/lightning/run_visualizer.py +21 -0
- endoreg_db/models/ai_model/model_meta.py +250 -0
- endoreg_db/models/ai_model/model_type.py +36 -0
- endoreg_db/models/ai_model/utils.py +8 -0
- endoreg_db/models/annotation/__init__.py +32 -0
- endoreg_db/models/annotation/anonymized_image_annotation.py +115 -0
- endoreg_db/models/annotation/binary_classification_annotation_task.py +117 -0
- endoreg_db/models/annotation/image_classification.py +86 -0
- endoreg_db/models/annotation/video_segmentation_annotation.py +52 -0
- endoreg_db/models/annotation/video_segmentation_labelset.py +20 -0
- endoreg_db/models/case/__init__.py +1 -0
- endoreg_db/models/case/case.py +34 -0
- endoreg_db/models/case_template/__init__.py +15 -0
- endoreg_db/models/case_template/case_template.py +125 -0
- endoreg_db/models/case_template/case_template_rule.py +276 -0
- endoreg_db/models/case_template/case_template_rule_value.py +88 -0
- endoreg_db/models/case_template/case_template_type.py +28 -0
- endoreg_db/models/center/__init__.py +11 -0
- endoreg_db/models/center/center.py +51 -0
- endoreg_db/models/center/center_product.py +33 -0
- endoreg_db/models/center/center_resource.py +33 -0
- endoreg_db/models/center/center_waste.py +16 -0
- endoreg_db/models/contraindication/__init__.py +21 -0
- endoreg_db/models/data_file/__init__.py +39 -0
- endoreg_db/models/data_file/base_classes/__init__.py +7 -0
- endoreg_db/models/data_file/base_classes/abstract_frame.py +100 -0
- endoreg_db/models/data_file/base_classes/abstract_pdf.py +136 -0
- endoreg_db/models/data_file/base_classes/abstract_video.py +807 -0
- endoreg_db/models/data_file/base_classes/frame_helpers.py +17 -0
- endoreg_db/models/data_file/base_classes/prepare_bulk_frames.py +19 -0
- endoreg_db/models/data_file/base_classes/utils.py +80 -0
- endoreg_db/models/data_file/frame.py +29 -0
- endoreg_db/models/data_file/import_classes/__init__.py +18 -0
- endoreg_db/models/data_file/import_classes/processing_functions/__init__.py +35 -0
- endoreg_db/models/data_file/import_classes/processing_functions/pdf.py +28 -0
- endoreg_db/models/data_file/import_classes/processing_functions/video.py +260 -0
- endoreg_db/models/data_file/import_classes/raw_pdf.py +260 -0
- endoreg_db/models/data_file/import_classes/raw_video.py +288 -0
- endoreg_db/models/data_file/metadata/__init__.py +13 -0
- endoreg_db/models/data_file/metadata/pdf_meta.py +74 -0
- endoreg_db/models/data_file/metadata/sensitive_meta.py +290 -0
- endoreg_db/models/data_file/metadata/video_meta.py +199 -0
- endoreg_db/models/data_file/report_file.py +56 -0
- endoreg_db/models/data_file/video/__init__.py +11 -0
- endoreg_db/models/data_file/video/import_meta.py +25 -0
- endoreg_db/models/data_file/video/video.py +196 -0
- endoreg_db/models/data_file/video_segment.py +214 -0
- endoreg_db/models/disease.py +79 -0
- endoreg_db/models/emission/__init__.py +5 -0
- endoreg_db/models/emission/emission_factor.py +85 -0
- endoreg_db/models/event.py +73 -0
- endoreg_db/models/examination/__init__.py +9 -0
- endoreg_db/models/examination/examination.py +67 -0
- endoreg_db/models/examination/examination_indication.py +170 -0
- endoreg_db/models/examination/examination_time.py +53 -0
- endoreg_db/models/examination/examination_time_type.py +48 -0
- endoreg_db/models/examination/examination_type.py +40 -0
- endoreg_db/models/finding/__init__.py +11 -0
- endoreg_db/models/finding/finding.py +75 -0
- endoreg_db/models/finding/finding_intervention.py +60 -0
- endoreg_db/models/finding/finding_location_classification.py +94 -0
- endoreg_db/models/finding/finding_morphology_classification.py +89 -0
- endoreg_db/models/finding/finding_type.py +22 -0
- endoreg_db/models/hardware/__init__.py +2 -0
- endoreg_db/models/hardware/endoscope.py +60 -0
- endoreg_db/models/hardware/endoscopy_processor.py +155 -0
- endoreg_db/models/information_source.py +29 -0
- endoreg_db/models/label/__init__.py +1 -0
- endoreg_db/models/label/label.py +112 -0
- endoreg_db/models/laboratory/__init__.py +1 -0
- endoreg_db/models/laboratory/lab_value.py +111 -0
- endoreg_db/models/logging/__init__.py +11 -0
- endoreg_db/models/logging/agl_service.py +19 -0
- endoreg_db/models/logging/base.py +22 -0
- endoreg_db/models/logging/log_type.py +23 -0
- endoreg_db/models/logging/network_device.py +27 -0
- endoreg_db/models/lx/__init__.py +4 -0
- endoreg_db/models/lx/client.py +57 -0
- endoreg_db/models/lx/identity.py +34 -0
- endoreg_db/models/lx/permission.py +18 -0
- endoreg_db/models/lx/user.py +16 -0
- endoreg_db/models/medication/__init__.py +19 -0
- endoreg_db/models/medication/medication.py +33 -0
- endoreg_db/models/medication/medication_indication.py +50 -0
- endoreg_db/models/medication/medication_indication_type.py +34 -0
- endoreg_db/models/medication/medication_intake_time.py +26 -0
- endoreg_db/models/medication/medication_schedule.py +37 -0
- endoreg_db/models/network/__init__.py +9 -0
- endoreg_db/models/network/agl_service.py +38 -0
- endoreg_db/models/network/network_device.py +58 -0
- endoreg_db/models/network/network_device_type.py +23 -0
- endoreg_db/models/organ/__init__.py +38 -0
- endoreg_db/models/other/__init__.py +23 -0
- endoreg_db/models/other/distribution/__init__.py +44 -0
- endoreg_db/models/other/distribution/base_value_distribution.py +20 -0
- endoreg_db/models/other/distribution/date_value_distribution.py +91 -0
- endoreg_db/models/other/distribution/multiple_categorical_value_distribution.py +32 -0
- endoreg_db/models/other/distribution/numeric_value_distribution.py +97 -0
- endoreg_db/models/other/distribution/single_categorical_value_distribution.py +22 -0
- endoreg_db/models/other/distribution.py +5 -0
- endoreg_db/models/other/material.py +20 -0
- endoreg_db/models/other/resource.py +18 -0
- endoreg_db/models/other/transport_route.py +22 -0
- endoreg_db/models/other/waste.py +20 -0
- endoreg_db/models/patient/__init__.py +24 -0
- endoreg_db/models/patient/patient_examination.py +182 -0
- endoreg_db/models/patient/patient_finding.py +143 -0
- endoreg_db/models/patient/patient_finding_intervention.py +26 -0
- endoreg_db/models/patient/patient_finding_location.py +120 -0
- endoreg_db/models/patient/patient_finding_morphology.py +166 -0
- endoreg_db/models/permissions/__init__.py +44 -0
- endoreg_db/models/persons/__init__.py +34 -0
- endoreg_db/models/persons/examiner/__init__.py +2 -0
- endoreg_db/models/persons/examiner/examiner.py +60 -0
- endoreg_db/models/persons/examiner/examiner_type.py +2 -0
- endoreg_db/models/persons/first_name.py +18 -0
- endoreg_db/models/persons/gender.py +22 -0
- endoreg_db/models/persons/last_name.py +20 -0
- endoreg_db/models/persons/patient/__init__.py +8 -0
- endoreg_db/models/persons/patient/patient.py +389 -0
- endoreg_db/models/persons/patient/patient_disease.py +22 -0
- endoreg_db/models/persons/patient/patient_event.py +52 -0
- endoreg_db/models/persons/patient/patient_examination_indication.py +32 -0
- endoreg_db/models/persons/patient/patient_lab_sample.py +108 -0
- endoreg_db/models/persons/patient/patient_lab_value.py +197 -0
- endoreg_db/models/persons/patient/patient_medication.py +59 -0
- endoreg_db/models/persons/patient/patient_medication_schedule.py +88 -0
- endoreg_db/models/persons/person.py +31 -0
- endoreg_db/models/persons/portal_user_information.py +27 -0
- endoreg_db/models/prediction/__init__.py +8 -0
- endoreg_db/models/prediction/image_classification.py +51 -0
- endoreg_db/models/prediction/video_prediction_meta.py +306 -0
- endoreg_db/models/product/__init__.py +14 -0
- endoreg_db/models/product/product.py +110 -0
- endoreg_db/models/product/product_group.py +27 -0
- endoreg_db/models/product/product_material.py +28 -0
- endoreg_db/models/product/product_weight.py +38 -0
- endoreg_db/models/product/reference_product.py +115 -0
- endoreg_db/models/questionnaires/__init__.py +114 -0
- endoreg_db/models/quiz/__init__.py +9 -0
- endoreg_db/models/quiz/quiz_answer.py +41 -0
- endoreg_db/models/quiz/quiz_question.py +54 -0
- endoreg_db/models/report_reader/__init__.py +7 -0
- endoreg_db/models/report_reader/report_reader_config.py +53 -0
- endoreg_db/models/report_reader/report_reader_flag.py +20 -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 +22 -0
- endoreg_db/queries/__init__.py +5 -0
- endoreg_db/queries/annotations/__init__.py +3 -0
- endoreg_db/queries/annotations/legacy.py +158 -0
- endoreg_db/queries/get/__init__.py +6 -0
- endoreg_db/queries/get/annotation.py +0 -0
- endoreg_db/queries/get/center.py +42 -0
- endoreg_db/queries/get/model.py +13 -0
- endoreg_db/queries/get/patient.py +14 -0
- endoreg_db/queries/get/patient_examination.py +20 -0
- endoreg_db/queries/get/prediction.py +0 -0
- endoreg_db/queries/get/report_file.py +33 -0
- endoreg_db/queries/get/video.py +31 -0
- endoreg_db/queries/get/video_import_meta.py +0 -0
- endoreg_db/queries/get/video_prediction_meta.py +0 -0
- endoreg_db/queries/sanity/__init_.py +0 -0
- endoreg_db/serializers/__init__.py +10 -0
- endoreg_db/serializers/ai_model.py +19 -0
- endoreg_db/serializers/annotation.py +14 -0
- endoreg_db/serializers/center.py +11 -0
- endoreg_db/serializers/examination.py +33 -0
- endoreg_db/serializers/frame.py +9 -0
- endoreg_db/serializers/hardware.py +21 -0
- endoreg_db/serializers/label.py +22 -0
- endoreg_db/serializers/patient.py +33 -0
- endoreg_db/serializers/prediction.py +10 -0
- endoreg_db/serializers/raw_video_meta_validation.py +13 -0
- endoreg_db/serializers/report_file.py +7 -0
- endoreg_db/serializers/video.py +20 -0
- endoreg_db/serializers/video_segmentation.py +492 -0
- endoreg_db/templates/admin/patient_finding_intervention.html +253 -0
- endoreg_db/templates/admin/start_examination.html +12 -0
- endoreg_db/templates/timeline.html +176 -0
- endoreg_db/utils/__init__.py +36 -0
- endoreg_db/utils/cropping.py +29 -0
- endoreg_db/utils/dataloader.py +118 -0
- endoreg_db/utils/dates.py +39 -0
- endoreg_db/utils/file_operations.py +30 -0
- endoreg_db/utils/hashs.py +152 -0
- endoreg_db/utils/legacy_ocr.py +201 -0
- endoreg_db/utils/names.py +74 -0
- endoreg_db/utils/ocr.py +190 -0
- endoreg_db/utils/parse_and_generate_yaml.py +46 -0
- endoreg_db/utils/pydantic_models/__init__.py +6 -0
- endoreg_db/utils/pydantic_models/db_config.py +57 -0
- endoreg_db/utils/uuid.py +4 -0
- endoreg_db/utils/validate_endo_roi.py +19 -0
- endoreg_db/utils/validate_subcategory_dict.py +91 -0
- endoreg_db/utils/video/__init__.py +13 -0
- endoreg_db/utils/video/extract_frames.py +121 -0
- endoreg_db/utils/video/transcode_videofile.py +111 -0
- endoreg_db/views/__init__.py +2 -0
- endoreg_db/views/csrf.py +7 -0
- endoreg_db/views/patient_views.py +90 -0
- endoreg_db/views/raw_video_meta_validation_views.py +38 -0
- endoreg_db/views/report_views.py +96 -0
- endoreg_db/views/video_segmentation_views.py +149 -0
- endoreg_db/views/views_for_timeline.py +46 -0
- {endoreg_db-0.6.0.dist-info → endoreg_db-0.6.2.dist-info}/METADATA +14 -4
- endoreg_db-0.6.2.dist-info/RECORD +420 -0
- {endoreg_db-0.6.0.dist-info → endoreg_db-0.6.2.dist-info}/WHEEL +1 -2
- endoreg_db-0.6.0.dist-info/RECORD +0 -11
- endoreg_db-0.6.0.dist-info/top_level.txt +0 -1
- {endoreg_db-0.6.0.dist-info → endoreg_db-0.6.2.dist-info/licenses}/LICENSE +0 -0
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
YES_NO_CHOICES = (
|
|
4
|
+
('yes', 'Ja'),
|
|
5
|
+
('no', 'Nein'),
|
|
6
|
+
)
|
|
7
|
+
|
|
8
|
+
class TtoQuestionnaire(models.Model):
|
|
9
|
+
|
|
10
|
+
# Patient Information
|
|
11
|
+
patient_name = models.CharField(max_length=255, verbose_name="Identifikation des Patienten (Name)")
|
|
12
|
+
birth_date = models.DateField(verbose_name="Geburtsdatum")
|
|
13
|
+
|
|
14
|
+
# Base documentation
|
|
15
|
+
emergency_patient = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Notfallpatient/kürzlich untersuchter Patient (Verzicht auf Team-Time-Out möglich)")
|
|
16
|
+
consent_signed = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Einverständniserklärung unterschrieben (Arzt, Patient)")
|
|
17
|
+
documents_present = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Alle Dokumente liegen vor (Labor, Befunde, etc.)")
|
|
18
|
+
communication_possible = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Kommunikation mit Patient möglich")
|
|
19
|
+
work_incapacity_certificate = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Arbeitsunfähigkeitsbescheinigung")
|
|
20
|
+
|
|
21
|
+
# priority items
|
|
22
|
+
pregnancy = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Schwangerschaft")
|
|
23
|
+
asa_classification_checked = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="ASA-Klassifikation/Komorbidität geprüft")
|
|
24
|
+
previous_anesthesia_complications = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Komplikationen bei bisherigen Narkosen?")
|
|
25
|
+
last_meal_over_6_hours_ago = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Zeitpunkt letzte Mahlzeit > 6 Stunden")
|
|
26
|
+
allergies = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Allergien, welche?")
|
|
27
|
+
outpatient_accompaniment_post_sedation = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Nur bei ambulanter Vorstellung: Begleitung nach Sedierung?")
|
|
28
|
+
|
|
29
|
+
# Possessions
|
|
30
|
+
dental_prosthesis = models.BooleanField(default=False, verbose_name="Zahnprothese")
|
|
31
|
+
glasses = models.BooleanField(default=False, verbose_name="Brille")
|
|
32
|
+
implants = models.BooleanField(default=False, verbose_name="Implantate")
|
|
33
|
+
hearing_aids = models.BooleanField(default=False, verbose_name="Hörgeräte")
|
|
34
|
+
|
|
35
|
+
|
|
36
|
+
# Medical History
|
|
37
|
+
anticoagulants_ass = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Antikoagulation, ASS")
|
|
38
|
+
blood_pressure_medication = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Blutdruckmedikamente")
|
|
39
|
+
glaucoma = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Glaukom")
|
|
40
|
+
|
|
41
|
+
metal_implants = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Metallimplantate")
|
|
42
|
+
pacemaker_icd = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Herzschrittmacher/ICD")
|
|
43
|
+
|
|
44
|
+
copd = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="COPD")
|
|
45
|
+
liver_cirrhosis = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Leberzirrhose")
|
|
46
|
+
|
|
47
|
+
ibd = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="CED (Chronisch entzündliche Darmerkrankungen)")
|
|
48
|
+
radiation = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Bestrahlung")
|
|
49
|
+
surgeries = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="OP´s")
|
|
50
|
+
|
|
51
|
+
# preflight
|
|
52
|
+
team_introduction = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Teamvorstellung mit Name und Aufgabe")
|
|
53
|
+
instruments_available = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Notwendige Instrumente vorhanden?")
|
|
54
|
+
monitoring_medications_equipment_checked = models.CharField(max_length=3, choices=YES_NO_CHOICES, verbose_name="Monitoring, Medikamente, Equipment zum Atemwegsmanagement zur Verfügung und überprüft?")
|
|
55
|
+
complete_documentation_inclusive_care_notes = models.TextField(verbose_name="Vollständige Dokumentation inklusive Hinweise für Nachsorge")
|
|
56
|
+
|
|
57
|
+
# notes
|
|
58
|
+
notes = models.TextField(verbose_name="Bemerkungen", default = "Keine Bemerkungen")
|
|
59
|
+
|
|
60
|
+
# postflight
|
|
61
|
+
specimens_secured = models.BooleanField(default=False, verbose_name="Histologische Proben gesichert")
|
|
62
|
+
patient_condition_documented = models.TextField(verbose_name="Patientenzustand dokumentiert (je nach Ausgangszustand)")
|
|
63
|
+
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
def __str__(self):
|
|
67
|
+
return f"Endoscopy Questionnaire for {self.patient_name} on {self.birth_date}"
|
|
68
|
+
|
|
69
|
+
def get_id_attributes(self):
|
|
70
|
+
_ = [self.patient_name, self.birth_date]
|
|
71
|
+
return _
|
|
72
|
+
|
|
73
|
+
def get_base_documentation_attributes(self):
|
|
74
|
+
_ = [self.emergency_patient, self.consent_signed, self.documents_present, self.communication_possible, self.work_incapacity_certificate]
|
|
75
|
+
return _
|
|
76
|
+
|
|
77
|
+
def get_priority_items_attributes(self):
|
|
78
|
+
_ = [self.pregnancy, self.asa_classification_checked, self.previous_anesthesia_complications, self.last_meal_over_6_hours_ago, self.allergies, self.outpatient_accompaniment_post_sedation]
|
|
79
|
+
return _
|
|
80
|
+
|
|
81
|
+
def get_possessions_attributes(self):
|
|
82
|
+
_ = [self.dental_prosthesis, self.glasses, self.implants, self.hearing_aids]
|
|
83
|
+
return _
|
|
84
|
+
|
|
85
|
+
def get_medical_history_attributes(self):
|
|
86
|
+
_ = [
|
|
87
|
+
self.anticoagulants_ass,
|
|
88
|
+
self.blood_pressure_medication,
|
|
89
|
+
self.glaucoma,
|
|
90
|
+
self.metal_implants,
|
|
91
|
+
self.pacemaker_icd,
|
|
92
|
+
self.copd,
|
|
93
|
+
self.liver_cirrhosis,
|
|
94
|
+
self.ibd,
|
|
95
|
+
self.radiation,
|
|
96
|
+
self.surgeries,
|
|
97
|
+
]
|
|
98
|
+
|
|
99
|
+
def get_preflight_attributes(self):
|
|
100
|
+
_ = [
|
|
101
|
+
self.team_introduction,
|
|
102
|
+
self.instruments_available,
|
|
103
|
+
self.monitoring_medications_equipment_checked,
|
|
104
|
+
]
|
|
105
|
+
|
|
106
|
+
def get_note_attributes(self):
|
|
107
|
+
_ = [self.notes]
|
|
108
|
+
|
|
109
|
+
def get_postflight_attributes(self):
|
|
110
|
+
_ = [
|
|
111
|
+
self.specimens_secured,
|
|
112
|
+
self.patient_condition_documented
|
|
113
|
+
]
|
|
114
|
+
|
|
@@ -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,53 @@
|
|
|
1
|
+
# ReportReaderConfig Class
|
|
2
|
+
# Description: This class is used to store the configuration of the ReportReader
|
|
3
|
+
|
|
4
|
+
# PATIENT_INFO_LINE_FLAG = "Patient: "
|
|
5
|
+
# ENDOSCOPE_INFO_LINE_FLAG = "Gerät: "
|
|
6
|
+
# EXAMINER_INFO_LINE_FLAG = "1. Unters.:"
|
|
7
|
+
# CUT_OFF_BELOW_LINE_FLAG = "________________"
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
# CUT_OFF_ABOVE_LINE_FLAGS = [
|
|
11
|
+
# ENDOSCOPE_INFO_LINE_FLAG,
|
|
12
|
+
# EXAMINER_INFO_LINE_FLAG,
|
|
13
|
+
# ]
|
|
14
|
+
|
|
15
|
+
# CUT_OFF_BELOW_LINE_FLAGS = [
|
|
16
|
+
# CUT_OFF_BELOW_LINE_FLAG
|
|
17
|
+
# ]
|
|
18
|
+
|
|
19
|
+
from django.db import models
|
|
20
|
+
from .report_reader_flag import ReportReaderFlag
|
|
21
|
+
from ..center import Center
|
|
22
|
+
from ..data_file import PdfType
|
|
23
|
+
|
|
24
|
+
class ReportReaderConfig(models.Model):
|
|
25
|
+
|
|
26
|
+
locale = models.CharField(default="de_DE", max_length=10)
|
|
27
|
+
first_names = models.ManyToManyField('FirstName', related_name='report_reader_configs')
|
|
28
|
+
last_names = models.ManyToManyField('LastName', related_name='report_reader_configs')
|
|
29
|
+
text_date_format = models.CharField(default = "%d.%m.%Y", max_length=10)
|
|
30
|
+
patient_info_line_flag = models.ForeignKey(ReportReaderFlag, related_name='report_reader_configs_patient_info_line', on_delete=models.CASCADE)
|
|
31
|
+
endoscope_info_line_flag = models.ForeignKey(ReportReaderFlag, related_name='report_reader_configs_endoscope_info_line', on_delete=models.CASCADE)
|
|
32
|
+
examiner_info_line_flag = models.ForeignKey(ReportReaderFlag, related_name='report_reader_configs_examiner_info_line', on_delete=models.CASCADE)
|
|
33
|
+
cut_off_below = models.ManyToManyField(ReportReaderFlag, related_name='report_reader_configs_cut_off_below')
|
|
34
|
+
cut_off_above = models.ManyToManyField(ReportReaderFlag, related_name='report_reader_configs_cut_off_above')
|
|
35
|
+
|
|
36
|
+
def __str__(self):
|
|
37
|
+
return self.locale
|
|
38
|
+
|
|
39
|
+
def update_names_by_center(self, center:Center, save = True):
|
|
40
|
+
self.first_names.set(center.first_names.all())
|
|
41
|
+
self.last_names.set(center.last_names.all())
|
|
42
|
+
if save:
|
|
43
|
+
self.save()
|
|
44
|
+
|
|
45
|
+
def update_flags_by_pdf_type(self, pdf_type:PdfType, save = True):
|
|
46
|
+
self.patient_info_line_flag = pdf_type.patient_info_line_flag
|
|
47
|
+
self.endoscope_info_line_flag = pdf_type.endoscope_info_line_flag
|
|
48
|
+
self.examiner_info_line_flag = pdf_type.examiner_info_line_flag
|
|
49
|
+
self.cut_off_below.set(pdf_type.cut_off_below.all())
|
|
50
|
+
self.cut_off_above.set(pdf_type.cut_off_above.all())
|
|
51
|
+
if save:
|
|
52
|
+
self.save()
|
|
53
|
+
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
# Django model for the report reader flag
|
|
2
|
+
# have name and value
|
|
3
|
+
# name is natural key
|
|
4
|
+
|
|
5
|
+
from django.db import models
|
|
6
|
+
|
|
7
|
+
class ReportReaderFlagManager(models.Manager):
|
|
8
|
+
def get_by_natural_key(self, name):
|
|
9
|
+
return self.get(name=name)
|
|
10
|
+
|
|
11
|
+
class ReportReaderFlag(models.Model):
|
|
12
|
+
objects = ReportReaderFlagManager()
|
|
13
|
+
name = models.CharField(max_length=255, unique=True)
|
|
14
|
+
value = models.CharField(max_length=255)
|
|
15
|
+
|
|
16
|
+
def natural_key(self):
|
|
17
|
+
return (self.name,)
|
|
18
|
+
|
|
19
|
+
def __str__(self):
|
|
20
|
+
return self.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
|
+
|
|
@@ -0,0 +1,224 @@
|
|
|
1
|
+
from .rule import Rule
|
|
2
|
+
import random
|
|
3
|
+
from django.core.exceptions import ValidationError
|
|
4
|
+
|
|
5
|
+
class RuleApplicator:
|
|
6
|
+
"""
|
|
7
|
+
A class to apply different types of rules to a Django model instance based on the Rule configuration.
|
|
8
|
+
"""
|
|
9
|
+
|
|
10
|
+
def get_rule_by_name(self, rule_name):
|
|
11
|
+
"""
|
|
12
|
+
A helper method to fetch a rule by name. This can be further customized.
|
|
13
|
+
"""
|
|
14
|
+
# get rule by name which is natural key
|
|
15
|
+
rule = Rule.objects.get_by_natural_key(rule_name)
|
|
16
|
+
return rule
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def apply(self, obj, rule):
|
|
20
|
+
"""
|
|
21
|
+
Applies a specified rule to an object based on the rule type and attributes.
|
|
22
|
+
|
|
23
|
+
Parameters:
|
|
24
|
+
obj (Django model instance): The object to which the rule is applied.
|
|
25
|
+
rule (Rule): The rule instance containing the rule_type and attribute details.
|
|
26
|
+
"""
|
|
27
|
+
rule_type_method = self.get_rule_type_method(rule.rule_type.name)
|
|
28
|
+
if rule_type_method:
|
|
29
|
+
rule_type_method(obj, rule)
|
|
30
|
+
|
|
31
|
+
else:
|
|
32
|
+
raise ValueError(f"Unsupported rule type: {rule.rule_type.name}")
|
|
33
|
+
|
|
34
|
+
def apply_rule_by_name(self, obj, rule_name):
|
|
35
|
+
"""
|
|
36
|
+
A helper method to apply a rule and get a value. This can be further customized.
|
|
37
|
+
"""
|
|
38
|
+
rule = self.get_rule_by_name(rule_name)
|
|
39
|
+
self.apply(obj, rule)
|
|
40
|
+
|
|
41
|
+
def apply_rules_by_name(self, obj, rule_names):
|
|
42
|
+
"""
|
|
43
|
+
A helper method to apply multiple rules and get values. This can be further customized.
|
|
44
|
+
"""
|
|
45
|
+
rules = [self.get_rule_by_name(rule_name) for rule_name in rule_names]
|
|
46
|
+
for rule in rules:
|
|
47
|
+
self.apply(obj, rule)
|
|
48
|
+
|
|
49
|
+
|
|
50
|
+
def get_rule_type_method(self, rule_type_name):
|
|
51
|
+
"""
|
|
52
|
+
Maps rule type name to the corresponding method.
|
|
53
|
+
"""
|
|
54
|
+
return getattr(self, f"handle_{rule_type_name}", None)
|
|
55
|
+
|
|
56
|
+
def parse_attribute_path(self, obj, attribute_key):
|
|
57
|
+
"""
|
|
58
|
+
Parses the attribute path and applies the value to the correct attribute of a nested object.
|
|
59
|
+
"""
|
|
60
|
+
parts = attribute_key.split('.')
|
|
61
|
+
model_name = parts[0]
|
|
62
|
+
if model_name.lower() != obj.__class__.__name__.lower():
|
|
63
|
+
raise ValidationError(f"Model type mismatch: expected {model_name}, got {obj.__class__.__name__}")
|
|
64
|
+
|
|
65
|
+
# Navigate through the nested attributes
|
|
66
|
+
target = obj
|
|
67
|
+
for part in parts[1:-1]:
|
|
68
|
+
target = getattr(target, part)
|
|
69
|
+
|
|
70
|
+
return target, parts[-1]
|
|
71
|
+
|
|
72
|
+
def set_attribute_value(self, obj, rule):
|
|
73
|
+
"""
|
|
74
|
+
Generic method to set attribute value considering nested paths.
|
|
75
|
+
"""
|
|
76
|
+
target, attribute = self.parse_attribute_path(obj, rule.attribute_key)
|
|
77
|
+
setattr(target, attribute, rule.attribute_dict['value'])
|
|
78
|
+
target.save()
|
|
79
|
+
|
|
80
|
+
#####
|
|
81
|
+
|
|
82
|
+
def handle_case_attribute_set_value(self, obj, rule):
|
|
83
|
+
"""
|
|
84
|
+
Sets a specific attribute to a given value.
|
|
85
|
+
"""
|
|
86
|
+
self.set_attribute_value(obj, rule.attribute_key, rule.attribute_dict["value"])
|
|
87
|
+
|
|
88
|
+
def handle_case_attribute_set_value_range_uniform(self, obj, rule):
|
|
89
|
+
"""
|
|
90
|
+
Sets an attribute to a value within a specified range, selected uniformly.
|
|
91
|
+
"""
|
|
92
|
+
if not hasattr(rule, 'attribute_dtype') or rule.attribute_dtype.name not in ['numeric', 'ordered_categorical']:
|
|
93
|
+
raise ValidationError("Attribute dtype must be numeric or ordered_categorical")
|
|
94
|
+
min_val = rule.attribute_dict["value_min"]
|
|
95
|
+
max_val = rule.attribute_dict["value_max"]
|
|
96
|
+
value = random.uniform(min_val, max_val)
|
|
97
|
+
self.set_attribute_value(obj, rule.attribute_key, value)
|
|
98
|
+
|
|
99
|
+
def handle_case_attribute_set_value_range_norm_dist(self, obj, rule):
|
|
100
|
+
"""
|
|
101
|
+
Sets an attribute to a value within a specified range, based on a normal distribution.
|
|
102
|
+
"""
|
|
103
|
+
if rule.attribute_dtype.name in ['float', "integer"]:
|
|
104
|
+
raise ValidationError("Attribute dtype must be float or integer")
|
|
105
|
+
mean = rule.attribute_dict["value_mean"]
|
|
106
|
+
std_dev = rule.attribute_dict["value_sd"]
|
|
107
|
+
value = random.normalvariate(mean, std_dev)
|
|
108
|
+
self.set_attribute_value(obj, rule.attribute_key, value)
|
|
109
|
+
|
|
110
|
+
|
|
111
|
+
def handle_case_attribute_set_from_list_uniform(self, obj, rule):
|
|
112
|
+
"""
|
|
113
|
+
Selects an attribute value uniformly from a list of choices.
|
|
114
|
+
"""
|
|
115
|
+
choices = rule.attribute_dict["value_choices"]
|
|
116
|
+
value = random.choice(choices)
|
|
117
|
+
self.set_attribute_value(obj, rule.attribute_key, value)
|
|
118
|
+
|
|
119
|
+
def handle_case_attribute_set_from_prop_tuple_list(self, obj, rule):
|
|
120
|
+
"""
|
|
121
|
+
Selects an attribute value based on a list of value-probability tuples.
|
|
122
|
+
"""
|
|
123
|
+
prop_list = rule.attribute_dict["value_prop_tuple_list"]
|
|
124
|
+
total = sum(prop for _, prop in prop_list)
|
|
125
|
+
pick = random.uniform(0, total)
|
|
126
|
+
current = 0
|
|
127
|
+
for value, prop in prop_list:
|
|
128
|
+
current += prop
|
|
129
|
+
if current > pick:
|
|
130
|
+
self.set_attribute_value(obj, rule.attribute_key, value)
|
|
131
|
+
break
|
|
132
|
+
|
|
133
|
+
|
|
134
|
+
#####
|
|
135
|
+
|
|
136
|
+
def handle_case_add_patient(self, obj, rule):
|
|
137
|
+
"""Function to add a patient to a case. If the patient already exists, raise an error. Requires the following
|
|
138
|
+
rules in the rule_dict:
|
|
139
|
+
- gender_rule
|
|
140
|
+
- dob_rule
|
|
141
|
+
- event_rules
|
|
142
|
+
- disease_rules
|
|
143
|
+
"""
|
|
144
|
+
from endoreg_db.models import Patient, PatientEvent, PatientDisease, Case
|
|
145
|
+
|
|
146
|
+
# Check if the
|
|
147
|
+
obj:Case = obj
|
|
148
|
+
if obj.patient:
|
|
149
|
+
raise ValueError("Patient already exists in the case")
|
|
150
|
+
|
|
151
|
+
patient = Patient()
|
|
152
|
+
self.apply_rule_by_name(patient, rule.attribute_dict["patient_gender_rule"])
|
|
153
|
+
self.apply_rule_by_name(patient, rule.attribute_dict["patient_dob_rule"])
|
|
154
|
+
self.apply_rules_by_name(patient, rule.attribute_dict["patient_event_rules"])
|
|
155
|
+
self.apply_rules_by_name(patient, rule.attribute_dict["patient_disease_rules"])
|
|
156
|
+
|
|
157
|
+
def handle_case_add_polyp(self, obj, rule):
|
|
158
|
+
"""
|
|
159
|
+
Adds a polyp to the case, processing various polyp characteristics. Requires Rules for each attribute:
|
|
160
|
+
- polyp_location_organ_rule (single rule)
|
|
161
|
+
- polyp_location_organ_part_rule (single rule)
|
|
162
|
+
- polyp_morphology_classification_shape_choices_rule (list of rules)
|
|
163
|
+
- polyp_morphology_classification_chromo_choices_rule (list of rules)
|
|
164
|
+
- polyp_intervention_type_rule (single rule)
|
|
165
|
+
- polyp_intervention_instrument_rule (single rule)
|
|
166
|
+
- polyp_size_mm_rule (single rule)
|
|
167
|
+
|
|
168
|
+
"""
|
|
169
|
+
# Example: create and add a polyp instance based on the rules
|
|
170
|
+
from endoreg_db.models import Polyp # Ensure you have a Polyp model and adjust the import
|
|
171
|
+
from endoreg_db.models import Location
|
|
172
|
+
from endoreg_db.models import EndoscopicIntervention
|
|
173
|
+
from endoreg_db.models import PolypMorphology
|
|
174
|
+
|
|
175
|
+
assert obj.examination.type != None, "Examination type is required"
|
|
176
|
+
|
|
177
|
+
location = Location()
|
|
178
|
+
location.organ = self.apply_rule_by_name(location, rule.attribute_dict["polyp_location_organ_rule"])
|
|
179
|
+
location.organ_part = self.apply_rule_by_name(location, rule.attribute_dict["polyp_location_organ_part_rule"])
|
|
180
|
+
location.save()
|
|
181
|
+
|
|
182
|
+
morphology = PolypMorphology()
|
|
183
|
+
# M2M to ClassificationChoice (linked to Classification which has name as natural key)
|
|
184
|
+
morphology.shape_classification_choices = self.apply_rules_by_name(morphology, rule.attribute_dict["polyp_morphology_classification_shape_choices_rules"])
|
|
185
|
+
# M2M to ClassificationChoice (linked to Classification which has name as natural key)
|
|
186
|
+
morphology.chromo_classification_choices = self.apply_rules_by_name(morphology, rule.attribute_dict["polyp_morphology_classification_chromo_choices_rules"])
|
|
187
|
+
morphology.save()
|
|
188
|
+
|
|
189
|
+
intervention = EndoscopicIntervention()
|
|
190
|
+
intervention.type = self.apply_rule_by_name(intervention, rule.attribute_dict["polyp_intervention_type_rule"])
|
|
191
|
+
intervention.instrument = self.apply_rule_by_name(intervention, rule.attribute_dict["polyp_intervention_instrument_rule"])
|
|
192
|
+
intervention.save()
|
|
193
|
+
|
|
194
|
+
polyp = Polyp()
|
|
195
|
+
polyp.size_mm = self.apply_rule_by_name(polyp, rule.attribute_dict["polyp_size_mm_rule"])
|
|
196
|
+
polyp.location = location
|
|
197
|
+
polyp.morphology = morphology
|
|
198
|
+
polyp.intervention = intervention
|
|
199
|
+
|
|
200
|
+
polyp.save()
|
|
201
|
+
|
|
202
|
+
obj.polyps.add(polyp)
|
|
203
|
+
obj.save()
|
|
204
|
+
|
|
205
|
+
|
|
206
|
+
# def handle_case_add_esophageal_varices(self, obj, rule):
|
|
207
|
+
# """
|
|
208
|
+
# Handles adding esophageal varices to a case based on the rule.
|
|
209
|
+
# """
|
|
210
|
+
# # Similar to the polyp case, create and add esophageal varices based on the attributes
|
|
211
|
+
# from endoreg_db.models import EsophagealVarices # Ensure this model exists
|
|
212
|
+
# varices = EsophagealVarices()
|
|
213
|
+
# varices.origin = self.apply_rule_by_name(obj, rule.attribute_dict["esophageal_varices_origin_rule"])
|
|
214
|
+
# varices.classification = self.apply_rule_by_name(obj, rule.attribute_dict["esophageal_varices_classification_rule"])
|
|
215
|
+
# varices.location_extent = self.apply_rule_by_name(obj, rule.attribute_dict["esophageal_varices_location_extent_rule"])
|
|
216
|
+
# varices.save()
|
|
217
|
+
# # Assuming a relationship setup
|
|
218
|
+
# obj.esophageal_varices.add(varices)
|
|
219
|
+
# obj.save()
|
|
220
|
+
|
|
221
|
+
|
|
222
|
+
|
|
223
|
+
|
|
224
|
+
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
|
|
4
|
+
class RuleAttributeDtypeManager(models.Manager):
|
|
5
|
+
def get_by_natural_key(self, name):
|
|
6
|
+
return self.get(name=name)
|
|
7
|
+
|
|
8
|
+
class RuleAttributeDType(models.Model):
|
|
9
|
+
name = models.CharField(max_length=255, unique=True)
|
|
10
|
+
name_de = models.CharField(max_length=255, blank=True, null=True)
|
|
11
|
+
name_en = models.CharField(max_length=255, blank=True, null=True)
|
|
12
|
+
|
|
13
|
+
objects = RuleAttributeDtypeManager()
|
|
14
|
+
|
|
15
|
+
def natural_key(self):
|
|
16
|
+
return (self.name,)
|
|
17
|
+
|
|
18
|
+
def __str__(self):
|
|
19
|
+
return self.name
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class RuleTypeManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
class RuleType(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
|
+
|
|
12
|
+
objects = RuleTypeManager()
|
|
13
|
+
|
|
14
|
+
def natural_key(self):
|
|
15
|
+
return (self.name,)
|
|
16
|
+
|
|
17
|
+
def __str__(self):
|
|
18
|
+
return self.name
|
|
19
|
+
|
|
20
|
+
class Meta:
|
|
21
|
+
verbose_name = 'Rule Type'
|
|
22
|
+
verbose_name_plural = 'Rule Types'
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class RulesetManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
class Ruleset(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
|
+
rules = models.ManyToManyField('Rule')
|
|
12
|
+
|
|
13
|
+
objects = RulesetManager()
|
|
14
|
+
|
|
15
|
+
def natural_key(self):
|
|
16
|
+
return (self.name,)
|
|
17
|
+
|
|
18
|
+
def __str__(self):
|
|
19
|
+
return self.name
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
from django.db import models
|
|
2
|
+
|
|
3
|
+
class UnitManager(models.Manager):
|
|
4
|
+
def get_by_natural_key(self, name):
|
|
5
|
+
return self.get(name=name)
|
|
6
|
+
|
|
7
|
+
class Unit(models.Model):
|
|
8
|
+
objects = UnitManager()
|
|
9
|
+
|
|
10
|
+
name = models.CharField(max_length=100) # e.g. "Centimeter"
|
|
11
|
+
name_de = models.CharField(max_length=100, blank=True, null=True) # e.g. "Zentimeter"
|
|
12
|
+
name_en = models.CharField(max_length=100, blank=True, null=True) # e.g. "Centimeter"
|
|
13
|
+
description = models.CharField(max_length=100, blank=True, null=True) # e.g. "centimeters", "milimeters", "inches"
|
|
14
|
+
abbreviation = models.CharField(max_length=25, blank=True, null=True) # e.g. "cm", "mm", "in"
|
|
15
|
+
|
|
16
|
+
def __str__(self):
|
|
17
|
+
if self.abbreviation:
|
|
18
|
+
return self.abbreviation
|
|
19
|
+
return self.name
|
|
20
|
+
|
|
21
|
+
def natural_key(self):
|
|
22
|
+
return (self.name,)
|