SearchService.php 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. <?php
  2. namespace App\Services;
  3. use App\Models\Address;
  4. use App\Models\Client;
  5. use App\Models\Media;
  6. use App\Models\Provider;
  7. use App\Rules\ScheduleBusinessRules;
  8. use Illuminate\Support\Facades\Auth;
  9. use Illuminate\Support\Facades\DB;
  10. use Illuminate\Support\Facades\Storage;
  11. class SearchService
  12. {
  13. public function __construct() {}
  14. public function buscaPrestadores(?string $name = null, ?string $date = null): array
  15. {
  16. $user = Auth::user();
  17. $cliente = Client::where('user_id', $user->id)->first();
  18. $blockedProviderIds = ScheduleBusinessRules::getBlockedProviderIdsForClient($cliente->id);
  19. $providersWithWorkingDays = ScheduleBusinessRules::getProviderIdsWithWorkingDays();
  20. $clientPrimaryAddress = Address::where('source', 'client')
  21. ->where('source_id', $cliente->id)
  22. ->orderBy('is_primary', 'desc')
  23. ->first();
  24. $distanceSelect = $this->distanceSelect(
  25. $clientPrimaryAddress?->latitude !== null ? (float) $clientPrimaryAddress->latitude : null,
  26. $clientPrimaryAddress?->longitude !== null ? (float) $clientPrimaryAddress->longitude : null,
  27. );
  28. $baseQuery = Provider::leftJoin('users as provider_user', 'provider_user.id', '=', 'providers.user_id')
  29. ->leftJoin(DB::raw("
  30. (
  31. SELECT DISTINCT ON (source_id)
  32. *
  33. FROM addresses
  34. WHERE source = 'provider'
  35. AND deleted_at IS NULL
  36. ORDER BY source_id, is_primary DESC
  37. ) as provider_address
  38. "), 'provider_address.source_id', '=', 'providers.id')
  39. ->whereNotNull('provider_address.id')
  40. ->whereNotIn('providers.id', $blockedProviderIds)
  41. ->whereIn('providers.id', $providersWithWorkingDays)
  42. ->whereNull('providers.deleted_at')
  43. ->whereNotNull('providers.daily_price_8h')
  44. ->whereNotNull('providers.daily_price_6h')
  45. ->whereNotNull('providers.daily_price_4h')
  46. ->whereNotNull('providers.daily_price_2h')
  47. ->when($name, fn ($q) => $q->where('provider_user.name', 'ILIKE', "%{$name}%"))
  48. ->select(
  49. 'providers.id as provider_id',
  50. 'providers.profile_media_id',
  51. 'provider_user.name as provider_name',
  52. 'provider_address.district',
  53. 'provider_address.latitude as provider_latitude',
  54. 'provider_address.longitude as provider_longitude',
  55. 'providers.average_rating',
  56. 'providers.total_services',
  57. 'providers.daily_price_8h',
  58. 'providers.daily_price_6h',
  59. 'providers.daily_price_4h',
  60. 'providers.daily_price_2h',
  61. 'providers.created_at',
  62. DB::raw("(
  63. SELECT COUNT(*)
  64. FROM reviews
  65. LEFT JOIN schedules ON schedules.id = reviews.schedule_id
  66. WHERE reviews.origin = 'provider'
  67. AND schedules.provider_id = providers.id
  68. ) as total_reviews"),
  69. $distanceSelect,
  70. )
  71. ->orderBy('providers.average_rating', 'desc');
  72. $providers = (clone $baseQuery)
  73. ->when(
  74. $clientPrimaryAddress?->city_id,
  75. fn ($query, int $cityId) => $query->where('provider_address.city_id', $cityId)
  76. )
  77. ->get();
  78. if ($providers->isEmpty() && $clientPrimaryAddress?->city_id) {
  79. $providers = $baseQuery->get();
  80. }
  81. $filtered = $providers->when(
  82. $date,
  83. fn ($collection) => $collection->whereIn(
  84. 'provider_id',
  85. ScheduleBusinessRules::getAvailableProviderIdsForDate(
  86. $date,
  87. $collection->pluck('provider_id')
  88. )->toArray()
  89. )->values()
  90. );
  91. $mediaIds = $filtered->pluck('profile_media_id')->filter()->unique()->values();
  92. $mediaUrls = [];
  93. if ($mediaIds->isNotEmpty()) {
  94. Media::whereIn('id', $mediaIds)->get()->each(function ($media) use (&$mediaUrls) {
  95. $mediaUrls[$media->id] = $media->path
  96. ? Storage::temporaryUrl($media->path, now()->addMinutes(60))
  97. : null;
  98. });
  99. }
  100. return $filtered->map(function ($item) use ($mediaUrls) {
  101. $arr = is_array($item) ? $item : $item->toArray();
  102. $arr['profile_media_url'] = $mediaUrls[$arr['profile_media_id'] ?? null] ?? null;
  103. unset($arr['profile_media_id']);
  104. return $arr;
  105. })->values()->toArray();
  106. }
  107. private function distanceSelect(?float $clientLatitude, ?float $clientLongitude): \Illuminate\Contracts\Database\Query\Expression
  108. {
  109. return DistanceService::sqlExpression($clientLatitude, $clientLongitude);
  110. }
  111. }