Trip planning depth 2

mail@pastecode.io avatar
unknown
php
5 months ago
13 kB
3
Indexable
<?php

namespace App\Services\TripPlanning;

use App\Models\Daljinar;
use App\Models\Linija;
use App\Models\OldDaljinar;
use App\Models\Stajaliste;
use App\Models\StajalisteRedVoznje;
use Illuminate\Support\Collection;
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Log;

class TripPlanningService
{
    public function findRoutesBetweenStops(Stajaliste $firstStop, Stajaliste $lastStop, $daljinari, $dist): array
    {
        $solutions = [];

        $startDaljinari = $firstStop->old_daljinari;
        $endDaljinari = $lastStop->old_daljinari;

        $allStartLinijaIds = $startDaljinari->pluck('linija_id');
        $allEndLinijaIds = $endDaljinari->pluck('linija_id');

        $allOtherRoutes = $daljinari->whereIn('linija_id', $allStartLinijaIds->merge($allEndLinijaIds)->unique());

        $routesByLinijaId = $allOtherRoutes->groupBy('linija_id');

        foreach ($startDaljinari as $startDaljinar) {
            if (!isset($routesByLinijaId[$startDaljinar->linija_id])) {
                continue;
            }

            $startStops = $routesByLinijaId[$startDaljinar->linija_id]->filter(function ($item) use ($startDaljinar) {
                return $item->smer === $startDaljinar->smer &&
                    $item->redni_broj_stajalista > $startDaljinar->redni_broj_stajalista;
            });

            foreach ($endDaljinari as $endDaljinar) {
                if (!isset($routesByLinijaId[$endDaljinar->linija_id])) {
                    continue;
                }

                $endStops = $routesByLinijaId[$endDaljinar->linija_id]->filter(function ($item) use ($endDaljinar) {
                    return $item->smer === $endDaljinar->smer &&
                        $item->redni_broj_stajalista < $endDaljinar->redni_broj_stajalista;
                });

                // Check for direct single line solution
                if ($startDaljinar->linija_id === $endDaljinar->linija_id &&
                    $startDaljinar->smer === $endDaljinar->smer &&
                    $startDaljinar->redni_broj_stajalista < $endDaljinar->redni_broj_stajalista) {
                    $solutions[] = (object)[
                        'segments' => [
                            (object)[
                                'linija' => $startDaljinar->linija,
                                'start_stop' => $startDaljinar->stajaliste,
                                'end_stop' => $endDaljinar->stajaliste,
                            ]
                        ]
                    ];
                }

                foreach ($endStops as $endStop) {
                    foreach ($startStops as $startStop) {
                        if ($this->calculateDistance(
                                $startStop->stajaliste->sirina, $startStop->stajaliste->duzina,
                                $endStop->stajaliste->sirina, $endStop->stajaliste->duzina
                            ) < $dist / 1000) {
                            $solutions[] = (object)[
                                'segments' => [
                                    (object)[
                                        'linija' => $startDaljinar->linija,
                                        'start_stop' => $startDaljinar->stajaliste,
                                        'end_stop' => $startStop->stajaliste,
                                    ],
                                    (object)[
                                        'linija' => $endDaljinar->linija,
                                        'start_stop' => $endStop->stajaliste,
                                        'end_stop' => $endDaljinar->stajaliste,
                                    ]
                                ]
                            ];
                        }
                    }
                }
            }
        }

        return $solutions;
    }

