Skip to content

Routes

Routing a URI to the code that should execute it is a fundamental part of any web application. In Monarch, routes are defined using a folder-based approach. Each URI is directly associated with a route file. The route file is a file that determines what happens when a user visits a specific URI.

An example folder structure might look like this:

routes/
   about.php
   index.php
   blog/
      index.php
      post[id].php
      archive[year][month].php

In this example, the index.php file would be responsible for handling the / URI. The about.php file would handle the /about URI. The blog/index.php file would handle the /blog URI, and the blog/post[id].php file would handle the /blog/post/123 URI.

Layouts

Special Layout files can be used to define the structure of a page. To define a master layout file to be used on all web files, create a file within the routes folder called +layout.php.

routes/
   +layout.php
   home.php

The +layout.php file defines the HTML structure of the page. It should include the <?= $content ?> tag to indicate where the content of the page should be placed.

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>My Site</title>
</head>
<body>
    <header>
        <h1>My Site</h1>
    </header>
    <main>
        <?= $content ?>
    </main>
    <footer>
        &copy; 2021 My Site
    </footer>
</body>
</html>

Nested Layouts

If a subfolder contains a +layout.php file, it will be inserted into the parent layout file. There is no maximum depth to the nesting of the layout files. All web files at that folder level will use the combined layout files. See the views documentation for more detailed information.

Route Types

By default, Monarch supports the following route types:

  • Web pages
  • Web page fragments
  • Rest API endpoints
  • Markdown content

Route types are defined within the app/config/routes.php file.

Route types are determined based on the route files' extension. The default route files are:

  • .php for web pages
  • -{name}.php for fragments
  • .api.php for Rest API endpoints
  • .md for Markdown files.

Errors

If a route file ends up throwing an error, the default error page found at /+error.php will be displayed to the user.

You can customize the main file to change the look of the error page. You can also create custom error pages within each of the route folders to customize the error page for that specific route.

Control Files

Control files are used to define the logic for a route. They are optional, and are only needed if you need to do more than just display a basic page.

Control files are named the same as the route file, but with a .control.php extension. For example, the about.php route file would use the about.control.php control file.

Within each control file, you can define an anonymous class that has methods that match the HTTP verbs you want to handle. For example, to handle a GET request, you would define a get() method. To handle a POST request, you would define a post() method.

<?php

return new class()
{
    public function get(): string
    {
        return 'Hello, World!';
    }
}

If the method returns a string, it will be made available to the route file as the $content variable. If it returns any other data type, it will be provided to the route file as the $data variable.

If you need to provide both the $content and $data variables, you can return an array with the content and data keys.

<?php

return new class()
{
    public function get(): array
    {
        return [
            'content' => 'Hello, World!',
            'data' => [1, 2, 3],
        ];
    }
}

Dynamic Routes

Dynamic routes are routes that contain variables within the URI. For example, a route that handles a blog post might look like this:

routes/
   blog/
      post[id].php

In this example, the post[id].php file would handle the /blog/post/123 URI. The id variable would be available within the control file as a parameter.

<?php

return new class()
{
    public function get(int $id): string
    {
        return 'The post ID is: ' . $id;
    }
}

You many include as many dynamic variables as you need within the route file. They will be passed to the control file in the order they are defined.

API Routes

API routes are similar to control files, but they are designed to return JSON data instead of HTML content. API routes are defined by using the .api.php extension. The class should extend the Monarch\API class, which provides helper methods for responding to API requests and handling errors.

<?php

use Monarch\API;

return new class() extends API
{
    public function get(): array
    {
        return $this->respond([
            'message' => 'Hello, World!',
        ]);
    }
}

Like control routes, API routes should have methods that match the HTTP verbs you want to handle. Any data returned must be either null or an array that can be converted to JSON.

Response Methods

API routes have access to the following response methods:

  • respond(array $data): array - Respond with a JSON object.
  • withStatus(int $status ?string $message=null): self - Set the status code of the response.
