Fix deployment issue, see DEPLOYMENT_MIGRATION_GUIDE_en.md

This commit is contained in:
2025-08-28 14:13:39 +09:00
parent bba0422efb
commit a180c1e258
29 changed files with 5750 additions and 1 deletions

View File

@ -1,245 +0,0 @@
# Generated by Django 3.2.9 on 2025-08-27 05:59
from django.conf import settings
from django.db import migrations, models
import django.db.models.deletion
import django.utils.timezone
class Migration(migrations.Migration):
dependencies = [
('rog', '0010_auto_20250827_1510'),
]
operations = [
migrations.CreateModel(
name='AppVersion',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('version', models.CharField(help_text='セマンティックバージョン (1.2.3)', max_length=20)),
('platform', models.CharField(choices=[('android', 'Android'), ('ios', 'iOS')], max_length=10)),
('build_number', models.CharField(blank=True, max_length=20, null=True)),
('is_latest', models.BooleanField(default=False, help_text='最新版フラグ')),
('is_required', models.BooleanField(default=False, help_text='強制更新フラグ')),
('update_message', models.TextField(blank=True, help_text='ユーザー向け更新メッセージ', null=True)),
('download_url', models.URLField(blank=True, help_text='アプリストアURL', null=True)),
('release_date', models.DateTimeField(default=django.utils.timezone.now)),
('created_at', models.DateTimeField(auto_now_add=True)),
],
options={
'db_table': 'app_versions',
},
),
migrations.CreateModel(
name='CheckinExtended',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('gps_latitude', models.DecimalField(blank=True, decimal_places=8, max_digits=10, null=True)),
('gps_longitude', models.DecimalField(blank=True, decimal_places=8, max_digits=11, null=True)),
('gps_accuracy', models.DecimalField(blank=True, decimal_places=2, help_text='GPS精度メートル', max_digits=6, null=True)),
('gps_timestamp', models.DateTimeField(blank=True, null=True)),
('camera_capture_time', models.DateTimeField(blank=True, null=True)),
('device_info', models.TextField(blank=True, null=True)),
('validation_status', models.CharField(choices=[('pending', 'Pending'), ('approved', 'Approved'), ('rejected', 'Rejected'), ('requires_review', 'Requires Review')], default='pending', max_length=20)),
('validation_comment', models.TextField(blank=True, null=True)),
('validated_at', models.DateTimeField(blank=True, null=True)),
('bonus_points', models.IntegerField(default=0)),
('scoring_breakdown', models.JSONField(blank=True, default=dict)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
],
options={
'db_table': 'rog_checkin_extended',
},
),
migrations.CreateModel(
name='UploadedImage',
fields=[
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('original_filename', models.CharField(max_length=255)),
('server_filename', models.CharField(max_length=255, unique=True)),
('file_url', models.URLField()),
('file_size', models.BigIntegerField()),
('mime_type', models.CharField(choices=[('image/jpeg', 'JPEG'), ('image/png', 'PNG'), ('image/heic', 'HEIC'), ('image/webp', 'WebP')], max_length=50)),
('event_code', models.CharField(blank=True, max_length=50, null=True)),
('team_name', models.CharField(blank=True, max_length=255, null=True)),
('cp_number', models.IntegerField(blank=True, null=True)),
('upload_source', models.CharField(choices=[('direct', 'Direct'), ('sharing_intent', 'Sharing Intent'), ('bulk_upload', 'Bulk Upload')], default='direct', max_length=50)),
('device_platform', models.CharField(blank=True, choices=[('ios', 'iOS'), ('android', 'Android'), ('web', 'Web')], max_length=20, null=True)),
('capture_timestamp', models.DateTimeField(blank=True, null=True)),
('upload_timestamp', models.DateTimeField(auto_now_add=True)),
('device_info', models.TextField(blank=True, null=True)),
('processing_status', models.CharField(choices=[('uploaded', 'Uploaded'), ('processing', 'Processing'), ('processed', 'Processed'), ('failed', 'Failed')], default='uploaded', max_length=20)),
('thumbnail_url', models.URLField(blank=True, null=True)),
('created_at', models.DateTimeField(auto_now_add=True)),
('updated_at', models.DateTimeField(auto_now=True)),
],
options={
'db_table': 'rog_uploaded_images',
},
),
# TeamStartテーブルのentryカラムを安全に削除
migrations.RunSQL(
sql="""
DO $$
BEGIN
IF EXISTS (SELECT FROM information_schema.tables WHERE table_name = 'rog_teamstart') THEN
IF EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_teamstart' AND column_name = 'entry_id') THEN
ALTER TABLE rog_teamstart DROP COLUMN entry_id;
END IF;
END IF;
END $$;
""",
reverse_sql="-- No reverse SQL needed for conditional operation"
),
# Entryテーブルに新しいフィールドを条件付きで追加
migrations.RunSQL(
sql="""
DO $$
BEGIN
-- can_access_private_events フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_entry' AND column_name = 'can_access_private_events') THEN
ALTER TABLE rog_entry ADD COLUMN can_access_private_events BOOLEAN DEFAULT FALSE;
COMMENT ON COLUMN rog_entry.can_access_private_events IS '非公開イベント参加権限';
END IF;
-- staff_privileges フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_entry' AND column_name = 'staff_privileges') THEN
ALTER TABLE rog_entry ADD COLUMN staff_privileges BOOLEAN DEFAULT FALSE;
COMMENT ON COLUMN rog_entry.staff_privileges IS 'スタッフ権限フラグ';
END IF;
-- team_validation_status フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_entry' AND column_name = 'team_validation_status') THEN
ALTER TABLE rog_entry ADD COLUMN team_validation_status VARCHAR(20) DEFAULT 'approved';
COMMENT ON COLUMN rog_entry.team_validation_status IS 'チーム承認状況';
END IF;
-- zekken_label フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_entry' AND column_name = 'zekken_label') THEN
ALTER TABLE rog_entry ADD COLUMN zekken_label VARCHAR(255);
END IF;
END $$;
""",
reverse_sql="-- No reverse SQL needed for conditional operation"
),
# GPSCheckinテーブルに新しいフィールドを条件付きで追加
migrations.RunSQL(
sql="""
DO $$
BEGIN
-- create_at フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_gpscheckin' AND column_name = 'create_at') THEN
ALTER TABLE rog_gpscheckin ADD COLUMN create_at TIMESTAMP WITH TIME ZONE;
COMMENT ON COLUMN rog_gpscheckin.create_at IS '作成日時:データの作成日時';
END IF;
-- create_user フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_gpscheckin' AND column_name = 'create_user') THEN
ALTER TABLE rog_gpscheckin ADD COLUMN create_user TEXT;
COMMENT ON COLUMN rog_gpscheckin.create_user IS '作成ユーザー';
END IF;
-- event_id フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_gpscheckin' AND column_name = 'event_id') THEN
ALTER TABLE rog_gpscheckin ADD COLUMN event_id INTEGER;
COMMENT ON COLUMN rog_gpscheckin.event_id IS 'イベントID';
END IF;
-- goal_time フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_gpscheckin' AND column_name = 'goal_time') THEN
ALTER TABLE rog_gpscheckin ADD COLUMN goal_time TEXT;
COMMENT ON COLUMN rog_gpscheckin.goal_time IS 'ゴール時刻=ゴール時のみ使用される。画像から時刻を読み取り設定する。';
END IF;
-- image_address フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_gpscheckin' AND column_name = 'image_address') THEN
ALTER TABLE rog_gpscheckin ADD COLUMN image_address TEXT;
COMMENT ON COLUMN rog_gpscheckin.image_address IS 'チェックイン画像のパス';
END IF;
END $$;
""",
reverse_sql="-- No reverse SQL needed for conditional operation"
),
# NewEvent2テーブルに新しいフィールドを条件付きで追加
migrations.RunSQL(
sql="""
DO $$
BEGIN
-- status フィールドの追加
IF NOT EXISTS (SELECT FROM information_schema.columns WHERE table_name = 'rog_newevent2' AND column_name = 'status') THEN
ALTER TABLE rog_newevent2 ADD COLUMN status VARCHAR(20) DEFAULT 'draft';
COMMENT ON COLUMN rog_newevent2.status IS 'イベントステータス';
END IF;
END $$;
""",
reverse_sql="-- No reverse SQL needed for conditional operation"
),
migrations.AlterModelTable(
name='gpslog',
table='gps_information',
),
migrations.DeleteModel(
name='TeamGoal',
),
migrations.DeleteModel(
name='TeamStart',
),
migrations.AddField(
model_name='uploadedimage',
name='entry',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='rog.entry'),
),
migrations.AddField(
model_name='uploadedimage',
name='gpslog',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='rog.gpscheckin'),
),
migrations.AddField(
model_name='checkinextended',
name='gpslog',
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='extended_info', to='rog.gpscheckin'),
),
migrations.AddField(
model_name='checkinextended',
name='validated_by',
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to=settings.AUTH_USER_MODEL),
),
migrations.AddIndex(
model_name='appversion',
index=models.Index(fields=['platform'], name='idx_app_versions_platform'),
),
migrations.AddIndex(
model_name='appversion',
index=models.Index(condition=models.Q(('is_latest', True)), fields=['is_latest'], name='idx_app_versions_latest_true'),
),
migrations.AlterUniqueTogether(
name='appversion',
unique_together={('version', 'platform')},
),
migrations.AddIndex(
model_name='uploadedimage',
index=models.Index(fields=['event_code', 'team_name'], name='idx_uploaded_event_team'),
),
migrations.AddIndex(
model_name='uploadedimage',
index=models.Index(fields=['cp_number'], name='idx_uploaded_cp_number'),
),
migrations.AddIndex(
model_name='uploadedimage',
index=models.Index(fields=['upload_timestamp'], name='idx_uploaded_timestamp'),
),
migrations.AddIndex(
model_name='uploadedimage',
index=models.Index(fields=['processing_status'], name='idx_uploaded_status'),
),
migrations.AddIndex(
model_name='checkinextended',
index=models.Index(fields=['validation_status'], name='idx_checkin_ext_valid'),
),
migrations.AddIndex(
model_name='checkinextended',
index=models.Index(fields=['created_at'], name='idx_checkin_ext_created'),
),
]