    public function findRoute2($startLat, $startLng, $endLat, $endLng, $dist): array
    {
        $solutions = [];

        $limit = 8;

        $initialStops = $this->findNearbyStops($startLat, $startLng, $dist, $limit);
        $finalStops = $this->findNearbyStops($endLat, $endLng, $dist, $limit);

        $stopsIds = array_merge($initialStops->pluck('id')->toArray(), $finalStops->pluck('id')->toArray());

        $daljinari = OldDaljinar::whereIn('stajaliste_id', $stopsIds)->get();

        $linijeIds = $daljinari->pluck('linija_id')->toArray();

        $activeLinijeIds = Linija::whereIn('kod_linije', $linijeIds)->active()->where('tip_linije', '!=', 'ноћне-линије')->pluck('kod_linije')->toArray();

        $daljinari = OldDaljinar::with(['linija.shapes', 'stajaliste'])->whereIn('linija_id', $activeLinijeIds)->get();

        foreach ($initialStops as $initialStop) {
            foreach ($finalStops as $finalStop) {
                $solutions = array_merge($solutions, $this->findRoutesBetweenStops($initialStop, $finalStop, $daljinari, $dist));
            }
        }

        return $this->groupSolutions($solutions);
    }

    private function decorateWithStopTimes()
    {

    }

    private function groupSolutions($solutions): array
    {
        $solutions = $this->groupSolutionsByStops($solutions);
        $solutions = $this->groupSolutionsByLines($solutions);

        return array_values($solutions);
    }

    private function groupSolutionsByStops($solutions): array
    {
        $groupedSolutions = [];

        foreach ($solutions as $solution) {
            $uniqueArray = [];

            foreach ($solution->segments as $segment) {
                $uniqueArray[] = $segment->start_stop->id;
                $uniqueArray[] = $segment->end_stop->id;
            }

            $uniqueString = implode('-', $uniqueArray);

            if (isset($groupedSolutions[$uniqueString])) {
                foreach ($groupedSolutions[$uniqueString]->segments as $index => $groupedSegment) {
                    $newLinija = $solution->segments[$index]->linija;

                    if (!in_array($newLinija, $groupedSegment->linija)) {
                        $groupedSegment->linija[] = $newLinija;
                    }
                }
            } else {
                $groupedSolutions[$uniqueString] = $solution;

                foreach ($groupedSolutions[$uniqueString]->segments as $segment) {
                    $segment->linija = [$segment->linija];
                }
            }
        }

        return array_values($groupedSolutions);
    }

    private function groupSolutionsByLines($solutions): array
    {
        $groupedSolutions = [];

        foreach ($solutions as $solution) {
            $linije = [];

            foreach ($solution->segments as $segment) {
                // Extract kod_linija from the linija array
                $linije[] = array_map(function($linija) {
                    return $linija->kod_linije;
                }, $segment->linija);
            }

            $alreadyIncluded = false;

            foreach ($groupedSolutions as $solutionIndex => $groupedSolution) {
                $groupedLinije = [];

                foreach ($groupedSolution->segments as $groupedSegment) {
                    $groupedLinije[] = array_map(function($linija) {
                        return $linija->kod_linije;
                    }, $groupedSegment->linija);
                }

                if (count($linije) === count($groupedLinije)) {
                    $includes = true;
                    $included = true;

                    foreach ($linije as $index => $linijaSegment) {
                        $status = $this->arrayContainsOrContained($groupedLinije[$index], $linijaSegment);
                        if ($status !== 1) {
                            $includes = false;
                        }
                        if ($status !== -1 && !($status === 1 && count($linijaSegment) === count($groupedLinije[$index]))) {
                            $included = false;
                        }
                    }

                    $allSegmentsMatch = $included || $includes;

                    if (!$includes && $included) {
                        $groupedSolutions[$solutionIndex] = $solution;
                    }

                    if ($allSegmentsMatch) {
                        $alreadyIncluded = true;
                        break;
                    }
                }
            }

            if (!$alreadyIncluded) {
                $groupedSolutions[] = $solution;
            }
        }

        return array_values($groupedSolutions);
    }

    /**
     * Returns 1 if $array1 contains $array2, -1 if $array2 contains $array1, and 0 if neither.
     *
     * @param array $array1
     * @param array $array2
     * @return int
     */
    public function arrayContainsOrContained(array $array1, array $array2): int
    {
        if ($this->isSubset($array2, $array1)) {
            return 1;
        }

        if ($this->isSubset($array1, $array2)) {
            return -1;
        }

        return 0;
    }

