phpvms/app/Http/Controllers/Api/PirepController.php

461 lines
14 KiB
PHP
Raw Normal View History

2017-08-15 12:36:49 +08:00
<?php
namespace App\Http\Controllers\Api;
use App\Models\PirepComment;
use Log;
2018-01-04 00:25:55 +08:00
use Auth;
use Illuminate\Database\Eloquent\ModelNotFoundException;
use Illuminate\Http\Request;
use App\Models\Acars;
use App\Models\Pirep;
use App\Models\Enums\AcarsType;
use App\Models\Enums\PirepState;
use App\Models\Enums\PirepStatus;
use App\Services\GeoService;
2017-12-27 04:54:28 +08:00
use App\Services\PIREPService;
use App\Repositories\AcarsRepository;
use App\Repositories\PirepRepository;
2017-08-15 12:36:49 +08:00
use App\Http\Resources\Pirep as PirepResource;
use App\Http\Resources\PirepComment as PirepCommentResource;
2018-01-05 09:33:23 +08:00
use App\Http\Resources\AcarsRoute as AcarsRouteResource;
2018-01-04 00:25:55 +08:00
use Symfony\Component\HttpKernel\Exception\BadRequestHttpException;
2017-08-15 12:36:49 +08:00
2018-01-05 09:33:23 +08:00
class PirepController extends RestController
2017-08-15 12:36:49 +08:00
{
2018-01-05 09:33:23 +08:00
protected $acarsRepo,
$geoSvc,
$pirepRepo,
$pirepSvc;
2018-01-05 09:33:23 +08:00
/**
* PirepController constructor.
* @param AcarsRepository $acarsRepo
* @param GeoService $geoSvc
* @param PirepRepository $pirepRepo
* @param PIREPService $pirepSvc
*/
public function __construct(
AcarsRepository $acarsRepo,
2017-12-28 10:52:37 +08:00
GeoService $geoSvc,
2017-12-27 04:54:28 +08:00
PirepRepository $pirepRepo,
PIREPService $pirepSvc
) {
$this->acarsRepo = $acarsRepo;
2017-12-28 10:52:37 +08:00
$this->geoSvc = $geoSvc;
$this->pirepRepo = $pirepRepo;
2017-12-27 04:54:28 +08:00
$this->pirepSvc = $pirepSvc;
}
2017-08-15 12:36:49 +08:00
public function get($id)
{
PirepResource::withoutWrapping();
return new PirepResource($this->pirepRepo->find($id));
2017-08-15 12:36:49 +08:00
}
/**
* Create a new PIREP and place it in a "inprogress" and "prefile" state
* Once ACARS updates are being processed, then it can go into an 'ENROUTE'
* status, and whatever other statuses may be defined
*
* TODO: Allow extra fields, etc to be set. Aircraft, etc
2018-01-05 09:33:23 +08:00
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
*/
public function prefile(Request $request)
{
2018-01-25 03:38:06 +08:00
Log::info('PIREP Prefile, user '.Auth::user()->id, $request->post());
2017-12-27 04:54:28 +08:00
$prefile_rules = [
'airline_id' => 'required|exists:airlines,id',
'aircraft_id' => 'required|exists:aircraft,id',
'dpt_airport_id' => 'required',
'arr_airport_id' => 'required',
'flight_id' => 'nullable',
'flight_number' => 'required',
'route_code' => 'nullable',
'route_leg' => 'nullable',
'flight_time' => ['nullable', 'integer'],
'planned_flight_time' => ['nullable', 'integer'],
'level' => 'required|integer',
'route' => 'nullable',
'notes' => 'nullable',
'created_at' => 'nullable|date',
];
$attrs = $this->getFromReq($request, $prefile_rules, [
2018-01-05 09:33:23 +08:00
'user_id' => Auth::user()->id,
'state' => PirepState::IN_PROGRESS,
'status' => PirepStatus::PREFILE,
]);
2017-12-27 04:54:28 +08:00
$pirep = new Pirep($attrs);
# Find if there's a duplicate, if so, let's work on that
$dupe_pirep = $this->pirepSvc->findDuplicate($pirep);
if($dupe_pirep !== false) {
$pirep = $dupe_pirep;
}
$pirep->save();
Log::info('PIREP PREFILED');
Log::info($pirep->id);
PirepResource::withoutWrapping();
2017-12-27 04:54:28 +08:00
return new PirepResource($pirep);
}
/**
* File the PIREP
* @param $id
* @param Request $request
* @return PirepResource
2018-01-04 00:25:55 +08:00
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
* @throws \Illuminate\Database\Eloquent\ModelNotFoundException
2017-12-27 04:54:28 +08:00
*/
public function file($id, Request $request)
{
2018-01-25 03:38:06 +08:00
Log::info('PIREP file, user ' . Auth::user()->id, $request->post());
2018-01-02 06:01:01 +08:00
2018-01-04 00:25:55 +08:00
$pirep = $this->pirepRepo->find($id);
if (empty($pirep)) {
throw new ModelNotFoundException('PIREP not found');
}
# Check if the status is cancelled...
if($pirep->state === PirepState::CANCELLED) {
throw new BadRequestHttpException('PIREP has been cancelled, updates can\'t be posted');
}
$file_rules = [
# actual flight time is required
'flight_time' => ['required', 'integer'],
'flight_number' => 'nullable',
'dpt_airport_id' => 'nullable',
'arr_airport_id' => 'nullable',
'airline_id' => 'nullable|exists:airlines,id',
'aircraft_id' => 'nullable|exists:aircraft,id',
'flight_id' => 'nullable',
'route_code' => 'nullable',
'route_leg' => 'nullable',
'planned_flight_time' => ['nullable', 'integer'],
'level' => 'nullable',
'route' => 'nullable',
'notes' => 'nullable',
'created_at' => 'nullable|date',
];
$attrs = $this->getFromReq($request, $file_rules, [
2018-01-05 09:33:23 +08:00
'state' => PirepState::PENDING,
'status' => PirepStatus::ARRIVED,
]);
2017-12-27 04:54:28 +08:00
2018-01-04 00:25:55 +08:00
$pirep_fields = [];
2017-12-27 04:54:28 +08:00
if($request->filled('fields')) {
$pirep_fields = $request->get('fields');
}
try {
$pirep = $this->pirepRepo->update($attrs, $id);
$pirep = $this->pirepSvc->create($pirep, $pirep_fields);
} catch (\Exception $e) {
Log::error($e);
}
PirepResource::withoutWrapping();
return new PirepResource($pirep);
}
2018-01-04 00:25:55 +08:00
/**
* Cancel the PIREP
* @param $id
* @param Request $request
* @return PirepResource
*/
public function cancel($id, Request $request)
{
2018-01-25 03:38:06 +08:00
Log::info('PIREP Cancel, user ' . Auth::user()->pilot_id, $request->post());
2018-01-04 00:25:55 +08:00
$attrs = [
'state' => PirepState::CANCELLED,
];
try {
$pirep = $this->pirepRepo->update($attrs, $id);
} catch (\Exception $e) {
Log::error($e);
}
PirepResource::withoutWrapping();
return new PirepResource($pirep);
}
/**
2017-12-29 04:35:28 +08:00
* Return the GeoJSON for the ACARS line
* @param $id
2017-12-29 04:35:28 +08:00
* @param Request $request
* @return \Illuminate\Contracts\Routing\ResponseFactory
*/
2018-01-05 09:33:23 +08:00
public function acars_geojson($id, Request $request)
{
2017-12-26 08:22:46 +08:00
$pirep = $this->pirepRepo->find($id);
2017-12-29 04:35:28 +08:00
$geodata = $this->geoSvc->getFeatureFromAcars($pirep);
2017-12-26 08:22:46 +08:00
2017-12-29 04:35:28 +08:00
return response(\json_encode($geodata), 200, [
'Content-Type' => 'application/json',
]);
}
2018-01-05 09:33:23 +08:00
/**
* Return the GeoJSON for the ACARS line
* @param $id
* @param Request $request
* @return AcarsRouteResource
*/
public function acars_get($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
AcarsRouteResource::withoutWrapping();
return new AcarsRouteResource(Acars::where([
'pirep_id' => $id,
'type' => AcarsType::FLIGHT_PATH
])->orderBy('created_at', 'asc')->get());
}
/**
* Post ACARS updates for a PIREP
* @param $id
* @param Request $request
* @return \Illuminate\Http\JsonResponse
2018-01-04 00:25:55 +08:00
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
*/
public function acars_store($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
2018-01-04 00:25:55 +08:00
# Check if the status is cancelled...
if ($pirep->state === PirepState::CANCELLED) {
throw new BadRequestHttpException('PIREP has been cancelled, updates can\'t be posted');
}
2018-01-25 03:38:06 +08:00
Log::info(
'Posting ACARS update (user: '.Auth::user()->pilot_id.', pirep id :'.$id.'): ',
$request->post()
);
2018-01-05 09:33:23 +08:00
$this->validate($request, ['positions' => 'required']);
$positions = $request->post('positions');
2018-01-05 09:33:23 +08:00
$acars_rules = [
'lat' => 'required|numeric',
'lon' => 'required|numeric',
'altitude' => 'nullable',
'level' => 'nullable',
'heading' => 'nullable',
'vs' => 'nullable',
'gs' => 'nullable',
'transponder' => 'nullable',
'autopilot' => 'nullable',
'fuel_flow' => 'nullable',
'log' => 'nullable',
'created_at' => 'nullable|date',
];
$count = 0;
2018-01-05 09:33:23 +08:00
foreach($positions as $position)
{
try {
$attrs = $this->getFromReq(
$position,
$acars_rules,
2018-01-05 09:33:23 +08:00
['pirep_id' => $id, 'type' => AcarsType::FLIGHT_PATH]
);
$update = Acars::create($attrs);
$update->save();
++$count;
2018-01-05 09:33:23 +08:00
} catch (\Exception $e) {
Log::error($e);
}
}
# Change the PIREP status
$pirep->status = PirepStatus::ENROUTE;
$pirep->save();
return $this->message($count . ' positions added', $count);
}
/**
* Post ACARS LOG update for a PIREP. These updates won't show up on the map
* But rather in a log file.
* @param $id
* @param Request $request
* @return \Illuminate\Http\JsonResponse
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
*/
public function acars_log($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
# Check if the status is cancelled...
if ($pirep->state === PirepState::CANCELLED) {
throw new BadRequestHttpException('PIREP has been cancelled, updates can\'t be posted');
}
2018-01-05 09:33:23 +08:00
Log::info('Posting ACARS log', $request->toArray());
$this->validate($request, ['logs' => 'required']);
$logs = $request->post('logs');
$rules = [
'log' => 'required',
'lat' => 'nullable',
'lon' => 'nullable',
'created_at' => 'nullable|date',
];
$count = 0;
foreach($logs as $log) {
$attrs = $this->getFromReq($log, $rules, ['pirep_id' => $id, 'type' => AcarsType::LOG]);
$acars = Acars::create($attrs);
$acars->save();
++$count;
}
return $this->message($count . ' logs added', $count);
2018-01-05 09:33:23 +08:00
}
/**
* Add a new comment
* @param $id
* @param Request $request
* @return \Illuminate\Http\Resources\Json\AnonymousResourceCollection
*/
public function comments_get($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
return PirepCommentResource::collection($pirep->comments);
}
/**
* Add a new comment
* @param $id
* @param Request $request
* @return PirepCommentResource
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
*/
public function comments_post($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
if ($pirep->state === PirepState::CANCELLED) {
throw new BadRequestHttpException('PIREP has been cancelled, comments can\'t be posted');
}
# validation
$this->validate($request, ['comment' => 'required']);
# Add it
$comment = new PirepComment($request->post());
$comment->pirep_id = $id;
$comment->user_id = Auth::user()->id;
$comment->save();
return new PirepCommentResource($comment);
}
2018-01-05 09:33:23 +08:00
/**
* @param $id
* @param Request $request
* @return AcarsRouteResource
*/
public function route_get($id, Request $request)
{
$this->pirepRepo->find($id);
AcarsRouteResource::withoutWrapping();
return new AcarsRouteResource(Acars::where([
'pirep_id' => $id,
'type' => AcarsType::ROUTE
])->orderBy('order', 'asc')->get());
}
/**
* Post the ROUTE for a PIREP, can be done from the ACARS log
* @param $id
* @param Request $request
* @return AcarsRouteResource
* @throws \Symfony\Component\HttpKernel\Exception\BadRequestHttpException
*/
public function route_post($id, Request $request)
{
$pirep = $this->pirepRepo->find($id);
# Check if the status is cancelled...
if ($pirep->state === PirepState::CANCELLED) {
throw new BadRequestHttpException('PIREP has been cancelled, updates can\'t be posted');
}
Log::info('Posting ACARS ROUTE', $request->toArray());
$this->validate($request, ['route' => 'required']);
2018-01-05 09:33:23 +08:00
$this->validate($request, [
'route.*.name' => 'required',
'route.*.order' => 'required|int',
'route.*.nav_type' => 'nullable|int',
'route.*.lat' => 'required|numeric',
'route.*.lon' => 'required|numeric',
]);
$route = $request->post('route', []);
2018-01-05 09:33:23 +08:00
foreach($route as $position) {
$attrs = [
'pirep_id' => $id,
'type' => AcarsType::ROUTE,
'name' => $position['name'],
'order' => $position['order'],
'lat' => $position['lat'],
'lon' => $position['lon'],
];
if(array_key_exists('nav_type', $position)) {
$attrs['nav_type'] = $position['nav_type'];
}
try {
$acars = Acars::create($attrs);
$acars->save();
} catch (\Exception $e) {
Log::error($e);
}
}
return $this->route_get($id, $request);
}
/**
* @param $id
* @param Request $request
* @return \Illuminate\Http\JsonResponse
*/
public function route_delete($id, Request $request)
{
$this->pirepRepo->find($id);
Acars::where([
'pirep_id' => $id,
'type' => AcarsType::ROUTE
])->delete();
2018-01-05 09:33:23 +08:00
return $this->message('Route deleted');
}
2017-08-15 12:36:49 +08:00
}