finish acceptance tests and refactor IotaCodesClient to service container

This commit is contained in:
Richard Morgan
2017-05-15 22:14:14 -04:00
parent 2c8c692890
commit e4e17dedc5
13 changed files with 338 additions and 24 deletions

View File

@@ -38,15 +38,18 @@ class AirportController extends Controller
*/ */
public function resourceList(Request $request) public function resourceList(Request $request)
{ {
$client = IotaClient::create(); $iotaCodesClient = app('IotaCodesClient');
$collection = $client->listAirports($request->input('autocomplete')); $airportsCollection = $iotaCodesClient->listAirports($request->input('autocomplete'));
if (! $airportsCollection) {
return $this->returnErrorMessage('failed to fetch airport list from IOTA api', 400);
}
//add pagination //add pagination
$per_page = $request->input('per_page', 10); $per_page = $request->input('per_page', 10);
$page = $request->input('page', 1); $page = $request->input('page', 1);
$paginator = new LengthAwarePaginator( $paginator = new LengthAwarePaginator(
$collection->forPage($page, $per_page), $airportsCollection->forPage($page, $per_page),
$collection->count(), $airportsCollection->count(),
$per_page, $per_page,
$page $page
); );
@@ -74,9 +77,9 @@ class AirportController extends Controller
*/ */
public function getAirport(Request $request, $code) public function getAirport(Request $request, $code)
{ {
$client = IotaClient::create(); $iotaCodesClient = app('IotaCodesClient');
$airport = $client->getAirport($code); $airport = $iotaCodesClient->getAirport($code);
if (is_null($airport)) { if (is_null($airport)) {
return $this->returnErrorMessage('not a valid airport code', 400); return $this->returnErrorMessage('not a valid airport code', 400);
} }

View File

@@ -19,7 +19,7 @@ class Controller extends BaseController
*/ */
protected function JsonApiResponse(ResourceInterface $resource, $statusCode, $includes = '') protected function JsonApiResponse(ResourceInterface $resource, $statusCode, $includes = '')
{ {
$manager = new Manager(); $manager = app('FractalManager');
$manager->setSerializer(new JsonApiSerializer('http://docker.dev:8080')); $manager->setSerializer(new JsonApiSerializer('http://docker.dev:8080'));
$manager->parseIncludes($includes); $manager->parseIncludes($includes);

View File

@@ -46,11 +46,9 @@ class FlightController extends Controller
$page $page
); );
$result = (new Collection($paginator, new TripTransformer(), 'trips')) $result = (new Collection($paginator, new FlightTransform(), 'trips'))
->setPaginator(new IlluminatePaginatorAdapter($paginator)); ->setPaginator(new IlluminatePaginatorAdapter($paginator));
$result = new Collection($flightCollection, new FlightTransform(), 'flights');
return $this->JsonApiResponse($result, 200); return $this->JsonApiResponse($result, 200);
} }
@@ -98,10 +96,10 @@ class FlightController extends Controller
{ {
/** @var \App\Libraries\Trips\Models\Flights $flight */ /** @var \App\Libraries\Trips\Models\Flights $flight */
$flight = Flights::find($id); $flight = Flights::find($id);
$deleted = $flight->delete(); if (! is_null($flight)) {
$deleted = $flight->delete();
}
$result = new Item($flight, new FlightTransform(), 'flights'); return response()->json(null, 204);
return $this->JsonApiResponse($result, 200);
} }
} }

View File