    /**
     * Checks if $subset is a subset of $set.
     *
     * @param array $subset
     * @param array $set
     * @return bool
     */
    private function isSubset(array $subset, array $set): bool
    {
        return empty(array_diff($subset, $set));
    }

    public function findNearbyStopsSquare($lat, $lng, $squareSide)
    {
        $latitudeDegree = 111320; // Meters
        $longitudeDegree = 78850; // Meters for Serbia (44-45 degrees)

        $latDiff = $squareSide / $latitudeDegree;
        $lngDiff = $squareSide / $longitudeDegree;

        return Stajaliste::where('sirina', '<', $lat + $latDiff)->where('sirina', '>', $lat - $latDiff)
            ->where('duzina', '<', $lng + $lngDiff)->where('duzina', '>', $lng - $lngDiff)
            ->with('old_daljinari.linija.shapes')->get();
    }

    private function calculateDistance2($lat1, $lon1, $lat2, $lon2)
    {
        $latitudeDegree = 111320; // Meters per degree of latitude
        $longitudeDegree = 78850; // Meters per degree of longitude (approximation for Serbia)

        // Convert latitude and longitude differences to meters
        $latDiff = abs($lat2 - $lat1) * $latitudeDegree;
        $lngDiff = abs($lon2 - $lon1) * $longitudeDegree;

        // Calculate the Euclidean distance
        return sqrt($latDiff ** 2 + $lngDiff ** 2); // Distance in meters
    }

    /**
     * Returns the close stops defined by the distance, given by $dist, in meters, and $limit.
     *
     * @param $lat
     * @param $lng
     * @param float $dist
     * @param int $limit
     * @param string $with
     * @return mixed
     */
    public function findNearbyStops($lat, $lng, float $dist = 0, int $limit = 10, $with = '')
    {
        $dist = $dist / 1000; //Convert meters into kilometers

        $query = Stajaliste::selectRaw("*, ( 6371 * acos( cos( radians(?) ) * cos( radians( sirina ) ) * cos( radians( duzina ) - radians(?) ) + sin( radians(?) ) * sin( radians( sirina ) ) ) ) AS distance", [$lat, $lng, $lat])
            ->orderBy('distance');

        if ($with) {
            $query->with('old_daljinari.linija.shapes');
        }

        if ($dist !== 0) {
            $query->having('distance', '<', $dist);
        }

        return $query->take($limit)->get();
    }

    private function calculateDistance($lat1, $lon1, $lat2, $lon2)
    {
        $earthRadius = 6371; // Kilometers

        $dLat = deg2rad($lat2 - $lat1);
        $dLon = deg2rad($lon2 - $lon1);

        $a = sin($dLat / 2) * sin($dLat / 2) +
            cos(deg2rad($lat1)) * cos(deg2rad($lat2)) *
            sin($dLon / 2) * sin($dLon / 2);

        $c = 2 * atan2(sqrt($a), sqrt(1 - $a));

        return $earthRadius * $c; // Distance in kilometers
    }

    public function decodePolyline($encoded) {
        $points = [];
        $index = 0;
        $lat = 0;
        $lng = 0;

        while ($index < strlen($encoded)) {
            $b = 0;
            $shift = 0;
            $result = 0;
            do {
                $b = ord($encoded[$index++]) - 63;
                $result |= ($b & 0x1f) << $shift;
                $shift += 5;
            } while ($b >= 0x20);
            $deltaLat = ($result & 1) ? ~(($result >> 1)) : ($result >> 1);
            $lat += $deltaLat;

            $b = 0;
            $shift = 0;
            $result = 0;
            do {
                $b = ord($encoded[$index++]) - 63;
                $result |= ($b & 0x1f) << $shift;
                $shift += 5;
            } while ($b >= 0x20);
            $deltaLng = ($result & 1) ? ~(($result >> 1)) : ($result >> 1);
            $lng += $deltaLng;

            $points[] = [($lat / 1E5), ($lng / 1E5)];
        }

        return $points;
    }
}
Leave a Comment