All checks were successful
Build and Push Docker Image / build (push) Successful in 1m5s
219 lines
15 KiB
Python
219 lines
15 KiB
Python
# Generated by Django 6.0.1 on 2026-05-11 20:32
|
|
|
|
import django.db.models.deletion
|
|
import uuid
|
|
from django.conf import settings
|
|
from django.db import migrations, models
|
|
|
|
|
|
class Migration(migrations.Migration):
|
|
|
|
initial = True
|
|
|
|
dependencies = [
|
|
('organizations', '0001_initial'),
|
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
|
]
|
|
|
|
operations = [
|
|
migrations.CreateModel(
|
|
name='KPI',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('name', models.CharField(max_length=200, unique=True)),
|
|
('name_ar', models.CharField(blank=True, max_length=200)),
|
|
('description', models.TextField(blank=True)),
|
|
('category', models.CharField(choices=[('patient_satisfaction', 'Patient Satisfaction'), ('complaint_management', 'Complaint Management'), ('action_management', 'Action Management'), ('sla_compliance', 'SLA Compliance'), ('survey_response', 'Survey Response'), ('operational', 'Operational')], db_index=True, max_length=100)),
|
|
('unit', models.CharField(help_text='Unit of measurement (%, count, hours, etc.)', max_length=50)),
|
|
('calculation_method', models.TextField(help_text='Description of how this KPI is calculated')),
|
|
('target_value', models.DecimalField(blank=True, decimal_places=2, help_text='Target value for this KPI', max_digits=10, null=True)),
|
|
('warning_threshold', models.DecimalField(blank=True, decimal_places=2, help_text='Warning threshold', max_digits=10, null=True)),
|
|
('critical_threshold', models.DecimalField(blank=True, decimal_places=2, help_text='Critical threshold', max_digits=10, null=True)),
|
|
('is_active', models.BooleanField(default=True)),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI',
|
|
'verbose_name_plural': 'KPIs',
|
|
'ordering': ['category', 'name'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIReport',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('report_type', models.CharField(choices=[('resolution_72h', 'MOH-2: 72-Hour Resolution Rate'), ('patient_experience', 'MOH-1: Patient Experience Score'), ('satisfaction_resolution', 'MOH-3: Overall Satisfaction with Resolution'), ('n_pad_001', 'N-PAD-001: Resolution to Patient Complaints'), ('response_rate', 'Dep-KPI-4: Department Response Rate'), ('activation_2h', 'KPI-6: Complaint Activation Within 2 Hours'), ('unactivated', 'KPI-7: Unactivated Filled Complaints Rate'), ('moh_24h', 'MOH-24h: 24-Hour MOH Complaint Resolution Rate'), ('chi_48h', 'CHI-48h: 48-Hour CHI Complaint Resolution Rate')], db_index=True, help_text='Type of KPI report', max_length=50)),
|
|
('year', models.IntegerField(db_index=True)),
|
|
('month', models.IntegerField(db_index=True)),
|
|
('report_date', models.DateField(help_text='Date the report was generated')),
|
|
('status', models.CharField(choices=[('pending', 'Pending'), ('generating', 'Generating'), ('completed', 'Completed'), ('failed', 'Failed')], db_index=True, default='pending', max_length=20)),
|
|
('generated_at', models.DateTimeField(blank=True, null=True)),
|
|
('target_percentage', models.DecimalField(decimal_places=2, default=95.0, help_text='Target percentage for this KPI', max_digits=5)),
|
|
('threshold_percentage', models.DecimalField(decimal_places=2, default=90.0, help_text='Threshold (minimum acceptable) percentage for this KPI', max_digits=5)),
|
|
('category', models.CharField(default='Organizational', help_text='Report category (e.g., Organizational, Clinical)', max_length=50)),
|
|
('kpi_type', models.CharField(default='Outcome', help_text='KPI type (e.g., Outcome, Process, Structure)', max_length=50)),
|
|
('risk_level', models.CharField(choices=[('High', 'High'), ('Medium', 'Medium'), ('Low', 'Low')], default='High', help_text='Risk level for this KPI', max_length=20)),
|
|
('data_collection_method', models.CharField(default='Retrospective', help_text='Data collection method', max_length=50)),
|
|
('data_collection_frequency', models.CharField(default='Monthly', help_text='How often data is collected', max_length=50)),
|
|
('reporting_frequency', models.CharField(default='Monthly', help_text='How often report is generated', max_length=50)),
|
|
('dimension', models.CharField(default='Efficiency', help_text='KPI dimension (e.g., Efficiency, Quality, Safety)', max_length=50)),
|
|
('collector_name', models.CharField(blank=True, help_text='Name of data collector', max_length=200)),
|
|
('analyzer_name', models.CharField(blank=True, help_text='Name of data analyzer', max_length=200)),
|
|
('total_numerator', models.IntegerField(default=0, help_text='Total count of successful outcomes')),
|
|
('total_denominator', models.IntegerField(default=0, help_text='Total count of all cases')),
|
|
('overall_result', models.DecimalField(decimal_places=2, default=0.0, help_text='Overall percentage result', max_digits=6)),
|
|
('error_message', models.TextField(blank=True)),
|
|
('ai_analysis', models.JSONField(blank=True, help_text='AI-generated analysis and recommendations for this report', null=True)),
|
|
('ai_analysis_generated_at', models.DateTimeField(blank=True, help_text='When the AI analysis was generated', null=True)),
|
|
('generated_by', models.ForeignKey(blank=True, help_text='User who generated the report (null for automated)', null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='generated_kpi_reports', to=settings.AUTH_USER_MODEL)),
|
|
('hospital', models.ForeignKey(help_text='Hospital this report belongs to', on_delete=django.db.models.deletion.CASCADE, related_name='kpi_reports', to='organizations.hospital')),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI Report',
|
|
'verbose_name_plural': 'KPI Reports',
|
|
'ordering': ['-year', '-month', 'report_type'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIReportDepartmentBreakdown',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('department_category', models.CharField(choices=[('medical', 'Medical Department'), ('nursing', 'Nursing Department'), ('non_medical', 'Non-Medical / Admin'), ('support_services', 'Support Services')], help_text='Category of department', max_length=50)),
|
|
('complaint_count', models.IntegerField(default=0)),
|
|
('resolved_count', models.IntegerField(default=0)),
|
|
('avg_resolution_days', models.DecimalField(blank=True, decimal_places=2, max_digits=5, null=True)),
|
|
('top_areas', models.TextField(blank=True, help_text='Top complaint areas or notes (newline-separated)')),
|
|
('details', models.JSONField(blank=True, default=dict)),
|
|
('kpi_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='department_breakdowns', to='analytics.kpireport')),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI Department Breakdown',
|
|
'verbose_name_plural': 'KPI Department Breakdowns',
|
|
'ordering': ['department_category'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIReportLocationBreakdown',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('location_type', models.CharField(choices=[('Inpatient', 'Inpatient'), ('Outpatient', 'Outpatient'), ('Emergency', 'Emergency')], help_text='Location type category', max_length=50)),
|
|
('complaint_count', models.IntegerField(default=0)),
|
|
('percentage', models.DecimalField(decimal_places=2, default=0.0, max_digits=5)),
|
|
('kpi_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='location_breakdowns', to='analytics.kpireport')),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI Location Breakdown',
|
|
'verbose_name_plural': 'KPI Location Breakdowns',
|
|
'ordering': ['location_type'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIReportMonthlyData',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('month', models.IntegerField(db_index=True, help_text='Month number (1-12), 0 for TOTAL')),
|
|
('numerator', models.IntegerField(default=0, help_text='Count of successful outcomes')),
|
|
('denominator', models.IntegerField(default=0, help_text='Count of all cases')),
|
|
('percentage', models.DecimalField(decimal_places=2, default=0.0, help_text='Calculated percentage', max_digits=6)),
|
|
('is_below_target', models.BooleanField(default=False, help_text='Whether this month is below target')),
|
|
('details', models.JSONField(blank=True, default=dict, help_text='Additional breakdown data (e.g., by source, department)')),
|
|
('kpi_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='monthly_data', to='analytics.kpireport')),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI Monthly Data',
|
|
'verbose_name_plural': 'KPI Monthly Data',
|
|
'ordering': ['month'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIReportSourceBreakdown',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('source_name', models.CharField(max_length=100)),
|
|
('complaint_count', models.IntegerField(default=0)),
|
|
('percentage', models.DecimalField(decimal_places=2, default=0.0, max_digits=5)),
|
|
('kpi_report', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='source_breakdowns', to='analytics.kpireport')),
|
|
],
|
|
options={
|
|
'verbose_name': 'KPI Source Breakdown',
|
|
'verbose_name_plural': 'KPI Source Breakdowns',
|
|
'ordering': ['-complaint_count'],
|
|
},
|
|
),
|
|
migrations.CreateModel(
|
|
name='KPIValue',
|
|
fields=[
|
|
('created_at', models.DateTimeField(auto_now_add=True, db_index=True)),
|
|
('updated_at', models.DateTimeField(auto_now=True)),
|
|
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
|
('value', models.DecimalField(decimal_places=2, max_digits=10)),
|
|
('period_start', models.DateTimeField(db_index=True)),
|
|
('period_end', models.DateTimeField(db_index=True)),
|
|
('period_type', models.CharField(choices=[('hourly', 'Hourly'), ('daily', 'Daily'), ('weekly', 'Weekly'), ('monthly', 'Monthly'), ('quarterly', 'Quarterly'), ('yearly', 'Yearly')], default='daily', max_length=20)),
|
|
('status', models.CharField(choices=[('on_target', 'On Target'), ('warning', 'Warning'), ('critical', 'Critical')], db_index=True, max_length=20)),
|
|
('metadata', models.JSONField(blank=True, default=dict, help_text='Additional calculation details')),
|
|
('department', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='kpi_values', to='organizations.department')),
|
|
('hospital', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='kpi_values', to='organizations.hospital')),
|
|
('kpi', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='values', to='analytics.kpi')),
|
|
],
|
|
options={
|
|
'ordering': ['-period_end'],
|
|
},
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpireport',
|
|
index=models.Index(fields=['report_type', '-year', '-month'], name='analytics_k_report__c0826c_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpireport',
|
|
index=models.Index(fields=['hospital', '-year', '-month'], name='analytics_k_hospita_ba868a_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpireport',
|
|
index=models.Index(fields=['status', '-created_at'], name='analytics_k_status_8c6a24_idx'),
|
|
),
|
|
migrations.AlterUniqueTogether(
|
|
name='kpireport',
|
|
unique_together={('report_type', 'hospital', 'year', 'month')},
|
|
),
|
|
migrations.AlterUniqueTogether(
|
|
name='kpireportdepartmentbreakdown',
|
|
unique_together={('kpi_report', 'department_category')},
|
|
),
|
|
migrations.AlterUniqueTogether(
|
|
name='kpireportlocationbreakdown',
|
|
unique_together={('kpi_report', 'location_type')},
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpireportmonthlydata',
|
|
index=models.Index(fields=['kpi_report', 'month'], name='analytics_k_kpi_rep_0c4281_idx'),
|
|
),
|
|
migrations.AlterUniqueTogether(
|
|
name='kpireportmonthlydata',
|
|
unique_together={('kpi_report', 'month')},
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpivalue',
|
|
index=models.Index(fields=['kpi', '-period_end'], name='analytics_k_kpi_id_f9c38d_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpivalue',
|
|
index=models.Index(fields=['hospital', 'kpi', '-period_end'], name='analytics_k_hospita_356dca_idx'),
|
|
),
|
|
migrations.AddIndex(
|
|
model_name='kpivalue',
|
|
index=models.Index(fields=['hospital', 'department', '-period_end'], name='analytics_k_hospita_ba95bd_idx'),
|
|
),
|
|
]
|