2024-09-07 13:02:52 -05:00
|
|
|
<?php
|
|
|
|
|
2024-09-07 16:39:57 -05:00
|
|
|
class TrieRouter implements Router
|
2024-09-07 13:02:52 -05:00
|
|
|
{
|
2024-09-07 16:39:57 -05:00
|
|
|
public array $root = [];
|
2024-09-07 13:02:52 -05:00
|
|
|
|
|
|
|
// Add route to the trie
|
2024-09-07 16:39:57 -05:00
|
|
|
public function add(string $method, string $route, callable $handler): Router
|
2024-09-07 13:02:52 -05:00
|
|
|
{
|
2024-09-07 16:39:57 -05:00
|
|
|
$node = &$this->root[$method];
|
2024-09-07 13:02:52 -05:00
|
|
|
$segments = explode('/', trim($route, '/'));
|
|
|
|
|
|
|
|
foreach ($segments as $segment) {
|
|
|
|
if (!isset($node[$segment])) {
|
|
|
|
$node[$segment] = ['_children' => [], '_handler' => null];
|
|
|
|
}
|
|
|
|
$node = &$node[$segment]['_children'];
|
|
|
|
}
|
|
|
|
|
|
|
|
$node['_handler'] = $handler;
|
2024-09-07 16:39:57 -05:00
|
|
|
|
|
|
|
return $this;
|
2024-09-07 13:02:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Find and handle the route
|
2024-09-07 16:39:57 -05:00
|
|
|
public function lookup(string $method, string $uri): int|array
|
2024-09-07 13:02:52 -05:00
|
|
|
{
|
2024-09-07 16:39:57 -05:00
|
|
|
$node = &$this->root[$method];
|
2024-09-07 13:02:52 -05:00
|
|
|
$segments = explode('/', trim($uri, '/'));
|
|
|
|
$params = [];
|
|
|
|
|
|
|
|
foreach ($segments as $segment) {
|
|
|
|
if (isset($node[$segment])) {
|
|
|
|
$node = &$node[$segment]['_children'];
|
|
|
|
} else {
|
|
|
|
// Handle dynamic parameters (e.g., :id)
|
|
|
|
$dynamicSegment = $this->matchDynamicSegment($node, $segment);
|
|
|
|
if ($dynamicSegment) {
|
|
|
|
$params[] = $segment;
|
|
|
|
$node = &$node[$dynamicSegment]['_children'];
|
|
|
|
} else {
|
2024-09-07 16:39:57 -05:00
|
|
|
return 404;
|
2024-09-07 13:02:52 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if a handler exists for the current node
|
2024-09-07 16:39:57 -05:00
|
|
|
if (isset($node['_handler'])) return [$node['_handler'], $params];
|
2024-09-07 13:02:52 -05:00
|
|
|
|
2024-09-07 16:39:57 -05:00
|
|
|
return 404;
|
2024-09-07 13:02:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Match dynamic route segments like ':id'
|
|
|
|
private function matchDynamicSegment(array $node, string $segment)
|
|
|
|
{
|
|
|
|
foreach ($node as $key => $value) {
|
|
|
|
if (strpos($key, ':') === 0) return $key;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2024-09-07 16:39:57 -05:00
|
|
|
public function clear(): Router
|
2024-09-07 13:02:52 -05:00
|
|
|
{
|
2024-09-07 16:39:57 -05:00
|
|
|
$this->root = [];
|
|
|
|
return $this;
|
2024-09-07 13:02:52 -05:00
|
|
|
}
|
|
|
|
}
|