create(); $this->actingAs($user, 'jwt') ->get('/api/user') ->assertJson([ 'code' => 0, 'data' => [ 'uid' => $user->uid, 'email' => $user->email, 'nickname' => $user->nickname, 'score' => $user->score, ], ]); } public function testIndex() { $user = factory(User::class)->create(); factory(\App\Models\Player::class)->create(['uid' => $user->uid]); $this->actingAs($user) ->get('/user') ->assertViewHas('statistics') ->assertSee((new Parsedown())->text(option_localized('announcement'))) ->assertSee((string) $user->score); $unverified = factory(User::class)->create(['verified' => false]); $this->actingAs($unverified) ->get('/user') ->assertDontSee(trans('user.verification.notice.title')); } public function testScoreInfo() { $user = factory(User::class)->create(); factory(\App\Models\Player::class)->create(['uid' => $user->uid]); $this->actingAs($user) ->get('/user/score-info') ->assertJson(['data' => [ 'user' => [ 'score' => $user->score, 'lastSignAt' => $user->last_sign_at, ], 'stats' => [ 'players' => [ 'used' => 1, 'total' => 11, 'percentage' => 1 / 11 * 100, ], 'storage' => [ 'used' => 0, 'total' => $user->score, 'percentage' => 0, ], ], 'signAfterZero' => option('sign_after_zero'), 'signGapTime' => option('sign_gap_time'), ]]); } public function testSign() { option(['sign_score' => '50,50']); $user = factory(User::class)->create(); // Success $this->actingAs($user) ->postJson('/user/sign') ->assertJson([ 'code' => 0, 'message' => trans('user.sign-success', ['score' => 50]), 'data' => [ 'score' => option('user_initial_score') + 50, 'storage' => [ 'percentage' => 0, 'total' => option('user_initial_score') + 50, 'used' => 0, ], 'remaining_time' => (int) option('sign_gap_time'), ], ]); // Remaining time is greater than 0 $user = factory(User::class)->create(['last_sign_at' => get_datetime_string()]); option(['sign_gap_time' => 2]); $this->actingAs($user) ->postJson('/user/sign') ->assertJson([ 'code' => 1, 'message' => trans( 'user.cant-sign-until', [ 'time' => 2, 'unit' => trans('user.time-unit-hour'), ] ), ]); // Can sign after 0 o'clock option(['sign_after_zero' => true]); $user = factory(User::class)->create(['last_sign_at' => get_datetime_string()]); $diff = \Carbon\Carbon::now()->diffInSeconds(\Carbon\Carbon::tomorrow()); if ($diff / 3600 >= 1) { $diff = round($diff / 3600); $unit = 'hour'; } else { $diff = round($diff / 60); $unit = 'min'; } $this->actingAs($user) ->postJson('/user/sign') ->assertJson([ 'code' => 1, 'message' => trans( 'user.cant-sign-until', [ 'time' => $diff, 'unit' => trans("user.time-unit-$unit"), ] ), ]); $user = factory(User::class)->create([ 'last_sign_at' => \Carbon\Carbon::today()->toDateTimeString(), ]); $this->actingAs($user)->postJson('/user/sign')->assertJson(['code' => 0]); } public function testSendVerificationEmail() { Mail::fake(); $unverified = factory(User::class)->create(['verified' => false]); $verified = factory(User::class)->create(); // Should be forbidden if account verification is disabled option(['require_verification' => false]); $this->actingAs($unverified) ->postJson('/user/email-verification') ->assertJson([ 'code' => 1, 'message' => trans('user.verification.disabled'), ]); option(['require_verification' => true]); // Too frequent $this->actingAs($unverified) ->withSession([ 'last_mail_time' => time() - 10, ]) ->postJson('/user/email-verification') ->assertJson([ 'code' => 1, 'message' => trans('user.verification.frequent-mail'), ]); $this->flushSession(); // Already verified $this->actingAs($verified) ->postJson('/user/email-verification') ->assertJson([ 'code' => 1, 'message' => trans('user.verification.verified'), ]); $this->actingAs($unverified) ->postJson('/user/email-verification') ->assertJson([ 'code' => 0, 'message' => trans('user.verification.success'), ]); Mail::assertSent(EmailVerification::class, function ($mail) use ($unverified) { return $mail->hasTo($unverified->email); }); // Should handle exception when sending email Mail::shouldReceive('to') ->once() ->andThrow(new \Mockery\Exception('A fake exception.')); $this->flushSession(); $this->actingAs($unverified) ->postJson('/user/email-verification') ->assertJson([ 'code' => 2, 'message' => trans('user.verification.failed', ['msg' => 'A fake exception.']), ]); // Addition: Mailable test $site_name = option_localized('site_name'); $mailable = new EmailVerification('url'); $mailable->build(); $this->assertTrue($mailable->hasFrom(config('mail.username'), $site_name)); $this->assertEquals(trans('user.verification.mail.title', ['sitename' => $site_name]), $mailable->subject); $this->assertEquals('mails.email-verification', $mailable->view); } public function testProfile() { $this->actAs('normal')->get('/user/profile')->assertViewIs('user.profile'); } public function testHandleProfile() { Event::fake(); $user = factory(User::class)->create(); $user->changePassword('12345678'); // Invalid action $this->actingAs($user) ->postJson('/user/profile') ->assertJson([ 'code' => 1, 'message' => trans('general.illegal-parameters'), ]); // Change nickname without `new_nickname` field $this->postJson('/user/profile', ['action' => 'nickname']) ->assertJsonValidationErrors('new_nickname'); // Invalid nickname $this->postJson('/user/profile', [ 'action' => 'nickname', 'new_nickname' => '\\', ])->assertJsonValidationErrors('new_nickname'); // Too long nickname $this->postJson('/user/profile', [ 'action' => 'nickname', 'new_nickname' => Str::random(256), ])->assertJsonValidationErrors('new_nickname'); // Single player option(['single_player' => true]); factory(\App\Models\Player::class)->create(['uid' => $user->uid]); $this->postJson('/user/profile', ['action' => 'nickname']) ->assertJson(['code' => 1, 'message' => trans('user.profile.nickname.single')]); option(['single_player' => false]); // Change nickname successfully $this->postJson('/user/profile', [ 'action' => 'nickname', 'new_nickname' => 'nickname', ])->assertJson([ 'code' => 0, 'message' => trans('user.profile.nickname.success', ['nickname' => 'nickname']), ]); $this->assertEquals('nickname', User::find($user->uid)->nickname); Event::assertDispatched(Events\UserProfileUpdated::class); // Change password without `current_password` field $this->postJson('/user/profile', ['action' => 'password']) ->assertJsonValidationErrors('current_password'); // Too short current password $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => '1', 'new_password' => '12345678', ])->assertJsonValidationErrors('current_password'); // Too long current password $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => Str::random(33), 'new_password' => '12345678', ])->assertJsonValidationErrors('current_password'); // Too short new password $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => '12345678', 'new_password' => '1', ])->assertJsonValidationErrors('new_password'); // Too long new password $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => '12345678', 'new_password' => Str::random(33), ])->assertJsonValidationErrors('new_password'); // Wrong old password $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => '1234567', 'new_password' => '87654321', ])->assertJson([ 'code' => 1, 'message' => trans('user.profile.password.wrong-password'), ]); // Change password successfully $this->postJson('/user/profile', [ 'action' => 'password', 'current_password' => '12345678', 'new_password' => '87654321', ])->assertJson([ 'code' => 0, 'message' => trans('user.profile.password.success'), ]); Event::assertDispatched(Events\EncryptUserPassword::class); $this->assertTrue(User::find($user->uid)->verifyPassword('87654321')); // After changed password, user should re-login. $this->assertGuest(); $user = User::find($user->uid); // Change email without `new_email` field $this->actingAs($user) ->postJson( '/user/profile', ['action' => 'email'] ) ->assertJsonValidationErrors('new_email'); // Invalid email $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => 'not_an_email', ])->assertJsonValidationErrors('new_email'); // Too short current password $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => 'a@b.c', 'password' => '1', ])->assertJsonValidationErrors('password'); // Too long current password $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => 'a@b.c', 'password' => Str::random(33), ])->assertJsonValidationErrors('password'); // Use a duplicated email $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => $user->email, 'password' => '87654321', ])->assertJson([ 'code' => 1, 'message' => trans('user.profile.email.existed'), ]); // Wrong password $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => 'a@b.c', 'password' => '7654321', ])->assertJson([ 'code' => 1, 'message' => trans('user.profile.email.wrong-password'), ]); // Change email successfully $this->postJson('/user/profile', [ 'action' => 'email', 'new_email' => 'a@b.c', 'password' => '87654321', ])->assertJson([ 'code' => 0, 'message' => trans('user.profile.email.success'), ]); $this->assertEquals('a@b.c', User::find($user->uid)->email); $this->assertEquals(0, User::find($user->uid)->verified); // After changed email, user should re-login. $this->assertGuest(); $user = User::find($user->uid); $user->verified = true; $user->save(); // Delete account without `password` field $this->actingAs($user) ->postJson( '/user/profile', ['action' => 'delete'] ) ->assertJsonValidationErrors('password'); // Too short current password $this->postJson('/user/profile', [ 'action' => 'delete', 'password' => '1', ])->assertJsonValidationErrors('password'); // Too long current password $this->postJson('/user/profile', [ 'action' => 'delete', 'password' => Str::random(33), ])->assertJsonValidationErrors('password'); // Wrong password $this->postJson('/user/profile', [ 'action' => 'delete', 'password' => '7654321', ])->assertJson([ 'code' => 1, 'message' => trans('user.profile.delete.wrong-password'), ]); // Delete account successfully $this->postJson('/user/profile', [ 'action' => 'delete', 'password' => '87654321', ])->assertJson([ 'code' => 0, 'message' => trans('user.profile.delete.success'), ]); $this->assertNull(User::find($user->uid)); // Administrator cannot be deleted $this->actAs('admin') ->postJson('/user/profile', [ 'action' => 'delete', 'password' => '87654321', ])->assertJson([ 'code' => 1, 'message' => trans('user.profile.delete.admin'), ]); } public function testSetAvatar() { $user = factory(User::class)->create(); $steve = factory(\App\Models\Texture::class)->create(); $cape = factory(\App\Models\Texture::class, 'cape')->create(); // Without `tid` field $this->actingAs($user) ->postJson('/user/profile/avatar') ->assertJsonValidationErrors('tid'); // TID is not a integer $this->actingAs($user) ->postJson('/user/profile/avatar', ['tid' => 'string']) ->assertJsonValidationErrors('tid'); // Texture cannot be found $this->actingAs($user) ->postJson('/user/profile/avatar', [ 'tid' => -1, ]) ->assertJson([ 'code' => 1, 'message' => trans('skinlib.non-existent'), ]); // Use cape $this->actingAs($user) ->postJson('/user/profile/avatar', [ 'tid' => $cape->tid, ]) ->assertJson([ 'code' => 1, 'message' => trans('user.profile.avatar.wrong-type'), ]); // Success $this->actingAs($user) ->postJson('/user/profile/avatar', [ 'tid' => $steve->tid, ]) ->assertJson([ 'code' => 0, 'message' => trans('user.profile.avatar.success'), ]); $this->assertEquals($steve->tid, User::find($user->uid)->avatar); // Reset avatar $this->postJson('/user/profile/avatar', ['tid' => 0]) ->assertJson(['code' => 0]); $this->assertEquals(0, User::find($user->uid)->avatar); } public function testReadNotification() { $user = factory(User::class)->create(); $user->notify(new Notifications\SiteMessage('Hyouka', 'Kotenbu?')); $user->refresh(); $notification = $user->unreadNotifications->first(); $this->actingAs($user) ->get('/user/notifications/'.$notification->id) ->assertJson([ 'title' => $notification->data['title'], 'content' => app('parsedown')->text($notification->data['content']), 'time' => $notification->created_at->toDateTimeString(), ]); $notification->refresh(); $this->assertNotNull($notification->read_at); } }