2021-01-12 22:47:58 +08:00
|
|
|
import gzip
|
|
|
|
import os
|
2016-01-27 19:15:38 +08:00
|
|
|
import warnings
|
2013-08-03 13:41:15 +08:00
|
|
|
|
2013-12-28 04:25:35 +08:00
|
|
|
from django.apps import apps
|
2007-12-17 14:53:15 +08:00
|
|
|
from django.core import serializers
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.core.management.base import BaseCommand, CommandError
|
2016-05-17 14:52:01 +08:00
|
|
|
from django.core.management.utils import parse_apps_and_model_labels
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.db import DEFAULT_DB_ALIAS, router
|
2007-08-16 14:06:55 +08:00
|
|
|
|
2021-01-12 22:47:58 +08:00
|
|
|
try:
|
|
|
|
import bz2
|
|
|
|
has_bz2 = True
|
|
|
|
except ImportError:
|
|
|
|
has_bz2 = False
|
|
|
|
|
|
|
|
try:
|
|
|
|
import lzma
|
|
|
|
has_lzma = True
|
|
|
|
except ImportError:
|
|
|
|
has_lzma = False
|
|
|
|
|
2007-09-10 05:57:59 +08:00
|
|
|
|
2016-01-27 19:15:38 +08:00
|
|
|
class ProxyModelWarning(Warning):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2007-08-16 14:06:55 +08:00
|
|
|
class Command(BaseCommand):
|
2016-03-29 06:33:29 +08:00
|
|
|
help = (
|
|
|
|
"Output the contents of the database as a fixture of the given format "
|
|
|
|
"(using each model's default manager unless --all is specified)."
|
|
|
|
)
|
2014-06-07 04:39:33 +08:00
|
|
|
|
|
|
|
def add_arguments(self, parser):
|
2016-03-29 06:33:29 +08:00
|
|
|
parser.add_argument(
|
|
|
|
'args', metavar='app_label[.ModelName]', nargs='*',
|
|
|
|
help='Restricts dumped data to the specified app_label or app_label.ModelName.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2018-07-03 05:54:57 +08:00
|
|
|
'--format', default='json',
|
2016-03-29 06:33:29 +08:00
|
|
|
help='Specifies the output serialization format for fixtures.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2018-07-03 05:54:57 +08:00
|
|
|
'--indent', type=int,
|
2016-03-29 06:33:29 +08:00
|
|
|
help='Specifies the indent level to use when pretty-printing output.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2018-07-03 05:54:57 +08:00
|
|
|
'--database',
|
2013-12-28 16:53:02 +08:00
|
|
|
default=DEFAULT_DB_ALIAS,
|
|
|
|
help='Nominates a specific database to dump fixtures from. '
|
2016-03-29 06:33:29 +08:00
|
|
|
'Defaults to the "default" database.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2018-07-03 05:54:57 +08:00
|
|
|
'-e', '--exclude', action='append', default=[],
|
2013-12-28 16:53:02 +08:00
|
|
|
help='An app_label or app_label.ModelName to exclude '
|
2016-03-29 06:33:29 +08:00
|
|
|
'(use multiple --exclude to exclude multiple apps/models).',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2017-04-02 08:03:56 +08:00
|
|
|
'--natural-foreign', action='store_true', dest='use_natural_foreign_keys',
|
2016-03-29 06:33:29 +08:00
|
|
|
help='Use natural foreign keys if they are available.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2017-04-02 08:03:56 +08:00
|
|
|
'--natural-primary', action='store_true', dest='use_natural_primary_keys',
|
2016-03-29 06:33:29 +08:00
|
|
|
help='Use natural primary keys if they are available.',
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2017-04-02 08:03:56 +08:00
|
|
|
'-a', '--all', action='store_true', dest='use_base_manager',
|
2013-12-28 16:53:02 +08:00
|
|
|
help="Use Django's base manager to dump all models stored in the database, "
|
2016-03-29 06:33:29 +08:00
|
|
|
"including those that would otherwise be filtered or modified by a custom manager.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
|
|
|
'--pks', dest='primary_keys',
|
|
|
|
help="Only dump objects with given primary keys. Accepts a comma-separated "
|
|
|
|
"list of keys. This option only works when you specify one model.",
|
|
|
|
)
|
|
|
|
parser.add_argument(
|
2018-07-03 05:54:57 +08:00
|
|
|
'-o', '--output',
|
2016-03-29 06:33:29 +08:00
|
|
|
help='Specifies file to which the output is written.'
|
|
|
|
)
|
2007-08-16 14:06:55 +08:00
|
|
|
|
|
|
|
def handle(self, *app_labels, **options):
|
2016-03-03 10:01:36 +08:00
|
|
|
format = options['format']
|
|
|
|
indent = options['indent']
|
|
|
|
using = options['database']
|
|
|
|
excludes = options['exclude']
|
|
|
|
output = options['output']
|
|
|
|
show_traceback = options['traceback']
|
|
|
|
use_natural_foreign_keys = options['use_natural_foreign_keys']
|
|
|
|
use_natural_primary_keys = options['use_natural_primary_keys']
|
|
|
|
use_base_manager = options['use_base_manager']
|
|
|
|
pks = options['primary_keys']
|
2013-05-19 18:39:14 +08:00
|
|
|
|
|
|
|
if pks:
|
2016-07-16 18:27:25 +08:00
|
|
|
primary_keys = [pk.strip() for pk in pks.split(',')]
|
2013-05-19 18:39:14 +08:00
|
|
|
else:
|
2013-05-30 01:47:49 +08:00
|
|
|
primary_keys = []
|
2008-06-11 22:01:35 +08:00
|
|
|
|
2016-05-17 14:52:01 +08:00
|
|
|
excluded_models, excluded_apps = parse_apps_and_model_labels(excludes)
|
2007-08-16 14:06:55 +08:00
|
|
|
|
2017-11-30 00:54:34 +08:00
|
|
|
if not app_labels:
|
2013-05-19 18:39:14 +08:00
|
|
|
if primary_keys:
|
|
|
|
raise CommandError("You can only use --pks option with one model")
|
2019-02-05 19:22:08 +08:00
|
|
|
app_list = dict.fromkeys(
|
2017-08-23 23:40:17 +08:00
|
|
|
app_config for app_config in apps.get_app_configs()
|
2016-03-29 06:33:29 +08:00
|
|
|
if app_config.models_module is not None and app_config not in excluded_apps
|
|
|
|
)
|
2007-08-16 14:06:55 +08:00
|
|
|
else:
|
2013-05-19 18:39:14 +08:00
|
|
|
if len(app_labels) > 1 and primary_keys:
|
|
|
|
raise CommandError("You can only use --pks option with one model")
|
2019-02-05 19:22:08 +08:00
|
|
|
app_list = {}
|
2009-02-28 13:35:22 +08:00
|
|
|
for label in app_labels:
|
|
|
|
try:
|
|
|
|
app_label, model_label = label.split('.')
|
|
|
|
try:
|
2013-12-28 04:25:35 +08:00
|
|
|
app_config = apps.get_app_config(app_label)
|
2015-05-12 17:02:23 +08:00
|
|
|
except LookupError as e:
|
|
|
|
raise CommandError(str(e))
|
2013-12-28 04:25:35 +08:00
|
|
|
if app_config.models_module is None or app_config in excluded_apps:
|
2010-08-07 00:48:07 +08:00
|
|
|
continue
|
2013-12-28 21:55:54 +08:00
|
|
|
try:
|
2014-01-26 19:57:08 +08:00
|
|
|
model = app_config.get_model(model_label)
|
2013-12-28 21:55:54 +08:00
|
|
|
except LookupError:
|
2009-02-28 13:35:22 +08:00
|
|
|
raise CommandError("Unknown model: %s.%s" % (app_label, model_label))
|
|
|
|
|
2013-12-28 04:25:35 +08:00
|
|
|
app_list_value = app_list.setdefault(app_config, [])
|
2014-02-11 22:59:57 +08:00
|
|
|
|
2019-04-14 15:44:56 +08:00
|
|
|
# We may have previously seen an "all-models" request for
|
2014-02-11 22:59:57 +08:00
|
|
|
# this app (no model qualifier was given). In this case
|
|
|
|
# there is no need adding specific models to the list.
|
2020-09-25 00:37:55 +08:00
|
|
|
if app_list_value is not None and model not in app_list_value:
|
|
|
|
app_list_value.append(model)
|
2009-02-28 13:35:22 +08:00
|
|
|
except ValueError:
|
2013-05-19 18:39:14 +08:00
|
|
|
if primary_keys:
|
|
|
|
raise CommandError("You can only use --pks option with one model")
|
2009-02-28 13:35:22 +08:00
|
|
|
# This is just an app - no model qualifier
|
|
|
|
app_label = label
|
|
|
|
try:
|
2013-12-28 04:25:35 +08:00
|
|
|
app_config = apps.get_app_config(app_label)
|
2015-05-12 17:02:23 +08:00
|
|
|
except LookupError as e:
|
|
|
|
raise CommandError(str(e))
|
2013-12-28 04:25:35 +08:00
|
|
|
if app_config.models_module is None or app_config in excluded_apps:
|
2010-08-07 00:48:07 +08:00
|
|
|
continue
|
2013-12-28 04:25:35 +08:00
|
|
|
app_list[app_config] = None
|
2007-08-16 14:06:55 +08:00
|
|
|
|
|
|
|
# Check that the serialization format exists; this is a shortcut to
|
|
|
|
# avoid collating all the objects and _then_ failing.
|
2007-12-17 19:09:56 +08:00
|
|
|
if format not in serializers.get_public_serializer_formats():
|
2013-09-07 03:24:55 +08:00
|
|
|
try:
|
|
|
|
serializers.get_serializer(format)
|
|
|
|
except serializers.SerializerDoesNotExist:
|
|
|
|
pass
|
2007-12-17 14:53:15 +08:00
|
|
|
|
2007-08-16 14:06:55 +08:00
|
|
|
raise CommandError("Unknown serialization format: %s" % format)
|
|
|
|
|
2015-07-22 05:24:32 +08:00
|
|
|
def get_objects(count_only=False):
|
|
|
|
"""
|
|
|
|
Collate the objects to be serialized. If count_only is True, just
|
|
|
|
count the number of objects to be serialized.
|
|
|
|
"""
|
2019-12-02 07:48:01 +08:00
|
|
|
if use_natural_foreign_keys:
|
|
|
|
models = serializers.sort_dependencies(app_list.items(), allow_cycles=True)
|
|
|
|
else:
|
|
|
|
# There is no need to sort dependencies when natural foreign
|
|
|
|
# keys are not used.
|
|
|
|
models = []
|
|
|
|
for (app_config, model_list) in app_list.items():
|
|
|
|
if model_list is None:
|
|
|
|
models.extend(app_config.get_models())
|
|
|
|
else:
|
|
|
|
models.extend(model_list)
|
2016-01-27 19:15:38 +08:00
|
|
|
for model in models:
|
2012-05-26 17:43:37 +08:00
|
|
|
if model in excluded_models:
|
|
|
|
continue
|
2016-02-06 01:14:32 +08:00
|
|
|
if model._meta.proxy and model._meta.proxy_for_model not in models:
|
2016-01-27 19:15:38 +08:00
|
|
|
warnings.warn(
|
|
|
|
"%s is a proxy model and won't be serialized." % model._meta.label,
|
|
|
|
category=ProxyModelWarning,
|
|
|
|
)
|
2015-02-19 15:27:58 +08:00
|
|
|
if not model._meta.proxy and router.allow_migrate_model(using, model):
|
2012-05-26 17:43:37 +08:00
|
|
|
if use_base_manager:
|
|
|
|
objects = model._base_manager
|
|
|
|
else:
|
|
|
|
objects = model._default_manager
|
2013-05-19 18:39:14 +08:00
|
|
|
|
|
|
|
queryset = objects.using(using).order_by(model._meta.pk.name)
|
|
|
|
if primary_keys:
|
|
|
|
queryset = queryset.filter(pk__in=primary_keys)
|
2015-07-22 05:24:32 +08:00
|
|
|
if count_only:
|
|
|
|
yield queryset.order_by().count()
|
|
|
|
else:
|
2017-02-24 09:06:01 +08:00
|
|
|
yield from queryset.iterator()
|
2009-02-28 13:35:22 +08:00
|
|
|
|
2007-08-16 14:06:55 +08:00
|
|
|
try:
|
2012-05-26 17:43:37 +08:00
|
|
|
self.stdout.ending = None
|
2015-07-22 05:24:32 +08:00
|
|
|
progress_output = None
|
|
|
|
object_count = 0
|
|
|
|
# If dumpdata is outputting to stdout, there is no way to display progress
|
2017-09-14 09:20:29 +08:00
|
|
|
if output and self.stdout.isatty() and options['verbosity'] > 0:
|
2015-07-22 05:24:32 +08:00
|
|
|
progress_output = self.stdout
|
|
|
|
object_count = sum(get_objects(count_only=True))
|
2021-01-12 22:47:58 +08:00
|
|
|
if output:
|
|
|
|
file_root, file_ext = os.path.splitext(output)
|
|
|
|
compression_formats = {
|
|
|
|
'.bz2': (open, {}, file_root),
|
|
|
|
'.gz': (gzip.open, {}, output),
|
|
|
|
'.lzma': (open, {}, file_root),
|
|
|
|
'.xz': (open, {}, file_root),
|
|
|
|
'.zip': (open, {}, file_root),
|
|
|
|
}
|
|
|
|
if has_bz2:
|
|
|
|
compression_formats['.bz2'] = (bz2.open, {}, output)
|
|
|
|
if has_lzma:
|
|
|
|
compression_formats['.lzma'] = (
|
|
|
|
lzma.open, {'format': lzma.FORMAT_ALONE}, output
|
|
|
|
)
|
|
|
|
compression_formats['.xz'] = (lzma.open, {}, output)
|
|
|
|
try:
|
|
|
|
open_method, kwargs, file_path = compression_formats[file_ext]
|
|
|
|
except KeyError:
|
|
|
|
open_method, kwargs, file_path = (open, {}, output)
|
|
|
|
if file_path != output:
|
|
|
|
file_name = os.path.basename(file_path)
|
|
|
|
warnings.warn(
|
|
|
|
f"Unsupported file extension ({file_ext}). "
|
|
|
|
f"Fixtures saved in '{file_name}'.",
|
|
|
|
RuntimeWarning,
|
|
|
|
)
|
|
|
|
stream = open_method(file_path, 'wt', **kwargs)
|
|
|
|
else:
|
|
|
|
stream = None
|
2014-05-26 04:08:05 +08:00
|
|
|
try:
|
2016-03-29 06:33:29 +08:00
|
|
|
serializers.serialize(
|
|
|
|
format, get_objects(), indent=indent,
|
|
|
|
use_natural_foreign_keys=use_natural_foreign_keys,
|
|
|
|
use_natural_primary_keys=use_natural_primary_keys,
|
|
|
|
stream=stream or self.stdout, progress_output=progress_output,
|
|
|
|
object_count=object_count,
|
|
|
|
)
|
2014-05-26 04:08:05 +08:00
|
|
|
finally:
|
|
|
|
if stream:
|
|
|
|
stream.close()
|
2012-04-29 00:09:37 +08:00
|
|
|
except Exception as e:
|
2007-12-17 19:09:32 +08:00
|
|
|
if show_traceback:
|
|
|
|
raise
|
2007-08-16 14:06:55 +08:00
|
|
|
raise CommandError("Unable to serialize database: %s" % e)
|