meta-edc 0.3.39__py3-none-any.whl → 1.4.0__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.
- meta_ae/action_items.py +38 -27
- meta_ae/admin/__init__.py +11 -0
- meta_ae/admin/ae_initial_admin.py +5 -2
- meta_ae/admin/ae_susar_admin.py +1 -1
- meta_ae/admin/death_report_admin.py +1 -1
- meta_ae/admin/modeladmin_mixins.py +14 -15
- meta_ae/baker_recipes.py +4 -6
- meta_ae/choices.py +1 -1
- meta_ae/forms/__init__.py +13 -0
- meta_ae/forms/death_report_form.py +1 -1
- meta_ae/forms/modelform_mixins.py +2 -2
- meta_ae/list_data.py +1 -1
- meta_ae/migrations/0001_initial.py +27 -27
- meta_ae/migrations/0006_aelocalreview_aesponsorreview.py +5 -5
- meta_ae/migrations/0017_auto_20221130_2257.py +1 -1
- meta_ae/migrations/0022_historicalhospitalization_hospitalization.py +5 -13
- meta_ae/migrations/0023_alter_aefollowup_action_identifier_and_more.py +2017 -0
- meta_ae/model_mixins/__init__.py +2 -0
- meta_ae/model_mixins/ae_review_model_mixin.py +6 -6
- meta_ae/model_mixins/death_report_model_mixin.py +3 -3
- meta_ae/models/__init__.py +13 -0
- meta_ae/models/hospitalization.py +3 -3
- meta_ae/pdf_reports/__init__.py +2 -0
- meta_ae/templatetags/meta_ae_extras.py +4 -6
- meta_analytics/.DS_Store +0 -0
- meta_analytics/README.rst +1 -2
- meta_analytics/dataframes/__init__.py +27 -0
- meta_analytics/dataframes/constants.py +5 -2
- meta_analytics/dataframes/get_eos_df.py +16 -4
- meta_analytics/dataframes/get_glucose_df.py +166 -0
- meta_analytics/dataframes/get_glucose_fbg_df.py +26 -0
- meta_analytics/dataframes/get_glucose_fbg_ogtt_df.py +21 -0
- meta_analytics/dataframes/get_last_imp_visits_df.py +12 -10
- meta_analytics/dataframes/glucose_endpoints/__init__.py +2 -0
- meta_analytics/dataframes/glucose_endpoints/endpoint_by_date.py +125 -124
- meta_analytics/dataframes/glucose_endpoints/glucose_endpoints_by_date.py +111 -235
- meta_analytics/dataframes/screening/__init__.py +2 -0
- meta_analytics/dataframes/screening/get_glucose_tested_only_df.py +1 -2
- meta_analytics/dataframes/screening/get_screening_df.py +9 -15
- meta_analytics/dataframes/utils.py +21 -12
- meta_analytics/get_tables.py +1 -2
- meta_analytics/tables/__init__.py +2 -0
- meta_analytics/tables/enrolled/glucose.py +2 -1
- meta_analytics/utils.py +81 -0
- meta_auth/auths.py +1 -1
- meta_consent/action_items.py +22 -3
- meta_consent/admin/__init__.py +7 -0
- meta_consent/admin/actions/__init__.py +2 -0
- meta_consent/admin/actions/create_missing_prescriptions.py +2 -2
- meta_consent/admin/list_filters.py +22 -0
- meta_consent/admin/modeladmin_mixins.py +4 -5
- meta_consent/admin/subject_consent_v1_ext_admin.py +93 -0
- meta_consent/baker_recipes.py +7 -7
- meta_consent/consents.py +15 -2
- meta_consent/constants.py +1 -0
- meta_consent/form_validators/__init__.py +2 -0
- meta_consent/forms/__init__.py +8 -0
- meta_consent/forms/subject_consent_v1_ext_form.py +47 -0
- meta_consent/forms/subject_reconsent_form.py +4 -4
- meta_consent/management/commands/create_missing_prescriptions.py +5 -3
- meta_consent/migrations/0001_initial.py +9 -9
- meta_consent/migrations/0024_historicalsubjectconsentv1.py +3 -8
- meta_consent/migrations/0026_historicalsubjectconsentv1ext_subjectconsentv1ext.py +535 -0
- meta_consent/migrations/0027_auto_20250111_0344.py +30 -0
- meta_consent/migrations/0028_historicalsubjectconsentv1ext_assessment_score_and_more.py +162 -0
- meta_consent/migrations/0029_alter_historicalsubjectconsentv1ext_agrees_to_extension_and_more.py +33 -0
- meta_consent/migrations/0030_auto_20250120_2114.py +40 -0
- meta_consent/migrations/0031_alter_historicalsubjectconsent_guardian_name_and_more.py +124 -0
- meta_consent/migrations/0032_alter_historicalsubjectconsent_device_created_and_more.py +678 -0
- meta_consent/migrations/0033_historicalsubjectconsentspfq_subjectconsentspfq.py +615 -0
- meta_consent/migrations/0034_remove_subjectconsentspfq_site_and_more.py +23 -0
- meta_consent/migrations/0035_alter_historicalsubjectconsent_consent_definition_name_and_more.py +43 -0
- meta_consent/models/__init__.py +10 -0
- meta_consent/models/model_mixins.py +1 -2
- meta_consent/models/signals.py +25 -11
- meta_consent/models/subject_consent.py +2 -2
- meta_consent/models/subject_consent_v1.py +0 -1
- meta_consent/models/subject_consent_v1_ext.py +34 -0
- meta_consent/models/subject_reconsent.py +4 -4
- meta_dashboard/navbars.py +2 -6
- meta_dashboard/patterns.py +1 -1
- meta_dashboard/templates/meta_dashboard/{bootstrap3/buttons → buttons}/eligibility_button.html +1 -1
- meta_dashboard/templates/meta_dashboard/{bootstrap3/buttons → buttons}/screening_button.html +1 -1
- meta_dashboard/templates/meta_dashboard/subject/dashboard/sidebar.html +24 -0
- meta_dashboard/templates/meta_dashboard/{bootstrap3/subject → subject}/dashboard/top_bar.html +1 -1
- meta_dashboard/templates/meta_dashboard/subject/dashboard.html +14 -0
- meta_dashboard/templatetags/meta_dashboard_extras.py +9 -14
- meta_dashboard/urls.py +5 -5
- meta_dashboard/view_utils/__init__.py +13 -0
- meta_dashboard/view_utils/subject_screening_button.py +13 -20
- meta_dashboard/views/__init__.py +8 -0
- meta_dashboard/views/ae/__init__.py +2 -0
- meta_dashboard/views/ae/ae_listboard_view.py +1 -1
- meta_dashboard/views/ae/death_report_listboard_view.py +1 -1
- meta_dashboard/views/screening/__init__.py +2 -0
- meta_dashboard/views/screening/listboard_view.py +2 -3
- meta_dashboard/views/subject/__init__.py +2 -0
- meta_dashboard/views/subject/dashboard/__init__.py +2 -0
- meta_dashboard/views/subject/dashboard/dashboard_view.py +19 -8
- meta_dashboard/views/subject/listboard/__init__.py +2 -0
- meta_dashboard/views/subject/listboard/listboard_view.py +2 -3
- meta_edc/__init__.py +5 -9
- meta_edc/admin.py +3 -4
- meta_edc/celery.py +2 -2
- meta_edc/celery_live.py +19 -0
- meta_edc/celery_uat.py +25 -0
- meta_edc/management/commands/update_forms_reference.py +16 -14
- meta_edc/meta_version.py +2 -2
- meta_edc/navbars.py +7 -5
- meta_edc/settings/debug.py +13 -4
- meta_edc/settings/defaults.py +55 -18
- meta_edc/settings/live.py +4 -1
- meta_edc/settings/logging.py +9 -4
- meta_edc/settings/minimal.py +4 -5
- meta_edc/settings/uat.py +3 -1
- meta_edc/templates/meta_edc/{bootstrap3/base.html → base.html} +1 -1
- meta_edc/templates/meta_edc/{bootstrap3/home.html → home.html} +2 -2
- meta_edc/urls.py +3 -1
- meta_edc/utils.py +3 -1
- meta_edc/views/__init__.py +2 -0
- meta_edc/views/home_view.py +1 -2
- meta_edc-1.4.0.dist-info/METADATA +174 -0
- {meta_edc-0.3.39.dist-info → meta_edc-1.4.0.dist-info}/RECORD +578 -586
- meta_edc-1.4.0.dist-info/WHEEL +4 -0
- meta_labs/list_data.py +2 -2
- meta_labs/reportables.py +80 -11
- meta_lists/list_data.py +13 -4
- meta_lists/migrations/0008_auto_20200528_1517.py +2 -2
- meta_lists/migrations/0019_auto_20250128_0143.py +48 -0
- meta_lists/migrations/0020_alter_abnormalfootappearanceobservations_extra_value_and_more.py +404 -0
- meta_pharmacy/admin/__init__.py +5 -0
- meta_pharmacy/admin/rx_admin.py +1 -0
- meta_pharmacy/admin/substitutions_admin.py +3 -3
- meta_pharmacy/constants.py +1 -1
- meta_pharmacy/forms/__init__.py +2 -0
- meta_pharmacy/forms/rx_form.py +0 -1
- meta_pharmacy/forms/substitutions_form.py +6 -4
- meta_pharmacy/labels/__init__.py +5 -2
- meta_pharmacy/labels/draw_label_for_subject_with_barcode.py +4 -1
- meta_pharmacy/labels/draw_label_with_test_data.py +2 -2
- meta_pharmacy/labels/label_data.py +1 -2
- meta_pharmacy/labels/print_sheets.py +4 -6
- meta_pharmacy/management/commands/update_initial_pharmacy_data.py +4 -3
- meta_pharmacy/migrations/0002_initial.py +7 -20
- meta_pharmacy/migrations/0003_auto_20240909_2335.py +3 -2
- meta_pharmacy/migrations/0006_lotnumber_label.py +5 -14
- meta_pharmacy/migrations/0008_remove_lotnumber_medication_and_more.py +5 -6
- meta_pharmacy/migrations/0010_alter_historicallabeldata_device_created_and_more.py +382 -0
- meta_pharmacy/models/__init__.py +7 -0
- meta_pharmacy/models/label_data.py +4 -5
- meta_pharmacy/models/rx_label.py +0 -1
- meta_pharmacy/models/substitutions.py +4 -4
- meta_pharmacy/prepare_meta_pharmacy.py +1 -1
- meta_pharmacy/utils/__init__.py +2 -0
- meta_pharmacy/utils/update_initial_pharmacy_data.py +1 -1
- meta_prn/action_items.py +44 -45
- meta_prn/admin/__init__.py +16 -0
- meta_prn/admin/dm_referral_admin.py +2 -1
- meta_prn/admin/end_of_study_admin.py +26 -15
- meta_prn/admin/loss_to_followup_admin.py +3 -2
- meta_prn/admin/off_study_medication_admin.py +5 -6
- meta_prn/admin/offschedule_admin.py +12 -11
- meta_prn/admin/offschedule_dm_referral_admin.py +11 -10
- meta_prn/admin/offschedule_postnatal_admin.py +15 -7
- meta_prn/admin/offschedule_pregnancy_admin.py +18 -9
- meta_prn/admin/onschedule_admin.py +7 -8
- meta_prn/admin/onschedule_dm_referral_admin.py +11 -12
- meta_prn/admin/pregnancy_notification_admin.py +5 -6
- meta_prn/admin/protocol_incident_admin.py +1 -1
- meta_prn/admin/subject_transfer_admin.py +1 -1
- meta_prn/baker_recipes.py +10 -10
- meta_prn/choices.py +11 -7
- meta_prn/constants.py +1 -0
- meta_prn/form_validators/__init__.py +5 -0
- meta_prn/form_validators/end_of_study.py +65 -20
- meta_prn/form_validators/protocol_incident.py +1 -1
- meta_prn/forms/__init__.py +13 -0
- meta_prn/forms/dm_referral_form.py +2 -8
- meta_prn/forms/end_of_study_form.py +1 -1
- meta_prn/forms/loss_to_followup_form.py +1 -1
- meta_prn/forms/off_study_medication_form.py +2 -2
- meta_prn/forms/offschedule_form.py +25 -0
- meta_prn/forms/pregnancy_notification_form.py +15 -17
- meta_prn/list_data.py +3 -3
- meta_prn/migrations/0001_initial.py +25 -25
- meta_prn/migrations/0017_auto_20220307_1929.py +5 -5
- meta_prn/migrations/0018_auto_20220309_2106.py +9 -9
- meta_prn/migrations/0021_auto_20220316_2147.py +13 -13
- meta_prn/migrations/0022_auto_20220318_0133.py +9 -9
- meta_prn/migrations/0032_historicalegfrnotification_egfrnotification.py +5 -13
- meta_prn/migrations/0034_auto_20220630_1110.py +1 -1
- meta_prn/migrations/0035_auto_20220630_1140.py +1 -1
- meta_prn/migrations/0038_alter_endofstudy_delivery_date_and_more.py +5 -13
- meta_prn/migrations/0041_endofstudy_transfer_date_and_more.py +5 -13
- meta_prn/migrations/0057_historicalonscheduledmreferral_and_more.py +13 -38
- meta_prn/migrations/0060_alter_onschedule_managers_and_more.py +55 -0
- meta_prn/migrations/0061_auto_20250115_2025.py +56 -0
- meta_prn/migrations/0062_alter_endofstudy_offstudy_reason_and_more.py +72 -0
- meta_prn/migrations/0063_historicaloffstudymedication_singleton_field_and_more.py +37 -0
- meta_prn/migrations/0064_auto_20250602_2143.py +18 -0
- meta_prn/migrations/0065_alter_historicaloffstudymedication_subject_identifier_and_more.py +23 -0
- meta_prn/migrations/0066_alter_historicallosstofollowup_subject_identifier_and_more.py +23 -0
- meta_prn/migrations/0067_alter_offschedule_managers_and_more.py +2557 -0
- meta_prn/migrations/0068_alter_dmreferral_referral_note_and_more.py +235 -0
- meta_prn/migrations/0069_alter_historicaloffstudymedication_singleton_field_and_more.py +37 -0
- meta_prn/models/__init__.py +20 -0
- meta_prn/models/dm_referral.py +2 -2
- meta_prn/models/end_of_study.py +28 -23
- meta_prn/models/loss_to_followup.py +8 -10
- meta_prn/models/off_study_medication.py +7 -4
- meta_prn/models/offschedule.py +4 -4
- meta_prn/models/pregnancy_notification.py +3 -5
- meta_prn/models/protocol_incident.py +5 -2
- meta_prn/models/signals.py +16 -14
- meta_prn/models/subject_transfer.py +7 -0
- meta_prn/templates/meta_prn/eos/additional_instructions.html +3 -0
- meta_prn/templates/meta_prn/offschedule/additional_instructions.html +2 -0
- meta_rando/migrations/0001_initial.py +5 -5
- meta_rando/migrations/0006_alter_historicalrandomizationlist_allocated_user_and_more.py +130 -0
- meta_rando/migrations/0007_spfqlist.py +197 -0
- meta_rando/migrations/0008_delete_spfqlist.py +16 -0
- meta_rando/models/__init__.py +1 -0
- meta_rando/{models.py → models/randomization_list.py} +3 -3
- meta_rando/randomizers.py +2 -2
- meta_reports/__init__.py +2 -0
- meta_reports/admin/__init__.py +16 -0
- meta_reports/admin/dbviews/__init__.py +13 -0
- meta_reports/admin/dbviews/glucose_summary_admin.py +7 -7
- meta_reports/admin/dbviews/imp_substitutions_admin.py +14 -13
- meta_reports/admin/dbviews/missing_screening_ogtt_admin/__init__.py +5 -0
- meta_reports/admin/dbviews/missing_screening_ogtt_admin/note_model_admin.py +27 -3
- meta_reports/admin/dbviews/missing_screening_ogtt_admin/unmanaged_model_admin.py +7 -7
- meta_reports/admin/dbviews/on_study_missing_lab_values_admin/__init__.py +2 -0
- meta_reports/admin/dbviews/on_study_missing_lab_values_admin/unmanaged_model_admin.py +0 -3
- meta_reports/admin/dbviews/on_study_missing_values_admin/__init__.py +2 -0
- meta_reports/admin/dbviews/on_study_missing_values_admin/unmanaged_model_admin.py +0 -1
- meta_reports/admin/dbviews/patient_history_missing_baseline_cd4_admin.py +9 -9
- meta_reports/admin/dbviews/unattended_three_in_row2_admin.py +6 -6
- meta_reports/admin/dbviews/unattended_three_in_row_admin.py +5 -5
- meta_reports/admin/dbviews/unattended_two_in_row_admin.py +5 -5
- meta_reports/admin/endpoints_admin.py +1 -1
- meta_reports/admin/endpoints_all_admin.py +0 -1
- meta_reports/admin/last_imp_refill_admin.py +33 -36
- meta_reports/admin/list_filters.py +3 -3
- meta_reports/admin/modeladmin_mixins.py +10 -17
- meta_reports/death_report.py +2 -2
- meta_reports/forms/__init__.py +2 -0
- meta_reports/forms/missing_ogtt_note_form.py +3 -4
- meta_reports/management/commands/generate_endpoints.py +5 -4
- meta_reports/migrations/0035_historicalmissingogttnote_missingogttnote.py +5 -14
- meta_reports/migrations/0054_auto_20250422_2003.py +81 -0
- meta_reports/migrations/0055_alter_glucosesummary_table.py +17 -0
- meta_reports/migrations/0056_auto_20250422_2214.py +54 -0
- meta_reports/migrations/0057_auto_20250422_2224.py +54 -0
- meta_reports/migrations/0058_auto_20250422_2232.py +54 -0
- meta_reports/migrations/0059_alter_endpoints_created_and_more.py +161 -0
- meta_reports/migrations/0060_auto_20250926_0242.py +366 -0
- meta_reports/migrations/0061_auto_20251004_0043.py +21 -0
- meta_reports/migrations/0062_auto_20251004_0106.py +21 -0
- meta_reports/models/__init__.py +17 -0
- meta_reports/models/dbviews/__init__.py +14 -0
- meta_reports/models/dbviews/glucose_summary/__init__.py +2 -0
- meta_reports/models/dbviews/glucose_summary/unmanaged_model.py +15 -3
- meta_reports/models/dbviews/glucose_summary/view_definition.py +8 -5
- meta_reports/models/dbviews/imp_substitutions/__init__.py +2 -0
- meta_reports/models/dbviews/imp_substitutions/unmanaged_model.py +1 -2
- meta_reports/models/dbviews/imp_substitutions/view_definition.py +1 -1
- meta_reports/models/dbviews/missing_screening_ogtt/__init__.py +2 -0
- meta_reports/models/dbviews/missing_screening_ogtt/note_model.py +2 -2
- meta_reports/models/dbviews/missing_screening_ogtt/unmanaged_model.py +4 -3
- meta_reports/models/dbviews/on_study_missing_lab_values/__init__.py +2 -0
- meta_reports/models/dbviews/on_study_missing_lab_values/qa_cases.py +13 -11
- meta_reports/models/dbviews/on_study_missing_lab_values/unmanged_model.py +0 -1
- meta_reports/models/dbviews/on_study_missing_values/__init__.py +2 -0
- meta_reports/models/dbviews/on_study_missing_values/qa_cases.py +19 -1
- meta_reports/models/dbviews/on_study_missing_values/unmanged_model.py +0 -1
- meta_reports/models/dbviews/patient_history_missing_baseline_cd4/__init__.py +2 -0
- meta_reports/models/dbviews/patient_history_missing_baseline_cd4/unmanaged_model.py +0 -1
- meta_reports/models/dbviews/unattended_three_in_row/__init__.py +2 -0
- meta_reports/models/dbviews/unattended_three_in_row/unmanaged_model.py +0 -1
- meta_reports/models/dbviews/unattended_three_in_row2/__init__.py +2 -0
- meta_reports/models/dbviews/unattended_three_in_row2/unmanaged_model.py +0 -1
- meta_reports/models/dbviews/unattended_two_in_row/__init__.py +2 -0
- meta_reports/models/dbviews/unattended_two_in_row/unmanaged_model.py +0 -1
- meta_reports/models/endpoints.py +8 -4
- meta_reports/models/last_imp_refill.py +2 -3
- meta_reports/pdf_report.py +2 -2
- meta_reports/tasks.py +4 -3
- meta_reports/templates/meta_reports/columns/subject_identifier_column.html +1 -1
- meta_reports/templates/meta_reports/endpoints_all_change_list_note.html +1 -1
- meta_reports/templates/meta_reports/endpoints_change_list_note.html +1 -1
- meta_reports/templates/meta_reports/last_imp_refill/changelist_note.html +13 -0
- meta_screening/admin/__init__.py +8 -0
- meta_screening/admin/fieldsets.py +13 -15
- meta_screening/admin/list_filters.py +7 -5
- meta_screening/admin/screening_part_one_admin.py +1 -3
- meta_screening/admin/screening_part_three_admin.py +2 -3
- meta_screening/admin/screening_part_two_admin.py +7 -10
- meta_screening/admin/subject_refusal_admin.py +5 -3
- meta_screening/admin/subject_screening_admin.py +21 -10
- meta_screening/baker_recipes.py +18 -10
- meta_screening/calculators.py +1 -1
- meta_screening/choices.py +1 -1
- meta_screening/constants.py +1 -1
- meta_screening/eligibility/__init__.py +9 -0
- meta_screening/eligibility/eligibility.py +21 -14
- meta_screening/eligibility/eligibility_part_one.py +1 -1
- meta_screening/eligibility/eligibility_part_three/__init__.py +2 -0
- meta_screening/eligibility/eligibility_part_three/base_eligibility_part_three.py +68 -49
- meta_screening/eligibility/eligibility_part_three/eligibility_part_three_phase_three.py +14 -15
- meta_screening/eligibility/eligibility_part_two.py +1 -1
- meta_screening/form_validators/__init__.py +8 -0
- meta_screening/form_validators/screening_part_one.py +1 -1
- meta_screening/form_validators/screening_part_three.py +6 -2
- meta_screening/form_validators/screening_part_two.py +1 -1
- meta_screening/form_validators/subject_refusal.py +1 -1
- meta_screening/forms/__init__.py +20 -0
- meta_screening/forms/field_lists.py +16 -18
- meta_screening/forms/screening_part_one_form.py +2 -2
- meta_screening/forms/screening_part_three_form.py +5 -3
- meta_screening/forms/screening_part_two_form.py +1 -5
- meta_screening/forms/subject_refusal_form.py +0 -4
- meta_screening/forms/subject_screening_form.py +0 -4
- meta_screening/migrations/0001_initial.py +15 -15
- meta_screening/migrations/0010_auto_20191106_0828.py +5 -5
- meta_screening/migrations/0012_auto_20191107_0427.py +1 -1
- meta_screening/migrations/0068_alter_historicalscreeningpartone_acute_condition_and_more.py +1579 -0
- meta_screening/model_mixins/__init__.py +8 -0
- meta_screening/model_mixins/calculated_model_mixin.py +2 -2
- meta_screening/model_mixins/creatinine_fields_model_mixin.py +1 -1
- meta_screening/model_mixins/eligibility_model_mixin.py +6 -4
- meta_screening/model_mixins/part_one_fields_model_mixin.py +16 -19
- meta_screening/model_mixins/part_three_fields_model_mixin.py +6 -7
- meta_screening/model_mixins/part_two_fields_model_mixin.py +19 -17
- meta_screening/models/__init__.py +9 -0
- meta_screening/models/icp_referral.py +5 -5
- meta_screening/models/proxy_models.py +1 -1
- meta_screening/models/signals.py +10 -11
- meta_screening/models/subject_refusal.py +1 -1
- meta_screening/models/subject_screening.py +2 -4
- meta_subject/action_items.py +18 -14
- meta_subject/admin/__init__.py +42 -0
- meta_subject/admin/birth_outcome_admin.py +7 -9
- meta_subject/admin/blood_results/__init__.py +9 -0
- meta_subject/admin/blood_results/blood_results_fbc_admin.py +1 -1
- meta_subject/admin/blood_results/blood_results_hba1c_admin.py +1 -1
- meta_subject/admin/blood_results/blood_results_ins_admin.py +1 -1
- meta_subject/admin/blood_results/blood_results_lft_admin.py +1 -1
- meta_subject/admin/blood_results/blood_results_lipids_admin.py +1 -1
- meta_subject/admin/blood_results/blood_results_rft_admin.py +3 -5
- meta_subject/admin/complications_glycemia_admin.py +1 -1
- meta_subject/admin/delivery_admin.py +8 -11
- meta_subject/admin/diabetes/__init__.py +2 -0
- meta_subject/admin/diabetes/dm_endpoint_admin.py +2 -2
- meta_subject/admin/diabetes/dm_followup_admin.py +3 -2
- meta_subject/admin/egfr_drop_notification_admin.py +1 -1
- meta_subject/admin/fields.py +5 -5
- meta_subject/admin/fieldsets.py +5 -5
- meta_subject/admin/followup_examination_admin.py +11 -9
- meta_subject/admin/followup_vitals_admin.py +31 -6
- meta_subject/admin/glucose_admin.py +4 -8
- meta_subject/admin/glucose_fbg_admin.py +18 -11
- meta_subject/admin/health_economics/__init__.py +2 -0
- meta_subject/admin/health_economics/health_economics_simple_admin.py +1 -1
- meta_subject/admin/health_economics/health_economics_update_admin.py +1 -1
- meta_subject/admin/hepatitis_test_admin.py +1 -1
- meta_subject/admin/hiv_exit_review_admin.py +55 -0
- meta_subject/admin/list_filters.py +5 -5
- meta_subject/admin/mnsi_admin.py +7 -5
- meta_subject/admin/next_appointment_admin.py +19 -0
- meta_subject/admin/other_arv_regimens_admin.py +3 -3
- meta_subject/admin/patient_history_admin.py +4 -4
- meta_subject/admin/physical_exam_admin.py +1 -1
- meta_subject/admin/pregnancy_update_admin.py +1 -1
- meta_subject/admin/study_medication_admin.py +8 -15
- meta_subject/admin/subject_requisition_admin.py +2 -2
- meta_subject/admin/subject_visit_admin.py +1 -1
- meta_subject/admin/subject_visit_missed_admin.py +1 -1
- meta_subject/admin/urine_dipstick_test_admin.py +1 -1
- meta_subject/admin/urine_pregnancy_admin.py +1 -1
- meta_subject/baker_recipes.py +15 -15
- meta_subject/choices.py +4 -3
- meta_subject/form_validators/__init__.py +15 -0
- meta_subject/form_validators/delivery_form_validator.py +3 -3
- meta_subject/form_validators/dm_endpoint_form_validator.py +3 -1
- meta_subject/form_validators/dm_followup_form_validator.py +8 -7
- meta_subject/form_validators/egfr_drop_notification_form_validator.py +1 -1
- meta_subject/form_validators/followup_examination_form_validator.py +1 -1
- meta_subject/form_validators/glucose_fbg_form_validator.py +76 -0
- meta_subject/form_validators/glucose_form_validator.py +10 -67
- meta_subject/form_validators/hiv_exit_review_form_validator.py +18 -0
- meta_subject/form_validators/mixins.py +95 -0
- meta_subject/forms/__init__.py +44 -0
- meta_subject/forms/blood_results/__init__.py +9 -0
- meta_subject/forms/blood_results/blood_results_hba1c_form.py +1 -1
- meta_subject/forms/blood_results/blood_results_rft_form.py +60 -4
- meta_subject/forms/delivery_form.py +2 -0
- meta_subject/forms/diabetes/__init__.py +2 -0
- meta_subject/forms/diabetes/dm_followup_form.py +2 -2
- meta_subject/forms/followup_vitals_form.py +27 -1
- meta_subject/forms/glucose_fbg_form.py +1 -46
- meta_subject/forms/health_economics/__init__.py +2 -0
- meta_subject/forms/hepatitis_test_form.py +1 -1
- meta_subject/forms/hiv_exit_review_form.py +13 -0
- meta_subject/forms/mixins.py +1 -1
- meta_subject/forms/next_appointment_form.py +36 -0
- meta_subject/forms/other_arv_regimens_form.py +1 -1
- meta_subject/forms/patient_history_form.py +1 -1
- meta_subject/forms/physical_exam_form.py +1 -1
- meta_subject/forms/pregnancy_update_form.py +1 -1
- meta_subject/forms/slider_widget.py +1 -1
- meta_subject/forms/study_medication_form.py +18 -12
- meta_subject/forms/subject_requisition_form.py +1 -1
- meta_subject/forms/subject_visit_missed_form.py +1 -1
- meta_subject/forms/urine_dipstick_test_form.py +1 -1
- meta_subject/forms/urine_pregnancy_form.py +1 -1
- meta_subject/management/commands/create_missing_refills.py +3 -3
- meta_subject/management/commands/create_missing_rx.py +2 -2
- meta_subject/management/commands/missed.py +20 -23
- meta_subject/metadata_rules/__init__.py +2 -0
- meta_subject/metadata_rules/metadata_rules.py +14 -0
- meta_subject/metadata_rules/predicates.py +44 -25
- meta_subject/migrations/0001_initial.py +61 -61
- meta_subject/migrations/0002_auto_20191021_0353.py +5 -5
- meta_subject/migrations/0012_auto_20200118_2334.py +5 -5
- meta_subject/migrations/0014_auto_20200120_1622.py +5 -5
- meta_subject/migrations/0018_coronakap_historicalcoronakap.py +5 -5
- meta_subject/migrations/0033_auto_20200516_2356.py +5 -5
- meta_subject/migrations/0038_auto_20200520_0020.py +5 -5
- meta_subject/migrations/0040_auto_20200527_2155.py +1 -1
- meta_subject/migrations/0045_auto_20200530_1801.py +1 -1
- meta_subject/migrations/0051_auto_20200617_2117.py +5 -5
- meta_subject/migrations/0063_auto_20210715_0337.py +5 -5
- meta_subject/migrations/0066_auto_20210721_0335.py +9 -9
- meta_subject/migrations/0067_auto_20210726_0340.py +5 -5
- meta_subject/migrations/0068_auto_20210728_1809.py +5 -5
- meta_subject/migrations/0072_auto_20210805_1545.py +7 -7
- meta_subject/migrations/0073_auto_20210809_0055.py +5 -5
- meta_subject/migrations/0077_auto_20210809_2323.py +3 -3
- meta_subject/migrations/0082_auto_20210823_1612.py +3 -3
- meta_subject/migrations/0083_auto_20210823_1620.py +3 -3
- meta_subject/migrations/0088_auto_20210924_0027.py +5 -5
- meta_subject/migrations/0090_auto_20210924_0424.py +5 -5
- meta_subject/migrations/0093_auto_20211117_0352.py +5 -5
- meta_subject/migrations/0095_auto_20220128_1719.py +5 -5
- meta_subject/migrations/0098_auto_20220309_2106.py +5 -5
- meta_subject/migrations/0101_auto_20220316_2147.py +13 -13
- meta_subject/migrations/0107_auto_20220415_0043.py +1 -1
- meta_subject/migrations/0115_historicalegfrnotification_egfrnotification.py +5 -13
- meta_subject/migrations/0128_auto_20220720_0055.py +1 -1
- meta_subject/migrations/0131_auto_20220722_0411.py +1 -1
- meta_subject/migrations/0135_auto_20220722_2212.py +2 -1
- meta_subject/migrations/0164_dmreferralfollowup_historicaldmreferralfollowup.py +5 -5
- meta_subject/migrations/0172_remove_historicalbloodresultsglu_action_item_and_more.py +1 -2
- meta_subject/migrations/0177_alter_bloodresultslft_alp_value_and_more.py +1 -2
- meta_subject/migrations/0178_historicalhealtheconomicsupdate_and_more.py +5 -14
- meta_subject/migrations/0186_healtheconomicsupdate_singleton_field_and_more.py +1 -2
- meta_subject/migrations/0187_dmdiagnosis_historicaldmdiagnosis_dmdxresult_and_more.py +5 -14
- meta_subject/migrations/0188_historicaldmdxresult_dmdxresult.py +5 -14
- meta_subject/migrations/0199_auto_20240516_0247.py +1 -1
- meta_subject/migrations/0209_remove_historicaldmdxresult_dm_diagnosis_and_more.py +1 -2
- meta_subject/migrations/0212_auto_20240827_2222.py +1 -1
- meta_subject/migrations/0216_historicalnextappointment_nextappointment.py +553 -0
- meta_subject/migrations/0217_alter_historicalnextappointment_appt_datetime_and_more.py +42 -0
- meta_subject/migrations/0218_alter_historicalnextappointment_appt_date_and_more.py +53 -0
- meta_subject/migrations/0219_remove_historicalnextappointment_allow_create_interim_and_more.py +92 -0
- meta_subject/migrations/0220_historicalbloodresultsgludummy_bloodresultsgludummy.py +825 -0
- meta_subject/migrations/0221_auto_20250402_1913.py +42 -0
- meta_subject/migrations/0222_alter_historicalstudymedication_stock_codes_and_more.py +46 -0
- meta_subject/migrations/0223_bloodresultsfbc_errors_bloodresultsgludummy_errors_and_more.py +83 -0
- meta_subject/migrations/0224_bloodresultsfbc_abnormal_summary_and_more.py +153 -0
- meta_subject/migrations/0225_followupvitals_waist_circumference_and_more.py +46 -0
- meta_subject/migrations/0226_followupvitals_waist_circumference_comment_and_more.py +97 -0
- meta_subject/migrations/0227_alter_followupvitals_waist_circumference_comment_and_more.py +97 -0
- meta_subject/migrations/0228_bloodresultshba1c_hba1c_datetime_and_more.py +2518 -0
- meta_subject/migrations/0229_alter_glucosefbg_consent_model_and_more.py +1918 -0
- meta_subject/migrations/0230_alter_historicaldelivery_action_identifier_and_more.py +1733 -0
- meta_subject/migrations/0231_alter_historicalmedicationadherence_consent_model_and_more.py +2054 -0
- meta_subject/migrations/0232_alter_patienthistory_concomitant_conditions_and_more.py +1170 -0
- meta_subject/migrations/0233_historicalspfq_spfq.py +1066 -0
- meta_subject/migrations/0234_remove_spfq_site_remove_spfq_subject_visit_and_more.py +27 -0
- meta_subject/migrations/0235_glucosefbg_endpoint_today_and_more.py +606 -0
- meta_subject/migrations/0236_alter_historicalhivexitreview_other_current_arv_regimen_and_more.py +58 -0
- meta_subject/migrations/0237_historicalhivexitreview_singleton_field_and_more.py +68 -0
- meta_subject/migrations/0238_historicalhivexitreview_available_and_more.py +88 -0
- meta_subject/model_mixins/__init__.py +10 -0
- meta_subject/model_mixins/arv_history_model_mixin.py +3 -44
- meta_subject/model_mixins/arv_review_model_mixin.py +53 -0
- meta_subject/model_mixins/search_slug_model_mixin.py +1 -2
- meta_subject/model_mixins/vitals_fields_model_mixin.py +33 -0
- meta_subject/models/__init__.py +52 -0
- meta_subject/models/birth_outcomes.py +3 -3
- meta_subject/models/blood_results/__init__.py +11 -0
- meta_subject/models/blood_results/blood_results_glu.py +29 -0
- meta_subject/models/blood_results/blood_results_hba1c.py +0 -1
- meta_subject/models/blood_results/blood_results_ins.py +0 -1
- meta_subject/models/blood_results/blood_results_lft.py +0 -1
- meta_subject/models/delivery.py +4 -6
- meta_subject/models/diabetes/__init__.py +2 -0
- meta_subject/models/diabetes/dm_endpoint.py +5 -5
- meta_subject/models/diabetes/dm_followup.py +7 -8
- meta_subject/models/diet_and_lifestyle.py +2 -2
- meta_subject/models/followup_examination.py +12 -14
- meta_subject/models/glucose.py +5 -5
- meta_subject/models/glucose_fbg.py +17 -4
- meta_subject/models/health_economics/__init__.py +2 -0
- meta_subject/models/health_economics/health_economics.py +8 -8
- meta_subject/models/health_economics/health_economics_simple.py +2 -2
- meta_subject/models/health_economics/health_economics_update.py +3 -2
- meta_subject/models/hepatitis_test.py +2 -2
- meta_subject/models/hiv_exit_review.py +44 -0
- meta_subject/models/next_appointment.py +15 -0
- meta_subject/models/other_arv_regimens_detail.py +1 -1
- meta_subject/models/patient_history.py +6 -7
- meta_subject/models/physical_exam.py +2 -2
- meta_subject/models/pregnancy_update.py +1 -1
- meta_subject/models/signals.py +15 -13
- meta_subject/models/subject_visit.py +2 -2
- meta_subject/models/todo.txt +1 -1
- meta_subject/models/urine_dipstick_test.py +2 -2
- meta_subject/models/urine_pregnancy.py +1 -1
- meta_subject/static/meta_subject/slider.css +1 -1
- meta_subject/templates/meta_subject/endpoint_review_instructions.html +1 -1
- meta_subject/templates/meta_subject/widgets/slider.html +0 -1
- meta_visit_schedule/visit_schedules/__init__.py +2 -0
- meta_visit_schedule/visit_schedules/phase_three/__init__.py +2 -0
- meta_visit_schedule/visit_schedules/phase_three/crfs.py +34 -1
- meta_visit_schedule/visit_schedules/phase_three/schedule.py +4 -4
- meta_visit_schedule/visit_schedules/phase_three/schedule_dm_referral.py +1 -2
- meta_visit_schedule/visit_schedules/phase_three/schedule_pregnancy.py +2 -3
- meta_ae/tests/holidays.csv +0 -15
- meta_ae/tests/tests/test_actions.py +0 -127
- meta_ae/tests/urls.py +0 -10
- meta_analytics/dataframes/enrolled/__init__.py +0 -1
- meta_analytics/dataframes/enrolled/get_glucose_df.py +0 -122
- meta_analytics/tests/__init__.py +0 -0
- meta_analytics/tests/test_endpoints_by_date.py +0 -94
- meta_consent/tests/__init__.py +0 -0
- meta_consent/tests/holidays.csv +0 -15
- meta_consent/tests/tests/__init__.py +0 -0
- meta_consent/tests/tests/test_form_validators.py +0 -110
- meta_consent/tests/tests/test_subject_consent.py +0 -10
- meta_consent/tests/urls.py +0 -17
- meta_dashboard/templates/meta_dashboard/bootstrap3/subject/dashboard.html +0 -11
- meta_dashboard/tests/__init__.py +0 -0
- meta_dashboard/tests/admin.py +0 -22
- meta_dashboard/tests/holidays.csv +0 -15
- meta_dashboard/tests/tests/__init__.py +0 -0
- meta_dashboard/tests/urls.py +0 -55
- meta_edc/tests/__init__.py +0 -0
- meta_edc/tests/tests/__init__.py +0 -0
- meta_edc/tests/tests/test_endpoints.py +0 -554
- meta_edc-0.3.39.dist-info/AUTHORS +0 -0
- meta_edc-0.3.39.dist-info/METADATA +0 -766
- meta_edc-0.3.39.dist-info/WHEEL +0 -5
- meta_edc-0.3.39.dist-info/top_level.txt +0 -20
- meta_labs/tests/__init__.py +0 -0
- meta_labs/tests/test_labs.py +0 -27
- meta_labs/tests/test_reportables.py +0 -16
- meta_labs/tests/urls.py +0 -4
- meta_lists/tests/__init__.py +0 -0
- meta_lists/tests/test_lists.py +0 -8
- meta_pharmacy/admin/actions.py +0 -38
- meta_prn/tests/__init__.py +0 -0
- meta_prn/tests/tests/__init__.py +0 -0
- meta_prn/tests/tests/test_actions.py +0 -97
- meta_prn/tests/tests/test_dm_referral.py +0 -206
- meta_prn/tests/tests/test_manager_order.py +0 -14
- meta_prn/tests/tests/test_pregnancy_notification.py +0 -93
- meta_prn/tests/urls.py +0 -10
- meta_rando/tests/__init__.py +0 -0
- meta_rando/tests/tests/__init__.py +0 -0
- meta_rando/tests/tests/test_randomizers.py +0 -57
- meta_reports/tests/__init__.py +0 -0
- meta_reports/tests/test_reports.py +0 -35
- meta_reports/tests/test_sql_gen.py +0 -5
- meta_reports/tests/urls.py +0 -4
- meta_screening/offline_models.py +0 -3
- meta_screening/tests/__init__.py +0 -0
- meta_screening/tests/holidays.csv +0 -15
- meta_screening/tests/meta_test_case_mixin.py +0 -216
- meta_screening/tests/options.py +0 -127
- meta_screening/tests/tests/__init__.py +0 -0
- meta_screening/tests/tests/test_forms.py +0 -397
- meta_screening/tests/tests/test_screening_part_one.py +0 -108
- meta_screening/tests/tests/test_screening_part_three.py +0 -436
- meta_screening/tests/tests/test_screening_part_two.py +0 -84
- meta_sites/tests/__init__.py +0 -0
- meta_sites/tests/test_sites.py +0 -12
- meta_sites/tests/urls.py +0 -4
- meta_stats/__init__.py +0 -0
- meta_stats/incidence.py +0 -16
- meta_stats/models.py +0 -0
- meta_stats/tests/__init__.py +0 -0
- meta_stats/tests/tests/__init__.py +0 -0
- meta_stats/tests/tests/test_incidence.py +0 -10
- meta_subject/tests/__init__.py +0 -0
- meta_subject/tests/holidays.csv +0 -15
- meta_subject/tests/tests/__init__.py +0 -0
- meta_subject/tests/tests/test_egfr.py +0 -234
- meta_subject/tests/tests/test_fixes.py +0 -64
- meta_subject/tests/tests/test_followup.py +0 -52
- meta_subject/tests/tests/test_manager_order.py +0 -11
- meta_subject/tests/tests/test_medication_adherence.py +0 -75
- meta_subject/tests/tests/test_metadata_rules.py +0 -135
- meta_subject/tests/tests/test_mnsi.py +0 -317
- meta_subject/tests/tests/test_patient_history_form.py +0 -74
- meta_subject/tests/tests/test_physical_exam.py +0 -84
- meta_subject/tests/tests/test_sf12.py +0 -173
- meta_subject/tests/tests/test_study_medication.py +0 -230
- meta_subject/tests/urls.py +0 -24
- meta_visit_schedule/tests/__init__.py +0 -0
- meta_visit_schedule/tests/tests/__init__.py +0 -0
- meta_visit_schedule/tests/tests/test_schedule.py +0 -181
- meta_visit_schedule/tests/urls.py +0 -4
- tests/__init__.py +0 -0
- tests/etc/randomization_list.csv +0 -241
- tests/etc/randomization_list_phase_three.csv +0 -241
- tests/etc/user-aes-local.key +0 -0
- tests/etc/user-aes-restricted.key +0 -1
- tests/etc/user-rsa-local-private.pem +0 -27
- tests/etc/user-rsa-local-public.pem +0 -9
- tests/etc/user-rsa-restricted-private.pem +0 -27
- tests/etc/user-rsa-restricted-public.pem +0 -9
- tests/etc/user-salt-local.key +0 -0
- tests/etc/user-salt-restricted.key +0 -0
- tests/holidays.csv +0 -15
- tests/test_settings.py +0 -186
- /meta_ae/templates/meta_ae/{bootstrap3/ae_initial_description.html → aeinitial_description.html} +0 -0
- /meta_dashboard/templates/meta_dashboard/{bootstrap3/buttons → buttons}/add_consent_button.html +0 -0
- /meta_dashboard/templates/meta_dashboard/{bootstrap3/buttons → buttons}/dashboard_button.html +0 -0
- /meta_dashboard/templates/meta_dashboard/{bootstrap3/buttons → buttons}/refusal_button.html +0 -0
- /meta_dashboard/templates/meta_dashboard/{bootstrap3/screening → screening}/listboard.html +0 -0
- /meta_dashboard/templates/meta_dashboard/{bootstrap3/subject → subject}/listboard.html +0 -0
- {meta_edc-0.3.39.dist-info → meta_edc-1.4.0.dist-info/licenses}/LICENSE +0 -0
- /meta_ae/tests/__init__.py → /meta_subject/management/__init__py.py +0 -0
- /meta_ae/tests/tests/__init__.py → /meta_subject/management/commands/__init__py.py +0 -0
|
@@ -1,44 +1,28 @@
|
|
|
1
1
|
import numpy as np
|
|
2
2
|
import pandas as pd
|
|
3
|
+
from clinicedc_constants import NULL_STRING, YES
|
|
3
4
|
from django.apps import apps as django_apps
|
|
4
|
-
from
|
|
5
|
-
from edc_pdutils.dataframes import (
|
|
6
|
-
get_crf,
|
|
7
|
-
get_eos,
|
|
8
|
-
get_subject_consent,
|
|
9
|
-
get_subject_visit,
|
|
10
|
-
)
|
|
11
|
-
from edc_utils import get_utcnow
|
|
5
|
+
from django.utils import timezone
|
|
12
6
|
|
|
13
7
|
from ..constants import (
|
|
14
8
|
CASE_EOS,
|
|
15
|
-
|
|
9
|
+
CASE_FBG_VERY_HIGH,
|
|
16
10
|
CASE_FBGS_WITH_FIRST_OGTT,
|
|
17
11
|
CASE_FBGS_WITH_SECOND_OGTT,
|
|
18
12
|
CASE_OGTT,
|
|
13
|
+
FBG_BEYOND_THRESHOLD,
|
|
19
14
|
endpoint_cases,
|
|
20
15
|
endpoint_columns,
|
|
21
16
|
)
|
|
17
|
+
from ..get_glucose_df import get_glucose_df
|
|
22
18
|
from ..utils import (
|
|
23
19
|
get_empty_endpoint_df,
|
|
24
20
|
get_test_string,
|
|
25
21
|
get_unique_subject_identifiers,
|
|
26
|
-
get_unique_visit_codes,
|
|
27
22
|
)
|
|
28
23
|
from .endpoint_by_date import EndpointByDate
|
|
29
24
|
|
|
30
25
|
|
|
31
|
-
def calculate_fasting_hrs(df: pd.DataFrame):
|
|
32
|
-
df.loc[(df["fasting"] == NO), "fasting_duration_delta"] = pd.NaT
|
|
33
|
-
if df.empty:
|
|
34
|
-
df["fasting_hrs"] = np.nan
|
|
35
|
-
else:
|
|
36
|
-
df["fasting_hrs"] = df["fasting_duration_delta"].apply(
|
|
37
|
-
lambda s: np.nan if pd.isna(s) else s.total_seconds() / 3600
|
|
38
|
-
)
|
|
39
|
-
return df
|
|
40
|
-
|
|
41
|
-
|
|
42
26
|
class GlucoseEndpointsByDate:
|
|
43
27
|
"""
|
|
44
28
|
Usage:
|
|
@@ -58,10 +42,11 @@ class GlucoseEndpointsByDate:
|
|
|
58
42
|
"""
|
|
59
43
|
|
|
60
44
|
fbg_threshhold = 7.0
|
|
45
|
+
fbg_beyond_threshold = FBG_BEYOND_THRESHOLD
|
|
61
46
|
ogtt_threshhold = 11.1
|
|
62
47
|
endpoint_cls = EndpointByDate
|
|
63
|
-
keep_cols = [
|
|
64
|
-
"
|
|
48
|
+
keep_cols = [ # noqa: RUF012
|
|
49
|
+
"fasted",
|
|
65
50
|
"fasting_hrs",
|
|
66
51
|
"fbg_value",
|
|
67
52
|
"fbg_units",
|
|
@@ -91,217 +76,86 @@ class GlucoseEndpointsByDate:
|
|
|
91
76
|
CASE_OGTT,
|
|
92
77
|
CASE_FBGS_WITH_FIRST_OGTT,
|
|
93
78
|
CASE_FBGS_WITH_SECOND_OGTT,
|
|
79
|
+
CASE_FBG_VERY_HIGH,
|
|
94
80
|
CASE_EOS,
|
|
95
81
|
]
|
|
96
82
|
self.endpoint_cases = {k: v for k, v in endpoint_cases.items() if k in self.case_list}
|
|
97
83
|
|
|
98
|
-
|
|
99
|
-
if self.glucose_fbg_ogtt_df.empty:
|
|
100
|
-
self.df = self.glucose_fbg_df.copy()
|
|
101
|
-
self.df[["ogtt_value", "ogtt_units"]] = np.nan
|
|
102
|
-
self.df[["ogtt_datetime"]] = pd.NaT
|
|
103
|
-
elif self.glucose_fbg_df.empty:
|
|
104
|
-
self.df = self.glucose_fbg_ogtt_df.copy()
|
|
105
|
-
else:
|
|
106
|
-
self.df = self.glucose_fbg_ogtt_df.merge(
|
|
107
|
-
self.glucose_fbg_df,
|
|
108
|
-
on=["subject_visit_id"],
|
|
109
|
-
how="outer",
|
|
110
|
-
indicator=True,
|
|
111
|
-
suffixes=("", "_y"),
|
|
112
|
-
)
|
|
113
|
-
# cast as ...
|
|
114
|
-
for col in ["fasting_hrs", "fbg_value"]:
|
|
115
|
-
self.df[col] = self.df[col].astype("float64")
|
|
116
|
-
if f"{col}_y" in self.df.columns:
|
|
117
|
-
self.df[f"{col}_y"] = self.df[f"{col}_y"].astype("float64")
|
|
118
|
-
for col in ["fasting", "fbg_units", "source"]:
|
|
119
|
-
self.df[col] = self.df[col].astype("object")
|
|
120
|
-
if f"{col}_y" in self.df.columns:
|
|
121
|
-
self.df[f"{col}_y"] = self.df[f"{col}_y"].astype("object")
|
|
122
|
-
self.df = self.df.drop(
|
|
123
|
-
columns=[col for col in self.df.columns if col.endswith("_y") or col == "_merge"]
|
|
124
|
-
)
|
|
125
|
-
self.df = self.df.reset_index(drop=True)
|
|
126
|
-
|
|
127
|
-
# merge w/ subject_visit
|
|
128
|
-
subject_visit_df = get_subject_visit(
|
|
129
|
-
"meta_subject.subjectvisit", subject_identifiers=self.subject_identifiers
|
|
130
|
-
)
|
|
131
|
-
self.df = subject_visit_df.merge(
|
|
132
|
-
self.df, on=["subject_visit_id"], how="left", suffixes=("", "_y")
|
|
133
|
-
)
|
|
134
|
-
self.df = self.df[[col for col in self.keep_cols]]
|
|
135
|
-
self.df = self.df.reset_index(drop=True)
|
|
136
|
-
|
|
137
|
-
# pivot right_only cols
|
|
138
|
-
cols = [
|
|
139
|
-
"fasting",
|
|
140
|
-
"fasting_hrs",
|
|
141
|
-
"fbg_value",
|
|
142
|
-
"fbg_units",
|
|
143
|
-
"fbg_datetime",
|
|
144
|
-
"source",
|
|
145
|
-
"report_datetime",
|
|
146
|
-
]
|
|
147
|
-
for col in cols:
|
|
148
|
-
if f"{col}_y" in self.df.columns and not self.df[f"{col}_y"].isnull().all():
|
|
149
|
-
self.df.loc[
|
|
150
|
-
(self.df["_merge"].isin(["both", "right_only"])) & (self.df[col].isna()),
|
|
151
|
-
col,
|
|
152
|
-
] = self.df[f"{col}_y"]
|
|
153
|
-
# if fbg_datetime still null, use visit datetime
|
|
154
|
-
if self.df["fbg_datetime"].isnull().all():
|
|
155
|
-
self["fbg_datetime"] = self.df["visit_datetime"]
|
|
156
|
-
else:
|
|
157
|
-
self.df.loc[(self.df["fbg_datetime"].isna()), "fbg_datetime"] = self.df[
|
|
158
|
-
"visit_datetime"
|
|
159
|
-
]
|
|
160
|
-
self.df = self.df.drop(
|
|
161
|
-
columns=[col for col in self.df.columns if col.endswith("_y") or col == "_merge"]
|
|
162
|
-
)
|
|
163
|
-
self.df = self.df.reset_index(drop=True)
|
|
164
|
-
|
|
165
|
-
self.merge_with_consent()
|
|
166
|
-
self.merge_with_eos()
|
|
167
|
-
|
|
168
|
-
self.add_calculated_days_from_baseline_to_event_columns()
|
|
84
|
+
self.df = get_glucose_df(subject_identifiers=self.subject_identifiers).copy()
|
|
169
85
|
|
|
170
86
|
# label rows by type of glu tests (ones with value)
|
|
171
87
|
self.df["test"] = self.df.apply(get_test_string, axis=1)
|
|
172
|
-
self.df = self.df.reset_index(drop=True)
|
|
173
|
-
|
|
174
|
-
self.visit_codes_df = get_unique_visit_codes(self.df)
|
|
175
|
-
self.subject_identifiers_df = get_unique_subject_identifiers(self.df)
|
|
176
|
-
|
|
177
|
-
self.df = self.df.sort_values(by=["subject_identifier", "fbg_datetime"])
|
|
178
|
-
self.df = self.df.reset_index(drop=True)
|
|
179
88
|
|
|
89
|
+
self.df = (
|
|
90
|
+
self.df.query("not (fbg_value.isna() and ogtt_value.isna())")
|
|
91
|
+
.sort_values(by=["subject_identifier", "fbg_datetime"])
|
|
92
|
+
.reset_index(drop=True)
|
|
93
|
+
)
|
|
180
94
|
self.working_df = self.df.copy()
|
|
181
95
|
self.working_df["endpoint"] = 0
|
|
182
96
|
self.endpoint_df = get_empty_endpoint_df()
|
|
183
97
|
|
|
184
98
|
def run(self):
|
|
185
|
-
self.
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
99
|
+
self.process_by_ogtt_only()
|
|
100
|
+
self.process_by_fbg_only()
|
|
101
|
+
subject_identifiers_df = get_unique_subject_identifiers(self.df)
|
|
102
|
+
for _, row in subject_identifiers_df.iterrows():
|
|
103
|
+
subject_df = self.endpoint_cls(
|
|
104
|
+
subject_df=self.get_subject_df(row["subject_identifier"]),
|
|
105
|
+
fbg_threshhold=self.fbg_threshhold,
|
|
106
|
+
ogtt_threshhold=self.ogtt_threshhold,
|
|
107
|
+
).subject_df
|
|
191
108
|
if len(subject_df.loc[subject_df["endpoint"] == 1]) == 1:
|
|
192
109
|
self.append_subject_to_endpoint_df(subject_df)
|
|
193
110
|
self.remove_subject_from_working_df(row)
|
|
194
|
-
|
|
195
|
-
if CASE_FBG_ONLY in self.endpoint_cases:
|
|
196
|
-
for index, row in self.subject_identifiers_df.iterrows():
|
|
197
|
-
subject_df = self.get_subject_df(row["subject_identifier"])
|
|
198
|
-
subject_df = self.check_endpoint_by_fbg_for_subject(
|
|
199
|
-
subject_df, case_list=[CASE_FBG_ONLY]
|
|
200
|
-
)
|
|
201
|
-
if len(subject_df.loc[subject_df["endpoint"] == 1]) == 1:
|
|
202
|
-
self.append_subject_to_endpoint_df(subject_df)
|
|
203
|
-
self.remove_subject_from_working_df(row)
|
|
204
|
-
|
|
205
111
|
self.post_check_endpoint()
|
|
206
112
|
self.merge_with_final_endpoints()
|
|
207
113
|
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
114
|
+
def process_by_fbg_only(self):
|
|
115
|
+
"""Flag subjects that meta endpoint by hitting the absurd FBG"""
|
|
116
|
+
subject_endpoint_df = self.working_df.loc[
|
|
117
|
+
(self.working_df["fbg_value"] >= self.fbg_beyond_threshold)
|
|
118
|
+
# & (self.working_df["fasted"] == YES)
|
|
119
|
+
].copy()
|
|
212
120
|
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
subject_identifiers=self.subject_identifiers,
|
|
219
|
-
# subject_visit_model="meta_subject.subjectvisit",
|
|
220
|
-
)
|
|
221
|
-
df["source"] = "meta_subject.glucosefbg"
|
|
222
|
-
df.rename(columns={"fbg_fasting": "fasting"}, inplace=True)
|
|
223
|
-
df.loc[(df["fasting"] == "fasting"), "fasting"] = YES
|
|
224
|
-
df.loc[(df["fasting"] == "non_fasting"), "fasting"] = NO
|
|
225
|
-
df = calculate_fasting_hrs(df)
|
|
226
|
-
# df = df[[col for col in self.keep_cols if not col.startswith("ogtt")]]
|
|
227
|
-
df = df.reset_index(drop=True)
|
|
228
|
-
self._glucose_fbg_df = df
|
|
229
|
-
return self._glucose_fbg_df
|
|
230
|
-
|
|
231
|
-
@property
|
|
232
|
-
def glucose_fbg_ogtt_df(self):
|
|
233
|
-
"""Returns a prepared Dataframe of CRF meta_subject.glucose.
|
|
234
|
-
|
|
235
|
-
Note: meta_subject.glucose has FBG and OGTT measures.
|
|
236
|
-
"""
|
|
237
|
-
if self._glucose_fbg_ogtt_df.empty:
|
|
238
|
-
df = get_crf(
|
|
239
|
-
model="meta_subject.glucose",
|
|
240
|
-
subject_identifiers=self.subject_identifiers,
|
|
241
|
-
# subject_visit_model="meta_subject.subjectvisit",
|
|
242
|
-
)
|
|
243
|
-
df["source"] = "meta_subject.glucose"
|
|
244
|
-
df = calculate_fasting_hrs(df)
|
|
245
|
-
# df = df[self.keep_cols]
|
|
246
|
-
df = df.reset_index(drop=True)
|
|
247
|
-
self._glucose_fbg_ogtt_df = df
|
|
248
|
-
return self._glucose_fbg_ogtt_df
|
|
249
|
-
|
|
250
|
-
def merge_with_consent(self):
|
|
251
|
-
"""Merge in consent DF."""
|
|
252
|
-
df_consent = get_subject_consent(
|
|
253
|
-
"meta_consent.subjectconsent", subject_identifiers=self.subject_identifiers
|
|
254
|
-
)
|
|
255
|
-
self.df = pd.merge(
|
|
256
|
-
self.df, df_consent, on="subject_identifier", how="left", suffixes=("", "_y")
|
|
121
|
+
subject_endpoint_df = (
|
|
122
|
+
subject_endpoint_df.sort_values(by=["subject_identifier", "fbg_datetime"])
|
|
123
|
+
.reset_index(drop=True)
|
|
124
|
+
.drop_duplicates(subset=["subject_identifier"], keep="first")
|
|
125
|
+
.reset_index(drop=True)
|
|
257
126
|
)
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
127
|
+
if not subject_endpoint_df.empty:
|
|
128
|
+
# flag the selected endpoint rows as endpoints
|
|
129
|
+
subject_endpoint_df["endpoint"] = 1
|
|
130
|
+
subject_endpoint_df["endpoint_label"] = self.endpoint_cases[CASE_FBG_VERY_HIGH]
|
|
131
|
+
subject_endpoint_df["endpoint_type"] = CASE_FBG_VERY_HIGH
|
|
132
|
+
subject_endpoint_df["interval_in_days"] = np.nan
|
|
263
133
|
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
134
|
+
# add back the others rows for these subjects
|
|
135
|
+
subjects_df = self.working_df.loc[
|
|
136
|
+
(
|
|
137
|
+
self.working_df["subject_identifier"].isin(
|
|
138
|
+
subject_endpoint_df["subject_identifier"]
|
|
139
|
+
)
|
|
140
|
+
& ~(
|
|
141
|
+
self.working_df["fbg_datetime"].isin(
|
|
142
|
+
subject_endpoint_df["fbg_datetime"]
|
|
143
|
+
)
|
|
144
|
+
)
|
|
145
|
+
)
|
|
146
|
+
].copy()
|
|
147
|
+
subjects_df = subjects_df.reset_index(drop=True)
|
|
148
|
+
subjects_df["endpoint"] = np.nan
|
|
149
|
+
subjects_df["endpoint_label"] = None
|
|
150
|
+
subjects_df["endpoint_type"] = None
|
|
151
|
+
subjects_df["interval_in_days"] = np.nan
|
|
152
|
+
subjects_df = pd.concat([subjects_df, subject_endpoint_df])
|
|
153
|
+
subjects_df = subjects_df.reset_index(drop=True)
|
|
279
154
|
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
baseline to visit, fbg, and ogtt.
|
|
283
|
-
"""
|
|
284
|
-
self.df["visit_days"] = np.nan
|
|
285
|
-
self.df["fbg_days"] = np.nan
|
|
286
|
-
self.df["ogtt_days"] = np.nan
|
|
287
|
-
self.df["test"] = np.nan
|
|
288
|
-
self.df["visit_days"] = (
|
|
289
|
-
self.df["visit_datetime"] - self.df["baseline_datetime"]
|
|
290
|
-
).dt.days
|
|
291
|
-
if not self.df["fbg_datetime"].isnull().all():
|
|
292
|
-
self.df["fbg_days"] = (
|
|
293
|
-
self.df["fbg_datetime"] - self.df["baseline_datetime"]
|
|
294
|
-
).dt.days
|
|
295
|
-
if not self.df["ogtt_datetime"].isnull().all():
|
|
296
|
-
self.df["ogtt_days"] = (
|
|
297
|
-
self.df["ogtt_datetime"] - self.df["baseline_datetime"]
|
|
298
|
-
).dt.days
|
|
299
|
-
self.df["visit_days"] = pd.to_numeric(self.df["visit_days"], downcast="integer")
|
|
300
|
-
self.df["fbg_days"] = pd.to_numeric(self.df["fbg_days"], downcast="integer")
|
|
301
|
-
self.df["ogtt_days"] = pd.to_numeric(self.df["ogtt_days"], downcast="integer")
|
|
302
|
-
self.df = self.df.reset_index(drop=True)
|
|
155
|
+
self.append_subject_to_endpoint_df(subjects_df[endpoint_columns])
|
|
156
|
+
self.remove_subjects_from_working_df(subjects_df)
|
|
303
157
|
|
|
304
|
-
def
|
|
158
|
+
def process_by_ogtt_only(self):
|
|
305
159
|
"""Flag subjects that met endpoint by hitting the OGTT
|
|
306
160
|
threshold.
|
|
307
161
|
|
|
@@ -322,15 +176,17 @@ class GlucoseEndpointsByDate:
|
|
|
322
176
|
"""
|
|
323
177
|
subject_endpoint_df = self.working_df.loc[
|
|
324
178
|
(self.working_df["ogtt_value"] >= self.ogtt_threshhold)
|
|
325
|
-
& (self.working_df["
|
|
179
|
+
& (self.working_df["ogtt_value"] <= 9999.99)
|
|
180
|
+
& (self.working_df["fasted"] == YES)
|
|
326
181
|
& (self.working_df["fbg_value"].notna())
|
|
327
182
|
].copy()
|
|
328
|
-
|
|
329
|
-
subject_endpoint_df =
|
|
330
|
-
|
|
331
|
-
|
|
183
|
+
|
|
184
|
+
subject_endpoint_df = (
|
|
185
|
+
subject_endpoint_df.sort_values(by=["subject_identifier", "fbg_datetime"])
|
|
186
|
+
.reset_index(drop=True)
|
|
187
|
+
.drop_duplicates(subset=["subject_identifier"], keep="first")
|
|
188
|
+
.reset_index(drop=True)
|
|
332
189
|
)
|
|
333
|
-
subject_endpoint_df = subject_endpoint_df.reset_index(drop=True)
|
|
334
190
|
if not subject_endpoint_df.empty:
|
|
335
191
|
# flag the selected endpoint rows as endpoints
|
|
336
192
|
subject_endpoint_df["endpoint"] = 1
|
|
@@ -372,8 +228,7 @@ class GlucoseEndpointsByDate:
|
|
|
372
228
|
self.endpoint_df = pd.concat([self.endpoint_df, subject_df])
|
|
373
229
|
self.endpoint_df = self.endpoint_df.sort_values(
|
|
374
230
|
by=["subject_identifier", "visit_code"]
|
|
375
|
-
)
|
|
376
|
-
self.endpoint_df = self.endpoint_df.reset_index(drop=True)
|
|
231
|
+
).reset_index(drop=True)
|
|
377
232
|
|
|
378
233
|
def remove_subject_from_working_df(self, row: pd.Series) -> None:
|
|
379
234
|
"""Removes one subject from the working DF given a Series with
|
|
@@ -383,8 +238,7 @@ class GlucoseEndpointsByDate:
|
|
|
383
238
|
index=self.working_df[
|
|
384
239
|
self.working_df["subject_identifier"] == row["subject_identifier"]
|
|
385
240
|
].index
|
|
386
|
-
)
|
|
387
|
-
self.working_df = self.working_df.reset_index(drop=True)
|
|
241
|
+
).reset_index(drop=True)
|
|
388
242
|
|
|
389
243
|
def remove_subjects_from_working_df(self, rows: pd.DataFrame) -> None:
|
|
390
244
|
"""Removes subjects from the working DF given a DF with
|
|
@@ -394,8 +248,7 @@ class GlucoseEndpointsByDate:
|
|
|
394
248
|
index=self.working_df.loc[
|
|
395
249
|
self.working_df["subject_identifier"].isin(rows["subject_identifier"])
|
|
396
250
|
].index
|
|
397
|
-
)
|
|
398
|
-
self.working_df = self.working_df.reset_index(drop=True)
|
|
251
|
+
).reset_index(drop=True)
|
|
399
252
|
|
|
400
253
|
def get_subject_df(self, subject_identifier: str) -> pd.DataFrame:
|
|
401
254
|
subject_df = self.working_df.loc[
|
|
@@ -407,22 +260,28 @@ class GlucoseEndpointsByDate:
|
|
|
407
260
|
subject_df["endpoint"] = 0
|
|
408
261
|
subject_df = subject_df[endpoint_columns]
|
|
409
262
|
subject_df = subject_df.sort_values(["subject_identifier", "fbg_datetime"])
|
|
410
|
-
subject_df = subject_df
|
|
411
|
-
|
|
263
|
+
subject_df[[col for col in subject_df if "value" in col]] = subject_df[
|
|
264
|
+
[col for col in subject_df if "value" in col]
|
|
265
|
+
].fillna(0.0)
|
|
266
|
+
|
|
267
|
+
return subject_df.reset_index(drop=True)
|
|
412
268
|
|
|
413
269
|
def check_endpoint_by_fbg_for_subject(
|
|
414
|
-
self,
|
|
270
|
+
self,
|
|
271
|
+
subject_df: pd.DataFrame,
|
|
272
|
+
case_list: list[int] | None = None, # noqa: ARG002
|
|
415
273
|
) -> pd.DataFrame:
|
|
416
|
-
case_list = case_list or [2, 3]
|
|
417
274
|
endpoint = self.endpoint_cls(
|
|
418
275
|
subject_df=subject_df,
|
|
419
276
|
fbg_threshhold=self.fbg_threshhold,
|
|
420
277
|
ogtt_threshhold=self.ogtt_threshhold,
|
|
421
|
-
case_list=case_list,
|
|
422
278
|
)
|
|
423
279
|
return endpoint.subject_df
|
|
424
280
|
|
|
425
281
|
def post_check_endpoint(self):
|
|
282
|
+
"""Add any who were taken off study before endpoint guidelines
|
|
283
|
+
were clearly defined.
|
|
284
|
+
"""
|
|
426
285
|
df_eos = self.working_df.loc[
|
|
427
286
|
self.working_df["offstudy_reason"] == "Patient developed diabetes"
|
|
428
287
|
].copy()
|
|
@@ -477,24 +336,37 @@ class GlucoseEndpointsByDate:
|
|
|
477
336
|
self.endpoint_only_df = pd.concat([df1, df2])
|
|
478
337
|
self.endpoint_only_df = self.endpoint_only_df.reset_index(drop=True)
|
|
479
338
|
|
|
480
|
-
self.df =
|
|
481
|
-
self.df,
|
|
339
|
+
self.df = self.df.merge(
|
|
482
340
|
self.endpoint_only_df[["subject_identifier", "visit_code", "endpoint"]],
|
|
483
341
|
on=["subject_identifier", "visit_code"],
|
|
484
342
|
how="left",
|
|
485
343
|
suffixes=("", "_y"),
|
|
486
344
|
)
|
|
487
|
-
self.df = self.df.sort_values(by=["subject_identifier", "fbg_datetime"])
|
|
488
|
-
|
|
345
|
+
self.df = self.df.sort_values(by=["subject_identifier", "fbg_datetime"]).reset_index(
|
|
346
|
+
drop=True
|
|
347
|
+
)
|
|
489
348
|
|
|
490
349
|
def to_model(self):
|
|
491
350
|
"""Write endpoint_only_df to the Endpoints model"""
|
|
492
351
|
df = self.endpoint_only_df
|
|
493
352
|
model = "meta_reports.endpoints"
|
|
494
|
-
now =
|
|
353
|
+
now = timezone.now()
|
|
495
354
|
model_cls = django_apps.get_model(model)
|
|
496
355
|
if self.subject_identifiers:
|
|
497
356
|
model_cls.objects.filter(subject_identifier__in=self.subject_identifiers).delete()
|
|
357
|
+
if self.endpoint_only_df.empty:
|
|
358
|
+
df = pd.DataFrame()
|
|
359
|
+
else:
|
|
360
|
+
df = (
|
|
361
|
+
self.endpoint_only_df[
|
|
362
|
+
self.endpoint_only_df["subject_identifier"].isin(
|
|
363
|
+
self.subject_identifiers
|
|
364
|
+
)
|
|
365
|
+
]
|
|
366
|
+
.copy()
|
|
367
|
+
.sort_values(by=["subject_identifier"])
|
|
368
|
+
.reset_index(drop=True)
|
|
369
|
+
)
|
|
498
370
|
else:
|
|
499
371
|
model_cls.objects.all().delete()
|
|
500
372
|
created = 0
|
|
@@ -512,15 +384,19 @@ class GlucoseEndpointsByDate:
|
|
|
512
384
|
fbg_value=(None if pd.isna(row["fbg_value"]) else row["fbg_value"]),
|
|
513
385
|
ogtt_value=None if pd.isna(row["ogtt_value"]) else row["ogtt_value"],
|
|
514
386
|
fbg_date=(None if pd.isna(row["fbg_datetime"]) else row["fbg_datetime"]),
|
|
515
|
-
fasting=(
|
|
387
|
+
fasting=(NULL_STRING if pd.isna(row["fasted"]) else row["fasted"]),
|
|
516
388
|
endpoint_label=(
|
|
517
|
-
|
|
389
|
+
NULL_STRING
|
|
390
|
+
if pd.isna(row["endpoint_label"])
|
|
391
|
+
else row["endpoint_label"]
|
|
518
392
|
),
|
|
519
393
|
offstudy_date=(
|
|
520
394
|
None if pd.isna(row["offstudy_datetime"]) else row["offstudy_datetime"]
|
|
521
395
|
),
|
|
522
396
|
offstudy_reason=(
|
|
523
|
-
|
|
397
|
+
NULL_STRING
|
|
398
|
+
if pd.isna(row["offstudy_reason"])
|
|
399
|
+
else row["offstudy_reason"]
|
|
524
400
|
),
|
|
525
401
|
report_model=model,
|
|
526
402
|
created=now,
|
|
@@ -72,7 +72,7 @@ def get_screening_df(df: pd.DataFrame | None = None) -> pd.DataFrame:
|
|
|
72
72
|
# condition to include any glucose test
|
|
73
73
|
|
|
74
74
|
# has_dm fillna with unk
|
|
75
|
-
df["has_dm"] = df["has_dm"].apply(lambda x:
|
|
75
|
+
df["has_dm"] = df["has_dm"].apply(lambda x: x if x else "unk")
|
|
76
76
|
|
|
77
77
|
na = "Not applicable, subject is not eligible based on the criteria above"
|
|
78
78
|
df["already_fasted"] = df["already_fasted"].apply(lambda x: "N/A" if x == na else x)
|
|
@@ -110,29 +110,26 @@ def get_screening_df(df: pd.DataFrame | None = None) -> pd.DataFrame:
|
|
|
110
110
|
|
|
111
111
|
# bmi
|
|
112
112
|
# subject SR9E8B4D has eligible part two == No but subject has a glucose value
|
|
113
|
-
|
|
113
|
+
# NOTE: update 10-02-2025: subject is not eligible by congestive_heart_failure
|
|
114
|
+
# df.loc[(df["screening_identifier"] == "SR9E8B4D"), "eligible_part_two"] = "Yes"
|
|
114
115
|
|
|
115
116
|
# merge with physical exam to get waist circumference if taken at baseline
|
|
116
117
|
subject_identifiers = list(df["subject_identifier"])
|
|
117
118
|
qs_subject_visit = SubjectVisit.objects.filter(subject_identifier__in=subject_identifiers)
|
|
118
119
|
df_subject_visit = read_frame(qs_subject_visit)
|
|
119
|
-
df_subject_visit.rename(columns={"id": "subject_visit"}
|
|
120
|
+
df_subject_visit = df_subject_visit.rename(columns={"id": "subject_visit"})
|
|
120
121
|
qs_physical_exam = PhysicalExam.objects.filter(
|
|
121
122
|
subject_visit__subject_identifier__in=subject_identifiers
|
|
122
123
|
)
|
|
123
124
|
df_physical_exam = read_frame(qs_physical_exam)
|
|
124
125
|
# merge w/ subject visit to get subject_identifier
|
|
125
|
-
df_physical_exam =
|
|
126
|
-
df_physical_exam,
|
|
126
|
+
df_physical_exam = df_physical_exam.merge(
|
|
127
127
|
df_subject_visit[
|
|
128
128
|
["subject_visit", "subject_identifier", "visit_code", "visit_code_sequence"]
|
|
129
129
|
],
|
|
130
130
|
on="subject_visit",
|
|
131
131
|
how="left",
|
|
132
|
-
)
|
|
133
|
-
df_physical_exam = df_physical_exam[
|
|
134
|
-
["subject_identifier", "visit_code", "visit_code_sequence", "waist_circumference"]
|
|
135
|
-
]
|
|
132
|
+
)[["subject_identifier", "visit_code", "visit_code_sequence", "waist_circumference"]]
|
|
136
133
|
df_physical_exam[["waist_circumference"]] = df_physical_exam[
|
|
137
134
|
["waist_circumference"]
|
|
138
135
|
].apply(pd.to_numeric)
|
|
@@ -143,12 +140,11 @@ def get_screening_df(df: pd.DataFrame | None = None) -> pd.DataFrame:
|
|
|
143
140
|
["waist_circumference_baseline"]
|
|
144
141
|
].apply(pd.to_numeric)
|
|
145
142
|
# merge on subject_identifier with main DF
|
|
146
|
-
df =
|
|
147
|
-
df,
|
|
143
|
+
df = df.merge(
|
|
148
144
|
df_physical_exam[["subject_identifier", "waist_circumference_baseline"]],
|
|
149
145
|
on="subject_identifier",
|
|
150
146
|
how="left",
|
|
151
|
-
)
|
|
147
|
+
).reset_index(drop=True)
|
|
152
148
|
# set waist_circumference=waist_circumference_baseline
|
|
153
149
|
# if `waist_circumference` is none and `waist_circumference_baseline` is not
|
|
154
150
|
df.loc[
|
|
@@ -157,6 +153,4 @@ def get_screening_df(df: pd.DataFrame | None = None) -> pd.DataFrame:
|
|
|
157
153
|
] = df["waist_circumference_baseline"]
|
|
158
154
|
|
|
159
155
|
# drop waist_circumference_baseline
|
|
160
|
-
df.drop(columns=["waist_circumference_baseline"]
|
|
161
|
-
|
|
162
|
-
return df
|
|
156
|
+
return df.drop(columns=["waist_circumference_baseline"])
|
|
@@ -1,4 +1,6 @@
|
|
|
1
|
+
import numpy as np
|
|
1
2
|
import pandas as pd
|
|
3
|
+
from clinicedc_constants import NO
|
|
2
4
|
|
|
3
5
|
from .constants import endpoint_columns
|
|
4
6
|
|
|
@@ -35,16 +37,12 @@ def get_empty_endpoint_df() -> pd.DataFrame:
|
|
|
35
37
|
"endpoint",
|
|
36
38
|
"endpoint_type",
|
|
37
39
|
]
|
|
38
|
-
].apply(
|
|
39
|
-
pd.to_numeric
|
|
40
|
-
)
|
|
40
|
+
].apply(pd.to_numeric)
|
|
41
41
|
endpoint_df[
|
|
42
42
|
["baseline_datetime", "visit_datetime", "fbg_datetime", "offstudy_datetime"]
|
|
43
43
|
] = endpoint_df[
|
|
44
44
|
["baseline_datetime", "visit_datetime", "fbg_datetime", "offstudy_datetime"]
|
|
45
|
-
].apply(
|
|
46
|
-
pd.to_datetime
|
|
47
|
-
)
|
|
45
|
+
].apply(pd.to_datetime)
|
|
48
46
|
endpoint_df["visit_code"] = endpoint_df["visit_code"].astype(float)
|
|
49
47
|
return endpoint_df
|
|
50
48
|
|
|
@@ -54,12 +52,23 @@ def get_unique_visit_codes(df: pd.DataFrame) -> pd.DataFrame:
|
|
|
54
52
|
stats_df = stats_df.reset_index()
|
|
55
53
|
stats_df["visit_code"] = stats_df["visit_code"].astype(float)
|
|
56
54
|
stats_df = stats_df.sort_values(["visit_code"])
|
|
57
|
-
|
|
58
|
-
return stats_df
|
|
55
|
+
return stats_df.reset_index(drop=True)
|
|
59
56
|
|
|
60
57
|
|
|
61
58
|
def get_unique_subject_identifiers(df: pd.DataFrame) -> pd.DataFrame:
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
59
|
+
return (
|
|
60
|
+
pd.DataFrame(df["subject_identifier"].unique(), columns=["subject_identifier"])
|
|
61
|
+
.sort_values(["subject_identifier"])
|
|
62
|
+
.reset_index()
|
|
63
|
+
)
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
def calculate_fasting_hrs(df: pd.DataFrame) -> pd.DataFrame:
|
|
67
|
+
df.loc[(df["fasting"] == NO), "fasting_duration_delta"] = pd.NaT
|
|
68
|
+
if df.empty:
|
|
69
|
+
df["fasting_hrs"] = np.nan
|
|
70
|
+
else:
|
|
71
|
+
df["fasting_hrs"] = df["fasting_duration_delta"].apply(
|
|
72
|
+
lambda s: np.nan if pd.isna(s) else s.total_seconds() / 3600
|
|
73
|
+
)
|
|
74
|
+
return df
|
meta_analytics/get_tables.py
CHANGED
|
@@ -35,8 +35,7 @@ def get_tables() -> dict[str, Data]:
|
|
|
35
35
|
& (df_not_tested["eligible_part_two"] == "Yes")
|
|
36
36
|
)
|
|
37
37
|
]
|
|
38
|
-
df_not_tested.drop(df_not_eligible_p1p2.index
|
|
39
|
-
df_not_tested.drop(df_fbg.index, inplace=True)
|
|
38
|
+
df_not_tested = df_not_tested.drop(df_not_eligible_p1p2.index).drop(df_fbg.index)
|
|
40
39
|
|
|
41
40
|
df_stats = pd.DataFrame(
|
|
42
41
|
{
|
|
@@ -17,10 +17,11 @@ class GlucoseTable(Table):
|
|
|
17
17
|
super().__init__(colname="", main_df=main_df, title="Glucose (enrolled)")
|
|
18
18
|
|
|
19
19
|
def build_table_df(self) -> None:
|
|
20
|
+
super().build_table_df()
|
|
20
21
|
df = self.main_df
|
|
21
22
|
s = df.groupby("subject_identifier")[["age_in_years", "gender"]].value_counts()
|
|
22
23
|
df_tmp = s.to_frame()
|
|
23
24
|
df_tmp = df_tmp.reset_index()
|
|
24
25
|
gender_tbl = GenderTable(main_df=df_tmp).table_df
|
|
25
26
|
age_tbl = AgeTable(main_df=df_tmp).table_df
|
|
26
|
-
|
|
27
|
+
self.table_df = pd.concat([self.table_df, gender_tbl, age_tbl]).reset_index(drop=True)
|