test_array.py 43 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148
  1. import decimal
  2. import enum
  3. import json
  4. import unittest
  5. import uuid
  6. from django import forms
  7. from django.core import checks, exceptions, serializers, validators
  8. from django.core.exceptions import FieldError
  9. from django.core.management import call_command
  10. from django.db import IntegrityError, connection, models
  11. from django.db.models.expressions import RawSQL
  12. from django.db.models.functions import Cast
  13. from django.test import TransactionTestCase, modify_settings, override_settings
  14. from django.test.utils import isolate_apps
  15. from django.utils import timezone
  16. from . import (
  17. PostgreSQLSimpleTestCase, PostgreSQLTestCase, PostgreSQLWidgetTestCase,
  18. )
  19. from .models import (
  20. ArrayEnumModel, ArrayFieldSubclass, CharArrayModel, DateTimeArrayModel,
  21. IntegerArrayModel, NestedIntegerArrayModel, NullableIntegerArrayModel,
  22. OtherTypesArrayModel, PostgreSQLModel, Tag,
  23. )
  24. try:
  25. from psycopg2.extras import NumericRange
  26. from django.contrib.postgres.aggregates import ArrayAgg
  27. from django.contrib.postgres.fields import ArrayField
  28. from django.contrib.postgres.fields.array import (
  29. IndexTransform, SliceTransform,
  30. )
  31. from django.contrib.postgres.forms import (
  32. SimpleArrayField, SplitArrayField, SplitArrayWidget,
  33. )
  34. from django.db.backends.postgresql.base import PSYCOPG2_VERSION
  35. except ImportError:
  36. pass
  37. @isolate_apps('postgres_tests')
  38. class BasicTests(PostgreSQLSimpleTestCase):
  39. def test_get_field_display(self):
  40. class MyModel(PostgreSQLModel):
  41. field = ArrayField(
  42. models.CharField(max_length=16),
  43. choices=[
  44. ['Media', [(['vinyl', 'cd'], 'Audio')]],
  45. (('mp3', 'mp4'), 'Digital'),
  46. ],
  47. )
  48. tests = (
  49. (['vinyl', 'cd'], 'Audio'),
  50. (('mp3', 'mp4'), 'Digital'),
  51. (('a', 'b'), "('a', 'b')"),
  52. (['c', 'd'], "['c', 'd']"),
  53. )
  54. for value, display in tests:
  55. with self.subTest(value=value, display=display):
  56. instance = MyModel(field=value)
  57. self.assertEqual(instance.get_field_display(), display)
  58. def test_get_field_display_nested_array(self):
  59. class MyModel(PostgreSQLModel):
  60. field = ArrayField(
  61. ArrayField(models.CharField(max_length=16)),
  62. choices=[
  63. [
  64. 'Media',
  65. [([['vinyl', 'cd'], ('x',)], 'Audio')],
  66. ],
  67. ((['mp3'], ('mp4',)), 'Digital'),
  68. ],
  69. )
  70. tests = (
  71. ([['vinyl', 'cd'], ('x',)], 'Audio'),
  72. ((['mp3'], ('mp4',)), 'Digital'),
  73. ((('a', 'b'), ('c',)), "(('a', 'b'), ('c',))"),
  74. ([['a', 'b'], ['c']], "[['a', 'b'], ['c']]"),
  75. )
  76. for value, display in tests:
  77. with self.subTest(value=value, display=display):
  78. instance = MyModel(field=value)
  79. self.assertEqual(instance.get_field_display(), display)
  80. class TestSaveLoad(PostgreSQLTestCase):
  81. def test_integer(self):
  82. instance = IntegerArrayModel(field=[1, 2, 3])
  83. instance.save()
  84. loaded = IntegerArrayModel.objects.get()
  85. self.assertEqual(instance.field, loaded.field)
  86. def test_char(self):
  87. instance = CharArrayModel(field=['hello', 'goodbye'])
  88. instance.save()
  89. loaded = CharArrayModel.objects.get()
  90. self.assertEqual(instance.field, loaded.field)
  91. def test_dates(self):
  92. instance = DateTimeArrayModel(
  93. datetimes=[timezone.now()],
  94. dates=[timezone.now().date()],
  95. times=[timezone.now().time()],
  96. )
  97. instance.save()
  98. loaded = DateTimeArrayModel.objects.get()
  99. self.assertEqual(instance.datetimes, loaded.datetimes)
  100. self.assertEqual(instance.dates, loaded.dates)
  101. self.assertEqual(instance.times, loaded.times)
  102. def test_tuples(self):
  103. instance = IntegerArrayModel(field=(1,))
  104. instance.save()
  105. loaded = IntegerArrayModel.objects.get()
  106. self.assertSequenceEqual(instance.field, loaded.field)
  107. def test_integers_passed_as_strings(self):
  108. # This checks that get_prep_value is deferred properly
  109. instance = IntegerArrayModel(field=['1'])
  110. instance.save()
  111. loaded = IntegerArrayModel.objects.get()
  112. self.assertEqual(loaded.field, [1])
  113. def test_default_null(self):
  114. instance = NullableIntegerArrayModel()
  115. instance.save()
  116. loaded = NullableIntegerArrayModel.objects.get(pk=instance.pk)
  117. self.assertIsNone(loaded.field)
  118. self.assertEqual(instance.field, loaded.field)
  119. def test_null_handling(self):
  120. instance = NullableIntegerArrayModel(field=None)
  121. instance.save()
  122. loaded = NullableIntegerArrayModel.objects.get()
  123. self.assertEqual(instance.field, loaded.field)
  124. instance = IntegerArrayModel(field=None)
  125. with self.assertRaises(IntegrityError):
  126. instance.save()
  127. def test_nested(self):
  128. instance = NestedIntegerArrayModel(field=[[1, 2], [3, 4]])
  129. instance.save()
  130. loaded = NestedIntegerArrayModel.objects.get()
  131. self.assertEqual(instance.field, loaded.field)
  132. def test_other_array_types(self):
  133. instance = OtherTypesArrayModel(
  134. ips=['192.168.0.1', '::1'],
  135. uuids=[uuid.uuid4()],
  136. decimals=[decimal.Decimal(1.25), 1.75],
  137. tags=[Tag(1), Tag(2), Tag(3)],
  138. json=[{'a': 1}, {'b': 2}],
  139. int_ranges=[NumericRange(10, 20), NumericRange(30, 40)],
  140. bigint_ranges=[
  141. NumericRange(7000000000, 10000000000),
  142. NumericRange(50000000000, 70000000000),
  143. ]
  144. )
  145. instance.save()
  146. loaded = OtherTypesArrayModel.objects.get()
  147. self.assertEqual(instance.ips, loaded.ips)
  148. self.assertEqual(instance.uuids, loaded.uuids)
  149. self.assertEqual(instance.decimals, loaded.decimals)
  150. self.assertEqual(instance.tags, loaded.tags)
  151. self.assertEqual(instance.json, loaded.json)
  152. self.assertEqual(instance.int_ranges, loaded.int_ranges)
  153. self.assertEqual(instance.bigint_ranges, loaded.bigint_ranges)
  154. def test_null_from_db_value_handling(self):
  155. instance = OtherTypesArrayModel.objects.create(
  156. ips=['192.168.0.1', '::1'],
  157. uuids=[uuid.uuid4()],
  158. decimals=[decimal.Decimal(1.25), 1.75],
  159. tags=None,
  160. )
  161. instance.refresh_from_db()
  162. self.assertIsNone(instance.tags)
  163. self.assertEqual(instance.json, [])
  164. self.assertIsNone(instance.int_ranges)
  165. self.assertIsNone(instance.bigint_ranges)
  166. def test_model_set_on_base_field(self):
  167. instance = IntegerArrayModel()
  168. field = instance._meta.get_field('field')
  169. self.assertEqual(field.model, IntegerArrayModel)
  170. self.assertEqual(field.base_field.model, IntegerArrayModel)
  171. def test_nested_nullable_base_field(self):
  172. if PSYCOPG2_VERSION < (2, 7, 5):
  173. self.skipTest('See https://github.com/psycopg/psycopg2/issues/325')
  174. instance = NullableIntegerArrayModel.objects.create(
  175. field_nested=[[None, None], [None, None]],
  176. )
  177. self.assertEqual(instance.field_nested, [[None, None], [None, None]])
  178. class TestQuerying(PostgreSQLTestCase):
  179. @classmethod
  180. def setUpTestData(cls):
  181. cls.objs = NullableIntegerArrayModel.objects.bulk_create([
  182. NullableIntegerArrayModel(field=[1]),
  183. NullableIntegerArrayModel(field=[2]),
  184. NullableIntegerArrayModel(field=[2, 3]),
  185. NullableIntegerArrayModel(field=[20, 30, 40]),
  186. NullableIntegerArrayModel(field=None),
  187. ])
  188. def test_empty_list(self):
  189. NullableIntegerArrayModel.objects.create(field=[])
  190. obj = NullableIntegerArrayModel.objects.annotate(
  191. empty_array=models.Value([], output_field=ArrayField(models.IntegerField())),
  192. ).filter(field=models.F('empty_array')).get()
  193. self.assertEqual(obj.field, [])
  194. self.assertEqual(obj.empty_array, [])
  195. def test_exact(self):
  196. self.assertSequenceEqual(
  197. NullableIntegerArrayModel.objects.filter(field__exact=[1]),
  198. self.objs[:1]
  199. )
  200. def test_exact_charfield(self):
  201. instance = CharArrayModel.objects.create(field=['text'])
  202. self.assertSequenceEqual(
  203. CharArrayModel.objects.filter(field=['text']),
  204. [instance]
  205. )
  206. def test_exact_nested(self):
  207. instance = NestedIntegerArrayModel.objects.create(field=[[1, 2], [3, 4]])
  208. self.assertSequenceEqual(
  209. NestedIntegerArrayModel.objects.filter(field=[[1, 2], [3, 4]]),
  210. [instance]
  211. )
  212. def test_isnull(self):
  213. self.assertSequenceEqual(
  214. NullableIntegerArrayModel.objects.filter(field__isnull=True),
  215. self.objs[-1:]
  216. )
  217. def test_gt(self):
  218. self.assertSequenceEqual(
  219. NullableIntegerArrayModel.objects.filter(field__gt=[0]),
  220. self.objs[:4]
  221. )
  222. def test_lt(self):
  223. self.assertSequenceEqual(
  224. NullableIntegerArrayModel.objects.filter(field__lt=[2]),
  225. self.objs[:1]
  226. )
  227. def test_in(self):
  228. self.assertSequenceEqual(
  229. NullableIntegerArrayModel.objects.filter(field__in=[[1], [2]]),
  230. self.objs[:2]
  231. )
  232. def test_in_subquery(self):
  233. IntegerArrayModel.objects.create(field=[2, 3])
  234. self.assertSequenceEqual(
  235. NullableIntegerArrayModel.objects.filter(
  236. field__in=IntegerArrayModel.objects.all().values_list('field', flat=True)
  237. ),
  238. self.objs[2:3]
  239. )
  240. @unittest.expectedFailure
  241. def test_in_including_F_object(self):
  242. # This test asserts that Array objects passed to filters can be
  243. # constructed to contain F objects. This currently doesn't work as the
  244. # psycopg2 mogrify method that generates the ARRAY() syntax is
  245. # expecting literals, not column references (#27095).
  246. self.assertSequenceEqual(
  247. NullableIntegerArrayModel.objects.filter(field__in=[[models.F('id')]]),
  248. self.objs[:2]
  249. )
  250. def test_in_as_F_object(self):
  251. self.assertSequenceEqual(
  252. NullableIntegerArrayModel.objects.filter(field__in=[models.F('field')]),
  253. self.objs[:4]
  254. )
  255. def test_contained_by(self):
  256. self.assertSequenceEqual(
  257. NullableIntegerArrayModel.objects.filter(field__contained_by=[1, 2]),
  258. self.objs[:2]
  259. )
  260. @unittest.expectedFailure
  261. def test_contained_by_including_F_object(self):
  262. # This test asserts that Array objects passed to filters can be
  263. # constructed to contain F objects. This currently doesn't work as the
  264. # psycopg2 mogrify method that generates the ARRAY() syntax is
  265. # expecting literals, not column references (#27095).
  266. self.assertSequenceEqual(
  267. NullableIntegerArrayModel.objects.filter(field__contained_by=[models.F('id'), 2]),
  268. self.objs[:2]
  269. )
  270. def test_contains(self):
  271. self.assertSequenceEqual(
  272. NullableIntegerArrayModel.objects.filter(field__contains=[2]),
  273. self.objs[1:3]
  274. )
  275. def test_icontains(self):
  276. # Using the __icontains lookup with ArrayField is inefficient.
  277. instance = CharArrayModel.objects.create(field=['FoO'])
  278. self.assertSequenceEqual(
  279. CharArrayModel.objects.filter(field__icontains='foo'),
  280. [instance]
  281. )
  282. def test_contains_charfield(self):
  283. # Regression for #22907
  284. self.assertSequenceEqual(
  285. CharArrayModel.objects.filter(field__contains=['text']),
  286. []
  287. )
  288. def test_contained_by_charfield(self):
  289. self.assertSequenceEqual(
  290. CharArrayModel.objects.filter(field__contained_by=['text']),
  291. []
  292. )
  293. def test_overlap_charfield(self):
  294. self.assertSequenceEqual(
  295. CharArrayModel.objects.filter(field__overlap=['text']),
  296. []
  297. )
  298. def test_lookups_autofield_array(self):
  299. qs = NullableIntegerArrayModel.objects.filter(
  300. field__0__isnull=False,
  301. ).values('field__0').annotate(
  302. arrayagg=ArrayAgg('id'),
  303. ).order_by('field__0')
  304. tests = (
  305. ('contained_by', [self.objs[1].pk, self.objs[2].pk, 0], [2]),
  306. ('contains', [self.objs[2].pk], [2]),
  307. ('exact', [self.objs[3].pk], [20]),
  308. ('overlap', [self.objs[1].pk, self.objs[3].pk], [2, 20]),
  309. )
  310. for lookup, value, expected in tests:
  311. with self.subTest(lookup=lookup):
  312. self.assertSequenceEqual(
  313. qs.filter(
  314. **{'arrayagg__' + lookup: value},
  315. ).values_list('field__0', flat=True),
  316. expected,
  317. )
  318. def test_index(self):
  319. self.assertSequenceEqual(
  320. NullableIntegerArrayModel.objects.filter(field__0=2),
  321. self.objs[1:3]
  322. )
  323. def test_index_chained(self):
  324. self.assertSequenceEqual(
  325. NullableIntegerArrayModel.objects.filter(field__0__lt=3),
  326. self.objs[0:3]
  327. )
  328. def test_index_nested(self):
  329. instance = NestedIntegerArrayModel.objects.create(field=[[1, 2], [3, 4]])
  330. self.assertSequenceEqual(
  331. NestedIntegerArrayModel.objects.filter(field__0__0=1),
  332. [instance]
  333. )
  334. @unittest.expectedFailure
  335. def test_index_used_on_nested_data(self):
  336. instance = NestedIntegerArrayModel.objects.create(field=[[1, 2], [3, 4]])
  337. self.assertSequenceEqual(
  338. NestedIntegerArrayModel.objects.filter(field__0=[1, 2]),
  339. [instance]
  340. )
  341. def test_index_transform_expression(self):
  342. expr = RawSQL("string_to_array(%s, ';')", ['1;2'])
  343. self.assertSequenceEqual(
  344. NullableIntegerArrayModel.objects.filter(
  345. field__0=Cast(
  346. IndexTransform(1, models.IntegerField, expr),
  347. output_field=models.IntegerField(),
  348. ),
  349. ),
  350. self.objs[:1],
  351. )
  352. def test_overlap(self):
  353. self.assertSequenceEqual(
  354. NullableIntegerArrayModel.objects.filter(field__overlap=[1, 2]),
  355. self.objs[0:3]
  356. )
  357. def test_len(self):
  358. self.assertSequenceEqual(
  359. NullableIntegerArrayModel.objects.filter(field__len__lte=2),
  360. self.objs[0:3]
  361. )
  362. def test_len_empty_array(self):
  363. obj = NullableIntegerArrayModel.objects.create(field=[])
  364. self.assertSequenceEqual(
  365. NullableIntegerArrayModel.objects.filter(field__len=0),
  366. [obj]
  367. )
  368. def test_slice(self):
  369. self.assertSequenceEqual(
  370. NullableIntegerArrayModel.objects.filter(field__0_1=[2]),
  371. self.objs[1:3]
  372. )
  373. self.assertSequenceEqual(
  374. NullableIntegerArrayModel.objects.filter(field__0_2=[2, 3]),
  375. self.objs[2:3]
  376. )
  377. def test_order_by_slice(self):
  378. more_objs = (
  379. NullableIntegerArrayModel.objects.create(field=[1, 637]),
  380. NullableIntegerArrayModel.objects.create(field=[2, 1]),
  381. NullableIntegerArrayModel.objects.create(field=[3, -98123]),
  382. NullableIntegerArrayModel.objects.create(field=[4, 2]),
  383. )
  384. self.assertSequenceEqual(
  385. NullableIntegerArrayModel.objects.order_by('field__1'),
  386. [
  387. more_objs[2], more_objs[1], more_objs[3], self.objs[2],
  388. self.objs[3], more_objs[0], self.objs[4], self.objs[1],
  389. self.objs[0],
  390. ]
  391. )
  392. @unittest.expectedFailure
  393. def test_slice_nested(self):
  394. instance = NestedIntegerArrayModel.objects.create(field=[[1, 2], [3, 4]])
  395. self.assertSequenceEqual(
  396. NestedIntegerArrayModel.objects.filter(field__0__0_1=[1]),
  397. [instance]
  398. )
  399. def test_slice_transform_expression(self):
  400. expr = RawSQL("string_to_array(%s, ';')", ['9;2;3'])
  401. self.assertSequenceEqual(
  402. NullableIntegerArrayModel.objects.filter(field__0_2=SliceTransform(2, 3, expr)),
  403. self.objs[2:3],
  404. )
  405. def test_usage_in_subquery(self):
  406. self.assertSequenceEqual(
  407. NullableIntegerArrayModel.objects.filter(
  408. id__in=NullableIntegerArrayModel.objects.filter(field__len=3)
  409. ),
  410. [self.objs[3]]
  411. )
  412. def test_enum_lookup(self):
  413. class TestEnum(enum.Enum):
  414. VALUE_1 = 'value_1'
  415. instance = ArrayEnumModel.objects.create(array_of_enums=[TestEnum.VALUE_1])
  416. self.assertSequenceEqual(
  417. ArrayEnumModel.objects.filter(array_of_enums__contains=[TestEnum.VALUE_1]),
  418. [instance]
  419. )
  420. def test_unsupported_lookup(self):
  421. msg = "Unsupported lookup '0_bar' for ArrayField or join on the field not permitted."
  422. with self.assertRaisesMessage(FieldError, msg):
  423. list(NullableIntegerArrayModel.objects.filter(field__0_bar=[2]))
  424. msg = "Unsupported lookup '0bar' for ArrayField or join on the field not permitted."
  425. with self.assertRaisesMessage(FieldError, msg):
  426. list(NullableIntegerArrayModel.objects.filter(field__0bar=[2]))
  427. def test_grouping_by_annotations_with_array_field_param(self):
  428. value = models.Value([1], output_field=ArrayField(models.IntegerField()))
  429. self.assertEqual(
  430. NullableIntegerArrayModel.objects.annotate(
  431. array_length=models.Func(
  432. value, 1, function='ARRAY_LENGTH', output_field=models.IntegerField(),
  433. ),
  434. ).values('array_length').annotate(
  435. count=models.Count('pk'),
  436. ).get()['array_length'],
  437. 1,
  438. )
  439. class TestDateTimeExactQuerying(PostgreSQLTestCase):
  440. @classmethod
  441. def setUpTestData(cls):
  442. now = timezone.now()
  443. cls.datetimes = [now]
  444. cls.dates = [now.date()]
  445. cls.times = [now.time()]
  446. cls.objs = [
  447. DateTimeArrayModel.objects.create(datetimes=cls.datetimes, dates=cls.dates, times=cls.times),
  448. ]
  449. def test_exact_datetimes(self):
  450. self.assertSequenceEqual(
  451. DateTimeArrayModel.objects.filter(datetimes=self.datetimes),
  452. self.objs
  453. )
  454. def test_exact_dates(self):
  455. self.assertSequenceEqual(
  456. DateTimeArrayModel.objects.filter(dates=self.dates),
  457. self.objs
  458. )
  459. def test_exact_times(self):
  460. self.assertSequenceEqual(
  461. DateTimeArrayModel.objects.filter(times=self.times),
  462. self.objs
  463. )
  464. class TestOtherTypesExactQuerying(PostgreSQLTestCase):
  465. @classmethod
  466. def setUpTestData(cls):
  467. cls.ips = ['192.168.0.1', '::1']
  468. cls.uuids = [uuid.uuid4()]
  469. cls.decimals = [decimal.Decimal(1.25), 1.75]
  470. cls.tags = [Tag(1), Tag(2), Tag(3)]
  471. cls.objs = [
  472. OtherTypesArrayModel.objects.create(
  473. ips=cls.ips,
  474. uuids=cls.uuids,
  475. decimals=cls.decimals,
  476. tags=cls.tags,
  477. )
  478. ]
  479. def test_exact_ip_addresses(self):
  480. self.assertSequenceEqual(
  481. OtherTypesArrayModel.objects.filter(ips=self.ips),
  482. self.objs
  483. )
  484. def test_exact_uuids(self):
  485. self.assertSequenceEqual(
  486. OtherTypesArrayModel.objects.filter(uuids=self.uuids),
  487. self.objs
  488. )
  489. def test_exact_decimals(self):
  490. self.assertSequenceEqual(
  491. OtherTypesArrayModel.objects.filter(decimals=self.decimals),
  492. self.objs
  493. )
  494. def test_exact_tags(self):
  495. self.assertSequenceEqual(
  496. OtherTypesArrayModel.objects.filter(tags=self.tags),
  497. self.objs
  498. )
  499. @isolate_apps('postgres_tests')
  500. class TestChecks(PostgreSQLSimpleTestCase):
  501. def test_field_checks(self):
  502. class MyModel(PostgreSQLModel):
  503. field = ArrayField(models.CharField())
  504. model = MyModel()
  505. errors = model.check()
  506. self.assertEqual(len(errors), 1)
  507. # The inner CharField is missing a max_length.
  508. self.assertEqual(errors[0].id, 'postgres.E001')
  509. self.assertIn('max_length', errors[0].msg)
  510. def test_invalid_base_fields(self):
  511. class MyModel(PostgreSQLModel):
  512. field = ArrayField(models.ManyToManyField('postgres_tests.IntegerArrayModel'))
  513. model = MyModel()
  514. errors = model.check()
  515. self.assertEqual(len(errors), 1)
  516. self.assertEqual(errors[0].id, 'postgres.E002')
  517. def test_invalid_default(self):
  518. class MyModel(PostgreSQLModel):
  519. field = ArrayField(models.IntegerField(), default=[])
  520. model = MyModel()
  521. self.assertEqual(model.check(), [
  522. checks.Warning(
  523. msg=(
  524. "ArrayField default should be a callable instead of an "
  525. "instance so that it's not shared between all field "
  526. "instances."
  527. ),
  528. hint='Use a callable instead, e.g., use `list` instead of `[]`.',
  529. obj=MyModel._meta.get_field('field'),
  530. id='fields.E010',
  531. )
  532. ])
  533. def test_valid_default(self):
  534. class MyModel(PostgreSQLModel):
  535. field = ArrayField(models.IntegerField(), default=list)
  536. model = MyModel()
  537. self.assertEqual(model.check(), [])
  538. def test_valid_default_none(self):
  539. class MyModel(PostgreSQLModel):
  540. field = ArrayField(models.IntegerField(), default=None)
  541. model = MyModel()
  542. self.assertEqual(model.check(), [])
  543. def test_nested_field_checks(self):
  544. """
  545. Nested ArrayFields are permitted.
  546. """
  547. class MyModel(PostgreSQLModel):
  548. field = ArrayField(ArrayField(models.CharField()))
  549. model = MyModel()
  550. errors = model.check()
  551. self.assertEqual(len(errors), 1)
  552. # The inner CharField is missing a max_length.
  553. self.assertEqual(errors[0].id, 'postgres.E001')
  554. self.assertIn('max_length', errors[0].msg)
  555. def test_choices_tuple_list(self):
  556. class MyModel(PostgreSQLModel):
  557. field = ArrayField(
  558. models.CharField(max_length=16),
  559. choices=[
  560. [
  561. 'Media',
  562. [(['vinyl', 'cd'], 'Audio'), (('vhs', 'dvd'), 'Video')],
  563. ],
  564. (['mp3', 'mp4'], 'Digital'),
  565. ],
  566. )
  567. self.assertEqual(MyModel._meta.get_field('field').check(), [])
  568. @unittest.skipUnless(connection.vendor == 'postgresql', "PostgreSQL specific tests")
  569. class TestMigrations(TransactionTestCase):
  570. available_apps = ['postgres_tests']
  571. def test_deconstruct(self):
  572. field = ArrayField(models.IntegerField())
  573. name, path, args, kwargs = field.deconstruct()
  574. new = ArrayField(*args, **kwargs)
  575. self.assertEqual(type(new.base_field), type(field.base_field))
  576. self.assertIsNot(new.base_field, field.base_field)
  577. def test_deconstruct_with_size(self):
  578. field = ArrayField(models.IntegerField(), size=3)
  579. name, path, args, kwargs = field.deconstruct()
  580. new = ArrayField(*args, **kwargs)
  581. self.assertEqual(new.size, field.size)
  582. def test_deconstruct_args(self):
  583. field = ArrayField(models.CharField(max_length=20))
  584. name, path, args, kwargs = field.deconstruct()
  585. new = ArrayField(*args, **kwargs)
  586. self.assertEqual(new.base_field.max_length, field.base_field.max_length)
  587. def test_subclass_deconstruct(self):
  588. field = ArrayField(models.IntegerField())
  589. name, path, args, kwargs = field.deconstruct()
  590. self.assertEqual(path, 'django.contrib.postgres.fields.ArrayField')
  591. field = ArrayFieldSubclass()
  592. name, path, args, kwargs = field.deconstruct()
  593. self.assertEqual(path, 'postgres_tests.models.ArrayFieldSubclass')
  594. @override_settings(MIGRATION_MODULES={
  595. "postgres_tests": "postgres_tests.array_default_migrations",
  596. })
  597. def test_adding_field_with_default(self):
  598. # See #22962
  599. table_name = 'postgres_tests_integerarraydefaultmodel'
  600. with connection.cursor() as cursor:
  601. self.assertNotIn(table_name, connection.introspection.table_names(cursor))
  602. call_command('migrate', 'postgres_tests', verbosity=0)
  603. with connection.cursor() as cursor:
  604. self.assertIn(table_name, connection.introspection.table_names(cursor))
  605. call_command('migrate', 'postgres_tests', 'zero', verbosity=0)
  606. with connection.cursor() as cursor:
  607. self.assertNotIn(table_name, connection.introspection.table_names(cursor))
  608. @override_settings(MIGRATION_MODULES={
  609. "postgres_tests": "postgres_tests.array_index_migrations",
  610. })
  611. def test_adding_arrayfield_with_index(self):
  612. """
  613. ArrayField shouldn't have varchar_patterns_ops or text_patterns_ops indexes.
  614. """
  615. table_name = 'postgres_tests_chartextarrayindexmodel'
  616. call_command('migrate', 'postgres_tests', verbosity=0)
  617. with connection.cursor() as cursor:
  618. like_constraint_columns_list = [
  619. v['columns']
  620. for k, v in list(connection.introspection.get_constraints(cursor, table_name).items())
  621. if k.endswith('_like')
  622. ]
  623. # Only the CharField should have a LIKE index.
  624. self.assertEqual(like_constraint_columns_list, [['char2']])
  625. # All fields should have regular indexes.
  626. with connection.cursor() as cursor:
  627. indexes = [
  628. c['columns'][0]
  629. for c in connection.introspection.get_constraints(cursor, table_name).values()
  630. if c['index'] and len(c['columns']) == 1
  631. ]
  632. self.assertIn('char', indexes)
  633. self.assertIn('char2', indexes)
  634. self.assertIn('text', indexes)
  635. call_command('migrate', 'postgres_tests', 'zero', verbosity=0)
  636. with connection.cursor() as cursor:
  637. self.assertNotIn(table_name, connection.introspection.table_names(cursor))
  638. class TestSerialization(PostgreSQLSimpleTestCase):
  639. test_data = (
  640. '[{"fields": {"field": "[\\"1\\", \\"2\\", null]"}, "model": "postgres_tests.integerarraymodel", "pk": null}]'
  641. )
  642. def test_dumping(self):
  643. instance = IntegerArrayModel(field=[1, 2, None])
  644. data = serializers.serialize('json', [instance])
  645. self.assertEqual(json.loads(data), json.loads(self.test_data))
  646. def test_loading(self):
  647. instance = list(serializers.deserialize('json', self.test_data))[0].object
  648. self.assertEqual(instance.field, [1, 2, None])
  649. class TestValidation(PostgreSQLSimpleTestCase):
  650. def test_unbounded(self):
  651. field = ArrayField(models.IntegerField())
  652. with self.assertRaises(exceptions.ValidationError) as cm:
  653. field.clean([1, None], None)
  654. self.assertEqual(cm.exception.code, 'item_invalid')
  655. self.assertEqual(
  656. cm.exception.message % cm.exception.params,
  657. 'Item 2 in the array did not validate: This field cannot be null.'
  658. )
  659. def test_blank_true(self):
  660. field = ArrayField(models.IntegerField(blank=True, null=True))
  661. # This should not raise a validation error
  662. field.clean([1, None], None)
  663. def test_with_size(self):
  664. field = ArrayField(models.IntegerField(), size=3)
  665. field.clean([1, 2, 3], None)
  666. with self.assertRaises(exceptions.ValidationError) as cm:
  667. field.clean([1, 2, 3, 4], None)
  668. self.assertEqual(cm.exception.messages[0], 'List contains 4 items, it should contain no more than 3.')
  669. def test_nested_array_mismatch(self):
  670. field = ArrayField(ArrayField(models.IntegerField()))
  671. field.clean([[1, 2], [3, 4]], None)
  672. with self.assertRaises(exceptions.ValidationError) as cm:
  673. field.clean([[1, 2], [3, 4, 5]], None)
  674. self.assertEqual(cm.exception.code, 'nested_array_mismatch')
  675. self.assertEqual(cm.exception.messages[0], 'Nested arrays must have the same length.')
  676. def test_with_base_field_error_params(self):
  677. field = ArrayField(models.CharField(max_length=2))
  678. with self.assertRaises(exceptions.ValidationError) as cm:
  679. field.clean(['abc'], None)
  680. self.assertEqual(len(cm.exception.error_list), 1)
  681. exception = cm.exception.error_list[0]
  682. self.assertEqual(
  683. exception.message,
  684. 'Item 1 in the array did not validate: Ensure this value has at most 2 characters (it has 3).'
  685. )
  686. self.assertEqual(exception.code, 'item_invalid')
  687. self.assertEqual(exception.params, {'nth': 1, 'value': 'abc', 'limit_value': 2, 'show_value': 3})
  688. def test_with_validators(self):
  689. field = ArrayField(models.IntegerField(validators=[validators.MinValueValidator(1)]))
  690. field.clean([1, 2], None)
  691. with self.assertRaises(exceptions.ValidationError) as cm:
  692. field.clean([0], None)
  693. self.assertEqual(len(cm.exception.error_list), 1)
  694. exception = cm.exception.error_list[0]
  695. self.assertEqual(
  696. exception.message,
  697. 'Item 1 in the array did not validate: Ensure this value is greater than or equal to 1.'
  698. )
  699. self.assertEqual(exception.code, 'item_invalid')
  700. self.assertEqual(exception.params, {'nth': 1, 'value': 0, 'limit_value': 1, 'show_value': 0})
  701. class TestSimpleFormField(PostgreSQLSimpleTestCase):
  702. def test_valid(self):
  703. field = SimpleArrayField(forms.CharField())
  704. value = field.clean('a,b,c')
  705. self.assertEqual(value, ['a', 'b', 'c'])
  706. def test_to_python_fail(self):
  707. field = SimpleArrayField(forms.IntegerField())
  708. with self.assertRaises(exceptions.ValidationError) as cm:
  709. field.clean('a,b,9')
  710. self.assertEqual(cm.exception.messages[0], 'Item 1 in the array did not validate: Enter a whole number.')
  711. def test_validate_fail(self):
  712. field = SimpleArrayField(forms.CharField(required=True))
  713. with self.assertRaises(exceptions.ValidationError) as cm:
  714. field.clean('a,b,')
  715. self.assertEqual(cm.exception.messages[0], 'Item 3 in the array did not validate: This field is required.')
  716. def test_validate_fail_base_field_error_params(self):
  717. field = SimpleArrayField(forms.CharField(max_length=2))
  718. with self.assertRaises(exceptions.ValidationError) as cm:
  719. field.clean('abc,c,defg')
  720. errors = cm.exception.error_list
  721. self.assertEqual(len(errors), 2)
  722. first_error = errors[0]
  723. self.assertEqual(
  724. first_error.message,
  725. 'Item 1 in the array did not validate: Ensure this value has at most 2 characters (it has 3).'
  726. )
  727. self.assertEqual(first_error.code, 'item_invalid')
  728. self.assertEqual(first_error.params, {'nth': 1, 'value': 'abc', 'limit_value': 2, 'show_value': 3})
  729. second_error = errors[1]
  730. self.assertEqual(
  731. second_error.message,
  732. 'Item 3 in the array did not validate: Ensure this value has at most 2 characters (it has 4).'
  733. )
  734. self.assertEqual(second_error.code, 'item_invalid')
  735. self.assertEqual(second_error.params, {'nth': 3, 'value': 'defg', 'limit_value': 2, 'show_value': 4})
  736. def test_validators_fail(self):
  737. field = SimpleArrayField(forms.RegexField('[a-e]{2}'))
  738. with self.assertRaises(exceptions.ValidationError) as cm:
  739. field.clean('a,bc,de')
  740. self.assertEqual(cm.exception.messages[0], 'Item 1 in the array did not validate: Enter a valid value.')
  741. def test_delimiter(self):
  742. field = SimpleArrayField(forms.CharField(), delimiter='|')
  743. value = field.clean('a|b|c')
  744. self.assertEqual(value, ['a', 'b', 'c'])
  745. def test_delimiter_with_nesting(self):
  746. field = SimpleArrayField(SimpleArrayField(forms.CharField()), delimiter='|')
  747. value = field.clean('a,b|c,d')
  748. self.assertEqual(value, [['a', 'b'], ['c', 'd']])
  749. def test_prepare_value(self):
  750. field = SimpleArrayField(forms.CharField())
  751. value = field.prepare_value(['a', 'b', 'c'])
  752. self.assertEqual(value, 'a,b,c')
  753. def test_max_length(self):
  754. field = SimpleArrayField(forms.CharField(), max_length=2)
  755. with self.assertRaises(exceptions.ValidationError) as cm:
  756. field.clean('a,b,c')
  757. self.assertEqual(cm.exception.messages[0], 'List contains 3 items, it should contain no more than 2.')
  758. def test_min_length(self):
  759. field = SimpleArrayField(forms.CharField(), min_length=4)
  760. with self.assertRaises(exceptions.ValidationError) as cm:
  761. field.clean('a,b,c')
  762. self.assertEqual(cm.exception.messages[0], 'List contains 3 items, it should contain no fewer than 4.')
  763. def test_required(self):
  764. field = SimpleArrayField(forms.CharField(), required=True)
  765. with self.assertRaises(exceptions.ValidationError) as cm:
  766. field.clean('')
  767. self.assertEqual(cm.exception.messages[0], 'This field is required.')
  768. def test_model_field_formfield(self):
  769. model_field = ArrayField(models.CharField(max_length=27))
  770. form_field = model_field.formfield()
  771. self.assertIsInstance(form_field, SimpleArrayField)
  772. self.assertIsInstance(form_field.base_field, forms.CharField)
  773. self.assertEqual(form_field.base_field.max_length, 27)
  774. def test_model_field_formfield_size(self):
  775. model_field = ArrayField(models.CharField(max_length=27), size=4)
  776. form_field = model_field.formfield()
  777. self.assertIsInstance(form_field, SimpleArrayField)
  778. self.assertEqual(form_field.max_length, 4)
  779. def test_model_field_choices(self):
  780. model_field = ArrayField(models.IntegerField(choices=((1, 'A'), (2, 'B'))))
  781. form_field = model_field.formfield()
  782. self.assertEqual(form_field.clean('1,2'), [1, 2])
  783. def test_already_converted_value(self):
  784. field = SimpleArrayField(forms.CharField())
  785. vals = ['a', 'b', 'c']
  786. self.assertEqual(field.clean(vals), vals)
  787. def test_has_changed(self):
  788. field = SimpleArrayField(forms.IntegerField())
  789. self.assertIs(field.has_changed([1, 2], [1, 2]), False)
  790. self.assertIs(field.has_changed([1, 2], '1,2'), False)
  791. self.assertIs(field.has_changed([1, 2], '1,2,3'), True)
  792. self.assertIs(field.has_changed([1, 2], 'a,b'), True)
  793. def test_has_changed_empty(self):
  794. field = SimpleArrayField(forms.CharField())
  795. self.assertIs(field.has_changed(None, None), False)
  796. self.assertIs(field.has_changed(None, ''), False)
  797. self.assertIs(field.has_changed(None, []), False)
  798. self.assertIs(field.has_changed([], None), False)
  799. self.assertIs(field.has_changed([], ''), False)
  800. class TestSplitFormField(PostgreSQLSimpleTestCase):
  801. def test_valid(self):
  802. class SplitForm(forms.Form):
  803. array = SplitArrayField(forms.CharField(), size=3)
  804. data = {'array_0': 'a', 'array_1': 'b', 'array_2': 'c'}
  805. form = SplitForm(data)
  806. self.assertTrue(form.is_valid())
  807. self.assertEqual(form.cleaned_data, {'array': ['a', 'b', 'c']})
  808. def test_required(self):
  809. class SplitForm(forms.Form):
  810. array = SplitArrayField(forms.CharField(), required=True, size=3)
  811. data = {'array_0': '', 'array_1': '', 'array_2': ''}
  812. form = SplitForm(data)
  813. self.assertFalse(form.is_valid())
  814. self.assertEqual(form.errors, {'array': ['This field is required.']})
  815. def test_remove_trailing_nulls(self):
  816. class SplitForm(forms.Form):
  817. array = SplitArrayField(forms.CharField(required=False), size=5, remove_trailing_nulls=True)
  818. data = {'array_0': 'a', 'array_1': '', 'array_2': 'b', 'array_3': '', 'array_4': ''}
  819. form = SplitForm(data)
  820. self.assertTrue(form.is_valid(), form.errors)
  821. self.assertEqual(form.cleaned_data, {'array': ['a', '', 'b']})
  822. def test_remove_trailing_nulls_not_required(self):
  823. class SplitForm(forms.Form):
  824. array = SplitArrayField(
  825. forms.CharField(required=False),
  826. size=2,
  827. remove_trailing_nulls=True,
  828. required=False,
  829. )
  830. data = {'array_0': '', 'array_1': ''}
  831. form = SplitForm(data)
  832. self.assertTrue(form.is_valid())
  833. self.assertEqual(form.cleaned_data, {'array': []})
  834. def test_required_field(self):
  835. class SplitForm(forms.Form):
  836. array = SplitArrayField(forms.CharField(), size=3)
  837. data = {'array_0': 'a', 'array_1': 'b', 'array_2': ''}
  838. form = SplitForm(data)
  839. self.assertFalse(form.is_valid())
  840. self.assertEqual(form.errors, {'array': ['Item 3 in the array did not validate: This field is required.']})
  841. def test_invalid_integer(self):
  842. msg = 'Item 2 in the array did not validate: Ensure this value is less than or equal to 100.'
  843. with self.assertRaisesMessage(exceptions.ValidationError, msg):
  844. SplitArrayField(forms.IntegerField(max_value=100), size=2).clean([0, 101])
  845. # To locate the widget's template.
  846. @modify_settings(INSTALLED_APPS={'append': 'django.contrib.postgres'})
  847. def test_rendering(self):
  848. class SplitForm(forms.Form):
  849. array = SplitArrayField(forms.CharField(), size=3)
  850. self.assertHTMLEqual(str(SplitForm()), '''
  851. <tr>
  852. <th><label for="id_array_0">Array:</label></th>
  853. <td>
  854. <input id="id_array_0" name="array_0" type="text" required>
  855. <input id="id_array_1" name="array_1" type="text" required>
  856. <input id="id_array_2" name="array_2" type="text" required>
  857. </td>
  858. </tr>
  859. ''')
  860. def test_invalid_char_length(self):
  861. field = SplitArrayField(forms.CharField(max_length=2), size=3)
  862. with self.assertRaises(exceptions.ValidationError) as cm:
  863. field.clean(['abc', 'c', 'defg'])
  864. self.assertEqual(cm.exception.messages, [
  865. 'Item 1 in the array did not validate: Ensure this value has at most 2 characters (it has 3).',
  866. 'Item 3 in the array did not validate: Ensure this value has at most 2 characters (it has 4).',
  867. ])
  868. def test_splitarraywidget_value_omitted_from_data(self):
  869. class Form(forms.ModelForm):
  870. field = SplitArrayField(forms.IntegerField(), required=False, size=2)
  871. class Meta:
  872. model = IntegerArrayModel
  873. fields = ('field',)
  874. form = Form({'field_0': '1', 'field_1': '2'})
  875. self.assertEqual(form.errors, {})
  876. obj = form.save(commit=False)
  877. self.assertEqual(obj.field, [1, 2])
  878. def test_splitarrayfield_has_changed(self):
  879. class Form(forms.ModelForm):
  880. field = SplitArrayField(forms.IntegerField(), required=False, size=2)
  881. class Meta:
  882. model = IntegerArrayModel
  883. fields = ('field',)
  884. tests = [
  885. ({}, {'field_0': '', 'field_1': ''}, True),
  886. ({'field': None}, {'field_0': '', 'field_1': ''}, True),
  887. ({'field': [1]}, {'field_0': '', 'field_1': ''}, True),
  888. ({'field': [1]}, {'field_0': '1', 'field_1': '0'}, True),
  889. ({'field': [1, 2]}, {'field_0': '1', 'field_1': '2'}, False),
  890. ({'field': [1, 2]}, {'field_0': 'a', 'field_1': 'b'}, True),
  891. ]
  892. for initial, data, expected_result in tests:
  893. with self.subTest(initial=initial, data=data):
  894. obj = IntegerArrayModel(**initial)
  895. form = Form(data, instance=obj)
  896. self.assertIs(form.has_changed(), expected_result)
  897. def test_splitarrayfield_remove_trailing_nulls_has_changed(self):
  898. class Form(forms.ModelForm):
  899. field = SplitArrayField(forms.IntegerField(), required=False, size=2, remove_trailing_nulls=True)
  900. class Meta:
  901. model = IntegerArrayModel
  902. fields = ('field',)
  903. tests = [
  904. ({}, {'field_0': '', 'field_1': ''}, False),
  905. ({'field': None}, {'field_0': '', 'field_1': ''}, False),
  906. ({'field': []}, {'field_0': '', 'field_1': ''}, False),
  907. ({'field': [1]}, {'field_0': '1', 'field_1': ''}, False),
  908. ]
  909. for initial, data, expected_result in tests:
  910. with self.subTest(initial=initial, data=data):
  911. obj = IntegerArrayModel(**initial)
  912. form = Form(data, instance=obj)
  913. self.assertIs(form.has_changed(), expected_result)
  914. class TestSplitFormWidget(PostgreSQLWidgetTestCase):
  915. def test_get_context(self):
  916. self.assertEqual(
  917. SplitArrayWidget(forms.TextInput(), size=2).get_context('name', ['val1', 'val2']),
  918. {
  919. 'widget': {
  920. 'name': 'name',
  921. 'is_hidden': False,
  922. 'required': False,
  923. 'value': "['val1', 'val2']",
  924. 'attrs': {},
  925. 'template_name': 'postgres/widgets/split_array.html',
  926. 'subwidgets': [
  927. {
  928. 'name': 'name_0',
  929. 'is_hidden': False,
  930. 'required': False,
  931. 'value': 'val1',
  932. 'attrs': {},
  933. 'template_name': 'django/forms/widgets/text.html',
  934. 'type': 'text',
  935. },
  936. {
  937. 'name': 'name_1',
  938. 'is_hidden': False,
  939. 'required': False,
  940. 'value': 'val2',
  941. 'attrs': {},
  942. 'template_name': 'django/forms/widgets/text.html',
  943. 'type': 'text',
  944. },
  945. ]
  946. }
  947. }
  948. )
  949. def test_checkbox_get_context_attrs(self):
  950. context = SplitArrayWidget(
  951. forms.CheckboxInput(),
  952. size=2,
  953. ).get_context('name', [True, False])
  954. self.assertEqual(context['widget']['value'], '[True, False]')
  955. self.assertEqual(
  956. [subwidget['attrs'] for subwidget in context['widget']['subwidgets']],
  957. [{'checked': True}, {}]
  958. )
  959. def test_render(self):
  960. self.check_html(
  961. SplitArrayWidget(forms.TextInput(), size=2), 'array', None,
  962. """
  963. <input name="array_0" type="text">
  964. <input name="array_1" type="text">
  965. """
  966. )
  967. def test_render_attrs(self):
  968. self.check_html(
  969. SplitArrayWidget(forms.TextInput(), size=2),
  970. 'array', ['val1', 'val2'], attrs={'id': 'foo'},
  971. html=(
  972. """
  973. <input id="foo_0" name="array_0" type="text" value="val1">
  974. <input id="foo_1" name="array_1" type="text" value="val2">
  975. """
  976. )
  977. )
  978. def test_value_omitted_from_data(self):
  979. widget = SplitArrayWidget(forms.TextInput(), size=2)
  980. self.assertIs(widget.value_omitted_from_data({}, {}, 'field'), True)
  981. self.assertIs(widget.value_omitted_from_data({'field_0': 'value'}, {}, 'field'), False)
  982. self.assertIs(widget.value_omitted_from_data({'field_1': 'value'}, {}, 'field'), False)
  983. self.assertIs(widget.value_omitted_from_data({'field_0': 'value', 'field_1': 'value'}, {}, 'field'), False)