endoreg-db 0.8.6.4__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/__init__.py +0 -0
- endoreg_db/admin.py +92 -0
- endoreg_db/api/serializers/finding_descriptions.py +0 -0
- endoreg_db/api/views/finding_descriptions.py +0 -0
- endoreg_db/api_urls.py +4 -0
- endoreg_db/apps.py +18 -0
- endoreg_db/assets/dummy_model.ckpt +1 -0
- endoreg_db/codemods/readme.md +88 -0
- endoreg_db/codemods/rename_datetime_fields.py +92 -0
- endoreg_db/config/__init__.py +0 -0
- endoreg_db/config/env.py +101 -0
- endoreg_db/data/__init__.py +144 -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/polyp_classification.yaml +52 -0
- endoreg_db/data/ai_model_label/label-set/data.yaml +40 -0
- endoreg_db/data/ai_model_label/label-set/polyp_classifications.yaml +25 -0
- endoreg_db/data/ai_model_label/label-type/data.yaml +7 -0
- endoreg_db/data/ai_model_meta/default_multilabel_classification.yaml +27 -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 +91 -0
- endoreg_db/data/center_resource/green_endoscopy_dashboard_CenterResource.yaml +144 -0
- endoreg_db/data/center_shift/ukw.yaml +9 -0
- endoreg_db/data/center_waste/green_endoscopy_dashboard_CenterWaste.yaml +48 -0
- endoreg_db/data/contraindication/bleeding.yaml +11 -0
- endoreg_db/data/db_summary.csv +58 -0
- endoreg_db/data/db_summary.xlsx +0 -0
- endoreg_db/data/disease/cardiovascular.yaml +37 -0
- endoreg_db/data/disease/hepatology.yaml +5 -0
- endoreg_db/data/disease/misc.yaml +5 -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 +50 -0
- endoreg_db/data/event/cardiology.yaml +15 -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 +72 -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 +17 -0
- endoreg_db/data/examination_indication/endoscopy.yaml +424 -0
- endoreg_db/data/examination_indication_classification/endoscopy.yaml +160 -0
- endoreg_db/data/examination_indication_classification_choice/endoscopy.yaml +101 -0
- endoreg_db/data/examination_requirement_set/colonoscopy.yaml +15 -0
- endoreg_db/data/finding/anatomy_colon.yaml +128 -0
- endoreg_db/data/finding/colonoscopy.yaml +40 -0
- endoreg_db/data/finding/colonoscopy_bowel_prep.yaml +56 -0
- endoreg_db/data/finding/complication.yaml +16 -0
- endoreg_db/data/finding/data.yaml +105 -0
- endoreg_db/data/finding/examination_setting.yaml +16 -0
- endoreg_db/data/finding/medication_related.yaml +18 -0
- endoreg_db/data/finding/outcome.yaml +12 -0
- endoreg_db/data/finding_classification/colonoscopy_bowel_preparation.yaml +95 -0
- endoreg_db/data/finding_classification/colonoscopy_jnet.yaml +22 -0
- endoreg_db/data/finding_classification/colonoscopy_kudo.yaml +25 -0
- endoreg_db/data/finding_classification/colonoscopy_lesion_circularity.yaml +20 -0
- endoreg_db/data/finding_classification/colonoscopy_lesion_planarity.yaml +24 -0
- endoreg_db/data/finding_classification/colonoscopy_lesion_size.yaml +68 -0
- endoreg_db/data/finding_classification/colonoscopy_lesion_surface.yaml +20 -0
- endoreg_db/data/finding_classification/colonoscopy_location.yaml +80 -0
- endoreg_db/data/finding_classification/colonoscopy_lst.yaml +21 -0
- endoreg_db/data/finding_classification/colonoscopy_nice.yaml +20 -0
- endoreg_db/data/finding_classification/colonoscopy_paris.yaml +26 -0
- endoreg_db/data/finding_classification/colonoscopy_sano.yaml +22 -0
- endoreg_db/data/finding_classification/colonoscopy_summary.yaml +53 -0
- endoreg_db/data/finding_classification/complication_generic.yaml +25 -0
- endoreg_db/data/finding_classification/examination_setting_generic.yaml +40 -0
- endoreg_db/data/finding_classification/histology_colo.yaml +51 -0
- endoreg_db/data/finding_classification/intervention_required.yaml +26 -0
- endoreg_db/data/finding_classification/medication_related.yaml +23 -0
- endoreg_db/data/finding_classification/visualized.yaml +33 -0
- endoreg_db/data/finding_classification_choice/bowel_preparation.yaml +78 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_circularity_default.yaml +32 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_jnet.yaml +15 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_kudo.yaml +23 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_lst.yaml +15 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_nice.yaml +17 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_paris.yaml +57 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_planarity_default.yaml +49 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_sano.yaml +14 -0
- endoreg_db/data/finding_classification_choice/colon_lesion_surface_intact_default.yaml +36 -0
- endoreg_db/data/finding_classification_choice/colonoscopy_location.yaml +229 -0
- endoreg_db/data/finding_classification_choice/colonoscopy_not_complete_reason.yaml +19 -0
- endoreg_db/data/finding_classification_choice/colonoscopy_size.yaml +82 -0
- endoreg_db/data/finding_classification_choice/colonoscopy_summary_worst_finding.yaml +15 -0
- endoreg_db/data/finding_classification_choice/complication_generic_types.yaml +15 -0
- endoreg_db/data/finding_classification_choice/examination_setting_generic_types.yaml +15 -0
- endoreg_db/data/finding_classification_choice/histology.yaml +24 -0
- endoreg_db/data/finding_classification_choice/histology_polyp.yaml +20 -0
- endoreg_db/data/finding_classification_choice/outcome.yaml +19 -0
- endoreg_db/data/finding_classification_choice/yes_no_na.yaml +11 -0
- endoreg_db/data/finding_classification_type/colonoscopy_basic.yaml +48 -0
- endoreg_db/data/finding_intervention/endoscopy.yaml +43 -0
- endoreg_db/data/finding_intervention/endoscopy_colonoscopy.yaml +168 -0
- endoreg_db/data/finding_intervention/endoscopy_egd.yaml +128 -0
- endoreg_db/data/finding_intervention/endoscopy_ercp.yaml +32 -0
- endoreg_db/data/finding_intervention/endoscopy_eus_lower.yaml +9 -0
- endoreg_db/data/finding_intervention/endoscopy_eus_upper.yaml +36 -0
- endoreg_db/data/finding_intervention_type/endoscopy.yaml +15 -0
- endoreg_db/data/finding_morphology_classification_type/colonoscopy.yaml +79 -0
- endoreg_db/data/finding_type/data.yaml +43 -0
- endoreg_db/data/gender/data.yaml +42 -0
- endoreg_db/data/information_source/annotation.yaml +6 -0
- endoreg_db/data/information_source/data.yaml +30 -0
- endoreg_db/data/information_source/endoscopy_guidelines.yaml +7 -0
- endoreg_db/data/information_source/medication.yaml +6 -0
- endoreg_db/data/information_source/prediction.yaml +7 -0
- endoreg_db/data/information_source_type/data.yaml +8 -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 +76 -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 +54 -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 +46 -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/qualification/endoscopy.yaml +36 -0
- endoreg_db/data/qualification/m2.yaml +39 -0
- endoreg_db/data/qualification/outpatient_clinic.yaml +35 -0
- endoreg_db/data/qualification/sonography.yaml +36 -0
- endoreg_db/data/qualification_type/base.yaml +29 -0
- endoreg_db/data/reference_product/green_endoscopy_dashboard_ReferenceProduct.yaml +55 -0
- endoreg_db/data/report_reader_flag/rkh-histology-generic.yaml +10 -0
- endoreg_db/data/report_reader_flag/ukw-examination-generic.yaml +30 -0
- endoreg_db/data/report_reader_flag/ukw-histology-generic.yaml +24 -0
- endoreg_db/data/requirement/age.yaml +26 -0
- endoreg_db/data/requirement/colonoscopy_baseline_austria.yaml +45 -0
- endoreg_db/data/requirement/disease_cardiovascular.yaml +79 -0
- endoreg_db/data/requirement/disease_classification_choice_cardiovascular.yaml +41 -0
- endoreg_db/data/requirement/disease_hepatology.yaml +12 -0
- endoreg_db/data/requirement/disease_misc.yaml +12 -0
- endoreg_db/data/requirement/disease_renal.yaml +96 -0
- endoreg_db/data/requirement/endoscopy_bleeding_risk.yaml +59 -0
- endoreg_db/data/requirement/event_cardiology.yaml +251 -0
- endoreg_db/data/requirement/event_requirements.yaml +145 -0
- endoreg_db/data/requirement/finding_colon_polyp.yaml +50 -0
- endoreg_db/data/requirement/gender.yaml +25 -0
- endoreg_db/data/requirement/lab_value.yaml +441 -0
- endoreg_db/data/requirement/medication.yaml +93 -0
- endoreg_db/data/requirement_operator/age.yaml +13 -0
- endoreg_db/data/requirement_operator/lab_operators.yaml +129 -0
- endoreg_db/data/requirement_operator/model_operators.yaml +96 -0
- endoreg_db/data/requirement_set/01_endoscopy_generic.yaml +48 -0
- endoreg_db/data/requirement_set/colonoscopy_austria_screening.yaml +57 -0
- endoreg_db/data/requirement_set/endoscopy_bleeding_risk.yaml +52 -0
- endoreg_db/data/requirement_set_type/data.yaml +20 -0
- endoreg_db/data/requirement_type/requirement_types.yaml +165 -0
- endoreg_db/data/resource/green_endoscopy_dashboard_Resource.yaml +15 -0
- endoreg_db/data/risk/bleeding.yaml +26 -0
- endoreg_db/data/risk/thrombosis.yaml +37 -0
- endoreg_db/data/risk_type/data.yaml +27 -0
- endoreg_db/data/setup_config.yaml +38 -0
- endoreg_db/data/shift/endoscopy.yaml +21 -0
- endoreg_db/data/shift/m2.yaml +0 -0
- endoreg_db/data/shift_type/base.yaml +35 -0
- endoreg_db/data/tag/requirement_set_tags.yaml +11 -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 +115 -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 +48 -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/exceptions.py +19 -0
- endoreg_db/factories/__init__.py +0 -0
- endoreg_db/forms/__init__.py +5 -0
- endoreg_db/forms/examination_form.py +11 -0
- endoreg_db/forms/patient_finding_intervention_form.py +18 -0
- endoreg_db/forms/patient_form.py +27 -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/helpers/__init__.py +0 -0
- endoreg_db/helpers/count_db.py +45 -0
- endoreg_db/helpers/data_loader.py +208 -0
- endoreg_db/helpers/default_objects.py +378 -0
- endoreg_db/helpers/download_segmentation_model.py +31 -0
- endoreg_db/helpers/interact.py +6 -0
- endoreg_db/helpers/test_video_helper.py +119 -0
- endoreg_db/logger_conf.py +140 -0
- endoreg_db/management/__init__.py +1 -0
- endoreg_db/management/commands/__init__.py +1 -0
- endoreg_db/management/commands/anonymize_video.py +0 -0
- endoreg_db/management/commands/check_auth.py +125 -0
- endoreg_db/management/commands/create_model_meta_from_huggingface.py +115 -0
- endoreg_db/management/commands/create_multilabel_model_meta.py +214 -0
- endoreg_db/management/commands/fix_missing_patient_data.py +172 -0
- endoreg_db/management/commands/fix_video_paths.py +165 -0
- endoreg_db/management/commands/import_fallback_video.py +203 -0
- endoreg_db/management/commands/import_report.py +298 -0
- endoreg_db/management/commands/import_video.py +423 -0
- endoreg_db/management/commands/import_video_with_classification.py +367 -0
- endoreg_db/management/commands/init_default_ai_model.py +112 -0
- endoreg_db/management/commands/load_ai_model_data.py +77 -0
- endoreg_db/management/commands/load_ai_model_label_data.py +59 -0
- endoreg_db/management/commands/load_base_db_data.py +192 -0
- endoreg_db/management/commands/load_center_data.py +68 -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 +86 -0
- endoreg_db/management/commands/load_finding_data.py +128 -0
- endoreg_db/management/commands/load_gender_data.py +44 -0
- endoreg_db/management/commands/load_green_endoscopy_wuerzburg_data.py +132 -0
- endoreg_db/management/commands/load_information_source.py +51 -0
- endoreg_db/management/commands/load_lab_value_data.py +50 -0
- endoreg_db/management/commands/load_medication_data.py +103 -0
- endoreg_db/management/commands/load_name_data.py +37 -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_qualification_data.py +59 -0
- endoreg_db/management/commands/load_report_reader_flag_data.py +46 -0
- endoreg_db/management/commands/load_requirement_data.py +180 -0
- endoreg_db/management/commands/load_risk_data.py +56 -0
- endoreg_db/management/commands/load_shift_data.py +60 -0
- endoreg_db/management/commands/load_tag_data.py +57 -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/management/commands/setup_endoreg_db.py +381 -0
- endoreg_db/management/commands/start_filewatcher.py +106 -0
- endoreg_db/management/commands/storage_management.py +548 -0
- endoreg_db/management/commands/summarize_db_content.py +189 -0
- endoreg_db/management/commands/validate_video.py +204 -0
- endoreg_db/management/commands/validate_video_files.py +161 -0
- endoreg_db/management/commands/video_validation.py +22 -0
- endoreg_db/mermaid/Overall_flow_patient_finding_intervention.md +10 -0
- endoreg_db/mermaid/anonymized_image_annotation.md +20 -0
- endoreg_db/mermaid/binary_classification_annotation.md +50 -0
- endoreg_db/mermaid/classification.md +8 -0
- endoreg_db/mermaid/examination.md +8 -0
- endoreg_db/mermaid/findings.md +7 -0
- endoreg_db/mermaid/image_classification.md +28 -0
- endoreg_db/mermaid/interventions.md +8 -0
- endoreg_db/mermaid/morphology.md +8 -0
- endoreg_db/mermaid/patient_creation.md +14 -0
- endoreg_db/mermaid/video_segmentation_annotation.md +17 -0
- endoreg_db/migrations/0001_initial.py +1857 -0
- endoreg_db/migrations/0002_add_video_correction_models.py +52 -0
- endoreg_db/migrations/0003_add_center_display_name.py +30 -0
- endoreg_db/migrations/__init__.py +0 -0
- endoreg_db/models/__init__.py +359 -0
- endoreg_db/models/administration/__init__.py +116 -0
- endoreg_db/models/administration/ai/__init__.py +9 -0
- endoreg_db/models/administration/ai/active_model.py +35 -0
- endoreg_db/models/administration/ai/ai_model.py +156 -0
- endoreg_db/models/administration/ai/model_type.py +41 -0
- endoreg_db/models/administration/case/__init__.py +19 -0
- endoreg_db/models/administration/case/case.py +114 -0
- endoreg_db/models/administration/case/case_template/__init__.py +15 -0
- endoreg_db/models/administration/case/case_template/case_template.py +125 -0
- endoreg_db/models/administration/case/case_template/case_template_rule.py +269 -0
- endoreg_db/models/administration/case/case_template/case_template_rule_value.py +86 -0
- endoreg_db/models/administration/case/case_template/case_template_type.py +26 -0
- endoreg_db/models/administration/center/__init__.py +13 -0
- endoreg_db/models/administration/center/center.py +67 -0
- endoreg_db/models/administration/center/center_product.py +64 -0
- endoreg_db/models/administration/center/center_resource.py +49 -0
- endoreg_db/models/administration/center/center_shift.py +88 -0
- endoreg_db/models/administration/center/center_waste.py +30 -0
- endoreg_db/models/administration/permissions/__init__.py +44 -0
- endoreg_db/models/administration/person/__init__.py +24 -0
- endoreg_db/models/administration/person/employee/__init__.py +3 -0
- endoreg_db/models/administration/person/employee/employee.py +35 -0
- endoreg_db/models/administration/person/employee/employee_qualification.py +39 -0
- endoreg_db/models/administration/person/employee/employee_type.py +42 -0
- endoreg_db/models/administration/person/examiner/__init__.py +4 -0
- endoreg_db/models/administration/person/examiner/examiner.py +54 -0
- endoreg_db/models/administration/person/names/__init__.py +0 -0
- endoreg_db/models/administration/person/names/first_name.py +18 -0
- endoreg_db/models/administration/person/names/last_name.py +19 -0
- endoreg_db/models/administration/person/patient/__init__.py +5 -0
- endoreg_db/models/administration/person/patient/patient.py +460 -0
- endoreg_db/models/administration/person/person.py +31 -0
- endoreg_db/models/administration/person/profession/__init__.py +24 -0
- endoreg_db/models/administration/person/user/__init__.py +5 -0
- endoreg_db/models/administration/person/user/portal_user_information.py +37 -0
- endoreg_db/models/administration/product/__init__.py +14 -0
- endoreg_db/models/administration/product/product.py +97 -0
- endoreg_db/models/administration/product/product_group.py +39 -0
- endoreg_db/models/administration/product/product_material.py +54 -0
- endoreg_db/models/administration/product/product_weight.py +47 -0
- endoreg_db/models/administration/product/reference_product.py +130 -0
- endoreg_db/models/administration/qualification/__init__.py +7 -0
- endoreg_db/models/administration/qualification/qualification.py +37 -0
- endoreg_db/models/administration/qualification/qualification_type.py +35 -0
- endoreg_db/models/administration/shift/__init__.py +9 -0
- endoreg_db/models/administration/shift/scheduled_days.py +69 -0
- endoreg_db/models/administration/shift/shift.py +51 -0
- endoreg_db/models/administration/shift/shift_type.py +108 -0
- endoreg_db/models/label/__init__.py +24 -0
- endoreg_db/models/label/annotation/__init__.py +12 -0
- endoreg_db/models/label/annotation/image_classification.py +84 -0
- endoreg_db/models/label/annotation/video_segmentation_annotation.py +66 -0
- endoreg_db/models/label/label.py +83 -0
- endoreg_db/models/label/label_set.py +53 -0
- endoreg_db/models/label/label_type.py +29 -0
- endoreg_db/models/label/label_video_segment/__init__.py +3 -0
- endoreg_db/models/label/label_video_segment/_create_from_video.py +41 -0
- endoreg_db/models/label/label_video_segment/label_video_segment.py +511 -0
- endoreg_db/models/label/video_segmentation_label.py +31 -0
- endoreg_db/models/label/video_segmentation_labelset.py +27 -0
- endoreg_db/models/media/__init__.py +16 -0
- endoreg_db/models/media/frame/__init__.py +3 -0
- endoreg_db/models/media/frame/frame.py +111 -0
- endoreg_db/models/media/pdf/__init__.py +11 -0
- endoreg_db/models/media/pdf/raw_pdf.py +757 -0
- endoreg_db/models/media/pdf/report_file.py +162 -0
- endoreg_db/models/media/pdf/report_reader/__init__.py +7 -0
- endoreg_db/models/media/pdf/report_reader/report_reader_config.py +77 -0
- endoreg_db/models/media/pdf/report_reader/report_reader_flag.py +20 -0
- endoreg_db/models/media/video/__init__.py +8 -0
- endoreg_db/models/media/video/create_from_file.py +358 -0
- endoreg_db/models/media/video/pipe_1.py +213 -0
- endoreg_db/models/media/video/pipe_2.py +105 -0
- endoreg_db/models/media/video/refactor_plan.md +0 -0
- endoreg_db/models/media/video/video_file.py +825 -0
- endoreg_db/models/media/video/video_file_ai.py +443 -0
- endoreg_db/models/media/video/video_file_anonymize.py +349 -0
- endoreg_db/models/media/video/video_file_frames/__init__.py +47 -0
- endoreg_db/models/media/video/video_file_frames/_bulk_create_frames.py +22 -0
- endoreg_db/models/media/video/video_file_frames/_create_frame_object.py +23 -0
- endoreg_db/models/media/video/video_file_frames/_delete_frames.py +104 -0
- endoreg_db/models/media/video/video_file_frames/_extract_frames.py +174 -0
- endoreg_db/models/media/video/video_file_frames/_get_frame.py +28 -0
- endoreg_db/models/media/video/video_file_frames/_get_frame_number.py +27 -0
- endoreg_db/models/media/video/video_file_frames/_get_frame_path.py +20 -0
- endoreg_db/models/media/video/video_file_frames/_get_frame_paths.py +27 -0
- endoreg_db/models/media/video/video_file_frames/_get_frame_range.py +34 -0
- endoreg_db/models/media/video/video_file_frames/_get_frames.py +27 -0
- endoreg_db/models/media/video/video_file_frames/_initialize_frames.py +129 -0
- endoreg_db/models/media/video/video_file_frames/_manage_frame_range.py +141 -0
- endoreg_db/models/media/video/video_file_frames/_mark_frames_extracted_status.py +65 -0
- endoreg_db/models/media/video/video_file_frames.py +0 -0
- endoreg_db/models/media/video/video_file_io.py +168 -0
- endoreg_db/models/media/video/video_file_meta/__init__.py +22 -0
- endoreg_db/models/media/video/video_file_meta/get_crop_template.py +45 -0
- endoreg_db/models/media/video/video_file_meta/get_endo_roi.py +39 -0
- endoreg_db/models/media/video/video_file_meta/get_fps.py +147 -0
- endoreg_db/models/media/video/video_file_meta/initialize_video_specs.py +143 -0
- endoreg_db/models/media/video/video_file_meta/text_meta.py +134 -0
- endoreg_db/models/media/video/video_file_meta/video_meta.py +70 -0
- endoreg_db/models/media/video/video_file_segments.py +209 -0
- endoreg_db/models/media/video/video_metadata.py +65 -0
- endoreg_db/models/media/video/video_processing.py +152 -0
- endoreg_db/models/medical/__init__.py +146 -0
- endoreg_db/models/medical/contraindication/__init__.py +17 -0
- endoreg_db/models/medical/disease.py +156 -0
- endoreg_db/models/medical/event.py +137 -0
- endoreg_db/models/medical/examination/__init__.py +9 -0
- endoreg_db/models/medical/examination/examination.py +148 -0
- endoreg_db/models/medical/examination/examination_indication.py +278 -0
- endoreg_db/models/medical/examination/examination_time.py +49 -0
- endoreg_db/models/medical/examination/examination_time_type.py +41 -0
- endoreg_db/models/medical/examination/examination_type.py +48 -0
- endoreg_db/models/medical/finding/__init__.py +18 -0
- endoreg_db/models/medical/finding/finding.py +96 -0
- endoreg_db/models/medical/finding/finding_classification.py +142 -0
- endoreg_db/models/medical/finding/finding_intervention.py +52 -0
- endoreg_db/models/medical/finding/finding_type.py +35 -0
- endoreg_db/models/medical/hardware/__init__.py +8 -0
- endoreg_db/models/medical/hardware/endoscope.py +65 -0
- endoreg_db/models/medical/hardware/endoscopy_processor.py +182 -0
- endoreg_db/models/medical/laboratory/__init__.py +5 -0
- endoreg_db/models/medical/laboratory/lab_value.py +419 -0
- endoreg_db/models/medical/medication/__init__.py +19 -0
- endoreg_db/models/medical/medication/medication.py +31 -0
- endoreg_db/models/medical/medication/medication_indication.py +50 -0
- endoreg_db/models/medical/medication/medication_indication_type.py +39 -0
- endoreg_db/models/medical/medication/medication_intake_time.py +44 -0
- endoreg_db/models/medical/medication/medication_schedule.py +45 -0
- endoreg_db/models/medical/organ/__init__.py +35 -0
- endoreg_db/models/medical/patient/__init__.py +56 -0
- endoreg_db/models/medical/patient/medication_examples.py +38 -0
- endoreg_db/models/medical/patient/patient_disease.py +63 -0
- endoreg_db/models/medical/patient/patient_event.py +75 -0
- endoreg_db/models/medical/patient/patient_examination.py +249 -0
- endoreg_db/models/medical/patient/patient_examination_indication.py +44 -0
- endoreg_db/models/medical/patient/patient_finding.py +357 -0
- endoreg_db/models/medical/patient/patient_finding_classification.py +207 -0
- endoreg_db/models/medical/patient/patient_finding_intervention.py +40 -0
- endoreg_db/models/medical/patient/patient_lab_sample.py +148 -0
- endoreg_db/models/medical/patient/patient_lab_value.py +222 -0
- endoreg_db/models/medical/patient/patient_medication.py +104 -0
- endoreg_db/models/medical/patient/patient_medication_schedule.py +136 -0
- endoreg_db/models/medical/risk/__init__.py +7 -0
- endoreg_db/models/medical/risk/risk.py +72 -0
- endoreg_db/models/medical/risk/risk_type.py +51 -0
- endoreg_db/models/metadata/__init__.py +19 -0
- endoreg_db/models/metadata/frame_ocr_result.py +0 -0
- endoreg_db/models/metadata/model_meta.py +206 -0
- endoreg_db/models/metadata/model_meta_logic.py +343 -0
- endoreg_db/models/metadata/pdf_meta.py +89 -0
- endoreg_db/models/metadata/sensitive_meta.py +288 -0
- endoreg_db/models/metadata/sensitive_meta_logic.py +1048 -0
- endoreg_db/models/metadata/video_meta.py +332 -0
- endoreg_db/models/metadata/video_prediction_logic.py +190 -0
- endoreg_db/models/metadata/video_prediction_meta.py +270 -0
- endoreg_db/models/other/__init__.py +40 -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 +89 -0
- endoreg_db/models/other/distribution/multiple_categorical_value_distribution.py +32 -0
- endoreg_db/models/other/distribution/numeric_value_distribution.py +125 -0
- endoreg_db/models/other/distribution/single_categorical_value_distribution.py +22 -0
- endoreg_db/models/other/emission/__init__.py +5 -0
- endoreg_db/models/other/emission/emission_factor.py +94 -0
- endoreg_db/models/other/gender.py +27 -0
- endoreg_db/models/other/information_source.py +159 -0
- endoreg_db/models/other/material.py +28 -0
- endoreg_db/models/other/resource.py +22 -0
- endoreg_db/models/other/tag.py +27 -0
- endoreg_db/models/other/transport_route.py +33 -0
- endoreg_db/models/other/unit.py +32 -0
- endoreg_db/models/other/waste.py +27 -0
- endoreg_db/models/requirement/__init__.py +11 -0
- endoreg_db/models/requirement/requirement.py +767 -0
- endoreg_db/models/requirement/requirement_evaluation/__init__.py +6 -0
- endoreg_db/models/requirement/requirement_evaluation/get_values.py +40 -0
- endoreg_db/models/requirement/requirement_evaluation/operator_evaluation_models.py +9 -0
- endoreg_db/models/requirement/requirement_evaluation/requirement_type_parser.py +95 -0
- endoreg_db/models/requirement/requirement_operator.py +176 -0
- endoreg_db/models/requirement/requirement_set.py +287 -0
- endoreg_db/models/rule/__init__.py +13 -0
- endoreg_db/models/rule/rule.py +27 -0
- endoreg_db/models/rule/rule_applicator.py +224 -0
- endoreg_db/models/rule/rule_attribute_dtype.py +17 -0
- endoreg_db/models/rule/rule_type.py +20 -0
- endoreg_db/models/rule/ruleset.py +17 -0
- endoreg_db/models/state/__init__.py +12 -0
- endoreg_db/models/state/abstract.py +11 -0
- endoreg_db/models/state/audit_ledger.py +150 -0
- endoreg_db/models/state/label_video_segment.py +22 -0
- endoreg_db/models/state/raw_pdf.py +187 -0
- endoreg_db/models/state/sensitive_meta.py +46 -0
- endoreg_db/models/state/video.py +232 -0
- endoreg_db/models/upload_job.py +99 -0
- endoreg_db/models/utils.py +135 -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/sanity/__init_.py +0 -0
- endoreg_db/renames.yml +8 -0
- endoreg_db/root_urls.py +9 -0
- endoreg_db/schemas/__init__.py +0 -0
- endoreg_db/schemas/examination_evaluation.py +27 -0
- endoreg_db/serializers/Frames_NICE_and_PARIS_classifications.py +775 -0
- endoreg_db/serializers/__init__.py +118 -0
- endoreg_db/serializers/_old/raw_pdf_meta_validation.py +223 -0
- endoreg_db/serializers/_old/raw_video_meta_validation.py +179 -0
- endoreg_db/serializers/_old/video.py +71 -0
- endoreg_db/serializers/administration/__init__.py +14 -0
- endoreg_db/serializers/administration/ai/__init__.py +10 -0
- endoreg_db/serializers/administration/ai/active_model.py +10 -0
- endoreg_db/serializers/administration/ai/ai_model.py +18 -0
- endoreg_db/serializers/administration/ai/model_type.py +10 -0
- endoreg_db/serializers/administration/center.py +9 -0
- endoreg_db/serializers/administration/gender.py +9 -0
- endoreg_db/serializers/anonymization.py +69 -0
- endoreg_db/serializers/evaluation/examination_evaluation.py +1 -0
- endoreg_db/serializers/examination/__init__.py +10 -0
- endoreg_db/serializers/examination/base.py +46 -0
- endoreg_db/serializers/examination/dropdown.py +21 -0
- endoreg_db/serializers/examination_serializer.py +12 -0
- endoreg_db/serializers/finding/__init__.py +5 -0
- endoreg_db/serializers/finding/finding.py +54 -0
- endoreg_db/serializers/finding_classification/__init__.py +7 -0
- endoreg_db/serializers/finding_classification/choice.py +19 -0
- endoreg_db/serializers/finding_classification/classification.py +13 -0
- endoreg_db/serializers/label/__init__.py +7 -0
- endoreg_db/serializers/label/image_classification_annotation.py +62 -0
- endoreg_db/serializers/label/label.py +15 -0
- endoreg_db/serializers/label_video_segment/__init__.py +7 -0
- endoreg_db/serializers/label_video_segment/_lvs_create.py +149 -0
- endoreg_db/serializers/label_video_segment/_lvs_update.py +138 -0
- endoreg_db/serializers/label_video_segment/_lvs_validate.py +149 -0
- endoreg_db/serializers/label_video_segment/label_video_segment.py +344 -0
- endoreg_db/serializers/label_video_segment/label_video_segment_annotation.py +99 -0
- endoreg_db/serializers/label_video_segment/label_video_segment_update.py +163 -0
- endoreg_db/serializers/meta/__init__.py +19 -0
- endoreg_db/serializers/meta/pdf_file_meta_extraction.py +115 -0
- endoreg_db/serializers/meta/report_meta.py +53 -0
- endoreg_db/serializers/meta/sensitive_meta_detail.py +162 -0
- endoreg_db/serializers/meta/sensitive_meta_update.py +148 -0
- endoreg_db/serializers/meta/sensitive_meta_verification.py +59 -0
- endoreg_db/serializers/meta/video_meta.py +39 -0
- endoreg_db/serializers/misc/__init__.py +14 -0
- endoreg_db/serializers/misc/file_overview.py +182 -0
- endoreg_db/serializers/misc/sensitive_patient_data.py +120 -0
- endoreg_db/serializers/misc/stats.py +33 -0
- endoreg_db/serializers/misc/translatable_field_mix_in.py +44 -0
- endoreg_db/serializers/misc/upload_job.py +71 -0
- endoreg_db/serializers/patient/__init__.py +11 -0
- endoreg_db/serializers/patient/patient.py +86 -0
- endoreg_db/serializers/patient/patient_dropdown.py +27 -0
- endoreg_db/serializers/patient_examination/__init__.py +7 -0
- endoreg_db/serializers/patient_examination/patient_examination.py +141 -0
- endoreg_db/serializers/patient_finding/__init__.py +15 -0
- endoreg_db/serializers/patient_finding/patient_finding.py +31 -0
- endoreg_db/serializers/patient_finding/patient_finding_classification.py +39 -0
- endoreg_db/serializers/patient_finding/patient_finding_detail.py +53 -0
- endoreg_db/serializers/patient_finding/patient_finding_intervention.py +26 -0
- endoreg_db/serializers/patient_finding/patient_finding_list.py +41 -0
- endoreg_db/serializers/patient_finding/patient_finding_write.py +126 -0
- endoreg_db/serializers/pdf/__init__.py +5 -0
- endoreg_db/serializers/pdf/anony_text_validation.py +85 -0
- endoreg_db/serializers/report/__init__.py +9 -0
- endoreg_db/serializers/report/mixins.py +45 -0
- endoreg_db/serializers/report/report.py +105 -0
- endoreg_db/serializers/report/report_list.py +22 -0
- endoreg_db/serializers/report/secure_file_url.py +26 -0
- endoreg_db/serializers/requirements/requirement_schema.py +25 -0
- endoreg_db/serializers/requirements/requirement_sets.py +29 -0
- endoreg_db/serializers/sensitive_meta_serializer.py +282 -0
- endoreg_db/serializers/video/__init__.py +7 -0
- endoreg_db/serializers/video/segmentation.py +263 -0
- endoreg_db/serializers/video/video_file_brief.py +10 -0
- endoreg_db/serializers/video/video_file_detail.py +83 -0
- endoreg_db/serializers/video/video_file_list.py +67 -0
- endoreg_db/serializers/video/video_metadata.py +105 -0
- endoreg_db/serializers/video/video_processing_history.py +153 -0
- endoreg_db/serializers/video_examination.py +198 -0
- endoreg_db/services/__init__.py +5 -0
- endoreg_db/services/anonymization.py +223 -0
- endoreg_db/services/examination_evaluation.py +149 -0
- endoreg_db/services/finding_description_service.py +0 -0
- endoreg_db/services/lookup_service.py +411 -0
- endoreg_db/services/lookup_store.py +266 -0
- endoreg_db/services/pdf_import.py +1382 -0
- endoreg_db/services/polling_coordinator.py +288 -0
- endoreg_db/services/pseudonym_service.py +89 -0
- endoreg_db/services/requirements_object.py +147 -0
- endoreg_db/services/segment_sync.py +155 -0
- endoreg_db/services/storage_aware_video_processor.py +344 -0
- endoreg_db/services/video_import.py +1259 -0
- endoreg_db/tasks/upload_tasks.py +207 -0
- endoreg_db/tasks/video_ingest.py +157 -0
- endoreg_db/tasks/video_processing_tasks.py +327 -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/urls/__init__.py +83 -0
- endoreg_db/urls/anonymization.py +32 -0
- endoreg_db/urls/auth.py +16 -0
- endoreg_db/urls/classification.py +39 -0
- endoreg_db/urls/examination.py +54 -0
- endoreg_db/urls/files.py +6 -0
- endoreg_db/urls/label_video_segment_validate.py +33 -0
- endoreg_db/urls/label_video_segments.py +46 -0
- endoreg_db/urls/media.py +227 -0
- endoreg_db/urls/patient.py +19 -0
- endoreg_db/urls/report.py +48 -0
- endoreg_db/urls/requirements.py +13 -0
- endoreg_db/urls/sensitive_meta.py +0 -0
- endoreg_db/urls/stats.py +46 -0
- endoreg_db/urls/upload.py +20 -0
- endoreg_db/urls/video.py +61 -0
- endoreg_db/urls.py +9 -0
- endoreg_db/utils/__init__.py +88 -0
- endoreg_db/utils/ai/__init__.py +9 -0
- endoreg_db/utils/ai/get.py +5 -0
- endoreg_db/utils/ai/inference_dataset.py +52 -0
- endoreg_db/utils/ai/multilabel_classification_net.py +159 -0
- endoreg_db/utils/ai/postprocess.py +63 -0
- endoreg_db/utils/ai/predict.py +291 -0
- endoreg_db/utils/ai/preprocess.py +68 -0
- endoreg_db/utils/calc_duration_seconds.py +24 -0
- endoreg_db/utils/case_generator/__init__.py +0 -0
- endoreg_db/utils/case_generator/case_generator.py +159 -0
- endoreg_db/utils/case_generator/lab_sample_factory.py +33 -0
- endoreg_db/utils/case_generator/utils.py +30 -0
- endoreg_db/utils/check_video_files.py +148 -0
- endoreg_db/utils/cropping.py +29 -0
- endoreg_db/utils/dataloader.py +175 -0
- endoreg_db/utils/dates.py +60 -0
- endoreg_db/utils/env.py +33 -0
- endoreg_db/utils/extract_specific_frames.py +72 -0
- endoreg_db/utils/file_operations.py +58 -0
- endoreg_db/utils/fix_video_path_direct.py +141 -0
- endoreg_db/utils/frame_anonymization_utils.py +463 -0
- endoreg_db/utils/hashs.py +153 -0
- endoreg_db/utils/links/__init__.py +0 -0
- endoreg_db/utils/links/requirement_link.py +193 -0
- endoreg_db/utils/mime_types.py +0 -0
- endoreg_db/utils/names.py +76 -0
- endoreg_db/utils/ocr.py +190 -0
- endoreg_db/utils/parse_and_generate_yaml.py +46 -0
- endoreg_db/utils/paths.py +95 -0
- endoreg_db/utils/permissions.py +143 -0
- endoreg_db/utils/pipelines/Readme.md +235 -0
- endoreg_db/utils/pipelines/__init__.py +0 -0
- endoreg_db/utils/pipelines/process_video_dir.py +120 -0
- endoreg_db/utils/product/__init__.py +0 -0
- endoreg_db/utils/product/sum_emissions.py +20 -0
- endoreg_db/utils/product/sum_weights.py +18 -0
- endoreg_db/utils/pydantic_models/__init__.py +6 -0
- endoreg_db/utils/pydantic_models/db_config.py +57 -0
- endoreg_db/utils/requirement_helpers.py +0 -0
- endoreg_db/utils/requirement_operator_logic/__init__.py +0 -0
- endoreg_db/utils/requirement_operator_logic/lab_value_operators.py +578 -0
- endoreg_db/utils/requirement_operator_logic/model_evaluators.py +368 -0
- endoreg_db/utils/setup_config.py +177 -0
- endoreg_db/utils/translation.py +27 -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/validate_video_detailed.py +357 -0
- endoreg_db/utils/video/__init__.py +26 -0
- endoreg_db/utils/video/extract_frames.py +88 -0
- endoreg_db/utils/video/ffmpeg_wrapper.py +835 -0
- endoreg_db/utils/video/names.py +42 -0
- endoreg_db/utils/video/streaming_processor.py +312 -0
- endoreg_db/utils/video/video_splitter.py +94 -0
- endoreg_db/views/Frames_NICE_and_PARIS_classifications_views.py +238 -0
- endoreg_db/views/__init__.py +274 -0
- endoreg_db/views/anonymization/__init__.py +27 -0
- endoreg_db/views/anonymization/media_management.py +454 -0
- endoreg_db/views/anonymization/overview.py +216 -0
- endoreg_db/views/anonymization/validate.py +107 -0
- endoreg_db/views/auth/__init__.py +13 -0
- endoreg_db/views/auth/keycloak.py +113 -0
- endoreg_db/views/examination/__init__.py +33 -0
- endoreg_db/views/examination/examination.py +37 -0
- endoreg_db/views/examination/examination_manifest_cache.py +26 -0
- endoreg_db/views/examination/get_finding_classification_choices.py +59 -0
- endoreg_db/views/examination/get_finding_classifications.py +36 -0
- endoreg_db/views/examination/get_findings.py +41 -0
- endoreg_db/views/examination/get_instruments.py +18 -0
- endoreg_db/views/examination/get_interventions.py +14 -0
- endoreg_db/views/finding/__init__.py +9 -0
- endoreg_db/views/finding/finding.py +112 -0
- endoreg_db/views/finding/get_classifications.py +14 -0
- endoreg_db/views/finding/get_interventions.py +17 -0
- endoreg_db/views/finding_classification/__init__.py +13 -0
- endoreg_db/views/finding_classification/base.py +0 -0
- endoreg_db/views/finding_classification/finding_classification.py +42 -0
- endoreg_db/views/finding_classification/get_classification_choices.py +55 -0
- endoreg_db/views/label/__init__.py +5 -0
- endoreg_db/views/label/label.py +15 -0
- endoreg_db/views/label_video_segment/__init__.py +16 -0
- endoreg_db/views/label_video_segment/create_lvs_from_annotation.py +44 -0
- endoreg_db/views/label_video_segment/get_lvs_by_name_and_video.py +50 -0
- endoreg_db/views/label_video_segment/label_video_segment.py +77 -0
- endoreg_db/views/label_video_segment/label_video_segment_by_label.py +174 -0
- endoreg_db/views/label_video_segment/label_video_segment_detail.py +73 -0
- endoreg_db/views/label_video_segment/update_lvs_from_annotation.py +46 -0
- endoreg_db/views/label_video_segment/validate.py +226 -0
- endoreg_db/views/media/__init__.py +45 -0
- endoreg_db/views/media/pdf_media.py +388 -0
- endoreg_db/views/media/segments.py +71 -0
- endoreg_db/views/media/sensitive_metadata.py +314 -0
- endoreg_db/views/media/video_media.py +272 -0
- endoreg_db/views/media/video_segments.py +524 -0
- endoreg_db/views/meta/__init__.py +15 -0
- endoreg_db/views/meta/available_files_list.py +146 -0
- endoreg_db/views/meta/report_meta.py +53 -0
- endoreg_db/views/meta/sensitive_meta_detail.py +148 -0
- endoreg_db/views/meta/sensitive_meta_list.py +104 -0
- endoreg_db/views/meta/sensitive_meta_verification.py +71 -0
- endoreg_db/views/misc/__init__.py +63 -0
- endoreg_db/views/misc/center.py +13 -0
- endoreg_db/views/misc/csrf.py +7 -0
- endoreg_db/views/misc/gender.py +14 -0
- endoreg_db/views/misc/secure_file_serving_view.py +80 -0
- endoreg_db/views/misc/secure_file_url_view.py +84 -0
- endoreg_db/views/misc/secure_url_validate.py +79 -0
- endoreg_db/views/misc/stats.py +220 -0
- endoreg_db/views/misc/translation.py +182 -0
- endoreg_db/views/misc/upload_views.py +240 -0
- endoreg_db/views/patient/__init__.py +5 -0
- endoreg_db/views/patient/patient.py +210 -0
- endoreg_db/views/patient_examination/DEPRECATED_video_backup.py +164 -0
- endoreg_db/views/patient_examination/__init__.py +11 -0
- endoreg_db/views/patient_examination/patient_examination.py +140 -0
- endoreg_db/views/patient_examination/patient_examination_create.py +63 -0
- endoreg_db/views/patient_examination/patient_examination_detail.py +66 -0
- endoreg_db/views/patient_examination/patient_examination_list.py +68 -0
- endoreg_db/views/patient_examination/video.py +194 -0
- endoreg_db/views/patient_finding/__init__.py +7 -0
- endoreg_db/views/patient_finding/base.py +0 -0
- endoreg_db/views/patient_finding/patient_finding.py +64 -0
- endoreg_db/views/patient_finding/patient_finding_optimized.py +259 -0
- endoreg_db/views/patient_finding_classification/__init__.py +5 -0
- endoreg_db/views/patient_finding_classification/pfc_create.py +67 -0
- endoreg_db/views/patient_finding_location/__init__.py +5 -0
- endoreg_db/views/patient_finding_location/pfl_create.py +70 -0
- endoreg_db/views/patient_finding_morphology/__init__.py +5 -0
- endoreg_db/views/patient_finding_morphology/pfm_create.py +70 -0
- endoreg_db/views/pdf/__init__.py +8 -0
- endoreg_db/views/pdf/pdf_stream.py +186 -0
- endoreg_db/views/pdf/reimport.py +177 -0
- endoreg_db/views/report/__init__.py +9 -0
- endoreg_db/views/report/report_list.py +112 -0
- endoreg_db/views/report/report_with_secure_url.py +28 -0
- endoreg_db/views/report/start_examination.py +7 -0
- endoreg_db/views/requirement/__init__.py +10 -0
- endoreg_db/views/requirement/evaluate.py +279 -0
- endoreg_db/views/requirement/lookup.py +367 -0
- endoreg_db/views/requirement/lookup_store.py +252 -0
- endoreg_db/views/requirement_lookup/lookup.py +0 -0
- endoreg_db/views/requirement_lookup/lookup_store.py +0 -0
- endoreg_db/views/stats/__init__.py +13 -0
- endoreg_db/views/stats/stats_views.py +229 -0
- endoreg_db/views/video/__init__.py +59 -0
- endoreg_db/views/video/correction.py +530 -0
- endoreg_db/views/video/reimport.py +195 -0
- endoreg_db/views/video/segmentation.py +274 -0
- endoreg_db/views/video/task_status.py +49 -0
- endoreg_db/views/video/timeline.py +46 -0
- endoreg_db/views/video/video_analyze.py +52 -0
- endoreg_db/views/video/video_apply_mask.py +48 -0
- endoreg_db/views/video/video_correction.py +21 -0
- endoreg_db/views/video/video_download_processed.py +58 -0
- endoreg_db/views/video/video_examination_viewset.py +242 -0
- endoreg_db/views/video/video_meta.py +29 -0
- endoreg_db/views/video/video_processing_history.py +24 -0
- endoreg_db/views/video/video_remove_frames.py +48 -0
- endoreg_db/views/video/video_stream.py +306 -0
- endoreg_db/views.py +0 -0
- endoreg_db-0.8.6.4.dist-info/METADATA +383 -0
- endoreg_db-0.8.6.4.dist-info/RECORD +793 -0
- endoreg_db-0.8.6.4.dist-info/WHEEL +4 -0
- endoreg_db-0.8.6.4.dist-info/licenses/LICENSE +674 -0
|
@@ -0,0 +1,524 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Modern Media Framework - Video Segments Views
|
|
3
|
+
Migrated from legacy label_video_segment views (October 14, 2025)
|
|
4
|
+
|
|
5
|
+
Provides RESTful endpoints for video segment management:
|
|
6
|
+
- Collection: GET/POST /api/media/videos/segments/
|
|
7
|
+
- Detail: GET/PATCH/DELETE /api/media/videos/<pk>/segments/<segment_id>/
|
|
8
|
+
- Video-specific: GET/POST /api/media/videos/<pk>/segments/
|
|
9
|
+
"""
|
|
10
|
+
|
|
11
|
+
import logging
|
|
12
|
+
|
|
13
|
+
from django.db import transaction
|
|
14
|
+
from django.db.models import Count
|
|
15
|
+
from django.shortcuts import get_object_or_404
|
|
16
|
+
from rest_framework import status
|
|
17
|
+
from rest_framework.decorators import api_view, permission_classes
|
|
18
|
+
from rest_framework.response import Response
|
|
19
|
+
|
|
20
|
+
from endoreg_db.models import Label, LabelVideoSegment, VideoFile
|
|
21
|
+
from endoreg_db.serializers.label_video_segment.label_video_segment import LabelVideoSegmentSerializer
|
|
22
|
+
from endoreg_db.utils.permissions import EnvironmentAwarePermission
|
|
23
|
+
|
|
24
|
+
logger = logging.getLogger(__name__)
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
@api_view(["GET"])
|
|
28
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
29
|
+
def video_segments_stats(request):
|
|
30
|
+
"""
|
|
31
|
+
Statistics endpoint for video segments.
|
|
32
|
+
|
|
33
|
+
GET /api/media/videos/segments/stats/
|
|
34
|
+
Returns aggregated statistics about video segments.
|
|
35
|
+
"""
|
|
36
|
+
try:
|
|
37
|
+
# Get all segments queryset
|
|
38
|
+
segments = LabelVideoSegment.objects.all()
|
|
39
|
+
|
|
40
|
+
# Calculate statistics
|
|
41
|
+
total_segments = segments.count()
|
|
42
|
+
|
|
43
|
+
# Segments by label
|
|
44
|
+
label_counts = segments.values("label__name").annotate(count=Count("id"))
|
|
45
|
+
|
|
46
|
+
# Videos with segments
|
|
47
|
+
videos_with_segments = segments.values("video_file").distinct().count()
|
|
48
|
+
|
|
49
|
+
stats = {
|
|
50
|
+
"total_segments": total_segments,
|
|
51
|
+
"videos_with_segments": videos_with_segments,
|
|
52
|
+
"by_label": {item["label__name"]: item["count"] for item in label_counts if item["label__name"]},
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
return Response(stats, status=status.HTTP_200_OK)
|
|
56
|
+
|
|
57
|
+
except Exception as e:
|
|
58
|
+
logger.error(f"Error fetching video segment stats: {e}")
|
|
59
|
+
return Response({"error": "Failed to fetch segment statistics"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
60
|
+
|
|
61
|
+
|
|
62
|
+
@api_view(["GET", "POST"])
|
|
63
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
64
|
+
def video_segments_collection(request):
|
|
65
|
+
"""
|
|
66
|
+
Collection endpoint for all video segments across all videos.
|
|
67
|
+
|
|
68
|
+
GET /api/media/videos/segments/
|
|
69
|
+
- Lists all segments, optionally filtered by video_id and/or label_id
|
|
70
|
+
- Query params: video_id, label_id
|
|
71
|
+
|
|
72
|
+
POST /api/media/videos/segments/
|
|
73
|
+
- Creates a new video segment
|
|
74
|
+
- Requires: video_id, label_id, start_frame_number, end_frame_number
|
|
75
|
+
|
|
76
|
+
Modern replacement for: /api/video-segments/
|
|
77
|
+
"""
|
|
78
|
+
if request.method == "POST":
|
|
79
|
+
logger.info(f"Creating new video segment with data: {request.data}")
|
|
80
|
+
|
|
81
|
+
with transaction.atomic():
|
|
82
|
+
serializer = LabelVideoSegmentSerializer(data=request.data)
|
|
83
|
+
if serializer.is_valid():
|
|
84
|
+
try:
|
|
85
|
+
segment = serializer.save()
|
|
86
|
+
logger.info(f"Successfully created video segment {segment.pk}")
|
|
87
|
+
return Response(LabelVideoSegmentSerializer(segment).data, status=status.HTTP_201_CREATED)
|
|
88
|
+
except Exception as e:
|
|
89
|
+
logger.error(f"Error creating video segment: {str(e)}")
|
|
90
|
+
return Response({"error": f"Failed to create segment: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
91
|
+
else:
|
|
92
|
+
logger.warning(f"Invalid data for video segment creation: {serializer.errors}")
|
|
93
|
+
return Response({"error": "Invalid data", "details": serializer.errors}, status=status.HTTP_400_BAD_REQUEST)
|
|
94
|
+
|
|
95
|
+
elif request.method == "GET":
|
|
96
|
+
# Optional filtering by video_id
|
|
97
|
+
video_id = request.GET.get("video_id")
|
|
98
|
+
label_id = request.GET.get("label_id")
|
|
99
|
+
|
|
100
|
+
queryset = LabelVideoSegment.objects.all()
|
|
101
|
+
|
|
102
|
+
if video_id:
|
|
103
|
+
try:
|
|
104
|
+
video = VideoFile.objects.get(id=video_id)
|
|
105
|
+
queryset = queryset.filter(video_file=video)
|
|
106
|
+
except VideoFile.DoesNotExist:
|
|
107
|
+
return Response({"error": f"Video with id {video_id} not found"}, status=status.HTTP_404_NOT_FOUND)
|
|
108
|
+
|
|
109
|
+
if label_id:
|
|
110
|
+
try:
|
|
111
|
+
label = Label.objects.get(id=label_id)
|
|
112
|
+
queryset = queryset.filter(label=label)
|
|
113
|
+
except Label.DoesNotExist:
|
|
114
|
+
return Response({"error": f"Label with id {label_id} not found"}, status=status.HTTP_404_NOT_FOUND)
|
|
115
|
+
|
|
116
|
+
# Order by video and start time for consistent results
|
|
117
|
+
segments = queryset.order_by("video_file__id", "start_frame_number")
|
|
118
|
+
serializer = LabelVideoSegmentSerializer(segments, many=True)
|
|
119
|
+
return Response(serializer.data)
|
|
120
|
+
|
|
121
|
+
|
|
122
|
+
@api_view(["GET", "POST"])
|
|
123
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
124
|
+
def video_segments_by_video(request, pk):
|
|
125
|
+
"""
|
|
126
|
+
Video-specific segments endpoint.
|
|
127
|
+
|
|
128
|
+
GET /api/media/videos/<pk>/segments/
|
|
129
|
+
- Lists all segments for a specific video
|
|
130
|
+
- Query params: label (label name filter)
|
|
131
|
+
- Note: This was already implemented in segments.py as video_segments_by_pk
|
|
132
|
+
|
|
133
|
+
POST /api/media/videos/<pk>/segments/
|
|
134
|
+
- Creates a new segment for this video
|
|
135
|
+
- Automatically sets video_id to pk
|
|
136
|
+
- Requires: label_id, start_frame_number, end_frame_number
|
|
137
|
+
|
|
138
|
+
Modern replacement for: /api/video-segments/?video_id=<pk>
|
|
139
|
+
"""
|
|
140
|
+
# Verify video exists
|
|
141
|
+
video = get_object_or_404(VideoFile, id=pk)
|
|
142
|
+
|
|
143
|
+
if request.method == "GET":
|
|
144
|
+
# This duplicates video_segments_by_pk functionality
|
|
145
|
+
# We keep both for compatibility during migration
|
|
146
|
+
label_name = request.GET.get("label")
|
|
147
|
+
|
|
148
|
+
queryset = LabelVideoSegment.objects.filter(video_file=video)
|
|
149
|
+
|
|
150
|
+
if label_name:
|
|
151
|
+
try:
|
|
152
|
+
label = Label.objects.get(name=label_name)
|
|
153
|
+
queryset = queryset.filter(label=label)
|
|
154
|
+
except Label.DoesNotExist:
|
|
155
|
+
return Response({"error": f'Label "{label_name}" not found'}, status=status.HTTP_404_NOT_FOUND)
|
|
156
|
+
|
|
157
|
+
segments = queryset.order_by("start_frame_number")
|
|
158
|
+
serializer = LabelVideoSegmentSerializer(segments, many=True)
|
|
159
|
+
return Response(serializer.data)
|
|
160
|
+
|
|
161
|
+
elif request.method == "POST":
|
|
162
|
+
logger.info(f"Creating new segment for video {pk} with data: {request.data}")
|
|
163
|
+
|
|
164
|
+
# Automatically set video_id to pk
|
|
165
|
+
data = request.data.copy()
|
|
166
|
+
data["video_id"] = pk
|
|
167
|
+
|
|
168
|
+
with transaction.atomic():
|
|
169
|
+
serializer = LabelVideoSegmentSerializer(data=data)
|
|
170
|
+
if serializer.is_valid():
|
|
171
|
+
try:
|
|
172
|
+
segment = serializer.save()
|
|
173
|
+
logger.info(f"Successfully created segment {segment.pk} for video {pk}")
|
|
174
|
+
return Response(LabelVideoSegmentSerializer(segment).data, status=status.HTTP_201_CREATED)
|
|
175
|
+
except Exception as e:
|
|
176
|
+
logger.error(f"Error creating segment for video {pk}: {str(e)}")
|
|
177
|
+
return Response({"error": f"Failed to create segment: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
178
|
+
else:
|
|
179
|
+
logger.warning(f"Invalid data for segment creation: {serializer.errors}")
|
|
180
|
+
return Response({"error": "Invalid data", "details": serializer.errors}, status=status.HTTP_400_BAD_REQUEST)
|
|
181
|
+
|
|
182
|
+
|
|
183
|
+
@api_view(["GET", "PATCH", "DELETE"])
|
|
184
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
185
|
+
def video_segment_detail(request, pk, segment_id):
|
|
186
|
+
"""
|
|
187
|
+
Detail endpoint for a specific video segment.
|
|
188
|
+
|
|
189
|
+
GET /api/media/videos/<pk>/segments/<segment_id>/
|
|
190
|
+
- Returns segment details
|
|
191
|
+
|
|
192
|
+
PATCH /api/media/videos/<pk>/segments/<segment_id>/
|
|
193
|
+
- Updates segment (partial update)
|
|
194
|
+
|
|
195
|
+
DELETE /api/media/videos/<pk>/segments/<segment_id>/
|
|
196
|
+
- Deletes segment
|
|
197
|
+
|
|
198
|
+
Modern replacement for: /api/video-segments/<segment_id>/
|
|
199
|
+
"""
|
|
200
|
+
# Verify video exists
|
|
201
|
+
video = get_object_or_404(VideoFile, id=pk)
|
|
202
|
+
|
|
203
|
+
# Get segment and verify it belongs to this video
|
|
204
|
+
segment = get_object_or_404(LabelVideoSegment, id=segment_id, video_file=video)
|
|
205
|
+
|
|
206
|
+
if request.method == "GET":
|
|
207
|
+
serializer = LabelVideoSegmentSerializer(segment)
|
|
208
|
+
return Response(serializer.data)
|
|
209
|
+
|
|
210
|
+
elif request.method == "PATCH":
|
|
211
|
+
logger.info(f"Updating segment {segment_id} for video {pk} with data: {request.data}")
|
|
212
|
+
|
|
213
|
+
with transaction.atomic():
|
|
214
|
+
serializer = LabelVideoSegmentSerializer(segment, data=request.data, partial=True)
|
|
215
|
+
if serializer.is_valid():
|
|
216
|
+
try:
|
|
217
|
+
segment = serializer.save()
|
|
218
|
+
logger.info(f"Successfully updated segment {segment_id}")
|
|
219
|
+
return Response(LabelVideoSegmentSerializer(segment).data)
|
|
220
|
+
except Exception as e:
|
|
221
|
+
logger.error(f"Error updating segment {segment_id}: {str(e)}")
|
|
222
|
+
return Response({"error": f"Failed to update segment: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
223
|
+
else:
|
|
224
|
+
logger.warning(f"Invalid data for segment update: {serializer.errors}")
|
|
225
|
+
return Response({"error": "Invalid data", "details": serializer.errors}, status=status.HTTP_400_BAD_REQUEST)
|
|
226
|
+
|
|
227
|
+
elif request.method == "DELETE":
|
|
228
|
+
logger.info(f"Deleting segment {segment_id} from video {pk}")
|
|
229
|
+
try:
|
|
230
|
+
with transaction.atomic():
|
|
231
|
+
segment.delete()
|
|
232
|
+
logger.info(f"Successfully deleted segment {segment_id}")
|
|
233
|
+
return Response({"message": f"Segment {segment_id} deleted successfully"}, status=status.HTTP_204_NO_CONTENT)
|
|
234
|
+
except Exception as e:
|
|
235
|
+
logger.error(f"Error deleting segment {segment_id}: {str(e)}")
|
|
236
|
+
return Response({"error": f"Failed to delete segment: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
237
|
+
|
|
238
|
+
|
|
239
|
+
# ============================================================================
|
|
240
|
+
# VIDEO SEGMENT VALIDATION ENDPOINTS (Modern Framework)
|
|
241
|
+
# Migrated from /api/label-video-segment/*/validate/ (October 14, 2025)
|
|
242
|
+
# ============================================================================
|
|
243
|
+
|
|
244
|
+
|
|
245
|
+
@api_view(["POST"])
|
|
246
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
247
|
+
def video_segment_validate(request, pk: int, segment_id: int):
|
|
248
|
+
"""
|
|
249
|
+
Validate a single video segment.
|
|
250
|
+
|
|
251
|
+
POST /api/media/videos/<pk>/segments/<segment_id>/validate/
|
|
252
|
+
|
|
253
|
+
Validates a single LabelVideoSegment and marks it as verified.
|
|
254
|
+
Used to confirm user-reviewed segment annotations.
|
|
255
|
+
|
|
256
|
+
Request Body (optional):
|
|
257
|
+
{
|
|
258
|
+
"is_validated": true, // optional, default true
|
|
259
|
+
"notes": "..." // optional, validation notes
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
Response:
|
|
263
|
+
{
|
|
264
|
+
"message": "Segment validated successfully",
|
|
265
|
+
"segment_id": 123,
|
|
266
|
+
"is_validated": true,
|
|
267
|
+
"label": "polyp",
|
|
268
|
+
"video_id": 456,
|
|
269
|
+
"start_frame": 100,
|
|
270
|
+
"end_frame": 200
|
|
271
|
+
}
|
|
272
|
+
"""
|
|
273
|
+
# Verify video exists
|
|
274
|
+
video = get_object_or_404(VideoFile, pk=pk)
|
|
275
|
+
|
|
276
|
+
# Get segment and verify it belongs to this video
|
|
277
|
+
segment = get_object_or_404(LabelVideoSegment.objects.select_related("state", "video_file", "label"), pk=segment_id, video_file=video)
|
|
278
|
+
|
|
279
|
+
try:
|
|
280
|
+
# Validation status from request (default: True)
|
|
281
|
+
is_validated = request.data.get("is_validated", True)
|
|
282
|
+
notes = request.data.get("notes", "")
|
|
283
|
+
|
|
284
|
+
# Get or create state object
|
|
285
|
+
if not hasattr(segment, "state") or segment.state is None:
|
|
286
|
+
return Response({"error": "Segment has no state object. Cannot validate."}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
287
|
+
|
|
288
|
+
# Update state
|
|
289
|
+
with transaction.atomic():
|
|
290
|
+
segment.state.is_validated = is_validated
|
|
291
|
+
if notes and hasattr(segment.state, "validation_notes"):
|
|
292
|
+
segment.state.validation_notes = notes
|
|
293
|
+
segment.state.save()
|
|
294
|
+
|
|
295
|
+
logger.info(f"Validated segment {segment_id} in video {pk}: {is_validated}")
|
|
296
|
+
|
|
297
|
+
return Response(
|
|
298
|
+
{
|
|
299
|
+
"message": f"Segment {segment_id} validation status updated",
|
|
300
|
+
"segment_id": segment_id,
|
|
301
|
+
"is_validated": is_validated,
|
|
302
|
+
"label": segment.label.name if segment.label else None,
|
|
303
|
+
"video_id": video.id,
|
|
304
|
+
"start_frame": segment.start_frame_number,
|
|
305
|
+
"end_frame": segment.end_frame_number,
|
|
306
|
+
},
|
|
307
|
+
status=status.HTTP_200_OK,
|
|
308
|
+
)
|
|
309
|
+
|
|
310
|
+
except Exception as e:
|
|
311
|
+
logger.error(f"Error validating segment {segment_id} in video {pk}: {e}")
|
|
312
|
+
return Response({"error": f"Validation failed: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
313
|
+
|
|
314
|
+
|
|
315
|
+
@api_view(["POST"])
|
|
316
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
317
|
+
def video_segments_validate_bulk(request, pk: int):
|
|
318
|
+
"""
|
|
319
|
+
Validate multiple video segments at once.
|
|
320
|
+
|
|
321
|
+
POST /api/media/videos/<pk>/segments/validate-bulk/
|
|
322
|
+
|
|
323
|
+
Validates multiple LabelVideoSegments simultaneously.
|
|
324
|
+
Useful for batch validation after review.
|
|
325
|
+
|
|
326
|
+
Request Body:
|
|
327
|
+
{
|
|
328
|
+
"segment_ids": [1, 2, 3, ...],
|
|
329
|
+
"is_validated": true, // optional, default true
|
|
330
|
+
"notes": "..." // optional, applies to all segments
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
Response:
|
|
334
|
+
{
|
|
335
|
+
"message": "Bulk validation completed. 3 segments updated.",
|
|
336
|
+
"updated_count": 3,
|
|
337
|
+
"requested_count": 3,
|
|
338
|
+
"is_validated": true,
|
|
339
|
+
"failed_ids": [] // only present if some failed
|
|
340
|
+
}
|
|
341
|
+
"""
|
|
342
|
+
# Verify video exists
|
|
343
|
+
video = get_object_or_404(VideoFile, pk=pk)
|
|
344
|
+
|
|
345
|
+
segment_ids = request.data.get("segment_ids", [])
|
|
346
|
+
is_validated = request.data.get("is_validated", True)
|
|
347
|
+
notes = request.data.get("notes", "")
|
|
348
|
+
|
|
349
|
+
if not segment_ids:
|
|
350
|
+
return Response({"error": "segment_ids is required"}, status=status.HTTP_400_BAD_REQUEST)
|
|
351
|
+
|
|
352
|
+
try:
|
|
353
|
+
# Get all segments for this video only
|
|
354
|
+
segments = LabelVideoSegment.objects.filter(pk__in=segment_ids, video_file=video).select_related("state")
|
|
355
|
+
|
|
356
|
+
if not segments.exists():
|
|
357
|
+
return Response({"error": "No segments found with provided IDs for this video"}, status=status.HTTP_404_NOT_FOUND)
|
|
358
|
+
|
|
359
|
+
updated_count = 0
|
|
360
|
+
failed_ids = []
|
|
361
|
+
|
|
362
|
+
with transaction.atomic():
|
|
363
|
+
for segment in segments:
|
|
364
|
+
try:
|
|
365
|
+
if segment.state:
|
|
366
|
+
segment.state.is_validated = is_validated
|
|
367
|
+
if notes and hasattr(segment.state, "validation_notes"):
|
|
368
|
+
segment.state.validation_notes = notes
|
|
369
|
+
segment.state.save()
|
|
370
|
+
updated_count += 1
|
|
371
|
+
else:
|
|
372
|
+
failed_ids.append(segment.id)
|
|
373
|
+
except Exception as e:
|
|
374
|
+
logger.error(f"Error validating segment {segment.id}: {e}")
|
|
375
|
+
failed_ids.append(segment.id)
|
|
376
|
+
|
|
377
|
+
logger.info(f"Bulk validated {updated_count} segments in video {pk}")
|
|
378
|
+
|
|
379
|
+
response_data = {
|
|
380
|
+
"message": f"Bulk validation completed. {updated_count} segments updated.",
|
|
381
|
+
"updated_count": updated_count,
|
|
382
|
+
"requested_count": len(segment_ids),
|
|
383
|
+
"is_validated": is_validated,
|
|
384
|
+
"video_id": pk,
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
if failed_ids:
|
|
388
|
+
response_data["failed_ids"] = failed_ids
|
|
389
|
+
response_data["warning"] = f"{len(failed_ids)} segments could not be validated"
|
|
390
|
+
|
|
391
|
+
return Response(response_data, status=status.HTTP_200_OK)
|
|
392
|
+
|
|
393
|
+
except Exception as e:
|
|
394
|
+
logger.error(f"Error in bulk validation for video {pk}: {e}")
|
|
395
|
+
return Response({"error": f"Bulk validation failed: {str(e)}"}, status=status.HTTP_500_INTERNAL_SERVER_ERROR)
|
|
396
|
+
|
|
397
|
+
|
|
398
|
+
@api_view(["GET", "POST"])
|
|
399
|
+
@permission_classes([EnvironmentAwarePermission])
|
|
400
|
+
def video_segments_validation_status(request, pk: int):
|
|
401
|
+
"""
|
|
402
|
+
Get or update validation status for all segments of a video.
|
|
403
|
+
|
|
404
|
+
GET /api/media/videos/<pk>/segments/validation-status/
|
|
405
|
+
Returns validation statistics for all segments.
|
|
406
|
+
|
|
407
|
+
POST /api/media/videos/<pk>/segments/validation-status/
|
|
408
|
+
Marks all segments (or filtered by label) as validated.
|
|
409
|
+
|
|
410
|
+
Query Parameters (GET):
|
|
411
|
+
- label_name: filter by label (optional)
|
|
412
|
+
|
|
413
|
+
Request Body (POST, optional):
|
|
414
|
+
{
|
|
415
|
+
"label_name": "...", // optional, only validate segments with this label
|
|
416
|
+
"notes": "..." // optional
|
|
417
|
+
}
|
|
418
|
+
|
|
419
|
+
Response (GET):
|
|
420
|
+
{
|
|
421
|
+
"video_id": 123,
|
|
422
|
+
"total_segments": 10,
|
|
423
|
+
"validated_count": 7,
|
|
424
|
+
"unvalidated_count": 3,
|
|
425
|
+
"validation_complete": false,
|
|
426
|
+
"by_label": {...}
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
Response (POST):
|
|
430
|
+
{
|
|
431
|
+
"message": "Video segment validation completed",
|
|
432
|
+
"video_id": 123,
|
|
433
|
+
"total_segments": 10,
|
|
434
|
+
"updated_count": 10,
|
|
435
|
+
"failed_count": 0
|
|
436
|
+
}
|
|
437
|
+
"""
|
|
438
|
+
# Verify video exists
|
|
439
|
+
video = get_object_or_404(VideoFile, pk=pk)
|
|
440
|
+
|
|
441
|
+
if request.method == "GET":
|
|
442
|
+
# Get validation status
|
|
443
|
+
label_name = request.query_params.get("label_name")
|
|
444
|
+
|
|
445
|
+
segments_query = LabelVideoSegment.objects.filter(video_file=video).select_related("state", "label")
|
|
446
|
+
|
|
447
|
+
if label_name:
|
|
448
|
+
segments_query = segments_query.filter(label__name=label_name)
|
|
449
|
+
|
|
450
|
+
segments = segments_query.all()
|
|
451
|
+
total_count = segments.count()
|
|
452
|
+
|
|
453
|
+
# Count validated segments
|
|
454
|
+
validated_count = sum(1 for s in segments if s.state and s.state.is_validated)
|
|
455
|
+
|
|
456
|
+
# By label breakdown
|
|
457
|
+
by_label = {}
|
|
458
|
+
for segment in segments:
|
|
459
|
+
label = segment.label.name if segment.label else "unknown"
|
|
460
|
+
if label not in by_label:
|
|
461
|
+
by_label[label] = {"total": 0, "validated": 0}
|
|
462
|
+
by_label[label]["total"] += 1
|
|
463
|
+
if segment.state and segment.state.is_validated:
|
|
464
|
+
by_label[label]["validated"] += 1
|
|
465
|
+
|
|
466
|
+
return Response(
|
|
467
|
+
{
|
|
468
|
+
"video_id": pk,
|
|
469
|
+
"total_segments": total_count,
|
|
470
|
+
"validated_count": validated_count,
|
|
471
|
+
"unvalidated_count": total_count - validated_count,
|
|
472
|
+
"validation_complete": validated_count == total_count and total_count > 0,
|
|
473
|
+
"by_label": by_label,
|
|
474
|
+
"label_filter": label_name,
|
|
475
|
+
},
|
|
476
|
+
status=status.HTTP_200_OK,
|
|
477
|
+
)
|
|
478
|
+
|
|
479
|
+
elif request.method == "POST":
|
|
480
|
+
# Mark all segments as validated
|
|
481
|
+
label_name = request.data.get("label_name")
|
|
482
|
+
notes = request.data.get("notes", "")
|
|
483
|
+
|
|
484
|
+
segments_query = LabelVideoSegment.objects.filter(video_file=video).select_related("state", "label")
|
|
485
|
+
|
|
486
|
+
if label_name:
|
|
487
|
+
segments_query = segments_query.filter(label__name=label_name)
|
|
488
|
+
|
|
489
|
+
segments = segments_query.all()
|
|
490
|
+
|
|
491
|
+
if not segments.exists():
|
|
492
|
+
return Response({"message": "No segments found to validate", "video_id": pk, "updated_count": 0}, status=status.HTTP_200_OK)
|
|
493
|
+
|
|
494
|
+
updated_count = 0
|
|
495
|
+
failed_count = 0
|
|
496
|
+
|
|
497
|
+
with transaction.atomic():
|
|
498
|
+
for segment in segments:
|
|
499
|
+
try:
|
|
500
|
+
if segment.state:
|
|
501
|
+
segment.state.is_validated = True
|
|
502
|
+
if notes and hasattr(segment.state, "validation_notes"):
|
|
503
|
+
segment.state.validation_notes = notes
|
|
504
|
+
segment.state.save()
|
|
505
|
+
updated_count += 1
|
|
506
|
+
else:
|
|
507
|
+
failed_count += 1
|
|
508
|
+
except Exception as e:
|
|
509
|
+
logger.error(f"Error validating segment {segment.id}: {e}")
|
|
510
|
+
failed_count += 1
|
|
511
|
+
|
|
512
|
+
logger.info(f"Completed validation for {updated_count} segments in video {pk}")
|
|
513
|
+
|
|
514
|
+
return Response(
|
|
515
|
+
{
|
|
516
|
+
"message": f"Video segment validation completed for video {pk}",
|
|
517
|
+
"video_id": pk,
|
|
518
|
+
"total_segments": len(segments),
|
|
519
|
+
"updated_count": updated_count,
|
|
520
|
+
"failed_count": failed_count,
|
|
521
|
+
"label_filter": label_name,
|
|
522
|
+
},
|
|
523
|
+
status=status.HTTP_200_OK,
|
|
524
|
+
)
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
from .available_files_list import AvailableFilesListView
|
|
2
|
+
from .sensitive_meta_detail import SensitiveMetaDetailView
|
|
3
|
+
from .sensitive_meta_list import SensitiveMetaListView
|
|
4
|
+
from .sensitive_meta_verification import SensitiveMetaVerificationView
|
|
5
|
+
from .report_meta import (
|
|
6
|
+
ReportFileMetadataView,
|
|
7
|
+
)
|
|
8
|
+
|
|
9
|
+
__all__ = [
|
|
10
|
+
"AvailableFilesListView",
|
|
11
|
+
"SensitiveMetaDetailView",
|
|
12
|
+
"SensitiveMetaListView",
|
|
13
|
+
"SensitiveMetaVerificationView",
|
|
14
|
+
"ReportFileMetadataView",
|
|
15
|
+
]
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
from endoreg_db.models import RawPdfFile, VideoFile
|
|
2
|
+
from endoreg_db.utils.permissions import DEBUG_PERMISSIONS
|
|
3
|
+
import logging
|
|
4
|
+
|
|
5
|
+
from rest_framework import status
|
|
6
|
+
from rest_framework.response import Response
|
|
7
|
+
from rest_framework.views import APIView
|
|
8
|
+
|
|
9
|
+
logger = logging.getLogger(__name__)
|
|
10
|
+
|
|
11
|
+
class AvailableFilesListView(APIView):
|
|
12
|
+
"""
|
|
13
|
+
API endpoint to list available PDFs and videos for anonymization selection.
|
|
14
|
+
|
|
15
|
+
GET: Returns lists of available PDF and video files with their metadata
|
|
16
|
+
"""
|
|
17
|
+
|
|
18
|
+
permission_classes = DEBUG_PERMISSIONS
|
|
19
|
+
|
|
20
|
+
def _validate_pagination_params(self, request):
|
|
21
|
+
"""Validates and returns pagination parameters, raising ValueError on failure."""
|
|
22
|
+
try:
|
|
23
|
+
limit = int(request.query_params.get('limit', 50))
|
|
24
|
+
offset = int(request.query_params.get('offset', 0))
|
|
25
|
+
except (ValueError, TypeError):
|
|
26
|
+
raise ValueError("Invalid 'limit' or 'offset' parameter. Must be integers.")
|
|
27
|
+
|
|
28
|
+
if limit < 0 or offset < 0:
|
|
29
|
+
raise ValueError("'limit' and 'offset' must be non-negative.")
|
|
30
|
+
|
|
31
|
+
if limit > 100:
|
|
32
|
+
logger.warning(f"Client requested limit of {limit}, capping at 100.")
|
|
33
|
+
limit = 100
|
|
34
|
+
|
|
35
|
+
return limit, offset
|
|
36
|
+
|
|
37
|
+
def get(self, request):
|
|
38
|
+
"""
|
|
39
|
+
List available PDF and video files for anonymization selection.
|
|
40
|
+
|
|
41
|
+
Query Parameters:
|
|
42
|
+
- type: Filter by file type ('pdf' or 'video')
|
|
43
|
+
- status: Filter by anonymization status
|
|
44
|
+
- limit: Number of results to return (default 50, max 100)
|
|
45
|
+
- offset: Offset for pagination (default 0)
|
|
46
|
+
|
|
47
|
+
Returns:
|
|
48
|
+
{
|
|
49
|
+
"pdfs": [...],
|
|
50
|
+
"videos": [...],
|
|
51
|
+
"total_pdfs": N,
|
|
52
|
+
"total_videos": N
|
|
53
|
+
}
|
|
54
|
+
"""
|
|
55
|
+
try:
|
|
56
|
+
limit, offset = self._validate_pagination_params(request)
|
|
57
|
+
file_type = request.query_params.get('type', 'all').lower()
|
|
58
|
+
except ValueError as e:
|
|
59
|
+
return Response({"error": str(e)}, status=status.HTTP_400_BAD_REQUEST)
|
|
60
|
+
|
|
61
|
+
try:
|
|
62
|
+
response_data = {}
|
|
63
|
+
|
|
64
|
+
# Get PDFs if requested
|
|
65
|
+
if file_type in ['all', 'pdf']:
|
|
66
|
+
pdf_queryset = RawPdfFile.objects.select_related('sensitive_meta').all()
|
|
67
|
+
total_pdfs = pdf_queryset.count()
|
|
68
|
+
paginated_pdfs = pdf_queryset[offset:offset + limit]
|
|
69
|
+
|
|
70
|
+
pdf_list = []
|
|
71
|
+
for pdf in paginated_pdfs:
|
|
72
|
+
# Safely handle missing file attribute
|
|
73
|
+
file_name = 'Unknown'
|
|
74
|
+
file_path = None
|
|
75
|
+
if hasattr(pdf, 'file') and pdf.file:
|
|
76
|
+
file_name = pdf.file.name.split('/')[-1]
|
|
77
|
+
file_path = pdf.file.name
|
|
78
|
+
pdf_data = {
|
|
79
|
+
'id': pdf.id,
|
|
80
|
+
'filename': file_name,
|
|
81
|
+
'file_path': file_path,
|
|
82
|
+
'sensitive_meta_id': pdf.sensitive_meta_id,
|
|
83
|
+
'anonymized_text': getattr(pdf, 'anonymized_text', None),
|
|
84
|
+
'created_at': pdf.date_created if hasattr(pdf, 'created_at') else None,
|
|
85
|
+
'patient_info': None
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
# Add patient info if available
|
|
89
|
+
if pdf.sensitive_meta:
|
|
90
|
+
pdf_data['patient_info'] = {
|
|
91
|
+
'patient_first_name': pdf.sensitive_meta.patient_first_name,
|
|
92
|
+
'patient_last_name': pdf.sensitive_meta.patient_last_name,
|
|
93
|
+
'patient_dob': pdf.sensitive_meta.patient_dob,
|
|
94
|
+
'examination_date': pdf.sensitive_meta.examination_date,
|
|
95
|
+
'center_name': pdf.sensitive_meta.center.name if pdf.sensitive_meta.center else None
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
pdf_list.append(pdf_data)
|
|
99
|
+
|
|
100
|
+
response_data['pdfs'] = pdf_list
|
|
101
|
+
response_data['total_pdfs'] = total_pdfs
|
|
102
|
+
|
|
103
|
+
# Get Videos if requested
|
|
104
|
+
if file_type in ['all', 'video']:
|
|
105
|
+
video_queryset = VideoFile.objects.select_related('sensitive_meta').all()
|
|
106
|
+
total_videos = video_queryset.count()
|
|
107
|
+
paginated_videos = video_queryset[offset:offset + limit]
|
|
108
|
+
|
|
109
|
+
video_list = []
|
|
110
|
+
for video in paginated_videos:
|
|
111
|
+
# Safely handle missing raw_file attribute
|
|
112
|
+
file_name = 'Unknown'
|
|
113
|
+
file_path = None
|
|
114
|
+
if hasattr(video, 'raw_file') and video.raw_file:
|
|
115
|
+
file_name = video.raw_file.name.split('/')[-1]
|
|
116
|
+
file_path = video.raw_file.name
|
|
117
|
+
video_data = {
|
|
118
|
+
'id': video.id,
|
|
119
|
+
'filename': file_name,
|
|
120
|
+
'file_path': file_path,
|
|
121
|
+
'sensitive_meta_id': video.sensitive_meta_id,
|
|
122
|
+
'patient_info': None
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
# Add patient info if available
|
|
126
|
+
if video.sensitive_meta:
|
|
127
|
+
video_data['patient_info'] = {
|
|
128
|
+
'patient_first_name': video.sensitive_meta.patient_first_name,
|
|
129
|
+
'patient_last_name': video.sensitive_meta.patient_last_name,
|
|
130
|
+
'patient_dob': video.sensitive_meta.patient_dob,
|
|
131
|
+
'examination_date': video.sensitive_meta.examination_date,
|
|
132
|
+
'center_name': video.sensitive_meta.center.name if video.sensitive_meta.center else None
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
video_list.append(video_data)
|
|
136
|
+
|
|
137
|
+
response_data['videos'] = video_list
|
|
138
|
+
response_data['total_videos'] = total_videos
|
|
139
|
+
|
|
140
|
+
return Response(response_data)
|
|
141
|
+
except Exception as e:
|
|
142
|
+
logger.error(f"Error listing available files: {e}", exc_info=True)
|
|
143
|
+
return Response(
|
|
144
|
+
{"error": "An unexpected error occurred while fetching files."},
|
|
145
|
+
status=status.HTTP_500_INTERNAL_SERVER_ERROR
|
|
146
|
+
)
|