test_useradmin_views.py 45 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266
  1. from django.contrib.auth import get_user_model
  2. from django.core import mail
  3. from django.urls import reverse
  4. from misago.acl.models import Role
  5. from misago.admin.testutils import AdminTestCase
  6. from misago.categories.models import Category
  7. from misago.legal.models import Agreement
  8. from misago.legal.utils import save_user_agreement_acceptance
  9. from misago.threads.testutils import post_thread, reply_thread
  10. from misago.users.datadownloads import request_user_data_download
  11. from misago.users.models import Ban, DataDownload, Rank
  12. from misago.users.testutils import create_test_user
  13. User = get_user_model()
  14. class UserAdminViewsTests(AdminTestCase):
  15. AJAX_HEADER = {'HTTP_X_REQUESTED_WITH': 'XMLHttpRequest'}
  16. def test_link_registered(self):
  17. """admin index view contains users link"""
  18. response = self.client.get(reverse('misago:admin:index'))
  19. self.assertContains(response, reverse('misago:admin:users:accounts:index'))
  20. def test_list_view(self):
  21. """users list view returns 200"""
  22. response = self.client.get(reverse('misago:admin:users:accounts:index'))
  23. self.assertEqual(response.status_code, 302)
  24. response = self.client.get(response['location'])
  25. self.assertEqual(response.status_code, 200)
  26. self.assertContains(response, self.user.username)
  27. def test_list_search(self):
  28. """users list is searchable"""
  29. response = self.client.get(reverse('misago:admin:users:accounts:index'))
  30. self.assertEqual(response.status_code, 302)
  31. link_base = response['location']
  32. response = self.client.get(link_base)
  33. self.assertEqual(response.status_code, 200)
  34. user_a = create_test_user('Tyrael', 't123@test.com')
  35. user_b = create_test_user('Tyrion', 't321@test.com')
  36. user_c = create_test_user('Karen', 't432@test.com')
  37. # Search both
  38. response = self.client.get('%s&username=tyr' % link_base)
  39. self.assertEqual(response.status_code, 200)
  40. self.assertContains(response, user_a.username)
  41. self.assertContains(response, user_b.username)
  42. # Search tyrion
  43. response = self.client.get('%s&username=tyrion' % link_base)
  44. self.assertEqual(response.status_code, 200)
  45. self.assertNotContains(response, user_a.username)
  46. self.assertContains(response, user_b.username)
  47. # Search tyrael
  48. response = self.client.get('%s&email=t123@test.com' % link_base)
  49. self.assertEqual(response.status_code, 200)
  50. self.assertContains(response, user_a.username)
  51. self.assertNotContains(response, user_b.username)
  52. # Search disabled
  53. user_c.is_active = False
  54. user_c.save()
  55. response = self.client.get('%s&disabled=1' % link_base)
  56. self.assertEqual(response.status_code, 200)
  57. self.assertNotContains(response, user_a.username)
  58. self.assertNotContains(response, user_b.username)
  59. self.assertContains(response, '<del>%s</del>' % user_c.username)
  60. # Search requested own account delete
  61. user_c.is_deleting_account = True
  62. user_c.save()
  63. response = self.client.get('%s&is_deleting_account=1' % link_base)
  64. self.assertEqual(response.status_code, 200)
  65. self.assertNotContains(response, user_a.username)
  66. self.assertNotContains(response, user_b.username)
  67. self.assertContains(response, '<del>%s</del>' % user_c.username)
  68. response = self.client.get('%s&disabled=1' % link_base)
  69. self.assertEqual(response.status_code, 200)
  70. self.assertNotContains(response, user_a.username)
  71. self.assertNotContains(response, user_b.username)
  72. self.assertContains(response, '<del>%s</del>' % user_c.username)
  73. def test_mass_activation(self):
  74. """users list activates multiple users"""
  75. user_pks = []
  76. for i in range(10):
  77. test_user = create_test_user(
  78. 'Bob%s' % i,
  79. 'bob%s@test.com' % i,
  80. requires_activation=1,
  81. )
  82. user_pks.append(test_user.pk)
  83. response = self.client.post(
  84. reverse('misago:admin:users:accounts:index'),
  85. data={
  86. 'action': 'activate',
  87. 'selected_items': user_pks,
  88. }
  89. )
  90. self.assertEqual(response.status_code, 302)
  91. inactive_qs = User.objects.filter(
  92. id__in=user_pks,
  93. requires_activation=1,
  94. )
  95. self.assertEqual(inactive_qs.count(), 0)
  96. self.assertIn("has been activated", mail.outbox[0].subject)
  97. def test_mass_ban(self):
  98. """users list bans multiple users"""
  99. user_pks = []
  100. for i in range(10):
  101. test_user = create_test_user(
  102. 'Bob%s' % i,
  103. 'bob%s@test.com' % i,
  104. requires_activation=1,
  105. )
  106. user_pks.append(test_user.pk)
  107. response = self.client.post(
  108. reverse('misago:admin:users:accounts:index'),
  109. data={
  110. 'action': 'ban',
  111. 'selected_items': user_pks,
  112. }
  113. )
  114. self.assertNotContains(response, 'value="ip"')
  115. self.assertNotContains(response, 'value="ip_first"')
  116. self.assertNotContains(response, 'value="ip_two"')
  117. response = self.client.post(
  118. reverse('misago:admin:users:accounts:index'),
  119. data={
  120. 'action': 'ban',
  121. 'selected_items': user_pks,
  122. 'ban_type': ['usernames', 'emails', 'domains'],
  123. 'finalize': '',
  124. },
  125. )
  126. self.assertEqual(response.status_code, 302)
  127. self.assertEqual(Ban.objects.count(), 21)
  128. def test_mass_ban_with_ips(self):
  129. """users list bans multiple users that also have ips"""
  130. user_pks = []
  131. for i in range(10):
  132. test_user = create_test_user(
  133. 'Bob%s' % i,
  134. 'bob%s@test.com' % i,
  135. joined_from_ip='73.95.67.27',
  136. requires_activation=1,
  137. )
  138. user_pks.append(test_user.pk)
  139. response = self.client.post(
  140. reverse('misago:admin:users:accounts:index'),
  141. data={
  142. 'action': 'ban',
  143. 'selected_items': user_pks,
  144. }
  145. )
  146. self.assertContains(response, 'value="ip"')
  147. self.assertContains(response, 'value="ip_first"')
  148. self.assertContains(response, 'value="ip_two"')
  149. response = self.client.post(
  150. reverse('misago:admin:users:accounts:index'),
  151. data={
  152. 'action': 'ban',
  153. 'selected_items': user_pks,
  154. 'ban_type': ['usernames', 'emails', 'domains', 'ip', 'ip_first', 'ip_two'],
  155. 'finalize': '',
  156. },
  157. )
  158. self.assertEqual(response.status_code, 302)
  159. self.assertEqual(Ban.objects.count(), 24)
  160. def test_mass_request_data_download(self):
  161. """users list requests data download for multiple users"""
  162. user_pks = []
  163. for i in range(10):
  164. test_user = create_test_user(
  165. 'Bob%s' % i,
  166. 'bob%s@test.com' % i,
  167. requires_activation=1,
  168. )
  169. user_pks.append(test_user.pk)
  170. response = self.client.post(
  171. reverse('misago:admin:users:accounts:index'),
  172. data={
  173. 'action': 'request_data_download',
  174. 'selected_items': user_pks,
  175. }
  176. )
  177. self.assertEqual(response.status_code, 302)
  178. self.assertEqual(DataDownload.objects.filter(user_id__in=user_pks).count(), len(user_pks))
  179. def test_mass_request_data_download_avoid_excessive_downloads(self):
  180. """users list avoids excessive data download requests for multiple users"""
  181. user_pks = []
  182. for i in range(10):
  183. test_user = create_test_user(
  184. 'Bob%s' % i,
  185. 'bob%s@test.com' % i,
  186. requires_activation=1,
  187. )
  188. request_user_data_download(test_user)
  189. user_pks.append(test_user.pk)
  190. response = self.client.post(
  191. reverse('misago:admin:users:accounts:index'),
  192. data={
  193. 'action': 'v',
  194. 'selected_items': user_pks,
  195. }
  196. )
  197. self.assertEqual(response.status_code, 302)
  198. self.assertEqual(DataDownload.objects.filter(user_id__in=user_pks).count(), len(user_pks))
  199. def test_mass_delete_accounts_self(self):
  200. """its impossible to delete oneself"""
  201. user_pks = [self.user.pk]
  202. response = self.client.post(
  203. reverse('misago:admin:users:accounts:index'),
  204. data={
  205. 'action': 'delete_accounts',
  206. 'selected_items': user_pks,
  207. }
  208. )
  209. self.assertEqual(response.status_code, 302)
  210. response = self.client.get(response['location'])
  211. self.assertContains(response, "delete yourself")
  212. def test_mass_delete_accounts_admin(self):
  213. """its impossible to delete admin account"""
  214. user_pks = []
  215. for i in range(10):
  216. test_user = create_test_user(
  217. 'Bob%s' % i,
  218. 'bob%s@test.com' % i,
  219. )
  220. user_pks.append(test_user.pk)
  221. test_user.is_staff = True
  222. test_user.save()
  223. response = self.client.post(
  224. reverse('misago:admin:users:accounts:index'),
  225. data={
  226. 'action': 'delete_accounts',
  227. 'selected_items': user_pks,
  228. }
  229. )
  230. self.assertEqual(response.status_code, 302)
  231. response = self.client.get(response['location'])
  232. self.assertContains(response, "is admin and can")
  233. self.assertContains(response, "be deleted.")
  234. self.assertEqual(User.objects.count(), 11)
  235. def test_mass_delete_accounts_superadmin(self):
  236. """its impossible to delete superadmin account"""
  237. user_pks = []
  238. for i in range(10):
  239. test_user = create_test_user(
  240. 'Bob%s' % i,
  241. 'bob%s@test.com' % i,
  242. )
  243. user_pks.append(test_user.pk)
  244. test_user.is_superuser = True
  245. test_user.save()
  246. response = self.client.post(
  247. reverse('misago:admin:users:accounts:index'),
  248. data={
  249. 'action': 'delete_accounts',
  250. 'selected_items': user_pks,
  251. }
  252. )
  253. self.assertEqual(response.status_code, 302)
  254. response = self.client.get(response['location'])
  255. self.assertContains(response, "is admin and can")
  256. self.assertContains(response, "be deleted.")
  257. self.assertEqual(User.objects.count(), 11)
  258. def test_mass_delete_accounts(self):
  259. """users list deletes users"""
  260. # create 10 users to delete
  261. user_pks = []
  262. for i in range(10):
  263. test_user = create_test_user(
  264. 'Bob%s' % i,
  265. 'bob%s@test.com' % i,
  266. requires_activation=0,
  267. )
  268. user_pks.append(test_user.pk)
  269. # create 10 more users that won't be deleted
  270. for i in range(10):
  271. test_user = create_test_user(
  272. 'Weebl%s' % i,
  273. 'weebl%s@test.com' % i,
  274. requires_activation=0,
  275. )
  276. response = self.client.post(
  277. reverse('misago:admin:users:accounts:index'),
  278. data={
  279. 'action': 'delete_accounts',
  280. 'selected_items': user_pks,
  281. }
  282. )
  283. self.assertEqual(response.status_code, 302)
  284. self.assertEqual(User.objects.count(), 11)
  285. def test_mass_delete_all_self(self):
  286. """its impossible to delete oneself with content"""
  287. user_pks = [self.user.pk]
  288. response = self.client.post(
  289. reverse('misago:admin:users:accounts:index'),
  290. data={
  291. 'action': 'delete_all',
  292. 'selected_items': user_pks,
  293. }
  294. )
  295. self.assertEqual(response.status_code, 302)
  296. response = self.client.get(response['location'])
  297. self.assertContains(response, "delete yourself")
  298. def test_mass_delete_all_admin(self):
  299. """its impossible to delete admin account and content"""
  300. user_pks = []
  301. for i in range(10):
  302. test_user = create_test_user(
  303. 'Bob%s' % i,
  304. 'bob%s@test.com' % i,
  305. )
  306. user_pks.append(test_user.pk)
  307. test_user.is_staff = True
  308. test_user.save()
  309. response = self.client.post(
  310. reverse('misago:admin:users:accounts:index'),
  311. data={
  312. 'action': 'delete_all',
  313. 'selected_items': user_pks,
  314. }
  315. )
  316. self.assertEqual(response.status_code, 302)
  317. response = self.client.get(response['location'])
  318. self.assertContains(response, "is admin and can")
  319. self.assertContains(response, "be deleted.")
  320. self.assertEqual(User.objects.count(), 11)
  321. def test_mass_delete_all_superadmin(self):
  322. """its impossible to delete superadmin account and content"""
  323. user_pks = []
  324. for i in range(10):
  325. test_user = create_test_user(
  326. 'Bob%s' % i,
  327. 'bob%s@test.com' % i,
  328. )
  329. user_pks.append(test_user.pk)
  330. test_user.is_superuser = True
  331. test_user.save()
  332. response = self.client.post(
  333. reverse('misago:admin:users:accounts:index'),
  334. data={
  335. 'action': 'delete_all',
  336. 'selected_items': user_pks,
  337. }
  338. )
  339. self.assertEqual(response.status_code, 302)
  340. response = self.client.get(response['location'])
  341. self.assertContains(response, "is admin and can")
  342. self.assertContains(response, "be deleted.")
  343. self.assertEqual(User.objects.count(), 11)
  344. def test_mass_delete_all(self):
  345. """users list mass deleting view has no showstoppers"""
  346. user_pks = []
  347. for i in range(10):
  348. test_user = create_test_user(
  349. 'Bob%s' % i,
  350. 'bob%s@test.com' % i,
  351. requires_activation=1,
  352. )
  353. user_pks.append(test_user.pk)
  354. response = self.client.post(
  355. reverse('misago:admin:users:accounts:index'),
  356. data={
  357. 'action': 'delete_all',
  358. 'selected_items': user_pks,
  359. }
  360. )
  361. self.assertEqual(response.status_code, 200)
  362. # asser that no user has been deleted, because actuall deleting happens in
  363. # dedicated views called via ajax from JavaScript
  364. self.assertEqual(User.objects.count(), 11)
  365. def test_new_view(self):
  366. """new user view creates account"""
  367. response = self.client.get(reverse('misago:admin:users:accounts:new'))
  368. self.assertEqual(response.status_code, 200)
  369. default_rank = Rank.objects.get_default()
  370. authenticated_role = Role.objects.get(special_role='authenticated')
  371. response = self.client.post(
  372. reverse('misago:admin:users:accounts:new'),
  373. data={
  374. 'username': 'Bawww',
  375. 'rank': str(default_rank.pk),
  376. 'roles': str(authenticated_role.pk),
  377. 'email': 'reg@stered.com',
  378. 'new_password': 'pass123',
  379. 'staff_level': '0',
  380. }
  381. )
  382. self.assertEqual(response.status_code, 302)
  383. User.objects.get_by_username('Bawww')
  384. test_user = User.objects.get_by_email('reg@stered.com')
  385. self.assertTrue(test_user.check_password('pass123'))
  386. def test_new_view_password_with_whitespaces(self):
  387. """new user view creates account with whitespaces password"""
  388. response = self.client.get(reverse('misago:admin:users:accounts:new'))
  389. self.assertEqual(response.status_code, 200)
  390. default_rank = Rank.objects.get_default()
  391. authenticated_role = Role.objects.get(special_role='authenticated')
  392. response = self.client.post(
  393. reverse('misago:admin:users:accounts:new'),
  394. data={
  395. 'username': 'Bawww',
  396. 'rank': str(default_rank.pk),
  397. 'roles': str(authenticated_role.pk),
  398. 'email': 'reg@stered.com',
  399. 'new_password': ' pass123 ',
  400. 'staff_level': '0',
  401. }
  402. )
  403. self.assertEqual(response.status_code, 302)
  404. User.objects.get_by_username('Bawww')
  405. test_user = User.objects.get_by_email('reg@stered.com')
  406. self.assertTrue(test_user.check_password(' pass123 '))
  407. def test_edit_view(self):
  408. """edit user view changes account"""
  409. test_user = create_test_user('Bob', 'bob@test.com')
  410. test_link = reverse(
  411. 'misago:admin:users:accounts:edit', kwargs={
  412. 'pk': test_user.pk,
  413. }
  414. )
  415. response = self.client.get(test_link)
  416. self.assertEqual(response.status_code, 200)
  417. response = self.client.post(
  418. test_link,
  419. data={
  420. 'username': 'Bawww',
  421. 'rank': str(test_user.rank_id),
  422. 'roles': str(test_user.roles.all()[0].pk),
  423. 'email': 'reg@stered.com',
  424. 'new_password': 'newpass123',
  425. 'staff_level': '0',
  426. 'signature': 'Hello world!',
  427. 'is_signature_locked': '1',
  428. 'is_hiding_presence': '0',
  429. 'limits_private_thread_invites_to': '0',
  430. 'signature_lock_staff_message': 'Staff message',
  431. 'signature_lock_user_message': 'User message',
  432. 'subscribe_to_started_threads': '2',
  433. 'subscribe_to_replied_threads': '2',
  434. }
  435. )
  436. self.assertEqual(response.status_code, 302)
  437. updated_user = User.objects.get(pk=test_user.pk)
  438. self.assertTrue(updated_user.check_password('newpass123'))
  439. self.assertEqual(updated_user.username, 'Bawww')
  440. self.assertEqual(updated_user.slug, 'bawww')
  441. User.objects.get_by_username('Bawww')
  442. User.objects.get_by_email('reg@stered.com')
  443. def test_edit_dont_change_username(self):
  444. """
  445. If username wasn't changed, don't touch user's username, slug or history
  446. This is regression test for issue #640
  447. """
  448. test_user = create_test_user('Bob', 'bob@test.com')
  449. test_link = reverse(
  450. 'misago:admin:users:accounts:edit', kwargs={
  451. 'pk': test_user.pk,
  452. }
  453. )
  454. response = self.client.get(test_link)
  455. self.assertEqual(response.status_code, 200)
  456. response = self.client.post(
  457. test_link,
  458. data={
  459. 'username': 'Bob',
  460. 'rank': str(test_user.rank_id),
  461. 'roles': str(test_user.roles.all()[0].pk),
  462. 'email': 'reg@stered.com',
  463. 'signature': 'Hello world!',
  464. 'is_signature_locked': '1',
  465. 'is_hiding_presence': '0',
  466. 'limits_private_thread_invites_to': '0',
  467. 'signature_lock_staff_message': 'Staff message',
  468. 'signature_lock_user_message': 'User message',
  469. 'subscribe_to_started_threads': '2',
  470. 'subscribe_to_replied_threads': '2',
  471. }
  472. )
  473. self.assertEqual(response.status_code, 302)
  474. updated_user = User.objects.get(pk=test_user.pk)
  475. self.assertEqual(updated_user.username, 'Bob')
  476. self.assertEqual(updated_user.slug, 'bob')
  477. self.assertEqual(updated_user.namechanges.count(), 0)
  478. def test_edit_change_password_whitespaces(self):
  479. """edit user view changes account password to include whitespaces"""
  480. test_user = create_test_user('Bob', 'bob@test.com')
  481. test_link = reverse(
  482. 'misago:admin:users:accounts:edit', kwargs={
  483. 'pk': test_user.pk,
  484. }
  485. )
  486. response = self.client.get(test_link)
  487. self.assertEqual(response.status_code, 200)
  488. response = self.client.post(
  489. test_link,
  490. data={
  491. 'username': 'Bawww',
  492. 'rank': str(test_user.rank_id),
  493. 'roles': str(test_user.roles.all()[0].pk),
  494. 'email': 'reg@stered.com',
  495. 'new_password': ' newpass123 ',
  496. 'staff_level': '0',
  497. 'signature': 'Hello world!',
  498. 'is_signature_locked': '1',
  499. 'is_hiding_presence': '0',
  500. 'limits_private_thread_invites_to': '0',
  501. 'signature_lock_staff_message': 'Staff message',
  502. 'signature_lock_user_message': 'User message',
  503. 'subscribe_to_started_threads': '2',
  504. 'subscribe_to_replied_threads': '2',
  505. }
  506. )
  507. self.assertEqual(response.status_code, 302)
  508. updated_user = User.objects.get(pk=test_user.pk)
  509. self.assertTrue(updated_user.check_password(' newpass123 '))
  510. self.assertEqual(updated_user.username, 'Bawww')
  511. self.assertEqual(updated_user.slug, 'bawww')
  512. User.objects.get_by_username('Bawww')
  513. User.objects.get_by_email('reg@stered.com')
  514. def test_edit_make_admin(self):
  515. """edit user view allows super admin to make other user admin"""
  516. test_user = create_test_user('Bob', 'bob@test.com')
  517. test_link = reverse(
  518. 'misago:admin:users:accounts:edit', kwargs={
  519. 'pk': test_user.pk,
  520. }
  521. )
  522. response = self.client.get(test_link)
  523. self.assertContains(response, 'id="id_is_staff_1"')
  524. self.assertContains(response, 'id="id_is_superuser_1"')
  525. response = self.client.post(
  526. test_link,
  527. data={
  528. 'username': 'Bawww',
  529. 'rank': str(test_user.rank_id),
  530. 'roles': str(test_user.roles.all()[0].pk),
  531. 'email': 'reg@stered.com',
  532. 'is_staff': '1',
  533. 'is_superuser': '0',
  534. 'signature': 'Hello world!',
  535. 'is_signature_locked': '1',
  536. 'is_hiding_presence': '0',
  537. 'limits_private_thread_invites_to': '0',
  538. 'signature_lock_staff_message': 'Staff message',
  539. 'signature_lock_user_message': 'User message',
  540. 'subscribe_to_started_threads': '2',
  541. 'subscribe_to_replied_threads': '2',
  542. }
  543. )
  544. self.assertEqual(response.status_code, 302)
  545. updated_user = User.objects.get(pk=test_user.pk)
  546. self.assertTrue(updated_user.is_staff)
  547. self.assertFalse(updated_user.is_superuser)
  548. def test_edit_make_superadmin_admin(self):
  549. """edit user view allows super admin to make other user super admin"""
  550. test_user = create_test_user('Bob', 'bob@test.com')
  551. test_link = reverse(
  552. 'misago:admin:users:accounts:edit', kwargs={
  553. 'pk': test_user.pk,
  554. }
  555. )
  556. response = self.client.get(test_link)
  557. self.assertContains(response, 'id="id_is_staff_1"')
  558. self.assertContains(response, 'id="id_is_superuser_1"')
  559. response = self.client.post(
  560. test_link,
  561. data={
  562. 'username': 'Bawww',
  563. 'rank': str(test_user.rank_id),
  564. 'roles': str(test_user.roles.all()[0].pk),
  565. 'email': 'reg@stered.com',
  566. 'is_staff': '0',
  567. 'is_superuser': '1',
  568. 'signature': 'Hello world!',
  569. 'is_signature_locked': '1',
  570. 'is_hiding_presence': '0',
  571. 'limits_private_thread_invites_to': '0',
  572. 'signature_lock_staff_message': 'Staff message',
  573. 'signature_lock_user_message': 'User message',
  574. 'subscribe_to_started_threads': '2',
  575. 'subscribe_to_replied_threads': '2',
  576. }
  577. )
  578. self.assertEqual(response.status_code, 302)
  579. updated_user = User.objects.get(pk=test_user.pk)
  580. self.assertFalse(updated_user.is_staff)
  581. self.assertTrue(updated_user.is_superuser)
  582. def test_edit_denote_superadmin(self):
  583. """edit user view allows super admin to denote other super admin"""
  584. test_user = create_test_user(
  585. 'Bob',
  586. 'bob@test.com',
  587. is_staff=True,
  588. is_superuser=True,
  589. )
  590. test_link = reverse(
  591. 'misago:admin:users:accounts:edit', kwargs={
  592. 'pk': test_user.pk,
  593. }
  594. )
  595. response = self.client.get(test_link)
  596. self.assertContains(response, 'id="id_is_staff_1"')
  597. self.assertContains(response, 'id="id_is_superuser_1"')
  598. response = self.client.post(
  599. test_link,
  600. data={
  601. 'username': 'Bawww',
  602. 'rank': str(test_user.rank_id),
  603. 'roles': str(test_user.roles.all()[0].pk),
  604. 'email': 'reg@stered.com',
  605. 'is_staff': '0',
  606. 'is_superuser': '0',
  607. 'signature': 'Hello world!',
  608. 'is_signature_locked': '1',
  609. 'is_hiding_presence': '0',
  610. 'limits_private_thread_invites_to': '0',
  611. 'signature_lock_staff_message': 'Staff message',
  612. 'signature_lock_user_message': 'User message',
  613. 'subscribe_to_started_threads': '2',
  614. 'subscribe_to_replied_threads': '2',
  615. }
  616. )
  617. self.assertEqual(response.status_code, 302)
  618. updated_user = User.objects.get(pk=test_user.pk)
  619. self.assertFalse(updated_user.is_staff)
  620. self.assertFalse(updated_user.is_superuser)
  621. def test_edit_cant_make_admin(self):
  622. """edit user view forbids admins from making other admins"""
  623. self.user.is_superuser = False
  624. self.user.save()
  625. test_user = create_test_user('Bob', 'bob@test.com')
  626. test_link = reverse(
  627. 'misago:admin:users:accounts:edit', kwargs={
  628. 'pk': test_user.pk,
  629. }
  630. )
  631. response = self.client.get(test_link)
  632. self.assertNotContains(response, 'id="id_is_staff_1"')
  633. self.assertNotContains(response, 'id="id_is_superuser_1"')
  634. response = self.client.post(
  635. test_link,
  636. data={
  637. 'username': 'Bawww',
  638. 'rank': str(test_user.rank_id),
  639. 'roles': str(test_user.roles.all()[0].pk),
  640. 'email': 'reg@stered.com',
  641. 'is_staff': '1',
  642. 'is_superuser': '1',
  643. 'signature': 'Hello world!',
  644. 'is_signature_locked': '1',
  645. 'is_hiding_presence': '0',
  646. 'limits_private_thread_invites_to': '0',
  647. 'signature_lock_staff_message': 'Staff message',
  648. 'signature_lock_user_message': 'User message',
  649. 'subscribe_to_started_threads': '2',
  650. 'subscribe_to_replied_threads': '2',
  651. }
  652. )
  653. self.assertEqual(response.status_code, 302)
  654. updated_user = User.objects.get(pk=test_user.pk)
  655. self.assertFalse(updated_user.is_staff)
  656. self.assertFalse(updated_user.is_superuser)
  657. def test_edit_disable_user(self):
  658. """edit user view allows admin to disable non admin"""
  659. self.user.is_superuser = False
  660. self.user.save()
  661. test_user = create_test_user('Bob', 'bob@test.com')
  662. test_link = reverse(
  663. 'misago:admin:users:accounts:edit', kwargs={
  664. 'pk': test_user.pk,
  665. }
  666. )
  667. response = self.client.get(test_link)
  668. self.assertContains(response, 'id="id_is_active_1"')
  669. self.assertContains(response, 'id="id_is_active_staff_message"')
  670. response = self.client.post(
  671. test_link,
  672. data={
  673. 'username': 'Bawww',
  674. 'rank': str(test_user.rank_id),
  675. 'roles': str(test_user.roles.all()[0].pk),
  676. 'email': 'reg@stered.com',
  677. 'is_staff': '0',
  678. 'is_superuser': '0',
  679. 'signature': 'Hello world!',
  680. 'is_signature_locked': '1',
  681. 'is_hiding_presence': '0',
  682. 'limits_private_thread_invites_to': '0',
  683. 'signature_lock_staff_message': 'Staff message',
  684. 'signature_lock_user_message': 'User message',
  685. 'subscribe_to_started_threads': '2',
  686. 'subscribe_to_replied_threads': '2',
  687. 'is_active': '0',
  688. 'is_active_staff_message': "Disabled in test!"
  689. }
  690. )
  691. self.assertEqual(response.status_code, 302)
  692. updated_user = User.objects.get(pk=test_user.pk)
  693. self.assertFalse(updated_user.is_active)
  694. self.assertEqual(updated_user.is_active_staff_message, "Disabled in test!")
  695. def test_edit_superuser_disable_admin(self):
  696. """edit user view allows admin to disable non admin"""
  697. self.user.is_superuser = True
  698. self.user.save()
  699. test_user = create_test_user('Bob', 'bob@test.com')
  700. test_user.is_staff = True
  701. test_user.save()
  702. test_link = reverse(
  703. 'misago:admin:users:accounts:edit', kwargs={
  704. 'pk': test_user.pk,
  705. }
  706. )
  707. response = self.client.get(test_link)
  708. self.assertContains(response, 'id="id_is_active_1"')
  709. self.assertContains(response, 'id="id_is_active_staff_message"')
  710. response = self.client.post(
  711. test_link,
  712. data={
  713. 'username': 'Bawww',
  714. 'rank': str(test_user.rank_id),
  715. 'roles': str(test_user.roles.all()[0].pk),
  716. 'email': 'reg@stered.com',
  717. 'is_staff': '1',
  718. 'is_superuser': '0',
  719. 'signature': 'Hello world!',
  720. 'is_signature_locked': '1',
  721. 'is_hiding_presence': '0',
  722. 'limits_private_thread_invites_to': '0',
  723. 'signature_lock_staff_message': 'Staff message',
  724. 'signature_lock_user_message': 'User message',
  725. 'subscribe_to_started_threads': '2',
  726. 'subscribe_to_replied_threads': '2',
  727. 'is_active': '0',
  728. 'is_active_staff_message': "Disabled in test!"
  729. }
  730. )
  731. self.assertEqual(response.status_code, 302)
  732. updated_user = User.objects.get(pk=test_user.pk)
  733. self.assertFalse(updated_user.is_active)
  734. self.assertEqual(updated_user.is_active_staff_message, "Disabled in test!")
  735. def test_edit_admin_cant_disable_admin(self):
  736. """edit user view disallows admin to disable admin"""
  737. self.user.is_superuser = False
  738. self.user.save()
  739. test_user = create_test_user('Bob', 'bob@test.com')
  740. test_user.is_staff = True
  741. test_user.save()
  742. test_link = reverse(
  743. 'misago:admin:users:accounts:edit', kwargs={
  744. 'pk': test_user.pk,
  745. }
  746. )
  747. response = self.client.get(test_link)
  748. self.assertNotContains(response, 'id="id_is_active_1"')
  749. self.assertNotContains(response, 'id="id_is_active_staff_message"')
  750. response = self.client.post(
  751. test_link,
  752. data={
  753. 'username': 'Bawww',
  754. 'rank': str(test_user.rank_id),
  755. 'roles': str(test_user.roles.all()[0].pk),
  756. 'email': 'reg@stered.com',
  757. 'is_staff': '1',
  758. 'is_superuser': '0',
  759. 'signature': 'Hello world!',
  760. 'is_signature_locked': '1',
  761. 'is_hiding_presence': '0',
  762. 'limits_private_thread_invites_to': '0',
  763. 'signature_lock_staff_message': 'Staff message',
  764. 'signature_lock_user_message': 'User message',
  765. 'subscribe_to_started_threads': '2',
  766. 'subscribe_to_replied_threads': '2',
  767. 'is_active': '0',
  768. 'is_active_staff_message': "Disabled in test!"
  769. }
  770. )
  771. self.assertEqual(response.status_code, 302)
  772. updated_user = User.objects.get(pk=test_user.pk)
  773. self.assertTrue(updated_user.is_active)
  774. self.assertFalse(updated_user.is_active_staff_message)
  775. def test_edit_is_deleting_account_cant_reactivate(self):
  776. """users deleting own accounts can't be reactivated"""
  777. test_user = create_test_user('Bob', 'bob@test.com')
  778. test_user.mark_for_delete()
  779. test_link = reverse(
  780. 'misago:admin:users:accounts:edit', kwargs={
  781. 'pk': test_user.pk,
  782. }
  783. )
  784. response = self.client.get(test_link)
  785. self.assertNotContains(response, 'id="id_is_active_1"')
  786. self.assertNotContains(response, 'id="id_is_active_staff_message"')
  787. response = self.client.post(
  788. test_link,
  789. data={
  790. 'username': 'Bawww',
  791. 'rank': str(test_user.rank_id),
  792. 'roles': str(test_user.roles.all()[0].pk),
  793. 'email': 'reg@stered.com',
  794. 'is_staff': '1',
  795. 'is_superuser': '0',
  796. 'signature': 'Hello world!',
  797. 'is_signature_locked': '1',
  798. 'is_hiding_presence': '0',
  799. 'limits_private_thread_invites_to': '0',
  800. 'signature_lock_staff_message': 'Staff message',
  801. 'signature_lock_user_message': 'User message',
  802. 'subscribe_to_started_threads': '2',
  803. 'subscribe_to_replied_threads': '2',
  804. 'is_active': '1',
  805. }
  806. )
  807. self.assertEqual(response.status_code, 302)
  808. updated_user = User.objects.get(pk=test_user.pk)
  809. self.assertFalse(updated_user.is_active)
  810. self.assertTrue(updated_user.is_deleting_account)
  811. def test_edit_unusable_password(self):
  812. """admin edit form handles unusable passwords and lets setting new password"""
  813. test_user = create_test_user('Bob', 'bob@test.com')
  814. self.assertFalse(test_user.has_usable_password())
  815. test_link = reverse(
  816. 'misago:admin:users:accounts:edit', kwargs={
  817. 'pk': test_user.pk,
  818. }
  819. )
  820. response = self.client.get(test_link)
  821. self.assertContains(response, 'id="div_id_has_usable_password"')
  822. response = self.client.post(
  823. test_link,
  824. data={
  825. 'username': 'Bawww',
  826. 'rank': str(test_user.rank_id),
  827. 'roles': str(test_user.roles.all()[0].pk),
  828. 'email': 'reg@stered.com',
  829. 'new_password': 'pass123',
  830. 'is_staff': '1',
  831. 'is_superuser': '0',
  832. 'signature': 'Hello world!',
  833. 'is_signature_locked': '1',
  834. 'is_hiding_presence': '0',
  835. 'limits_private_thread_invites_to': '0',
  836. 'signature_lock_staff_message': 'Staff message',
  837. 'signature_lock_user_message': 'User message',
  838. 'subscribe_to_started_threads': '2',
  839. 'subscribe_to_replied_threads': '2',
  840. 'is_active': '1',
  841. }
  842. )
  843. self.assertEqual(response.status_code, 302)
  844. updated_user = User.objects.get(pk=test_user.pk)
  845. self.assertTrue(updated_user.has_usable_password())
  846. def test_edit_keep_unusable_password(self):
  847. """admin edit form handles unusable passwords and lets admin leave them unchanged"""
  848. test_user = create_test_user('Bob', 'bob@test.com')
  849. self.assertFalse(test_user.has_usable_password())
  850. test_link = reverse(
  851. 'misago:admin:users:accounts:edit', kwargs={
  852. 'pk': test_user.pk,
  853. }
  854. )
  855. response = self.client.get(test_link)
  856. self.assertContains(response, 'id="div_id_has_usable_password"')
  857. response = self.client.post(
  858. test_link,
  859. data={
  860. 'username': 'Bawww',
  861. 'rank': str(test_user.rank_id),
  862. 'roles': str(test_user.roles.all()[0].pk),
  863. 'email': 'reg@stered.com',
  864. 'is_staff': '1',
  865. 'is_superuser': '0',
  866. 'signature': 'Hello world!',
  867. 'is_signature_locked': '1',
  868. 'is_hiding_presence': '0',
  869. 'limits_private_thread_invites_to': '0',
  870. 'signature_lock_staff_message': 'Staff message',
  871. 'signature_lock_user_message': 'User message',
  872. 'subscribe_to_started_threads': '2',
  873. 'subscribe_to_replied_threads': '2',
  874. 'is_active': '1',
  875. }
  876. )
  877. self.assertEqual(response.status_code, 302)
  878. updated_user = User.objects.get(pk=test_user.pk)
  879. self.assertFalse(updated_user.has_usable_password())
  880. def test_edit_agreements_list(self):
  881. """edit view displays list of user's agreements"""
  882. test_user = create_test_user('Bob', 'bob@test.com')
  883. test_link = reverse(
  884. 'misago:admin:users:accounts:edit', kwargs={
  885. 'pk': test_user.pk,
  886. }
  887. )
  888. agreement = Agreement.objects.create(
  889. type=Agreement.TYPE_TOS,
  890. title="Test agreement!",
  891. text="Lorem ipsum!",
  892. is_active=True,
  893. )
  894. response = self.client.get(test_link)
  895. self.assertEqual(response.status_code, 200)
  896. self.assertNotContains(response, agreement.title)
  897. save_user_agreement_acceptance(test_user, agreement, commit=True)
  898. response = self.client.get(test_link)
  899. self.assertEqual(response.status_code, 200)
  900. self.assertContains(response, agreement.title)
  901. def test_delete_threads_view_self(self):
  902. """delete user threads view validates if user deletes self"""
  903. test_link = reverse(
  904. 'misago:admin:users:accounts:delete-threads', kwargs={
  905. 'pk': self.user.pk,
  906. }
  907. )
  908. response = self.client.post(test_link, **self.AJAX_HEADER)
  909. self.assertEqual(response.status_code, 302)
  910. response = self.client.get(reverse('misago:admin:index'))
  911. self.assertContains(response, "delete yourself");
  912. def test_delete_threads_view_staff(self):
  913. """delete user threads view validates if user deletes staff"""
  914. test_user = create_test_user('Bob', 'bob@test.com')
  915. test_user.is_staff = True
  916. test_user.save()
  917. test_link = reverse(
  918. 'misago:admin:users:accounts:delete-threads', kwargs={
  919. 'pk': test_user.pk,
  920. }
  921. )
  922. response = self.client.post(test_link, **self.AJAX_HEADER)
  923. self.assertEqual(response.status_code, 302)
  924. response = self.client.get(reverse('misago:admin:index'))
  925. self.assertContains(response, "is admin and");
  926. def test_delete_threads_view_superuser(self):
  927. """delete user threads view validates if user deletes superuser"""
  928. test_user = create_test_user('Bob', 'bob@test.com')
  929. test_user.is_superuser = True
  930. test_user.save()
  931. test_link = reverse(
  932. 'misago:admin:users:accounts:delete-threads', kwargs={
  933. 'pk': test_user.pk,
  934. }
  935. )
  936. response = self.client.post(test_link, **self.AJAX_HEADER)
  937. self.assertEqual(response.status_code, 302)
  938. response = self.client.get(reverse('misago:admin:index'))
  939. self.assertContains(response, "is admin and");
  940. def test_delete_threads_view(self):
  941. """delete user threads view deletes threads"""
  942. test_user = create_test_user('Bob', 'bob@test.com')
  943. test_link = reverse(
  944. 'misago:admin:users:accounts:delete-threads', kwargs={
  945. 'pk': test_user.pk,
  946. }
  947. )
  948. category = Category.objects.all_categories()[:1][0]
  949. [post_thread(category, poster=test_user) for _ in range(10)]
  950. response = self.client.post(test_link, **self.AJAX_HEADER)
  951. self.assertEqual(response.status_code, 200)
  952. response_dict = response.json()
  953. self.assertEqual(response_dict['deleted_count'], 10)
  954. self.assertFalse(response_dict['is_completed'])
  955. response = self.client.post(test_link, **self.AJAX_HEADER)
  956. self.assertEqual(response.status_code, 200)
  957. response_dict = response.json()
  958. self.assertEqual(response_dict['deleted_count'], 0)
  959. self.assertTrue(response_dict['is_completed'])
  960. def test_delete_posts_view_self(self):
  961. """delete user posts view validates if user deletes self"""
  962. test_link = reverse(
  963. 'misago:admin:users:accounts:delete-posts', kwargs={
  964. 'pk': self.user.pk,
  965. }
  966. )
  967. response = self.client.post(test_link, **self.AJAX_HEADER)
  968. self.assertEqual(response.status_code, 302)
  969. response = self.client.get(reverse('misago:admin:index'))
  970. self.assertContains(response, "delete yourself");
  971. def test_delete_posts_view_staff(self):
  972. """delete user posts view validates if user deletes staff"""
  973. test_user = create_test_user('Bob', 'bob@test.com')
  974. test_user.is_staff = True
  975. test_user.save()
  976. test_link = reverse(
  977. 'misago:admin:users:accounts:delete-posts', kwargs={
  978. 'pk': test_user.pk,
  979. }
  980. )
  981. response = self.client.post(test_link, **self.AJAX_HEADER)
  982. self.assertEqual(response.status_code, 302)
  983. response = self.client.get(reverse('misago:admin:index'))
  984. self.assertContains(response, "is admin and");
  985. def test_delete_posts_view_superuser(self):
  986. """delete user posts view validates if user deletes superuser"""
  987. test_user = create_test_user('Bob', 'bob@test.com')
  988. test_user.is_superuser = True
  989. test_user.save()
  990. test_link = reverse(
  991. 'misago:admin:users:accounts:delete-posts', kwargs={
  992. 'pk': test_user.pk,
  993. }
  994. )
  995. response = self.client.post(test_link, **self.AJAX_HEADER)
  996. self.assertEqual(response.status_code, 302)
  997. response = self.client.get(reverse('misago:admin:index'))
  998. self.assertContains(response, "is admin and");
  999. def test_delete_posts_view(self):
  1000. """delete user posts view deletes posts"""
  1001. test_user = create_test_user('Bob', 'bob@test.com')
  1002. test_link = reverse(
  1003. 'misago:admin:users:accounts:delete-posts', kwargs={
  1004. 'pk': test_user.pk,
  1005. }
  1006. )
  1007. category = Category.objects.all_categories()[:1][0]
  1008. thread = post_thread(category)
  1009. [reply_thread(thread, poster=test_user) for _ in range(10)]
  1010. response = self.client.post(test_link, **self.AJAX_HEADER)
  1011. self.assertEqual(response.status_code, 200)
  1012. response_dict = response.json()
  1013. self.assertEqual(response_dict['deleted_count'], 10)
  1014. self.assertFalse(response_dict['is_completed'])
  1015. response = self.client.post(test_link, **self.AJAX_HEADER)
  1016. self.assertEqual(response.status_code, 200)
  1017. response_dict = response.json()
  1018. self.assertEqual(response_dict['deleted_count'], 0)
  1019. self.assertTrue(response_dict['is_completed'])
  1020. def test_delete_account_view_self(self):
  1021. """delete user account view validates if user deletes self"""
  1022. test_link = reverse(
  1023. 'misago:admin:users:accounts:delete-account', kwargs={
  1024. 'pk': self.user.pk,
  1025. }
  1026. )
  1027. response = self.client.post(test_link, **self.AJAX_HEADER)
  1028. self.assertEqual(response.status_code, 302)
  1029. response = self.client.get(reverse('misago:admin:index'))
  1030. self.assertContains(response, "delete yourself");
  1031. def test_delete_account_view_staff(self):
  1032. """delete user account view validates if user deletes staff"""
  1033. test_user = create_test_user('Bob', 'bob@test.com')
  1034. test_user.is_staff = True
  1035. test_user.save()
  1036. test_link = reverse(
  1037. 'misago:admin:users:accounts:delete-account', kwargs={
  1038. 'pk': test_user.pk,
  1039. }
  1040. )
  1041. response = self.client.post(test_link, **self.AJAX_HEADER)
  1042. self.assertEqual(response.status_code, 302)
  1043. response = self.client.get(reverse('misago:admin:index'))
  1044. self.assertContains(response, "is admin and");
  1045. def test_delete_account_view_superuser(self):
  1046. """delete user account view validates if user deletes superuser"""
  1047. test_user = create_test_user('Bob', 'bob@test.com')
  1048. test_user.is_superuser = True
  1049. test_user.save()
  1050. test_link = reverse(
  1051. 'misago:admin:users:accounts:delete-account', kwargs={
  1052. 'pk': test_user.pk,
  1053. }
  1054. )
  1055. response = self.client.post(test_link, **self.AJAX_HEADER)
  1056. self.assertEqual(response.status_code, 302)
  1057. response = self.client.get(reverse('misago:admin:index'))
  1058. self.assertContains(response, "is admin and");
  1059. def test_delete_account_view(self):
  1060. """delete user account view deletes user account"""
  1061. test_user = create_test_user('Bob', 'bob@test.com')
  1062. test_link = reverse(
  1063. 'misago:admin:users:accounts:delete-account', kwargs={
  1064. 'pk': test_user.pk,
  1065. }
  1066. )
  1067. response = self.client.post(test_link, **self.AJAX_HEADER)
  1068. self.assertEqual(response.status_code, 200)
  1069. response_dict = response.json()
  1070. self.assertTrue(response_dict['is_completed'])