You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1570 lines
40 KiB
1570 lines
40 KiB
<?php
|
|
/**
|
|
* CodeIgniter
|
|
*
|
|
* An open source application development framework for PHP
|
|
*
|
|
* This content is released under the MIT License (MIT)
|
|
*
|
|
* Copyright (c) 2014-2019 British Columbia Institute of Technology
|
|
* Copyright (c) 2019 CodeIgniter Foundation
|
|
*
|
|
* 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.
|
|
*
|
|
* @package CodeIgniter
|
|
* @author CodeIgniter Dev Team
|
|
* @copyright 2019 CodeIgniter Foundation
|
|
* @license https://opensource.org/licenses/MIT MIT License
|
|
* @link https://codeigniter.com
|
|
* @since Version 4.0.0
|
|
* @filesource
|
|
*/
|
|
|
|
namespace CodeIgniter\Router;
|
|
|
|
use CodeIgniter\HTTP\Request;
|
|
use Config\Services;
|
|
use CodeIgniter\Autoloader\FileLocator;
|
|
use CodeIgniter\Router\Exceptions\RouterException;
|
|
|
|
/**
|
|
* Class RouteCollection
|
|
*
|
|
* @todo Implement nested resource routing (See CakePHP)
|
|
*
|
|
* @package CodeIgniter\Router
|
|
*/
|
|
class RouteCollection implements RouteCollectionInterface
|
|
{
|
|
|
|
/**
|
|
* The namespace to be added to any Controllers.
|
|
* Defaults to the global namespaces (\)
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $defaultNamespace = '\\';
|
|
|
|
/**
|
|
* The name of the default controller to use
|
|
* when no other controller is specified.
|
|
*
|
|
* Not used here. Pass-thru value for Router class.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $defaultController = 'Home';
|
|
|
|
/**
|
|
* The name of the default method to use
|
|
* when no other method has been specified.
|
|
*
|
|
* Not used here. Pass-thru value for Router class.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $defaultMethod = 'index';
|
|
|
|
/**
|
|
* The placeholder used when routing 'resources'
|
|
* when no other placeholder has been specified.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $defaultPlaceholder = 'any';
|
|
|
|
/**
|
|
* Whether to convert dashes to underscores in URI.
|
|
*
|
|
* Not used here. Pass-thru value for Router class.
|
|
*
|
|
* @var boolean
|
|
*/
|
|
protected $translateURIDashes = false;
|
|
|
|
/**
|
|
* Whether to match URI against Controllers
|
|
* when it doesn't match defined routes.
|
|
*
|
|
* Not used here. Pass-thru value for Router class.
|
|
*
|
|
* @var boolean
|
|
*/
|
|
protected $autoRoute = true;
|
|
|
|
/**
|
|
* A callable that will be shown
|
|
* when the route cannot be matched.
|
|
*
|
|
* @var string|\Closure
|
|
*/
|
|
protected $override404;
|
|
|
|
/**
|
|
* Defined placeholders that can be used
|
|
* within the
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $placeholders = [
|
|
'any' => '.*',
|
|
'segment' => '[^/]+',
|
|
'alphanum' => '[a-zA-Z0-9]+',
|
|
'num' => '[0-9]+',
|
|
'alpha' => '[a-zA-Z]+',
|
|
'hash' => '[^/]+',
|
|
];
|
|
|
|
/**
|
|
* An array of all routes and their mappings.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $routes = [
|
|
'*' => [],
|
|
'options' => [],
|
|
'get' => [],
|
|
'head' => [],
|
|
'post' => [],
|
|
'put' => [],
|
|
'delete' => [],
|
|
'trace' => [],
|
|
'connect' => [],
|
|
'cli' => [],
|
|
];
|
|
|
|
/**
|
|
* Array of routes options
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $routesOptions = [];
|
|
|
|
/**
|
|
* The current method that the script is being called by.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $HTTPVerb;
|
|
|
|
/**
|
|
* The default list of HTTP methods (and CLI for command line usage)
|
|
* that is allowed if no other method is provided.
|
|
*
|
|
* @var array
|
|
*/
|
|
protected $defaultHTTPMethods = [
|
|
'options',
|
|
'get',
|
|
'head',
|
|
'post',
|
|
'put',
|
|
'delete',
|
|
'trace',
|
|
'connect',
|
|
'cli',
|
|
];
|
|
|
|
/**
|
|
* The name of the current group, if any.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $group = null;
|
|
|
|
/**
|
|
* The current subdomain.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $currentSubdomain = null;
|
|
|
|
/**
|
|
* Stores copy of current options being
|
|
* applied during creation.
|
|
*
|
|
* @var null
|
|
*/
|
|
protected $currentOptions = null;
|
|
|
|
/**
|
|
* A little performance booster.
|
|
*
|
|
* @var boolean
|
|
*/
|
|
protected $didDiscover = false;
|
|
|
|
/**
|
|
* Handle to the file locator to use.
|
|
*
|
|
* @var \CodeIgniter\Autoloader\FileLocator
|
|
*/
|
|
protected $fileLocator;
|
|
|
|
/**
|
|
* Handle to the modules config.
|
|
*
|
|
* @var \Config\Modules
|
|
*/
|
|
protected $moduleConfig;
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Constructor
|
|
*
|
|
* @param FileLocator $locator
|
|
* @param \Config\Modules $moduleConfig
|
|
*/
|
|
public function __construct(FileLocator $locator, $moduleConfig)
|
|
{
|
|
$this->fileLocator = $locator;
|
|
$this->moduleConfig = $moduleConfig;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Registers a new constraint with the system. Constraints are used
|
|
* by the routes as placeholders for regular expressions to make defining
|
|
* the routes more human-friendly.
|
|
*
|
|
* You can pass an associative array as $placeholder, and have
|
|
* multiple placeholders added at once.
|
|
*
|
|
* @param string|array $placeholder
|
|
* @param string $pattern
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function addPlaceholder($placeholder, string $pattern = null): RouteCollectionInterface
|
|
{
|
|
if (! is_array($placeholder))
|
|
{
|
|
$placeholder = [$placeholder => $pattern];
|
|
}
|
|
|
|
$this->placeholders = array_merge($this->placeholders, $placeholder);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the default namespace to use for Controllers when no other
|
|
* namespace has been specified.
|
|
*
|
|
* @param $value
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function setDefaultNamespace(string $value): RouteCollectionInterface
|
|
{
|
|
$this->defaultNamespace = filter_var($value, FILTER_SANITIZE_STRING);
|
|
$this->defaultNamespace = rtrim($this->defaultNamespace, '\\') . '\\';
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the default controller to use when no other controller has been
|
|
* specified.
|
|
*
|
|
* @param $value
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function setDefaultController(string $value): RouteCollectionInterface
|
|
{
|
|
$this->defaultController = filter_var($value, FILTER_SANITIZE_STRING);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the default method to call on the controller when no other
|
|
* method has been set in the route.
|
|
*
|
|
* @param $value
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function setDefaultMethod(string $value): RouteCollectionInterface
|
|
{
|
|
$this->defaultMethod = filter_var($value, FILTER_SANITIZE_STRING);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Tells the system whether to convert dashes in URI strings into
|
|
* underscores. In some search engines, including Google, dashes
|
|
* create more meaning and make it easier for the search engine to
|
|
* find words and meaning in the URI for better SEO. But it
|
|
* doesn't work well with PHP method names....
|
|
*
|
|
* @param boolean $value
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function setTranslateURIDashes(bool $value): RouteCollectionInterface
|
|
{
|
|
$this->translateURIDashes = $value;
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* If TRUE, the system will attempt to match the URI against
|
|
* Controllers by matching each segment against folders/files
|
|
* in APPPATH/Controllers, when a match wasn't found against
|
|
* defined routes.
|
|
*
|
|
* If FALSE, will stop searching and do NO automatic routing.
|
|
*
|
|
* @param boolean $value
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function setAutoRoute(bool $value): RouteCollectionInterface
|
|
{
|
|
$this->autoRoute = $value;
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the class/method that should be called if routing doesn't
|
|
* find a match. It can be either a closure or the controller/method
|
|
* name exactly like a route is defined: Users::index
|
|
*
|
|
* This setting is passed to the Router class and handled there.
|
|
*
|
|
* @param callable|null $callable
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function set404Override($callable = null): RouteCollectionInterface
|
|
{
|
|
$this->override404 = $callable;
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the 404 Override setting, which can be null, a closure
|
|
* or the controller/string.
|
|
*
|
|
* @return string|\Closure|null
|
|
*/
|
|
public function get404Override()
|
|
{
|
|
return $this->override404;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Will attempt to discover any additional routes, either through
|
|
* the local PSR4 namespaces, or through selected Composer packages.
|
|
*/
|
|
protected function discoverRoutes()
|
|
{
|
|
if ($this->didDiscover)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// We need this var in local scope
|
|
// so route files can access it.
|
|
$routes = $this;
|
|
|
|
if ($this->moduleConfig->shouldDiscover('routes'))
|
|
{
|
|
$files = $this->fileLocator->search('Config/Routes.php');
|
|
|
|
foreach ($files as $file)
|
|
{
|
|
// Don't include our main file again...
|
|
if ($file === APPPATH . 'Config/Routes.php')
|
|
{
|
|
continue;
|
|
}
|
|
|
|
include $file;
|
|
}
|
|
}
|
|
|
|
$this->didDiscover = true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the default constraint to be used in the system. Typically
|
|
* for use with the 'resource' method.
|
|
*
|
|
* @param string $placeholder
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function setDefaultConstraint(string $placeholder): RouteCollectionInterface
|
|
{
|
|
if (array_key_exists($placeholder, $this->placeholders))
|
|
{
|
|
$this->defaultPlaceholder = $placeholder;
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the name of the default controller. With Namespace.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getDefaultController(): string
|
|
{
|
|
return $this->defaultController;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the name of the default method to use within the controller.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getDefaultMethod(): string
|
|
{
|
|
return $this->defaultMethod;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the default namespace as set in the Routes config file.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getDefaultNamespace(): string
|
|
{
|
|
return $this->defaultNamespace;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the current value of the translateURIDashes setting.
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function shouldTranslateURIDashes(): bool
|
|
{
|
|
return $this->translateURIDashes;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the flag that tells whether to autoRoute URI against Controllers.
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function shouldAutoRoute(): bool
|
|
{
|
|
return $this->autoRoute;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the raw array of available routes.
|
|
*
|
|
* @param mixed $verb
|
|
*
|
|
* @return array
|
|
*/
|
|
public function getRoutes($verb = null): array
|
|
{
|
|
if (empty($verb))
|
|
{
|
|
$verb = $this->getHTTPVerb();
|
|
}
|
|
|
|
// Since this is the entry point for the Router,
|
|
// take a moment to do any route discovery
|
|
// we might need to do.
|
|
$this->discoverRoutes();
|
|
|
|
$routes = [];
|
|
|
|
if (isset($this->routes[$verb]))
|
|
{
|
|
// Keep current verb's routes at the beginning so they're matched
|
|
// before any of the generic, "add" routes.
|
|
if (isset($this->routes['*']))
|
|
{
|
|
$extraRules = array_diff_key($this->routes['*'], $this->routes[$verb]);
|
|
$collection = array_merge($this->routes[$verb], $extraRules);
|
|
}
|
|
foreach ($collection as $r)
|
|
{
|
|
$key = key($r['route']);
|
|
$routes[$key] = $r['route'][$key];
|
|
}
|
|
}
|
|
|
|
return $routes;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns one or all routes options
|
|
*
|
|
* @param string $from
|
|
*
|
|
* @return array
|
|
*/
|
|
public function getRoutesOptions(string $from = null): array
|
|
{
|
|
return $from ? $this->routesOptions[$from] ?? [] : $this->routesOptions;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the current HTTP Verb being used.
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getHTTPVerb(): string
|
|
{
|
|
return $this->HTTPVerb;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Sets the current HTTP verb.
|
|
* Used primarily for testing.
|
|
*
|
|
* @param string $verb
|
|
*
|
|
* @return $this
|
|
*/
|
|
public function setHTTPVerb(string $verb)
|
|
{
|
|
$this->HTTPVerb = $verb;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* A shortcut method to add a number of routes at a single time.
|
|
* It does not allow any options to be set on the route, or to
|
|
* define the method used.
|
|
*
|
|
* @param array $routes
|
|
* @param array $options
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function map(array $routes = [], array $options = null): RouteCollectionInterface
|
|
{
|
|
foreach ($routes as $from => $to)
|
|
{
|
|
$this->add($from, $to, $options);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Adds a single route to the collection.
|
|
*
|
|
* Example:
|
|
* $routes->add('news', 'Posts::index');
|
|
*
|
|
* @param string $from
|
|
* @param array|string $to
|
|
* @param array $options
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function add(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('*', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Adds a temporary redirect from one route to another. Used for
|
|
* redirecting traffic from old, non-existing routes to the new
|
|
* moved routes.
|
|
*
|
|
* @param string $from The pattern to match against
|
|
* @param string $to Either a route name or a URI to redirect to
|
|
* @param integer $status The HTTP status code that should be returned with this redirect
|
|
*
|
|
* @return RouteCollection
|
|
*/
|
|
public function addRedirect(string $from, string $to, int $status = 302)
|
|
{
|
|
// Use the named route's pattern if this is a named route.
|
|
if (array_key_exists($to, $this->routes['*']))
|
|
{
|
|
$to = $this->routes['*'][$to]['route'];
|
|
}
|
|
|
|
$this->create('*', $from, $to, ['redirect' => $status]);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Determines if the route is a redirecting route.
|
|
*
|
|
* @param string $from
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function isRedirect(string $from): bool
|
|
{
|
|
foreach ($this->routes['*'] as $name => $route)
|
|
{
|
|
// Named route?
|
|
if ($name === $from || key($route['route']) === $from)
|
|
{
|
|
return isset($route['redirect']) && is_numeric($route['redirect']);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Grabs the HTTP status code from a redirecting Route.
|
|
*
|
|
* @param string $from
|
|
*
|
|
* @return integer
|
|
*/
|
|
public function getRedirectCode(string $from): int
|
|
{
|
|
foreach ($this->routes['*'] as $name => $route)
|
|
{
|
|
// Named route?
|
|
if ($name === $from || key($route['route']) === $from)
|
|
{
|
|
return $route['redirect'] ?? 0;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
//--------------------------------------------------------------------
|
|
// Grouping Routes
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Group a series of routes under a single URL segment. This is handy
|
|
* for grouping items into an admin area, like:
|
|
*
|
|
* Example:
|
|
* // Creates route: admin/users
|
|
* $route->group('admin', function() {
|
|
* $route->resource('users');
|
|
* });
|
|
*
|
|
* @param string $name The name to group/prefix the routes with.
|
|
* @param $params
|
|
*
|
|
* @return void
|
|
*/
|
|
public function group(string $name, ...$params)
|
|
{
|
|
$oldGroup = $this->group;
|
|
$oldOptions = $this->currentOptions;
|
|
|
|
// To register a route, we'll set a flag so that our router
|
|
// so it will see the group name.
|
|
$this->group = ltrim($oldGroup . '/' . $name, '/');
|
|
|
|
$callback = array_pop($params);
|
|
|
|
if ($params && is_array($params[0]))
|
|
{
|
|
$this->currentOptions = array_shift($params);
|
|
}
|
|
|
|
if (is_callable($callback))
|
|
{
|
|
$callback($this);
|
|
}
|
|
|
|
$this->group = $oldGroup;
|
|
$this->currentOptions = $oldOptions;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
//--------------------------------------------------------------------
|
|
// HTTP Verb-based routing
|
|
//--------------------------------------------------------------------
|
|
// Routing works here because, as the routes Config file is read in,
|
|
// the various HTTP verb-based routes will only be added to the in-memory
|
|
// routes if it is a call that should respond to that verb.
|
|
//
|
|
// The options array is typically used to pass in an 'as' or var, but may
|
|
// be expanded in the future. See the docblock for 'add' method above for
|
|
// current list of globally available options.
|
|
//
|
|
|
|
/**
|
|
* Creates a collections of HTTP-verb based routes for a controller.
|
|
*
|
|
* Possible Options:
|
|
* 'controller' - Customize the name of the controller used in the 'to' route
|
|
* 'placeholder' - The regex used by the Router. Defaults to '(:any)'
|
|
* 'websafe' - - '1' if only GET and POST HTTP verbs are supported
|
|
*
|
|
* Example:
|
|
*
|
|
* $route->resource('photos');
|
|
*
|
|
* // Generates the following routes:
|
|
* HTTP Verb | Path | Action | Used for...
|
|
* ----------+-------------+---------------+-----------------
|
|
* GET /photos index an array of photo objects
|
|
* GET /photos/new new an empty photo object, with default properties
|
|
* GET /photos/{id}/edit edit a specific photo object, editable properties
|
|
* GET /photos/{id} show a specific photo object, all properties
|
|
* POST /photos create a new photo object, to add to the resource
|
|
* DELETE /photos/{id} delete deletes the specified photo object
|
|
* PUT/PATCH /photos/{id} update replacement properties for existing photo
|
|
*
|
|
* If 'websafe' option is present, the following paths are also available:
|
|
*
|
|
* POST /photos/{id}/delete delete
|
|
* POST /photos/{id} update
|
|
*
|
|
* @param string $name The name of the resource/controller to route to.
|
|
* @param array $options An list of possible ways to customize the routing.
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function resource(string $name, array $options = null): RouteCollectionInterface
|
|
{
|
|
// In order to allow customization of the route the
|
|
// resources are sent to, we need to have a new name
|
|
// to store the values in.
|
|
$new_name = ucfirst($name);
|
|
|
|
// If a new controller is specified, then we replace the
|
|
// $name value with the name of the new controller.
|
|
if (isset($options['controller']))
|
|
{
|
|
$new_name = ucfirst(filter_var($options['controller'], FILTER_SANITIZE_STRING));
|
|
}
|
|
|
|
// In order to allow customization of allowed id values
|
|
// we need someplace to store them.
|
|
$id = $this->placeholders[$this->defaultPlaceholder] ?? '(:segment)';
|
|
|
|
if (isset($options['placeholder']))
|
|
{
|
|
$id = $options['placeholder'];
|
|
}
|
|
|
|
// Make sure we capture back-references
|
|
$id = '(' . trim($id, '()') . ')';
|
|
|
|
$methods = isset($options['only']) ? is_string($options['only']) ? explode(',', $options['only']) : $options['only'] : ['index', 'show', 'create', 'update', 'delete', 'new', 'edit'];
|
|
|
|
if (isset($options['except']))
|
|
{
|
|
$options['except'] = is_array($options['except']) ? $options['except'] : explode(',', $options['except']);
|
|
$c = count($methods);
|
|
for ($i = 0; $i < $c; $i ++)
|
|
{
|
|
if (in_array($methods[$i], $options['except']))
|
|
{
|
|
unset($methods[$i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (in_array('index', $methods))
|
|
{
|
|
$this->get($name, $new_name . '::index', $options);
|
|
}
|
|
if (in_array('new', $methods))
|
|
{
|
|
$this->get($name . '/new', $new_name . '::new', $options);
|
|
}
|
|
if (in_array('edit', $methods))
|
|
{
|
|
$this->get($name . '/' . $id . '/edit', $new_name . '::edit/$1', $options);
|
|
}
|
|
if (in_array('show', $methods))
|
|
{
|
|
$this->get($name . '/' . $id, $new_name . '::show/$1', $options);
|
|
}
|
|
if (in_array('create', $methods))
|
|
{
|
|
$this->post($name, $new_name . '::create', $options);
|
|
}
|
|
if (in_array('update', $methods))
|
|
{
|
|
$this->put($name . '/' . $id, $new_name . '::update/$1', $options);
|
|
$this->patch($name . '/' . $id, $new_name . '::update/$1', $options);
|
|
}
|
|
if (in_array('delete', $methods))
|
|
{
|
|
$this->delete($name . '/' . $id, $new_name . '::delete/$1', $options);
|
|
}
|
|
|
|
// Web Safe? delete needs checking before update because of method name
|
|
if (isset($options['websafe']))
|
|
{
|
|
if (in_array('delete', $methods))
|
|
{
|
|
$this->post($name . '/' . $id . '/delete', $new_name . '::delete/$1', $options);
|
|
}
|
|
if (in_array('update', $methods))
|
|
{
|
|
$this->post($name . '/' . $id, $new_name . '::update/$1', $options);
|
|
}
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Creates a collections of HTTP-verb based routes for a presenter controller.
|
|
*
|
|
* Possible Options:
|
|
* 'controller' - Customize the name of the controller used in the 'to' route
|
|
* 'placeholder' - The regex used by the Router. Defaults to '(:any)'
|
|
*
|
|
* Example:
|
|
*
|
|
* $route->presenter('photos');
|
|
*
|
|
* // Generates the following routes:
|
|
* HTTP Verb | Path | Action | Used for...
|
|
* ----------+-------------+---------------+-----------------
|
|
* GET /photos index showing all array of photo objects
|
|
* GET /photos/show/{id} show showing a specific photo object, all properties
|
|
* GET /photos/new new showing a form for an empty photo object, with default properties
|
|
* POST /photos/create create processing the form for a new photo
|
|
* GET /photos/edit/{id} edit show an editing form for a specific photo object, editable properties
|
|
* POST /photos/update/{id} update process the editing form data
|
|
* GET /photos/remove/{id} remove show a form to confirm deletion of a specific photo object
|
|
* POST /photos/delete/{id} delete deleting the specified photo object
|
|
*
|
|
* @param string $name The name of the controller to route to.
|
|
* @param array $options An list of possible ways to customize the routing.
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function presenter(string $name, array $options = null): RouteCollectionInterface
|
|
{
|
|
// In order to allow customization of the route the
|
|
// resources are sent to, we need to have a new name
|
|
// to store the values in.
|
|
$newName = ucfirst($name);
|
|
|
|
// If a new controller is specified, then we replace the
|
|
// $name value with the name of the new controller.
|
|
if (isset($options['controller']))
|
|
{
|
|
$newName = ucfirst(filter_var($options['controller'], FILTER_SANITIZE_STRING));
|
|
}
|
|
|
|
// In order to allow customization of allowed id values
|
|
// we need someplace to store them.
|
|
$id = $this->placeholders[$this->defaultPlaceholder] ?? '(:segment)';
|
|
|
|
if (isset($options['placeholder']))
|
|
{
|
|
$id = $options['placeholder'];
|
|
}
|
|
|
|
// Make sure we capture back-references
|
|
$id = '(' . trim($id, '()') . ')';
|
|
|
|
$methods = isset($options['only']) ? is_string($options['only']) ? explode(',', $options['only']) : $options['only'] : ['index', 'show', 'new', 'create', 'edit', 'update', 'remove', 'delete'];
|
|
|
|
if (isset($options['except']))
|
|
{
|
|
$options['except'] = is_array($options['except']) ? $options['except'] : explode(',', $options['except']);
|
|
$c = count($methods);
|
|
for ($i = 0; $i < $c; $i ++)
|
|
{
|
|
if (in_array($methods[$i], $options['except']))
|
|
{
|
|
unset($methods[$i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (in_array('index', $methods))
|
|
{
|
|
$this->get($name, $newName . '::index', $options);
|
|
}
|
|
if (in_array('show', $methods))
|
|
{
|
|
$this->get($name . '/show/' . $id, $newName . '::show/$1', $options);
|
|
}
|
|
if (in_array('new', $methods))
|
|
{
|
|
$this->get($name . '/new', $newName . '::new', $options);
|
|
}
|
|
if (in_array('create', $methods))
|
|
{
|
|
$this->post($name . '/create', $newName . '::create', $options);
|
|
}
|
|
if (in_array('edit', $methods))
|
|
{
|
|
$this->get($name . '/edit/' . $id, $newName . '::edit/$1', $options);
|
|
}
|
|
if (in_array('update', $methods))
|
|
{
|
|
$this->post($name . '/update/' . $id, $newName . '::update/$1', $options);
|
|
}
|
|
if (in_array('remove', $methods))
|
|
{
|
|
$this->get($name . '/remove/' . $id, $newName . '::remove/$1', $options);
|
|
}
|
|
if (in_array('delete', $methods))
|
|
{
|
|
$this->post($name . '/delete/' . $id, $newName . '::delete/$1', $options);
|
|
}
|
|
if (in_array('show', $methods))
|
|
{
|
|
$this->get($name . '/' . $id, $newName . '::show/$1', $options);
|
|
}
|
|
if (in_array('create', $methods))
|
|
{
|
|
$this->post($name, $newName . '::create', $options);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a single route to match for multiple HTTP Verbs.
|
|
*
|
|
* Example:
|
|
* $route->match( ['get', 'post'], 'users/(:num)', 'users/$1);
|
|
*
|
|
* @param array $verbs
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function match(array $verbs = [], string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
foreach ($verbs as $verb)
|
|
{
|
|
$verb = strtolower($verb);
|
|
|
|
$this->{$verb}($from, $to, $options);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to GET requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function get(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('get', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to POST requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function post(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('post', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to PUT requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function put(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('put', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to DELETE requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function delete(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('delete', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to HEAD requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function head(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('head', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to PATCH requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function patch(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('patch', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to OPTIONS requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function options(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('options', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Specifies a route that is only available to command-line requests.
|
|
*
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*
|
|
* @return \CodeIgniter\Router\RouteCollectionInterface
|
|
*/
|
|
public function cli(string $from, $to, array $options = null): RouteCollectionInterface
|
|
{
|
|
$this->create('cli', $from, $to, $options);
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Limits the routes to a specified ENVIRONMENT or they won't run.
|
|
*
|
|
* @param string $env
|
|
* @param \Closure $callback
|
|
*
|
|
* @return RouteCollectionInterface
|
|
*/
|
|
public function environment(string $env, \Closure $callback): RouteCollectionInterface
|
|
{
|
|
if (ENVIRONMENT === $env)
|
|
{
|
|
$callback($this);
|
|
}
|
|
|
|
return $this;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Attempts to look up a route based on it's destination.
|
|
*
|
|
* If a route exists:
|
|
*
|
|
* 'path/(:any)/(:any)' => 'Controller::method/$1/$2'
|
|
*
|
|
* This method allows you to know the Controller and method
|
|
* and get the route that leads to it.
|
|
*
|
|
* // Equals 'path/$param1/$param2'
|
|
* reverseRoute('Controller::method', $param1, $param2);
|
|
*
|
|
* @param string $search
|
|
* @param array ...$params
|
|
*
|
|
* @return string|false
|
|
*/
|
|
public function reverseRoute(string $search, ...$params)
|
|
{
|
|
// Named routes get higher priority.
|
|
foreach ($this->routes as $verb => $collection)
|
|
{
|
|
if (array_key_exists($search, $collection))
|
|
{
|
|
return $this->fillRouteParams(key($collection[$search]['route']), $params);
|
|
}
|
|
}
|
|
|
|
// If it's not a named route, then loop over
|
|
// all routes to find a match.
|
|
foreach ($this->routes as $verb => $collection)
|
|
{
|
|
foreach ($collection as $route)
|
|
{
|
|
$from = key($route['route']);
|
|
$to = $route['route'][$from];
|
|
|
|
// ignore closures
|
|
if (! is_string($to))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Lose any namespace slash at beginning of strings
|
|
// to ensure more consistent match.
|
|
$to = ltrim($to, '\\');
|
|
$search = ltrim($search, '\\');
|
|
|
|
// If there's any chance of a match, then it will
|
|
// be with $search at the beginning of the $to string.
|
|
if (strpos($to, $search) !== 0)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Ensure that the number of $params given here
|
|
// matches the number of back-references in the route
|
|
if (substr_count($to, '$') !== count($params))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
return $this->fillRouteParams($from, $params);
|
|
}
|
|
}
|
|
|
|
// If we're still here, then we did not find a match.
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Checks a route (using the "from") to see if it's filtered or not.
|
|
*
|
|
* @param string $search
|
|
*
|
|
* @return boolean
|
|
*/
|
|
public function isFiltered(string $search): bool
|
|
{
|
|
return isset($this->routesOptions[$search]['filter']);
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Returns the filter that should be applied for a single route, along
|
|
* with any parameters it might have. Parameters are found by splitting
|
|
* the parameter name on a colon to separate the filter name from the parameter list,
|
|
* and the splitting the result on commas. So:
|
|
*
|
|
* 'role:admin,manager'
|
|
*
|
|
* has a filter of "role", with parameters of ['admin', 'manager'].
|
|
*
|
|
* @param string $search
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getFilterForRoute(string $search): string
|
|
{
|
|
if (! $this->isFiltered($search))
|
|
{
|
|
return '';
|
|
}
|
|
|
|
return $this->routesOptions[$search]['filter'];
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Given a
|
|
*
|
|
* @param string $from
|
|
* @param array|null $params
|
|
*
|
|
* @return string
|
|
*/
|
|
protected function fillRouteParams(string $from, array $params = null): string
|
|
{
|
|
// Find all of our back-references in the original route
|
|
preg_match_all('/\(([^)]+)\)/', $from, $matches);
|
|
|
|
if (empty($matches[0]))
|
|
{
|
|
return '/' . ltrim($from, '/');
|
|
}
|
|
|
|
// Build our resulting string, inserting the $params in
|
|
// the appropriate places.
|
|
foreach ($matches[0] as $index => $pattern)
|
|
{
|
|
// Ensure that the param we're inserting matches
|
|
// the expected param type.
|
|
$pos = strpos($from, $pattern);
|
|
|
|
if (preg_match("|{$pattern}|", $params[$index]))
|
|
{
|
|
$from = substr_replace($from, $params[$index], $pos, strlen($pattern));
|
|
}
|
|
else
|
|
{
|
|
throw RouterException::forInvalidParameterType();
|
|
}
|
|
}
|
|
|
|
return '/' . ltrim($from, '/');
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Does the heavy lifting of creating an actual route. You must specify
|
|
* the request method(s) that this route will work for. They can be separated
|
|
* by a pipe character "|" if there is more than one.
|
|
*
|
|
* @param string $verb
|
|
* @param string $from
|
|
* @param string|array $to
|
|
* @param array $options
|
|
*/
|
|
protected function create(string $verb, string $from, $to, array $options = null)
|
|
{
|
|
$overwrite = false;
|
|
$prefix = is_null($this->group) ? '' : $this->group . '/';
|
|
|
|
$from = filter_var($prefix . $from, FILTER_SANITIZE_STRING);
|
|
|
|
// While we want to add a route within a group of '/',
|
|
// it doesn't work with matching, so remove them...
|
|
if ($from !== '/')
|
|
{
|
|
$from = trim($from, '/');
|
|
}
|
|
|
|
$options = array_merge((array) $this->currentOptions, (array) $options);
|
|
|
|
// Hostname limiting?
|
|
if (! empty($options['hostname']))
|
|
{
|
|
// @todo determine if there's a way to whitelist hosts?
|
|
if (isset($_SERVER['HTTP_HOST']) && strtolower($_SERVER['HTTP_HOST']) !== strtolower($options['hostname']))
|
|
{
|
|
return;
|
|
}
|
|
|
|
$overwrite = true;
|
|
}
|
|
|
|
// Limiting to subdomains?
|
|
else if (isset($options['subdomain']) && ! empty($options['subdomain']))
|
|
{
|
|
// If we don't match the current subdomain, then
|
|
// we don't need to add the route.
|
|
if (! $this->checkSubdomains($options['subdomain']))
|
|
{
|
|
return;
|
|
}
|
|
|
|
$overwrite = true;
|
|
}
|
|
|
|
// Are we offsetting the binds?
|
|
// If so, take care of them here in one
|
|
// fell swoop.
|
|
if (isset($options['offset']) && is_string($to))
|
|
{
|
|
// Get a constant string to work with.
|
|
$to = preg_replace('/(\$\d+)/', '$X', $to);
|
|
|
|
for ($i = (int) $options['offset'] + 1; $i < (int) $options['offset'] + 7; $i ++)
|
|
{
|
|
$to = preg_replace_callback(
|
|
'/\$X/', function ($m) use ($i) {
|
|
return '$' . $i;
|
|
}, $to, 1
|
|
);
|
|
}
|
|
}
|
|
|
|
// Replace our regex pattern placeholders with the actual thing
|
|
// so that the Router doesn't need to know about any of this.
|
|
foreach ($this->placeholders as $tag => $pattern)
|
|
{
|
|
$from = str_ireplace(':' . $tag, $pattern, $from);
|
|
}
|
|
|
|
// If no namespace found, add the default namespace
|
|
if (is_string($to) && strpos($to, '\\') === false)
|
|
{
|
|
$namespace = $options['namespace'] ?? $this->defaultNamespace;
|
|
$to = trim($namespace, '\\') . '\\' . $to;
|
|
}
|
|
|
|
// Always ensure that we escape our namespace so we're not pointing to
|
|
// \CodeIgniter\Routes\Controller::method.
|
|
if (is_string($to))
|
|
{
|
|
$to = '\\' . ltrim($to, '\\');
|
|
}
|
|
|
|
$name = $options['as'] ?? $from;
|
|
|
|
// Don't overwrite any existing 'froms' so that auto-discovered routes
|
|
// do not overwrite any app/Config/Routes settings. The app
|
|
// routes should always be the "source of truth".
|
|
// this works only because discovered routes are added just prior
|
|
// to attempting to route the request.
|
|
if (isset($this->routes[$verb][$name]) && ! $overwrite)
|
|
{
|
|
return;
|
|
}
|
|
|
|
$this->routes[$verb][$name] = [
|
|
'route' => [$from => $to],
|
|
];
|
|
|
|
$this->routesOptions[$from] = $options;
|
|
|
|
// Is this a redirect?
|
|
if (isset($options['redirect']) && is_numeric($options['redirect']))
|
|
{
|
|
$this->routes['*'][$name]['redirect'] = $options['redirect'];
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Compares the subdomain(s) passed in against the current subdomain
|
|
* on this page request.
|
|
*
|
|
* @param mixed $subdomains
|
|
*
|
|
* @return boolean
|
|
*/
|
|
private function checkSubdomains($subdomains): bool
|
|
{
|
|
// CLI calls can't be on subdomain.
|
|
if (! isset($_SERVER['HTTP_HOST']))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (is_null($this->currentSubdomain))
|
|
{
|
|
$this->currentSubdomain = $this->determineCurrentSubdomain();
|
|
}
|
|
|
|
if (! is_array($subdomains))
|
|
{
|
|
$subdomains = [$subdomains];
|
|
}
|
|
|
|
// Routes can be limited to any sub-domain. In that case, though,
|
|
// it does require a sub-domain to be present.
|
|
if (! empty($this->currentSubdomain) && in_array('*', $subdomains))
|
|
{
|
|
return true;
|
|
}
|
|
|
|
foreach ($subdomains as $subdomain)
|
|
{
|
|
if ($subdomain === $this->currentSubdomain)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Examines the HTTP_HOST to get a best match for the subdomain. It
|
|
* won't be perfect, but should work for our needs.
|
|
*
|
|
* It's especially not perfect since it's possible to register a domain
|
|
* with a period (.) as part of the domain name.
|
|
*/
|
|
private function determineCurrentSubdomain()
|
|
{
|
|
// We have to ensure that a scheme exists
|
|
// on the URL else parse_url will mis-interpret
|
|
// 'host' as the 'path'.
|
|
$url = $_SERVER['HTTP_HOST'];
|
|
if (strpos($url, 'http') !== 0)
|
|
{
|
|
$url = 'http://' . $url;
|
|
}
|
|
|
|
$parsedUrl = parse_url($url);
|
|
|
|
$host = explode('.', $parsedUrl['host']);
|
|
|
|
if ($host[0] === 'www')
|
|
{
|
|
unset($host[0]);
|
|
}
|
|
|
|
// Get rid of any domains, which will be the last
|
|
unset($host[count($host)]);
|
|
|
|
// Account for .co.uk, .co.nz, etc. domains
|
|
if (end($host) === 'co')
|
|
{
|
|
$host = array_slice($host, 0, -1);
|
|
}
|
|
|
|
// If we only have 1 part left, then we don't have a sub-domain.
|
|
if (count($host) === 1)
|
|
{
|
|
// Set it to false so we don't make it back here again.
|
|
return false;
|
|
}
|
|
|
|
return array_shift($host);
|
|
}
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
/**
|
|
* Reset the routes, so that a FeatureTestCase can provide the
|
|
* explicit ones needed for it.
|
|
*/
|
|
public function resetRoutes()
|
|
{
|
|
$this->routes = ['*' => []];
|
|
foreach ($this->defaultHTTPMethods as $verb)
|
|
{
|
|
$this->routes[$verb] = [];
|
|
}
|
|
}
|
|
|
|
}
|
|
|