2016-11-22 00:26:30 -06:00
|
|
|
<?php namespace CodeIgniter\API;
|
|
|
|
|
2017-01-20 00:57:53 -08:00
|
|
|
/**
|
|
|
|
* CodeIgniter
|
|
|
|
*
|
|
|
|
* An open source application development framework for PHP
|
|
|
|
*
|
|
|
|
* This content is released under the MIT License (MIT)
|
|
|
|
*
|
2018-12-26 21:49:51 -08:00
|
|
|
* Copyright (c) 2014-2019 British Columbia Institute of Technology
|
2017-01-20 00:57:53 -08:00
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*
|
2018-11-10 02:57:39 -02:00
|
|
|
* @package CodeIgniter
|
|
|
|
* @author CodeIgniter Dev Team
|
2018-12-26 21:49:51 -08:00
|
|
|
* @copyright 2014-2019 British Columbia Institute of Technology (https://bcit.ca/)
|
2018-11-10 02:57:39 -02:00
|
|
|
* @license https://opensource.org/licenses/MIT MIT License
|
|
|
|
* @link https://codeigniter.com
|
|
|
|
* @since Version 3.0.0
|
2017-01-20 00:57:53 -08:00
|
|
|
* @filesource
|
|
|
|
*/
|
2018-11-10 02:57:39 -02:00
|
|
|
|
2017-04-02 21:57:04 -05:00
|
|
|
use Config\Format;
|
2017-03-03 22:26:20 +00:00
|
|
|
use CodeIgniter\HTTP\Response;
|
|
|
|
|
2016-11-22 00:26:30 -06:00
|
|
|
/**
|
2017-03-03 22:26:20 +00:00
|
|
|
* Response trait.
|
2016-11-22 00:26:30 -06:00
|
|
|
*
|
|
|
|
* Provides common, more readable, methods to provide
|
|
|
|
* consistent HTTP responses under a variety of common
|
|
|
|
* situations when working as an API.
|
|
|
|
*
|
2017-07-22 21:19:59 -03:00
|
|
|
* @property \CodeIgniter\HTTP\IncomingRequest $request
|
|
|
|
* @property \CodeIgniter\HTTP\Response $response
|
2016-11-22 00:26:30 -06:00
|
|
|
*
|
|
|
|
* @package CodeIgniter\API
|
|
|
|
*/
|
|
|
|
trait ResponseTrait
|
|
|
|
{
|
2017-07-05 14:42:45 -07:00
|
|
|
|
2017-02-04 16:36:08 +09:00
|
|
|
/**
|
|
|
|
* Allows child classes to override the
|
|
|
|
* status code that is used in their API.
|
|
|
|
*
|
|
|
|
* @var array
|
|
|
|
*/
|
|
|
|
protected $codes = [
|
2018-11-10 02:57:39 -02:00
|
|
|
'created' => 201,
|
|
|
|
'deleted' => 200,
|
|
|
|
'invalid_request' => 400,
|
|
|
|
'unsupported_response_type' => 400,
|
|
|
|
'invalid_scope' => 400,
|
|
|
|
'temporarily_unavailable' => 400,
|
|
|
|
'invalid_grant' => 400,
|
|
|
|
'invalid_credentials' => 400,
|
|
|
|
'invalid_refresh' => 400,
|
|
|
|
'no_data' => 400,
|
|
|
|
'invalid_data' => 400,
|
|
|
|
'access_denied' => 401,
|
|
|
|
'unauthorized' => 401,
|
|
|
|
'invalid_client' => 401,
|
|
|
|
'forbidden' => 403,
|
|
|
|
'resource_not_found' => 404,
|
|
|
|
'not_acceptable' => 406,
|
|
|
|
'resource_exists' => 409,
|
|
|
|
'conflict' => 409,
|
|
|
|
'resource_gone' => 410,
|
|
|
|
'payload_too_large' => 413,
|
|
|
|
'unsupported_media_type' => 415,
|
|
|
|
'too_many_requests' => 429,
|
|
|
|
'server_error' => 500,
|
|
|
|
'unsupported_grant_type' => 501,
|
|
|
|
'not_implemented' => 501,
|
2017-02-04 16:36:08 +09:00
|
|
|
];
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides a single, simple method to return an API response, formatted
|
|
|
|
* to match the requested format, with proper content-type and status code.
|
|
|
|
*
|
2018-11-10 02:57:39 -02:00
|
|
|
* @param null $data
|
|
|
|
* @param integer $status
|
|
|
|
* @param string $message
|
2017-02-04 16:36:08 +09:00
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function respond($data = null, int $status = null, string $message = '')
|
|
|
|
{
|
|
|
|
// If data is null and status code not provided, exit and bail
|
|
|
|
if ($data === null && $status === null)
|
|
|
|
{
|
|
|
|
$status = 404;
|
|
|
|
|
|
|
|
// Create the output var here in case of $this->response([]);
|
|
|
|
$output = null;
|
|
|
|
} // If data is null but status provided, keep the output empty.
|
|
|
|
elseif ($data === null && is_numeric($status))
|
|
|
|
{
|
|
|
|
$output = null;
|
2017-07-05 14:42:45 -07:00
|
|
|
}
|
|
|
|
else
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
$status = empty($status) ? 200 : $status;
|
|
|
|
$output = $this->format($data);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->response->setBody($output)
|
2017-07-05 14:42:45 -07:00
|
|
|
->setStatusCode($status, $message);
|
2017-02-04 16:36:08 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used for generic failures that no custom methods exist for.
|
|
|
|
*
|
2017-07-22 21:14:21 -03:00
|
|
|
* @param string|array $messages
|
2018-11-10 02:57:39 -02:00
|
|
|
* @param integer|null $status HTTP status code
|
|
|
|
* @param string|null $code Custom, API-specific, error code
|
2017-07-22 21:14:21 -03:00
|
|
|
* @param string $customMessage
|
2017-02-04 16:36:08 +09:00
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
public function fail($messages, int $status = 400, string $code = null, string $customMessage = '')
|
|
|
|
{
|
2018-11-10 02:57:39 -02:00
|
|
|
if (! is_array($messages))
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
2017-11-07 03:37:23 -02:00
|
|
|
$messages = ['error' => $messages];
|
2017-02-04 16:36:08 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
$response = [
|
2018-11-10 02:57:39 -02:00
|
|
|
'status' => $status,
|
|
|
|
'error' => $code === null ? $status : $code,
|
|
|
|
'messages' => $messages,
|
2017-02-04 16:36:08 +09:00
|
|
|
];
|
|
|
|
|
|
|
|
return $this->respond($response, $status, $customMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// Response Helpers
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used after successfully creating a new resource.
|
|
|
|
*
|
2017-03-10 23:31:20 +00:00
|
|
|
* @param mixed $data Data.
|
|
|
|
* @param string $message Message.
|
2017-02-04 16:36:08 +09:00
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-03-10 23:31:20 +00:00
|
|
|
public function respondCreated($data = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->respond($data, $this->codes['created'], $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used after a resource has been successfully deleted.
|
|
|
|
*
|
2017-03-10 23:31:20 +00:00
|
|
|
* @param mixed $data Data.
|
|
|
|
* @param string $message Message.
|
2017-02-04 16:36:08 +09:00
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-03-10 23:31:20 +00:00
|
|
|
public function respondDeleted($data = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->respond($data, $this->codes['deleted'], $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used when the client is either didn't send authorization information,
|
|
|
|
* or had bad authorization credentials. User is encouraged to try again
|
|
|
|
* with the proper information.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failUnauthorized(string $description = 'Unauthorized', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['unauthorized'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used when access is always denied to this resource and no amount
|
|
|
|
* of trying again will help.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failForbidden(string $description = 'Forbidden', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['forbidden'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used when a specified resource cannot be found.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failNotFound(string $description = 'Not Found', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['resource_not_found'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used when the data provided by the client cannot be validated.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failValidationError(string $description = 'Bad Request', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['invalid_data'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use when trying to create a new resource and it already exists.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failResourceExists(string $description = 'Conflict', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['resource_exists'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use when a resource was previously deleted. This is different than
|
|
|
|
* Not Found, because here we know the data previously existed, but is now gone,
|
|
|
|
* where Not Found means we simply cannot find any information about it.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failResourceGone(string $description = 'Gone', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['resource_gone'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used when the user has made too many requests for the resource recently.
|
|
|
|
*
|
|
|
|
* @param string $description
|
2017-07-24 23:03:42 -03:00
|
|
|
* @param string $code
|
2017-02-04 16:36:08 +09:00
|
|
|
* @param string $message
|
|
|
|
*
|
|
|
|
* @return mixed
|
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failTooManyRequests(string $description = 'Too Many Requests', string $code = null, string $message = '')
|
2017-02-04 16:36:08 +09:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['too_many_requests'], $code, $message);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
2017-03-03 22:26:20 +00:00
|
|
|
/**
|
|
|
|
* Used when there is a server error.
|
|
|
|
*
|
2017-03-06 09:57:45 +00:00
|
|
|
* @param string $description The error message to show the user.
|
|
|
|
* @param string|null $code A custom, API-specific, error code.
|
|
|
|
* @param string $message A custom "reason" message to return.
|
2017-03-03 22:26:20 +00:00
|
|
|
*
|
2017-03-06 09:57:45 +00:00
|
|
|
* @return Response The value of the Response's send() method.
|
2017-03-03 22:26:20 +00:00
|
|
|
*/
|
2017-11-07 03:31:31 -02:00
|
|
|
public function failServerError(string $description = 'Internal Server Error', string $code = null, string $message = ''): Response
|
2017-03-03 22:26:20 +00:00
|
|
|
{
|
|
|
|
return $this->fail($description, $this->codes['server_error'], $code, $message);
|
|
|
|
}
|
2017-02-04 16:36:08 +09:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
// Utility Methods
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles formatting a response. Currently makes some heavy assumptions
|
|
|
|
* and needs updating! :)
|
|
|
|
*
|
|
|
|
* @param null $data
|
|
|
|
*
|
|
|
|
* @return null|string
|
|
|
|
*/
|
|
|
|
protected function format($data = null)
|
|
|
|
{
|
|
|
|
// If the data is a string, there's not much we can do to it...
|
|
|
|
if (is_string($data))
|
|
|
|
{
|
2017-07-05 14:42:45 -07:00
|
|
|
// The content type should be text/... and not application/...
|
|
|
|
$contentType = $this->response->getHeaderLine('Content-Type');
|
|
|
|
$contentType = str_replace('application/json', 'text/html', $contentType);
|
|
|
|
$contentType = str_replace('application/', 'text/', $contentType);
|
|
|
|
$this->response->setContentType($contentType);
|
2017-02-04 16:36:08 +09:00
|
|
|
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
2018-05-15 22:30:09 -05:00
|
|
|
// Determine correct response type through content negotiation
|
|
|
|
$config = new Format();
|
2018-10-09 11:36:06 +08:00
|
|
|
$format = $this->request->negotiate('media', $config->supportedResponseFormats, false);
|
2018-05-15 22:30:09 -05:00
|
|
|
|
|
|
|
$this->response->setContentType($format);
|
|
|
|
|
2017-07-05 14:42:45 -07:00
|
|
|
// if we don't have a formatter, make one
|
2018-11-10 02:57:39 -02:00
|
|
|
if (! isset($this->formatter))
|
2017-07-05 14:42:45 -07:00
|
|
|
{
|
|
|
|
// if no formatter, use the default
|
|
|
|
$this->formatter = $config->getFormatter($format);
|
|
|
|
}
|
2017-02-04 16:36:08 +09:00
|
|
|
|
2018-01-01 14:40:20 -02:00
|
|
|
if ($format !== 'application/json')
|
|
|
|
{
|
|
|
|
// Recursively convert objects into associative arrays
|
|
|
|
// Conversion not required for JSONFormatter
|
|
|
|
$data = json_decode(json_encode($data), true);
|
|
|
|
}
|
2017-08-04 08:26:15 -03:00
|
|
|
|
2017-07-05 14:42:45 -07:00
|
|
|
return $this->formatter->format($data);
|
2017-02-04 16:36:08 +09:00
|
|
|
}
|
|
|
|
|
2016-11-22 00:26:30 -06:00
|
|
|
}
|