新しいDjangoプロジェクトのテーブルをセットアップしようとしています(つまり、テーブルはデータベースにまだ存在していません)。 Djangoバージョンは1.7で、dbバックエンドはPostgreSQLです。プロジェクトの名前はcrudです。移行試行の結果は次のとおりです。
python manage.py makemigrations crud
Migrations for 'crud':
0001_initial.py:
- Create model AddressPoint
- Create model CrudPermission
- Create model CrudUser
- Create model LDAPGroup
- Create model LogEntry
- Add field ldap_groups to cruduser
- Alter unique_together for crudpermission (1 constraint(s))
python manage.py migrate crud
Operations to perform:
Apply all migrations: crud
Running migrations:
Applying crud.0001_initial...Traceback (most recent call last):
File "manage.py", line 18, in <module>
execute_from_command_line(sys.argv)
File "/usr/local/lib/python2.7/dist-packages/Django/core/management/__init__.py", line 385, in execute_from_command_line
utility.execute()
File "/usr/local/lib/python2.7/dist-packages/Django/core/management/__init__.py", line 377, in execute
self.fetch_command(subcommand).run_from_argv(self.argv)
File "/usr/local/lib/python2.7/dist-packages/Django/core/management/base.py", line 288, in run_from_argv
self.execute(*args, **options.__dict__)
File "/usr/local/lib/python2.7/dist-packages/Django/core/management/base.py", line 338, in execute
output = self.handle(*args, **options)
File "/usr/local/lib/python2.7/dist-packages/Django/core/management/commands/migrate.py", line 161, in handle
executor.migrate(targets, plan, fake=options.get("fake", False))
File "/usr/local/lib/python2.7/dist-packages/Django/db/migrations/executor.py", line 68, in migrate
self.apply_migration(migration, fake=fake)
File "/usr/local/lib/python2.7/dist-packages/Django/db/migrations/executor.py", line 102, in apply_migration
migration.apply(project_state, schema_editor)
File "/usr/local/lib/python2.7/dist-packages/Django/db/migrations/migration.py", line 108, in apply
operation.database_forwards(self.app_label, schema_editor, project_state, new_state)
File "/usr/local/lib/python2.7/dist-packages/Django/db/migrations/operations/models.py", line 36, in database_forwards
schema_editor.create_model(model)
File "/usr/local/lib/python2.7/dist-packages/Django/db/backends/schema.py", line 262, in create_model
self.execute(sql, params)
File "/usr/local/lib/python2.7/dist-packages/Django/db/backends/schema.py", line 103, in execute
cursor.execute(sql, params)
File "/usr/local/lib/python2.7/dist-packages/Django/db/backends/utils.py", line 82, in execute
return super(CursorDebugWrapper, self).execute(sql, params)
File "/usr/local/lib/python2.7/dist-packages/Django/db/backends/utils.py", line 66, in execute
return self.cursor.execute(sql, params)
File "/usr/local/lib/python2.7/dist-packages/Django/db/utils.py", line 94, in __exit__
six.reraise(dj_exc_type, dj_exc_value, traceback)
File "/usr/local/lib/python2.7/dist-packages/Django/db/backends/utils.py", line 66, in execute
return self.cursor.execute(sql, params)
Django.db.utils.ProgrammingError: relation "crud_crudpermission" already exists
移行ファイルからのいくつかのハイライト:
dependencies = [
('auth', '0001_initial'),
('contenttypes', '0001_initial'),
]
migrations.CreateModel(
name='CrudPermission',
fields=[
('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)),
('_created_by', models.CharField(default=b'', max_length=64, null=True, editable=False, blank=True)),
('_last_updated_by', models.CharField(default=b'', max_length=64, null=True, editable=False, blank=True)),
('_created', models.DateTimeField(null=True, editable=False, blank=True)),
('_last_updated', models.DateTimeField(null=True, editable=False, blank=True)),
('domain', models.CharField(max_length=32, choices=[(b'town', b'Town'), (b'boe', b'BOE'), (b'police', b'Police')])),
('ldap_group', models.CharField(max_length=128, verbose_name=b'LDAP group')),
('can_add', models.BooleanField(default=False, verbose_name=b'add')),
('can_change', models.BooleanField(default=False, verbose_name=b'change')),
('restrict_change_to_own', models.BooleanField(default=False)),
('can_delete', models.BooleanField(default=False, verbose_name=b'delete')),
('restrict_delete_to_own', models.BooleanField(default=False)),
('models', models.ManyToManyField(to='contenttypes.ContentType', null=True, blank=True)),
],
options={
'verbose_name': 'CRUD permission',
},
bases=(models.Model,),
),
migrations.AlterUniqueTogether(
name='crudpermission',
unique_together=set([('ldap_group', 'can_add', 'can_change', 'can_delete', 'domain')]),
)
、
Crudアプリは実際に何もするものではありませんが、別のアプリを使用するため、そのアプリから移行しようとすると、上記の問題が発生します。
同様の問題を持つ人々のウェブ上で他の例を見つけましたが、彼らのケースはどれも当てはまらないようです。
根本的な問題を見つけるために次にどこを見るべきですか?
最初にpython manage.py migrate --fake
を実行します。
読む https://docs.djangoproject.com/en/1.9/ref/Django-admin/#Django-admin-migrate
同様の問題に直面して、最終的に移行フォルダー内のすべての.pyファイルを削除し(Django 1.7が自動的に作成します)、その後完全に機能しました。
既存のモデルにいくつかの新しいフィールドを追加すると、同様の問題に直面しました。私はDjango 1.9を使用していますが、これは 導入済み--run-syncdb
オプションです。 manage.py migrate --run-syncdb
を実行すると、テーブルが修正されました。
列名を変更した同様の問題に直面していました。私は彼の質問で提供されたスタックトレースで言及されたのと同じエラーを受け取っていました。
これが私がしたことです。
最初に偽の移行を実行しました。次に、(実行したい移行)エントリをDjango_migrationsテーブルから削除し、移行を再度実行しました(今回は偽物ではありません)。
予想どおり変更が表示されました。
これが役立つことを願っています。
Djangoには--fake-initial
オプションが用意されており、これを使用すると効果的です。 Django Migration Documentation から:
-偽の初期
その移行のすべてのCreateModel操作によって作成されたすべてのモデルの名前を持つすべてのデータベーステーブルが既に存在する場合、Djangoがアプリの初期移行をスキップできるようにします。このオプションは、移行の使用が既存のデータベースに対して最初に移行を実行するときに使用することを目的としています。ただし、このオプションは、一致するテーブル名以外の一致するデータベーススキーマをチェックしないため、既存のスキーマが最初の移行で記録されたものと一致する場合にのみ安全に使用できます。
使用するために、バージョン管理からプロジェクトを取得し、いくつかの新しいモデルフィールドを追加する準備をしていました。フィールドを追加し、./manage.py makemigrations
を実行し、./manage.py migrate
を実行しようとしましたが、予想どおり、多くのフィールドが既存のデータベースにすでに存在していたため、エラーがスローされました。
私がすべきだったのは、バージョン管理からプロジェクトをプルした直後にmakemigrations
を実行して、既存のモデルの状態のスナップショットを作成することでした。次に、./manage.py migrate --fake-initial
を実行することが次のステップになります。
その後、追加して、通常どおりmakemigrations
> migrate
を追加できます。
注:--fake-initial
が既存のフィールドをスキップし、add newもの。その時点までに作成した新しいフィールドをコメントアウトすることを選択し、--fake-initial
を実行して、バージョン管理からプルした後の最初の操作であるかのように実行しましたthen次の移行で更新されたフィールドに追加されました。
その他の関連ドキュメント: https://docs.djangoproject.com/en/dev/topics/migrations/#initial-migrations
これで(Django 1.9を使用しています)、次のように作成できます。
./ manage.py [--database DATABASE] --fake [app_label] [migration_name]
このようにして、より正確に問題をターゲットにし、特定のデータベースで問題のある移行のみを偽造できます。
したがって、質問を見て、次のことができます。
./ manage.py --database default --fake crud crud.0001_initial
この問題はweb2pyframework
のmodels/config.py
で見つかりました。
変化する
settings.base.migrate = True
設定ファイルに
settings.base.migrate = False
問題が解決しました。
member
という名前の外部キーフィールドを別のテーブルを指すように変更しているときに、Django 1.10プロジェクトでこのエラーの特定の例を見つけて解決しました。 3つの異なるモデルでこれを変更していましたが、すべてのモデルでこのエラーが発生しました。最初の試行で、member
の名前をmember_user
に変更し、新しいテーブルを指す外部キーとして新しいフィールドmember
を作成しようとしましたが、これは機能しませんでした。
私が見つけたのは、member
列の名前を変更したとき、<app>_<model>_<hash>
形式のインデックス名は変更されず、新しいmember
列を作成しようとしても同じものを作成しようとしたことです。名前のハッシュ部分が同じだったため、インデックス名。
新しいmember_user
関係を一時的に作成し、データをコピーすることで問題を解決しました。これにより、異なるハッシュを持つ新しいインデックスが作成されました。次にmember
を削除し、新しいテーブルとそれと競合するインデックス名を指すように再作成しました。それが完了したら、RunPython
ステップを実行して、新しいmember
列に適切なテーブルへの参照を追加しました。 RemoveField
マイグレーションを追加して、一時的なmember_user
列をクリーンアップしました。
このエラーを受け取ったため、移行を2つのファイルに分割する必要がありました。
psycopg2.OperationalError:保留中のトリガーイベントがあるため、TABLE "<table_name>"を変更できません
データを作成してmember_user
にコピーした後、同じ移行トランザクションでmember
を削除できませんでした。これはpostgres固有の制限かもしれませんが、別のトランザクションを作成し、member_user
を作成して2番目の移行にコピーした後にすべてを移動することで簡単に解決しました。