123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035 |
- import datetime
- import decimal
- import enum
- import functools
- import math
- import os
- import pathlib
- import re
- import sys
- import time
- import uuid
- import zoneinfo
- from types import NoneType
- from unittest import mock
- import custom_migration_operations.more_operations
- import custom_migration_operations.operations
- from django import get_version
- from django.conf import SettingsReference, settings
- from django.core.validators import EmailValidator, RegexValidator
- from django.db import migrations, models
- from django.db.migrations.serializer import BaseSerializer
- from django.db.migrations.writer import MigrationWriter, OperationWriter
- from django.test import SimpleTestCase
- from django.utils.deconstruct import deconstructible
- from django.utils.functional import SimpleLazyObject
- from django.utils.timezone import get_default_timezone, get_fixed_timezone
- from django.utils.translation import gettext_lazy as _
- from .models import FoodManager, FoodQuerySet
- class DeconstructibleInstances:
- def deconstruct(self):
- return ("DeconstructibleInstances", [], {})
- class Money(decimal.Decimal):
- def deconstruct(self):
- return (
- "%s.%s" % (self.__class__.__module__, self.__class__.__name__),
- [str(self)],
- {},
- )
- class TestModel1:
- def upload_to(self):
- return "/somewhere/dynamic/"
- thing = models.FileField(upload_to=upload_to)
- class TextEnum(enum.Enum):
- A = "a-value"
- B = "value-b"
- class TextTranslatedEnum(enum.Enum):
- A = _("a-value")
- B = _("value-b")
- class BinaryEnum(enum.Enum):
- A = b"a-value"
- B = b"value-b"
- class IntEnum(enum.IntEnum):
- A = 1
- B = 2
- class IntFlagEnum(enum.IntFlag):
- A = 1
- B = 2
- class OperationWriterTests(SimpleTestCase):
- def test_empty_signature(self):
- operation = custom_migration_operations.operations.TestOperation()
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.TestOperation(\n),",
- )
- def test_args_signature(self):
- operation = custom_migration_operations.operations.ArgsOperation(1, 2)
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ArgsOperation(\n"
- " arg1=1,\n"
- " arg2=2,\n"
- "),",
- )
- def test_kwargs_signature(self):
- operation = custom_migration_operations.operations.KwargsOperation(kwarg1=1)
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.KwargsOperation(\n"
- " kwarg1=1,\n"
- "),",
- )
- def test_args_kwargs_signature(self):
- operation = custom_migration_operations.operations.ArgsKwargsOperation(
- 1, 2, kwarg2=4
- )
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ArgsKwargsOperation(\n"
- " arg1=1,\n"
- " arg2=2,\n"
- " kwarg2=4,\n"
- "),",
- )
- def test_nested_args_signature(self):
- operation = custom_migration_operations.operations.ArgsOperation(
- custom_migration_operations.operations.ArgsOperation(1, 2),
- custom_migration_operations.operations.KwargsOperation(kwarg1=3, kwarg2=4),
- )
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ArgsOperation(\n"
- " arg1=custom_migration_operations.operations.ArgsOperation(\n"
- " arg1=1,\n"
- " arg2=2,\n"
- " ),\n"
- " arg2=custom_migration_operations.operations.KwargsOperation(\n"
- " kwarg1=3,\n"
- " kwarg2=4,\n"
- " ),\n"
- "),",
- )
- def test_multiline_args_signature(self):
- operation = custom_migration_operations.operations.ArgsOperation(
- "test\n arg1", "test\narg2"
- )
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ArgsOperation(\n"
- " arg1='test\\n arg1',\n"
- " arg2='test\\narg2',\n"
- "),",
- )
- def test_expand_args_signature(self):
- operation = custom_migration_operations.operations.ExpandArgsOperation([1, 2])
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ExpandArgsOperation(\n"
- " arg=[\n"
- " 1,\n"
- " 2,\n"
- " ],\n"
- "),",
- )
- def test_nested_operation_expand_args_signature(self):
- operation = custom_migration_operations.operations.ExpandArgsOperation(
- arg=[
- custom_migration_operations.operations.KwargsOperation(
- kwarg1=1,
- kwarg2=2,
- ),
- ]
- )
- buff, imports = OperationWriter(operation, indentation=0).serialize()
- self.assertEqual(imports, {"import custom_migration_operations.operations"})
- self.assertEqual(
- buff,
- "custom_migration_operations.operations.ExpandArgsOperation(\n"
- " arg=[\n"
- " custom_migration_operations.operations.KwargsOperation(\n"
- " kwarg1=1,\n"
- " kwarg2=2,\n"
- " ),\n"
- " ],\n"
- "),",
- )
- class WriterTests(SimpleTestCase):
- """
- Tests the migration writer (makes migration files from Migration instances)
- """
- class NestedEnum(enum.IntEnum):
- A = 1
- B = 2
- class NestedChoices(models.TextChoices):
- X = "X", "X value"
- Y = "Y", "Y value"
- def safe_exec(self, string, value=None):
- d = {}
- try:
- exec(string, globals(), d)
- except Exception as e:
- if value:
- self.fail(
- "Could not exec %r (from value %r): %s" % (string.strip(), value, e)
- )
- else:
- self.fail("Could not exec %r: %s" % (string.strip(), e))
- return d
- def serialize_round_trip(self, value):
- string, imports = MigrationWriter.serialize(value)
- return self.safe_exec(
- "%s\ntest_value_result = %s" % ("\n".join(imports), string), value
- )["test_value_result"]
- def assertSerializedEqual(self, value):
- self.assertEqual(self.serialize_round_trip(value), value)
- def assertSerializedResultEqual(self, value, target):
- self.assertEqual(MigrationWriter.serialize(value), target)
- def assertSerializedFieldEqual(self, value):
- new_value = self.serialize_round_trip(value)
- self.assertEqual(value.__class__, new_value.__class__)
- self.assertEqual(value.max_length, new_value.max_length)
- self.assertEqual(value.null, new_value.null)
- self.assertEqual(value.unique, new_value.unique)
- def test_serialize_numbers(self):
- self.assertSerializedEqual(1)
- self.assertSerializedEqual(1.2)
- self.assertTrue(math.isinf(self.serialize_round_trip(float("inf"))))
- self.assertTrue(math.isinf(self.serialize_round_trip(float("-inf"))))
- self.assertTrue(math.isnan(self.serialize_round_trip(float("nan"))))
- self.assertSerializedEqual(decimal.Decimal("1.3"))
- self.assertSerializedResultEqual(
- decimal.Decimal("1.3"), ("Decimal('1.3')", {"from decimal import Decimal"})
- )
- self.assertSerializedEqual(Money("1.3"))
- self.assertSerializedResultEqual(
- Money("1.3"),
- ("migrations.test_writer.Money('1.3')", {"import migrations.test_writer"}),
- )
- def test_serialize_constants(self):
- self.assertSerializedEqual(None)
- self.assertSerializedEqual(True)
- self.assertSerializedEqual(False)
- def test_serialize_strings(self):
- self.assertSerializedEqual(b"foobar")
- string, imports = MigrationWriter.serialize(b"foobar")
- self.assertEqual(string, "b'foobar'")
- self.assertSerializedEqual("föobár")
- string, imports = MigrationWriter.serialize("foobar")
- self.assertEqual(string, "'foobar'")
- def test_serialize_multiline_strings(self):
- self.assertSerializedEqual(b"foo\nbar")
- string, imports = MigrationWriter.serialize(b"foo\nbar")
- self.assertEqual(string, "b'foo\\nbar'")
- self.assertSerializedEqual("föo\nbár")
- string, imports = MigrationWriter.serialize("foo\nbar")
- self.assertEqual(string, "'foo\\nbar'")
- def test_serialize_collections(self):
- self.assertSerializedEqual({1: 2})
- self.assertSerializedEqual(["a", 2, True, None])
- self.assertSerializedEqual({2, 3, "eighty"})
- self.assertSerializedEqual({"lalalala": ["yeah", "no", "maybe"]})
- self.assertSerializedEqual(_("Hello"))
- def test_serialize_builtin_types(self):
- self.assertSerializedEqual([list, tuple, dict, set, frozenset])
- self.assertSerializedResultEqual(
- [list, tuple, dict, set, frozenset],
- ("[list, tuple, dict, set, frozenset]", set()),
- )
- def test_serialize_lazy_objects(self):
- pattern = re.compile(r"^foo$")
- lazy_pattern = SimpleLazyObject(lambda: pattern)
- self.assertEqual(self.serialize_round_trip(lazy_pattern), pattern)
- def test_serialize_enums(self):
- self.assertSerializedResultEqual(
- TextEnum.A,
- ("migrations.test_writer.TextEnum['A']", {"import migrations.test_writer"}),
- )
- self.assertSerializedResultEqual(
- TextTranslatedEnum.A,
- (
- "migrations.test_writer.TextTranslatedEnum['A']",
- {"import migrations.test_writer"},
- ),
- )
- self.assertSerializedResultEqual(
- BinaryEnum.A,
- (
- "migrations.test_writer.BinaryEnum['A']",
- {"import migrations.test_writer"},
- ),
- )
- self.assertSerializedResultEqual(
- IntEnum.B,
- ("migrations.test_writer.IntEnum['B']", {"import migrations.test_writer"}),
- )
- self.assertSerializedResultEqual(
- self.NestedEnum.A,
- (
- "migrations.test_writer.WriterTests.NestedEnum['A']",
- {"import migrations.test_writer"},
- ),
- )
- self.assertSerializedEqual(self.NestedEnum.A)
- field = models.CharField(
- default=TextEnum.B, choices=[(m.value, m) for m in TextEnum]
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.CharField(choices=["
- "('a-value', migrations.test_writer.TextEnum['A']), "
- "('value-b', migrations.test_writer.TextEnum['B'])], "
- "default=migrations.test_writer.TextEnum['B'])",
- )
- field = models.CharField(
- default=TextTranslatedEnum.A,
- choices=[(m.value, m) for m in TextTranslatedEnum],
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.CharField(choices=["
- "('a-value', migrations.test_writer.TextTranslatedEnum['A']), "
- "('value-b', migrations.test_writer.TextTranslatedEnum['B'])], "
- "default=migrations.test_writer.TextTranslatedEnum['A'])",
- )
- field = models.CharField(
- default=BinaryEnum.B, choices=[(m.value, m) for m in BinaryEnum]
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.CharField(choices=["
- "(b'a-value', migrations.test_writer.BinaryEnum['A']), "
- "(b'value-b', migrations.test_writer.BinaryEnum['B'])], "
- "default=migrations.test_writer.BinaryEnum['B'])",
- )
- field = models.IntegerField(
- default=IntEnum.A, choices=[(m.value, m) for m in IntEnum]
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.IntegerField(choices=["
- "(1, migrations.test_writer.IntEnum['A']), "
- "(2, migrations.test_writer.IntEnum['B'])], "
- "default=migrations.test_writer.IntEnum['A'])",
- )
- def test_serialize_enum_flags(self):
- self.assertSerializedResultEqual(
- IntFlagEnum.A,
- (
- "migrations.test_writer.IntFlagEnum['A']",
- {"import migrations.test_writer"},
- ),
- )
- self.assertSerializedResultEqual(
- IntFlagEnum.B,
- (
- "migrations.test_writer.IntFlagEnum['B']",
- {"import migrations.test_writer"},
- ),
- )
- field = models.IntegerField(
- default=IntFlagEnum.A, choices=[(m.value, m) for m in IntFlagEnum]
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.IntegerField(choices=["
- "(1, migrations.test_writer.IntFlagEnum['A']), "
- "(2, migrations.test_writer.IntFlagEnum['B'])], "
- "default=migrations.test_writer.IntFlagEnum['A'])",
- )
- self.assertSerializedResultEqual(
- IntFlagEnum.A | IntFlagEnum.B,
- (
- "migrations.test_writer.IntFlagEnum['A'] | "
- "migrations.test_writer.IntFlagEnum['B']",
- {"import migrations.test_writer"},
- ),
- )
- def test_serialize_choices(self):
- class TextChoices(models.TextChoices):
- A = "A", "A value"
- B = "B", "B value"
- class IntegerChoices(models.IntegerChoices):
- A = 1, "One"
- B = 2, "Two"
- class DateChoices(datetime.date, models.Choices):
- DATE_1 = 1969, 7, 20, "First date"
- DATE_2 = 1969, 11, 19, "Second date"
- self.assertSerializedResultEqual(TextChoices.A, ("'A'", set()))
- self.assertSerializedResultEqual(IntegerChoices.A, ("1", set()))
- self.assertSerializedResultEqual(
- DateChoices.DATE_1,
- ("datetime.date(1969, 7, 20)", {"import datetime"}),
- )
- field = models.CharField(default=TextChoices.B, choices=TextChoices)
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.CharField(choices=[('A', 'A value'), ('B', 'B value')], "
- "default='B')",
- )
- field = models.IntegerField(default=IntegerChoices.B, choices=IntegerChoices)
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.IntegerField(choices=[(1, 'One'), (2, 'Two')], default=2)",
- )
- field = models.DateField(default=DateChoices.DATE_2, choices=DateChoices)
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.DateField(choices=["
- "(datetime.date(1969, 7, 20), 'First date'), "
- "(datetime.date(1969, 11, 19), 'Second date')], "
- "default=datetime.date(1969, 11, 19))",
- )
- def test_serialize_nested_class(self):
- for nested_cls in [self.NestedEnum, self.NestedChoices]:
- cls_name = nested_cls.__name__
- with self.subTest(cls_name):
- self.assertSerializedResultEqual(
- nested_cls,
- (
- "migrations.test_writer.WriterTests.%s" % cls_name,
- {"import migrations.test_writer"},
- ),
- )
- def test_serialize_uuid(self):
- self.assertSerializedEqual(uuid.uuid1())
- self.assertSerializedEqual(uuid.uuid4())
- uuid_a = uuid.UUID("5c859437-d061-4847-b3f7-e6b78852f8c8")
- uuid_b = uuid.UUID("c7853ec1-2ea3-4359-b02d-b54e8f1bcee2")
- self.assertSerializedResultEqual(
- uuid_a,
- ("uuid.UUID('5c859437-d061-4847-b3f7-e6b78852f8c8')", {"import uuid"}),
- )
- self.assertSerializedResultEqual(
- uuid_b,
- ("uuid.UUID('c7853ec1-2ea3-4359-b02d-b54e8f1bcee2')", {"import uuid"}),
- )
- field = models.UUIDField(
- choices=((uuid_a, "UUID A"), (uuid_b, "UUID B")), default=uuid_a
- )
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(
- string,
- "models.UUIDField(choices=["
- "(uuid.UUID('5c859437-d061-4847-b3f7-e6b78852f8c8'), 'UUID A'), "
- "(uuid.UUID('c7853ec1-2ea3-4359-b02d-b54e8f1bcee2'), 'UUID B')], "
- "default=uuid.UUID('5c859437-d061-4847-b3f7-e6b78852f8c8'))",
- )
- def test_serialize_pathlib(self):
- # Pure path objects work in all platforms.
- self.assertSerializedEqual(pathlib.PurePosixPath())
- self.assertSerializedEqual(pathlib.PureWindowsPath())
- path = pathlib.PurePosixPath("/path/file.txt")
- expected = ("pathlib.PurePosixPath('/path/file.txt')", {"import pathlib"})
- self.assertSerializedResultEqual(path, expected)
- path = pathlib.PureWindowsPath("A:\\File.txt")
- expected = ("pathlib.PureWindowsPath('A:/File.txt')", {"import pathlib"})
- self.assertSerializedResultEqual(path, expected)
- # Concrete path objects work on supported platforms.
- if sys.platform == "win32":
- self.assertSerializedEqual(pathlib.WindowsPath.cwd())
- path = pathlib.WindowsPath("A:\\File.txt")
- expected = ("pathlib.PureWindowsPath('A:/File.txt')", {"import pathlib"})
- self.assertSerializedResultEqual(path, expected)
- else:
- self.assertSerializedEqual(pathlib.PosixPath.cwd())
- path = pathlib.PosixPath("/path/file.txt")
- expected = ("pathlib.PurePosixPath('/path/file.txt')", {"import pathlib"})
- self.assertSerializedResultEqual(path, expected)
- field = models.FilePathField(path=pathlib.PurePosixPath("/home/user"))
- string, imports = MigrationWriter.serialize(field)
- self.assertEqual(
- string,
- "models.FilePathField(path=pathlib.PurePosixPath('/home/user'))",
- )
- self.assertIn("import pathlib", imports)
- def test_serialize_path_like(self):
- with os.scandir(os.path.dirname(__file__)) as entries:
- path_like = list(entries)[0]
- expected = (repr(path_like.path), {})
- self.assertSerializedResultEqual(path_like, expected)
- field = models.FilePathField(path=path_like)
- string = MigrationWriter.serialize(field)[0]
- self.assertEqual(string, "models.FilePathField(path=%r)" % path_like.path)
- def test_serialize_functions(self):
- with self.assertRaisesMessage(ValueError, "Cannot serialize function: lambda"):
- self.assertSerializedEqual(lambda x: 42)
- self.assertSerializedEqual(models.SET_NULL)
- string, imports = MigrationWriter.serialize(models.SET(42))
- self.assertEqual(string, "models.SET(42)")
- self.serialize_round_trip(models.SET(42))
- def test_serialize_datetime(self):
- self.assertSerializedEqual(datetime.datetime.now())
- self.assertSerializedEqual(datetime.datetime.now)
- self.assertSerializedEqual(datetime.datetime.today())
- self.assertSerializedEqual(datetime.datetime.today)
- self.assertSerializedEqual(datetime.date.today())
- self.assertSerializedEqual(datetime.date.today)
- self.assertSerializedEqual(datetime.datetime.now().time())
- self.assertSerializedEqual(
- datetime.datetime(2014, 1, 1, 1, 1, tzinfo=get_default_timezone())
- )
- self.assertSerializedEqual(
- datetime.datetime(2013, 12, 31, 22, 1, tzinfo=get_fixed_timezone(180))
- )
- self.assertSerializedResultEqual(
- datetime.datetime(2014, 1, 1, 1, 1),
- ("datetime.datetime(2014, 1, 1, 1, 1)", {"import datetime"}),
- )
- self.assertSerializedResultEqual(
- datetime.datetime(2012, 1, 1, 1, 1, tzinfo=datetime.timezone.utc),
- (
- "datetime.datetime(2012, 1, 1, 1, 1, tzinfo=datetime.timezone.utc)",
- {"import datetime"},
- ),
- )
- self.assertSerializedResultEqual(
- datetime.datetime(
- 2012, 1, 1, 2, 1, tzinfo=zoneinfo.ZoneInfo("Europe/Paris")
- ),
- (
- "datetime.datetime(2012, 1, 1, 1, 1, tzinfo=datetime.timezone.utc)",
- {"import datetime"},
- ),
- )
- def test_serialize_fields(self):
- self.assertSerializedFieldEqual(models.CharField(max_length=255))
- self.assertSerializedResultEqual(
- models.CharField(max_length=255),
- ("models.CharField(max_length=255)", {"from django.db import models"}),
- )
- self.assertSerializedFieldEqual(models.TextField(null=True, blank=True))
- self.assertSerializedResultEqual(
- models.TextField(null=True, blank=True),
- (
- "models.TextField(blank=True, null=True)",
- {"from django.db import models"},
- ),
- )
- def test_serialize_settings(self):
- self.assertSerializedEqual(
- SettingsReference(settings.AUTH_USER_MODEL, "AUTH_USER_MODEL")
- )
- self.assertSerializedResultEqual(
- SettingsReference("someapp.model", "AUTH_USER_MODEL"),
- ("settings.AUTH_USER_MODEL", {"from django.conf import settings"}),
- )
- def test_serialize_iterators(self):
- self.assertSerializedResultEqual(
- ((x, x * x) for x in range(3)), ("((0, 0), (1, 1), (2, 4))", set())
- )
- def test_serialize_compiled_regex(self):
- """
- Make sure compiled regex can be serialized.
- """
- regex = re.compile(r"^\w+$")
- self.assertSerializedEqual(regex)
- def test_serialize_class_based_validators(self):
- """
- Ticket #22943: Test serialization of class-based validators, including
- compiled regexes.
- """
- validator = RegexValidator(message="hello")
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string, "django.core.validators.RegexValidator(message='hello')"
- )
- self.serialize_round_trip(validator)
- # Test with a compiled regex.
- validator = RegexValidator(regex=re.compile(r"^\w+$"))
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string,
- "django.core.validators.RegexValidator(regex=re.compile('^\\\\w+$'))",
- )
- self.serialize_round_trip(validator)
- # Test a string regex with flag
- validator = RegexValidator(r"^[0-9]+$", flags=re.S)
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string,
- "django.core.validators.RegexValidator('^[0-9]+$', "
- "flags=re.RegexFlag['DOTALL'])",
- )
- self.serialize_round_trip(validator)
- # Test message and code
- validator = RegexValidator("^[-a-zA-Z0-9_]+$", "Invalid", "invalid")
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string,
- "django.core.validators.RegexValidator('^[-a-zA-Z0-9_]+$', 'Invalid', "
- "'invalid')",
- )
- self.serialize_round_trip(validator)
- # Test with a subclass.
- validator = EmailValidator(message="hello")
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string, "django.core.validators.EmailValidator(message='hello')"
- )
- self.serialize_round_trip(validator)
- validator = deconstructible(path="migrations.test_writer.EmailValidator")(
- EmailValidator
- )(message="hello")
- string = MigrationWriter.serialize(validator)[0]
- self.assertEqual(
- string, "migrations.test_writer.EmailValidator(message='hello')"
- )
- validator = deconstructible(path="custom.EmailValidator")(EmailValidator)(
- message="hello"
- )
- with self.assertRaisesMessage(ImportError, "No module named 'custom'"):
- MigrationWriter.serialize(validator)
- validator = deconstructible(path="django.core.validators.EmailValidator2")(
- EmailValidator
- )(message="hello")
- with self.assertRaisesMessage(
- ValueError,
- "Could not find object EmailValidator2 in django.core.validators.",
- ):
- MigrationWriter.serialize(validator)
- def test_serialize_complex_func_index(self):
- index = models.Index(
- models.Func("rating", function="ABS"),
- models.Case(
- models.When(name="special", then=models.Value("X")),
- default=models.Value("other"),
- ),
- models.ExpressionWrapper(
- models.F("pages"),
- output_field=models.IntegerField(),
- ),
- models.OrderBy(models.F("name").desc()),
- name="complex_func_index",
- )
- string, imports = MigrationWriter.serialize(index)
- self.assertEqual(
- string,
- "models.Index(models.Func('rating', function='ABS'), "
- "models.Case(models.When(name='special', then=models.Value('X')), "
- "default=models.Value('other')), "
- "models.ExpressionWrapper("
- "models.F('pages'), output_field=models.IntegerField()), "
- "models.OrderBy(models.OrderBy(models.F('name'), descending=True)), "
- "name='complex_func_index')",
- )
- self.assertEqual(imports, {"from django.db import models"})
- def test_serialize_empty_nonempty_tuple(self):
- """
- Ticket #22679: makemigrations generates invalid code for (an empty
- tuple) default_permissions = ()
- """
- empty_tuple = ()
- one_item_tuple = ("a",)
- many_items_tuple = ("a", "b", "c")
- self.assertSerializedEqual(empty_tuple)
- self.assertSerializedEqual(one_item_tuple)
- self.assertSerializedEqual(many_items_tuple)
- def test_serialize_range(self):
- string, imports = MigrationWriter.serialize(range(1, 5))
- self.assertEqual(string, "range(1, 5)")
- self.assertEqual(imports, set())
- def test_serialize_builtins(self):
- string, imports = MigrationWriter.serialize(range)
- self.assertEqual(string, "range")
- self.assertEqual(imports, set())
- def test_serialize_unbound_method_reference(self):
- """An unbound method used within a class body can be serialized."""
- self.serialize_round_trip(TestModel1.thing)
- def test_serialize_local_function_reference(self):
- """A reference in a local scope can't be serialized."""
- class TestModel2:
- def upload_to(self):
- return "somewhere dynamic"
- thing = models.FileField(upload_to=upload_to)
- with self.assertRaisesMessage(
- ValueError, "Could not find function upload_to in migrations.test_writer"
- ):
- self.serialize_round_trip(TestModel2.thing)
- def test_serialize_managers(self):
- self.assertSerializedEqual(models.Manager())
- self.assertSerializedResultEqual(
- FoodQuerySet.as_manager(),
- (
- "migrations.models.FoodQuerySet.as_manager()",
- {"import migrations.models"},
- ),
- )
- self.assertSerializedEqual(FoodManager("a", "b"))
- self.assertSerializedEqual(FoodManager("x", "y", c=3, d=4))
- def test_serialize_frozensets(self):
- self.assertSerializedEqual(frozenset())
- self.assertSerializedEqual(frozenset("let it go"))
- self.assertSerializedResultEqual(
- frozenset("cba"), ("frozenset(['a', 'b', 'c'])", set())
- )
- def test_serialize_set(self):
- self.assertSerializedEqual(set())
- self.assertSerializedResultEqual(set(), ("set()", set()))
- self.assertSerializedEqual({"a"})
- self.assertSerializedResultEqual({"a"}, ("{'a'}", set()))
- self.assertSerializedEqual({"c", "b", "a"})
- self.assertSerializedResultEqual({"c", "b", "a"}, ("{'a', 'b', 'c'}", set()))
- def test_serialize_timedelta(self):
- self.assertSerializedEqual(datetime.timedelta())
- self.assertSerializedEqual(datetime.timedelta(minutes=42))
- def test_serialize_functools_partial(self):
- value = functools.partial(datetime.timedelta, 1, seconds=2)
- result = self.serialize_round_trip(value)
- self.assertEqual(result.func, value.func)
- self.assertEqual(result.args, value.args)
- self.assertEqual(result.keywords, value.keywords)
- def test_serialize_functools_partialmethod(self):
- value = functools.partialmethod(datetime.timedelta, 1, seconds=2)
- result = self.serialize_round_trip(value)
- self.assertIsInstance(result, functools.partialmethod)
- self.assertEqual(result.func, value.func)
- self.assertEqual(result.args, value.args)
- self.assertEqual(result.keywords, value.keywords)
- def test_serialize_type_none(self):
- self.assertSerializedEqual(NoneType)
- def test_serialize_type_model(self):
- self.assertSerializedEqual(models.Model)
- self.assertSerializedResultEqual(
- MigrationWriter.serialize(models.Model),
- ("('models.Model', {'from django.db import models'})", set()),
- )
- def test_simple_migration(self):
- """
- Tests serializing a simple migration.
- """
- fields = {
- "charfield": models.DateTimeField(default=datetime.datetime.now),
- "datetimefield": models.DateTimeField(default=datetime.datetime.now),
- }
- options = {
- "verbose_name": "My model",
- "verbose_name_plural": "My models",
- }
- migration = type(
- "Migration",
- (migrations.Migration,),
- {
- "operations": [
- migrations.CreateModel(
- "MyModel", tuple(fields.items()), options, (models.Model,)
- ),
- migrations.CreateModel(
- "MyModel2", tuple(fields.items()), bases=(models.Model,)
- ),
- migrations.CreateModel(
- name="MyModel3",
- fields=tuple(fields.items()),
- options=options,
- bases=(models.Model,),
- ),
- migrations.DeleteModel("MyModel"),
- migrations.AddField(
- "OtherModel", "datetimefield", fields["datetimefield"]
- ),
- ],
- "dependencies": [("testapp", "some_other_one")],
- },
- )
- writer = MigrationWriter(migration)
- output = writer.as_string()
- # We don't test the output formatting - that's too fragile.
- # Just make sure it runs for now, and that things look alright.
- result = self.safe_exec(output)
- self.assertIn("Migration", result)
- def test_migration_path(self):
- test_apps = [
- "migrations.migrations_test_apps.normal",
- "migrations.migrations_test_apps.with_package_model",
- "migrations.migrations_test_apps.without_init_file",
- ]
- base_dir = os.path.dirname(os.path.dirname(__file__))
- for app in test_apps:
- with self.modify_settings(INSTALLED_APPS={"append": app}):
- migration = migrations.Migration("0001_initial", app.split(".")[-1])
- expected_path = os.path.join(
- base_dir, *(app.split(".") + ["migrations", "0001_initial.py"])
- )
- writer = MigrationWriter(migration)
- self.assertEqual(writer.path, expected_path)
- def test_custom_operation(self):
- migration = type(
- "Migration",
- (migrations.Migration,),
- {
- "operations": [
- custom_migration_operations.operations.TestOperation(),
- custom_migration_operations.operations.CreateModel(),
- migrations.CreateModel("MyModel", (), {}, (models.Model,)),
- custom_migration_operations.more_operations.TestOperation(),
- ],
- "dependencies": [],
- },
- )
- writer = MigrationWriter(migration)
- output = writer.as_string()
- result = self.safe_exec(output)
- self.assertIn("custom_migration_operations", result)
- self.assertNotEqual(
- result["custom_migration_operations"].operations.TestOperation,
- result["custom_migration_operations"].more_operations.TestOperation,
- )
- def test_sorted_dependencies(self):
- migration = type(
- "Migration",
- (migrations.Migration,),
- {
- "operations": [
- migrations.AddField("mymodel", "myfield", models.IntegerField()),
- ],
- "dependencies": [
- ("testapp10", "0005_fifth"),
- ("testapp02", "0005_third"),
- ("testapp02", "0004_sixth"),
- ("testapp01", "0001_initial"),
- ],
- },
- )
- output = MigrationWriter(migration, include_header=False).as_string()
- self.assertIn(
- " dependencies = [\n"
- " ('testapp01', '0001_initial'),\n"
- " ('testapp02', '0004_sixth'),\n"
- " ('testapp02', '0005_third'),\n"
- " ('testapp10', '0005_fifth'),\n"
- " ]",
- output,
- )
- def test_sorted_imports(self):
- """
- #24155 - Tests ordering of imports.
- """
- migration = type(
- "Migration",
- (migrations.Migration,),
- {
- "operations": [
- migrations.AddField(
- "mymodel",
- "myfield",
- models.DateTimeField(
- default=datetime.datetime(
- 2012, 1, 1, 1, 1, tzinfo=datetime.timezone.utc
- ),
- ),
- ),
- migrations.AddField(
- "mymodel",
- "myfield2",
- models.FloatField(default=time.time),
- ),
- ]
- },
- )
- writer = MigrationWriter(migration)
- output = writer.as_string()
- self.assertIn(
- "import datetime\nimport time\nfrom django.db import migrations, models\n",
- output,
- )
- def test_migration_file_header_comments(self):
- """
- Test comments at top of file.
- """
- migration = type("Migration", (migrations.Migration,), {"operations": []})
- dt = datetime.datetime(2015, 7, 31, 4, 40, 0, 0, tzinfo=datetime.timezone.utc)
- with mock.patch("django.db.migrations.writer.now", lambda: dt):
- for include_header in (True, False):
- with self.subTest(include_header=include_header):
- writer = MigrationWriter(migration, include_header)
- output = writer.as_string()
- self.assertEqual(
- include_header,
- output.startswith(
- "# Generated by Django %s on 2015-07-31 04:40\n\n"
- % get_version()
- ),
- )
- if not include_header:
- # Make sure the output starts with something that's not
- # a comment or indentation or blank line
- self.assertRegex(
- output.splitlines(keepends=True)[0], r"^[^#\s]+"
- )
- def test_models_import_omitted(self):
- """
- django.db.models shouldn't be imported if unused.
- """
- migration = type(
- "Migration",
- (migrations.Migration,),
- {
- "operations": [
- migrations.AlterModelOptions(
- name="model",
- options={
- "verbose_name": "model",
- "verbose_name_plural": "models",
- },
- ),
- ]
- },
- )
- writer = MigrationWriter(migration)
- output = writer.as_string()
- self.assertIn("from django.db import migrations\n", output)
- def test_deconstruct_class_arguments(self):
- # Yes, it doesn't make sense to use a class as a default for a
- # CharField. It does make sense for custom fields though, for example
- # an enumfield that takes the enum class as an argument.
- string = MigrationWriter.serialize(
- models.CharField(default=DeconstructibleInstances)
- )[0]
- self.assertEqual(
- string,
- "models.CharField(default=migrations.test_writer.DeconstructibleInstances)",
- )
- def test_register_serializer(self):
- class ComplexSerializer(BaseSerializer):
- def serialize(self):
- return "complex(%r)" % self.value, {}
- MigrationWriter.register_serializer(complex, ComplexSerializer)
- self.assertSerializedEqual(complex(1, 2))
- MigrationWriter.unregister_serializer(complex)
- with self.assertRaisesMessage(ValueError, "Cannot serialize: (1+2j)"):
- self.assertSerializedEqual(complex(1, 2))
- def test_register_non_serializer(self):
- with self.assertRaisesMessage(
- ValueError, "'TestModel1' must inherit from 'BaseSerializer'."
- ):
- MigrationWriter.register_serializer(complex, TestModel1)
|