From aa908ab7bd03818a2cb1f9b4686034f08736d1cb Mon Sep 17 00:00:00 2001 From: Daniel Weipert Date: Thu, 19 Feb 2026 14:52:15 +0100 Subject: initial commit --- src/Config.php | 19 +++++++++ src/ERPHP.php | 43 ++++++++++++++++++++ src/Hook.php | 70 ++++++++++++++++++++++++++++++++ src/Router/Route.php | 39 ++++++++++++++++++ src/Router/Router.php | 68 +++++++++++++++++++++++++++++++ src/Router/RoutesDefinitionInterface.php | 11 +++++ src/routes.php | 20 +++++++++ 7 files changed, 270 insertions(+) create mode 100644 src/Config.php create mode 100644 src/ERPHP.php create mode 100644 src/Hook.php create mode 100644 src/Router/Route.php create mode 100644 src/Router/Router.php create mode 100644 src/Router/RoutesDefinitionInterface.php create mode 100644 src/routes.php (limited to 'src') diff --git a/src/Config.php b/src/Config.php new file mode 100644 index 0000000..06b32d0 --- /dev/null +++ b/src/Config.php @@ -0,0 +1,19 @@ +loadPlugins(); + + // load routes + Hook::addFilter("Router::getRouteFiles", function (array $routeFiles) { + $routeFiles[] = __DIR__ . "/routes.php"; + return $routeFiles; + }); + + Router::getInstance()->run(); + } + + private function loadPlugins(): void + { + $pluginsDirectory = Config::get("plugins_directory", dirname(__DIR__) . "/plugins"); + $pluginDirectories = scandir($pluginsDirectory); + + foreach ($pluginDirectories as $directory) { + if ($directory === "." || $directory === "..") { + continue; + } + + $composerConfig = file_get_contents("$pluginsDirectory/$directory/composer.json"); + include_once "$pluginsDirectory/$directory/index.php"; + } + } +} diff --git a/src/Hook.php b/src/Hook.php new file mode 100644 index 0000000..0b292ce --- /dev/null +++ b/src/Hook.php @@ -0,0 +1,70 @@ +> $actions + */ + private static array $actions; + + /** + * @var array> $filters + */ + private static array $filters; + + /** + * @param string $name + * @param callable $function + */ + public static function addAction(string $name, callable $function, int $priority = 10): void + { + self::$actions[$name][$priority][] = $function; + } + + /** + * @param string $name + * @param array $parameters + */ + public static function runAction(string $name, ...$parameters): void + { + $actions = self::$actions[$name] ?? []; + + foreach ($actions as $priority => $functions) { + foreach ($functions as $function) { + call_user_func_array($function, $parameters); + } + } + } + + /** + * @param string $name + * @param callable $function + */ + public static function addFilter(string $name, callable $function, int $priority = 10): void + { + self::$filters[$name][$priority][] = $function; + } + + /** + * @param string $name + * @param mixed $value + * @param array $parameters + * + * @return mixed + */ + public static function applyFilter(string $name, mixed $value, ...$parameters): mixed + { + $filteredValue = $value; + $filters = self::$filters[$name] ?? []; + + foreach ($filters as $priority => $functions) { + foreach ($functions as $function) { + $filteredValue = call_user_func_array($function, [$value, ...$parameters]); + } + } + + return $filteredValue; + } +} diff --git a/src/Router/Route.php b/src/Router/Route.php new file mode 100644 index 0000000..ee9637b --- /dev/null +++ b/src/Router/Route.php @@ -0,0 +1,39 @@ +name = $path; + } + + $defaults = array_merge([ + "_" => [ + "action" => $action, + ], + ], $defaults); + + parent::__construct($path, $defaults, $requirements, $options, $host, $schemes, $methods, $condition); + } + + public function getName(): string + { + return $this->name; + } +} diff --git a/src/Router/Router.php b/src/Router/Router.php new file mode 100644 index 0000000..d770f15 --- /dev/null +++ b/src/Router/Router.php @@ -0,0 +1,68 @@ +routes = new RouteCollection(); + } + + public function run(): void + { + // get defined routes + $routeFiles = []; + $routeFiles = Hook::applyFilter("Router::getRouteFiles", $routeFiles); + + foreach ($routeFiles as $routesFile) { + /** @var RoutesDefinitionInterface $routesDefinition */ + $routesDefinition = include $routesFile; + $routes = $routesDefinition->get(); + + foreach ($routes as $route) { + /** @var Route[] $route */ + if (! is_null($this->routes->get($route->getName()))) { + throw new \Exception("Route \"{$route->getName()}\" already defined."); + } + + $this->routes->add($route->getName(), $route); + } + } + + $request = Request::createFromGlobals(); + $response = new Response(); + + // set semantically correct request method + if ($request->get("_method", false)) { + $request->setMethod($request->get("_method")); + } + + $matcher = new UrlMatcher($this->routes, new RequestContext()); + $match = $matcher->matchRequest($request); + + /** @var Response $response */ + $response = call_user_func_array($match["_"]["action"], [$request]); + $response->send(); + } +} diff --git a/src/Router/RoutesDefinitionInterface.php b/src/Router/RoutesDefinitionInterface.php new file mode 100644 index 0000000..614c7cd --- /dev/null +++ b/src/Router/RoutesDefinitionInterface.php @@ -0,0 +1,11 @@ +