Fixed #24550 -- Added migration operation description to sqlmigrate output

Thanks Tim Graham for the review.
This commit is contained in:
Markus Holtermann 2015-03-29 16:59:35 +02:00
parent dc27f3ee0c
commit c5cc332bf2
6 changed files with 83 additions and 21 deletions

View File

@ -91,13 +91,16 @@ class Migration(object):
for operation in self.operations: for operation in self.operations:
# If this operation cannot be represented as SQL, place a comment # If this operation cannot be represented as SQL, place a comment
# there instead # there instead
if collect_sql and not operation.reduces_to_sql: if collect_sql:
schema_editor.collected_sql.append("--") schema_editor.collected_sql.append("--")
schema_editor.collected_sql.append("-- MIGRATION NOW PERFORMS OPERATION THAT CANNOT BE " if not operation.reduces_to_sql:
"WRITTEN AS SQL:") schema_editor.collected_sql.append(
"-- MIGRATION NOW PERFORMS OPERATION THAT CANNOT BE WRITTEN AS SQL:"
)
schema_editor.collected_sql.append("-- %s" % operation.describe()) schema_editor.collected_sql.append("-- %s" % operation.describe())
schema_editor.collected_sql.append("--") schema_editor.collected_sql.append("--")
continue if not operation.reduces_to_sql:
continue
# Save the state before the operation has run # Save the state before the operation has run
old_state = project_state.clone() old_state = project_state.clone()
operation.state_forwards(self.app_label, project_state) operation.state_forwards(self.app_label, project_state)
@ -142,12 +145,14 @@ class Migration(object):
# Phase 2 # Phase 2
for operation, to_state, from_state in to_run: for operation, to_state, from_state in to_run:
if collect_sql: if collect_sql:
schema_editor.collected_sql.append("--")
if not operation.reduces_to_sql:
schema_editor.collected_sql.append(
"-- MIGRATION NOW PERFORMS OPERATION THAT CANNOT BE WRITTEN AS SQL:"
)
schema_editor.collected_sql.append("-- %s" % operation.describe())
schema_editor.collected_sql.append("--")
if not operation.reduces_to_sql: if not operation.reduces_to_sql:
schema_editor.collected_sql.append("--")
schema_editor.collected_sql.append("-- MIGRATION NOW PERFORMS OPERATION THAT CANNOT BE "
"WRITTEN AS SQL:")
schema_editor.collected_sql.append("-- %s" % operation.describe())
schema_editor.collected_sql.append("--")
continue continue
if not schema_editor.connection.features.can_rollback_ddl and operation.atomic: if not schema_editor.connection.features.can_rollback_ddl and operation.atomic:
# We're forcing a transaction on a non-transactional-DDL backend # We're forcing a transaction on a non-transactional-DDL backend

View File

@ -446,8 +446,8 @@ You should see something similar to the following:
Migrations for 'polls': Migrations for 'polls':
0001_initial.py: 0001_initial.py:
- Create model Question
- Create model Choice - Create model Choice
- Create model Question
- Add field question to choice - Add field question to choice
By running ``makemigrations``, you're telling Django that you've made By running ``makemigrations``, you're telling Django that you've made
@ -476,16 +476,25 @@ readability):
.. code-block:: sql .. code-block:: sql
BEGIN; BEGIN;
--
-- Create model Choice
--
CREATE TABLE "polls_choice" ( CREATE TABLE "polls_choice" (
"id" serial NOT NULL PRIMARY KEY, "id" serial NOT NULL PRIMARY KEY,
"choice_text" varchar(200) NOT NULL, "choice_text" varchar(200) NOT NULL,
"votes" integer NOT NULL "votes" integer NOT NULL
); );
--
-- Create model Question
--
CREATE TABLE "polls_question" ( CREATE TABLE "polls_question" (
"id" serial NOT NULL PRIMARY KEY, "id" serial NOT NULL PRIMARY KEY,
"question_text" varchar(200) NOT NULL, "question_text" varchar(200) NOT NULL,
"pub_date" timestamp with time zone NOT NULL "pub_date" timestamp with time zone NOT NULL
); );
--
-- Add field question to choice
--
ALTER TABLE "polls_choice" ADD COLUMN "question_id" integer NOT NULL; ALTER TABLE "polls_choice" ADD COLUMN "question_id" integer NOT NULL;
ALTER TABLE "polls_choice" ALTER COLUMN "question_id" DROP DEFAULT; ALTER TABLE "polls_choice" ALTER COLUMN "question_id" DROP DEFAULT;
CREATE INDEX "polls_choice_7aa0f6ee" ON "polls_choice" ("question_id"); CREATE INDEX "polls_choice_7aa0f6ee" ON "polls_choice" ("question_id");

View File

