UserControllerTest.php 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. <?php
  2. namespace Tests\Feature\Http\Auth;
  3. use App\Http\Controllers\Auth\UserController;
  4. use App\Http\Middleware\RejectIfDemoMode;
  5. use App\Http\Requests\UserUpdateRequest;
  6. use App\Models\User;
  7. use App\Observers\UserObserver;
  8. use App\Policies\UserPolicy;
  9. use Illuminate\Support\Facades\Config;
  10. use PHPUnit\Framework\Attributes\CoversClass;
  11. use PHPUnit\Framework\Attributes\Test;
  12. use Tests\FeatureTestCase;
  13. /**
  14. * UserControllerTest test class
  15. */
  16. #[CoversClass(UserController::class)]
  17. #[CoversClass(UserObserver::class)]
  18. #[CoversClass(UserPolicy::class)]
  19. #[CoversClass(RejectIfDemoMode::class)]
  20. #[CoversClass(UserUpdateRequest::class)]
  21. class UserControllerTest extends FeatureTestCase
  22. {
  23. /**
  24. * @var \App\Models\User|\Illuminate\Contracts\Auth\Authenticatable
  25. */
  26. protected $user;
  27. private const NEW_USERNAME = 'Jane DOE';
  28. private const NEW_EMAIL = 'janedoe@example.org';
  29. private const PASSWORD = 'password';
  30. public function setUp() : void
  31. {
  32. parent::setUp();
  33. $this->user = User::factory()->create();
  34. }
  35. #[Test]
  36. public function test_update_user_returns_success()
  37. {
  38. $response = $this->actingAs($this->user, 'web-guard')
  39. ->json('PUT', '/user', [
  40. 'name' => self::NEW_USERNAME,
  41. 'email' => self::NEW_EMAIL,
  42. 'password' => self::PASSWORD,
  43. ])
  44. ->assertOk()
  45. ->assertJsonFragment([
  46. 'name' => self::NEW_USERNAME,
  47. 'id' => $this->user->id,
  48. 'email' => self::NEW_EMAIL,
  49. 'is_admin' => false,
  50. ])
  51. ->assertJsonStructure([
  52. 'preferences',
  53. ]);
  54. $this->assertDatabaseHas('users', [
  55. 'name' => self::NEW_USERNAME,
  56. 'id' => $this->user->id,
  57. 'email' => self::NEW_EMAIL,
  58. 'is_admin' => false,
  59. ]);
  60. }
  61. #[Test]
  62. public function test_update_user_without_changing_email_returns_success()
  63. {
  64. $response = $this->actingAs($this->user, 'web-guard')
  65. ->json('PUT', '/user', [
  66. 'name' => self::NEW_USERNAME,
  67. 'email' => $this->user->email,
  68. 'password' => self::PASSWORD,
  69. ])
  70. ->assertOk()
  71. ->assertJsonFragment([
  72. 'name' => self::NEW_USERNAME,
  73. 'id' => $this->user->id,
  74. 'email' => $this->user->email,
  75. 'is_admin' => false,
  76. ]);
  77. $this->assertDatabaseHas('users', [
  78. 'name' => self::NEW_USERNAME,
  79. 'id' => $this->user->id,
  80. 'email' => $this->user->email,
  81. 'is_admin' => false,
  82. ]);
  83. }
  84. #[Test]
  85. public function test_update_user_without_changing_name_returns_success()
  86. {
  87. $response = $this->actingAs($this->user, 'web-guard')
  88. ->json('PUT', '/user', [
  89. 'name' => $this->user->name,
  90. 'email' => self::NEW_EMAIL,
  91. 'password' => self::PASSWORD,
  92. ])
  93. ->assertOk()
  94. ->assertJsonFragment([
  95. 'name' => $this->user->name,
  96. 'id' => $this->user->id,
  97. 'email' => self::NEW_EMAIL,
  98. 'is_admin' => false,
  99. ]);
  100. $this->assertDatabaseHas('users', [
  101. 'name' => $this->user->name,
  102. 'id' => $this->user->id,
  103. 'email' => self::NEW_EMAIL,
  104. 'is_admin' => false,
  105. ]);
  106. }
  107. #[Test]
  108. public function test_update_user_with_uppercased_email_returns_success()
  109. {
  110. $response = $this->actingAs($this->user, 'web-guard')
  111. ->json('PUT', '/user', [
  112. 'name' => self::NEW_USERNAME,
  113. 'email' => strtoupper(self::NEW_EMAIL),
  114. 'password' => self::PASSWORD,
  115. ])
  116. ->assertOk()
  117. ->assertJsonFragment([
  118. 'name' => self::NEW_USERNAME,
  119. 'id' => $this->user->id,
  120. 'email' => self::NEW_EMAIL,
  121. 'is_admin' => false,
  122. ]);
  123. $this->assertDatabaseHas('users', [
  124. 'name' => self::NEW_USERNAME,
  125. 'id' => $this->user->id,
  126. 'email' => self::NEW_EMAIL,
  127. 'is_admin' => false,
  128. ]);
  129. }
  130. #[Test]
  131. public function test_update_user_in_demo_mode_returns_unchanged_user()
  132. {
  133. Config::set('2fauth.config.isDemoApp', true);
  134. $name = $this->user->name;
  135. $email = $this->user->email;
  136. $response = $this->actingAs($this->user, 'web-guard')
  137. ->json('PUT', '/user', [
  138. 'name' => self::NEW_USERNAME,
  139. 'email' => self::NEW_EMAIL,
  140. 'password' => self::PASSWORD,
  141. ])
  142. ->assertOk()
  143. ->assertJsonFragment([
  144. 'name' => $name,
  145. 'id' => $this->user->id,
  146. 'email' => $email,
  147. 'is_admin' => $this->user->is_admin,
  148. ]);
  149. $this->assertDatabaseHas('users', [
  150. 'name' => $name,
  151. 'id' => $this->user->id,
  152. 'email' => $email,
  153. ]);
  154. }
  155. #[Test]
  156. public function test_update_user_passing_wrong_password_returns_bad_request()
  157. {
  158. $response = $this->actingAs($this->user, 'web-guard')
  159. ->json('PUT', '/user', [
  160. 'name' => self::NEW_USERNAME,
  161. 'email' => self::NEW_EMAIL,
  162. 'password' => 'wrongPassword',
  163. ])
  164. ->assertStatus(400);
  165. }
  166. #[Test]
  167. public function test_update_user_with_invalid_data_returns_validation_error()
  168. {
  169. $response = $this->actingAs($this->user, 'web-guard')
  170. ->json('PUT', '/user', [
  171. 'name' => '',
  172. 'email' => '',
  173. 'password' => self::PASSWORD,
  174. ])
  175. ->assertStatus(422);
  176. }
  177. #[Test]
  178. public function test_delete_user_returns_success()
  179. {
  180. $this->actingAs($this->user, 'web-guard')
  181. ->json('DELETE', '/user', [
  182. 'password' => self::PASSWORD,
  183. ])
  184. ->assertNoContent();
  185. }
  186. #[Test]
  187. public function test_delete_user_in_demo_mode_returns_unauthorized()
  188. {
  189. Config::set('2fauth.config.isDemoApp', true);
  190. $response = $this->actingAs($this->user, 'web-guard')
  191. ->json('DELETE', '/user', [
  192. 'password' => self::PASSWORD,
  193. ])
  194. ->assertUnauthorized()
  195. ->assertJsonStructure([
  196. 'message',
  197. ]);
  198. $this->assertDatabaseHas('users', [
  199. 'id' => $this->user->id,
  200. ]);
  201. }
  202. #[Test]
  203. public function test_delete_user_passing_wrong_password_returns_bad_request()
  204. {
  205. $response = $this->actingAs($this->user, 'web-guard')
  206. ->json('DELETE', '/user', [
  207. 'password' => 'wrongPassword',
  208. ])
  209. ->assertStatus(400);
  210. $this->assertDatabaseHas('users', [
  211. 'id' => $this->user->id,
  212. ]);
  213. }
  214. #[Test]
  215. public function test_delete_the_only_admin_returns_bad_request()
  216. {
  217. /**
  218. * @var \App\Models\User|\Illuminate\Contracts\Auth\Authenticatable
  219. */
  220. $admin = User::factory()->administrator()->create();
  221. $this->assertDatabaseCount('users', 2);
  222. $this->assertEquals(1, User::admins()->count());
  223. $response = $this->actingAs($admin, 'web-guard')
  224. ->json('DELETE', '/user', [
  225. 'password' => self::PASSWORD,
  226. ])
  227. ->assertStatus(400);
  228. $this->assertDatabaseHas('users', [
  229. 'id' => $admin->id,
  230. ]);
  231. }
  232. }