@@ -115,7 +115,7 @@ class TripController extends Controller
return $this->returnErrorMessage('destination not set', 400); return $this->returnErrorMessage('destination not set', 400);
} }
$client = IotaClient::create(); $client = app('IotaCodesClient');
$airport = $client->getAirport($destination); $airport = $client->getAirport($destination);
if (is_null($airport)) { if (is_null($airport)) {
return $this->returnErrorMessage('destination not a valid airport code', 400); return $this->returnErrorMessage('destination not a valid airport code', 400);

View File

@@ -3,17 +3,33 @@ namespace App\Libraries\IotaCodes;
class Airport class Airport
{ {
/**
* @var string
*/
public $name; public $name;
/**
* @var string
*/
public $code; public $code;
/**
* Airport constructor.
* @param $name
* @param $code
*/
protected function __construct($name, $code) protected function __construct($name, $code)
{ {
$this->name = $name; $this->name = $name;
$this->code = $code; $this->code = $code;
} }
/**
* Airport factory method
* @param $name
* @param $code
* @return \App\Libraries\IotaCodes\Airport
*/
public static function create($name, $code) public static function create($name, $code)
{ {
return new self($name, $code); return new self($name, $code);

View File

@@ -5,6 +5,12 @@ use League\Fractal\TransformerAbstract;
class AirportTransformer extends TransformerAbstract class AirportTransformer extends TransformerAbstract
{ {
/**
* Fractal Transformer for Airport Resource
*
* @param \App\Libraries\IotaCodes\Airport $airport
* @return array
*/
public function transform(Airport $airport) public function transform(Airport $airport)
{ {
return [ return [

View File

@@ -22,6 +22,11 @@ class Client
$this->guzzle = $guzzle; $this->guzzle = $guzzle;
} }
/**
* Factory Method
*
* @return \App\Libraries\IotaCodes\Client
*/
public static function create() public static function create()
{ {
return new self(new Guzzle([ return new self(new Guzzle([
@@ -31,6 +36,12 @@ class Client
])); ]));
} }
/**
* Get list of airports
*
* @param string $autoComplete
* @return \Illuminate\Support\Collection|false
*/
public function listAirports($autoComplete = '') public function listAirports($autoComplete = '')
{ {
$uri = '/api/v6/airports'; $uri = '/api/v6/airports';
@@ -48,14 +59,20 @@ class Client
]); ]);
} catch (\Exception $e) { } catch (\Exception $e) {
//todo handle error return false;
} }
$result = $response->getBody()->getContents(); $result = $response->getBody()->getContents();
if (empty($result)) {
return false;
}
app('cache')->put($key, $result, $minutes); app('cache')->put($key, $result, $minutes);
return $result; return $result;
}); });
if(! $response) {
return false;
}
$result = $this->toAirportCollection($response); $result = $this->toAirportCollection($response);
if ($autoComplete != '') { if ($autoComplete != '') {
$result = $this->autoCompleteMap($result, $autoComplete); $result = $this->autoCompleteMap($result, $autoComplete);
@@ -71,7 +88,7 @@ class Client
{ {
$uri = '/api/v6/airports'; $uri = '/api/v6/airports';
$cacheKey = 'airport.'.$code; $cacheKey = 'airport.'.$code;
$cacheMinutes = 1; $cacheMinutes = 60;
$response = app('cache')->get($cacheKey, function () use ($cacheKey, $cacheMinutes, $uri, $code) { $response = app('cache')->get($cacheKey, function () use ($cacheKey, $cacheMinutes, $uri, $code) {
try { try {
@@ -103,6 +120,9 @@ class Client
private function toAirportCollection($json) private function toAirportCollection($json)
{ {
$data = json_decode($json); $data = json_decode($json);
if (! $data) {
return new Collection();
}
$result = (new Collection($data->response)) $result = (new Collection($data->response))
->sortBy('name') ->sortBy('name')
->map(function ($item, $key) { ->map(function ($item, $key) {
@@ -112,6 +132,11 @@ class Client
return $result; return $result;
} }
/**
* @param \Illuminate\Support\Collection $collection
* @param $autoComplete
* @return Collection
*/
private function autoCompleteMap(Collection $collection, $autoComplete) private function autoCompleteMap(Collection $collection, $autoComplete)
{ {
return $collection->filter(function ($value, $key) use ($autoComplete) { return $collection->filter(function ($value, $key) use ($autoComplete) {

View File

@@ -0,0 +1,20 @@
<?php
namespace App\Libraries\IotaCodes;
use Illuminate\Support\ServiceProvider;
class IotaCodesServiceProvider extends ServiceProvider
{
/**
* Register any application services.
*
* @return void
*/
public function register()
{
$this->app->singleton('IotaCodesClient', function ($app) {
return Client::create();
});
}
}

View File

@@ -3,6 +3,7 @@
namespace App\Providers; namespace App\Providers;
use Illuminate\Support\ServiceProvider; use Illuminate\Support\ServiceProvider;
use League\Fractal\Manager;
class AppServiceProvider extends ServiceProvider class AppServiceProvider extends ServiceProvider
{ {
@@ -13,6 +14,8 @@ class AppServiceProvider extends ServiceProvider
*/ */
public function register() public function register()
{ {
// $this->app->singleton('FractalManager', function ($app) {
return new Manager();
});
} }
} }

View File

@@ -78,10 +78,11 @@ $app->singleton(
| |
*/ */
// $app->register(App\Providers\AppServiceProvider::class); $app->register(App\Providers\AppServiceProvider::class);
// $app->register(App\Providers\AuthServiceProvider::class); // $app->register(App\Providers\AuthServiceProvider::class);
// $app->register(App\Providers\EventServiceProvider::class); // $app->register(App\Providers\EventServiceProvider::class);
$app->register(Illuminate\Redis\RedisServiceProvider::class); $app->register(Illuminate\Redis\RedisServiceProvider::class);
$app->register(App\Libraries\IotaCodes\IotaCodesServiceProvider::class);
/* /*
|-------------------------------------------------------------------------- |--------------------------------------------------------------------------

View File

@@ -3,6 +3,8 @@
class AirportsCest class AirportsCest
{ {
protected $uri = '/airports';
public function _before(AcceptanceTester $I) public function _before(AcceptanceTester $I)
{ {
} }
@@ -12,7 +14,81 @@ class AirportsCest
} }
// tests // tests
public function tryToTest(AcceptanceTester $I) public function listAirportsReturnsJson(AcceptanceTester $I)
{ {
$params = [];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
}
public function listAirportsReturns1ItemWhenPerPageIs1(AcceptanceTester $I)
{
$params = [
'page' => 1,
'per_page' => 1
];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r);
$I->assertCount(1, $data->data, json_encode($data));
}
public function listAirportsReturns2ItemWhenPerPageIs2(AcceptanceTester $I)
{
$params = [
'page' => 1,
'per_page' => 2
];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r);
$I->assertCount(2, $data->data, json_encode($data));
}
public function getAirportReturnsTripJson(AcceptanceTester $I)
{
$I->sendGET($this->uri, []);
$r = $I->grabResponse();
$data = json_decode($r, true);
$airportId = $data['data'][0]['id'];
$params = [];
$uri = $this->uri . '/' . $airportId;
$I->sendGET($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r, true);
$I->assertEquals('airports', $data['data']['type']);
$I->assertEquals($airportId, $data['data']['id']);
}
public function getAirportReturns400IfTripIdNotFound(AcceptanceTester $I)
{
$params = [];
$uri = $this->uri . '/999999';
$I->sendGET($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(400, $r);
$I->seeResponseIsJson();
} }
} }

View File

@@ -3,6 +3,8 @@
class FlightsCest class FlightsCest
{ {
protected $uri = '/flights';
public function _before(AcceptanceTester $I) public function _before(AcceptanceTester $I)
{ {
} }
@@ -12,7 +14,133 @@ class FlightsCest
} }
// tests // tests
public function tryToTest(AcceptanceTester $I) public function listFlightsReturnsJson(AcceptanceTester $I)
{ {
$params = [];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
}
public function listFlightsReturns1ItemWhenPerPageIs1(AcceptanceTester $I)
{
$params = [
'page' => 1,
'per_page' => 1
];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r);
$I->assertCount(1, $data->data, json_encode($data));
}
public function listFlightsReturns2ItemWhenPerPageIs2(AcceptanceTester $I)
{
$params = [
'page' => 1,
'per_page' => 2
];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r);
$I->assertCount(2, $data->data, json_encode($data));
}
public function listFlightsReturns0ItemWhenPageIsBeyondEndOfCollection(AcceptanceTester $I)
{
$params = [
'page' => 99999,
'per_page' => 2
];
$I->sendGET($this->uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r);
$I->assertCount(0, $data->data, json_encode($data));
}
public function getFlightReturnsTripJson(AcceptanceTester $I)
{
$I->sendGET($this->uri, []);
$r = $I->grabResponse();
$data = json_decode($r, true);
$tripId = $data['data'][0]['id'];
$params = [];
$uri = $this->uri . '/' . $tripId;
$I->sendGET($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(200, $r);
$I->seeResponseIsJson();
$data = json_decode($r, true);
$I->assertEquals('flights', $data['data']['type']);
$I->assertEquals($tripId, $data['data']['id']);
}
public function getFlightReturns400IfTripIdNotFound(AcceptanceTester $I)
{
$params = [];
$uri = $this->uri . '/999999';
$I->sendGET($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(400, $r);
$I->seeResponseIsJson();
}
public function removeFlightRemovesFlightFromTrips(AcceptanceTester $I)
{
$tripId = $this->getFirstTripId($I);
$r = $this->addFlightToTrip($I, 'YUL', $tripId);
$data = json_decode($r, true);
$flightId = $data['data']['attributes']['flights'][0]['id'];
$uri = $this->uri .'/'.$flightId;
$I->sendDELETE($uri);
$I->seeResponseCodeIs(204);
}
private function getFirstTripId(AcceptanceTester $I)
{
$uri = '/trips/';
$I->sendGET($this->uri, []);
$r = $I->grabResponse();
$data = json_decode($r, true);
$tripId = $data['data'][0]['id'];
return $tripId;
}
private function addFlightToTrip(AcceptanceTester $I, $destination, $tripId)
{
$uri = '/trips/' . $tripId .'/flights';
$I->sendPost($uri, [
'destination' => $destination
]);
return $I->grabResponse();
} }
} }

View File

@@ -39,7 +39,7 @@ class TripsCest
$I->seeResponseIsJson(); $I->seeResponseIsJson();
$data = json_decode($r); $data = json_decode($r);
$I->assertCount(1, $data->data, $data); $I->assertCount(1, $data->data, json_encode($data));
} }
public function listTripReturns2ItemWhenPerPageIs2(AcceptanceTester $I) public function listTripReturns2ItemWhenPerPageIs2(AcceptanceTester $I)
@@ -56,7 +56,7 @@ class TripsCest
$I->seeResponseIsJson(); $I->seeResponseIsJson();
$data = json_decode($r); $data = json_decode($r);
$I->assertCount(2, $data->data, $data); $I->assertCount(2, $data->data, json_encode($data));
} }
public function getTripReturnsTripJson(AcceptanceTester $I) public function getTripReturnsTripJson(AcceptanceTester $I)
@@ -114,4 +114,42 @@ class TripsCest
$data = json_decode($r, true); $data = json_decode($r, true);
$I->assertCount($nbFlights+1, $data['data']['attributes']['flights']); $I->assertCount($nbFlights+1, $data['data']['attributes']['flights']);
} }
public function addFlightReturns400WhenDestinationInvalid(AcceptanceTester $I)
{
$I->sendGET($this->uri, []);
$r = $I->grabResponse();
$data = json_decode($r, true);
$tripId = $data['data'][0]['id'];
$nbFlights = count($data['data'][0]['attributes']['flights']);
$params = [
'destination' => 'BlueFin'
];
$uri = $this->uri . '/' . $tripId .'/flights';
$I->sendPost($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(400, $r);
$I->seeResponseIsJson();
}
public function addFlightReturns400WhenDestinationNotSet(AcceptanceTester $I)
{
$I->sendGET($this->uri, []);
$r = $I->grabResponse();
$data = json_decode($r, true);
$tripId = $data['data'][0]['id'];
$nbFlights = count($data['data'][0]['attributes']['flights']);
$params = [];
$uri = $this->uri . '/' . $tripId .'/flights';
$I->sendPost($uri, $params);
$r = $I->grabResponse();
$I->seeResponseCodeIs(400, $r);
$I->seeResponseIsJson();
}
} }