123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105 |
- from __future__ import unicode_literals
- import sys
- from importlib import import_module
- from django.apps import apps
- from django.db import connections, router, transaction, DEFAULT_DB_ALIAS
- from django.core.management import call_command
- from django.core.management.base import BaseCommand, CommandError
- from django.core.management.color import no_style
- from django.core.management.sql import sql_flush, emit_post_migrate_signal
- from django.utils.six.moves import input
- from django.utils import six
- class Command(BaseCommand):
- help = ('Removes ALL DATA from the database, including data added during '
- 'migrations. Unmigrated apps will also have their initial_data '
- 'fixture reloaded. Does not achieve a "fresh install" state.')
- def add_arguments(self, parser):
- parser.add_argument('--noinput', action='store_false', dest='interactive', default=True,
- help='Tells Django to NOT prompt the user for input of any kind.')
- parser.add_argument('--database', action='store', dest='database',
- default=DEFAULT_DB_ALIAS,
- help='Nominates a database to flush. Defaults to the "default" database.')
- parser.add_argument('--no-initial-data', action='store_false',
- dest='load_initial_data', default=True,
- help='Tells Django not to load any initial data after database synchronization.')
- def handle(self, **options):
- database = options.get('database')
- connection = connections[database]
- verbosity = options.get('verbosity')
- interactive = options.get('interactive')
- # The following are stealth options used by Django's internals.
- reset_sequences = options.get('reset_sequences', True)
- allow_cascade = options.get('allow_cascade', False)
- inhibit_post_migrate = options.get('inhibit_post_migrate', False)
- self.style = no_style()
- # Import the 'management' module within each installed app, to register
- # dispatcher events.
- for app_config in apps.get_app_configs():
- try:
- import_module('.management', app_config.name)
- except ImportError:
- pass
- sql_list = sql_flush(self.style, connection, only_django=True,
- reset_sequences=reset_sequences,
- allow_cascade=allow_cascade)
- if interactive:
- confirm = input("""You have requested a flush of the database.
- This will IRREVERSIBLY DESTROY all data currently in the %r database,
- and return each table to an empty state.
- Are you sure you want to do this?
- Type 'yes' to continue, or 'no' to cancel: """ % connection.settings_dict['NAME'])
- else:
- confirm = 'yes'
- if confirm == 'yes':
- try:
- with transaction.atomic(using=database,
- savepoint=connection.features.can_rollback_ddl):
- with connection.cursor() as cursor:
- for sql in sql_list:
- cursor.execute(sql)
- except Exception as e:
- new_msg = (
- "Database %s couldn't be flushed. Possible reasons:\n"
- " * The database isn't running or isn't configured correctly.\n"
- " * At least one of the expected database tables doesn't exist.\n"
- " * The SQL was invalid.\n"
- "Hint: Look at the output of 'django-admin sqlflush'. "
- "That's the SQL this command wasn't able to run.\n"
- "The full error: %s") % (connection.settings_dict['NAME'], e)
- six.reraise(CommandError, CommandError(new_msg), sys.exc_info()[2])
- if not inhibit_post_migrate:
- self.emit_post_migrate(verbosity, interactive, database)
- # Reinstall the initial_data fixture.
- if options.get('load_initial_data'):
- # Reinstall the initial_data fixture for apps without migrations.
- from django.db.migrations.executor import MigrationExecutor
- executor = MigrationExecutor(connection)
- app_options = options.copy()
- for app_label in executor.loader.unmigrated_apps:
- app_options['app_label'] = app_label
- call_command('loaddata', 'initial_data', **app_options)
- else:
- self.stdout.write("Flush cancelled.\n")
- @staticmethod
- def emit_post_migrate(verbosity, interactive, database):
- # Emit the post migrate signal. This allows individual applications to
- # respond as if the database had been migrated from scratch.
- all_models = []
- for app_config in apps.get_app_configs():
- all_models.extend(router.get_migratable_models(app_config, database, include_auto_created=True))
- emit_post_migrate_signal(verbosity, interactive, database)
|