Merge pull request #439 from masroore/master

PHP 7.4/8.0/8.1 compatibility
pull/457/head
Mike Cao 3 years ago committed by GitHub
commit 0ae1f9f484
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -17,7 +17,7 @@ Flight::start();
# Requirements # Requirements
Flight requires `PHP 5.3` or greater. Flight requires `PHP 7.4` or greater.
# License # License

@ -12,12 +12,13 @@
} }
], ],
"require": { "require": {
"php": ">=5.3.0" "php": "^7.4|^8.0|^8.1",
"ext-json": "*"
}, },
"autoload": { "autoload": {
"files": [ "flight/autoload.php", "flight/Flight.php" ] "files": [ "flight/autoload.php", "flight/Flight.php" ]
}, },
"require-dev": { "require-dev": {
"phpunit/phpunit": "~4.6" "phpunit/phpunit": "^9.5"
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -8,8 +10,15 @@
namespace flight; namespace flight;
use flight\core\Loader; use ErrorException;
use Exception;
use flight\core\Dispatcher; use flight\core\Dispatcher;
use flight\core\Loader;
use flight\net\Request;
use flight\net\Response;
use flight\net\Router;
use flight\template\View;
use Throwable;
/** /**
* The Engine class contains the core functionality of the framework. * The Engine class contains the core functionality of the framework.
@ -17,23 +26,21 @@ use flight\core\Dispatcher;
* and generating an HTTP response. * and generating an HTTP response.
* *
* Core methods * Core methods
*
* @method void start() Starts engine * @method void start() Starts engine
* @method void stop() Stops framework and outputs current response * @method void stop() Stops framework and outputs current response
* @method void halt(int $code = 200, string $message = '') Stops processing and returns a given response. * @method void halt(int $code = 200, string $message = '') Stops processing and returns a given response.
*
*
* Routing
* @method void route(string $pattern, callable $callback, bool $pass_route = false) Routes a URL to a callback function. * @method void route(string $pattern, callable $callback, bool $pass_route = false) Routes a URL to a callback function.
* @method \flight\net\Router router() Gets router * @method Router router() Gets router
* *
* Views * Views
* @method void render(string $file, array $data = null, string $key = null) Renders template * @method void render(string $file, array $data = null, string $key = null) Renders template
* @method \flight\template\View view() Gets current view * @method View view() Gets current view
* *
* Request-response * Request-response
* @method \flight\net\Request request() Gets current request * @method Request request() Gets current request
* @method \flight\net\Response response() Gets current response * @method Response response() Gets current response
* @method void error(\Exception $e) Sends an HTTP 500 response for any errors. * @method void error(Exception $e) Sends an HTTP 500 response for any errors.
* @method void notFound() Sends an HTTP 404 response when a URL is not found. * @method void notFound() Sends an HTTP 404 response when a URL is not found.
* @method void redirect(string $url, int $code = 303) Redirects the current request to another URL. * @method void redirect(string $url, int $code = 303) Redirects the current request to another URL.
* @method void json(mixed $data, int $code = 200, bool $encode = true, string $charset = 'utf-8', int $option = 0) Sends a JSON response. * @method void json(mixed $data, int $code = 200, bool $encode = true, string $charset = 'utf-8', int $option = 0) Sends a JSON response.
@ -43,33 +50,29 @@ use flight\core\Dispatcher;
* @method void etag($id, string $type = 'strong') Handles ETag HTTP caching. * @method void etag($id, string $type = 'strong') Handles ETag HTTP caching.
* @method void lastModified(int $time) Handles last modified HTTP caching. * @method void lastModified(int $time) Handles last modified HTTP caching.
*/ */
class Engine { class Engine
{
/** /**
* Stored variables. * Stored variables.
*
* @var array
*/ */
protected $vars; protected array $vars;
/** /**
* Class loader. * Class loader.
*
* @var Loader
*/ */
protected $loader; protected Loader $loader;
/** /**
* Event dispatcher. * Event dispatcher.
*
* @var Dispatcher
*/ */
protected $dispatcher; protected Dispatcher $dispatcher;
/** /**
* Constructor. * Constructor.
*/ */
public function __construct() { public function __construct()
$this->vars = array(); {
$this->vars = [];
$this->loader = new Loader(); $this->loader = new Loader();
$this->dispatcher = new Dispatcher(); $this->dispatcher = new Dispatcher();
@ -82,60 +85,65 @@ class Engine {
* *
* @param string $name Method name * @param string $name Method name
* @param array $params Method parameters * @param array $params Method parameters
*
* @throws Exception
*
* @return mixed Callback results * @return mixed Callback results
* @throws \Exception
*/ */
public function __call($name, $params) { public function __call(string $name, array $params)
{
$callback = $this->dispatcher->get($name); $callback = $this->dispatcher->get($name);
if (is_callable($callback)) { if (\is_callable($callback)) {
return $this->dispatcher->run($name, $params); return $this->dispatcher->run($name, $params);
} }
if (!$this->loader->get($name)) { if (!$this->loader->get($name)) {
throw new \Exception("{$name} must be a mapped method."); throw new Exception("{$name} must be a mapped method.");
} }
$shared = (!empty($params)) ? (bool)$params[0] : true; $shared = empty($params) || $params[0];
return $this->loader->load($name, $shared); return $this->loader->load($name, $shared);
} }
/*** Core Methods ***/ // Core Methods
/** /**
* Initializes the framework. * Initializes the framework.
*/ */
public function init() { public function init(): void
{
static $initialized = false; static $initialized = false;
$self = $this; $self = $this;
if ($initialized) { if ($initialized) {
$this->vars = array(); $this->vars = [];
$this->loader->reset(); $this->loader->reset();
$this->dispatcher->reset(); $this->dispatcher->reset();
} }
// Register default components // Register default components
$this->loader->register('request', '\flight\net\Request'); $this->loader->register('request', Request::class);
$this->loader->register('response', '\flight\net\Response'); $this->loader->register('response', Response::class);
$this->loader->register('router', '\flight\net\Router'); $this->loader->register('router', Router::class);
$this->loader->register('view', '\flight\template\View', array(), function($view) use ($self) { $this->loader->register('view', View::class, [], function ($view) use ($self) {
$view->path = $self->get('flight.views.path'); $view->path = $self->get('flight.views.path');
$view->extension = $self->get('flight.views.extension'); $view->extension = $self->get('flight.views.extension');
}); });
// Register framework methods // Register framework methods
$methods = array( $methods = [
'start', 'stop', 'route', 'halt', 'error', 'notFound', 'start', 'stop', 'route', 'halt', 'error', 'notFound',
'render','redirect','etag','lastModified','json','jsonp' 'render', 'redirect', 'etag', 'lastModified', 'json', 'jsonp',
); 'post', 'put', 'patch', 'delete',
];
foreach ($methods as $name) { foreach ($methods as $name) {
$this->dispatcher->set($name, array($this, '_'.$name)); $this->dispatcher->set($name, [$this, '_' . $name]);
} }
// Default configuration settings // Default configuration settings
$this->set('flight.base_url', null); $this->set('flight.base_url');
$this->set('flight.case_sensitive', false); $this->set('flight.case_sensitive', false);
$this->set('flight.handle_errors', true); $this->set('flight.handle_errors', true);
$this->set('flight.log_errors', false); $this->set('flight.log_errors', false);
@ -147,8 +155,8 @@ class Engine {
$this->before('start', function () use ($self) { $this->before('start', function () use ($self) {
// Enable error handling // Enable error handling
if ($self->get('flight.handle_errors')) { if ($self->get('flight.handle_errors')) {
set_error_handler(array($self, 'handleError')); set_error_handler([$self, 'handleError']);
set_exception_handler(array($self, 'handleException')); set_exception_handler([$self, 'handleException']);
} }
// Set case-sensitivity // Set case-sensitivity
@ -164,23 +172,26 @@ class Engine {
* Custom error handler. Converts errors into exceptions. * Custom error handler. Converts errors into exceptions.
* *
* @param int $errno Error number * @param int $errno Error number
* @param int $errstr Error string * @param string $errstr Error string
* @param int $errfile Error file name * @param string $errfile Error file name
* @param int $errline Error file line number * @param int $errline Error file line number
* @throws \ErrorException *
* @throws ErrorException
*/ */
public function handleError($errno, $errstr, $errfile, $errline) { public function handleError(int $errno, string $errstr, string $errfile, int $errline)
{
if ($errno & error_reporting()) { if ($errno & error_reporting()) {
throw new \ErrorException($errstr, $errno, 0, $errfile, $errline); throw new ErrorException($errstr, $errno, 0, $errfile, $errline);
} }
} }
/** /**
* Custom exception handler. Logs exceptions. * Custom exception handler. Logs exceptions.
* *
* @param \Exception $e Thrown exception * @param Exception $e Thrown exception
*/ */
public function handleException($e) { public function handleException($e): void
{
if ($this->get('flight.log_errors')) { if ($this->get('flight.log_errors')) {
error_log($e->getMessage()); error_log($e->getMessage());
} }
@ -193,11 +204,13 @@ class Engine {
* *
* @param string $name Method name * @param string $name Method name
* @param callback $callback Callback function * @param callback $callback Callback function
* @throws \Exception If trying to map over a framework method *
* @throws Exception If trying to map over a framework method
*/ */
public function map($name, $callback) { public function map(string $name, callable $callback): void
{
if (method_exists($this, $name)) { if (method_exists($this, $name)) {
throw new \Exception('Cannot override an existing framework method.'); throw new Exception('Cannot override an existing framework method.');
} }
$this->dispatcher->set($name, $callback); $this->dispatcher->set($name, $callback);
@ -209,12 +222,14 @@ class Engine {
* @param string $name Method name * @param string $name Method name
* @param string $class Class name * @param string $class Class name
* @param array $params Class initialization parameters * @param array $params Class initialization parameters
* @param callback $callback Function to call after object instantiation * @param callable|null $callback $callback Function to call after object instantiation
* @throws \Exception If trying to map over a framework method *
* @throws Exception If trying to map over a framework method
*/ */
public function register($name, $class, array $params = array(), $callback = null) { public function register(string $name, string $class, array $params = [], ?callable $callback = null): void
{
if (method_exists($this, $name)) { if (method_exists($this, $name)) {
throw new \Exception('Cannot override an existing framework method.'); throw new Exception('Cannot override an existing framework method.');
} }
$this->loader->register($name, $class, $params, $callback); $this->loader->register($name, $class, $params, $callback);
@ -226,7 +241,8 @@ class Engine {
* @param string $name Method name * @param string $name Method name
* @param callback $callback Callback function * @param callback $callback Callback function
*/ */
public function before($name, $callback) { public function before(string $name, callable $callback): void
{
$this->dispatcher->hook($name, 'before', $callback); $this->dispatcher->hook($name, 'before', $callback);
} }
@ -236,35 +252,40 @@ class Engine {
* @param string $name Method name * @param string $name Method name
* @param callback $callback Callback function * @param callback $callback Callback function
*/ */
public function after($name, $callback) { public function after(string $name, callable $callback): void
{
$this->dispatcher->hook($name, 'after', $callback); $this->dispatcher->hook($name, 'after', $callback);
} }
/** /**
* Gets a variable. * Gets a variable.
* *
* @param string $key Key * @param string|null $key Key
* @return mixed *
* @return array|mixed|null
*/ */
public function get($key = null) { public function get(?string $key = null)
if ($key === null) return $this->vars; {
if (null === $key) {
return $this->vars;
}
return isset($this->vars[$key]) ? $this->vars[$key] : null; return $this->vars[$key] ?? null;
} }
/** /**
* Sets a variable. * Sets a variable.
* *
* @param mixed $key Key * @param mixed $key Key
* @param string $value Value * @param mixed|null $value Value
*/ */
public function set($key, $value = null) { public function set($key, $value = null): void
if (is_array($key) || is_object($key)) { {
if (\is_array($key) || \is_object($key)) {
foreach ($key as $k => $v) { foreach ($key as $k => $v) {
$this->vars[$k] = $v; $this->vars[$k] = $v;
} }
} } else {
else {
$this->vars[$key] = $value; $this->vars[$key] = $value;
} }
} }
@ -273,22 +294,24 @@ class Engine {
* Checks if a variable has been set. * Checks if a variable has been set.
* *
* @param string $key Key * @param string $key Key
*
* @return bool Variable status * @return bool Variable status
*/ */
public function has($key) { public function has(string $key): bool
{
return isset($this->vars[$key]); return isset($this->vars[$key]);
} }
/** /**
* Unsets a variable. If no key is passed in, clear all variables. * Unsets a variable. If no key is passed in, clear all variables.
* *
* @param string $key Key * @param string|null $key Key
*/ */
public function clear($key = null) { public function clear(?string $key = null): void
if (is_null($key)) { {
$this->vars = array(); if (null === $key) {
} $this->vars = [];
else { } else {
unset($this->vars[$key]); unset($this->vars[$key]);
} }
} }
@ -298,17 +321,20 @@ class Engine {
* *
* @param string $dir Directory path * @param string $dir Directory path
*/ */
public function path($dir) { public function path(string $dir): void
{
$this->loader->addDirectory($dir); $this->loader->addDirectory($dir);
} }
/*** Extensible Methods ***/ // Extensible Methods
/** /**
* Starts the framework. * Starts the framework.
* @throws \Exception *
* @throws Exception
*/ */
public function _start() { public function _start(): void
{
$dispatched = false; $dispatched = false;
$self = $this; $self = $this;
$request = $this->request(); $request = $this->request();
@ -345,7 +371,9 @@ class Engine {
$dispatched = true; $dispatched = true;
if (!$continue) break; if (!$continue) {
break;
}
$router->next(); $router->next();
@ -357,17 +385,45 @@ class Engine {
} }
} }
/**
* Sends an HTTP 500 response for any errors.
*
* @param Throwable $e Thrown exception
*/
public function _error($e): void
{
$msg = sprintf('<h1>500 Internal Server Error</h1>' .
'<h3>%s (%s)</h3>' .
'<pre>%s</pre>',
$e->getMessage(),
$e->getCode(),
$e->getTraceAsString()
);
try {
$this->response()
->clear()
->status(500)
->write($msg)
->send();
} catch (Throwable $t) {
exit($msg);
}
}
/** /**
* Stops the framework and outputs the current response. * Stops the framework and outputs the current response.
* *
* @param int $code HTTP status code * @param int|null $code HTTP status code
* @throws \Exception *
* @throws Exception
*/ */
public function _stop($code = null) { public function _stop(?int $code = null): void
{
$response = $this->response(); $response = $this->response();
if (!$response->sent()) { if (!$response->sent()) {
if ($code !== null) { if (null !== $code) {
$response->status($code); $response->status($code);
} }
@ -382,59 +438,82 @@ class Engine {
* *
* @param string $pattern URL pattern to match * @param string $pattern URL pattern to match
* @param callback $callback Callback function * @param callback $callback Callback function
* @param boolean $pass_route Pass the matching route object to the callback * @param bool $pass_route Pass the matching route object to the callback
*/ */
public function _route($pattern, $callback, $pass_route = false) { public function _route(string $pattern, callable $callback, bool $pass_route = false): void
{
$this->router()->map($pattern, $callback, $pass_route); $this->router()->map($pattern, $callback, $pass_route);
} }
/** /**
* Stops processing and returns a given response. * Routes a URL to a callback function.
* *
* @param int $code HTTP status code * @param string $pattern URL pattern to match
* @param string $message Response message * @param callback $callback Callback function
* @param bool $pass_route Pass the matching route object to the callback
*/ */
public function _halt($code = 200, $message = '') { public function _post(string $pattern, callable $callback, bool $pass_route = false): void
$this->response() {
->clear() $this->router()->map('POST ' . $pattern, $callback, $pass_route);
->status($code)
->write($message)
->send();
exit();
} }
/** /**
* Sends an HTTP 500 response for any errors. * Routes a URL to a callback function.
* *
* @param \Exception|\Throwable $e Thrown exception * @param string $pattern URL pattern to match
* @param callback $callback Callback function
* @param bool $pass_route Pass the matching route object to the callback
*/ */
public function _error($e) { public function _put(string $pattern, callable $callback, bool $pass_route = false): void
$msg = sprintf('<h1>500 Internal Server Error</h1>'. {
'<h3>%s (%s)</h3>'. $this->router()->map('PUT ' . $pattern, $callback, $pass_route);
'<pre>%s</pre>', }
$e->getMessage(),
$e->getCode(),
$e->getTraceAsString()
);
try { /**
* Routes a URL to a callback function.
*
* @param string $pattern URL pattern to match
* @param callback $callback Callback function
* @param bool $pass_route Pass the matching route object to the callback
*/
public function _patch(string $pattern, callable $callback, bool $pass_route = false): void
{
$this->router()->map('PATCH ' . $pattern, $callback, $pass_route);
}
/**
* Routes a URL to a callback function.
*
* @param string $pattern URL pattern to match
* @param callback $callback Callback function
* @param bool $pass_route Pass the matching route object to the callback
*/
public function _delete(string $pattern, callable $callback, bool $pass_route = false): void
{
$this->router()->map('DELETE ' . $pattern, $callback, $pass_route);
}
/**
* Stops processing and returns a given response.
*
* @param int $code HTTP status code
* @param string $message Response message
*/
public function _halt(int $code = 200, string $message = ''): void
{
$this->response() $this->response()
->clear() ->clear()
->status(500) ->status($code)
->write($msg) ->write($message)
->send(); ->send();
} exit();
catch (\Throwable $t) { // PHP 7.0+
exit($msg);
} catch(\Exception $e) { // PHP < 7
exit($msg);
}
} }
/** /**
* Sends an HTTP 404 response when a URL is not found. * Sends an HTTP 404 response when a URL is not found.
*/ */
public function _notFound() { public function _notFound(): void
{
$this->response() $this->response()
->clear() ->clear()
->status(404) ->status(404)
@ -452,15 +531,16 @@ class Engine {
* @param string $url URL * @param string $url URL
* @param int $code HTTP status code * @param int $code HTTP status code
*/ */
public function _redirect($url, $code = 303) { public function _redirect(string $url, int $code = 303): void
{
$base = $this->get('flight.base_url'); $base = $this->get('flight.base_url');
if ($base === null) { if (null === $base) {
$base = $this->request()->base; $base = $this->request()->base;
} }
// Append base url to redirect url // Append base url to redirect url
if ($base != '/' && strpos($url, '://') === false) { if ('/' !== $base && false === strpos($url, '://')) {
$url = $base . preg_replace('#/+#', '/', '/' . $url); $url = $base . preg_replace('#/+#', '/', '/' . $url);
} }
@ -475,15 +555,16 @@ class Engine {
* Renders a template. * Renders a template.
* *
* @param string $file Template file * @param string $file Template file
* @param array $data Template data * @param array|null $data Template data
* @param string $key View variable name * @param string|null $key View variable name
* @throws \Exception *
* @throws Exception
*/ */
public function _render($file, $data = null, $key = null) { public function _render(string $file, ?array $data = null, ?string $key = null): void
if ($key !== null) { {
if (null !== $key) {
$this->view()->set($key, $this->view()->fetch($file, $data)); $this->view()->set($key, $this->view()->fetch($file, $data));
} } else {
else {
$this->view()->render($file, $data); $this->view()->render($file, $data);
} }
} }
@ -496,16 +577,17 @@ class Engine {
* @param bool $encode Whether to perform JSON encoding * @param bool $encode Whether to perform JSON encoding
* @param string $charset Charset * @param string $charset Charset
* @param int $option Bitmask Json constant such as JSON_HEX_QUOT * @param int $option Bitmask Json constant such as JSON_HEX_QUOT
* @throws \Exception *
* @throws Exception
*/ */
public function _json( public function _json(
$data, $data,
$code = 200, int $code = 200,
$encode = true, bool $encode = true,
$charset = 'utf-8', string $charset = 'utf-8',
$option = 0 int $option = 0
) { ): void {
$json = ($encode) ? json_encode($data, $option) : $data; $json = $encode ? json_encode($data, $option) : $data;
$this->response() $this->response()
->status($code) ->status($code)
@ -523,17 +605,18 @@ class Engine {
* @param bool $encode Whether to perform JSON encoding * @param bool $encode Whether to perform JSON encoding
* @param string $charset Charset * @param string $charset Charset
* @param int $option Bitmask Json constant such as JSON_HEX_QUOT * @param int $option Bitmask Json constant such as JSON_HEX_QUOT
* @throws \Exception *
* @throws Exception
*/ */
public function _jsonp( public function _jsonp(
$data, $data,
$param = 'jsonp', string $param = 'jsonp',
$code = 200, int $code = 200,
$encode = true, bool $encode = true,
$charset = 'utf-8', string $charset = 'utf-8',
$option = 0 int $option = 0
) { ): void {
$json = ($encode) ? json_encode($data, $option) : $data; $json = $encode ? json_encode($data, $option) : $data;
$callback = $this->request()->query[$param]; $callback = $this->request()->query[$param];
@ -550,8 +633,9 @@ class Engine {
* @param string $id ETag identifier * @param string $id ETag identifier
* @param string $type ETag type * @param string $type ETag type
*/ */
public function _etag($id, $type = 'strong') { public function _etag(string $id, string $type = 'strong'): void
$id = (($type === 'weak') ? 'W/' : '').$id; {
$id = (('weak' === $type) ? 'W/' : '') . $id;
$this->response()->header('ETag', $id); $this->response()->header('ETag', $id);
@ -566,7 +650,8 @@ class Engine {
* *
* @param int $time Unix timestamp * @param int $time Unix timestamp
*/ */
public function _lastModified($time) { public function _lastModified(int $time): void
{
$this->response()->header('Last-Modified', gmdate('D, d M Y H:i:s \G\M\T', $time)); $this->response()->header('Last-Modified', gmdate('D, d M Y H:i:s \G\M\T', $time));
if (isset($_SERVER['HTTP_IF_MODIFIED_SINCE']) && if (isset($_SERVER['HTTP_IF_MODIFIED_SINCE']) &&

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -6,10 +8,18 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\core\Dispatcher;
use flight\Engine;
use flight\net\Request;
use flight\net\Response;
use flight\net\Router;
use flight\template\View;
/** /**
* The Flight class is a static representation of the framework. * The Flight class is a static representation of the framework.
* *
* Core. * Core.
*
* @method static void start() Starts the framework. * @method static void start() Starts the framework.
* @method static void path($path) Adds a path for autoloading classes. * @method static void path($path) Adds a path for autoloading classes.
* @method static void stop() Stops the framework and sends a response. * @method static void stop() Stops the framework and sends a response.
@ -17,7 +27,7 @@
* *
* Routing. * Routing.
* @method static void route($pattern, $callback) Maps a URL pattern to a callback. * @method static void route($pattern, $callback) Maps a URL pattern to a callback.
* @method static \flight\net\Router router() Returns Router instance. * @method static Router router() Returns Router instance.
* *
* Extending & Overriding. * Extending & Overriding.
* @method static void map($name, $callback) Creates a custom framework method. * @method static void map($name, $callback) Creates a custom framework method.
@ -35,11 +45,11 @@
* *
* Views. * Views.
* @method static void render($file, array $data = null, $key = null) Renders a template file. * @method static void render($file, array $data = null, $key = null) Renders a template file.
* @method static \flight\template\View view() Returns View instance. * @method static View view() Returns View instance.
* *
* Request & Response. * Request & Response.
* @method static \flight\net\Request request() Returns Request instance. * @method static Request request() Returns Request instance.
* @method static \flight\net\Response response() Returns Response instance. * @method static Response response() Returns Response instance.
* @method static void redirect($url, $code = 303) Redirects to another URL. * @method static void redirect($url, $code = 303) Redirects to another URL.
* @method static void json($data, $code = 200, $encode = true, $charset = "utf8", $encodeOption = 0, $encodeDepth = 512) Sends a JSON response. * @method static void json($data, $code = 200, $encode = true, $charset = "utf8", $encodeOption = 0, $encodeDepth = 512) Sends a JSON response.
* @method static void jsonp($data, $param = 'jsonp', $code = 200, $encode = true, $charset = "utf8", $encodeOption = 0, $encodeDepth = 512) Sends a JSONP response. * @method static void jsonp($data, $param = 'jsonp', $code = 200, $encode = true, $charset = "utf8", $encodeOption = 0, $encodeDepth = 512) Sends a JSONP response.
@ -50,43 +60,54 @@
* @method static void etag($id, $type = 'strong') Performs ETag HTTP caching. * @method static void etag($id, $type = 'strong') Performs ETag HTTP caching.
* @method static void lastModified($time) Performs last modified HTTP caching. * @method static void lastModified($time) Performs last modified HTTP caching.
*/ */
class Flight { class Flight
{
/** /**
* Framework engine. * Framework engine.
*
* @var \flight\Engine
*/ */
private static $engine; private static Engine $engine;
// Don't allow object instantiation // Don't allow object instantiation
private function __construct() {} private function __construct()
private function __destruct() {} {
private function __clone() {} }
private function __destruct()
{
}
private function __clone()
{
}
/** /**
* Handles calls to static methods. * Handles calls to static methods.
* *
* @param string $name Method name * @param string $name Method name
* @param array $params Method parameters * @param array $params Method parameters
*
* @throws Exception
*
* @return mixed Callback results * @return mixed Callback results
* @throws \Exception
*/ */
public static function __callStatic($name, $params) { public static function __callStatic(string $name, array $params)
$app = Flight::app(); {
$app = self::app();
return \flight\core\Dispatcher::invokeMethod(array($app, $name), $params); return Dispatcher::invokeMethod([$app, $name], $params);
} }
/** /**
* @return \flight\Engine Application instance * @return Engine Application instance
*/ */
public static function app() { public static function app(): Engine
{
static $initialized = false; static $initialized = false;
if (!$initialized) { if (!$initialized) {
require_once __DIR__ . '/autoload.php'; require_once __DIR__ . '/autoload.php';
self::$engine = new \flight\Engine(); self::$engine = new Engine();
$initialized = true; $initialized = true;
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -6,6 +8,8 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\core\Loader;
require_once __DIR__ . '/core/Loader.php'; require_once __DIR__ . '/core/Loader.php';
\flight\core\Loader::autoload(true, dirname(__DIR__)); Loader::autoload(true, [dirname(__DIR__)]);

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -8,36 +10,39 @@
namespace flight\core; namespace flight\core;
use Exception;
use InvalidArgumentException;
/** /**
* The Dispatcher class is responsible for dispatching events. Events * The Dispatcher class is responsible for dispatching events. Events
* are simply aliases for class methods or functions. The Dispatcher * are simply aliases for class methods or functions. The Dispatcher
* allows you to hook other functions to an event that can modify the * allows you to hook other functions to an event that can modify the
* input parameters and/or the output. * input parameters and/or the output.
*/ */
class Dispatcher { class Dispatcher
{
/** /**
* Mapped events. * Mapped events.
*
* @var array
*/ */
protected $events = array(); protected array $events = [];
/** /**
* Method filters. * Method filters.
*
* @var array
*/ */
protected $filters = array(); protected array $filters = [];
/** /**
* Dispatches an event. * Dispatches an event.
* *
* @param string $name Event name * @param string $name Event name
* @param array $params Callback parameters * @param array $params Callback parameters
* @return string Output of callback *
* @throws \Exception *@throws Exception
*
* @return mixed|null Output of callback
*/ */
public function run($name, array $params = array()) { final public function run(string $name, array $params = [])
{
$output = ''; $output = '';
// Run pre-filters // Run pre-filters
@ -46,7 +51,7 @@ class Dispatcher {
} }
// Run requested method // Run requested method
$output = $this->execute($this->get($name), $params); $output = self::execute($this->get($name), $params);
// Run post-filters // Run post-filters
if (!empty($this->filters[$name]['after'])) { if (!empty($this->filters[$name]['after'])) {
@ -62,7 +67,8 @@ class Dispatcher {
* @param string $name Event name * @param string $name Event name
* @param callback $callback Callback function * @param callback $callback Callback function
*/ */
public function set($name, $callback) { final public function set(string $name, callable $callback): void
{
$this->events[$name] = $callback; $this->events[$name] = $callback;
} }
@ -70,19 +76,23 @@ class Dispatcher {
* Gets an assigned callback. * Gets an assigned callback.
* *
* @param string $name Event name * @param string $name Event name
*
* @return callback $callback Callback function * @return callback $callback Callback function
*/ */
public function get($name) { final public function get(string $name): ?callable
return isset($this->events[$name]) ? $this->events[$name] : null; {
return $this->events[$name] ?? null;
} }
/** /**
* Checks if an event has been set. * Checks if an event has been set.
* *
* @param string $name Event name * @param string $name Event name
*
* @return bool Event status * @return bool Event status
*/ */
public function has($name) { final public function has(string $name): bool
{
return isset($this->events[$name]); return isset($this->events[$name]);
} }
@ -90,16 +100,16 @@ class Dispatcher {
* Clears an event. If no name is given, * Clears an event. If no name is given,
* all events are removed. * all events are removed.
* *
* @param string $name Event name * @param string|null $name Event name
*/ */
public function clear($name = null) { final public function clear(?string $name = null): void
if ($name !== null) { {
if (null !== $name) {
unset($this->events[$name]); unset($this->events[$name]);
unset($this->filters[$name]); unset($this->filters[$name]);
} } else {
else { $this->events = [];
$this->events = array(); $this->filters = [];
$this->filters = array();
} }
} }
@ -110,7 +120,8 @@ class Dispatcher {
* @param string $type Filter type * @param string $type Filter type
* @param callback $callback Callback function * @param callback $callback Callback function
*/ */
public function hook($name, $type, $callback) { final public function hook(string $name, string $type, callable $callback): void
{
$this->filters[$name][$type][] = $callback; $this->filters[$name][$type][] = $callback;
} }
@ -120,49 +131,57 @@ class Dispatcher {
* @param array $filters Chain of filters * @param array $filters Chain of filters
* @param array $params Method parameters * @param array $params Method parameters
* @param mixed $output Method output * @param mixed $output Method output
* @throws \Exception *
* @throws Exception
*/ */
public function filter($filters, &$params, &$output) { final public function filter(array $filters, array &$params, &$output): void
$args = array(&$params, &$output); {
$args = [&$params, &$output];
foreach ($filters as $callback) { foreach ($filters as $callback) {
$continue = $this->execute($callback, $args); $continue = self::execute($callback, $args);
if ($continue === false) break; if (false === $continue) {
break;
}
} }
} }
/** /**
* Executes a callback function. * Executes a callback function.
* *
* @param callback $callback Callback function * @param array|callback $callback Callback function
* @param array $params Function parameters * @param array $params Function parameters
*
*@throws Exception
*
* @return mixed Function results * @return mixed Function results
* @throws \Exception
*/ */
public static function execute($callback, array &$params = array()) { public static function execute($callback, array &$params = [])
if (is_callable($callback)) { {
return is_array($callback) ? if (\is_callable($callback)) {
return \is_array($callback) ?
self::invokeMethod($callback, $params) : self::invokeMethod($callback, $params) :
self::callFunction($callback, $params); self::callFunction($callback, $params);
} }
else {
throw new \Exception('Invalid callback specified.'); throw new InvalidArgumentException('Invalid callback specified.');
}
} }
/** /**
* Calls a function. * Calls a function.
* *
* @param string $func Name of function to call * @param callable|string $func Name of function to call
* @param array $params Function parameters * @param array $params Function parameters
*
* @return mixed Function results * @return mixed Function results
*/ */
public static function callFunction($func, array &$params = array()) { public static function callFunction($func, array &$params = [])
{
// Call static method // Call static method
if (is_string($func) && strpos($func, '::') !== false) { if (\is_string($func) && false !== strpos($func, '::')) {
return call_user_func_array($func, $params); return \call_user_func_array($func, $params);
} }
switch (count($params)) { switch (\count($params)) {
case 0: case 0:
return $func(); return $func();
case 1: case 1:
@ -176,7 +195,7 @@ class Dispatcher {
case 5: case 5:
return $func($params[0], $params[1], $params[2], $params[3], $params[4]); return $func($params[0], $params[1], $params[2], $params[3], $params[4]);
default: default:
return call_user_func_array($func, $params); return \call_user_func_array($func, $params);
} }
} }
@ -185,14 +204,16 @@ class Dispatcher {
* *
* @param mixed $func Class method * @param mixed $func Class method
* @param array $params Class method parameters * @param array $params Class method parameters
*
* @return mixed Function results * @return mixed Function results
*/ */
public static function invokeMethod($func, array &$params = array()) { public static function invokeMethod($func, array &$params = [])
list($class, $method) = $func; {
[$class, $method] = $func;
$instance = is_object($class); $instance = \is_object($class);
switch (count($params)) { switch (\count($params)) {
case 0: case 0:
return ($instance) ? return ($instance) ?
$class->$method() : $class->$method() :
@ -218,15 +239,16 @@ class Dispatcher {
$class->$method($params[0], $params[1], $params[2], $params[3], $params[4]) : $class->$method($params[0], $params[1], $params[2], $params[3], $params[4]) :
$class::$method($params[0], $params[1], $params[2], $params[3], $params[4]); $class::$method($params[0], $params[1], $params[2], $params[3], $params[4]);
default: default:
return call_user_func_array($func, $params); return \call_user_func_array($func, $params);
} }
} }
/** /**
* Resets the object to the initial state. * Resets the object to the initial state.
*/ */
public function reset() { final public function reset(): void
$this->events = array(); {
$this->filters = array(); $this->events = [];
$this->filters = [];
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -8,46 +10,46 @@
namespace flight\core; namespace flight\core;
use Exception;
use ReflectionClass;
use ReflectionException;
/** /**
* The Loader class is responsible for loading objects. It maintains * The Loader class is responsible for loading objects. It maintains
* a list of reusable class instances and can generate a new class * a list of reusable class instances and can generate a new class
* instances with custom initialization parameters. It also performs * instances with custom initialization parameters. It also performs
* class autoloading. * class autoloading.
*/ */
class Loader { class Loader
{
/** /**
* Registered classes. * Registered classes.
*
* @var array
*/ */
protected $classes = array(); protected array $classes = [];
/** /**
* Class instances. * Class instances.
*
* @var array
*/ */
protected $instances = array(); protected array $instances = [];
/** /**
* Autoload directories. * Autoload directories.
*
* @var array
*/ */
protected static $dirs = array(); protected static array $dirs = [];
/** /**
* Registers a class. * Registers a class.
* *
* @param string $name Registry name * @param string $name Registry name
* @param string|callable $class Class name or function to instantiate class * @param callable|string $class Class name or function to instantiate class
* @param array $params Class initialization parameters * @param array $params Class initialization parameters
* @param callback $callback Function to call after object instantiation * @param callable|null $callback $callback Function to call after object instantiation
*/ */
public function register($name, $class, array $params = array(), $callback = null) { public function register(string $name, $class, array $params = [], ?callable $callback = null): void
{
unset($this->instances[$name]); unset($this->instances[$name]);
$this->classes[$name] = array($class, $params, $callback); $this->classes[$name] = [$class, $params, $callback];
} }
/** /**
@ -55,7 +57,8 @@ class Loader {
* *
* @param string $name Registry name * @param string $name Registry name
*/ */
public function unregister($name) { public function unregister(string $name): void
{
unset($this->classes[$name]); unset($this->classes[$name]);
} }
@ -64,14 +67,17 @@ class Loader {
* *
* @param string $name Method name * @param string $name Method name
* @param bool $shared Shared instance * @param bool $shared Shared instance
*
* @throws Exception
*
* @return object Class instance * @return object Class instance
* @throws \Exception
*/ */
public function load($name, $shared = true) { public function load(string $name, bool $shared = true): ?object
{
$obj = null; $obj = null;
if (isset($this->classes[$name])) { if (isset($this->classes[$name])) {
list($class, $params, $callback) = $this->classes[$name]; [$class, $params, $callback] = $this->classes[$name];
$exists = isset($this->instances[$name]); $exists = isset($this->instances[$name]);
@ -83,14 +89,13 @@ class Loader {
if (!$exists) { if (!$exists) {
$this->instances[$name] = $obj; $this->instances[$name] = $obj;
} }
} } else {
else {
$obj = $this->newInstance($class, $params); $obj = $this->newInstance($class, $params);
} }
if ($callback && (!$shared || !$exists)) { if ($callback && (!$shared || !$exists)) {
$ref = array(&$obj); $ref = [&$obj];
call_user_func_array($callback, $ref); \call_user_func_array($callback, $ref);
} }
} }
@ -101,26 +106,31 @@ class Loader {
* Gets a single instance of a class. * Gets a single instance of a class.
* *
* @param string $name Instance name * @param string $name Instance name
*
* @return object Class instance * @return object Class instance
*/ */
public function getInstance($name) { public function getInstance(string $name): ?object
return isset($this->instances[$name]) ? $this->instances[$name] : null; {
return $this->instances[$name] ?? null;
} }
/** /**
* Gets a new instance of a class. * Gets a new instance of a class.
* *
* @param string|callable $class Class name or callback function to instantiate class * @param callable|string $class Class name or callback function to instantiate class
* @param array $params Class initialization parameters * @param array $params Class initialization parameters
*
* @throws Exception
*
* @return object Class instance * @return object Class instance
* @throws \Exception
*/ */
public function newInstance($class, array $params = array()) { public function newInstance($class, array $params = []): object
if (is_callable($class)) { {
return call_user_func_array($class, $params); if (\is_callable($class)) {
return \call_user_func_array($class, $params);
} }
switch (count($params)) { switch (\count($params)) {
case 0: case 0:
return new $class(); return new $class();
case 1: case 1:
@ -135,44 +145,48 @@ class Loader {
return new $class($params[0], $params[1], $params[2], $params[3], $params[4]); return new $class($params[0], $params[1], $params[2], $params[3], $params[4]);
default: default:
try { try {
$refClass = new \ReflectionClass($class); $refClass = new ReflectionClass($class);
return $refClass->newInstanceArgs($params); return $refClass->newInstanceArgs($params);
} catch (\ReflectionException $e) { } catch (ReflectionException $e) {
throw new \Exception("Cannot instantiate {$class}", 0, $e); throw new Exception("Cannot instantiate {$class}", 0, $e);
} }
} }
} }
/** /**
* @param string $name Registry name * @param string $name Registry name
*
* @return mixed Class information or null if not registered * @return mixed Class information or null if not registered
*/ */
public function get($name) { public function get(string $name)
return isset($this->classes[$name]) ? $this->classes[$name] : null; {
return $this->classes[$name] ?? null;
} }
/** /**
* Resets the object to the initial state. * Resets the object to the initial state.
*/ */
public function reset() { public function reset(): void
$this->classes = array(); {
$this->instances = array(); $this->classes = [];
$this->instances = [];
} }
/*** Autoloading Functions ***/ // Autoloading Functions
/** /**
* Starts/stops autoloader. * Starts/stops autoloader.
* *
* @param bool $enabled Enable/disable autoloading * @param bool $enabled Enable/disable autoloading
* @param array $dirs Autoload directories * @param mixed $dirs Autoload directories
*/ */
public static function autoload($enabled = true, $dirs = array()) { public static function autoload(bool $enabled = true, $dirs = []): void
{
if ($enabled) { if ($enabled) {
spl_autoload_register(array(__CLASS__, 'loadClass')); spl_autoload_register([__CLASS__, 'loadClass']);
} } else {
else { spl_autoload_unregister([__CLASS__, 'loadClass']);
spl_autoload_unregister(array(__CLASS__, 'loadClass'));
} }
if (!empty($dirs)) { if (!empty($dirs)) {
@ -185,13 +199,15 @@ class Loader {
* *
* @param string $class Class name * @param string $class Class name
*/ */
public static function loadClass($class) { public static function loadClass(string $class): void
$class_file = str_replace(array('\\', '_'), '/', $class).'.php'; {
$class_file = str_replace(['\\', '_'], '/', $class) . '.php';
foreach (self::$dirs as $dir) { foreach (self::$dirs as $dir) {
$file = $dir . '/' . $class_file; $file = $dir . '/' . $class_file;
if (file_exists($file)) { if (file_exists($file)) {
require $file; require $file;
return; return;
} }
} }
@ -202,14 +218,16 @@ class Loader {
* *
* @param mixed $dir Directory path * @param mixed $dir Directory path
*/ */
public static function addDirectory($dir) { public static function addDirectory($dir): void
if (is_array($dir) || is_object($dir)) { {
if (\is_array($dir) || \is_object($dir)) {
foreach ($dir as $value) { foreach ($dir as $value) {
self::addDirectory($value); self::addDirectory($value);
} }
} elseif (\is_string($dir)) {
if (!\in_array($dir, self::$dirs, true)) {
self::$dirs[] = $dir;
} }
else if (is_string($dir)) {
if (!in_array($dir, self::$dirs)) self::$dirs[] = $dir;
} }
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -34,112 +36,114 @@ use flight\util\Collection;
* accept - HTTP accept parameters * accept - HTTP accept parameters
* proxy_ip - Proxy IP address of the client * proxy_ip - Proxy IP address of the client
*/ */
class Request { final class Request
{
/** /**
* @var string URL being requested * @var string URL being requested
*/ */
public $url; public string $url;
/** /**
* @var string Parent subdirectory of the URL * @var string Parent subdirectory of the URL
*/ */
public $base; public string $base;
/** /**
* @var string Request method (GET, POST, PUT, DELETE) * @var string Request method (GET, POST, PUT, DELETE)
*/ */
public $method; public string $method;
/** /**
* @var string Referrer URL * @var string Referrer URL
*/ */
public $referrer; public string $referrer;
/** /**
* @var string IP address of the client * @var string IP address of the client
*/ */
public $ip; public string $ip;
/** /**
* @var bool Whether the request is an AJAX request * @var bool Whether the request is an AJAX request
*/ */
public $ajax; public bool $ajax;
/** /**
* @var string Server protocol (http, https) * @var string Server protocol (http, https)
*/ */
public $scheme; public string $scheme;
/** /**
* @var string Browser information * @var string Browser information
*/ */
public $user_agent; public string $user_agent;
/** /**
* @var string Content type * @var string Content type
*/ */
public $type; public string $type;
/** /**
* @var int Content length * @var int Content length
*/ */
public $length; public int $length;
/** /**
* @var \flight\util\Collection Query string parameters * @var Collection Query string parameters
*/ */
public $query; public Collection $query;
/** /**
* @var \flight\util\Collection Post parameters * @var Collection Post parameters
*/ */
public $data; public Collection $data;
/** /**
* @var \flight\util\Collection Cookie parameters * @var Collection Cookie parameters
*/ */
public $cookies; public Collection $cookies;
/** /**
* @var \flight\util\Collection Uploaded files * @var Collection Uploaded files
*/ */
public $files; public Collection $files;
/** /**
* @var bool Whether the connection is secure * @var bool Whether the connection is secure
*/ */
public $secure; public bool $secure;
/** /**
* @var string HTTP accept parameters * @var string HTTP accept parameters
*/ */
public $accept; public string $accept;
/** /**
* @var string Proxy IP address of the client * @var string Proxy IP address of the client
*/ */
public $proxy_ip; public string $proxy_ip;
/** /**
* @var string HTTP host name * @var string HTTP host name
*/ */
public $host; public string $host;
/** /**
* Constructor. * Constructor.
* *
* @param array $config Request configuration * @param array $config Request configuration
*/ */
public function __construct($config = array()) { public function __construct(array $config = [])
{
// Default properties // Default properties
if (empty($config)) { if (empty($config)) {
$config = array( $config = [
'url' => str_replace('@', '%40', self::getVar('REQUEST_URI', '/')), 'url' => str_replace('@', '%40', self::getVar('REQUEST_URI', '/')),
'base' => str_replace(array('\\',' '), array('/','%20'), dirname(self::getVar('SCRIPT_NAME'))), 'base' => str_replace(['\\', ' '], ['/', '%20'], \dirname(self::getVar('SCRIPT_NAME'))),
'method' => self::getMethod(), 'method' => self::getMethod(),
'referrer' => self::getVar('HTTP_REFERER'), 'referrer' => self::getVar('HTTP_REFERER'),
'ip' => self::getVar('REMOTE_ADDR'), 'ip' => self::getVar('REMOTE_ADDR'),
'ajax' => self::getVar('HTTP_X_REQUESTED_WITH') == 'XMLHttpRequest', 'ajax' => 'XMLHttpRequest' === self::getVar('HTTP_X_REQUESTED_WITH'),
'scheme' => self::getScheme(), 'scheme' => self::getScheme(),
'user_agent' => self::getVar('HTTP_USER_AGENT'), 'user_agent' => self::getVar('HTTP_USER_AGENT'),
'type' => self::getVar('CONTENT_TYPE'), 'type' => self::getVar('CONTENT_TYPE'),
@ -148,11 +152,11 @@ class Request {
'data' => new Collection($_POST), 'data' => new Collection($_POST),
'cookies' => new Collection($_COOKIE), 'cookies' => new Collection($_COOKIE),
'files' => new Collection($_FILES), 'files' => new Collection($_FILES),
'secure' => self::getScheme() == 'https', 'secure' => 'https' === self::getScheme(),
'accept' => self::getVar('HTTP_ACCEPT'), 'accept' => self::getVar('HTTP_ACCEPT'),
'proxy_ip' => self::getProxyIpAddress(), 'proxy_ip' => self::getProxyIpAddress(),
'host' => self::getVar('HTTP_HOST'), 'host' => self::getVar('HTTP_HOST'),
); ];
} }
$this->init($config); $this->init($config);
@ -163,32 +167,32 @@ class Request {
* *
* @param array $properties Array of request properties * @param array $properties Array of request properties
*/ */
public function init($properties = array()) { public function init(array $properties = [])
{
// Set all the defined properties // Set all the defined properties
foreach ($properties as $name => $value) { foreach ($properties as $name => $value) {
$this->$name = $value; $this->$name = $value;
} }
// Get the requested URL without the base directory // Get the requested URL without the base directory
if ($this->base != '/' && strlen($this->base) > 0 && strpos($this->url, $this->base) === 0) { if ('/' !== $this->base && '' !== $this->base && 0 === strpos($this->url, $this->base)) {
$this->url = substr($this->url, strlen($this->base)); $this->url = substr($this->url, \strlen($this->base));
} }
// Default url // Default url
if (empty($this->url)) { if (empty($this->url)) {
$this->url = '/'; $this->url = '/';
} } else {
// Merge URL query parameters with $_GET // Merge URL query parameters with $_GET
else { $_GET = array_merge($_GET, self::parseQuery($this->url));
$_GET += self::parseQuery($this->url);
$this->query->setData($_GET); $this->query->setData($_GET);
} }
// Check for JSON input // Check for JSON input
if (strpos($this->type, 'application/json') === 0) { if (0 === strpos($this->type, 'application/json')) {
$body = $this->getBody(); $body = self::getBody();
if ($body != '') { if ('' !== $body) {
$data = json_decode($body, true); $data = json_decode($body, true);
if (is_array($data)) { if (is_array($data)) {
$this->data->setData($data); $this->data->setData($data);
@ -202,16 +206,17 @@ class Request {
* *
* @return string Raw HTTP request body * @return string Raw HTTP request body
*/ */
public static function getBody() { public static function getBody(): ?string
{
static $body; static $body;
if (!is_null($body)) { if (null !== $body) {
return $body; return $body;
} }
$method = self::getMethod(); $method = self::getMethod();
if ($method == 'POST' || $method == 'PUT' || $method == 'DELETE' || $method == 'PATCH') { if ('POST' === $method || 'PUT' === $method || 'DELETE' === $method || 'PATCH' === $method) {
$body = file_get_contents('php://input'); $body = file_get_contents('php://input');
} }
@ -220,16 +225,14 @@ class Request {
/** /**
* Gets the request method. * Gets the request method.
*
* @return string
*/ */
public static function getMethod() { public static function getMethod(): string
{
$method = self::getVar('REQUEST_METHOD', 'GET'); $method = self::getVar('REQUEST_METHOD', 'GET');
if (isset($_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'])) { if (isset($_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'])) {
$method = $_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE']; $method = $_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'];
} } elseif (isset($_REQUEST['_method'])) {
elseif (isset($_REQUEST['_method'])) {
$method = $_REQUEST['_method']; $method = $_REQUEST['_method'];
} }
@ -241,22 +244,23 @@ class Request {
* *
* @return string IP address * @return string IP address
*/ */
public static function getProxyIpAddress() { public static function getProxyIpAddress(): string
static $forwarded = array( {
static $forwarded = [
'HTTP_CLIENT_IP', 'HTTP_CLIENT_IP',
'HTTP_X_FORWARDED_FOR', 'HTTP_X_FORWARDED_FOR',
'HTTP_X_FORWARDED', 'HTTP_X_FORWARDED',
'HTTP_X_CLUSTER_CLIENT_IP', 'HTTP_X_CLUSTER_CLIENT_IP',
'HTTP_FORWARDED_FOR', 'HTTP_FORWARDED_FOR',
'HTTP_FORWARDED' 'HTTP_FORWARDED',
); ];
$flags = \FILTER_FLAG_NO_PRIV_RANGE | \FILTER_FLAG_NO_RES_RANGE; $flags = \FILTER_FLAG_NO_PRIV_RANGE | \FILTER_FLAG_NO_RES_RANGE;
foreach ($forwarded as $key) { foreach ($forwarded as $key) {
if (array_key_exists($key, $_SERVER)) { if (\array_key_exists($key, $_SERVER)) {
sscanf($_SERVER[$key], '%[^,]', $ip); sscanf($_SERVER[$key], '%[^,]', $ip);
if (filter_var($ip, \FILTER_VALIDATE_IP, $flags) !== false) { if (false !== filter_var($ip, \FILTER_VALIDATE_IP, $flags)) {
return $ip; return $ip;
} }
} }
@ -269,21 +273,25 @@ class Request {
* Gets a variable from $_SERVER using $default if not provided. * Gets a variable from $_SERVER using $default if not provided.
* *
* @param string $var Variable name * @param string $var Variable name
* @param string $default Default value to substitute * @param mixed $default Default value to substitute
* @return string Server variable value *
* @return mixed Server variable value
*/ */
public static function getVar($var, $default = '') { public static function getVar(string $var, $default = '')
return isset($_SERVER[$var]) ? $_SERVER[$var] : $default; {
return $_SERVER[$var] ?? $default;
} }
/** /**
* Parse query parameters from a URL. * Parse query parameters from a URL.
* *
* @param string $url URL string * @param string $url URL string
*
* @return array Query parameters * @return array Query parameters
*/ */
public static function parseQuery($url) { public static function parseQuery(string $url): array
$params = array(); {
$params = [];
$args = parse_url($url); $args = parse_url($url);
if (isset($args['query'])) { if (isset($args['query'])) {
@ -293,18 +301,20 @@ class Request {
return $params; return $params;
} }
public static function getScheme() { public static function getScheme(): string
{
if ( if (
(isset($_SERVER['HTTPS']) && strtolower($_SERVER['HTTPS']) === 'on') (isset($_SERVER['HTTPS']) && 'on' === strtolower($_SERVER['HTTPS']))
|| ||
(isset($_SERVER['HTTP_X_FORWARDED_PROTO']) && $_SERVER['HTTP_X_FORWARDED_PROTO'] === 'https') (isset($_SERVER['HTTP_X_FORWARDED_PROTO']) && 'https' === $_SERVER['HTTP_X_FORWARDED_PROTO'])
|| ||
(isset($_SERVER['HTTP_FRONT_END_HTTPS']) && $_SERVER['HTTP_FRONT_END_HTTPS'] === 'on') (isset($_SERVER['HTTP_FRONT_END_HTTPS']) && 'on' === $_SERVER['HTTP_FRONT_END_HTTPS'])
|| ||
(isset($_SERVER['REQUEST_SCHEME']) && $_SERVER['REQUEST_SCHEME'] === 'https') (isset($_SERVER['REQUEST_SCHEME']) && 'https' === $_SERVER['REQUEST_SCHEME'])
) { ) {
return 'https'; return 'https';
} }
return 'http'; return 'http';
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -8,43 +10,24 @@
namespace flight\net; namespace flight\net;
use Exception;
/** /**
* The Response class represents an HTTP response. The object * The Response class represents an HTTP response. The object
* contains the response headers, HTTP status code, and response * contains the response headers, HTTP status code, and response
* body. * body.
*/ */
class Response { class Response
/** {
* @var int HTTP status
*/
protected $status = 200;
/**
* @var array HTTP headers
*/
protected $headers = array();
/** /**
* @var string HTTP response body * header Content-Length.
*/ */
protected $body; public bool $content_length = true;
/**
* @var bool HTTP response sent
*/
protected $sent = false;
/**
* header Content-Length
*
* @var boolean
*/
public $content_length = true;
/** /**
* @var array HTTP status codes * @var array HTTP status codes
*/ */
public static $codes = array( public static array $codes = [
100 => 'Continue', 100 => 'Continue',
101 => 'Switching Protocols', 101 => 'Switching Protocols',
102 => 'Processing', 102 => 'Processing',
@ -112,26 +95,47 @@ class Response {
508 => 'Loop Detected', 508 => 'Loop Detected',
510 => 'Not Extended', 510 => 'Not Extended',
511 => 'Network Authentication Required' 511 => 'Network Authentication Required',
); ];
/**
* @var int HTTP status
*/
protected int $status = 200;
/**
* @var array HTTP headers
*/
protected array $headers = [];
/**
* @var string HTTP response body
*/
protected string $body = '';
/**
* @var bool HTTP response sent
*/
protected bool $sent = false;
/** /**
* Sets the HTTP status of the response. * Sets the HTTP status of the response.
* *
* @param int $code HTTP status code. * @param int|null $code HTTP status code.
* @return object|int Self reference *
* @throws \Exception If invalid status code * @throws Exception If invalid status code
*
* @return int|object Self reference
*/ */
public function status($code = null) { public function status(?int $code = null)
if ($code === null) { {
if (null === $code) {
return $this->status; return $this->status;
} }
if (array_key_exists($code, self::$codes)) { if (\array_key_exists($code, self::$codes)) {
$this->status = $code; $this->status = $code;
} } else {
else { throw new Exception('Invalid status code.');
throw new \Exception('Invalid status code.');
} }
return $this; return $this;
@ -140,17 +144,18 @@ class Response {
/** /**
* Adds a header to the response. * Adds a header to the response.
* *
* @param string|array $name Header name or array of names and values * @param array|string $name Header name or array of names and values
* @param string $value Header value * @param string|null $value Header value
*
* @return object Self reference * @return object Self reference
*/ */
public function header($name, $value = null) { public function header($name, ?string $value = null)
if (is_array($name)) { {
if (\is_array($name)) {
foreach ($name as $k => $v) { foreach ($name as $k => $v) {
$this->headers[$k] = $v; $this->headers[$k] = $v;
} }
} } else {
else {
$this->headers[$name] = $value; $this->headers[$name] = $value;
} }
@ -158,10 +163,12 @@ class Response {
} }
/** /**
* Returns the headers from the response * Returns the headers from the response.
*
* @return array * @return array
*/ */
public function headers() { public function headers()
{
return $this->headers; return $this->headers;
} }
@ -169,9 +176,11 @@ class Response {
* Writes content to the response body. * Writes content to the response body.
* *
* @param string $str Response content * @param string $str Response content
* @return object Self reference *
* @return Response Self reference
*/ */
public function write($str) { public function write(string $str): self
{
$this->body .= $str; $this->body .= $str;
return $this; return $this;
@ -180,11 +189,12 @@ class Response {
/** /**
* Clears the response. * Clears the response.
* *
* @return object Self reference * @return Response Self reference
*/ */
public function clear() { public function clear(): self
{
$this->status = 200; $this->status = 200;
$this->headers = array(); $this->headers = [];
$this->body = ''; $this->body = '';
return $this; return $this;
@ -194,37 +204,40 @@ class Response {
* Sets caching headers for the response. * Sets caching headers for the response.
* *
* @param int|string $expires Expiration time * @param int|string $expires Expiration time
* @return object Self reference *
* @return Response Self reference
*/ */
public function cache($expires) { public function cache($expires): self
if ($expires === false) { {
if (false === $expires) {
$this->headers['Expires'] = 'Mon, 26 Jul 1997 05:00:00 GMT'; $this->headers['Expires'] = 'Mon, 26 Jul 1997 05:00:00 GMT';
$this->headers['Cache-Control'] = array( $this->headers['Cache-Control'] = [
'no-store, no-cache, must-revalidate', 'no-store, no-cache, must-revalidate',
'post-check=0, pre-check=0', 'post-check=0, pre-check=0',
'max-age=0' 'max-age=0',
); ];
$this->headers['Pragma'] = 'no-cache'; $this->headers['Pragma'] = 'no-cache';
} } else {
else { $expires = \is_int($expires) ? $expires : strtotime($expires);
$expires = is_int($expires) ? $expires : strtotime($expires);
$this->headers['Expires'] = gmdate('D, d M Y H:i:s', $expires) . ' GMT'; $this->headers['Expires'] = gmdate('D, d M Y H:i:s', $expires) . ' GMT';
$this->headers['Cache-Control'] = 'max-age=' . ($expires - time()); $this->headers['Cache-Control'] = 'max-age=' . ($expires - time());
if (isset($this->headers['Pragma']) && $this->headers['Pragma'] == 'no-cache'){ if (isset($this->headers['Pragma']) && 'no-cache' == $this->headers['Pragma']) {
unset($this->headers['Pragma']); unset($this->headers['Pragma']);
} }
} }
return $this; return $this;
} }
/** /**
* Sends HTTP headers. * Sends HTTP headers.
* *
* @return object Self reference * @return Response Self reference
*/ */
public function sendHeaders() { public function sendHeaders(): self
{
// Send status code header // Send status code header
if (strpos(php_sapi_name(), 'cgi') !== false) { if (false !== strpos(\PHP_SAPI, 'cgi')) {
header( header(
sprintf( sprintf(
'Status: %d %s', 'Status: %d %s',
@ -233,12 +246,11 @@ class Response {
), ),
true true
); );
} } else {
else {
header( header(
sprintf( sprintf(
'%s %d %s', '%s %d %s',
(isset($_SERVER['SERVER_PROTOCOL']) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.1'), $_SERVER['SERVER_PROTOCOL'] ?? 'HTTP/1.1',
$this->status, $this->status,
self::$codes[$this->status]), self::$codes[$this->status]),
true, true,
@ -248,12 +260,11 @@ class Response {
// Send other headers // Send other headers
foreach ($this->headers as $field => $value) { foreach ($this->headers as $field => $value) {
if (is_array($value)) { if (\is_array($value)) {
foreach ($value as $v) { foreach ($value as $v) {
header($field . ': ' . $v, false); header($field . ': ' . $v, false);
} }
} } else {
else {
header($field . ': ' . $value); header($field . ': ' . $value);
} }
} }
@ -273,25 +284,28 @@ class Response {
/** /**
* Gets the content length. * Gets the content length.
* *
* @return string Content length * @return int Content length
*/ */
public function getContentLength() { public function getContentLength(): int
return extension_loaded('mbstring') ? {
return \extension_loaded('mbstring') ?
mb_strlen($this->body, 'latin1') : mb_strlen($this->body, 'latin1') :
strlen($this->body); \strlen($this->body);
} }
/** /**
* Gets whether response was sent. * Gets whether response was sent.
*/ */
public function sent() { public function sent(): bool
{
return $this->sent; return $this->sent;
} }
/** /**
* Sends a HTTP response. * Sends a HTTP response.
*/ */
public function send() { public function send(): void
{
if (ob_get_length() > 0) { if (ob_get_length() > 0) {
ob_end_clean(); ob_end_clean();
} }
@ -305,4 +319,3 @@ class Response {
$this->sent = true; $this->sent = true;
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -13,11 +15,12 @@ namespace flight\net;
* an assigned callback function. The Router tries to match the * an assigned callback function. The Router tries to match the
* requested URL against a series of URL patterns. * requested URL against a series of URL patterns.
*/ */
class Route { final class Route
{
/** /**
* @var string URL pattern * @var string URL pattern
*/ */
public $pattern; public string $pattern;
/** /**
* @var mixed Callback function * @var mixed Callback function
@ -27,27 +30,27 @@ class Route {
/** /**
* @var array HTTP methods * @var array HTTP methods
*/ */
public $methods = array(); public array $methods = [];
/** /**
* @var array Route parameters * @var array Route parameters
*/ */
public $params = array(); public array $params = [];
/** /**
* @var string Matching regular expression * @var string|null Matching regular expression
*/ */
public $regex; public ?string $regex = null;
/** /**
* @var string URL splat content * @var string URL splat content
*/ */
public $splat = ''; public string $splat = '';
/** /**
* @var boolean Pass self in callback parameters * @var bool Pass self in callback parameters
*/ */
public $pass = false; public bool $pass = false;
/** /**
* Constructor. * Constructor.
@ -55,9 +58,10 @@ class Route {
* @param string $pattern URL pattern * @param string $pattern URL pattern
* @param mixed $callback Callback function * @param mixed $callback Callback function
* @param array $methods HTTP methods * @param array $methods HTTP methods
* @param boolean $pass Pass self in callback parameters * @param bool $pass Pass self in callback parameters
*/ */
public function __construct($pattern, $callback, $methods, $pass) { public function __construct(string $pattern, $callback, array $methods, bool $pass)
{
$this->pattern = $pattern; $this->pattern = $pattern;
$this->callback = $callback; $this->callback = $callback;
$this->methods = $methods; $this->methods = $methods;
@ -68,52 +72,58 @@ class Route {
* Checks if a URL matches the route pattern. Also parses named parameters in the URL. * Checks if a URL matches the route pattern. Also parses named parameters in the URL.
* *
* @param string $url Requested URL * @param string $url Requested URL
* @param boolean $case_sensitive Case sensitive matching * @param bool $case_sensitive Case sensitive matching
* @return boolean Match status *
* @return bool Match status
*/ */
public function matchUrl($url, $case_sensitive = false) { public function matchUrl(string $url, bool $case_sensitive = false): bool
{
// Wildcard or exact match // Wildcard or exact match
if ($this->pattern === '*' || $this->pattern === $url) { if ('*' === $this->pattern || $this->pattern === $url) {
return true; return true;
} }
$ids = array(); $ids = [];
$last_char = substr($this->pattern, -1); $last_char = substr($this->pattern, -1);
// Get splat // Get splat
if ($last_char === '*') { if ('*' === $last_char) {
$n = 0; $n = 0;
$len = strlen($url); $len = \strlen($url);
$count = substr_count($this->pattern, '/'); $count = substr_count($this->pattern, '/');
for ($i = 0; $i < $len; $i++) { for ($i = 0; $i < $len; $i++) {
if ($url[$i] == '/') $n++; if ('/' === $url[$i]) {
if ($n == $count) break; $n++;
}
if ($n === $count) {
break;
}
} }
$this->splat = (string) substr($url, $i + 1); $this->splat = (string) substr($url, $i + 1);
} }
// Build the regex for matching // Build the regex for matching
$regex = str_replace(array(')','/*'), array(')?','(/?|/.*?)'), $this->pattern); $regex = str_replace([')', '/*'], [')?', '(/?|/.*?)'], $this->pattern);
$regex = preg_replace_callback( $regex = preg_replace_callback(
'#@([\w]+)(:([^/\(\)]*))?#', '#@([\w]+)(:([^/\(\)]*))?#',
function($matches) use (&$ids) { static function ($matches) use (&$ids) {
$ids[$matches[1]] = null; $ids[$matches[1]] = null;
if (isset($matches[3])) { if (isset($matches[3])) {
return '(?P<' . $matches[1] . '>' . $matches[3] . ')'; return '(?P<' . $matches[1] . '>' . $matches[3] . ')';
} }
return '(?P<' . $matches[1] . '>[^/\?]+)'; return '(?P<' . $matches[1] . '>[^/\?]+)';
}, },
$regex $regex
); );
// Fix trailing slash // Fix trailing slash
if ($last_char === '/') { if ('/' === $last_char) {
$regex .= '?'; $regex .= '?';
} } // Allow trailing slash
// Allow trailing slash
else { else {
$regex .= '/?'; $regex .= '/?';
} }
@ -121,7 +131,7 @@ class Route {
// Attempt to match route and named parameters // Attempt to match route and named parameters
if (preg_match('#^' . $regex . '(?:\?.*)?$#' . (($case_sensitive) ? '' : 'i'), $url, $matches)) { if (preg_match('#^' . $regex . '(?:\?.*)?$#' . (($case_sensitive) ? '' : 'i'), $url, $matches)) {
foreach ($ids as $k => $v) { foreach ($ids as $k => $v) {
$this->params[$k] = (array_key_exists($k, $matches)) ? urldecode($matches[$k]) : null; $this->params[$k] = (\array_key_exists($k, $matches)) ? urldecode($matches[$k]) : null;
} }
$this->regex = $regex; $this->regex = $regex;
@ -136,9 +146,11 @@ class Route {
* Checks if an HTTP method matches the route methods. * Checks if an HTTP method matches the route methods.
* *
* @param string $method HTTP method * @param string $method HTTP method
*
* @return bool Match status * @return bool Match status
*/ */
public function matchMethod($method) { public function matchMethod(string $method): bool
return count(array_intersect(array($method, '*'), $this->methods)) > 0; {
return \count(array_intersect([$method, '*'], $this->methods)) > 0;
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -13,42 +15,38 @@ namespace flight\net;
* an assigned callback function. The Router tries to match the * an assigned callback function. The Router tries to match the
* requested URL against a series of URL patterns. * requested URL against a series of URL patterns.
*/ */
class Router { class Router
{
/** /**
* Mapped routes. * Case sensitive matching.
*
* @var array
*/ */
protected $routes = array(); public bool $case_sensitive = false;
/** /**
* Pointer to current route. * Mapped routes.
*
* @var int
*/ */
protected $index = 0; protected array $routes = [];
/** /**
* Case sensitive matching. * Pointer to current route.
*
* @var boolean
*/ */
public $case_sensitive = false; protected int $index = 0;
/** /**
* Gets mapped routes. * Gets mapped routes.
* *
* @return array Array of routes * @return array Array of routes
*/ */
public function getRoutes() { public function getRoutes(): array
{
return $this->routes; return $this->routes;
} }
/** /**
* Clears all routes in the router. * Clears all routes in the router.
*/ */
public function clear() { public function clear(): void
$this->routes = array(); {
$this->routes = [];
} }
/** /**
@ -56,14 +54,15 @@ class Router {
* *
* @param string $pattern URL pattern to match * @param string $pattern URL pattern to match
* @param callback $callback Callback function * @param callback $callback Callback function
* @param boolean $pass_route Pass the matching route object to the callback * @param bool $pass_route Pass the matching route object to the callback
*/ */
public function map($pattern, $callback, $pass_route = false) { public function map(string $pattern, callable $callback, bool $pass_route = false): void
$url = $pattern; {
$methods = array('*'); $url = trim($pattern);
$methods = ['*'];
if (strpos($pattern, ' ') !== false) { if (false !== strpos($url, ' ')) {
list($method, $url) = explode(' ', trim($pattern), 2); [$method, $url] = explode(' ', $url, 2);
$url = trim($url); $url = trim($url);
$methods = explode('|', $method); $methods = explode('|', $method);
} }
@ -75,12 +74,14 @@ class Router {
* Routes the current request. * Routes the current request.
* *
* @param Request $request Request object * @param Request $request Request object
* @return Route|bool Matching route or false if no match *
* @return bool|Route Matching route or false if no match
*/ */
public function route(Request $request) { public function route(Request $request)
{
$url_decoded = urldecode($request->url); $url_decoded = urldecode($request->url);
while ($route = $this->current()) { while ($route = $this->current()) {
if ($route !== false && $route->matchMethod($request->method) && $route->matchUrl($url_decoded, $this->case_sensitive)) { if (false !== $route && $route->matchMethod($request->method) && $route->matchUrl($url_decoded, $this->case_sensitive)) {
return $route; return $route;
} }
$this->next(); $this->next();
@ -92,26 +93,26 @@ class Router {
/** /**
* Gets the current route. * Gets the current route.
* *
* @return Route * @return bool|Route
*/ */
public function current() { public function current()
return isset($this->routes[$this->index]) ? $this->routes[$this->index] : false; {
return $this->routes[$this->index] ?? false;
} }
/** /**
* Gets the next route. * Gets the next route.
*
* @return Route
*/ */
public function next() { public function next(): void
{
$this->index++; $this->index++;
} }
/** /**
* Reset to the first route. * Reset to the first route.
*/ */
public function reset() { public function reset(): void
{
$this->index = 0; $this->index = 0;
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -13,7 +15,8 @@ namespace flight\template;
* methods for managing view data and inserts the data into * methods for managing view data and inserts the data into
* view templates upon rendering. * view templates upon rendering.
*/ */
class View { class View
{
/** /**
* Location of view templates. * Location of view templates.
* *
@ -33,7 +36,7 @@ class View {
* *
* @var array * @var array
*/ */
protected $vars = array(); protected $vars = [];
/** /**
* Template file. * Template file.
@ -47,7 +50,8 @@ class View {
* *
* @param string $path Path to templates directory * @param string $path Path to templates directory
*/ */
public function __construct($path = '.') { public function __construct($path = '.')
{
$this->path = $path; $this->path = $path;
} }
@ -55,10 +59,12 @@ class View {
* Gets a template variable. * Gets a template variable.
* *
* @param string $key Key * @param string $key Key
*
* @return mixed Value * @return mixed Value
*/ */
public function get($key) { public function get($key)
return isset($this->vars[$key]) ? $this->vars[$key] : null; {
return $this->vars[$key] ?? null;
} }
/** /**
@ -67,13 +73,13 @@ class View {
* @param mixed $key Key * @param mixed $key Key
* @param string $value Value * @param string $value Value
*/ */
public function set($key, $value = null) { public function set($key, $value = null)
if (is_array($key) || is_object($key)) { {
if (\is_array($key) || \is_object($key)) {
foreach ($key as $k => $v) { foreach ($key as $k => $v) {
$this->vars[$k] = $v; $this->vars[$k] = $v;
} }
} } else {
else {
$this->vars[$key] = $value; $this->vars[$key] = $value;
} }
} }
@ -82,9 +88,11 @@ class View {
* Checks if a template variable is set. * Checks if a template variable is set.
* *
* @param string $key Key * @param string $key Key
* @return boolean If key exists *
* @return bool If key exists
*/ */
public function has($key) { public function has($key)
{
return isset($this->vars[$key]); return isset($this->vars[$key]);
} }
@ -93,11 +101,11 @@ class View {
* *
* @param string $key Key * @param string $key Key
*/ */
public function clear($key = null) { public function clear($key = null)
if (is_null($key)) { {
$this->vars = array(); if (null === $key) {
} $this->vars = [];
else { } else {
unset($this->vars[$key]); unset($this->vars[$key]);
} }
} }
@ -107,16 +115,18 @@ class View {
* *
* @param string $file Template file * @param string $file Template file
* @param array $data Template data * @param array $data Template data
*
* @throws \Exception If template not found * @throws \Exception If template not found
*/ */
public function render($file, $data = null) { public function render($file, $data = null)
{
$this->template = $this->getTemplate($file); $this->template = $this->getTemplate($file);
if (!file_exists($this->template)) { if (!file_exists($this->template)) {
throw new \Exception("Template file not found: {$this->template}."); throw new \Exception("Template file not found: {$this->template}.");
} }
if (is_array($data)) { if (\is_array($data)) {
$this->vars = array_merge($this->vars, $data); $this->vars = array_merge($this->vars, $data);
} }
@ -130,24 +140,27 @@ class View {
* *
* @param string $file Template file * @param string $file Template file
* @param array $data Template data * @param array $data Template data
*
* @return string Output of template * @return string Output of template
*/ */
public function fetch($file, $data = null) { public function fetch($file, $data = null)
{
ob_start(); ob_start();
$this->render($file, $data); $this->render($file, $data);
$output = ob_get_clean();
return $output; return ob_get_clean();
} }
/** /**
* Checks if a template file exists. * Checks if a template file exists.
* *
* @param string $file Template file * @param string $file Template file
*
* @return bool Template file exists * @return bool Template file exists
*/ */
public function exists($file) { public function exists($file)
{
return file_exists($this->getTemplate($file)); return file_exists($this->getTemplate($file));
} }
@ -155,16 +168,18 @@ class View {
* Gets the full path to a template file. * Gets the full path to a template file.
* *
* @param string $file Template file * @param string $file Template file
*
* @return string Template file location * @return string Template file location
*/ */
public function getTemplate($file) { public function getTemplate($file)
{
$ext = $this->extension; $ext = $this->extension;
if (!empty($ext) && (substr($file, -1 * strlen($ext)) != $ext)) { if (!empty($ext) && (substr($file, -1 * \strlen($ext)) != $ext)) {
$file .= $ext; $file .= $ext;
} }
if ((substr($file, 0, 1) == '/')) { if (('/' == substr($file, 0, 1))) {
return $file; return $file;
} }
@ -175,10 +190,11 @@ class View {
* Displays escaped output. * Displays escaped output.
* *
* @param string $str String to escape * @param string $str String to escape
*
* @return string Escaped string * @return string Escaped string
*/ */
public function e($str) { public function e($str)
{
echo htmlentities($str); echo htmlentities($str);
} }
} }

@ -1,4 +1,6 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
@ -8,28 +10,34 @@
namespace flight\util; namespace flight\util;
use ArrayAccess;
use function count;
use Countable;
use Iterator;
use JsonSerializable;
if (!interface_exists('JsonSerializable')) { if (!interface_exists('JsonSerializable')) {
require_once dirname(__FILE__) . '/LegacyJsonSerializable.php'; require_once __DIR__ . '/LegacyJsonSerializable.php';
} }
/** /**
* The Collection class allows you to access a set of data * The Collection class allows you to access a set of data
* using both array and object notation. * using both array and object notation.
*/ */
class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializable { final class Collection implements ArrayAccess, Iterator, Countable, JsonSerializable
{
/** /**
* Collection data. * Collection data.
*
* @var array
*/ */
private $data; private array $data;
/** /**
* Constructor. * Constructor.
* *
* @param array $data Initial data * @param array $data Initial data
*/ */
public function __construct(array $data = array()) { public function __construct(array $data = [])
{
$this->data = $data; $this->data = $data;
} }
@ -37,10 +45,12 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* Gets an item. * Gets an item.
* *
* @param string $key Key * @param string $key Key
*
* @return mixed Value * @return mixed Value
*/ */
public function __get($key) { public function __get(string $key)
return isset($this->data[$key]) ? $this->data[$key] : null; {
return $this->data[$key] ?? null;
} }
/** /**
@ -49,7 +59,8 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* @param string $key Key * @param string $key Key
* @param mixed $value Value * @param mixed $value Value
*/ */
public function __set($key, $value) { public function __set(string $key, $value): void
{
$this->data[$key] = $value; $this->data[$key] = $value;
} }
@ -57,9 +68,11 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* Checks if an item exists. * Checks if an item exists.
* *
* @param string $key Key * @param string $key Key
*
* @return bool Item status * @return bool Item status
*/ */
public function __isset($key) { public function __isset(string $key): bool
{
return isset($this->data[$key]); return isset($this->data[$key]);
} }
@ -68,7 +81,8 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @param string $key Key * @param string $key Key
*/ */
public function __unset($key) { public function __unset(string $key): void
{
unset($this->data[$key]); unset($this->data[$key]);
} }
@ -76,10 +90,13 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* Gets an item at the offset. * Gets an item at the offset.
* *
* @param string $offset Offset * @param string $offset Offset
*
* @return mixed Value * @return mixed Value
*/ */
public function offsetGet($offset) { #[\ReturnTypeWillChange]
return isset($this->data[$offset]) ? $this->data[$offset] : null; public function offsetGet($offset)
{
return $this->data[$offset] ?? null;
} }
/** /**
@ -88,11 +105,12 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* @param string $offset Offset * @param string $offset Offset
* @param mixed $value Value * @param mixed $value Value
*/ */
public function offsetSet($offset, $value) { #[\ReturnTypeWillChange]
if (is_null($offset)) { public function offsetSet($offset, $value)
{
if (null === $offset) {
$this->data[] = $value; $this->data[] = $value;
} } else {
else {
$this->data[$offset] = $value; $this->data[$offset] = $value;
} }
} }
@ -101,9 +119,11 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* Checks if an item exists at the offset. * Checks if an item exists at the offset.
* *
* @param string $offset Offset * @param string $offset Offset
*
* @return bool Item status * @return bool Item status
*/ */
public function offsetExists($offset) { public function offsetExists($offset): bool
{
return isset($this->data[$offset]); return isset($this->data[$offset]);
} }
@ -112,14 +132,16 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @param string $offset Offset * @param string $offset Offset
*/ */
public function offsetUnset($offset) { public function offsetUnset($offset): void
{
unset($this->data[$offset]); unset($this->data[$offset]);
} }
/** /**
* Resets the collection. * Resets the collection.
*/ */
public function rewind() { public function rewind(): void
{
reset($this->data); reset($this->data);
} }
@ -128,7 +150,9 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return mixed Value * @return mixed Value
*/ */
public function current() { #[\ReturnTypeWillChange]
public function current()
{
return current($this->data); return current($this->data);
} }
@ -137,7 +161,9 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return mixed Value * @return mixed Value
*/ */
public function key() { #[\ReturnTypeWillChange]
public function key()
{
return key($this->data); return key($this->data);
} }
@ -146,6 +172,7 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return mixed Value * @return mixed Value
*/ */
#[\ReturnTypeWillChange]
public function next() public function next()
{ {
return next($this->data); return next($this->data);
@ -156,10 +183,11 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return bool Key status * @return bool Key status
*/ */
public function valid() public function valid(): bool
{ {
$key = key($this->data); $key = key($this->data);
return ($key !== NULL && $key !== FALSE);
return null !== $key && false !== $key;
} }
/** /**
@ -167,8 +195,9 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return int Collection size * @return int Collection size
*/ */
public function count() { public function count(): int
return sizeof($this->data); {
return \count($this->data);
} }
/** /**
@ -176,7 +205,8 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return array Collection keys * @return array Collection keys
*/ */
public function keys() { public function keys(): array
{
return array_keys($this->data); return array_keys($this->data);
} }
@ -185,7 +215,8 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @return array Collection data * @return array Collection data
*/ */
public function getData() { public function getData(): array
{
return $this->data; return $this->data;
} }
@ -194,23 +225,26 @@ class Collection implements \ArrayAccess, \Iterator, \Countable, \JsonSerializab
* *
* @param array $data New collection data * @param array $data New collection data
*/ */
public function setData(array $data) { public function setData(array $data): void
{
$this->data = $data; $this->data = $data;
} }
/** /**
* Gets the collection data which can be serialized to JSON * Gets the collection data which can be serialized to JSON.
* *
* @return array Collection data which can be serialized by <b>json_encode</b> * @return array Collection data which can be serialized by <b>json_encode</b>
*/ */
public function jsonSerialize() { public function jsonSerialize(): array
{
return $this->data; return $this->data;
} }
/** /**
* Removes all items from the collection. * Removes all items from the collection.
*/ */
public function clear() { public function clear(): void
$this->data = array(); {
$this->data = [];
} }
} }

@ -1,11 +1,13 @@
<?php <?php
declare(strict_types=1);
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
* @copyright Copyright (c) 2011, Mike Cao <mike@mikecao.com> * @copyright Copyright (c) 2011, Mike Cao <mike@mikecao.com>
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
interface LegacyJsonSerializable
interface JsonSerializable { {
public function jsonSerialize(); public function jsonSerialize();
} }

@ -1,4 +1,5 @@
<?php <?php
require 'flight/Flight.php'; require 'flight/Flight.php';
Flight::route('/', function () { Flight::route('/', function () {

@ -6,36 +6,41 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class AutoloadTest extends PHPUnit_Framework_TestCase class AutoloadTest extends PHPUnit\Framework\TestCase
{ {
/** /**
* @var \flight\Engine * @var Engine
*/ */
private $app; private $app;
function setUp() { protected function setUp(): void
$this->app = new \flight\Engine(); {
$this->app = new Engine();
$this->app->path(__DIR__ . '/classes'); $this->app->path(__DIR__ . '/classes');
} }
// Autoload a class // Autoload a class
function testAutoload(){ public function testAutoload()
{
$this->app->register('user', 'User'); $this->app->register('user', 'User');
$loaders = spl_autoload_functions(); $loaders = spl_autoload_functions();
$user = $this->app->user(); $user = $this->app->user();
$this->assertTrue(sizeof($loaders) > 0); self::assertTrue(count($loaders) > 0);
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
} }
// Check autoload failure // Check autoload failure
function testMissingClass(){ public function testMissingClass()
{
$test = null; $test = null;
$this->app->register('test', 'NonExistentClass'); $this->app->register('test', 'NonExistentClass');
@ -43,6 +48,6 @@ class AutoloadTest extends PHPUnit_Framework_TestCase
$test = $this->app->test(); $test = $this->app->test();
} }
$this->assertEquals(null, $test); self::assertNull($test);
} }
} }

@ -6,64 +6,72 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\core\Dispatcher;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/classes/Hello.php'; require_once __DIR__ . '/classes/Hello.php';
class DispatcherTest extends PHPUnit_Framework_TestCase class DispatcherTest extends PHPUnit\Framework\TestCase
{ {
/** /**
* @var \flight\core\Dispatcher * @var Dispatcher|null
*/ */
private $dispatcher; private Dispatcher $dispatcher;
function setUp(){ protected function setUp(): void
$this->dispatcher = new \flight\core\Dispatcher(); {
$this->dispatcher = new Dispatcher();
} }
// Map a closure // Map a closure
function testClosureMapping(){ public function testClosureMapping()
{
$this->dispatcher->set('map1', function () { $this->dispatcher->set('map1', function () {
return 'hello'; return 'hello';
}); });
$result = $this->dispatcher->run('map1'); $result = $this->dispatcher->run('map1');
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a function // Map a function
function testFunctionMapping(){ public function testFunctionMapping()
{
$this->dispatcher->set('map2', function () { $this->dispatcher->set('map2', function () {
return 'hello'; return 'hello';
}); });
$result = $this->dispatcher->run('map2'); $result = $this->dispatcher->run('map2');
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a class method // Map a class method
function testClassMethodMapping(){ public function testClassMethodMapping()
{
$h = new Hello(); $h = new Hello();
$this->dispatcher->set('map3', array($h, 'sayHi')); $this->dispatcher->set('map3', [$h, 'sayHi']);
$result = $this->dispatcher->run('map3'); $result = $this->dispatcher->run('map3');
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a static class method // Map a static class method
function testStaticClassMethodMapping(){ public function testStaticClassMethodMapping()
$this->dispatcher->set('map4', array('Hello', 'sayBye')); {
$this->dispatcher->set('map4', ['Hello', 'sayBye']);
$result = $this->dispatcher->run('map4'); $result = $this->dispatcher->run('map4');
$this->assertEquals('goodbye', $result); self::assertEquals('goodbye', $result);
} }
// Run before and after filters // Run before and after filters
function testBeforeAndAfter() { public function testBeforeAndAfter()
{
$this->dispatcher->set('hello', function ($name) { $this->dispatcher->set('hello', function ($name) {
return "Hello, $name!"; return "Hello, $name!";
}); });
@ -75,18 +83,19 @@ class DispatcherTest extends PHPUnit_Framework_TestCase
$this->dispatcher->hook('hello', 'after', function (&$params, &$output) { $this->dispatcher->hook('hello', 'after', function (&$params, &$output) {
// Manipulate the output // Manipulate the output
$output .= " Have a nice day!"; $output .= ' Have a nice day!';
}); });
$result = $this->dispatcher->run('hello', array('Bob')); $result = $this->dispatcher->run('hello', ['Bob']);
$this->assertEquals('Hello, Fred! Have a nice day!', $result); self::assertEquals('Hello, Fred! Have a nice day!', $result);
} }
// Test an invalid callback // Test an invalid callback
function testInvalidCallback() { public function testInvalidCallback()
$this->setExpectedException('Exception', 'Invalid callback specified.'); {
$this->expectException(Exception::class);
$this->dispatcher->execute(array('NonExistentClass', 'nonExistentMethod')); $this->dispatcher->execute(['NonExistentClass', 'nonExistentMethod']);
} }
} }

@ -6,22 +6,26 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class FilterTest extends PHPUnit_Framework_TestCase class FilterTest extends PHPUnit\Framework\TestCase
{ {
/** /**
* @var \flight\Engine * @var Engine
*/ */
private $app; private $app;
function setUp() { protected function setUp(): void
$this->app = new \flight\Engine(); {
$this->app = new Engine();
} }
// Run before and after filters // Run before and after filters
function testBeforeAndAfter() { public function testBeforeAndAfter()
{
$this->app->map('hello', function ($name) { $this->app->map('hello', function ($name) {
return "Hello, $name!"; return "Hello, $name!";
}); });
@ -33,7 +37,7 @@ class FilterTest extends PHPUnit_Framework_TestCase
$this->app->after('hello', function (&$params, &$output) { $this->app->after('hello', function (&$params, &$output) {
// Manipulate the output // Manipulate the output
$output .= " Have a nice day!"; $output .= ' Have a nice day!';
}); });
$result = $this->app->hello('Bob'); $result = $this->app->hello('Bob');
@ -42,7 +46,8 @@ class FilterTest extends PHPUnit_Framework_TestCase
} }
// Break out of a filter chain by returning false // Break out of a filter chain by returning false
function testFilterChaining() { public function testFilterChaining()
{
$this->app->map('bye', function ($name) { $this->app->map('bye', function ($name) {
return "Bye, $name!"; return "Bye, $name!";
}); });
@ -52,6 +57,7 @@ class FilterTest extends PHPUnit_Framework_TestCase
}); });
$this->app->before('bye', function (&$params, &$output) { $this->app->before('bye', function (&$params, &$output) {
$params[0] = 'Fred'; $params[0] = 'Fred';
return false; return false;
}); });
$this->app->before('bye', function (&$params, &$output) { $this->app->before('bye', function (&$params, &$output) {

@ -1,35 +1,43 @@
<?php <?php
use flight\net\Request;
use flight\net\Response;
use flight\net\Router;
use flight\template\View;
/** /**
* Flight: An extensible micro-framework. * Flight: An extensible micro-framework.
* *
* @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com> * @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com>
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/Flight.php'; require_once __DIR__ . '/../flight/Flight.php';
class FlightTest extends PHPUnit_Framework_TestCase class FlightTest extends PHPUnit\Framework\TestCase
{
protected function setUp(): void
{ {
function setUp() {
Flight::init(); Flight::init();
} }
// Checks that default components are loaded // Checks that default components are loaded
function testDefaultComponents(){ public function testDefaultComponents()
{
$request = Flight::request(); $request = Flight::request();
$response = Flight::response(); $response = Flight::response();
$router = Flight::router(); $router = Flight::router();
$view = Flight::view(); $view = Flight::view();
$this->assertEquals('flight\net\Request', get_class($request)); $this->assertEquals(Request::class, get_class($request));
$this->assertEquals('flight\net\Response', get_class($response)); $this->assertEquals(Response::class, get_class($response));
$this->assertEquals('flight\net\Router', get_class($router)); $this->assertEquals(Router::class, get_class($router));
$this->assertEquals('flight\template\View', get_class($view)); $this->assertEquals(View::class, get_class($view));
} }
// Test get/set of variables // Test get/set of variables
function testGetAndSet(){ public function testGetAndSet()
{
Flight::set('a', 1); Flight::set('a', 1);
$var = Flight::get('a'); $var = Flight::get('a');
@ -38,19 +46,20 @@ class FlightTest extends PHPUnit_Framework_TestCase
Flight::clear(); Flight::clear();
$vars = Flight::get(); $vars = Flight::get();
$this->assertEquals(0, count($vars)); $this->assertCount(0, $vars);
Flight::set('a', 1); Flight::set('a', 1);
Flight::set('b', 2); Flight::set('b', 2);
$vars = Flight::get(); $vars = Flight::get();
$this->assertEquals(2, count($vars)); $this->assertCount(2, $vars);
$this->assertEquals(1, $vars['a']); $this->assertEquals(1, $vars['a']);
$this->assertEquals(2, $vars['b']); $this->assertEquals(2, $vars['b']);
} }
// Register a class // Register a class
function testRegister(){ public function testRegister()
{
Flight::path(__DIR__ . '/classes'); Flight::path(__DIR__ . '/classes');
Flight::register('user', 'User'); Flight::register('user', 'User');
@ -58,25 +67,28 @@ class FlightTest extends PHPUnit_Framework_TestCase
$loaders = spl_autoload_functions(); $loaders = spl_autoload_functions();
$this->assertTrue(sizeof($loaders) > 0); self::assertTrue(count($loaders) > 0);
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
} }
// Map a function // Map a function
function testMap(){ public function testMap()
{
Flight::map('map1', function () { Flight::map('map1', function () {
return 'hello'; return 'hello';
}); });
$result = Flight::map1(); $result = Flight::map1();
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Unmapped method // Unmapped method
function testUnmapped() { public function testUnmapped()
$this->setExpectedException('Exception', 'doesNotExist must be a mapped method.'); {
$this->expectException(Exception::class);
$this->expectExceptionMessage('doesNotExist must be a mapped method.');
Flight::doesNotExist(); Flight::doesNotExist();
} }

@ -6,109 +6,116 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\core\Loader;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/classes/User.php'; require_once __DIR__ . '/classes/User.php';
require_once __DIR__ . '/classes/Factory.php'; require_once __DIR__ . '/classes/Factory.php';
class LoaderTest extends PHPUnit_Framework_TestCase class LoaderTest extends PHPUnit\Framework\TestCase
{ {
/** private Loader $loader;
* @var \flight\core\Loader
*/
private $loader;
function setUp(){ protected function setUp(): void
$this->loader = new \flight\core\Loader(); {
$this->loader = new Loader();
$this->loader->autoload(true, __DIR__ . '/classes'); $this->loader->autoload(true, __DIR__ . '/classes');
} }
// Autoload a class // Autoload a class
function testAutoload(){ public function testAutoload()
{
$this->loader->register('tests', 'User'); $this->loader->register('tests', 'User');
$test = $this->loader->load('tests'); $test = $this->loader->load('tests');
$this->assertTrue(is_object($test)); self::assertIsObject($test);
$this->assertEquals('User', get_class($test)); self::assertInstanceOf(User::class, $test);
} }
// Register a class // Register a class
function testRegister(){ public function testRegister()
{
$this->loader->register('a', 'User'); $this->loader->register('a', 'User');
$user = $this->loader->load('a'); $user = $this->loader->load('a');
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('', $user->name); self::assertEquals('', $user->name);
} }
// Register a class with constructor parameters // Register a class with constructor parameters
function testRegisterWithConstructor(){ public function testRegisterWithConstructor()
$this->loader->register('b', 'User', array('Bob')); {
$this->loader->register('b', 'User', ['Bob']);
$user = $this->loader->load('b'); $user = $this->loader->load('b');
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('Bob', $user->name); self::assertEquals('Bob', $user->name);
} }
// Register a class with initialization // Register a class with initialization
function testRegisterWithInitialization(){ public function testRegisterWithInitialization()
$this->loader->register('c', 'User', array('Bob'), function($user){ {
$this->loader->register('c', 'User', ['Bob'], function ($user) {
$user->name = 'Fred'; $user->name = 'Fred';
}); });
$user = $this->loader->load('c'); $user = $this->loader->load('c');
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('Fred', $user->name); self::assertEquals('Fred', $user->name);
} }
// Get a non-shared instance of a class // Get a non-shared instance of a class
function testSharedInstance() { public function testSharedInstance()
{
$this->loader->register('d', 'User'); $this->loader->register('d', 'User');
$user1 = $this->loader->load('d'); $user1 = $this->loader->load('d');
$user2 = $this->loader->load('d'); $user2 = $this->loader->load('d');
$user3 = $this->loader->load('d', false); $user3 = $this->loader->load('d', false);
$this->assertTrue($user1 === $user2); self::assertSame($user1, $user2);
$this->assertTrue($user1 !== $user3); self::assertNotSame($user1, $user3);
} }
// Gets an object from a factory method // Gets an object from a factory method
function testRegisterUsingCallable(){ public function testRegisterUsingCallable()
$this->loader->register('e', array('Factory','create')); {
$this->loader->register('e', ['Factory', 'create']);
$obj = $this->loader->load('e'); $obj = $this->loader->load('e');
$this->assertTrue(is_object($obj)); self::assertIsObject($obj);
$this->assertEquals('Factory', get_class($obj)); self::assertInstanceOf(Factory::class, $obj);
$obj2 = $this->loader->load('e'); $obj2 = $this->loader->load('e');
$this->assertTrue(is_object($obj2)); self::assertIsObject($obj2);
$this->assertEquals('Factory', get_class($obj2)); self::assertInstanceOf(Factory::class, $obj2);
$this->assertTrue($obj === $obj2); self::assertSame($obj, $obj2);
$obj3 = $this->loader->load('e', false); $obj3 = $this->loader->load('e', false);
$this->assertTrue(is_object($obj3)); self::assertIsObject($obj3);
$this->assertEquals('Factory', get_class($obj3)); self::assertInstanceOf(Factory::class, $obj3);
$this->assertTrue($obj !== $obj3); self::assertNotSame($obj, $obj3);
} }
// Gets an object from a callback function // Gets an object from a callback function
function testRegisterUsingCallback(){ public function testRegisterUsingCallback()
{
$this->loader->register('f', function () { $this->loader->register('f', function () {
return Factory::create(); return Factory::create();
}); });
$obj = $this->loader->load('f'); $obj = $this->loader->load('f');
$this->assertTrue(is_object($obj)); self::assertIsObject($obj);
$this->assertEquals('Factory', get_class($obj)); self::assertInstanceOf(Factory::class, $obj);
} }
} }

@ -6,66 +6,72 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
require_once __DIR__ . '/classes/Hello.php'; require_once __DIR__ . '/classes/Hello.php';
class MapTest extends PHPUnit_Framework_TestCase class MapTest extends PHPUnit\Framework\TestCase
{ {
/** private Engine $app;
* @var \flight\Engine
*/
private $app;
function setUp() { protected function setUp(): void
$this->app = new \flight\Engine(); {
$this->app = new Engine();
} }
// Map a closure // Map a closure
function testClosureMapping(){ public function testClosureMapping()
{
$this->app->map('map1', function () { $this->app->map('map1', function () {
return 'hello'; return 'hello';
}); });
$result = $this->app->map1(); $result = $this->app->map1();
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a function // Map a function
function testFunctionMapping(){ public function testFunctionMapping()
{
$this->app->map('map2', function () { $this->app->map('map2', function () {
return 'hello'; return 'hello';
}); });
$result = $this->app->map2(); $result = $this->app->map2();
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a class method // Map a class method
function testClassMethodMapping(){ public function testClassMethodMapping()
{
$h = new Hello(); $h = new Hello();
$this->app->map('map3', array($h, 'sayHi')); $this->app->map('map3', [$h, 'sayHi']);
$result = $this->app->map3(); $result = $this->app->map3();
$this->assertEquals('hello', $result); self::assertEquals('hello', $result);
} }
// Map a static class method // Map a static class method
function testStaticClassMethodMapping(){ public function testStaticClassMethodMapping()
$this->app->map('map4', array('Hello', 'sayBye')); {
$this->app->map('map4', ['Hello', 'sayBye']);
$result = $this->app->map4(); $result = $this->app->map4();
$this->assertEquals('goodbye', $result); self::assertEquals('goodbye', $result);
} }
// Unmapped method // Unmapped method
function testUnmapped() { public function testUnmapped()
$this->setExpectedException('Exception', 'doesNotExist must be a mapped method.'); {
$this->expectException(Exception::class);
$this->expectExceptionMessage('doesNotExist must be a mapped method.');
$this->app->doesNotExist(); $this->app->doesNotExist();
} }

@ -6,72 +6,77 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class RedirectTest extends PHPUnit_Framework_TestCase class RedirectTest extends PHPUnit\Framework\TestCase
{ {
/** private Engine $app;
* @var \flight\Engine
*/
private $app;
function getBaseUrl($base, $url){ protected function setUp(): void
if ($base != '/' && strpos($url, '://') === false) { {
$url = preg_replace('#/+#', '/', $base.'/'.$url); $_SERVER['SCRIPT_NAME'] = '/subdir/index.php';
}
return $url; $this->app = new Engine();
$this->app->set('flight.base_url', '/testdir');
} }
function setUp() { public function getBaseUrl($base, $url)
$_SERVER['SCRIPT_NAME'] = '/subdir/index.php'; {
if ('/' !== $base && false === strpos($url, '://')) {
$url = preg_replace('#/+#', '/', $base . '/' . $url);
}
$this->app = new \flight\Engine(); return $url;
$this->app->set('flight.base_url', '/testdir');
} }
// The base should be the subdirectory // The base should be the subdirectory
function testBase(){ public function testBase()
{
$base = $this->app->request()->base; $base = $this->app->request()->base;
$this->assertEquals('/subdir', $base); self::assertEquals('/subdir', $base);
} }
// Absolute URLs should include the base // Absolute URLs should include the base
function testAbsoluteUrl(){ public function testAbsoluteUrl()
{
$url = '/login'; $url = '/login';
$base = $this->app->request()->base; $base = $this->app->request()->base;
$this->assertEquals('/subdir/login', $this->getBaseUrl($base, $url)); self::assertEquals('/subdir/login', $this->getBaseUrl($base, $url));
} }
// Relative URLs should include the base // Relative URLs should include the base
function testRelativeUrl(){ public function testRelativeUrl()
{
$url = 'login'; $url = 'login';
$base = $this->app->request()->base; $base = $this->app->request()->base;
$this->assertEquals('/subdir/login', $this->getBaseUrl($base, $url)); self::assertEquals('/subdir/login', $this->getBaseUrl($base, $url));
} }
// External URLs should ignore the base // External URLs should ignore the base
function testHttpUrl(){ public function testHttpUrl()
{
$url = 'http://www.yahoo.com'; $url = 'http://www.yahoo.com';
$base = $this->app->request()->base; $base = $this->app->request()->base;
$this->assertEquals('http://www.yahoo.com', $this->getBaseUrl($base, $url)); self::assertEquals('http://www.yahoo.com', $this->getBaseUrl($base, $url));
} }
// Configuration should override derived value // Configuration should override derived value
function testBaseOverride(){ public function testBaseOverride()
{
$url = 'login'; $url = 'login';
if ($this->app->get('flight.base_url') !== null) { if (null !== $this->app->get('flight.base_url')) {
$base = $this->app->get('flight.base_url'); $base = $this->app->get('flight.base_url');
} } else {
else {
$base = $this->app->request()->base; $base = $this->app->request()->base;
} }
$this->assertEquals('/testdir/login', $this->getBaseUrl($base, $url)); self::assertEquals('/testdir/login', $this->getBaseUrl($base, $url));
} }
} }

@ -6,75 +6,80 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
require_once __DIR__ . '/classes/User.php'; require_once __DIR__ . '/classes/User.php';
class RegisterTest extends PHPUnit_Framework_TestCase class RegisterTest extends PHPUnit\Framework\TestCase
{ {
/** private Engine $app;
* @var \flight\Engine
*/
private $app;
function setUp() { protected function setUp(): void
$this->app = new \flight\Engine(); {
$this->app = new Engine();
} }
// Register a class // Register a class
function testRegister(){ public function testRegister()
{
$this->app->register('reg1', 'User'); $this->app->register('reg1', 'User');
$user = $this->app->reg1(); $user = $this->app->reg1();
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('', $user->name); self::assertEquals('', $user->name);
} }
// Register a class with constructor parameters // Register a class with constructor parameters
function testRegisterWithConstructor(){ public function testRegisterWithConstructor()
$this->app->register('reg2', 'User', array('Bob')); {
$this->app->register('reg2', 'User', ['Bob']);
$user = $this->app->reg2(); $user = $this->app->reg2();
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('Bob', $user->name); self::assertEquals('Bob', $user->name);
} }
// Register a class with initialization // Register a class with initialization
function testRegisterWithInitialization(){ public function testRegisterWithInitialization()
$this->app->register('reg3', 'User', array('Bob'), function($user){ {
$this->app->register('reg3', 'User', ['Bob'], function ($user) {
$user->name = 'Fred'; $user->name = 'Fred';
}); });
$user = $this->app->reg3(); $user = $this->app->reg3();
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->assertEquals('User', get_class($user)); self::assertInstanceOf(User::class, $user);
$this->assertEquals('Fred', $user->name); self::assertEquals('Fred', $user->name);
} }
// Get a non-shared instance of a class // Get a non-shared instance of a class
function testSharedInstance() { public function testSharedInstance()
{
$this->app->register('reg4', 'User'); $this->app->register('reg4', 'User');
$user1 = $this->app->reg4(); $user1 = $this->app->reg4();
$user2 = $this->app->reg4(); $user2 = $this->app->reg4();
$user3 = $this->app->reg4(false); $user3 = $this->app->reg4(false);
$this->assertTrue($user1 === $user2); self::assertSame($user1, $user2);
$this->assertTrue($user1 !== $user3); self::assertNotSame($user1, $user3);
} }
// Map method takes precedence over register // Map method takes precedence over register
function testMapOverridesRegister(){ public function testMapOverridesRegister()
{
$this->app->register('reg5', 'User'); $this->app->register('reg5', 'User');
$user = $this->app->reg5(); $user = $this->app->reg5();
$this->assertTrue(is_object($user)); self::assertIsObject($user);
$this->app->map('reg5', function () { $this->app->map('reg5', function () {
return 123; return 123;
@ -82,6 +87,6 @@ class RegisterTest extends PHPUnit_Framework_TestCase
$user = $this->app->reg5(); $user = $this->app->reg5();
$this->assertEquals(123, $user); self::assertEquals(123, $user);
} }
} }

@ -6,31 +6,33 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\Engine;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/Flight.php'; require_once __DIR__ . '/../flight/Flight.php';
class RenderTest extends PHPUnit_Framework_TestCase class RenderTest extends PHPUnit\Framework\TestCase
{ {
/** private Engine $app;
* @var \flight\Engine
*/
private $app;
function setUp() { protected function setUp(): void
$this->app = new \flight\Engine(); {
$this->app = new Engine();
$this->app->set('flight.views.path', __DIR__ . '/views'); $this->app->set('flight.views.path', __DIR__ . '/views');
} }
// Render a view // Render a view
function testRenderView(){ public function testRenderView()
$this->app->render('hello', array('name' => 'Bob')); {
$this->app->render('hello', ['name' => 'Bob']);
$this->expectOutputString('Hello, Bob!'); $this->expectOutputString('Hello, Bob!');
} }
// Renders a view into a layout // Renders a view into a layout
function testRenderLayout(){ public function testRenderLayout()
$this->app->render('hello', array('name' => 'Bob'), 'content'); {
$this->app->render('hello', ['name' => 'Bob'], 'content');
$this->app->render('layouts/layout'); $this->app->render('layouts/layout');
$this->expectOutputString('<html>Hello, Bob!</html>'); $this->expectOutputString('<html>Hello, Bob!</html>');

@ -6,17 +6,17 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\net\Request;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class RequestTest extends PHPUnit_Framework_TestCase class RequestTest extends PHPUnit\Framework\TestCase
{ {
/** private Request $request;
* @var \flight\net\Request
*/
private $request;
function setUp() { protected function setUp(): void
{
$_SERVER['REQUEST_URI'] = '/'; $_SERVER['REQUEST_URI'] = '/';
$_SERVER['SCRIPT_NAME'] = '/index.php'; $_SERVER['SCRIPT_NAME'] = '/index.php';
$_SERVER['REQUEST_METHOD'] = 'GET'; $_SERVER['REQUEST_METHOD'] = 'GET';
@ -25,47 +25,52 @@ class RequestTest extends PHPUnit_Framework_TestCase
$_SERVER['HTTP_X_FORWARDED_FOR'] = '32.32.32.32'; $_SERVER['HTTP_X_FORWARDED_FOR'] = '32.32.32.32';
$_SERVER['HTTP_HOST'] = 'example.com'; $_SERVER['HTTP_HOST'] = 'example.com';
$this->request = new \flight\net\Request(); $this->request = new Request();
} }
function testDefaults() { public function testDefaults()
$this->assertEquals('/', $this->request->url); {
$this->assertEquals('/', $this->request->base); self::assertEquals('/', $this->request->url);
$this->assertEquals('GET', $this->request->method); self::assertEquals('/', $this->request->base);
$this->assertEquals('', $this->request->referrer); self::assertEquals('GET', $this->request->method);
$this->assertEquals(true, $this->request->ajax); self::assertEquals('', $this->request->referrer);
$this->assertEquals('http', $this->request->scheme); self::assertTrue($this->request->ajax);
$this->assertEquals('', $this->request->type); self::assertEquals('http', $this->request->scheme);
$this->assertEquals(0, $this->request->length); self::assertEquals('', $this->request->type);
$this->assertEquals(false, $this->request->secure); self::assertEquals(0, $this->request->length);
$this->assertEquals('', $this->request->accept); self::assertFalse($this->request->secure);
$this->assertEquals('example.com', $this->request->host); self::assertEquals('', $this->request->accept);
self::assertEquals('example.com', $this->request->host);
} }
function testIpAddress() { public function testIpAddress()
$this->assertEquals('8.8.8.8', $this->request->ip); {
$this->assertEquals('32.32.32.32', $this->request->proxy_ip); self::assertEquals('8.8.8.8', $this->request->ip);
self::assertEquals('32.32.32.32', $this->request->proxy_ip);
} }
function testSubdirectory() { public function testSubdirectory()
{
$_SERVER['SCRIPT_NAME'] = '/subdir/index.php'; $_SERVER['SCRIPT_NAME'] = '/subdir/index.php';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('/subdir', $request->base); self::assertEquals('/subdir', $request->base);
} }
function testQueryParameters() { public function testQueryParameters()
{
$_SERVER['REQUEST_URI'] = '/page?id=1&name=bob'; $_SERVER['REQUEST_URI'] = '/page?id=1&name=bob';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('/page?id=1&name=bob', $request->url); self::assertEquals('/page?id=1&name=bob', $request->url);
$this->assertEquals(1, $request->query->id); self::assertEquals(1, $request->query->id);
$this->assertEquals('bob', $request->query->name); self::assertEquals('bob', $request->query->name);
} }
function testCollections() { public function testCollections()
{
$_SERVER['REQUEST_URI'] = '/page?id=1'; $_SERVER['REQUEST_URI'] = '/page?id=1';
$_GET['q'] = 1; $_GET['q'] = 1;
@ -73,58 +78,61 @@ class RequestTest extends PHPUnit_Framework_TestCase
$_COOKIE['q'] = 1; $_COOKIE['q'] = 1;
$_FILES['q'] = 1; $_FILES['q'] = 1;
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals(1, $request->query->q); self::assertEquals(1, $request->query->q);
$this->assertEquals(1, $request->query->id); self::assertEquals(1, $request->query->id);
$this->assertEquals(1, $request->data->q); self::assertEquals(1, $request->data->q);
$this->assertEquals(1, $request->cookies->q); self::assertEquals(1, $request->cookies->q);
$this->assertEquals(1, $request->files->q); self::assertEquals(1, $request->files->q);
} }
function testMethodOverrideWithHeader() { public function testMethodOverrideWithHeader()
{
$_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'] = 'PUT'; $_SERVER['HTTP_X_HTTP_METHOD_OVERRIDE'] = 'PUT';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('PUT', $request->method); self::assertEquals('PUT', $request->method);
} }
function testMethodOverrideWithPost() { public function testMethodOverrideWithPost()
{
$_REQUEST['_method'] = 'PUT'; $_REQUEST['_method'] = 'PUT';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('PUT', $request->method); self::assertEquals('PUT', $request->method);
} }
function testHttps() { public function testHttps()
{
$_SERVER['HTTPS'] = 'on'; $_SERVER['HTTPS'] = 'on';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('https', $request->scheme); self::assertEquals('https', $request->scheme);
$_SERVER['HTTPS'] = 'off'; $_SERVER['HTTPS'] = 'off';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('http', $request->scheme); self::assertEquals('http', $request->scheme);
$_SERVER['HTTP_X_FORWARDED_PROTO'] = 'https'; $_SERVER['HTTP_X_FORWARDED_PROTO'] = 'https';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('https', $request->scheme); self::assertEquals('https', $request->scheme);
$_SERVER['HTTP_X_FORWARDED_PROTO'] = 'http'; $_SERVER['HTTP_X_FORWARDED_PROTO'] = 'http';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('http', $request->scheme); self::assertEquals('http', $request->scheme);
$_SERVER['HTTP_FRONT_END_HTTPS'] = 'on'; $_SERVER['HTTP_FRONT_END_HTTPS'] = 'on';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('https', $request->scheme); self::assertEquals('https', $request->scheme);
$_SERVER['HTTP_FRONT_END_HTTPS'] = 'off'; $_SERVER['HTTP_FRONT_END_HTTPS'] = 'off';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('http', $request->scheme); self::assertEquals('http', $request->scheme);
$_SERVER['REQUEST_SCHEME'] = 'https'; $_SERVER['REQUEST_SCHEME'] = 'https';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('https', $request->scheme); self::assertEquals('https', $request->scheme);
$_SERVER['REQUEST_SCHEME'] = 'http'; $_SERVER['REQUEST_SCHEME'] = 'http';
$request = new \flight\net\Request(); $request = new Request();
$this->assertEquals('http', $request->scheme); self::assertEquals('http', $request->scheme);
} }
} }

@ -6,39 +6,37 @@
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
use flight\core\Dispatcher;
use flight\net\Request;
use flight\net\Router;
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class RouterTest extends PHPUnit_Framework_TestCase class RouterTest extends PHPUnit\Framework\TestCase
{ {
/** private Router $router;
* @var \flight\net\Router
*/
private $router;
/** private Request $request;
* @var \flight\net\Request
*/
private $request;
/** private Dispatcher $dispatcher;
* @var \flight\core\Dispatcher
*/
private $dispatcher;
function setUp(){ protected function setUp(): void
$this->router = new \flight\net\Router(); {
$this->request = new \flight\net\Request(); $this->router = new Router();
$this->dispatcher = new \flight\core\Dispatcher(); $this->request = new Request();
$this->dispatcher = new Dispatcher();
} }
// Simple output // Simple output
function ok(){ public function ok()
{
echo 'OK'; echo 'OK';
} }
// Checks if a route was matched with a given output // Checks if a route was matched with a given output
function check($str = '') { public function check($str = '')
{
/* /*
$route = $this->router->route($this->request); $route = $this->router->route($this->request);
@ -53,7 +51,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
$this->expectOutputString($str); $this->expectOutputString($str);
} }
function routeRequest() { public function routeRequest()
{
$dispatched = false; $dispatched = false;
while ($route = $this->router->route($this->request)) { while ($route = $this->router->route($this->request)) {
@ -70,7 +69,9 @@ class RouterTest extends PHPUnit_Framework_TestCase
$dispatched = true; $dispatched = true;
if (!$continue) break; if (!$continue) {
break;
}
$this->router->next(); $this->router->next();
@ -83,24 +84,27 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Default route // Default route
function testDefaultRoute(){ public function testDefaultRoute()
$this->router->map('/', array($this, 'ok')); {
$this->router->map('/', [$this, 'ok']);
$this->request->url = '/'; $this->request->url = '/';
$this->check('OK'); $this->check('OK');
} }
// Simple path // Simple path
function testPathRoute(){ public function testPathRoute()
$this->router->map('/path', array($this, 'ok')); {
$this->router->map('/path', [$this, 'ok']);
$this->request->url = '/path'; $this->request->url = '/path';
$this->check('OK'); $this->check('OK');
} }
// POST route // POST route
function testPostRoute(){ public function testPostRoute()
$this->router->map('POST /', array($this, 'ok')); {
$this->router->map('POST /', [$this, 'ok']);
$this->request->url = '/'; $this->request->url = '/';
$this->request->method = 'POST'; $this->request->method = 'POST';
@ -108,8 +112,9 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Either GET or POST route // Either GET or POST route
function testGetPostRoute(){ public function testGetPostRoute()
$this->router->map('GET|POST /', array($this, 'ok')); {
$this->router->map('GET|POST /', [$this, 'ok']);
$this->request->url = '/'; $this->request->url = '/';
$this->request->method = 'GET'; $this->request->method = 'GET';
@ -117,15 +122,17 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Test regular expression matching // Test regular expression matching
function testRegEx(){ public function testRegEx()
$this->router->map('/num/[0-9]+', array($this, 'ok')); {
$this->router->map('/num/[0-9]+', [$this, 'ok']);
$this->request->url = '/num/1234'; $this->request->url = '/num/1234';
$this->check('OK'); $this->check('OK');
} }
// Passing URL parameters // Passing URL parameters
function testUrlParameters(){ public function testUrlParameters()
{
$this->router->map('/user/@id', function ($id) { $this->router->map('/user/@id', function ($id) {
echo $id; echo $id;
}); });
@ -135,7 +142,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Passing URL parameters matched with regular expression // Passing URL parameters matched with regular expression
function testRegExParameters(){ public function testRegExParameters()
{
$this->router->map('/test/@name:[a-z]+', function ($name) { $this->router->map('/test/@name:[a-z]+', function ($name) {
echo $name; echo $name;
}); });
@ -145,7 +153,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Optional parameters // Optional parameters
function testOptionalParameters(){ public function testOptionalParameters()
{
$this->router->map('/blog(/@year(/@month(/@day)))', function ($year, $month, $day) { $this->router->map('/blog(/@year(/@month(/@day)))', function ($year, $month, $day) {
echo "$year,$month,$day"; echo "$year,$month,$day";
}); });
@ -155,7 +164,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Regex in optional parameters // Regex in optional parameters
function testRegexOptionalParameters(){ public function testRegexOptionalParameters()
{
$this->router->map('/@controller/@method(/@id:[0-9]+)', function ($controller, $method, $id) { $this->router->map('/@controller/@method(/@id:[0-9]+)', function ($controller, $method, $id) {
echo "$controller,$method,$id"; echo "$controller,$method,$id";
}); });
@ -165,7 +175,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Regex in optional parameters // Regex in optional parameters
function testRegexEmptyOptionalParameters(){ public function testRegexEmptyOptionalParameters()
{
$this->router->map('/@controller/@method(/@id:[0-9]+)', function ($controller, $method, $id) { $this->router->map('/@controller/@method(/@id:[0-9]+)', function ($controller, $method, $id) {
echo "$controller,$method,$id"; echo "$controller,$method,$id";
}); });
@ -175,22 +186,24 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Wildcard matching // Wildcard matching
function testWildcard(){ public function testWildcard()
$this->router->map('/account/*', array($this, 'ok')); {
$this->router->map('/account/*', [$this, 'ok']);
$this->request->url = '/account/123/abc/xyz'; $this->request->url = '/account/123/abc/xyz';
$this->check('OK'); $this->check('OK');
} }
// Check if route object was passed // Check if route object was passed
function testRouteObjectPassing(){ public function testRouteObjectPassing()
{
$this->router->map('/yes_route', function ($route) { $this->router->map('/yes_route', function ($route) {
$this->assertTrue(is_object($route)); $this->assertIsObject($route);
$this->assertTrue(is_array($route->methods)); $this->assertIsArray($route->methods);
$this->assertTrue(is_array($route->params)); $this->assertIsArray($route->params);
$this->assertEquals(sizeof($route->params), 0); $this->assertCount(0, $route->params);
$this->assertEquals($route->regex, null); $this->assertNull($route->regex);
$this->assertEquals($route->splat, ''); $this->assertEquals('', $route->splat);
$this->assertTrue($route->pass); $this->assertTrue($route->pass);
}, true); }, true);
$this->request->url = '/yes_route'; $this->request->url = '/yes_route';
@ -198,16 +211,17 @@ class RouterTest extends PHPUnit_Framework_TestCase
$this->check(); $this->check();
$this->router->map('/no_route', function ($route = null) { $this->router->map('/no_route', function ($route = null) {
$this->assertTrue(is_null($route)); $this->assertNull($route);
}, false); }, false);
$this->request->url = '/no_route'; $this->request->url = '/no_route';
$this->check(); $this->check();
} }
function testRouteWithParameters() { public function testRouteWithParameters()
{
$this->router->map('/@one/@two', function ($one, $two, $route) { $this->router->map('/@one/@two', function ($one, $two, $route) {
$this->assertEquals(sizeof($route->params), 2); $this->assertCount(2, $route->params);
$this->assertEquals($route->params['one'], $one); $this->assertEquals($route->params['one'], $one);
$this->assertEquals($route->params['two'], $two); $this->assertEquals($route->params['two'], $two);
}, true); }, true);
@ -217,7 +231,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Test splat // Test splat
function testSplatWildcard(){ public function testSplatWildcard()
{
$this->router->map('/account/*', function ($route) { $this->router->map('/account/*', function ($route) {
echo $route->splat; echo $route->splat;
}, true); }, true);
@ -227,7 +242,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Test splat without trailing slash // Test splat without trailing slash
function testSplatWildcardTrailingSlash(){ public function testSplatWildcardTrailingSlash()
{
$this->router->map('/account/*', function ($route) { $this->router->map('/account/*', function ($route) {
echo $route->splat; echo $route->splat;
}, true); }, true);
@ -237,7 +253,8 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Test splat with named parameters // Test splat with named parameters
function testSplatNamedPlusWildcard(){ public function testSplatNamedPlusWildcard()
{
$this->router->map('/account/@name/*', function ($name, $route) { $this->router->map('/account/@name/*', function ($name, $route) {
echo $route->splat; echo $route->splat;
$this->assertEquals('abc', $name); $this->assertEquals('abc', $name);
@ -248,25 +265,27 @@ class RouterTest extends PHPUnit_Framework_TestCase
} }
// Test not found // Test not found
function testNotFound() { public function testNotFound()
$this->router->map('/does_exist', array($this, 'ok')); {
$this->router->map('/does_exist', [$this, 'ok']);
$this->request->url = '/does_not_exist'; $this->request->url = '/does_not_exist';
$this->check('404'); $this->check('404');
} }
// Test case sensitivity // Test case sensitivity
function testCaseSensitivity() { public function testCaseSensitivity()
$this->router->map('/hello', array($this, 'ok')); {
$this->router->map('/hello', [$this, 'ok']);
$this->request->url = '/HELLO'; $this->request->url = '/HELLO';
$this->router->case_sensitive = true; $this->router->case_sensitive = true;
$this->check('404'); $this->check('404');
} }
// Passing URL parameters matched with regular expression for a URL containing Cyrillic letters: // Passing URL parameters matched with regular expression for a URL containing Cyrillic letters:
function testRegExParametersCyrillic(){ public function testRegExParametersCyrillic()
{
$this->router->map('/категория/@name:[абвгдеёжзийклмнопрстуфхцчшщъыьэюя]+', function ($name) { $this->router->map('/категория/@name:[абвгдеёжзийклмнопрстуфхцчшщъыьэюя]+', function ($name) {
echo $name; echo $name;
}); });
@ -274,5 +293,4 @@ class RouterTest extends PHPUnit_Framework_TestCase
$this->check('цветя'); $this->check('цветя');
} }
} }

@ -5,45 +5,50 @@
* @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com> * @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com>
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class VariableTest extends PHPUnit_Framework_TestCase class VariableTest extends PHPUnit\Framework\TestCase
{ {
/** /**
* @var \flight\Engine * @var \flight\Engine
*/ */
private $app; private $app;
function setUp() { protected function setUp(): void
{
$this->app = new \flight\Engine(); $this->app = new \flight\Engine();
} }
// Set and get a variable // Set and get a variable
function testSetAndGet() { public function testSetAndGet()
{
$this->app->set('a', 1); $this->app->set('a', 1);
$var = $this->app->get('a'); $var = $this->app->get('a');
$this->assertEquals(1, $var); $this->assertEquals(1, $var);
} }
// Clear a specific variable // Clear a specific variable
function testClear() { public function testClear()
{
$this->app->set('b', 1); $this->app->set('b', 1);
$this->app->clear('b'); $this->app->clear('b');
$var = $this->app->get('b'); $var = $this->app->get('b');
$this->assertEquals(null, $var); $this->assertNull($var);
} }
// Clear all variables // Clear all variables
function testClearAll() { public function testClearAll()
{
$this->app->set('c', 1); $this->app->set('c', 1);
$this->app->clear(); $this->app->clear();
$var = $this->app->get('c'); $var = $this->app->get('c');
$this->assertEquals(null, $var); $this->assertNull($var);
} }
// Check if a variable exists // Check if a variable exists
function testHas() { public function testHas()
{
$this->app->set('d', 1); $this->app->set('d', 1);
$this->assertTrue($this->app->has('d')); $this->assertTrue($this->app->has('d'));
} }

@ -5,24 +5,25 @@
* @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com> * @copyright Copyright (c) 2012, Mike Cao <mike@mikecao.com>
* @license MIT, http://flightphp.com/license * @license MIT, http://flightphp.com/license
*/ */
require_once 'vendor/autoload.php'; require_once 'vendor/autoload.php';
require_once __DIR__ . '/../flight/autoload.php'; require_once __DIR__ . '/../flight/autoload.php';
class ViewTest extends PHPUnit_Framework_TestCase class ViewTest extends PHPUnit\Framework\TestCase
{ {
/** /**
* @var \flight\template\View * @var \flight\template\View
*/ */
private $view; private $view;
function setUp() { protected function setUp(): void
{
$this->view = new \flight\template\View(); $this->view = new \flight\template\View();
$this->view->path = __DIR__ . '/views'; $this->view->path = __DIR__ . '/views';
} }
// Set template variables // Set template variables
function testVariables() { public function testVariables()
{
$this->view->set('test', 123); $this->view->set('test', 123);
$this->assertEquals(123, $this->view->get('test')); $this->assertEquals(123, $this->view->get('test'));
@ -32,31 +33,35 @@ class ViewTest extends PHPUnit_Framework_TestCase
$this->view->clear('test'); $this->view->clear('test');
$this->assertEquals(null, $this->view->get('test')); $this->assertNull($this->view->get('test'));
} }
// Check if template files exist // Check if template files exist
function testTemplateExists() { public function testTemplateExists()
{
$this->assertTrue($this->view->exists('hello.php')); $this->assertTrue($this->view->exists('hello.php'));
$this->assertTrue(!$this->view->exists('unknown.php')); $this->assertTrue(!$this->view->exists('unknown.php'));
} }
// Render a template // Render a template
function testRender() { public function testRender()
$this->view->render('hello', array('name' => 'Bob')); {
$this->view->render('hello', ['name' => 'Bob']);
$this->expectOutputString('Hello, Bob!'); $this->expectOutputString('Hello, Bob!');
} }
// Fetch template output // Fetch template output
function testFetch() { public function testFetch()
$output = $this->view->fetch('hello', array('name' => 'Bob')); {
$output = $this->view->fetch('hello', ['name' => 'Bob']);
$this->assertEquals('Hello, Bob!', $output); $this->assertEquals('Hello, Bob!', $output);
} }
// Default extension // Default extension
function testTemplateWithExtension() { public function testTemplateWithExtension()
{
$this->view->set('name', 'Bob'); $this->view->set('name', 'Bob');
$this->view->render('hello.php'); $this->view->render('hello.php');
@ -65,7 +70,8 @@ class ViewTest extends PHPUnit_Framework_TestCase
} }
// Custom extension // Custom extension
function testTemplateWithCustomExtension() { public function testTemplateWithCustomExtension()
{
$this->view->set('name', 'Bob'); $this->view->set('name', 'Bob');
$this->view->extension = '.html'; $this->view->extension = '.html';

@ -1,11 +1,14 @@
<?php <?php
class Factory {
// Cannot be instantiated
private function __construct() {
class Factory
{
// Cannot be instantiated
private function __construct()
{
} }
public static function create() { public static function create()
{
return new self(); return new self();
} }
} }

@ -1,10 +1,14 @@
<?php <?php
class Hello {
public function sayHi() { class Hello
{
public function sayHi()
{
return 'hello'; return 'hello';
} }
public static function sayBye() { public static function sayBye()
{
return 'goodbye'; return 'goodbye';
} }
} }

@ -1,8 +1,11 @@
<?php <?php
class User {
class User
{
public $name; public $name;
public function __construct($name = ''){ public function __construct($name = '')
{
$this->name = $name; $this->name = $name;
} }
} }
Loading…
Cancel
Save