test_user_changeemail_api.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. from django.contrib.auth import get_user_model
  2. from django.core import mail
  3. from django.urls import reverse
  4. from misago.users.testutils import AuthenticatedUserTestCase
  5. UserModel = get_user_model()
  6. class UserChangeEmailTests(AuthenticatedUserTestCase):
  7. """tests for user change email RPC (/api/users/1/change-email/)"""
  8. def setUp(self):
  9. super().setUp()
  10. self.link = '/api/users/%s/change-email/' % self.user.pk
  11. def test_unsupported_methods(self):
  12. """api isn't supporting GET"""
  13. response = self.client.get(self.link)
  14. self.assertEqual(response.status_code, 405)
  15. def test_empty_input(self):
  16. """api errors correctly for empty input"""
  17. response = self.client.post(self.link, data={})
  18. self.assertEqual(response.status_code, 400)
  19. self.assertEqual(
  20. response.json(), {
  21. 'new_email': ["This field is required."],
  22. 'password': ["This field is required."],
  23. }
  24. )
  25. def test_invalid_password(self):
  26. """api errors correctly for invalid password"""
  27. response = self.client.post(
  28. self.link,
  29. data={
  30. 'new_email': 'new@email.com',
  31. 'password': 'Lor3mIpsum',
  32. },
  33. )
  34. self.assertEqual(response.status_code, 400)
  35. self.assertEqual(response.json(), {
  36. 'password': ["Entered password is invalid."]
  37. })
  38. def test_invalid_input(self):
  39. """api errors correctly for invalid input"""
  40. response = self.client.post(
  41. self.link,
  42. data={
  43. 'new_email': '',
  44. 'password': self.USER_PASSWORD,
  45. },
  46. )
  47. self.assertEqual(response.status_code, 400)
  48. self.assertEqual(response.json(), {
  49. 'new_email': ["This field may not be blank."],
  50. })
  51. response = self.client.post(
  52. self.link,
  53. data={
  54. 'new_email': 'newmail',
  55. 'password': self.USER_PASSWORD,
  56. },
  57. )
  58. self.assertEqual(response.status_code, 400)
  59. self.assertEqual(response.json(), {
  60. 'new_email': ["Enter a valid email address."],
  61. })
  62. def test_email_taken(self):
  63. """api validates email usage"""
  64. UserModel.objects.create_user('BobBoberson', 'new@email.com', 'Pass.123')
  65. response = self.client.post(
  66. self.link,
  67. data={
  68. 'new_email': 'new@email.com',
  69. 'password': self.USER_PASSWORD,
  70. },
  71. )
  72. self.assertEqual(response.status_code, 400)
  73. self.assertEqual(response.json(), {
  74. 'new_email': ["This e-mail address is not available."]
  75. })
  76. def test_change_email(self):
  77. """api allows users to change their e-mail addresses"""
  78. new_email = 'new@email.com'
  79. response = self.client.post(
  80. self.link,
  81. data={
  82. 'new_email': new_email,
  83. 'password': self.USER_PASSWORD,
  84. },
  85. )
  86. self.assertEqual(response.status_code, 200)
  87. self.assertIn('Confirm e-mail change', mail.outbox[0].subject)
  88. for line in [l.strip() for l in mail.outbox[0].body.splitlines()]:
  89. if line.startswith('http://'):
  90. token = line.rstrip('/').split('/')[-1]
  91. break
  92. else:
  93. self.fail("E-mail sent didn't contain confirmation url")
  94. response = self.client.get(
  95. reverse(
  96. 'misago:options-confirm-email-change',
  97. kwargs={
  98. 'token': token,
  99. },
  100. )
  101. )
  102. self.assertEqual(response.status_code, 200)
  103. self.reload_user()
  104. self.assertEqual(self.user.email, new_email)
  105. def test_change_email_user_password_whitespace(self):
  106. """api supports users with whitespace around their passwords"""
  107. user_password = ' old password '
  108. new_password = ' N3wP@55w0rd '
  109. new_email = 'new@email.com'
  110. self.user.set_password(user_password)
  111. self.user.save()
  112. self.login_user(self.user)
  113. response = self.client.post(
  114. self.link,
  115. data={
  116. 'new_email': new_email,
  117. 'password': user_password,
  118. },
  119. )
  120. self.assertEqual(response.status_code, 200)
  121. self.assertIn('Confirm e-mail change', mail.outbox[0].subject)
  122. for line in [l.strip() for l in mail.outbox[0].body.splitlines()]:
  123. if line.startswith('http://'):
  124. token = line.rstrip('/').split('/')[-1]
  125. break
  126. else:
  127. self.fail("E-mail sent didn't contain confirmation url")
  128. response = self.client.get(
  129. reverse(
  130. 'misago:options-confirm-email-change',
  131. kwargs={
  132. 'token': token,
  133. },
  134. )
  135. )
  136. self.assertEqual(response.status_code, 200)
  137. self.reload_user()
  138. self.assertEqual(self.user.email, new_email)