test_autocomplete_view.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572
  1. import datetime
  2. import json
  3. from contextlib import contextmanager
  4. from django.contrib import admin
  5. from django.contrib.admin.tests import AdminSeleniumTestCase
  6. from django.contrib.admin.views.autocomplete import AutocompleteJsonView
  7. from django.contrib.auth.models import Permission, User
  8. from django.contrib.contenttypes.models import ContentType
  9. from django.core.exceptions import PermissionDenied
  10. from django.http import Http404
  11. from django.test import RequestFactory, override_settings
  12. from django.urls import reverse, reverse_lazy
  13. from .admin import AnswerAdmin, QuestionAdmin
  14. from .models import (
  15. Answer,
  16. Author,
  17. Authorship,
  18. Bonus,
  19. Book,
  20. Employee,
  21. Manager,
  22. Parent,
  23. PKChild,
  24. Question,
  25. Toy,
  26. WorkHour,
  27. )
  28. from .tests import AdminViewBasicTestCase
  29. PAGINATOR_SIZE = AutocompleteJsonView.paginate_by
  30. class AuthorAdmin(admin.ModelAdmin):
  31. ordering = ["id"]
  32. search_fields = ["id"]
  33. class AuthorshipInline(admin.TabularInline):
  34. model = Authorship
  35. autocomplete_fields = ["author"]
  36. class BookAdmin(admin.ModelAdmin):
  37. inlines = [AuthorshipInline]
  38. site = admin.AdminSite(name="autocomplete_admin")
  39. site.register(Question, QuestionAdmin)
  40. site.register(Answer, AnswerAdmin)
  41. site.register(Author, AuthorAdmin)
  42. site.register(Book, BookAdmin)
  43. site.register(Employee, search_fields=["name"])
  44. site.register(WorkHour, autocomplete_fields=["employee"])
  45. site.register(Manager, search_fields=["name"])
  46. site.register(Bonus, autocomplete_fields=["recipient"])
  47. site.register(PKChild, search_fields=["name"])
  48. site.register(Toy, autocomplete_fields=["child"])
  49. @contextmanager
  50. def model_admin(model, model_admin, admin_site=site):
  51. org_admin = admin_site._registry.get(model)
  52. if org_admin:
  53. admin_site.unregister(model)
  54. admin_site.register(model, model_admin)
  55. try:
  56. yield
  57. finally:
  58. if org_admin:
  59. admin_site._registry[model] = org_admin
  60. class AutocompleteJsonViewTests(AdminViewBasicTestCase):
  61. as_view_args = {"admin_site": site}
  62. opts = {
  63. "app_label": Answer._meta.app_label,
  64. "model_name": Answer._meta.model_name,
  65. "field_name": "question",
  66. }
  67. factory = RequestFactory()
  68. url = reverse_lazy("autocomplete_admin:autocomplete")
  69. @classmethod
  70. def setUpTestData(cls):
  71. cls.user = User.objects.create_user(
  72. username="user",
  73. password="secret",
  74. email="user@example.com",
  75. is_staff=True,
  76. )
  77. super().setUpTestData()
  78. def test_success(self):
  79. q = Question.objects.create(question="Is this a question?")
  80. request = self.factory.get(self.url, {"term": "is", **self.opts})
  81. request.user = self.superuser
  82. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  83. self.assertEqual(response.status_code, 200)
  84. data = json.loads(response.content.decode("utf-8"))
  85. self.assertEqual(
  86. data,
  87. {
  88. "results": [{"id": str(q.pk), "text": q.question}],
  89. "pagination": {"more": False},
  90. },
  91. )
  92. def test_custom_to_field(self):
  93. q = Question.objects.create(question="Is this a question?")
  94. request = self.factory.get(
  95. self.url,
  96. {"term": "is", **self.opts, "field_name": "question_with_to_field"},
  97. )
  98. request.user = self.superuser
  99. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  100. self.assertEqual(response.status_code, 200)
  101. data = json.loads(response.content.decode("utf-8"))
  102. self.assertEqual(
  103. data,
  104. {
  105. "results": [{"id": str(q.uuid), "text": q.question}],
  106. "pagination": {"more": False},
  107. },
  108. )
  109. def test_custom_to_field_permission_denied(self):
  110. Question.objects.create(question="Is this a question?")
  111. request = self.factory.get(
  112. self.url,
  113. {"term": "is", **self.opts, "field_name": "question_with_to_field"},
  114. )
  115. request.user = self.user
  116. with self.assertRaises(PermissionDenied):
  117. AutocompleteJsonView.as_view(**self.as_view_args)(request)
  118. def test_custom_to_field_custom_pk(self):
  119. q = Question.objects.create(question="Is this a question?")
  120. opts = {
  121. "app_label": Question._meta.app_label,
  122. "model_name": Question._meta.model_name,
  123. "field_name": "related_questions",
  124. }
  125. request = self.factory.get(self.url, {"term": "is", **opts})
  126. request.user = self.superuser
  127. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  128. self.assertEqual(response.status_code, 200)
  129. data = json.loads(response.content.decode("utf-8"))
  130. self.assertEqual(
  131. data,
  132. {
  133. "results": [{"id": str(q.big_id), "text": q.question}],
  134. "pagination": {"more": False},
  135. },
  136. )
  137. def test_to_field_resolution_with_mti(self):
  138. """
  139. to_field resolution should correctly resolve for target models using
  140. MTI. Tests for single and multi-level cases.
  141. """
  142. tests = [
  143. (Employee, WorkHour, "employee"),
  144. (Manager, Bonus, "recipient"),
  145. ]
  146. for Target, Remote, related_name in tests:
  147. with self.subTest(
  148. target_model=Target, remote_model=Remote, related_name=related_name
  149. ):
  150. o = Target.objects.create(
  151. name="Frida Kahlo", gender=2, code="painter", alive=False
  152. )
  153. opts = {
  154. "app_label": Remote._meta.app_label,
  155. "model_name": Remote._meta.model_name,
  156. "field_name": related_name,
  157. }
  158. request = self.factory.get(self.url, {"term": "frida", **opts})
  159. request.user = self.superuser
  160. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  161. self.assertEqual(response.status_code, 200)
  162. data = json.loads(response.content.decode("utf-8"))
  163. self.assertEqual(
  164. data,
  165. {
  166. "results": [{"id": str(o.pk), "text": o.name}],
  167. "pagination": {"more": False},
  168. },
  169. )
  170. def test_to_field_resolution_with_fk_pk(self):
  171. p = Parent.objects.create(name="Bertie")
  172. c = PKChild.objects.create(parent=p, name="Anna")
  173. opts = {
  174. "app_label": Toy._meta.app_label,
  175. "model_name": Toy._meta.model_name,
  176. "field_name": "child",
  177. }
  178. request = self.factory.get(self.url, {"term": "anna", **opts})
  179. request.user = self.superuser
  180. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  181. self.assertEqual(response.status_code, 200)
  182. data = json.loads(response.content.decode("utf-8"))
  183. self.assertEqual(
  184. data,
  185. {
  186. "results": [{"id": str(c.pk), "text": c.name}],
  187. "pagination": {"more": False},
  188. },
  189. )
  190. def test_field_does_not_exist(self):
  191. request = self.factory.get(
  192. self.url, {"term": "is", **self.opts, "field_name": "does_not_exist"}
  193. )
  194. request.user = self.superuser
  195. with self.assertRaises(PermissionDenied):
  196. AutocompleteJsonView.as_view(**self.as_view_args)(request)
  197. def test_field_no_related_field(self):
  198. request = self.factory.get(
  199. self.url, {"term": "is", **self.opts, "field_name": "answer"}
  200. )
  201. request.user = self.superuser
  202. with self.assertRaises(PermissionDenied):
  203. AutocompleteJsonView.as_view(**self.as_view_args)(request)
  204. def test_field_does_not_allowed(self):
  205. request = self.factory.get(
  206. self.url, {"term": "is", **self.opts, "field_name": "related_questions"}
  207. )
  208. request.user = self.superuser
  209. with self.assertRaises(PermissionDenied):
  210. AutocompleteJsonView.as_view(**self.as_view_args)(request)
  211. def test_limit_choices_to(self):
  212. # Answer.question_with_to_field defines limit_choices_to to "those not
  213. # starting with 'not'".
  214. q = Question.objects.create(question="Is this a question?")
  215. Question.objects.create(question="Not a question.")
  216. request = self.factory.get(
  217. self.url,
  218. {"term": "is", **self.opts, "field_name": "question_with_to_field"},
  219. )
  220. request.user = self.superuser
  221. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  222. self.assertEqual(response.status_code, 200)
  223. data = json.loads(response.content.decode("utf-8"))
  224. self.assertEqual(
  225. data,
  226. {
  227. "results": [{"id": str(q.uuid), "text": q.question}],
  228. "pagination": {"more": False},
  229. },
  230. )
  231. def test_must_be_logged_in(self):
  232. response = self.client.get(self.url, {"term": "", **self.opts})
  233. self.assertEqual(response.status_code, 200)
  234. self.client.logout()
  235. response = self.client.get(self.url, {"term": "", **self.opts})
  236. self.assertEqual(response.status_code, 302)
  237. def test_has_view_or_change_permission_required(self):
  238. """
  239. Users require the change permission for the related model to the
  240. autocomplete view for it.
  241. """
  242. request = self.factory.get(self.url, {"term": "is", **self.opts})
  243. request.user = self.user
  244. with self.assertRaises(PermissionDenied):
  245. AutocompleteJsonView.as_view(**self.as_view_args)(request)
  246. for permission in ("view", "change"):
  247. with self.subTest(permission=permission):
  248. self.user.user_permissions.clear()
  249. p = Permission.objects.get(
  250. content_type=ContentType.objects.get_for_model(Question),
  251. codename="%s_question" % permission,
  252. )
  253. self.user.user_permissions.add(p)
  254. request.user = User.objects.get(pk=self.user.pk)
  255. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  256. self.assertEqual(response.status_code, 200)
  257. def test_search_use_distinct(self):
  258. """
  259. Searching across model relations use QuerySet.distinct() to avoid
  260. duplicates.
  261. """
  262. q1 = Question.objects.create(question="question 1")
  263. q2 = Question.objects.create(question="question 2")
  264. q2.related_questions.add(q1)
  265. q3 = Question.objects.create(question="question 3")
  266. q3.related_questions.add(q1)
  267. request = self.factory.get(self.url, {"term": "question", **self.opts})
  268. request.user = self.superuser
  269. class DistinctQuestionAdmin(QuestionAdmin):
  270. search_fields = ["related_questions__question", "question"]
  271. with model_admin(Question, DistinctQuestionAdmin):
  272. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  273. self.assertEqual(response.status_code, 200)
  274. data = json.loads(response.content.decode("utf-8"))
  275. self.assertEqual(len(data["results"]), 3)
  276. def test_missing_search_fields(self):
  277. class EmptySearchAdmin(QuestionAdmin):
  278. search_fields = []
  279. with model_admin(Question, EmptySearchAdmin):
  280. msg = "EmptySearchAdmin must have search_fields for the autocomplete_view."
  281. with self.assertRaisesMessage(Http404, msg):
  282. site.autocomplete_view(
  283. self.factory.get(self.url, {"term": "", **self.opts})
  284. )
  285. def test_get_paginator(self):
  286. """Search results are paginated."""
  287. class PKOrderingQuestionAdmin(QuestionAdmin):
  288. ordering = ["pk"]
  289. Question.objects.bulk_create(
  290. Question(question=str(i)) for i in range(PAGINATOR_SIZE + 10)
  291. )
  292. # The first page of results.
  293. request = self.factory.get(self.url, {"term": "", **self.opts})
  294. request.user = self.superuser
  295. with model_admin(Question, PKOrderingQuestionAdmin):
  296. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  297. self.assertEqual(response.status_code, 200)
  298. data = json.loads(response.content.decode("utf-8"))
  299. self.assertEqual(
  300. data,
  301. {
  302. "results": [
  303. {"id": str(q.pk), "text": q.question}
  304. for q in Question.objects.all()[:PAGINATOR_SIZE]
  305. ],
  306. "pagination": {"more": True},
  307. },
  308. )
  309. # The second page of results.
  310. request = self.factory.get(self.url, {"term": "", "page": "2", **self.opts})
  311. request.user = self.superuser
  312. with model_admin(Question, PKOrderingQuestionAdmin):
  313. response = AutocompleteJsonView.as_view(**self.as_view_args)(request)
  314. self.assertEqual(response.status_code, 200)
  315. data = json.loads(response.content.decode("utf-8"))
  316. self.assertEqual(
  317. data,
  318. {
  319. "results": [
  320. {"id": str(q.pk), "text": q.question}
  321. for q in Question.objects.all()[PAGINATOR_SIZE:]
  322. ],
  323. "pagination": {"more": False},
  324. },
  325. )
  326. def test_serialize_result(self):
  327. class AutocompleteJsonSerializeResultView(AutocompleteJsonView):
  328. def serialize_result(self, obj, to_field_name):
  329. return {
  330. **super().serialize_result(obj, to_field_name),
  331. "posted": str(obj.posted),
  332. }
  333. Question.objects.create(question="Question 1", posted=datetime.date(2021, 8, 9))
  334. Question.objects.create(question="Question 2", posted=datetime.date(2021, 8, 7))
  335. request = self.factory.get(self.url, {"term": "question", **self.opts})
  336. request.user = self.superuser
  337. response = AutocompleteJsonSerializeResultView.as_view(**self.as_view_args)(
  338. request
  339. )
  340. self.assertEqual(response.status_code, 200)
  341. data = json.loads(response.content.decode("utf-8"))
  342. self.assertEqual(
  343. data,
  344. {
  345. "results": [
  346. {"id": str(q.pk), "text": q.question, "posted": str(q.posted)}
  347. for q in Question.objects.order_by("-posted")
  348. ],
  349. "pagination": {"more": False},
  350. },
  351. )
  352. @override_settings(ROOT_URLCONF="admin_views.urls")
  353. class SeleniumTests(AdminSeleniumTestCase):
  354. available_apps = ["admin_views"] + AdminSeleniumTestCase.available_apps
  355. def setUp(self):
  356. self.superuser = User.objects.create_superuser(
  357. username="super",
  358. password="secret",
  359. email="super@example.com",
  360. )
  361. self.admin_login(
  362. username="super",
  363. password="secret",
  364. login_url=reverse("autocomplete_admin:index"),
  365. )
  366. @contextmanager
  367. def select2_ajax_wait(self, timeout=10):
  368. from selenium.common.exceptions import NoSuchElementException
  369. from selenium.webdriver.common.by import By
  370. from selenium.webdriver.support import expected_conditions as ec
  371. yield
  372. with self.disable_implicit_wait():
  373. try:
  374. loading_element = self.selenium.find_element(
  375. By.CSS_SELECTOR, "li.select2-results__option.loading-results"
  376. )
  377. except NoSuchElementException:
  378. pass
  379. else:
  380. self.wait_until(ec.staleness_of(loading_element), timeout=timeout)
  381. def test_select(self):
  382. from selenium.webdriver.common.by import By
  383. from selenium.webdriver.common.keys import Keys
  384. from selenium.webdriver.support.ui import Select
  385. self.selenium.get(
  386. self.live_server_url + reverse("autocomplete_admin:admin_views_answer_add")
  387. )
  388. elem = self.selenium.find_element(By.CSS_SELECTOR, ".select2-selection")
  389. elem.click() # Open the autocomplete dropdown.
  390. results = self.selenium.find_element(By.CSS_SELECTOR, ".select2-results")
  391. self.assertTrue(results.is_displayed())
  392. option = self.selenium.find_element(By.CSS_SELECTOR, ".select2-results__option")
  393. self.assertEqual(option.text, "No results found")
  394. elem.click() # Close the autocomplete dropdown.
  395. q1 = Question.objects.create(question="Who am I?")
  396. Question.objects.bulk_create(
  397. Question(question=str(i)) for i in range(PAGINATOR_SIZE + 10)
  398. )
  399. elem.click() # Reopen the dropdown now that some objects exist.
  400. result_container = self.selenium.find_element(
  401. By.CSS_SELECTOR, ".select2-results"
  402. )
  403. self.assertTrue(result_container.is_displayed())
  404. # PAGINATOR_SIZE results and "Loading more results".
  405. self.assertCountSeleniumElements(
  406. ".select2-results__option",
  407. PAGINATOR_SIZE + 1,
  408. root_element=result_container,
  409. )
  410. search = self.selenium.find_element(By.CSS_SELECTOR, ".select2-search__field")
  411. # Load next page of results by scrolling to the bottom of the list.
  412. with self.select2_ajax_wait():
  413. for _ in range(PAGINATOR_SIZE + 1):
  414. search.send_keys(Keys.ARROW_DOWN)
  415. # All objects are now loaded.
  416. self.assertCountSeleniumElements(
  417. ".select2-results__option",
  418. PAGINATOR_SIZE + 11,
  419. root_element=result_container,
  420. )
  421. # Limit the results with the search field.
  422. with self.select2_ajax_wait():
  423. search.send_keys("Who")
  424. # Ajax request is delayed.
  425. self.assertTrue(result_container.is_displayed())
  426. self.assertCountSeleniumElements(
  427. ".select2-results__option",
  428. PAGINATOR_SIZE + 12,
  429. root_element=result_container,
  430. )
  431. self.assertTrue(result_container.is_displayed())
  432. self.assertCountSeleniumElements(
  433. ".select2-results__option", 1, root_element=result_container
  434. )
  435. # Select the result.
  436. search.send_keys(Keys.RETURN)
  437. select = Select(self.selenium.find_element(By.ID, "id_question"))
  438. self.assertEqual(
  439. select.first_selected_option.get_attribute("value"), str(q1.pk)
  440. )
  441. def test_select_multiple(self):
  442. from selenium.webdriver.common.by import By
  443. from selenium.webdriver.common.keys import Keys
  444. from selenium.webdriver.support.ui import Select
  445. self.selenium.get(
  446. self.live_server_url
  447. + reverse("autocomplete_admin:admin_views_question_add")
  448. )
  449. elem = self.selenium.find_element(By.CSS_SELECTOR, ".select2-selection")
  450. with self.select2_ajax_wait():
  451. elem.click() # Open the autocomplete dropdown.
  452. results = self.selenium.find_element(By.CSS_SELECTOR, ".select2-results")
  453. self.assertTrue(results.is_displayed())
  454. option = self.selenium.find_element(By.CSS_SELECTOR, ".select2-results__option")
  455. self.assertEqual(option.text, "No results found")
  456. elem.click() # Close the autocomplete dropdown.
  457. Question.objects.create(question="Who am I?")
  458. Question.objects.bulk_create(
  459. Question(question=str(i)) for i in range(PAGINATOR_SIZE + 10)
  460. )
  461. elem.click() # Reopen the dropdown now that some objects exist.
  462. result_container = self.selenium.find_element(
  463. By.CSS_SELECTOR, ".select2-results"
  464. )
  465. self.assertTrue(result_container.is_displayed())
  466. self.assertCountSeleniumElements(
  467. ".select2-results__option",
  468. PAGINATOR_SIZE + 1,
  469. root_element=result_container,
  470. )
  471. search = self.selenium.find_element(By.CSS_SELECTOR, ".select2-search__field")
  472. # Load next page of results by scrolling to the bottom of the list.
  473. with self.select2_ajax_wait():
  474. for _ in range(PAGINATOR_SIZE + 1):
  475. search.send_keys(Keys.ARROW_DOWN)
  476. self.assertCountSeleniumElements(
  477. ".select2-results__option", 31, root_element=result_container
  478. )
  479. # Limit the results with the search field.
  480. with self.select2_ajax_wait():
  481. search.send_keys("Who")
  482. # Ajax request is delayed.
  483. self.assertTrue(result_container.is_displayed())
  484. self.assertCountSeleniumElements(
  485. ".select2-results__option", 32, root_element=result_container
  486. )
  487. self.assertTrue(result_container.is_displayed())
  488. self.assertCountSeleniumElements(
  489. ".select2-results__option", 1, root_element=result_container
  490. )
  491. # Select the result.
  492. search.send_keys(Keys.RETURN)
  493. # Reopen the dropdown and add the first result to the selection.
  494. elem.click()
  495. search.send_keys(Keys.ARROW_DOWN)
  496. search.send_keys(Keys.RETURN)
  497. select = Select(self.selenium.find_element(By.ID, "id_related_questions"))
  498. self.assertEqual(len(select.all_selected_options), 2)
  499. def test_inline_add_another_widgets(self):
  500. from selenium.webdriver.common.by import By
  501. def assertNoResults(row):
  502. elem = row.find_element(By.CSS_SELECTOR, ".select2-selection")
  503. elem.click() # Open the autocomplete dropdown.
  504. results = self.selenium.find_element(By.CSS_SELECTOR, ".select2-results")
  505. self.assertTrue(results.is_displayed())
  506. option = self.selenium.find_element(
  507. By.CSS_SELECTOR, ".select2-results__option"
  508. )
  509. self.assertEqual(option.text, "No results found")
  510. # Autocomplete works in rows present when the page loads.
  511. self.selenium.get(
  512. self.live_server_url + reverse("autocomplete_admin:admin_views_book_add")
  513. )
  514. rows = self.selenium.find_elements(By.CSS_SELECTOR, ".dynamic-authorship_set")
  515. self.assertEqual(len(rows), 3)
  516. assertNoResults(rows[0])
  517. # Autocomplete works in rows added using the "Add another" button.
  518. self.selenium.find_element(By.LINK_TEXT, "Add another Authorship").click()
  519. rows = self.selenium.find_elements(By.CSS_SELECTOR, ".dynamic-authorship_set")
  520. self.assertEqual(len(rows), 4)
  521. assertNoResults(rows[-1])