test_foreignkey.py 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. from decimal import Decimal
  2. from django.apps import apps
  3. from django.core import checks
  4. from django.db import models
  5. from django.test import TestCase, skipIfDBFeature
  6. from django.test.utils import isolate_apps
  7. from .models import Bar, FkToChar, Foo, PrimaryKeyCharModel
  8. class ForeignKeyTests(TestCase):
  9. def test_callable_default(self):
  10. """A lazy callable may be used for ForeignKey.default."""
  11. a = Foo.objects.create(id=1, a='abc', d=Decimal('12.34'))
  12. b = Bar.objects.create(b='bcd')
  13. self.assertEqual(b.a, a)
  14. @skipIfDBFeature('interprets_empty_strings_as_nulls')
  15. def test_empty_string_fk(self):
  16. """
  17. Empty strings foreign key values don't get converted to None (#19299).
  18. """
  19. char_model_empty = PrimaryKeyCharModel.objects.create(string='')
  20. fk_model_empty = FkToChar.objects.create(out=char_model_empty)
  21. fk_model_empty = FkToChar.objects.select_related('out').get(id=fk_model_empty.pk)
  22. self.assertEqual(fk_model_empty.out, char_model_empty)
  23. @isolate_apps('model_fields')
  24. def test_warning_when_unique_true_on_fk(self):
  25. class Foo(models.Model):
  26. pass
  27. class FKUniqueTrue(models.Model):
  28. fk_field = models.ForeignKey(Foo, models.CASCADE, unique=True)
  29. model = FKUniqueTrue()
  30. expected_warnings = [
  31. checks.Warning(
  32. 'Setting unique=True on a ForeignKey has the same effect as using a OneToOneField.',
  33. hint='ForeignKey(unique=True) is usually better served by a OneToOneField.',
  34. obj=FKUniqueTrue.fk_field.field,
  35. id='fields.W342',
  36. )
  37. ]
  38. warnings = model.check()
  39. self.assertEqual(warnings, expected_warnings)
  40. def test_related_name_converted_to_text(self):
  41. rel_name = Bar._meta.get_field('a').remote_field.related_name
  42. self.assertIsInstance(rel_name, str)
  43. def test_abstract_model_pending_operations(self):
  44. """
  45. Foreign key fields declared on abstract models should not add lazy
  46. relations to resolve relationship declared as string (#24215).
  47. """
  48. pending_ops_before = list(apps._pending_operations.items())
  49. class AbstractForeignKeyModel(models.Model):
  50. fk = models.ForeignKey('missing.FK', models.CASCADE)
  51. class Meta:
  52. abstract = True
  53. self.assertIs(AbstractForeignKeyModel._meta.apps, apps)
  54. self.assertEqual(
  55. pending_ops_before,
  56. list(apps._pending_operations.items()),
  57. 'Pending lookup added for a foreign key on an abstract model'
  58. )
  59. @isolate_apps('model_fields', 'model_fields.tests')
  60. def test_abstract_model_app_relative_foreign_key(self):
  61. class AbstractReferent(models.Model):
  62. reference = models.ForeignKey('Referred', on_delete=models.CASCADE)
  63. class Meta:
  64. app_label = 'model_fields'
  65. abstract = True
  66. def assert_app_model_resolved(label):
  67. class Referred(models.Model):
  68. class Meta:
  69. app_label = label
  70. class ConcreteReferent(AbstractReferent):
  71. class Meta:
  72. app_label = label
  73. self.assertEqual(ConcreteReferent._meta.get_field('reference').related_model, Referred)
  74. assert_app_model_resolved('model_fields')
  75. assert_app_model_resolved('tests')
  76. @isolate_apps('model_fields')
  77. def test_to_python(self):
  78. class Foo(models.Model):
  79. pass
  80. class Bar(models.Model):
  81. fk = models.ForeignKey(Foo, models.CASCADE)
  82. self.assertEqual(Bar._meta.get_field('fk').to_python('1'), 1)
  83. @isolate_apps('model_fields')
  84. def test_fk_to_fk_get_col_output_field(self):
  85. class Foo(models.Model):
  86. pass
  87. class Bar(models.Model):
  88. foo = models.ForeignKey(Foo, models.CASCADE, primary_key=True)
  89. class Baz(models.Model):
  90. bar = models.ForeignKey(Bar, models.CASCADE, primary_key=True)
  91. col = Baz._meta.get_field('bar').get_col('alias')
  92. self.assertIs(col.output_field, Foo._meta.pk)
  93. @isolate_apps('model_fields')
  94. def test_recursive_fks_get_col(self):
  95. class Foo(models.Model):
  96. bar = models.ForeignKey('Bar', models.CASCADE, primary_key=True)
  97. class Bar(models.Model):
  98. foo = models.ForeignKey(Foo, models.CASCADE, primary_key=True)
  99. with self.assertRaisesMessage(ValueError, 'Cannot resolve output_field'):
  100. Foo._meta.get_field('bar').get_col('alias')