| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235 |
- <?php
- namespace App\Services;
- use App\Enums\UserTypeEnum;
- use App\Models\Address;
- use App\Models\City;
- use App\Models\Provider;
- use App\Models\ProviderServicesType;
- use App\Models\ProviderWorkingDay;
- use App\Models\State;
- use App\Models\User;
- use Illuminate\Database\Eloquent\Collection;
- use Illuminate\Support\Facades\DB;
- use Illuminate\Support\Facades\Log;
- class ProviderService
- {
- public function __construct(
- private readonly AuthService $authService,
- ) {}
- public function getAll(): Collection
- {
- $providers = Provider::query()
- ->with(['user', 'profileMedia'])
- ->orderBy('created_at', 'desc')
- ->get();
- return $providers;
- }
- public function findById(int $id): ?Provider
- {
- return Provider::with(['user', 'profileMedia'])->find($id);
- }
- public function create(array $data): Provider
- {
- return Provider::create($data);
- }
- public function update(int $id, array $data): ?Provider
- {
- $model = $this->findById($id);
- if (!$model) {
- return null;
- }
- $model->update($data);
- return $model->fresh(['user', 'profileMedia']);
- }
- public function delete(int $id): bool
- {
- $model = $this->findById($id);
- if (!$model) {
- return false;
- }
- return $model->delete();
- }
- public function register(array $data): ?array
- {
- try {
- DB::beginTransaction();
- $email = $data['email'] ?? null;
- $phone = $data['phone'] ?? null;
- $code = $data['code'] ?? null;
- $user = User::query()
- ->where('type', UserTypeEnum::PROVIDER->value)
- ->where('code', $code)
- ->where(function ($query) use ($email, $phone) {
- if (!empty($email)) {
- $query->orWhere('email', $email);
- }
- if (!empty($phone)) {
- $query->orWhere('phone', $phone);
- }
- })
- ->latest('id')
- ->first();
- if (!$user) {
- throw new \Exception(__('messages.user_not_found_or_code_not_validated'));
- }
- $user->name = $data['name'];
- if (empty($user->email) && !empty($email)) {
- $user->email = $email;
- }
- if (empty($user->phone) && !empty($phone)) {
- $user->phone = $phone;
- }
- $user->save();
- $provider = new Provider();
- $provider->user_id = $user->id;
- $provider->rg = $data['rg'] ?? null;
- $provider->document = $this->sanitizeDigits($data['document'] ?? null);
- $provider->birth_date = $data['birth_date'] ?? null;
- $provider->daily_price_8h = $data['daily_price_8h'] ?? null;
- $provider->daily_price_6h = $data['daily_price_6h'] ?? null;
- $provider->daily_price_4h = $data['daily_price_4h'] ?? null;
- $provider->daily_price_2h = $data['daily_price_2h'] ?? null;
- $provider->is_approved = false;
- $provider->selfie_media_base64 = $data['selfie_base64'] ?? null;
- $provider->document_front_media_base64 = $data['document_front_base64'] ?? null;
- $provider->document_back_media_base64 = $data['document_back_base64'] ?? null;
- $provider->save();
- $provider->refresh();
- $this->createProviderAddress($provider->id, $data);
- $this->createProviderServicesTypes($provider->id, $data);
- $this->createProviderWorkingDays($provider->id, $data);
- if (empty($user->email) || empty($user->code)) {
- throw new \Exception(__('messages.user_not_found_or_code_not_validated'));
- }
- $result = $this->authService->loginWithEmail(
- email: $user->email,
- code: $user->code,
- );
- DB::commit();
- return $result;
- } catch (\Exception $e) {
- DB::rollBack();
- Log::error('Error registering provider: ' . $e->getMessage(), [
- 'data' => $data,
- ]);
- throw $e;
- }
- }
- private function createProviderAddress(int $providerId, array $data): void
- {
- $state = null;
- $city = null;
- if (!empty($data['state'])) {
- $state = State::query()
- ->whereRaw('LOWER(code) = ?', [mb_strtolower($data['state'])])
- ->first();
- }
- if (!empty($data['city'])) {
- $cityQuery = City::query()
- ->whereRaw('LOWER(name) = ?', [mb_strtolower($data['city'])]);
- if ($state) {
- $cityQuery->where('state_id', $state->id);
- }
- $city = $cityQuery->first();
- }
- $address = new Address();
- $address->source = 'provider';
- $address->source_id = $providerId;
- $address->zip_code = $this->sanitizeDigits($data['zip_code'] ?? null);
- $address->address = $data['address'] ?? null;
- $address->has_complement = (bool) ($data['has_complement'] ?? false);
- $address->complement = $data['complement'] ?? null;
- $address->nickname = $data['nickname'] ?? null;
- $address->instructions = $data['instructions'] ?? null;
- $address->address_type = $data['address_type'] ?? 'home';
- $address->state_id = $state?->id;
- $address->city_id = $city?->id;
- $address->save();
- }
- private function createProviderServicesTypes(int $providerId, array $data): void
- {
- $serviceTypeIds = $data['services_types_ids'] ?? $data['service_types_ids'] ?? [];
- $uniqueIds = array_values(array_unique(array_map('intval', $serviceTypeIds)));
- foreach ($uniqueIds as $serviceTypeId) {
- ProviderServicesType::create([
- 'provider_id' => $providerId,
- 'service_type_id' => $serviceTypeId,
- ]);
- }
- }
- private function createProviderWorkingDays(int $providerId, array $data): void
- {
- $workingDays = $data['working_days'] ?? [];
- $seen = [];
- foreach ($workingDays as $workingDay) {
- $day = (int) ($workingDay['day'] ?? -1);
- $period = $workingDay['period'] ?? null;
- if ($day < 0 || $day > 6 || !in_array($period, ['morning', 'afternoon'], true)) {
- continue;
- }
- $uniqueKey = $day . '-' . $period;
- if (isset($seen[$uniqueKey])) {
- continue;
- }
- $seen[$uniqueKey] = true;
- ProviderWorkingDay::create([
- 'provider_id' => $providerId,
- 'day' => $day,
- 'period' => $period,
- ]);
- }
- }
- private function sanitizeDigits(?string $value): ?string
- {
- if ($value === null) {
- return null;
- }
- $digits = preg_replace('/\D+/', '', $value);
- return $digits === '' ? null : $digits;
- }
- }
|