<?php
/**
* @file classes/core/APIRouter.php
*
* Copyright (c) 2014-2021 Simon Fraser University
* Copyright (c) 2000-2021 John Willinsky
* Distributed under the GNU GPL v3. For full terms see the file docs/COPYING.
*
* @class APIRouter
*
* @ingroup core
*
* @brief Map HTTP requests to a REST API using the Slim microframework.
*
* Requests for [index.php]/api are intercepted for site-level API requests,
* and requests for [index.php]/{contextPath}/api are intercepted for
* context-level API requests.
*/
namespace PKP\core;
use APP\core\Application;
use Exception;
use PKP\handler\APIHandler;
use PKP\session\SessionManager;
class APIRouter extends PKPRouter
{
/**
* Determines path info parts
*
*/
protected function getPathInfoParts(): array
{
return explode('/', trim($_SERVER['PATH_INFO'] ?? '', '/'));
}
/**
* Determines whether this router can route the given request.
*
* @param PKPRequest $request
*
* @return bool true, if the router supports this request, otherwise false
*/
public function supports($request): bool
{
$pathInfoParts = $this->getPathInfoParts();
if (!is_null($pathInfoParts) && count($pathInfoParts) >= 2 && $pathInfoParts[1] == 'api') {
// Context-specific API requests: [index.php]/{contextPath}/api
return true;
}
return false;
}
/**
* Get the API version
*
* @return string
*/
public function getVersion()
{
$pathInfoParts = $this->getPathInfoParts();
return Core::cleanFileVar($pathInfoParts[2] ?? '');
}
/**
* Get the entity being requested
*
* @return string
*/
public function getEntity()
{
$pathInfoParts = $this->getPathInfoParts();
return Core::cleanFileVar($pathInfoParts[3] ?? '');
}
//
// Implement template methods from PKPRouter
//
/**
* @copydoc PKPRouter::route()
*/
public function route($request)
{
// Ensure slim library is available
require_once('lib/pkp/lib/vendor/autoload.php');
$sourceFile = sprintf('api/%s/%s/index.php', $this->getVersion(), $this->getEntity());
if (!file_exists($sourceFile)) {
http_response_code('404');
header('Content-Type: application/json');
echo json_encode([
'error' => 'api.404.endpointNotFound',
'errorMessage' => __('api.404.endpointNotFound'),
]);
exit;
}
if (!SessionManager::isDisabled()) {
// Initialize session
SessionManager::getManager();
}
$handler = require('./' . $sourceFile);
$this->setHandler($handler);
$handler->getApp()->run();
}
/**
* Get the requested operation
*
* @param PKPRequest $request
*
* @return string
*/
public function getRequestedOp($request)
{
/** @var APIHandler */
$handler = $this->getHandler();
$container = $handler->getApp()->getContainer();
$router = $container->get('router');
$slimRequest = $handler->getSlimRequest();
$routeInfo = $router->dispatch($slimRequest);
if (isset($routeInfo[1])) {
$route = $router->lookupRoute($routeInfo[1]);
$callable = $route->getCallable();
if (is_array($callable) && count($callable) == 2) {
return $callable[1];
}
}
return '';
}
/**
* @copydoc PKPRouter::handleAuthorizationFailure()
*/
public function handleAuthorizationFailure(
$request,
$authorizationMessage,
array $messageParams = []
) {
http_response_code('403');
header('Content-Type: application/json');
echo json_encode([
'error' => $authorizationMessage,
'errorMessage' => __($authorizationMessage, $messageParams),
]);
exit;
}
/**
* @copydoc PKPRouter::url()
*
* @param null|mixed $newContext
* @param null|mixed $endpoint
* @param null|mixed $op
* @param null|mixed $path
* @param null|mixed $params
* @param null|mixed $anchor
*/
public function url(
PKPRequest $request,
?string $newContext = null,
$endpoint = null,
$op = null,
$path = null,
$params = null,
$anchor = null,
$escape = false
) {
// APIHandlers do not understand $op, $path or $anchor. All routing is baked
// into the $endpoint string. It only accepts a string as the $newContext,
// since it relies on this when path info is disabled.
if (!is_null($op) || !is_null($path) || !is_null($anchor) || !is_scalar($newContext)) {
throw new Exception('APIRouter::url() should not be called with an op, path or anchor. If a new context is passed, the context path must be passed instead of the context object.');
}
//
// Base URL and Context
//
$baseUrlAndContext = $this->_urlGetBaseAndContext($request, $newContext);
$baseUrl = array_shift($baseUrlAndContext);
$context = array_shift($baseUrlAndContext);
//
// Additional query parameters
//
$additionalParameters = $this->_urlGetAdditionalParameters($request, $params, $escape);
//
// Assemble URL
//
$pathInfoArray = array_merge(
$context,
['api', Application::API_VERSION, $endpoint]
);
$queryParametersArray = $additionalParameters;
return $this->_urlFromParts($baseUrl, $pathInfoArray, $queryParametersArray, $anchor, $escape);
}
}
if (!PKP_STRICT_MODE) {
class_alias('\PKP\core\APIRouter', '\APIRouter');
}
|