endoreg-db 0.8.6.1__py3-none-any.whl → 0.8.8.9__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/authz/auth.py +74 -0
- endoreg_db/authz/backends.py +168 -0
- endoreg_db/authz/management/commands/list_routes.py +18 -0
- endoreg_db/authz/middleware.py +83 -0
- endoreg_db/authz/permissions.py +127 -0
- endoreg_db/authz/policy.py +218 -0
- endoreg_db/authz/views_auth.py +66 -0
- endoreg_db/config/env.py +13 -8
- endoreg_db/data/__init__.py +2 -11
- endoreg_db/data/ai_model_meta/default_multilabel_classification.yaml +3 -3
- endoreg_db/data/event_classification/data.yaml +4 -0
- endoreg_db/data/event_classification_choice/data.yaml +9 -0
- endoreg_db/data/examination/examinations/data.yaml +114 -14
- endoreg_db/data/examination/time-type/data.yaml +0 -3
- endoreg_db/data/examination_indication/endoscopy.yaml +108 -173
- endoreg_db/data/examination_indication_classification/endoscopy.yaml +0 -70
- endoreg_db/data/examination_indication_classification_choice/endoscopy.yaml +33 -37
- endoreg_db/data/finding/00_generic.yaml +35 -0
- endoreg_db/data/finding/00_generic_complication.yaml +9 -0
- endoreg_db/data/finding/01_gastroscopy_baseline.yaml +88 -0
- endoreg_db/data/finding/01_gastroscopy_observation.yaml +113 -0
- endoreg_db/data/finding/02_colonoscopy_baseline.yaml +53 -0
- endoreg_db/data/finding/02_colonoscopy_hidden.yaml +119 -0
- endoreg_db/data/finding/02_colonoscopy_observation.yaml +152 -0
- endoreg_db/data/finding_classification/00_generic.yaml +44 -0
- endoreg_db/data/finding_classification/00_generic_histology.yaml +28 -0
- endoreg_db/data/finding_classification/00_generic_lesion.yaml +52 -0
- endoreg_db/data/finding_classification/02_colonoscopy_baseline.yaml +83 -0
- endoreg_db/data/finding_classification/02_colonoscopy_histology.yaml +13 -0
- endoreg_db/data/finding_classification/02_colonoscopy_other.yaml +12 -0
- endoreg_db/data/finding_classification/02_colonoscopy_polyp.yaml +101 -0
- endoreg_db/data/finding_classification_choice/{yes_no_na.yaml → 00_generic.yaml} +5 -1
- endoreg_db/data/finding_classification_choice/{examination_setting_generic_types.yaml → 00_generic_baseline.yaml} +10 -2
- endoreg_db/data/finding_classification_choice/{complication_generic_types.yaml → 00_generic_complication.yaml} +1 -1
- endoreg_db/data/finding_classification_choice/{histology.yaml → 00_generic_histology.yaml} +1 -4
- endoreg_db/data/finding_classification_choice/00_generic_lesion.yaml +158 -0
- endoreg_db/data/finding_classification_choice/{bowel_preparation.yaml → 02_colonoscopy_bowel_preparation.yaml} +1 -30
- endoreg_db/data/finding_classification_choice/{colonoscopy_not_complete_reason.yaml → 02_colonoscopy_generic.yaml} +1 -1
- endoreg_db/data/finding_classification_choice/{histology_polyp.yaml → 02_colonoscopy_histology.yaml} +1 -1
- endoreg_db/data/finding_classification_choice/{colonoscopy_location.yaml → 02_colonoscopy_location.yaml} +23 -4
- endoreg_db/data/finding_classification_choice/02_colonoscopy_other.yaml +34 -0
- endoreg_db/data/finding_classification_choice/02_colonoscopy_polyp_advanced_imaging.yaml +76 -0
- endoreg_db/data/finding_classification_choice/{colon_lesion_paris.yaml → 02_colonoscopy_polyp_morphology.yaml} +26 -8
- endoreg_db/data/finding_classification_choice/02_colonoscopy_size.yaml +27 -0
- endoreg_db/data/finding_classification_type/{colonoscopy_basic.yaml → 00_generic.yaml} +18 -13
- endoreg_db/data/finding_classification_type/02_colonoscopy.yaml +9 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy.yaml +59 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy_ablation.yaml +44 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy_bleeding.yaml +55 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy_resection.yaml +85 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy_stenosis.yaml +17 -0
- endoreg_db/data/finding_intervention/00_generic_endoscopy_stent.yaml +9 -0
- endoreg_db/data/finding_intervention/01_gastroscopy.yaml +19 -0
- endoreg_db/data/finding_intervention/04_eus.yaml +39 -0
- endoreg_db/data/finding_intervention/05_ercp.yaml +3 -0
- endoreg_db/data/finding_type/data.yaml +8 -12
- endoreg_db/data/requirement/01_patient_data.yaml +93 -0
- endoreg_db/data/requirement/old/colon_polyp_intervention.yaml +49 -0
- endoreg_db/data/requirement/old/coloreg_colon_polyp.yaml +49 -0
- endoreg_db/data/requirement_operator/new_operators.yaml +36 -0
- endoreg_db/data/requirement_set/01_endoscopy_generic.yaml +29 -12
- endoreg_db/data/requirement_set/01_laboratory.yaml +13 -0
- endoreg_db/data/requirement_set/{endoscopy_bleeding_risk.yaml → 02_endoscopy_bleeding_risk.yaml} +0 -6
- endoreg_db/data/requirement_set/90_coloreg.yaml +190 -0
- endoreg_db/data/requirement_set/_old_ +109 -0
- endoreg_db/data/requirement_set_type/data.yaml +21 -0
- endoreg_db/data/setup_config.yaml +4 -4
- endoreg_db/data/tag/requirement_set_tags.yaml +21 -0
- endoreg_db/exceptions.py +4 -2
- endoreg_db/forms/examination_form.py +1 -1
- endoreg_db/helpers/data_loader.py +125 -53
- endoreg_db/helpers/default_objects.py +116 -81
- endoreg_db/import_files/__init__.py +27 -0
- endoreg_db/import_files/context/__init__.py +7 -0
- endoreg_db/import_files/context/default_sensitive_meta.py +81 -0
- endoreg_db/import_files/context/ensure_center.py +17 -0
- endoreg_db/import_files/context/file_lock.py +66 -0
- endoreg_db/import_files/context/import_context.py +43 -0
- endoreg_db/import_files/context/validate_directories.py +56 -0
- endoreg_db/import_files/file_storage/__init__.py +15 -0
- endoreg_db/import_files/file_storage/create_report_file.py +76 -0
- endoreg_db/import_files/file_storage/create_video_file.py +75 -0
- endoreg_db/import_files/file_storage/sensitive_meta_storage.py +39 -0
- endoreg_db/import_files/file_storage/state_management.py +400 -0
- endoreg_db/import_files/file_storage/storage.py +36 -0
- endoreg_db/import_files/import_service.md +26 -0
- endoreg_db/import_files/processing/__init__.py +11 -0
- endoreg_db/import_files/processing/report_processing/report_anonymization.py +94 -0
- endoreg_db/import_files/processing/sensitive_meta_adapter.py +51 -0
- endoreg_db/import_files/processing/video_processing/video_anonymization.py +107 -0
- endoreg_db/import_files/processing/video_processing/video_cleanup_on_error.py +119 -0
- endoreg_db/import_files/pseudonymization/fake.py +52 -0
- endoreg_db/import_files/pseudonymization/k_anonymity.py +182 -0
- endoreg_db/import_files/pseudonymization/k_pseudonymity.py +128 -0
- endoreg_db/import_files/report_import_service.py +141 -0
- endoreg_db/import_files/video_import_service.py +150 -0
- endoreg_db/management/commands/create_model_meta_from_huggingface.py +21 -10
- endoreg_db/management/commands/create_multilabel_model_meta.py +299 -129
- endoreg_db/management/commands/import_report.py +130 -65
- endoreg_db/management/commands/import_video.py +9 -10
- endoreg_db/management/commands/import_video_with_classification.py +2 -2
- endoreg_db/management/commands/list_routes.py +18 -0
- endoreg_db/management/commands/load_ai_model_data.py +5 -5
- endoreg_db/management/commands/load_ai_model_label_data.py +9 -7
- endoreg_db/management/commands/load_base_db_data.py +5 -134
- endoreg_db/management/commands/load_center_data.py +12 -12
- endoreg_db/management/commands/load_contraindication_data.py +14 -16
- endoreg_db/management/commands/load_disease_classification_choices_data.py +15 -18
- endoreg_db/management/commands/load_disease_classification_data.py +15 -18
- endoreg_db/management/commands/load_disease_data.py +25 -28
- endoreg_db/management/commands/load_endoscope_data.py +20 -27
- endoreg_db/management/commands/load_event_data.py +14 -16
- endoreg_db/management/commands/load_examination_data.py +31 -44
- endoreg_db/management/commands/load_examination_indication_data.py +20 -21
- endoreg_db/management/commands/load_finding_data.py +52 -80
- endoreg_db/management/commands/load_information_source.py +21 -23
- endoreg_db/management/commands/load_lab_value_data.py +17 -26
- endoreg_db/management/commands/load_medication_data.py +13 -12
- endoreg_db/management/commands/load_organ_data.py +15 -19
- endoreg_db/management/commands/load_pdf_type_data.py +19 -18
- endoreg_db/management/commands/load_profession_data.py +14 -17
- endoreg_db/management/commands/load_qualification_data.py +20 -23
- endoreg_db/management/commands/load_report_reader_flag_data.py +17 -19
- endoreg_db/management/commands/load_requirement_data.py +62 -39
- endoreg_db/management/commands/load_requirement_set_tags.py +95 -0
- endoreg_db/management/commands/load_risk_data.py +7 -6
- endoreg_db/management/commands/load_shift_data.py +20 -23
- endoreg_db/management/commands/load_tag_data.py +8 -11
- endoreg_db/management/commands/load_unit_data.py +17 -19
- endoreg_db/management/commands/setup_endoreg_db.py +3 -3
- endoreg_db/management/commands/start_filewatcher.py +46 -37
- endoreg_db/management/commands/storage_management.py +271 -203
- endoreg_db/management/commands/validate_video_files.py +1 -5
- endoreg_db/migrations/0001_initial.py +297 -250
- endoreg_db/models/__init__.py +78 -123
- endoreg_db/models/administration/__init__.py +21 -42
- endoreg_db/models/administration/ai/active_model.py +2 -2
- endoreg_db/models/administration/ai/ai_model.py +7 -6
- endoreg_db/models/administration/case/__init__.py +1 -15
- endoreg_db/models/administration/case/case.py +3 -3
- endoreg_db/models/administration/case/case_template/__init__.py +2 -14
- endoreg_db/models/administration/case/case_template/case_template.py +2 -124
- endoreg_db/models/administration/case/case_template/case_template_rule.py +2 -268
- endoreg_db/models/administration/case/case_template/case_template_rule_value.py +2 -85
- endoreg_db/models/administration/case/case_template/case_template_type.py +2 -25
- endoreg_db/models/administration/center/center.py +33 -19
- endoreg_db/models/administration/center/center_product.py +12 -9
- endoreg_db/models/administration/center/center_resource.py +25 -19
- endoreg_db/models/administration/center/center_shift.py +21 -17
- endoreg_db/models/administration/center/center_waste.py +16 -8
- endoreg_db/models/administration/person/__init__.py +2 -0
- endoreg_db/models/administration/person/employee/employee.py +10 -5
- endoreg_db/models/administration/person/employee/employee_qualification.py +9 -4
- endoreg_db/models/administration/person/employee/employee_type.py +12 -6
- endoreg_db/models/administration/person/examiner/examiner.py +13 -11
- endoreg_db/models/administration/person/patient/__init__.py +2 -0
- endoreg_db/models/administration/person/patient/patient.py +129 -100
- endoreg_db/models/administration/person/patient/patient_external_id.py +37 -0
- endoreg_db/models/administration/person/person.py +4 -0
- endoreg_db/models/administration/person/profession/__init__.py +8 -4
- endoreg_db/models/administration/person/user/portal_user_information.py +11 -7
- endoreg_db/models/administration/product/product.py +20 -15
- endoreg_db/models/administration/product/product_material.py +17 -18
- endoreg_db/models/administration/product/product_weight.py +12 -8
- endoreg_db/models/administration/product/reference_product.py +23 -55
- endoreg_db/models/administration/qualification/qualification.py +7 -3
- endoreg_db/models/administration/qualification/qualification_type.py +7 -3
- endoreg_db/models/administration/shift/scheduled_days.py +8 -5
- endoreg_db/models/administration/shift/shift.py +16 -12
- endoreg_db/models/administration/shift/shift_type.py +23 -31
- endoreg_db/models/label/__init__.py +8 -9
- endoreg_db/models/label/annotation/image_classification.py +10 -9
- endoreg_db/models/label/annotation/video_segmentation_annotation.py +23 -28
- endoreg_db/models/label/label.py +15 -15
- endoreg_db/models/label/label_set.py +19 -6
- endoreg_db/models/label/label_type.py +1 -1
- endoreg_db/models/label/label_video_segment/_create_from_video.py +5 -8
- endoreg_db/models/label/label_video_segment/label_video_segment.py +98 -102
- endoreg_db/models/label/video_segmentation_label.py +4 -0
- endoreg_db/models/label/video_segmentation_labelset.py +4 -3
- endoreg_db/models/media/frame/frame.py +22 -22
- endoreg_db/models/media/pdf/raw_pdf.py +194 -194
- endoreg_db/models/media/pdf/report_file.py +25 -29
- endoreg_db/models/media/pdf/report_reader/report_reader_config.py +55 -47
- endoreg_db/models/media/pdf/report_reader/report_reader_flag.py +23 -7
- endoreg_db/models/media/processing_history/__init__.py +5 -0
- endoreg_db/models/media/processing_history/processing_history.py +96 -0
- endoreg_db/models/media/video/__init__.py +1 -0
- endoreg_db/models/media/video/create_from_file.py +139 -77
- endoreg_db/models/media/video/pipe_2.py +8 -9
- endoreg_db/models/media/video/video_file.py +174 -112
- endoreg_db/models/media/video/video_file_ai.py +288 -74
- endoreg_db/models/media/video/video_file_anonymize.py +38 -38
- endoreg_db/models/media/video/video_file_frames/__init__.py +3 -1
- endoreg_db/models/media/video/video_file_frames/_bulk_create_frames.py +6 -8
- endoreg_db/models/media/video/video_file_frames/_create_frame_object.py +7 -9
- endoreg_db/models/media/video/video_file_frames/_delete_frames.py +9 -8
- endoreg_db/models/media/video/video_file_frames/_extract_frames.py +38 -45
- endoreg_db/models/media/video/video_file_frames/_get_frame.py +6 -8
- endoreg_db/models/media/video/video_file_frames/_get_frame_number.py +4 -18
- endoreg_db/models/media/video/video_file_frames/_get_frame_path.py +4 -3
- endoreg_db/models/media/video/video_file_frames/_get_frame_paths.py +7 -6
- endoreg_db/models/media/video/video_file_frames/_get_frame_range.py +6 -8
- endoreg_db/models/media/video/video_file_frames/_get_frames.py +6 -8
- endoreg_db/models/media/video/video_file_frames/_initialize_frames.py +15 -25
- endoreg_db/models/media/video/video_file_frames/_manage_frame_range.py +26 -23
- endoreg_db/models/media/video/video_file_frames/_mark_frames_extracted_status.py +23 -14
- endoreg_db/models/media/video/video_file_io.py +113 -61
- endoreg_db/models/media/video/video_file_meta/get_crop_template.py +3 -3
- endoreg_db/models/media/video/video_file_meta/get_endo_roi.py +5 -3
- endoreg_db/models/media/video/video_file_meta/get_fps.py +37 -34
- endoreg_db/models/media/video/video_file_meta/initialize_video_specs.py +19 -25
- endoreg_db/models/media/video/video_file_meta/text_meta.py +41 -38
- endoreg_db/models/media/video/video_file_meta/video_meta.py +14 -7
- endoreg_db/models/media/video/video_file_segments.py +24 -17
- endoreg_db/models/media/video/video_metadata.py +19 -35
- endoreg_db/models/media/video/video_processing.py +96 -95
- endoreg_db/models/medical/contraindication/README.md +1 -0
- endoreg_db/models/medical/contraindication/__init__.py +13 -3
- endoreg_db/models/medical/disease.py +22 -16
- endoreg_db/models/medical/event.py +31 -18
- endoreg_db/models/medical/examination/__init__.py +13 -6
- endoreg_db/models/medical/examination/examination.py +39 -20
- endoreg_db/models/medical/examination/examination_indication.py +30 -95
- endoreg_db/models/medical/examination/examination_time.py +23 -8
- endoreg_db/models/medical/examination/examination_time_type.py +9 -6
- endoreg_db/models/medical/examination/examination_type.py +3 -4
- endoreg_db/models/medical/finding/finding.py +32 -40
- endoreg_db/models/medical/finding/finding_classification.py +42 -72
- endoreg_db/models/medical/finding/finding_intervention.py +25 -22
- endoreg_db/models/medical/finding/finding_type.py +13 -12
- endoreg_db/models/medical/hardware/endoscope.py +26 -26
- endoreg_db/models/medical/hardware/endoscopy_processor.py +2 -2
- endoreg_db/models/medical/laboratory/lab_value.py +62 -91
- endoreg_db/models/medical/medication/medication.py +22 -10
- endoreg_db/models/medical/medication/medication_indication.py +29 -3
- endoreg_db/models/medical/medication/medication_indication_type.py +25 -14
- endoreg_db/models/medical/medication/medication_intake_time.py +31 -19
- endoreg_db/models/medical/medication/medication_schedule.py +27 -16
- endoreg_db/models/medical/organ/__init__.py +15 -12
- endoreg_db/models/medical/patient/medication_examples.py +6 -6
- endoreg_db/models/medical/patient/patient_disease.py +20 -23
- endoreg_db/models/medical/patient/patient_event.py +19 -22
- endoreg_db/models/medical/patient/patient_examination.py +48 -54
- endoreg_db/models/medical/patient/patient_examination_indication.py +16 -14
- endoreg_db/models/medical/patient/patient_finding.py +122 -139
- endoreg_db/models/medical/patient/patient_finding_classification.py +44 -49
- endoreg_db/models/medical/patient/patient_finding_intervention.py +8 -19
- endoreg_db/models/medical/patient/patient_lab_sample.py +28 -23
- endoreg_db/models/medical/patient/patient_lab_value.py +82 -89
- endoreg_db/models/medical/patient/patient_medication.py +27 -38
- endoreg_db/models/medical/patient/patient_medication_schedule.py +28 -36
- endoreg_db/models/medical/risk/risk.py +7 -6
- endoreg_db/models/medical/risk/risk_type.py +8 -5
- endoreg_db/models/metadata/model_meta.py +60 -29
- endoreg_db/models/metadata/model_meta_logic.py +125 -18
- endoreg_db/models/metadata/pdf_meta.py +31 -24
- endoreg_db/models/metadata/sensitive_meta.py +105 -85
- endoreg_db/models/metadata/sensitive_meta_logic.py +198 -103
- endoreg_db/models/metadata/video_meta.py +51 -31
- endoreg_db/models/metadata/video_prediction_logic.py +16 -23
- endoreg_db/models/metadata/video_prediction_meta.py +29 -33
- endoreg_db/models/other/distribution/date_value_distribution.py +89 -29
- endoreg_db/models/other/distribution/multiple_categorical_value_distribution.py +21 -5
- endoreg_db/models/other/distribution/numeric_value_distribution.py +114 -53
- endoreg_db/models/other/distribution/single_categorical_value_distribution.py +4 -3
- endoreg_db/models/other/emission/emission_factor.py +18 -8
- endoreg_db/models/other/gender.py +10 -5
- endoreg_db/models/other/information_source.py +50 -29
- endoreg_db/models/other/material.py +9 -5
- endoreg_db/models/other/resource.py +6 -4
- endoreg_db/models/other/tag.py +10 -5
- endoreg_db/models/other/transport_route.py +13 -8
- endoreg_db/models/other/unit.py +10 -6
- endoreg_db/models/other/waste.py +6 -5
- endoreg_db/models/report/report.py +6 -0
- endoreg_db/models/requirement/requirement.py +329 -361
- endoreg_db/models/requirement/requirement_error.py +85 -0
- endoreg_db/models/requirement/requirement_evaluation/evaluate_with_dependencies.py +268 -0
- endoreg_db/models/requirement/requirement_evaluation/operator_evaluation_models.py +3 -6
- endoreg_db/models/requirement/requirement_evaluation/requirement_type_parser.py +90 -64
- endoreg_db/models/requirement/requirement_operator.py +103 -112
- endoreg_db/models/requirement/requirement_set.py +74 -57
- endoreg_db/models/state/__init__.py +4 -4
- endoreg_db/models/state/abstract.py +2 -2
- endoreg_db/models/state/anonymization.py +12 -0
- endoreg_db/models/state/audit_ledger.py +49 -51
- endoreg_db/models/state/label_video_segment.py +9 -0
- endoreg_db/models/state/raw_pdf.py +101 -68
- endoreg_db/models/state/sensitive_meta.py +6 -2
- endoreg_db/models/state/video.py +110 -90
- endoreg_db/models/upload_job.py +35 -34
- endoreg_db/models/utils.py +28 -25
- endoreg_db/queries/__init__.py +3 -1
- endoreg_db/root_urls.py +21 -2
- endoreg_db/schemas/examination_evaluation.py +1 -1
- endoreg_db/serializers/__init__.py +2 -10
- endoreg_db/serializers/anonymization.py +18 -10
- endoreg_db/serializers/label_video_segment/label_video_segment.py +2 -29
- endoreg_db/serializers/meta/__init__.py +1 -6
- endoreg_db/serializers/meta/sensitive_meta_detail.py +63 -118
- endoreg_db/serializers/misc/file_overview.py +11 -99
- endoreg_db/serializers/misc/sensitive_patient_data.py +50 -26
- endoreg_db/serializers/patient_examination/patient_examination.py +3 -3
- endoreg_db/serializers/pdf/anony_text_validation.py +39 -23
- endoreg_db/serializers/requirements/requirement_sets.py +92 -22
- endoreg_db/serializers/video/segmentation.py +2 -1
- endoreg_db/serializers/video/video_file_list.py +65 -34
- endoreg_db/serializers/video/video_processing_history.py +20 -5
- endoreg_db/services/__old/pdf_import.py +1487 -0
- endoreg_db/services/__old/video_import.py +1306 -0
- endoreg_db/services/anonymization.py +128 -89
- endoreg_db/services/lookup_service.py +65 -52
- endoreg_db/services/lookup_store.py +2 -2
- endoreg_db/services/pdf_import.py +0 -1382
- endoreg_db/services/report_import.py +10 -0
- endoreg_db/services/video_import.py +6 -1255
- endoreg_db/tasks/upload_tasks.py +79 -70
- endoreg_db/tasks/video_ingest.py +8 -4
- endoreg_db/urls/__init__.py +5 -32
- endoreg_db/urls/ai.py +32 -0
- endoreg_db/urls/media.py +121 -83
- endoreg_db/urls/root_urls.py +29 -0
- endoreg_db/utils/__init__.py +15 -5
- endoreg_db/utils/ai/multilabel_classification_net.py +116 -20
- endoreg_db/utils/case_generator/__init__.py +3 -0
- endoreg_db/utils/dataloader.py +142 -40
- endoreg_db/utils/defaults/set_default_center.py +32 -0
- endoreg_db/utils/names.py +22 -16
- endoreg_db/utils/paths.py +110 -46
- endoreg_db/utils/permissions.py +2 -1
- endoreg_db/utils/pipelines/Readme.md +1 -1
- endoreg_db/utils/pipelines/process_video_dir.py +1 -1
- endoreg_db/utils/requirement_operator_logic/_old/model_evaluators.py +655 -0
- endoreg_db/utils/requirement_operator_logic/new_operator_logic.py +97 -0
- endoreg_db/utils/setup_config.py +8 -5
- endoreg_db/utils/storage.py +115 -0
- endoreg_db/utils/validate_endo_roi.py +8 -2
- endoreg_db/utils/video/ffmpeg_wrapper.py +184 -188
- endoreg_db/views/__init__.py +85 -183
- endoreg_db/views/ai/__init__.py +8 -0
- endoreg_db/views/ai/label.py +155 -0
- endoreg_db/views/anonymization/media_management.py +202 -166
- endoreg_db/views/anonymization/overview.py +99 -67
- endoreg_db/views/anonymization/validate.py +182 -44
- endoreg_db/views/media/__init__.py +7 -20
- endoreg_db/views/media/pdf_media.py +197 -174
- endoreg_db/views/media/sensitive_metadata.py +193 -138
- endoreg_db/views/media/video_media.py +89 -82
- endoreg_db/views/meta/__init__.py +0 -8
- endoreg_db/views/misc/__init__.py +1 -7
- endoreg_db/views/misc/upload_views.py +94 -93
- endoreg_db/views/patient/patient.py +5 -4
- endoreg_db/views/report/__init__.py +5 -7
- endoreg_db/views/{pdf → report}/reimport.py +22 -22
- endoreg_db/views/{pdf/pdf_stream.py → report/report_stream.py} +46 -39
- endoreg_db/views/requirement/evaluate.py +188 -187
- endoreg_db/views/requirement/lookup.py +17 -3
- endoreg_db/views/requirement/lookup_store.py +22 -90
- endoreg_db/views/requirement/requirement_utils.py +89 -0
- endoreg_db/views/video/__init__.py +23 -24
- endoreg_db/views/video/correction.py +201 -172
- endoreg_db/views/video/reimport.py +1 -1
- endoreg_db/views/{media/video_segments.py → video/segments_crud.py} +77 -40
- endoreg_db/views/video/{video_meta.py → video_meta_stats.py} +2 -2
- endoreg_db/views/video/video_stream.py +7 -8
- {endoreg_db-0.8.6.1.dist-info → endoreg_db-0.8.8.9.dist-info}/METADATA +7 -3
- {endoreg_db-0.8.6.1.dist-info → endoreg_db-0.8.8.9.dist-info}/RECORD +391 -413
- {endoreg_db-0.8.6.1.dist-info → endoreg_db-0.8.8.9.dist-info}/WHEEL +1 -1
- endoreg_db/data/finding/anatomy_colon.yaml +0 -128
- endoreg_db/data/finding/colonoscopy.yaml +0 -40
- endoreg_db/data/finding/colonoscopy_bowel_prep.yaml +0 -56
- endoreg_db/data/finding/complication.yaml +0 -16
- endoreg_db/data/finding/data.yaml +0 -105
- endoreg_db/data/finding/examination_setting.yaml +0 -16
- endoreg_db/data/finding/medication_related.yaml +0 -18
- endoreg_db/data/finding/outcome.yaml +0 -12
- endoreg_db/data/finding_classification/colonoscopy_bowel_preparation.yaml +0 -95
- endoreg_db/data/finding_classification/colonoscopy_jnet.yaml +0 -22
- endoreg_db/data/finding_classification/colonoscopy_kudo.yaml +0 -25
- endoreg_db/data/finding_classification/colonoscopy_lesion_circularity.yaml +0 -20
- endoreg_db/data/finding_classification/colonoscopy_lesion_planarity.yaml +0 -24
- endoreg_db/data/finding_classification/colonoscopy_lesion_size.yaml +0 -68
- endoreg_db/data/finding_classification/colonoscopy_lesion_surface.yaml +0 -20
- endoreg_db/data/finding_classification/colonoscopy_location.yaml +0 -80
- endoreg_db/data/finding_classification/colonoscopy_lst.yaml +0 -21
- endoreg_db/data/finding_classification/colonoscopy_nice.yaml +0 -20
- endoreg_db/data/finding_classification/colonoscopy_paris.yaml +0 -26
- endoreg_db/data/finding_classification/colonoscopy_sano.yaml +0 -22
- endoreg_db/data/finding_classification/colonoscopy_summary.yaml +0 -53
- endoreg_db/data/finding_classification/complication_generic.yaml +0 -25
- endoreg_db/data/finding_classification/examination_setting_generic.yaml +0 -40
- endoreg_db/data/finding_classification/histology_colo.yaml +0 -51
- endoreg_db/data/finding_classification/intervention_required.yaml +0 -26
- endoreg_db/data/finding_classification/medication_related.yaml +0 -23
- endoreg_db/data/finding_classification/visualized.yaml +0 -33
- endoreg_db/data/finding_classification_choice/colon_lesion_circularity_default.yaml +0 -32
- endoreg_db/data/finding_classification_choice/colon_lesion_jnet.yaml +0 -15
- endoreg_db/data/finding_classification_choice/colon_lesion_kudo.yaml +0 -23
- endoreg_db/data/finding_classification_choice/colon_lesion_lst.yaml +0 -15
- endoreg_db/data/finding_classification_choice/colon_lesion_nice.yaml +0 -17
- endoreg_db/data/finding_classification_choice/colon_lesion_planarity_default.yaml +0 -49
- endoreg_db/data/finding_classification_choice/colon_lesion_sano.yaml +0 -14
- endoreg_db/data/finding_classification_choice/colon_lesion_surface_intact_default.yaml +0 -36
- endoreg_db/data/finding_classification_choice/colonoscopy_size.yaml +0 -82
- endoreg_db/data/finding_classification_choice/colonoscopy_summary_worst_finding.yaml +0 -15
- endoreg_db/data/finding_classification_choice/outcome.yaml +0 -19
- endoreg_db/data/finding_intervention/endoscopy.yaml +0 -43
- endoreg_db/data/finding_intervention/endoscopy_colonoscopy.yaml +0 -168
- endoreg_db/data/finding_intervention/endoscopy_egd.yaml +0 -128
- endoreg_db/data/finding_intervention/endoscopy_ercp.yaml +0 -32
- endoreg_db/data/finding_intervention/endoscopy_eus_lower.yaml +0 -9
- endoreg_db/data/finding_intervention/endoscopy_eus_upper.yaml +0 -36
- endoreg_db/data/finding_morphology_classification_type/colonoscopy.yaml +0 -79
- endoreg_db/data/requirement/age.yaml +0 -26
- endoreg_db/data/requirement/gender.yaml +0 -25
- endoreg_db/management/commands/init_default_ai_model.py +0 -112
- endoreg_db/management/commands/reset_celery_schedule.py +0 -9
- endoreg_db/management/commands/validate_video.py +0 -204
- endoreg_db/migrations/0002_add_video_correction_models.py +0 -52
- endoreg_db/migrations/0003_add_center_display_name.py +0 -30
- endoreg_db/models/administration/permissions/__init__.py +0 -44
- endoreg_db/models/rule/__init__.py +0 -13
- endoreg_db/models/rule/rule.py +0 -27
- endoreg_db/models/rule/rule_applicator.py +0 -224
- endoreg_db/models/rule/rule_attribute_dtype.py +0 -17
- endoreg_db/models/rule/rule_type.py +0 -20
- endoreg_db/models/rule/ruleset.py +0 -17
- endoreg_db/renames.yml +0 -8
- endoreg_db/serializers/_old/raw_pdf_meta_validation.py +0 -223
- endoreg_db/serializers/_old/raw_video_meta_validation.py +0 -179
- endoreg_db/serializers/_old/video.py +0 -71
- endoreg_db/serializers/meta/pdf_file_meta_extraction.py +0 -115
- endoreg_db/serializers/meta/report_meta.py +0 -53
- endoreg_db/serializers/report/__init__.py +0 -9
- endoreg_db/serializers/report/mixins.py +0 -45
- endoreg_db/serializers/report/report.py +0 -105
- endoreg_db/serializers/report/report_list.py +0 -22
- endoreg_db/serializers/report/secure_file_url.py +0 -26
- endoreg_db/serializers/video/video_metadata.py +0 -105
- endoreg_db/services/requirements_object.py +0 -147
- endoreg_db/services/storage_aware_video_processor.py +0 -344
- endoreg_db/urls/files.py +0 -6
- endoreg_db/urls/label_video_segment_validate.py +0 -33
- endoreg_db/urls/label_video_segments.py +0 -46
- endoreg_db/urls/report.py +0 -48
- endoreg_db/urls/video.py +0 -61
- endoreg_db/utils/case_generator/case_generator.py +0 -159
- endoreg_db/utils/case_generator/utils.py +0 -30
- endoreg_db/utils/requirement_operator_logic/model_evaluators.py +0 -368
- endoreg_db/views/label/__init__.py +0 -5
- endoreg_db/views/label/label.py +0 -15
- endoreg_db/views/label_video_segment/__init__.py +0 -16
- endoreg_db/views/label_video_segment/create_lvs_from_annotation.py +0 -44
- endoreg_db/views/label_video_segment/get_lvs_by_name_and_video.py +0 -50
- endoreg_db/views/label_video_segment/label_video_segment.py +0 -77
- endoreg_db/views/label_video_segment/label_video_segment_by_label.py +0 -174
- endoreg_db/views/label_video_segment/label_video_segment_detail.py +0 -73
- endoreg_db/views/label_video_segment/update_lvs_from_annotation.py +0 -46
- endoreg_db/views/label_video_segment/validate.py +0 -226
- endoreg_db/views/media/segments.py +0 -71
- endoreg_db/views/meta/available_files_list.py +0 -146
- endoreg_db/views/meta/report_meta.py +0 -53
- endoreg_db/views/meta/sensitive_meta_detail.py +0 -148
- endoreg_db/views/misc/secure_file_serving_view.py +0 -80
- endoreg_db/views/misc/secure_file_url_view.py +0 -84
- endoreg_db/views/misc/secure_url_validate.py +0 -79
- endoreg_db/views/patient_examination/DEPRECATED_video_backup.py +0 -164
- endoreg_db/views/patient_finding_location/__init__.py +0 -5
- endoreg_db/views/patient_finding_location/pfl_create.py +0 -70
- endoreg_db/views/patient_finding_morphology/__init__.py +0 -5
- endoreg_db/views/patient_finding_morphology/pfm_create.py +0 -70
- endoreg_db/views/pdf/__init__.py +0 -8
- endoreg_db/views/report/report_list.py +0 -112
- endoreg_db/views/report/report_with_secure_url.py +0 -28
- endoreg_db/views/report/start_examination.py +0 -7
- endoreg_db/views/video/segmentation.py +0 -274
- endoreg_db/views/video/task_status.py +0 -49
- endoreg_db/views/video/timeline.py +0 -46
- endoreg_db/views/video/video_analyze.py +0 -52
- endoreg_db/views.py +0 -0
- /endoreg_db/data/requirement/{colonoscopy_baseline_austria.yaml → old/colonoscopy_baseline_austria.yaml} +0 -0
- /endoreg_db/data/requirement/{disease_cardiovascular.yaml → old/disease_cardiovascular.yaml} +0 -0
- /endoreg_db/data/requirement/{disease_classification_choice_cardiovascular.yaml → old/disease_classification_choice_cardiovascular.yaml} +0 -0
- /endoreg_db/data/requirement/{disease_hepatology.yaml → old/disease_hepatology.yaml} +0 -0
- /endoreg_db/data/requirement/{disease_misc.yaml → old/disease_misc.yaml} +0 -0
- /endoreg_db/data/requirement/{disease_renal.yaml → old/disease_renal.yaml} +0 -0
- /endoreg_db/data/requirement/{endoscopy_bleeding_risk.yaml → old/endoscopy_bleeding_risk.yaml} +0 -0
- /endoreg_db/data/requirement/{event_cardiology.yaml → old/event_cardiology.yaml} +0 -0
- /endoreg_db/data/requirement/{event_requirements.yaml → old/event_requirements.yaml} +0 -0
- /endoreg_db/data/requirement/{finding_colon_polyp.yaml → old/finding_colon_polyp.yaml} +0 -0
- /endoreg_db/{migrations/__init__.py → data/requirement/old/gender.yaml} +0 -0
- /endoreg_db/data/requirement/{lab_value.yaml → old/lab_value.yaml} +0 -0
- /endoreg_db/data/requirement/{medication.yaml → old/medication.yaml} +0 -0
- /endoreg_db/data/requirement_operator/{age.yaml → _old/age.yaml} +0 -0
- /endoreg_db/data/requirement_operator/{lab_operators.yaml → _old/lab_operators.yaml} +0 -0
- /endoreg_db/data/requirement_operator/{model_operators.yaml → _old/model_operators.yaml} +0 -0
- /endoreg_db/{models/media/video/refactor_plan.md → import_files/pseudonymization/__init__.py} +0 -0
- /endoreg_db/{models/media/video/video_file_frames.py → import_files/pseudonymization/pseudonymize.py} +0 -0
- /endoreg_db/models/{metadata/frame_ocr_result.py → report/__init__.py} +0 -0
- /endoreg_db/{urls/sensitive_meta.py → models/report/images.py} +0 -0
- /endoreg_db/utils/requirement_operator_logic/{lab_value_operators.py → _old/lab_value_operators.py} +0 -0
- {endoreg_db-0.8.6.1.dist-info → endoreg_db-0.8.8.9.dist-info}/licenses/LICENSE +0 -0
|
@@ -1,214 +1,384 @@
|
|
|
1
1
|
"""
|
|
2
|
-
|
|
2
|
+
Management command for creating ModelMeta entries for multilabel classification models.
|
|
3
|
+
|
|
4
|
+
Supports two workflows:
|
|
5
|
+
1. Registering a local `.safetensors` weights file.
|
|
6
|
+
2. Generating metadata from a YAML template, downloading weights from Hugging Face.
|
|
3
7
|
"""
|
|
4
8
|
|
|
9
|
+
import logging
|
|
10
|
+
import tempfile
|
|
5
11
|
from pathlib import Path
|
|
6
|
-
from
|
|
7
|
-
|
|
12
|
+
from typing import Any, Dict, Iterable, List
|
|
13
|
+
|
|
14
|
+
import yaml
|
|
15
|
+
from django.core.management import BaseCommand, CommandError
|
|
16
|
+
from huggingface_hub import hf_hub_download
|
|
8
17
|
|
|
18
|
+
from endoreg_db.data import AI_MODEL_META_DATA_DIR
|
|
19
|
+
from endoreg_db.models import AiModel, LabelSet, ModelMeta
|
|
9
20
|
|
|
10
|
-
|
|
11
|
-
# python manage.py create_multilabel_model_meta --model_path "./data/models/colo_segmentation_RegNetX800MF_6.ckpt"
|
|
12
|
-
# python manage.py create_multilabel_model_meta --model_path "./libs/endoreg-db/tests/assets/colo_segmentation_RegNetX800MF_6.ckpt"
|
|
21
|
+
logger = logging.getLogger(__name__)
|
|
13
22
|
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
23
|
+
DEFAULT_MODEL_PATH = (
|
|
24
|
+
Path(__file__).resolve().parents[3]
|
|
25
|
+
/ "tests"
|
|
26
|
+
/ "assets"
|
|
27
|
+
/ "colo_segmentation_RegNetX800MF_6.safetensors"
|
|
28
|
+
)
|
|
18
29
|
|
|
19
30
|
|
|
20
31
|
class Command(BaseCommand):
|
|
21
|
-
help =
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
def add_arguments(self, parser):
|
|
29
|
-
"""
|
|
30
|
-
Adds command-line arguments for configuring model meta creation.
|
|
31
|
-
|
|
32
|
-
This method defines all necessary arguments for specifying model details, label set selection, normalization parameters, image dimensions, axes configuration, batch size, worker count, and versioning options for the Django management command.
|
|
33
|
-
"""
|
|
32
|
+
help = (
|
|
33
|
+
"Create or update ModelMeta entries for multilabel classification models using "
|
|
34
|
+
"either a local safetensor file or a YAML template with Hugging Face download support."
|
|
35
|
+
)
|
|
36
|
+
|
|
37
|
+
def add_arguments(self, parser): # noqa: D401 - inherited docstring is sufficient
|
|
34
38
|
parser.add_argument(
|
|
35
39
|
"--model_name",
|
|
36
40
|
type=str,
|
|
37
41
|
default="image_multilabel_classification_colonoscopy_default",
|
|
38
|
-
help="Name of the
|
|
42
|
+
help="Name of the AiModel to attach metadata to.",
|
|
39
43
|
)
|
|
40
|
-
|
|
41
|
-
# model_path
|
|
42
44
|
parser.add_argument(
|
|
43
45
|
"--model_path",
|
|
44
46
|
type=str,
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
47
|
+
default=str(DEFAULT_MODEL_PATH),
|
|
48
|
+
help=(
|
|
49
|
+
"Path to a local .safetensors weights file. If provided (or left as default) "
|
|
50
|
+
"the command registers the local weights."
|
|
51
|
+
),
|
|
52
|
+
)
|
|
53
|
+
parser.add_argument(
|
|
54
|
+
"--template_path",
|
|
55
|
+
type=str,
|
|
56
|
+
default=None,
|
|
57
|
+
help="Absolute or relative path to a model meta YAML template.",
|
|
58
|
+
)
|
|
59
|
+
parser.add_argument(
|
|
60
|
+
"--template_name",
|
|
61
|
+
type=str,
|
|
62
|
+
default=None,
|
|
63
|
+
help=(
|
|
64
|
+
"Name of a built-in template file in endoreg_db/data/ai_model_meta (without extension)."
|
|
65
|
+
),
|
|
66
|
+
)
|
|
67
|
+
parser.add_argument(
|
|
68
|
+
"--template_entry_name",
|
|
69
|
+
type=str,
|
|
70
|
+
default=None,
|
|
71
|
+
help="Optional entry selector when the template file defines multiple models.",
|
|
48
72
|
)
|
|
49
|
-
|
|
50
|
-
# model_meta_version: int = 1
|
|
51
73
|
parser.add_argument(
|
|
52
74
|
"--model_meta_version",
|
|
53
|
-
type=
|
|
54
|
-
default=
|
|
55
|
-
help=
|
|
75
|
+
type=str,
|
|
76
|
+
default=None,
|
|
77
|
+
help=(
|
|
78
|
+
"Version to assign to the metadata. When omitted the command uses the template value "
|
|
79
|
+
"or defaults to '1' for local registrations."
|
|
80
|
+
),
|
|
56
81
|
)
|
|
57
|
-
|
|
58
82
|
parser.add_argument(
|
|
59
83
|
"--image_classification_labelset_name",
|
|
60
84
|
type=str,
|
|
61
85
|
default="multilabel_classification_colonoscopy_default",
|
|
62
|
-
help="Name of the LabelSet
|
|
86
|
+
help="Name of the LabelSet used by the model.",
|
|
63
87
|
)
|
|
64
|
-
|
|
65
88
|
parser.add_argument(
|
|
66
89
|
"--image_classification_labelset_version",
|
|
67
90
|
type=int,
|
|
68
91
|
default=-1,
|
|
69
|
-
help="
|
|
92
|
+
help="Specific LabelSet version. Use -1 to select the latest available version.",
|
|
70
93
|
)
|
|
71
|
-
|
|
72
|
-
# activation: str = "sigmoid"
|
|
73
94
|
parser.add_argument(
|
|
74
95
|
"--activation_function_name",
|
|
75
96
|
type=str,
|
|
76
97
|
default="sigmoid",
|
|
77
|
-
help="Activation function
|
|
98
|
+
help="Activation function applied to model outputs.",
|
|
78
99
|
)
|
|
79
|
-
|
|
80
|
-
# mean: str = "0.45211223,0.27139644,0.19264949"
|
|
81
100
|
parser.add_argument(
|
|
82
101
|
"--mean",
|
|
83
102
|
type=str,
|
|
84
103
|
default="0.45211223,0.27139644,0.19264949",
|
|
85
|
-
help="
|
|
104
|
+
help="Comma-separated mean values for input normalization.",
|
|
86
105
|
)
|
|
87
|
-
|
|
88
106
|
parser.add_argument(
|
|
89
107
|
"--std",
|
|
90
108
|
type=str,
|
|
91
109
|
default="0.31418097,0.21088019,0.16059452",
|
|
92
|
-
help="
|
|
110
|
+
help="Comma-separated std values for input normalization.",
|
|
93
111
|
)
|
|
94
|
-
|
|
95
|
-
# size_x: int = 716
|
|
96
112
|
parser.add_argument(
|
|
97
113
|
"--size_x",
|
|
98
114
|
type=int,
|
|
99
115
|
default=716,
|
|
100
|
-
help="
|
|
116
|
+
help="Input width expected by the model.",
|
|
101
117
|
)
|
|
102
|
-
|
|
103
|
-
# size_y: int = 716
|
|
104
118
|
parser.add_argument(
|
|
105
119
|
"--size_y",
|
|
106
120
|
type=int,
|
|
107
121
|
default=716,
|
|
108
|
-
help="
|
|
122
|
+
help="Input height expected by the model.",
|
|
109
123
|
)
|
|
110
|
-
|
|
111
124
|
parser.add_argument(
|
|
112
125
|
"--axes",
|
|
113
126
|
type=str,
|
|
114
127
|
default="2,0,1",
|
|
115
|
-
help="
|
|
128
|
+
help="Comma-separated axis order expected by the model (e.g. '2,0,1' for CHW).",
|
|
116
129
|
)
|
|
117
|
-
|
|
118
|
-
# batchsize: int = 16
|
|
119
130
|
parser.add_argument(
|
|
120
131
|
"--batchsize",
|
|
121
132
|
type=int,
|
|
122
133
|
default=16,
|
|
123
|
-
help="
|
|
134
|
+
help="Default batch size for inference.",
|
|
124
135
|
)
|
|
125
|
-
|
|
126
|
-
# num_workers: int = 0
|
|
127
136
|
parser.add_argument(
|
|
128
137
|
"--num_workers",
|
|
129
138
|
type=int,
|
|
130
139
|
default=0,
|
|
131
|
-
help="
|
|
140
|
+
help="Default number of data loading workers.",
|
|
141
|
+
)
|
|
142
|
+
parser.add_argument(
|
|
143
|
+
"--description",
|
|
144
|
+
type=str,
|
|
145
|
+
default="",
|
|
146
|
+
help="Optional description to store on the ModelMeta record.",
|
|
132
147
|
)
|
|
133
|
-
|
|
134
|
-
# bump_version: bool = False
|
|
135
148
|
parser.add_argument(
|
|
136
149
|
"--bump_version",
|
|
137
150
|
action="store_true",
|
|
138
|
-
help="
|
|
151
|
+
help="If the requested version exists, bump to the next available version instead of failing.",
|
|
152
|
+
)
|
|
153
|
+
parser.add_argument(
|
|
154
|
+
"--huggingface_token",
|
|
155
|
+
type=str,
|
|
156
|
+
default=None,
|
|
157
|
+
help="Optional Hugging Face token for private repositories.",
|
|
158
|
+
)
|
|
159
|
+
|
|
160
|
+
def handle(self, *args, **options): # noqa: D401 - inherited docstring is sufficient
|
|
161
|
+
use_template = options.get("template_path") or options.get("template_name")
|
|
162
|
+
|
|
163
|
+
try:
|
|
164
|
+
if use_template:
|
|
165
|
+
model_meta = self._create_from_template(options)
|
|
166
|
+
else:
|
|
167
|
+
model_meta = self._create_from_local_file(options)
|
|
168
|
+
except CommandError:
|
|
169
|
+
raise
|
|
170
|
+
except Exception as exc: # pragma: no cover - defensive logging
|
|
171
|
+
logger.exception("Failed to create ModelMeta", exc_info=exc)
|
|
172
|
+
raise CommandError(str(exc)) from exc
|
|
173
|
+
|
|
174
|
+
self.stdout.write(
|
|
175
|
+
self.style.SUCCESS(f"ModelMeta ready: {model_meta.name} (v{model_meta.version}) for {model_meta.model.name}")
|
|
139
176
|
)
|
|
140
177
|
|
|
141
|
-
def
|
|
142
|
-
""
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
Retrieves and validates the specified LabelSet and AiModel, ensures the model file exists, and invokes ModelMeta.create_from_file with the provided configuration. Raises a ValueError if the requested LabelSet does not exist.
|
|
146
|
-
"""
|
|
178
|
+
def _create_from_local_file(self, options: Dict[str, Any]) -> ModelMeta:
|
|
179
|
+
weights_path = Path(options["model_path"]).expanduser().resolve()
|
|
180
|
+
self._validate_safetensors_path(weights_path)
|
|
181
|
+
|
|
147
182
|
model_name = options["model_name"]
|
|
183
|
+
self._ensure_ai_model_exists(model_name)
|
|
184
|
+
|
|
185
|
+
labelset = self._resolve_labelset(
|
|
186
|
+
options["image_classification_labelset_name"],
|
|
187
|
+
options.get("image_classification_labelset_version"),
|
|
188
|
+
)
|
|
148
189
|
|
|
149
|
-
|
|
150
|
-
activation_function_name = options["activation_function_name"]
|
|
190
|
+
requested_version = options.get("model_meta_version") or "1"
|
|
151
191
|
|
|
152
|
-
|
|
153
|
-
|
|
192
|
+
model_meta = ModelMeta.create_from_file(
|
|
193
|
+
meta_name=model_name,
|
|
194
|
+
model_name=model_name,
|
|
195
|
+
labelset_name=labelset.name,
|
|
196
|
+
labelset_version=labelset.version,
|
|
197
|
+
weights_file=weights_path.as_posix(),
|
|
198
|
+
requested_version=str(requested_version),
|
|
199
|
+
bump_if_exists=options.get("bump_version", False),
|
|
200
|
+
**self._collect_local_kwargs(options),
|
|
201
|
+
)
|
|
154
202
|
|
|
155
|
-
|
|
156
|
-
size_y = options["size_y"]
|
|
203
|
+
return model_meta
|
|
157
204
|
|
|
158
|
-
|
|
205
|
+
def _create_from_template(self, options: Dict[str, Any]) -> ModelMeta:
|
|
206
|
+
template_path = self._resolve_template_path(options)
|
|
207
|
+
entries = self._load_template_entries(template_path)
|
|
208
|
+
entry = self._select_template_entry(entries, options)
|
|
159
209
|
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
210
|
+
fields = entry.get("fields", {})
|
|
211
|
+
if not fields:
|
|
212
|
+
raise CommandError("Template entry is missing a 'fields' section.")
|
|
163
213
|
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
]
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
)
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
+
meta_name = fields.get("name") or options["model_name"]
|
|
215
|
+
model_name = fields.get("model") or options["model_name"]
|
|
216
|
+
labelset_name = fields.get("labelset") or options["image_classification_labelset_name"]
|
|
217
|
+
labelset_version = fields.get("labelset_version", options.get("image_classification_labelset_version"))
|
|
218
|
+
|
|
219
|
+
self._ensure_ai_model_exists(model_name)
|
|
220
|
+
labelset = self._resolve_labelset(labelset_name, labelset_version)
|
|
221
|
+
|
|
222
|
+
requested_version = options.get("model_meta_version") or fields.get("version")
|
|
223
|
+
if not requested_version:
|
|
224
|
+
raise CommandError("Provide --model_meta_version or include a 'version' in the template entry.")
|
|
225
|
+
|
|
226
|
+
hf_config = entry.get("setup_config", {}).get("huggingface_fallback", {})
|
|
227
|
+
repo_id = hf_config.get("repo_id")
|
|
228
|
+
filename = hf_config.get("filename")
|
|
229
|
+
|
|
230
|
+
if not repo_id or not filename:
|
|
231
|
+
raise CommandError(
|
|
232
|
+
"Template entry must define setup_config.huggingface_fallback.repo_id and filename for weight download."
|
|
233
|
+
)
|
|
234
|
+
|
|
235
|
+
if not filename.endswith(".safetensors"):
|
|
236
|
+
raise CommandError("Only .safetensors files are supported when downloading from Hugging Face.")
|
|
237
|
+
|
|
238
|
+
token = options.get("huggingface_token")
|
|
239
|
+
|
|
240
|
+
with tempfile.TemporaryDirectory(prefix="hf-multilabel-") as download_dir:
|
|
241
|
+
download_kwargs = {
|
|
242
|
+
"repo_id": repo_id,
|
|
243
|
+
"filename": filename,
|
|
244
|
+
"local_dir": download_dir,
|
|
245
|
+
"local_dir_use_symlinks": False,
|
|
246
|
+
}
|
|
247
|
+
if token:
|
|
248
|
+
download_kwargs["token"] = token
|
|
249
|
+
|
|
250
|
+
weights_path = Path(hf_hub_download(**download_kwargs)).resolve()
|
|
251
|
+
|
|
252
|
+
self._validate_safetensors_path(weights_path)
|
|
253
|
+
|
|
254
|
+
model_meta = ModelMeta.create_from_file(
|
|
255
|
+
meta_name=meta_name,
|
|
256
|
+
model_name=model_name,
|
|
257
|
+
labelset_name=labelset.name,
|
|
258
|
+
labelset_version=labelset.version,
|
|
259
|
+
weights_file=weights_path.as_posix(),
|
|
260
|
+
requested_version=str(requested_version),
|
|
261
|
+
bump_if_exists=options.get("bump_version", False),
|
|
262
|
+
**self._collect_template_kwargs(fields, options),
|
|
263
|
+
)
|
|
264
|
+
|
|
265
|
+
return model_meta
|
|
266
|
+
|
|
267
|
+
def _resolve_template_path(self, options: Dict[str, Any]) -> Path:
|
|
268
|
+
template_path = options.get("template_path")
|
|
269
|
+
template_name = options.get("template_name")
|
|
270
|
+
|
|
271
|
+
if template_path:
|
|
272
|
+
resolved = Path(template_path).expanduser().resolve()
|
|
273
|
+
elif template_name:
|
|
274
|
+
resolved = (AI_MODEL_META_DATA_DIR / f"{template_name}.yaml").resolve()
|
|
275
|
+
else: # pragma: no cover - guarded by caller
|
|
276
|
+
raise CommandError("Template mode requires --template_path or --template_name.")
|
|
277
|
+
|
|
278
|
+
if not resolved.exists():
|
|
279
|
+
raise CommandError(f"Template file not found: {resolved}")
|
|
280
|
+
|
|
281
|
+
return resolved
|
|
282
|
+
|
|
283
|
+
@staticmethod
|
|
284
|
+
def _load_template_entries(template_path: Path) -> List[Dict[str, Any]]:
|
|
285
|
+
with template_path.open("r", encoding="utf-8") as handle:
|
|
286
|
+
data = yaml.safe_load(handle) or []
|
|
287
|
+
|
|
288
|
+
if isinstance(data, dict):
|
|
289
|
+
return [data]
|
|
290
|
+
if isinstance(data, list):
|
|
291
|
+
return [entry for entry in data if isinstance(entry, dict)]
|
|
292
|
+
|
|
293
|
+
raise CommandError(f"Template {template_path} must define a mapping or list of mappings.")
|
|
294
|
+
|
|
295
|
+
def _select_template_entry(self, entries: Iterable[Dict[str, Any]], options: Dict[str, Any]) -> Dict[str, Any]:
|
|
296
|
+
target = options.get("template_entry_name") or options.get("model_name")
|
|
297
|
+
|
|
298
|
+
for entry in entries:
|
|
299
|
+
fields = entry.get("fields", {})
|
|
300
|
+
if not fields:
|
|
301
|
+
continue
|
|
302
|
+
if target and (fields.get("name") == target or fields.get("model") == target):
|
|
303
|
+
return entry
|
|
304
|
+
|
|
305
|
+
entries = list(entries)
|
|
306
|
+
if len(entries) == 1:
|
|
307
|
+
return entries[0]
|
|
308
|
+
|
|
309
|
+
raise CommandError(
|
|
310
|
+
"Unable to determine which template entry to use. Specify --template_entry_name to disambiguate."
|
|
311
|
+
)
|
|
312
|
+
|
|
313
|
+
def _collect_local_kwargs(self, options: Dict[str, Any]) -> Dict[str, Any]:
|
|
314
|
+
return self._filter_none(
|
|
315
|
+
{
|
|
316
|
+
"activation": options.get("activation_function_name"),
|
|
317
|
+
"mean": options.get("mean"),
|
|
318
|
+
"std": options.get("std"),
|
|
319
|
+
"size_x": options.get("size_x"),
|
|
320
|
+
"size_y": options.get("size_y"),
|
|
321
|
+
"axes": options.get("axes"),
|
|
322
|
+
"batchsize": options.get("batchsize"),
|
|
323
|
+
"num_workers": options.get("num_workers"),
|
|
324
|
+
"description": options.get("description"),
|
|
325
|
+
}
|
|
326
|
+
)
|
|
327
|
+
|
|
328
|
+
def _collect_template_kwargs(self, fields: Dict[str, Any], options: Dict[str, Any]) -> Dict[str, Any]:
|
|
329
|
+
def numeric(value):
|
|
330
|
+
return int(value) if value is not None else value
|
|
331
|
+
|
|
332
|
+
return self._filter_none(
|
|
333
|
+
{
|
|
334
|
+
"activation": fields.get("activation") or options.get("activation_function_name"),
|
|
335
|
+
"mean": self._normalise_sequence(fields.get("mean")) or options.get("mean"),
|
|
336
|
+
"std": self._normalise_sequence(fields.get("std")) or options.get("std"),
|
|
337
|
+
"size_x": numeric(fields.get("size_x")) if fields.get("size_x") is not None else options.get("size_x"),
|
|
338
|
+
"size_y": numeric(fields.get("size_y")) if fields.get("size_y") is not None else options.get("size_y"),
|
|
339
|
+
"axes": fields.get("axes") or options.get("axes"),
|
|
340
|
+
"batchsize": numeric(fields.get("batchsize")) if fields.get("batchsize") is not None else options.get("batchsize"),
|
|
341
|
+
"num_workers": numeric(fields.get("num_workers")) if fields.get("num_workers") is not None else options.get("num_workers"),
|
|
342
|
+
"description": fields.get("description") or options.get("description"),
|
|
343
|
+
}
|
|
214
344
|
)
|
|
345
|
+
|
|
346
|
+
@staticmethod
|
|
347
|
+
def _normalise_sequence(value: Any) -> str | None:
|
|
348
|
+
if value is None:
|
|
349
|
+
return None
|
|
350
|
+
if isinstance(value, str):
|
|
351
|
+
return value
|
|
352
|
+
if isinstance(value, (list, tuple)):
|
|
353
|
+
return ",".join(str(item) for item in value)
|
|
354
|
+
return str(value)
|
|
355
|
+
|
|
356
|
+
@staticmethod
|
|
357
|
+
def _filter_none(payload: Dict[str, Any]) -> Dict[str, Any]:
|
|
358
|
+
return {key: value for key, value in payload.items() if value not in (None, "")}
|
|
359
|
+
|
|
360
|
+
@staticmethod
|
|
361
|
+
def _validate_safetensors_path(path: Path) -> None:
|
|
362
|
+
if path.suffix != ".safetensors":
|
|
363
|
+
raise CommandError(f"Expected a .safetensors file, got: {path}")
|
|
364
|
+
if not path.exists():
|
|
365
|
+
raise CommandError(f"Weights file not found: {path}")
|
|
366
|
+
|
|
367
|
+
@staticmethod
|
|
368
|
+
def _ensure_ai_model_exists(model_name: str) -> None:
|
|
369
|
+
if not AiModel.objects.filter(name=model_name).exists():
|
|
370
|
+
raise CommandError(f"AiModel not found: {model_name}. Load ai model data before running this command.")
|
|
371
|
+
|
|
372
|
+
@staticmethod
|
|
373
|
+
def _resolve_labelset(name: str, version: Any) -> LabelSet:
|
|
374
|
+
queryset = LabelSet.objects.filter(name=name)
|
|
375
|
+
|
|
376
|
+
if version in (None, -1):
|
|
377
|
+
labelset = queryset.order_by("-version").first()
|
|
378
|
+
else:
|
|
379
|
+
labelset = queryset.filter(version=version).first()
|
|
380
|
+
|
|
381
|
+
if not labelset:
|
|
382
|
+
raise CommandError(f"LabelSet not found for name='{name}' and version='{version}'.")
|
|
383
|
+
|
|
384
|
+
return labelset
|