292 lines
16 KiB
Python
292 lines
16 KiB
Python
|
|
#!/usr/bin/env python
|
|||
|
|
"""
|
|||
|
|
Migration ファイルをシンプルにリセットするスクリプト
|
|||
|
|
|
|||
|
|
使用方法:
|
|||
|
|
1. 現在のmigrationテーブルの状態をバックアップ
|
|||
|
|
2. migrationファイルを削除
|
|||
|
|
3. 新しいシンプルなmigrationファイルを作成
|
|||
|
|
4. データベースのmigration履歴をリセット
|
|||
|
|
"""
|
|||
|
|
|
|||
|
|
import os
|
|||
|
|
import shutil
|
|||
|
|
from datetime import datetime
|
|||
|
|
|
|||
|
|
def backup_migrations():
|
|||
|
|
"""現在のmigrationファイルをバックアップ"""
|
|||
|
|
migrations_dir = "rog/migrations"
|
|||
|
|
backup_dir = f"rog/migrations_backup_{datetime.now().strftime('%Y%m%d_%H%M%S')}"
|
|||
|
|
|
|||
|
|
if os.path.exists(migrations_dir):
|
|||
|
|
shutil.copytree(migrations_dir, backup_dir)
|
|||
|
|
print(f"Migrationファイルを {backup_dir} にバックアップしました")
|
|||
|
|
|
|||
|
|
def reset_migrations():
|
|||
|
|
"""migrationファイルをリセット"""
|
|||
|
|
migrations_dir = "rog/migrations"
|
|||
|
|
|
|||
|
|
# __init__.py以外のファイルを削除
|
|||
|
|
if os.path.exists(migrations_dir):
|
|||
|
|
for file in os.listdir(migrations_dir):
|
|||
|
|
if file != "__init__.py" and file.endswith(".py"):
|
|||
|
|
os.remove(os.path.join(migrations_dir, file))
|
|||
|
|
print(f"削除: {file}")
|
|||
|
|
|
|||
|
|
def create_simple_initial_migration():
|
|||
|
|
"""シンプルな初期migrationファイルを作成"""
|
|||
|
|
migration_content = '''# -*- coding: utf-8 -*-
|
|||
|
|
# Generated by reset_migrations_simple.py
|
|||
|
|
|
|||
|
|
from django.conf import settings
|
|||
|
|
from django.db import migrations, models
|
|||
|
|
import django.contrib.gis.db.models.fields
|
|||
|
|
import django.contrib.postgres.indexes
|
|||
|
|
import django.db.models.deletion
|
|||
|
|
import django.utils.timezone
|
|||
|
|
import rog.models
|
|||
|
|
|
|||
|
|
|
|||
|
|
class Migration(migrations.Migration):
|
|||
|
|
|
|||
|
|
initial = True
|
|||
|
|
|
|||
|
|
dependencies = [
|
|||
|
|
('auth', '0012_alter_user_first_name_max_length'),
|
|||
|
|
]
|
|||
|
|
|
|||
|
|
operations = [
|
|||
|
|
# 基本的なモデルのみ作成(順番に依存関係を考慮)
|
|||
|
|
|
|||
|
|
# 1. 地理情報テーブル(managed=False)
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='GifuAreas',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('geom', django.contrib.gis.db.models.fields.MultiPolygonField(blank=True, null=True, srid=4326)),
|
|||
|
|
('adm0_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('area_nm', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'gifu_areas',
|
|||
|
|
'managed': False,
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='JpnAdminMainPerf',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('geom', django.contrib.gis.db.models.fields.MultiPolygonField(blank=True, null=True, srid=4326)),
|
|||
|
|
('adm0_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'jpn_admin_main_perf',
|
|||
|
|
'managed': False,
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='JpnSubPerf',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('geom', django.contrib.gis.db.models.fields.MultiPolygonField(blank=True, null=True, srid=4326)),
|
|||
|
|
('adm0_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm0_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm1_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_ja', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_en', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('adm2_pcode', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('name_modified', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('area_name', models.CharField(blank=True, max_length=254, null=True)),
|
|||
|
|
('list_order', models.IntegerField(default=0)),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'jpn_sub_perf',
|
|||
|
|
'managed': False,
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 2. ユーザー関連モデル
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='CustomUser',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('password', models.CharField(max_length=128, verbose_name='password')),
|
|||
|
|
('last_login', models.DateTimeField(blank=True, null=True, verbose_name='last login')),
|
|||
|
|
('is_superuser', models.BooleanField(default=False, help_text='Designates that this user has all permissions without explicitly assigning them.', verbose_name='superuser status')),
|
|||
|
|
('email', models.EmailField(max_length=254, unique=True)),
|
|||
|
|
('firstname', models.CharField(blank=True, max_length=255, null=True)),
|
|||
|
|
('lastname', models.CharField(blank=True, max_length=255, null=True)),
|
|||
|
|
('date_of_birth', models.DateField(blank=True, null=True)),
|
|||
|
|
('female', models.BooleanField(default=False)),
|
|||
|
|
('group', models.CharField(blank=True, max_length=255)),
|
|||
|
|
('is_active', models.BooleanField(default=True)),
|
|||
|
|
('is_staff', models.BooleanField(default=False)),
|
|||
|
|
('date_joined', models.DateTimeField(default=django.utils.timezone.now)),
|
|||
|
|
('is_rogaining', models.BooleanField(default=False)),
|
|||
|
|
('groups', models.ManyToManyField(blank=True, help_text='The groups this user belongs to. A user will get all permissions granted to each of their groups.', related_name='user_set', related_query_name='user', to='auth.Group', verbose_name='groups')),
|
|||
|
|
('user_permissions', models.ManyToManyField(blank=True, help_text='Specific permissions for this user.', related_name='user_set', related_query_name='user', to='auth.Permission', verbose_name='user permissions')),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'abstract': False,
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 3. カテゴリモデル
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='Category',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('category_name', models.CharField(max_length=255, verbose_name='カテゴリ名')),
|
|||
|
|
('parent_category', models.CharField(blank=True, max_length=255, null=True)),
|
|||
|
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
|||
|
|
('last_updated_at', models.DateTimeField(auto_now=True)),
|
|||
|
|
],
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 4. イベント関連モデル
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='NewEvent',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('event_code', models.CharField(max_length=255, verbose_name='イベントコード')),
|
|||
|
|
('event_name', models.CharField(max_length=255, verbose_name='イベント名')),
|
|||
|
|
('event_date', models.DateField(verbose_name='イベント日')),
|
|||
|
|
('start_time', models.TimeField(blank=True, null=True, verbose_name='開始時刻')),
|
|||
|
|
('end_time', models.TimeField(blank=True, null=True, verbose_name='終了時刻')),
|
|||
|
|
('description', models.TextField(blank=True, null=True, verbose_name='説明')),
|
|||
|
|
('is_active', models.BooleanField(default=True)),
|
|||
|
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
|||
|
|
('last_updated_at', models.DateTimeField(auto_now=True)),
|
|||
|
|
('category', models.ForeignKey(default=1, on_delete=django.db.models.deletion.DO_NOTHING, to='rog.category')),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'rog_newevent',
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 5. チーム関連モデル
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='Team',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('team_name', models.CharField(max_length=255, verbose_name='チーム名')),
|
|||
|
|
('team_member_num', models.IntegerField(blank=True, default=1, null=True, verbose_name='チーム人数')),
|
|||
|
|
('score', models.IntegerField(blank=True, default=0, null=True, verbose_name='スコア')),
|
|||
|
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
|||
|
|
('last_updated_at', models.DateTimeField(auto_now=True)),
|
|||
|
|
('event', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='rog.newevent')),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'rog_team',
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 6. ロケーション関連モデル(基本的なもの)
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='Location',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('location_id', models.IntegerField(blank=True, db_index=True, null=True, verbose_name='Location id')),
|
|||
|
|
('sub_loc_id', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Sub location id')),
|
|||
|
|
('cp', models.FloatField(blank=False, default=0, null=True, verbose_name='Check Point')),
|
|||
|
|
('location_name', models.CharField(default='--- 場所をお願いします --', max_length=2048, verbose_name='Location Name')),
|
|||
|
|
('category', models.CharField(blank=True, db_index=True, max_length=2048, null=True, verbose_name='Category')),
|
|||
|
|
('subcategory', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Sub Category')),
|
|||
|
|
('zip', models.CharField(blank=True, max_length=12, null=True, verbose_name='Zip code')),
|
|||
|
|
('address', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Address')),
|
|||
|
|
('prefecture', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Prefecture')),
|
|||
|
|
('area', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Area')),
|
|||
|
|
('city', models.CharField(blank=True, max_length=2048, null=True, verbose_name='City')),
|
|||
|
|
('latitude', models.FloatField(blank=True, null=True, verbose_name='Latitude')),
|
|||
|
|
('longitude', models.FloatField(blank=True, null=True, verbose_name='Latitude')),
|
|||
|
|
('photos', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Photos')),
|
|||
|
|
('videos', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Videos')),
|
|||
|
|
('webcontents', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Web Content')),
|
|||
|
|
('status', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Status')),
|
|||
|
|
('portal', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Portal')),
|
|||
|
|
('group', models.CharField(blank=True, db_index=True, max_length=2048, null=True, verbose_name='Group')),
|
|||
|
|
('phone', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Phone')),
|
|||
|
|
('fax', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Fax')),
|
|||
|
|
('email', models.EmailField(blank=True, max_length=2048, null=True, verbose_name='Email')),
|
|||
|
|
('facility', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Facility')),
|
|||
|
|
('remark', models.CharField(blank=True, max_length=2048, null=True, verbose_name='Remarks')),
|
|||
|
|
('tags', models.CharField(blank=True, max_length=512, null=True, verbose_name='Tags')),
|
|||
|
|
('parammeters', models.CharField(blank=True, max_length=512, null=True, verbose_name='Parameters')),
|
|||
|
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
|||
|
|
('last_updated_at', models.DateTimeField(auto_now=True)),
|
|||
|
|
('last_updated_user', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='location_updated_user', to=settings.AUTH_USER_MODEL)),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'rog_location',
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# 7. エントリー関連モデル
|
|||
|
|
migrations.CreateModel(
|
|||
|
|
name='Entry',
|
|||
|
|
fields=[
|
|||
|
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|||
|
|
('start_time', models.DateTimeField(blank=True, null=True, verbose_name='Start time')),
|
|||
|
|
('goal_time', models.DateTimeField(blank=True, null=True, verbose_name='Goal time')),
|
|||
|
|
('check_point', models.IntegerField(blank=True, null=True, verbose_name='Check Point')),
|
|||
|
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
|||
|
|
('last_updated_at', models.DateTimeField(auto_now=True)),
|
|||
|
|
('location', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='rog.location')),
|
|||
|
|
('team', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='rog.team')),
|
|||
|
|
('last_updated_user', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.DO_NOTHING, related_name='entry_updated_user', to=settings.AUTH_USER_MODEL)),
|
|||
|
|
],
|
|||
|
|
options={
|
|||
|
|
'db_table': 'rog_entry',
|
|||
|
|
},
|
|||
|
|
),
|
|||
|
|
|
|||
|
|
# インデックスの追加
|
|||
|
|
migrations.AddIndex(
|
|||
|
|
model_name='gifu_areas',
|
|||
|
|
index=models.Index(fields=['geom'], name='gifu_areas_geom_idx'),
|
|||
|
|
),
|
|||
|
|
]
|
|||
|
|
'''
|
|||
|
|
|
|||
|
|
with open("rog/migrations/0001_simple_initial.py", "w", encoding="utf-8") as f:
|
|||
|
|
f.write(migration_content)
|
|||
|
|
|
|||
|
|
print("シンプルな初期migrationファイル 0001_simple_initial.py を作成しました")
|
|||
|
|
|
|||
|
|
def main():
|
|||
|
|
print("=== Migration リセットスクリプト ===")
|
|||
|
|
print("1. Migrationファイルをバックアップします")
|
|||
|
|
backup_migrations()
|
|||
|
|
|
|||
|
|
print("\\n2. 既存のmigrationファイルを削除します")
|
|||
|
|
reset_migrations()
|
|||
|
|
|
|||
|
|
print("\\n3. シンプルな初期migrationファイルを作成します")
|
|||
|
|
create_simple_initial_migration()
|
|||
|
|
|
|||
|
|
print("\\n=== 完了 ===")
|
|||
|
|
print("次の手順:")
|
|||
|
|
print("1. docker compose exec app python manage.py migrate --fake-initial")
|
|||
|
|
print("2. 必要に応じて追加のmigrationファイルを作成")
|
|||
|
|
|
|||
|
|
if __name__ == "__main__":
|
|||
|
|
main()
|