acarsRepo = $acarsRepo; $this->geoSvc = $geoSvc; $this->pilotSvc = $pilotSvc; $this->navRepo = $navRepo; $this->pirepRepo = $pirepRepo; } /** * Find if there are duplicates to a given PIREP. Ideally, the passed * in PIREP hasn't been saved or gone through the create() method * @param Pirep $pirep * @return bool|Pirep */ public function findDuplicate(Pirep $pirep) { $minutes = setting('pireps.duplicate_check_time', 10); $time_limit = Carbon::now()->subMinutes($minutes)->toDateTimeString(); $where = [ 'user_id' => $pirep->user_id, 'airline_id' => $pirep->airline_id, 'flight_number' => $pirep->flight_number, ]; if(filled($pirep->route_code)) { $where['route_code'] = $pirep->route_code; } if(filled($pirep->route_leg)) { $where['route_leg'] = $pirep->route_leg; } try { $found_pireps = Pirep::where($where) ->where('state', '!=', PirepState::CANCELLED) ->where('created_at', '>=', $time_limit) ->get(); if($found_pireps->count() === 0) { return false; } return $found_pireps[0]; } catch (ModelNotFoundException $e) { return false; } } /** * Save the route into the ACARS table with AcarsType::ROUTE * @param Pirep $pirep * @return Pirep * @throws \Exception */ public function saveRoute(Pirep $pirep): Pirep { # Delete all the existing nav points Acars::where([ 'pirep_id' => $pirep->id, 'type' => AcarsType::ROUTE, ])->delete(); # See if a route exists if (!filled($pirep->route)) { return $pirep; } if (!filled($pirep->dpt_airport)) { Log::error('saveRoute: dpt_airport not found: ' . $pirep->dpt_airport_id); return $pirep; } $route = $this->geoSvc->getCoordsFromRoute( $pirep->dpt_airport_id, $pirep->arr_airport_id, [$pirep->dpt_airport->lat, $pirep->dpt_airport->lon], $pirep->route ); /** * @var $point Navdata */ $point_count = 1; foreach ($route as $point) { $acars = new Acars(); $acars->pirep_id = $pirep->id; $acars->type = AcarsType::ROUTE; $acars->nav_type = $point->type; $acars->order = $point_count; $acars->name = $point->id; $acars->lat = $point->lat; $acars->lon = $point->lon; $acars->save(); ++$point_count; } return $pirep; } /** * Create a new PIREP with some given fields * * @param Pirep $pirep * @param array [PirepFieldValues] $field_values * * @return Pirep */ public function create(Pirep $pirep, array $field_values = []): Pirep { if (empty($field_values)) { $field_values = []; } # Figure out what default state should be. Look at the default # behavior from the rank that the pilot is assigned to $default_state = PirepState::PENDING; if ($pirep->source === PirepSource::ACARS) { if ($pirep->pilot->rank->auto_approve_acars) { $default_state = PirepState::ACCEPTED; } } else { if ($pirep->pilot->rank->auto_approve_manual) { $default_state = PirepState::ACCEPTED; } } $pirep->save(); $pirep->refresh(); if(\count($field_values) > 0) { $this->updateCustomFields($pirep->id, $field_values); } Log::info('New PIREP filed', [$pirep]); event(new PirepFiled($pirep)); # only update the pilot last state if they are accepted if ($default_state === PirepState::ACCEPTED) { $pirep = $this->accept($pirep); $this->setPilotState($pirep->pilot, $pirep); } return $pirep; } /** * Update any custom PIREP fields * @param $pirep_id * @param array $field_values */ public function updateCustomFields($pirep_id, array $field_values) { foreach ($field_values as $fv) { PirepFieldValues::updateOrCreate( [ 'pirep_id' => $pirep_id, 'name' => $fv['name'] ], [ 'value' => $fv['value'], 'source' => $fv['source'] ] ); } } /** * Save the list of fares * @param $pirep_id * @param array $fares ['field_id', 'count'] * @throws \Exception */ public function saveFares($pirep_id, array $fares) { if(!$fares) { return; } # Remove all the previous fares PirepFare::where('pirep_id', $pirep_id)->delete(); # Add them in foreach($fares as $fare) { $fare['pirep_id'] = $pirep_id; # other fields: ['fare_id', 'count'] $field = new PirepFare($fare); $field->save(); } } /** * @param Pirep $pirep * @param int $new_state * @return Pirep */ public function changeState(Pirep $pirep, int $new_state) { Log::info('PIREP ' . $pirep->id . ' state change from ' . $pirep->state . ' to ' . $new_state); if ($pirep->state === $new_state) { return $pirep; } /** * Move from a PENDING status into either ACCEPTED or REJECTED */ if ($pirep->state === PirepState::PENDING) { if ($new_state === PirepState::ACCEPTED) { return $this->accept($pirep); } elseif ($new_state === PirepState::REJECTED) { return $this->reject($pirep); } else { return $pirep; } } /* * Move from a ACCEPTED to REJECTED status */ elseif ($pirep->state === PirepState::ACCEPTED) { $pirep = $this->reject($pirep); return $pirep; } /** * Move from REJECTED to ACCEPTED */ elseif ($pirep->state === PirepState::REJECTED) { $pirep = $this->accept($pirep); return $pirep; } return $pirep->refresh(); } /** * @param Pirep $pirep * @return Pirep */ public function accept(Pirep $pirep): Pirep { # moving from a REJECTED state to ACCEPTED, reconcile statuses if ($pirep->state === PirepState::ACCEPTED) { return $pirep; } $ft = $pirep->flight_time; $pilot = $pirep->pilot; $this->pilotSvc->adjustFlightTime($pilot, $ft); $this->pilotSvc->adjustFlightCount($pilot, +1); $this->pilotSvc->calculatePilotRank($pilot); $pirep->pilot->refresh(); # Change the status $pirep->state = PirepState::ACCEPTED; $pirep->save(); $pirep->refresh(); $this->setPilotState($pilot, $pirep); Log::info('PIREP ' . $pirep->id . ' state change to ACCEPTED'); # Update the aircraft $pirep->aircraft->flight_time += $pirep->flight_time; $pirep->aircraft->airport_id = $pirep->arr_airport_id; $pirep->aircraft->landing_time = $pirep->updated_at; $pirep->aircraft->save(); event(new PirepAccepted($pirep)); return $pirep; } /** * @param Pirep $pirep * @return Pirep */ public function reject(Pirep $pirep): Pirep { # If this was previously ACCEPTED, then reconcile the flight hours # that have already been counted, etc if ($pirep->state === PirepState::ACCEPTED) { $pilot = $pirep->pilot; $ft = $pirep->flight_time * -1; $this->pilotSvc->adjustFlightTime($pilot, $ft); $this->pilotSvc->adjustFlightCount($pilot, -1); $this->pilotSvc->calculatePilotRank($pilot); $pirep->pilot->refresh(); } # Change the status $pirep->state = PirepState::REJECTED; $pirep->save(); $pirep->refresh(); $pirep->aircraft->flight_time -= $pirep->flight_time; $pirep->aircraft->save(); Log::info('PIREP ' . $pirep->id . ' state change to REJECTED'); event(new PirepRejected($pirep)); return $pirep; } /** * @param Pirep $pirep */ public function setPilotState(User $pilot, Pirep $pirep) { $pilot->refresh(); $previous_airport = $pilot->curr_airport_id; $pilot->curr_airport_id = $pirep->arr_airport_id; $pilot->last_pirep_id = $pirep->id; $pilot->save(); $pirep->refresh(); event(new UserStatsChanged($pilot, 'airport', $previous_airport)); } }