AuthService.php 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. <?php
  2. namespace App\Services;
  3. use App\Enums\UserTypeEnum;
  4. use App\Mail\PasswordResetCodeMail;
  5. use App\Models\User;
  6. use App\Models\PersonalAccessToken;
  7. use Carbon\Carbon;
  8. use Exception;
  9. use Illuminate\Support\Facades\Auth;
  10. use Illuminate\Support\Facades\DB;
  11. use Illuminate\Support\Facades\Mail;
  12. use Illuminate\Support\Str;
  13. class AuthService
  14. {
  15. public function login(string $email, string $password, string $origem): ?array
  16. {
  17. if (!Auth::attempt(["email" => $email, "password" => $password])) {
  18. return null;
  19. }
  20. $user = User::where("email", $email)->first();
  21. if ($origem === 'admin') {
  22. if ($user->user_type !== UserTypeEnum::ADMIN) {
  23. throw new Exception('credenciais invalidas para acesso de franqueadora');
  24. }
  25. }
  26. $user->update(['last_login_at' => now()]);
  27. $user->load('units');
  28. $deviceId = Str::uuid()->toString();
  29. $accessToken = $user->createAccessToken($deviceId);
  30. $refreshToken = $user->createRefreshToken($deviceId);
  31. return [
  32. "payload" => [
  33. "access_token" => $accessToken,
  34. "user" => $user,
  35. ],
  36. "refreshToken" => $refreshToken,
  37. ];
  38. }
  39. public function refresh(string $refreshToken): ?array
  40. {
  41. if (!$refreshToken) {
  42. return null;
  43. }
  44. $tokenModel = PersonalAccessToken::findToken($refreshToken);
  45. if (
  46. !$tokenModel ||
  47. !in_array("refresh", $tokenModel->abilities) ||
  48. $tokenModel->expires_at < now()
  49. ) {
  50. return null;
  51. }
  52. $user = $tokenModel->tokenable;
  53. if (!$user) {
  54. return null;
  55. }
  56. $user->load('units');
  57. $deviceId = Str::afterLast($tokenModel->name, "_");
  58. $tokens = $this->refreshTokenTransaction($tokenModel, $user, $deviceId);
  59. return [
  60. "payload" => [
  61. "access_token" => $tokens["access_token"],
  62. "user" => $user,
  63. ],
  64. "refreshToken" => $tokens["refresh_token"],
  65. ];
  66. }
  67. public function forgotPassword(string $email, string $portal = 'franchisee'): bool
  68. {
  69. $user = User::where('email', $email)->first();
  70. if (!$user) {
  71. return false;
  72. }
  73. $code = str_pad((string) random_int(0, 999999), 6, '0', STR_PAD_LEFT);
  74. DB::table('password_reset_tokens')->updateOrInsert(
  75. ['email' => $email],
  76. [
  77. 'token' => $code,
  78. 'created_at' => now(),
  79. 'expires_at' => now()->addMinutes(30),
  80. ]
  81. );
  82. $baseUrl = $portal === 'franchisor'
  83. ? config('app.franchisor_url')
  84. : config('app.franchisee_url');
  85. $recoveryLink = $baseUrl . '/recovery-password?email=' . urlencode($email);
  86. Mail::to($email)->send(new PasswordResetCodeMail($code, $recoveryLink));
  87. return true;
  88. }
  89. public function resetPassword(string $email, string $code, string $password): bool
  90. {
  91. if (!$this->verifyPasswordCode($email, $code)) {
  92. return false;
  93. }
  94. $user = User::where('email', $email)->first();
  95. if (!$user) {
  96. return false;
  97. }
  98. $user->update(['password' => $password]);
  99. DB::table('password_reset_tokens')->where('email', $email)->delete();
  100. return true;
  101. }
  102. public function verifyPasswordCode(string $email, string $code): bool
  103. {
  104. $record = DB::table('password_reset_tokens')
  105. ->where('email', $email)
  106. ->where('token', $code)
  107. ->first();
  108. if (!$record) {
  109. return false;
  110. }
  111. if (Carbon::parse($record->expires_at)->isPast()) {
  112. return false;
  113. }
  114. return true;
  115. }
  116. public function logout(): void
  117. {
  118. $user = Auth::user();
  119. if (!$user) {
  120. return;
  121. }
  122. $tokenName = $user->currentAccessToken()->name;
  123. $deviceId = Str::afterLast($tokenName, "_");
  124. $user
  125. ->tokens()
  126. ->where("name", "like", "%_{$deviceId}")
  127. ->delete();
  128. }
  129. protected function refreshTokenTransaction(
  130. PersonalAccessToken $tokenModel, User $user, string $deviceId,
  131. ): array {
  132. return DB::transaction(function () use (
  133. $tokenModel,
  134. $user,
  135. $deviceId,
  136. ): array {
  137. $tokenModel->update(["expires_at" => Carbon::now()]);
  138. $accessToken = $user->createAccessToken($deviceId);
  139. $refreshToken = $user->createRefreshToken($deviceId);
  140. return [
  141. "access_token" => $accessToken,
  142. "refresh_token" => $refreshToken,
  143. ];
  144. });
  145. }
  146. }