@ -286,6 +286,9 @@ This command should produce the following output:
.. code-block:: sql .. code-block:: sql
BEGIN; BEGIN;
--
-- Create model WorldBorder
--
CREATE TABLE "world_worldborder" ( CREATE TABLE "world_worldborder" (
"id" serial NOT NULL PRIMARY KEY, "id" serial NOT NULL PRIMARY KEY,
"name" varchar(50) NOT NULL, "name" varchar(50) NOT NULL,

View File

@ -984,6 +984,12 @@ By default, the SQL created is for running the migration in the forwards
direction. Pass ``--backwards`` to generate the SQL for direction. Pass ``--backwards`` to generate the SQL for
unapplying the migration instead. unapplying the migration instead.
.. versionchanged:: 1.9
To increase the readability of the overall SQL output the SQL code
generated for each migration operation is preceded by the operation's
description.
sqlsequencereset <app_label app_label ...> sqlsequencereset <app_label app_label ...>
------------------------------------------ ------------------------------------------

View File

@ -152,6 +152,10 @@ Management Commands
* The new :djadmin:`sendtestemail` command lets you send a test email to * The new :djadmin:`sendtestemail` command lets you send a test email to
easily confirm that email sending through Django is working. easily confirm that email sending through Django is working.
* To increase the readability of the SQL code generated by
:djadmin:`sqlmigrate`, the SQL code generated for each migration operation is
preceded by the operation's description.
Models Models
^^^^^^ ^^^^^^

View File

@ -289,29 +289,64 @@ class MigrateTests(MigrationTestBase):
) )
@override_settings(MIGRATION_MODULES={"migrations": "migrations.test_migrations"}) @override_settings(MIGRATION_MODULES={"migrations": "migrations.test_migrations"})
def test_sqlmigrate(self): def test_sqlmigrate_forwards(self):
""" """
Makes sure that sqlmigrate does something. Makes sure that sqlmigrate does something.
""" """
# Make sure the output is wrapped in a transaction
out = six.StringIO() out = six.StringIO()
call_command("sqlmigrate", "migrations", "0001", stdout=out) call_command("sqlmigrate", "migrations", "0001", stdout=out)
output = out.getvalue() output = out.getvalue().lower()
self.assertIn(connection.ops.start_transaction_sql(), output)
self.assertIn(connection.ops.end_transaction_sql(), output)
# Test forwards. All the databases agree on CREATE TABLE, at least. index_tx_start = output.find(connection.ops.start_transaction_sql().lower())
out = six.StringIO() index_op_desc_author = output.find('-- create model author')
call_command("sqlmigrate", "migrations", "0001", stdout=out) index_create_table = output.find('create table')
self.assertIn("create table", out.getvalue().lower()) index_op_desc_tribble = output.find('-- create model tribble')
index_op_desc_unique_together = output.find('-- alter unique_together')
index_tx_end = output.find(connection.ops.end_transaction_sql().lower())
self.assertGreater(index_tx_start, -1, "Transaction start not found")
self.assertGreater(index_op_desc_author, index_tx_start,
"Operation description (author) not found or found before transaction start")
self.assertGreater(index_create_table, index_op_desc_author,
"CREATE TABLE not found or found before operation description (author)")
self.assertGreater(index_op_desc_tribble, index_create_table,
"Operation description (tribble) not found or found before CREATE TABLE (author)")
self.assertGreater(index_op_desc_unique_together, index_op_desc_tribble,
"Operation description (unique_together) not found or found before operation description (tribble)")
self.assertGreater(index_tx_end, index_op_desc_unique_together,
"Transaction end not found or found before operation description (unique_together)")
@override_settings(MIGRATION_MODULES={"migrations": "migrations.test_migrations"})
def test_sqlmigrate_backwards(self):
"""
Makes sure that sqlmigrate does something.
"""
# Cannot generate the reverse SQL unless we've applied the migration. # Cannot generate the reverse SQL unless we've applied the migration.
call_command("migrate", "migrations", verbosity=0) call_command("migrate", "migrations", verbosity=0)
# And backwards is a DROP TABLE
out = six.StringIO() out = six.StringIO()
call_command("sqlmigrate", "migrations", "0001", stdout=out, backwards=True) call_command("sqlmigrate", "migrations", "0001", stdout=out, backwards=True)
self.assertIn("drop table", out.getvalue().lower()) output = out.getvalue().lower()
index_tx_start = output.find(connection.ops.start_transaction_sql().lower())
index_op_desc_unique_together = output.find('-- alter unique_together')
index_op_desc_tribble = output.find('-- create model tribble')
index_op_desc_author = output.find('-- create model author')
index_drop_table = output.rfind('drop table')
index_tx_end = output.find(connection.ops.end_transaction_sql().lower())
self.assertGreater(index_tx_start, -1, "Transaction start not found")
self.assertGreater(index_op_desc_unique_together, index_tx_start,
"Operation description (unique_together) not found or found before transaction start")
self.assertGreater(index_op_desc_tribble, index_op_desc_unique_together,
"Operation description (tribble) not found or found before operation description (unique_together)")
self.assertGreater(index_op_desc_author, index_op_desc_tribble,
"Operation description (author) not found or found before operation description (tribble)")
self.assertGreater(index_drop_table, index_op_desc_author,
"DROP TABLE not found or found before operation description (author)")
self.assertGreater(index_tx_end, index_op_desc_unique_together,
"Transaction end not found or found before DROP TABLE")
# Cleanup by unmigrating everything # Cleanup by unmigrating everything
call_command("migrate", "migrations", "zero", verbosity=0) call_command("migrate", "migrations", "zero", verbosity=0)