test_user_create_api.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. from django.contrib.auth import get_user_model
  2. from django.core import mail
  3. from django.urls import reverse
  4. from misago.conf.test import override_dynamic_settings
  5. from misago.legal.models import Agreement
  6. from misago.users.models import Ban, Online
  7. from misago.users.testutils import UserTestCase
  8. User = get_user_model()
  9. class UserCreateTests(UserTestCase):
  10. """tests for new user registration (POST to /api/users/)"""
  11. def setUp(self):
  12. super().setUp()
  13. Agreement.objects.invalidate_cache()
  14. self.api_link = "/api/users/"
  15. def tearDown(self):
  16. Agreement.objects.invalidate_cache()
  17. def test_empty_request(self):
  18. """empty request errors with code 400"""
  19. response = self.client.post(self.api_link)
  20. self.assertEqual(response.status_code, 400)
  21. self.assertEqual(
  22. response.json(),
  23. {
  24. "username": ["This field is required."],
  25. "email": ["This field is required."],
  26. "password": ["This field is required."],
  27. },
  28. )
  29. def test_invalid_data(self):
  30. """invalid request data errors with code 400"""
  31. response = self.client.post(
  32. self.api_link, "false", content_type="application/json"
  33. )
  34. self.assertEqual(response.status_code, 400)
  35. self.assertEqual(
  36. response.json(),
  37. {
  38. "username": ["This field is required."],
  39. "email": ["This field is required."],
  40. "password": ["This field is required."],
  41. },
  42. )
  43. def test_authenticated_request(self):
  44. """authentiated user request errors with code 403"""
  45. self.login_user(self.get_authenticated_user())
  46. response = self.client.post(self.api_link)
  47. self.assertEqual(response.status_code, 403)
  48. self.assertEqual(
  49. response.json(),
  50. {"detail": "This action is not available to signed in users."},
  51. )
  52. @override_dynamic_settings(account_activation="closed")
  53. def test_registration_off_request(self):
  54. """registrations off request errors with code 403"""
  55. response = self.client.post(self.api_link)
  56. self.assertEqual(response.status_code, 403)
  57. self.assertEqual(
  58. response.json(), {"detail": "New users registrations are currently closed."}
  59. )
  60. def test_registration_validates_ip_ban(self):
  61. """api validates ip ban"""
  62. Ban.objects.create(
  63. check_type=Ban.IP,
  64. banned_value="127.*",
  65. user_message="You can't register account like this.",
  66. )
  67. response = self.client.post(
  68. self.api_link,
  69. data={
  70. "username": "totallyNew",
  71. "email": "loremipsum@dolor.met",
  72. "password": "LoremP4ssword",
  73. },
  74. )
  75. self.assertEqual(response.status_code, 403)
  76. def test_registration_validates_ip_registration_ban(self):
  77. """api validates ip registration-only ban"""
  78. Ban.objects.create(
  79. check_type=Ban.IP,
  80. banned_value="127.*",
  81. user_message="You can't register account like this.",
  82. registration_only=True,
  83. )
  84. response = self.client.post(
  85. self.api_link,
  86. data={
  87. "username": "totallyNew",
  88. "email": "loremipsum@dolor.met",
  89. "password": "LoremP4ssword",
  90. },
  91. )
  92. self.assertEqual(response.status_code, 400)
  93. self.assertEqual(
  94. response.json(), {"__all__": ["You can't register account like this."]}
  95. )
  96. def test_registration_validates_username(self):
  97. """api validates usernames"""
  98. user = self.get_authenticated_user()
  99. response = self.client.post(
  100. self.api_link,
  101. data={
  102. "username": user.username,
  103. "email": "loremipsum@dolor.met",
  104. "password": "LoremP4ssword",
  105. },
  106. )
  107. self.assertEqual(response.status_code, 400)
  108. self.assertEqual(
  109. response.json(), {"username": ["This username is not available."]}
  110. )
  111. def test_registration_validates_username_ban(self):
  112. """api validates username ban"""
  113. Ban.objects.create(
  114. banned_value="totally*",
  115. user_message="You can't register account like this.",
  116. )
  117. response = self.client.post(
  118. self.api_link,
  119. data={
  120. "username": "totallyNew",
  121. "email": "loremipsum@dolor.met",
  122. "password": "LoremP4ssword",
  123. },
  124. )
  125. self.assertEqual(response.status_code, 400)
  126. self.assertEqual(
  127. response.json(), {"username": ["You can't register account like this."]}
  128. )
  129. def test_registration_validates_username_registration_ban(self):
  130. """api validates username registration-only ban"""
  131. Ban.objects.create(
  132. banned_value="totally*",
  133. user_message="You can't register account like this.",
  134. registration_only=True,
  135. )
  136. response = self.client.post(
  137. self.api_link,
  138. data={
  139. "username": "totallyNew",
  140. "email": "loremipsum@dolor.met",
  141. "password": "LoremP4ssword",
  142. },
  143. )
  144. self.assertEqual(response.status_code, 400)
  145. self.assertEqual(
  146. response.json(), {"username": ["You can't register account like this."]}
  147. )
  148. def test_registration_validates_email(self):
  149. """api validates usernames"""
  150. user = self.get_authenticated_user()
  151. response = self.client.post(
  152. self.api_link,
  153. data={
  154. "username": "totallyNew",
  155. "email": user.email,
  156. "password": "LoremP4ssword",
  157. },
  158. )
  159. self.assertEqual(response.status_code, 400)
  160. self.assertEqual(
  161. response.json(), {"email": ["This e-mail address is not available."]}
  162. )
  163. def test_registration_validates_email_ban(self):
  164. """api validates email ban"""
  165. Ban.objects.create(
  166. check_type=Ban.EMAIL,
  167. banned_value="lorem*",
  168. user_message="You can't register account like this.",
  169. )
  170. response = self.client.post(
  171. self.api_link,
  172. data={
  173. "username": "totallyNew",
  174. "email": "loremipsum@dolor.met",
  175. "password": "LoremP4ssword",
  176. },
  177. )
  178. self.assertEqual(response.status_code, 400)
  179. self.assertEqual(
  180. response.json(), {"email": ["You can't register account like this."]}
  181. )
  182. def test_registration_validates_email_registration_ban(self):
  183. """api validates email registration-only ban"""
  184. Ban.objects.create(
  185. check_type=Ban.EMAIL,
  186. banned_value="lorem*",
  187. user_message="You can't register account like this.",
  188. registration_only=True,
  189. )
  190. response = self.client.post(
  191. self.api_link,
  192. data={
  193. "username": "totallyNew",
  194. "email": "loremipsum@dolor.met",
  195. "password": "LoremP4ssword",
  196. },
  197. )
  198. self.assertEqual(response.status_code, 400)
  199. self.assertEqual(
  200. response.json(), {"email": ["You can't register account like this."]}
  201. )
  202. def test_registration_requires_password(self):
  203. """api uses django's validate_password to validate registrations"""
  204. response = self.client.post(
  205. self.api_link,
  206. data={"username": "Bob", "email": "loremipsum@dolor.met", "password": ""},
  207. )
  208. self.assertEqual(response.status_code, 400)
  209. self.assertEqual(response.json(), {"password": ["This field is required."]})
  210. def test_registration_validates_password(self):
  211. """api uses django's validate_password to validate registrations"""
  212. response = self.client.post(
  213. self.api_link,
  214. data={
  215. "username": "Bob",
  216. "email": "l.o.r.e.m.i.p.s.u.m@gmail.com",
  217. "password": "123",
  218. },
  219. )
  220. self.assertEqual(response.status_code, 400)
  221. self.assertEqual(
  222. response.json(),
  223. {
  224. "email": ["This email is not allowed."],
  225. "password": [
  226. "This password is too short. It must contain at least 7 characters.",
  227. "This password is entirely numeric.",
  228. ],
  229. },
  230. )
  231. def test_registration_validates_password_similiarity(self):
  232. """api uses validate_password to validate registrations"""
  233. response = self.client.post(
  234. self.api_link,
  235. data={
  236. "username": "BobBoberson",
  237. "email": "l.o.r.e.m.i.p.s.u.m@gmail.com",
  238. "password": "BobBoberson",
  239. },
  240. )
  241. self.assertEqual(response.status_code, 400)
  242. self.assertEqual(
  243. response.json(),
  244. {
  245. "email": ["This email is not allowed."],
  246. "password": ["The password is too similar to the username."],
  247. },
  248. )
  249. @override_dynamic_settings(
  250. captcha_type="qa", qa_question="Test", qa_answers="Lorem\nIpsum"
  251. )
  252. def test_registration_validates_captcha(self):
  253. """api validates captcha"""
  254. response = self.client.post(
  255. self.api_link,
  256. data={
  257. "username": "totallyNew",
  258. "email": "loremipsum@dolor.met",
  259. "password": "LoremP4ssword",
  260. "captcha": "dolor",
  261. },
  262. )
  263. self.assertEqual(response.status_code, 400)
  264. self.assertEqual(response.json(), {"captcha": ["Entered answer is incorrect."]})
  265. # valid captcha
  266. response = self.client.post(
  267. self.api_link,
  268. data={
  269. "username": "totallyNew",
  270. "email": "loremipsum@dolor.met",
  271. "password": "LoremP4ssword",
  272. "captcha": "ipSUM",
  273. },
  274. )
  275. self.assertEqual(response.status_code, 200)
  276. @override_dynamic_settings(
  277. captcha_type="qa", qa_question="", qa_answers="Lorem\n\nIpsum"
  278. )
  279. def test_qacaptcha_handles_empty_answers(self):
  280. """api validates captcha"""
  281. response = self.client.post(
  282. self.api_link,
  283. data={
  284. "username": "totallyNew",
  285. "email": "loremipsum@dolor.met",
  286. "password": "LoremP4ssword",
  287. "captcha": "",
  288. },
  289. )
  290. self.assertEqual(response.status_code, 400)
  291. self.assertEqual(response.json(), {"captcha": ["Entered answer is incorrect."]})
  292. def test_registration_check_agreement(self):
  293. """api checks agreement"""
  294. agreement = Agreement.objects.create(
  295. type=Agreement.TYPE_TOS, text="Lorem ipsum", is_active=True
  296. )
  297. response = self.client.post(
  298. self.api_link,
  299. data={
  300. "username": "totallyNew",
  301. "email": "loremipsum@dolor.met",
  302. "password": "LoremP4ssword",
  303. },
  304. )
  305. self.assertEqual(response.status_code, 400)
  306. self.assertEqual(
  307. response.json(), {"terms_of_service": ["This agreement is required."]}
  308. )
  309. # invalid agreement id
  310. response = self.client.post(
  311. self.api_link,
  312. data={
  313. "username": "totallyNew",
  314. "email": "loremipsum@dolor.met",
  315. "password": "LoremP4ssword",
  316. "terms_of_service": agreement.id + 1,
  317. },
  318. )
  319. self.assertEqual(response.status_code, 400)
  320. self.assertEqual(
  321. response.json(), {"terms_of_service": ["This agreement is required."]}
  322. )
  323. # valid agreement id
  324. response = self.client.post(
  325. self.api_link,
  326. data={
  327. "username": "totallyNew",
  328. "email": "loremipsum@dolor.met",
  329. "password": "LoremP4ssword",
  330. "terms_of_service": agreement.id,
  331. },
  332. )
  333. self.assertEqual(response.status_code, 200)
  334. user = User.objects.get(email="loremipsum@dolor.met")
  335. self.assertEqual(user.agreements, [agreement.id])
  336. self.assertEqual(user.useragreement_set.count(), 1)
  337. def test_registration_ignore_inactive_agreement(self):
  338. """api ignores inactive agreement"""
  339. Agreement.objects.create(
  340. type=Agreement.TYPE_TOS, text="Lorem ipsum", is_active=False
  341. )
  342. response = self.client.post(
  343. self.api_link,
  344. data={
  345. "username": "totallyNew",
  346. "email": "loremipsum@dolor.met",
  347. "password": "LoremP4ssword",
  348. "terms_of_service": "",
  349. },
  350. )
  351. self.assertEqual(response.status_code, 200)
  352. user = User.objects.get(email="loremipsum@dolor.met")
  353. self.assertEqual(user.agreements, [])
  354. self.assertEqual(user.useragreement_set.count(), 0)
  355. def test_registration_calls_validate_new_registration(self):
  356. """api uses validate_new_registration to validate registrations"""
  357. response = self.client.post(
  358. self.api_link,
  359. data={
  360. "username": "Bob",
  361. "email": "l.o.r.e.m.i.p.s.u.m@gmail.com",
  362. "password": "pas123",
  363. },
  364. )
  365. self.assertEqual(response.status_code, 400)
  366. self.assertEqual(
  367. response.json(),
  368. {
  369. "email": ["This email is not allowed."],
  370. "password": [
  371. "This password is too short. It must contain at least 7 characters."
  372. ],
  373. },
  374. )
  375. @override_dynamic_settings(account_activation="none")
  376. def test_registration_creates_active_user(self):
  377. """api creates active and signed in user on POST"""
  378. response = self.client.post(
  379. self.api_link,
  380. data={"username": "Bob", "email": "bob@bob.com", "password": "pass123"},
  381. )
  382. self.assertEqual(response.status_code, 200)
  383. self.assertEqual(
  384. response.json(),
  385. {"activation": "active", "username": "Bob", "email": "bob@bob.com"},
  386. )
  387. User.objects.get_by_username("Bob")
  388. test_user = User.objects.get_by_email("bob@bob.com")
  389. self.assertEqual(Online.objects.filter(user=test_user).count(), 1)
  390. self.assertTrue(test_user.check_password("pass123"))
  391. auth_json = self.client.get(reverse("misago:api:auth")).json()
  392. self.assertTrue(auth_json["is_authenticated"])
  393. self.assertEqual(auth_json["username"], "Bob")
  394. self.assertIn("Welcome", mail.outbox[0].subject)
  395. self.assertEqual(test_user.audittrail_set.count(), 1)
  396. @override_dynamic_settings(account_activation="user")
  397. def test_registration_creates_inactive_user(self):
  398. """api creates inactive user on POST"""
  399. response = self.client.post(
  400. self.api_link,
  401. data={"username": "Bob", "email": "bob@bob.com", "password": "pass123"},
  402. )
  403. self.assertEqual(response.status_code, 200)
  404. self.assertEqual(
  405. response.json(),
  406. {"activation": "user", "username": "Bob", "email": "bob@bob.com"},
  407. )
  408. auth_json = self.client.get(reverse("misago:api:auth")).json()
  409. self.assertFalse(auth_json["is_authenticated"])
  410. User.objects.get_by_username("Bob")
  411. User.objects.get_by_email("bob@bob.com")
  412. self.assertIn("Welcome", mail.outbox[0].subject)
  413. @override_dynamic_settings(account_activation="admin")
  414. def test_registration_creates_admin_activated_user(self):
  415. """api creates admin activated user on POST"""
  416. response = self.client.post(
  417. self.api_link,
  418. data={"username": "Bob", "email": "bob@bob.com", "password": "pass123"},
  419. )
  420. self.assertEqual(response.status_code, 200)
  421. self.assertEqual(
  422. response.json(),
  423. {"activation": "admin", "username": "Bob", "email": "bob@bob.com"},
  424. )
  425. auth_json = self.client.get(reverse("misago:api:auth")).json()
  426. self.assertFalse(auth_json["is_authenticated"])
  427. User.objects.get_by_username("Bob")
  428. User.objects.get_by_email("bob@bob.com")
  429. self.assertIn("Welcome", mail.outbox[0].subject)
  430. @override_dynamic_settings(account_activation="none")
  431. def test_registration_creates_user_with_whitespace_password(self):
  432. """api creates user with spaces around password"""
  433. response = self.client.post(
  434. self.api_link,
  435. data={"username": "Bob", "email": "bob@bob.com", "password": " pass123 "},
  436. )
  437. self.assertEqual(response.status_code, 200)
  438. self.assertEqual(
  439. response.json(),
  440. {"activation": "active", "username": "Bob", "email": "bob@bob.com"},
  441. )
  442. User.objects.get_by_username("Bob")
  443. test_user = User.objects.get_by_email("bob@bob.com")
  444. self.assertEqual(Online.objects.filter(user=test_user).count(), 1)
  445. self.assertTrue(test_user.check_password(" pass123 "))
  446. self.assertIn("Welcome", mail.outbox[0].subject)