AuthService.php 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. <?php
  2. namespace App\Services;
  3. use App\Models\User;
  4. use App\Models\PersonalAccessToken;
  5. use Carbon\Carbon;
  6. use Illuminate\Support\Facades\Auth;
  7. use Illuminate\Support\Facades\DB;
  8. use Illuminate\Support\Facades\Log;
  9. use Illuminate\Support\Str;
  10. class AuthService
  11. {
  12. public function __construct(
  13. private readonly EmailService $emailService,
  14. ) {}
  15. public function login(string $email, string $password): ?array
  16. {
  17. if (!Auth::attempt(["email" => $email, "password" => $password]) || (User::where("email", $email)->first()->type == 'CLIENT' || User::where("email", $email)->first()->type == 'PROVIDER')) {
  18. return null;
  19. }
  20. $user = User::where("email", $email)->first();
  21. $deviceId = Str::uuid()->toString();
  22. $accessToken = $user->createAccessToken($deviceId);
  23. $refreshToken = $user->createRefreshToken($deviceId);
  24. return [
  25. "payload" => [
  26. "access_token" => $accessToken,
  27. "user" => $user,
  28. ],
  29. "refreshToken" => $refreshToken,
  30. ];
  31. }
  32. public function refresh(string $refreshToken): ?array
  33. {
  34. if (!$refreshToken) {
  35. return null;
  36. }
  37. $tokenModel = PersonalAccessToken::findToken($refreshToken);
  38. if (
  39. !$tokenModel ||
  40. !in_array("refresh", $tokenModel->abilities) ||
  41. $tokenModel->expires_at < now()
  42. ) {
  43. return null;
  44. }
  45. $user = $tokenModel->tokenable;
  46. if (!$user) {
  47. return null;
  48. }
  49. $deviceId = Str::afterLast($tokenModel->name, "_");
  50. $tokens = $this->refreshTokenTransaction($tokenModel, $user, $deviceId);
  51. return [
  52. "payload" => [
  53. "access_token" => $tokens["access_token"],
  54. "user" => $user,
  55. ],
  56. "refreshToken" => $tokens["refresh_token"],
  57. ];
  58. }
  59. public function logout(): void
  60. {
  61. $user = Auth::user();
  62. if (!$user) {
  63. return;
  64. }
  65. $tokenName = $user->currentAccessToken()->name;
  66. $deviceId = Str::afterLast($tokenName, "_");
  67. $user
  68. ->tokens()
  69. ->where("name", "like", "%_{$deviceId}")
  70. ->delete();
  71. }
  72. protected function refreshTokenTransaction(
  73. PersonalAccessToken $tokenModel,
  74. User $user,
  75. string $deviceId,
  76. ): array {
  77. return DB::transaction(function () use (
  78. $tokenModel,
  79. $user,
  80. $deviceId,
  81. ): array {
  82. $tokenModel->update(["expires_at" => Carbon::now()]);
  83. $accessToken = $user->createAccessToken($deviceId);
  84. $refreshToken = $user->createRefreshToken($deviceId);
  85. return [
  86. "access_token" => $accessToken,
  87. "refresh_token" => $refreshToken,
  88. ];
  89. });
  90. }
  91. public function sendCode(array $data): ?bool
  92. {
  93. try {
  94. DB::beginTransaction();
  95. $code = str_pad((string) random_int(0, 999999), 6, '0', STR_PAD_LEFT);
  96. $user = User::where(function ($query) use ($data) {
  97. $query->when(!empty($data['email']), function ($q) use ($data) {
  98. $q->where('email', $data['email']);
  99. })
  100. ->when(!empty($data['phone']), function ($q) use ($data) {
  101. $q->where('phone', $data['phone']);
  102. });
  103. })
  104. ->first();
  105. $isLogin = false;
  106. if ($user) {
  107. $user->code = $code;
  108. $user->validated_code = false;
  109. $user->save();
  110. $isLogin = true;
  111. } else {
  112. $user = new User();
  113. $user->fill($data);
  114. $user->code = $code;
  115. $user->name = $data['name'] ?? 'Usuário';
  116. $user->type = $data['type'] ?? 'USER';
  117. $user->save();
  118. }
  119. if (!empty($data['email'])) {
  120. $this->emailService->sendVerificationCode(
  121. email: $data['email'],
  122. code: $code,
  123. recipientName: $data['name'] ?? '',
  124. );
  125. } elseif (!empty($data['phone'])) {
  126. Log::info('SMS: envio de código por telefone ainda não implementado.', [
  127. 'phone' => $data['phone'],
  128. ]);
  129. }
  130. DB::commit();
  131. return $isLogin;
  132. } catch (\Exception $e) {
  133. DB::rollBack();
  134. Log::error('Erro ao enviar código de verificação.', [
  135. 'error' => $e->getMessage(),
  136. 'data' => $data,
  137. ]);
  138. return false;
  139. }
  140. }
  141. public function validateCode(array $data, bool $isLogin): bool|array
  142. {
  143. $email = $data['email'] ?? null;
  144. $phone = $data['phone'] ?? null;
  145. $code = $data['code'] ?? '';
  146. $user = User::where(function ($query) use ($email, $phone) {
  147. $query->when($email, function ($q) use ($email) {
  148. $q->where('email', $email);
  149. })
  150. ->when($phone, function ($q) use ($phone) {
  151. $q->where('phone', $phone);
  152. });
  153. })
  154. ->where('code', $code)
  155. ->first();
  156. if (!$user) {
  157. return false;
  158. }
  159. if($isLogin) {
  160. $resultLogin = $this->loginWithEmail($user->email, $code);
  161. return $resultLogin;
  162. }
  163. return true;
  164. }
  165. public function loginWithEmail(string $email, string $code): ?array
  166. {
  167. $user = User::where('email', $email)
  168. ->where('code', $code)
  169. ->first();
  170. if (!$user) {
  171. return null;
  172. }
  173. $deviceId = Str::uuid()->toString();
  174. $accessToken = $user->createAccessTokenApp($deviceId);
  175. $refreshToken = $user->createRefreshTokenApp($deviceId);
  176. $user->validated_code = true;
  177. $user->code = null;
  178. $user->save();
  179. return [
  180. "payload" => [
  181. "access_token" => $accessToken,
  182. "user" => $user,
  183. ],
  184. "refreshToken" => $refreshToken,
  185. ];
  186. }
  187. }