test_model_checks.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. from unittest import mock
  2. from django.core import checks
  3. from django.core.checks import Error, Warning
  4. from django.db import models
  5. from django.test import SimpleTestCase, TestCase, skipUnlessDBFeature
  6. from django.test.utils import (
  7. isolate_apps, modify_settings, override_settings, override_system_checks,
  8. )
  9. class EmptyRouter:
  10. pass
  11. @isolate_apps('check_framework', attr_name='apps')
  12. @override_system_checks([checks.model_checks.check_all_models])
  13. class DuplicateDBTableTests(SimpleTestCase):
  14. def test_collision_in_same_app(self):
  15. class Model1(models.Model):
  16. class Meta:
  17. db_table = 'test_table'
  18. class Model2(models.Model):
  19. class Meta:
  20. db_table = 'test_table'
  21. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  22. Error(
  23. "db_table 'test_table' is used by multiple models: "
  24. "check_framework.Model1, check_framework.Model2.",
  25. obj='test_table',
  26. id='models.E028',
  27. )
  28. ])
  29. @override_settings(DATABASE_ROUTERS=['check_framework.test_model_checks.EmptyRouter'])
  30. def test_collision_in_same_app_database_routers_installed(self):
  31. class Model1(models.Model):
  32. class Meta:
  33. db_table = 'test_table'
  34. class Model2(models.Model):
  35. class Meta:
  36. db_table = 'test_table'
  37. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  38. Warning(
  39. "db_table 'test_table' is used by multiple models: "
  40. "check_framework.Model1, check_framework.Model2.",
  41. hint=(
  42. 'You have configured settings.DATABASE_ROUTERS. Verify '
  43. 'that check_framework.Model1, check_framework.Model2 are '
  44. 'correctly routed to separate databases.'
  45. ),
  46. obj='test_table',
  47. id='models.W035',
  48. )
  49. ])
  50. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  51. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  52. def test_collision_across_apps(self, apps):
  53. class Model1(models.Model):
  54. class Meta:
  55. app_label = 'basic'
  56. db_table = 'test_table'
  57. class Model2(models.Model):
  58. class Meta:
  59. app_label = 'check_framework'
  60. db_table = 'test_table'
  61. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [
  62. Error(
  63. "db_table 'test_table' is used by multiple models: "
  64. "basic.Model1, check_framework.Model2.",
  65. obj='test_table',
  66. id='models.E028',
  67. )
  68. ])
  69. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  70. @override_settings(DATABASE_ROUTERS=['check_framework.test_model_checks.EmptyRouter'])
  71. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  72. def test_collision_across_apps_database_routers_installed(self, apps):
  73. class Model1(models.Model):
  74. class Meta:
  75. app_label = 'basic'
  76. db_table = 'test_table'
  77. class Model2(models.Model):
  78. class Meta:
  79. app_label = 'check_framework'
  80. db_table = 'test_table'
  81. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [
  82. Warning(
  83. "db_table 'test_table' is used by multiple models: "
  84. "basic.Model1, check_framework.Model2.",
  85. hint=(
  86. 'You have configured settings.DATABASE_ROUTERS. Verify '
  87. 'that basic.Model1, check_framework.Model2 are correctly '
  88. 'routed to separate databases.'
  89. ),
  90. obj='test_table',
  91. id='models.W035',
  92. )
  93. ])
  94. def test_no_collision_for_unmanaged_models(self):
  95. class Unmanaged(models.Model):
  96. class Meta:
  97. db_table = 'test_table'
  98. managed = False
  99. class Managed(models.Model):
  100. class Meta:
  101. db_table = 'test_table'
  102. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  103. def test_no_collision_for_proxy_models(self):
  104. class Model(models.Model):
  105. class Meta:
  106. db_table = 'test_table'
  107. class ProxyModel(Model):
  108. class Meta:
  109. proxy = True
  110. self.assertEqual(Model._meta.db_table, ProxyModel._meta.db_table)
  111. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  112. @isolate_apps('check_framework', attr_name='apps')
  113. @override_system_checks([checks.model_checks.check_all_models])
  114. class IndexNameTests(SimpleTestCase):
  115. def test_collision_in_same_model(self):
  116. index = models.Index(fields=['id'], name='foo')
  117. class Model(models.Model):
  118. class Meta:
  119. indexes = [index, index]
  120. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  121. Error(
  122. "index name 'foo' is not unique for model check_framework.Model.",
  123. id='models.E029',
  124. ),
  125. ])
  126. def test_collision_in_different_models(self):
  127. index = models.Index(fields=['id'], name='foo')
  128. class Model1(models.Model):
  129. class Meta:
  130. indexes = [index]
  131. class Model2(models.Model):
  132. class Meta:
  133. indexes = [index]
  134. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  135. Error(
  136. "index name 'foo' is not unique among models: "
  137. "check_framework.Model1, check_framework.Model2.",
  138. id='models.E030',
  139. ),
  140. ])
  141. def test_collision_abstract_model(self):
  142. class AbstractModel(models.Model):
  143. class Meta:
  144. indexes = [models.Index(fields=['id'], name='foo')]
  145. abstract = True
  146. class Model1(AbstractModel):
  147. pass
  148. class Model2(AbstractModel):
  149. pass
  150. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  151. Error(
  152. "index name 'foo' is not unique among models: "
  153. "check_framework.Model1, check_framework.Model2.",
  154. id='models.E030',
  155. ),
  156. ])
  157. def test_no_collision_abstract_model_interpolation(self):
  158. class AbstractModel(models.Model):
  159. name = models.CharField(max_length=20)
  160. class Meta:
  161. indexes = [models.Index(fields=['name'], name='%(app_label)s_%(class)s_foo')]
  162. abstract = True
  163. class Model1(AbstractModel):
  164. pass
  165. class Model2(AbstractModel):
  166. pass
  167. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  168. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  169. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  170. def test_collision_across_apps(self, apps):
  171. index = models.Index(fields=['id'], name='foo')
  172. class Model1(models.Model):
  173. class Meta:
  174. app_label = 'basic'
  175. indexes = [index]
  176. class Model2(models.Model):
  177. class Meta:
  178. app_label = 'check_framework'
  179. indexes = [index]
  180. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [
  181. Error(
  182. "index name 'foo' is not unique among models: basic.Model1, "
  183. "check_framework.Model2.",
  184. id='models.E030',
  185. ),
  186. ])
  187. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  188. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  189. def test_no_collision_across_apps_interpolation(self, apps):
  190. index = models.Index(fields=['id'], name='%(app_label)s_%(class)s_foo')
  191. class Model1(models.Model):
  192. class Meta:
  193. app_label = 'basic'
  194. constraints = [index]
  195. class Model2(models.Model):
  196. class Meta:
  197. app_label = 'check_framework'
  198. constraints = [index]
  199. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [])
  200. @isolate_apps('check_framework', attr_name='apps')
  201. @override_system_checks([checks.model_checks.check_all_models])
  202. @skipUnlessDBFeature('supports_table_check_constraints')
  203. class ConstraintNameTests(TestCase):
  204. def test_collision_in_same_model(self):
  205. class Model(models.Model):
  206. class Meta:
  207. constraints = [
  208. models.CheckConstraint(check=models.Q(id__gt=0), name='foo'),
  209. models.CheckConstraint(check=models.Q(id__lt=100), name='foo'),
  210. ]
  211. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  212. Error(
  213. "constraint name 'foo' is not unique for model "
  214. "check_framework.Model.",
  215. id='models.E031',
  216. ),
  217. ])
  218. def test_collision_in_different_models(self):
  219. constraint = models.CheckConstraint(check=models.Q(id__gt=0), name='foo')
  220. class Model1(models.Model):
  221. class Meta:
  222. constraints = [constraint]
  223. class Model2(models.Model):
  224. class Meta:
  225. constraints = [constraint]
  226. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  227. Error(
  228. "constraint name 'foo' is not unique among models: "
  229. "check_framework.Model1, check_framework.Model2.",
  230. id='models.E032',
  231. ),
  232. ])
  233. def test_collision_abstract_model(self):
  234. class AbstractModel(models.Model):
  235. class Meta:
  236. constraints = [models.CheckConstraint(check=models.Q(id__gt=0), name='foo')]
  237. abstract = True
  238. class Model1(AbstractModel):
  239. pass
  240. class Model2(AbstractModel):
  241. pass
  242. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  243. Error(
  244. "constraint name 'foo' is not unique among models: "
  245. "check_framework.Model1, check_framework.Model2.",
  246. id='models.E032',
  247. ),
  248. ])
  249. def test_no_collision_abstract_model_interpolation(self):
  250. class AbstractModel(models.Model):
  251. class Meta:
  252. constraints = [
  253. models.CheckConstraint(check=models.Q(id__gt=0), name='%(app_label)s_%(class)s_foo'),
  254. ]
  255. abstract = True
  256. class Model1(AbstractModel):
  257. pass
  258. class Model2(AbstractModel):
  259. pass
  260. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  261. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  262. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  263. def test_collision_across_apps(self, apps):
  264. constraint = models.CheckConstraint(check=models.Q(id__gt=0), name='foo')
  265. class Model1(models.Model):
  266. class Meta:
  267. app_label = 'basic'
  268. constraints = [constraint]
  269. class Model2(models.Model):
  270. class Meta:
  271. app_label = 'check_framework'
  272. constraints = [constraint]
  273. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [
  274. Error(
  275. "constraint name 'foo' is not unique among models: "
  276. "basic.Model1, check_framework.Model2.",
  277. id='models.E032',
  278. ),
  279. ])
  280. @modify_settings(INSTALLED_APPS={'append': 'basic'})
  281. @isolate_apps('basic', 'check_framework', kwarg_name='apps')
  282. def test_no_collision_across_apps_interpolation(self, apps):
  283. constraint = models.CheckConstraint(check=models.Q(id__gt=0), name='%(app_label)s_%(class)s_foo')
  284. class Model1(models.Model):
  285. class Meta:
  286. app_label = 'basic'
  287. constraints = [constraint]
  288. class Model2(models.Model):
  289. class Meta:
  290. app_label = 'check_framework'
  291. constraints = [constraint]
  292. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [])
  293. def mocked_is_overridden(self, setting):
  294. # Force treating DEFAULT_AUTO_FIELD = 'django.db.models.AutoField' as a not
  295. # overridden setting.
  296. return (
  297. setting != 'DEFAULT_AUTO_FIELD' or
  298. self.DEFAULT_AUTO_FIELD != 'django.db.models.AutoField'
  299. )
  300. @mock.patch('django.conf.UserSettingsHolder.is_overridden', mocked_is_overridden)
  301. @override_settings(DEFAULT_AUTO_FIELD='django.db.models.AutoField')
  302. @isolate_apps('check_framework.apps.CheckDefaultPKConfig', attr_name='apps')
  303. @override_system_checks([checks.model_checks.check_all_models])
  304. class ModelDefaultAutoFieldTests(SimpleTestCase):
  305. def test_auto_created_pk(self):
  306. class Model(models.Model):
  307. pass
  308. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [
  309. Warning(
  310. "Auto-created primary key used when not defining a primary "
  311. "key type, by default 'django.db.models.AutoField'.",
  312. hint=(
  313. "Configure the DEFAULT_AUTO_FIELD setting or the "
  314. "CheckDefaultPKConfig.default_auto_field attribute to "
  315. "point to a subclass of AutoField, e.g. "
  316. "'django.db.models.BigAutoField'."
  317. ),
  318. obj=Model,
  319. id='models.W042',
  320. ),
  321. ])
  322. @override_settings(DEFAULT_AUTO_FIELD='django.db.models.BigAutoField')
  323. def test_default_auto_field_setting(self):
  324. class Model(models.Model):
  325. pass
  326. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  327. def test_explicit_pk(self):
  328. class Model(models.Model):
  329. id = models.BigAutoField(primary_key=True)
  330. self.assertEqual(checks.run_checks(app_configs=self.apps.get_app_configs()), [])
  331. @isolate_apps('check_framework.apps.CheckPKConfig', kwarg_name='apps')
  332. def test_app_default_auto_field(self, apps):
  333. class ModelWithPkViaAppConfig(models.Model):
  334. class Meta:
  335. app_label = 'check_framework.apps.CheckPKConfig'
  336. self.assertEqual(checks.run_checks(app_configs=apps.get_app_configs()), [])