mirror of
https://github.com/codeigniter4/CodeIgniter4.git
synced 2025-02-20 11:44:28 +08:00
363 lines
12 KiB
PHP
363 lines
12 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
/**
|
|
* This file is part of CodeIgniter 4 framework.
|
|
*
|
|
* (c) CodeIgniter Foundation <admin@codeigniter.com>
|
|
*
|
|
* For the full copyright and license information, please view
|
|
* the LICENSE file that was distributed with this source code.
|
|
*/
|
|
|
|
namespace CodeIgniter\API;
|
|
|
|
use CodeIgniter\Format\FormatterInterface;
|
|
use CodeIgniter\HTTP\IncomingRequest;
|
|
use CodeIgniter\HTTP\ResponseInterface;
|
|
|
|
/**
|
|
* Provides common, more readable, methods to provide
|
|
* consistent HTTP responses under a variety of common
|
|
* situations when working as an API.
|
|
*
|
|
* @property bool $stringAsHtml Whether to treat string data as HTML in JSON response.
|
|
* Setting `true` is only for backward compatibility.
|
|
*/
|
|
trait ResponseTrait
|
|
{
|
|
/**
|
|
* Allows child classes to override the
|
|
* status code that is used in their API.
|
|
*
|
|
* @var array<string, int>
|
|
*/
|
|
protected $codes = [
|
|
'created' => 201,
|
|
'deleted' => 200,
|
|
'updated' => 200,
|
|
'no_content' => 204,
|
|
'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,
|
|
];
|
|
|
|
/**
|
|
* How to format the response data.
|
|
* Either 'json' or 'xml'. If null is set, it will be determined through
|
|
* content negotiation.
|
|
*
|
|
* @var string|null
|
|
* @phpstan-var 'html'|'json'|'xml'|null
|
|
*/
|
|
protected $format = 'json';
|
|
|
|
/**
|
|
* Current Formatter instance. This is usually set by ResponseTrait::format
|
|
*
|
|
* @var FormatterInterface|null
|
|
*/
|
|
protected $formatter;
|
|
|
|
/**
|
|
* Provides a single, simple method to return an API response, formatted
|
|
* to match the requested format, with proper content-type and status code.
|
|
*
|
|
* @param array|string|null $data
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function respond($data = null, ?int $status = null, string $message = '')
|
|
{
|
|
if ($data === null && $status === null) {
|
|
$status = 404;
|
|
$output = null;
|
|
$this->format($data);
|
|
} elseif ($data === null && is_numeric($status)) {
|
|
$output = null;
|
|
$this->format($data);
|
|
} else {
|
|
$status ??= 200;
|
|
$output = $this->format($data);
|
|
}
|
|
|
|
if ($output !== null) {
|
|
if ($this->format === 'json') {
|
|
return $this->response->setJSON($output)->setStatusCode($status, $message);
|
|
}
|
|
|
|
if ($this->format === 'xml') {
|
|
return $this->response->setXML($output)->setStatusCode($status, $message);
|
|
}
|
|
}
|
|
|
|
return $this->response->setBody($output)->setStatusCode($status, $message);
|
|
}
|
|
|
|
/**
|
|
* Used for generic failures that no custom methods exist for.
|
|
*
|
|
* @param array|string $messages
|
|
* @param int $status HTTP status code
|
|
* @param string|null $code Custom, API-specific, error code
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function fail($messages, int $status = 400, ?string $code = null, string $customMessage = '')
|
|
{
|
|
if (! is_array($messages)) {
|
|
$messages = ['error' => $messages];
|
|
}
|
|
|
|
$response = [
|
|
'status' => $status,
|
|
'error' => $code ?? $status,
|
|
'messages' => $messages,
|
|
];
|
|
|
|
return $this->respond($response, $status, $customMessage);
|
|
}
|
|
|
|
// --------------------------------------------------------------------
|
|
// Response Helpers
|
|
// --------------------------------------------------------------------
|
|
|
|
/**
|
|
* Used after successfully creating a new resource.
|
|
*
|
|
* @param array|string|null $data
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function respondCreated($data = null, string $message = '')
|
|
{
|
|
return $this->respond($data, $this->codes['created'], $message);
|
|
}
|
|
|
|
/**
|
|
* Used after a resource has been successfully deleted.
|
|
*
|
|
* @param array|string|null $data
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function respondDeleted($data = null, string $message = '')
|
|
{
|
|
return $this->respond($data, $this->codes['deleted'], $message);
|
|
}
|
|
|
|
/**
|
|
* Used after a resource has been successfully updated.
|
|
*
|
|
* @param array|string|null $data
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function respondUpdated($data = null, string $message = '')
|
|
{
|
|
return $this->respond($data, $this->codes['updated'], $message);
|
|
}
|
|
|
|
/**
|
|
* Used after a command has been successfully executed but there is no
|
|
* meaningful reply to send back to the client.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function respondNoContent(string $message = 'No Content')
|
|
{
|
|
return $this->respond(null, $this->codes['no_content'], $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.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failUnauthorized(string $description = 'Unauthorized', ?string $code = null, string $message = '')
|
|
{
|
|
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.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failForbidden(string $description = 'Forbidden', ?string $code = null, string $message = '')
|
|
{
|
|
return $this->fail($description, $this->codes['forbidden'], $code, $message);
|
|
}
|
|
|
|
/**
|
|
* Used when a specified resource cannot be found.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failNotFound(string $description = 'Not Found', ?string $code = null, string $message = '')
|
|
{
|
|
return $this->fail($description, $this->codes['resource_not_found'], $code, $message);
|
|
}
|
|
|
|
/**
|
|
* Used when the data provided by the client cannot be validated on one or more fields.
|
|
*
|
|
* @param list<string>|string $errors
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failValidationErrors($errors, ?string $code = null, string $message = '')
|
|
{
|
|
return $this->fail($errors, $this->codes['invalid_data'], $code, $message);
|
|
}
|
|
|
|
/**
|
|
* Use when trying to create a new resource and it already exists.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failResourceExists(string $description = 'Conflict', ?string $code = null, string $message = '')
|
|
{
|
|
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.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failResourceGone(string $description = 'Gone', ?string $code = null, string $message = '')
|
|
{
|
|
return $this->fail($description, $this->codes['resource_gone'], $code, $message);
|
|
}
|
|
|
|
/**
|
|
* Used when the user has made too many requests for the resource recently.
|
|
*
|
|
* @return ResponseInterface
|
|
*/
|
|
protected function failTooManyRequests(string $description = 'Too Many Requests', ?string $code = null, string $message = '')
|
|
{
|
|
return $this->fail($description, $this->codes['too_many_requests'], $code, $message);
|
|
}
|
|
|
|
/**
|
|
* Used when there is a server error.
|
|
*
|
|
* @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.
|
|
*/
|
|
protected function failServerError(string $description = 'Internal Server Error', ?string $code = null, string $message = ''): ResponseInterface
|
|
{
|
|
return $this->fail($description, $this->codes['server_error'], $code, $message);
|
|
}
|
|
|
|
// --------------------------------------------------------------------
|
|
// Utility Methods
|
|
// --------------------------------------------------------------------
|
|
|
|
/**
|
|
* Handles formatting a response. Currently, makes some heavy assumptions
|
|
* and needs updating! :)
|
|
*
|
|
* @param array|string|null $data
|
|
*
|
|
* @return string|null
|
|
*/
|
|
protected function format($data = null)
|
|
{
|
|
$format = service('format');
|
|
|
|
$mime = ($this->format === null) ? $format->getConfig()->supportedResponseFormats[0]
|
|
: "application/{$this->format}";
|
|
|
|
// Determine correct response type through content negotiation if not explicitly declared
|
|
if (
|
|
! in_array($this->format, ['json', 'xml'], true)
|
|
&& $this->request instanceof IncomingRequest
|
|
) {
|
|
$mime = $this->request->negotiate(
|
|
'media',
|
|
$format->getConfig()->supportedResponseFormats,
|
|
false,
|
|
);
|
|
}
|
|
|
|
$this->response->setContentType($mime);
|
|
|
|
// if we don't have a formatter, make one
|
|
if (! isset($this->formatter)) {
|
|
// if no formatter, use the default
|
|
$this->formatter = $format->getFormatter($mime);
|
|
}
|
|
|
|
$asHtml = $this->stringAsHtml ?? false;
|
|
|
|
// Returns as HTML.
|
|
if (
|
|
($mime === 'application/json' && $asHtml && is_string($data))
|
|
|| ($mime !== 'application/json' && is_string($data))
|
|
) {
|
|
// 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);
|
|
$this->format = 'html';
|
|
|
|
return $data;
|
|
}
|
|
|
|
if ($mime !== 'application/json') {
|
|
// Recursively convert objects into associative arrays
|
|
// Conversion not required for JSONFormatter
|
|
$data = json_decode(json_encode($data), true);
|
|
}
|
|
|
|
return $this->formatter->format($data);
|
|
}
|
|
|
|
/**
|
|
* Sets the format the response should be in.
|
|
*
|
|
* @param string|null $format Response format
|
|
* @phpstan-param 'json'|'xml' $format
|
|
*
|
|
* @return $this
|
|
*/
|
|
protected function setResponseFormat(?string $format = null)
|
|
{
|
|
$this->format = ($format === null) ? null : strtolower($format);
|
|
|
|
return $this;
|
|
}
|
|
}
|