mitas1c
mitas1c

Reputation: 335

PHP routing - How can I implement 404 page on wrong url routes?

everyone.

I have a basic router created in PHP.

I can redirect to any page I want, if there is a callback function the callback function gets executed and if there is a page (String instead of a function) the page loads the correct file. However I can't figure out how to implement 404 page on non-existing route.

I tried to reuse the preg_match() function, but that gave me no results and if I place the notFound() (404 page) in the else block, it always gets executed regardless of the correct url or not.

if(preg_match($pattern, $path, $matches) && $httpMethod === $route['method']) {

}else{
     self::notFound(); //THIS GETS EXECUTED ON EVERY ROUTE
}

This is my Code.

<?php

class Router{

    public static $routes = [];

    public static function get($route, $callback){
        self::$routes[] = [
            'route' => $route,
            'callback' => $callback,
            'method' => 'GET'
        ];

       
    }

    public static function resolve(){
        $path = $_SERVER['REQUEST_URI'];
        $httpMethod = $_SERVER['REQUEST_METHOD'];

        $methodMatch = false;
        $routeMatch = false;

        foreach(self::$routes as $route){

            // convert urls like '/users/:uid/posts/:pid' to regular expression
            $pattern = "@^" . preg_replace('/\\\:[a-zA-Z0-9\_\-]+/', '([a-zA-Z0-9\-\_]+)', preg_quote($route['route'])) . "$@D";
            $matches = Array();


            // check if the current request matches the expression
            if(preg_match($pattern, $path, $matches) && $httpMethod === $route['method']) {
                // remove the first match
                array_shift($matches);
                // call the callback with the matched positions as params

                if(is_callable($route['callback'])){
                    call_user_func_array($route['callback'], $matches);
                }else{
                    self::render($route['callback']);

                }
            }
        }
      
    }

    public static function render($file, $viewsFolder='./views/'){
        include($viewsFolder . $file);

    }

    public static function notFound(){
        http_response_code(400);
        include('./views/404.php');
        exit();
    }
}

Router::get("/", "home.php");
Router::get("/user/:id", function($val1) {
    $data = array(
        "Nicole",
        "Sarah",
        "Jinx",
        "Sarai"
    );

    echo $data[$val1] ?? "No data";
});

Router::get("/user/profile/:id", "admin.php");
Router::resolve();

?>

Upvotes: 0

Views: 213

Answers (1)

Kpro
Kpro

Reputation: 71

You can add notFound() at the very end of resolve() method, and a return when you hit a match:

public static function resolve(){
    $path = $_SERVER['REQUEST_URI'];
    $httpMethod = $_SERVER['REQUEST_METHOD'];

    $methodMatch = false;
    $routeMatch = false;

    foreach(self::$routes as $route){
        $pattern = "@^" . preg_replace('/\\\:[a-zA-Z0-9\_\-]+/', '([a-zA-Z0-9\-\_]+)', preg_quote($route['route'])) . "$@D";
        $matches = Array();

        if(preg_match($pattern, $path, $matches) && $httpMethod === $route['method']) {                
            array_shift($matches);
            if(is_callable($route['callback'])){
                call_user_func_array($route['callback'], $matches);
            }else{
                self::render($route['callback']);
            }
            return;
        }           
    }
    notFound();
}

Upvotes: 0

Related Questions