test_user_changeemail_api.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  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(UserChangeEmailTests, self).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(
  36. response.json(), {
  37. 'password': ["Entered password is invalid."],
  38. },
  39. )
  40. def test_invalid_input(self):
  41. """api errors correctly for invalid input"""
  42. response = self.client.post(
  43. self.link,
  44. data={
  45. 'new_email': '',
  46. 'password': self.USER_PASSWORD,
  47. },
  48. )
  49. self.assertEqual(response.status_code, 400)
  50. self.assertEqual(response.json(), {
  51. 'new_email': ["This field may not be blank."],
  52. })
  53. response = self.client.post(
  54. self.link,
  55. data={
  56. 'new_email': 'newmail',
  57. 'password': self.USER_PASSWORD,
  58. },
  59. )
  60. self.assertEqual(response.status_code, 400)
  61. self.assertEqual(response.json(), {
  62. 'new_email': ["Enter a valid email address."],
  63. })
  64. def test_email_taken(self):
  65. """api validates email usage"""
  66. UserModel.objects.create_user('BobBoberson', 'new@email.com', 'Pass.123')
  67. response = self.client.post(
  68. self.link,
  69. data={
  70. 'new_email': 'new@email.com',
  71. 'password': self.USER_PASSWORD,
  72. },
  73. )
  74. self.assertEqual(response.status_code, 400)
  75. self.assertEqual(
  76. response.json(), {
  77. 'new_email': ["This e-mail address is not available."],
  78. },
  79. )
  80. def test_change_email(self):
  81. """api allows users to change their e-mail addresses"""
  82. new_email = 'new@email.com'
  83. response = self.client.post(
  84. self.link,
  85. data={
  86. 'new_email': new_email,
  87. 'password': self.USER_PASSWORD,
  88. },
  89. )
  90. self.assertEqual(response.status_code, 204)
  91. self.assertIn('Confirm e-mail change', mail.outbox[0].subject)
  92. for line in [l.strip() for l in mail.outbox[0].body.splitlines()]:
  93. if line.startswith('http://'):
  94. token = line.rstrip('/').split('/')[-1]
  95. break
  96. else:
  97. self.fail("E-mail sent didn't contain confirmation url")
  98. response = self.client.get(
  99. reverse(
  100. 'misago:options-confirm-email-change',
  101. kwargs={
  102. 'token': token,
  103. },
  104. )
  105. )
  106. self.assertEqual(response.status_code, 200)
  107. self.reload_user()
  108. self.assertEqual(self.user.email, new_email)
  109. def test_change_email_user_password_whitespace(self):
  110. """api supports users with whitespace around their passwords"""
  111. user_password = ' old password '
  112. new_email = 'new@email.com'
  113. self.user.set_password(user_password)
  114. self.user.save()
  115. self.login_user(self.user)
  116. response = self.client.post(
  117. self.link,
  118. data={
  119. 'new_email': new_email,
  120. 'password': user_password,
  121. },
  122. )
  123. self.assertEqual(response.status_code, 204)
  124. self.assertIn('Confirm e-mail change', mail.outbox[0].subject)
  125. for line in [l.strip() for l in mail.outbox[0].body.splitlines()]:
  126. if line.startswith('http://'):
  127. token = line.rstrip('/').split('/')[-1]
  128. break
  129. else:
  130. self.fail("E-mail sent didn't contain confirmation url")
  131. response = self.client.get(
  132. reverse(
  133. 'misago:options-confirm-email-change',
  134. kwargs={
  135. 'token': token,
  136. },
  137. )
  138. )
  139. self.assertEqual(response.status_code, 200)
  140. self.reload_user()
  141. self.assertEqual(self.user.email, new_email)