public function get()
{
    return $this->respond([
        'message' => 'Resource created',
    ])->withStatus(200);
}
  • fail(?string $description=null) - Respond with an error message. If a description is provided, it will be included in the response, otherwise the message will be "Unknown Error". The response also includes the timestamp of the error, the status code, and the URI path that caused the error.
public function get()
{
    $resource = $this->getResource();

    if (! $resource) {
        return $this->fail('Resource not found');
    }
}

// Returns the following JSON:
{
    "error": "Resource not found",
    "message": "Resource not found",
    "timestamp": "2021-10-01 12:00:00",
    "status": 404,
    "path": "/api/resource/123"
}
  • respondCreated(array $body, ?string $message = null) - Respond with a 201 status code and a JSON object. The body of the response is the data provided in the $body parameter. If a message is provided, it will be included in the response.
public function post()
{
    $resource = $this->createResource();

    return $this->respondCreated($resource, 'Resource created');
}
  • respondDeleted(array $body, ?string $message = null) - Respond with a 200 status code and a JSON object. The body of the response is the data provided in the $body parameter. If a message is provided, it will be included in the response.
public function delete()
{
    $resource = $this->deleteResource();

    return $this->respondDeleted($resource, 'Resource deleted');
}
  • respondUpdated(array $body, ?string $message = null) - Respond with a 200 status code and a JSON object. The body of the response is the data provided in the $body parameter. If a message is provided, it will be included in the response.
public function put()
{
    $resource = $this->updateResource();

    return $this->respondUpdated($resource, 'Resource updated');
}
  • respondNoContent(?string $message = null) - Respond with a 204 status code and no content. If a message is provided, it will be included in the response.
public function delete()
{
    $this->deleteResource();

    return $this->respondNoContent('Resource deleted');
}
  • failUnauthorized(?string $error = null) - Respond with a 401 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Unauthorized".
public function get()
{
    if (! $this->isAuthorized()) {
        return $this->failUnauthorized('Unauthorized');
    }
}
  • failForbidden(?string $error = null) - Respond with a 403 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Forbidden".
public function get()
{
    if (! $this->isAuthorized()) {
        return $this->failForbidden('Forbidden');
    }
}
  • failNotFound(?string $error = null) - Respond with a 404 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Not Found".
public function get()
{
    $resource = $this->getResource();

    if (! $resource) {
        return $this->failNotFound('Resource not found');
    }
}
  • failValidationError(string $error = 'Bad Request') - Respond with a 400 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Bad Request". Used when the data provided by the client cannot be validated.
public function post()
{
    $data = $this->getRequestData();

    if (! $this->validateData($data)) {
        return $this->failValidationError('Invalid data');
    }
}
  • failValidationErrors(array $errors) - Respond with a 400 status code and an array of error messages. The error message is a JSON object that contains the errors provided in the $errors parameter. Used when the data provided by the client cannot be validated for one or more fields.
public function post()
{
    $errors = $this->validateData($data);

    if ($errors) {
        return $this->failValidationErrors($errors);
    }
}
  • failResourceExists(string $description = 'Conflict') - Respond with a 409 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Conflict". Used when the resource already exists and cannot be created.
public function post()
{
    $resource = $this->getResource();

    if ($resource) {
        return $this->failResourceExists('Resource already exists');
    }
}
  • failResourceGone(string $description = 'Gone') - Respond with a 410 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Gone". Used when the resource has been deleted and cannot be accessed.
public function get()
{
    $resource = $this->getResource();

    if (! $resource) {
        return $this->failResourceGone('Resource has been deleted');
    }
}
  • failTooManyRequests(string $description = 'Too Many Requests') - Respond with a 429 status code and an error message. If an error string is provided, it will be included in the response, otherwise the message will be "Too Many Requests". Used when the client has sent too many requests in a given amount of time.
public function get()
{
    if ($this->isRateLimited()) {
        return $this->failTooManyRequests('Rate limit exceeded');
    }
}

Monarch Routes

When the DEBUG constant is set to true, Monarch reserves a set of routes to provide tools for you to use. These all use the _/ folder to avoid conflicts with your own routes. You should not use a base folder in your routes folder called _ otherwise that will override the Monarch routes.