On demand

Plugin to render pages on demand in Deno Deploy

Options See on deno.land

routesPath string

The file path to save the routes

Default:
"/_routes.json"
preloadPath string

The file path to save the preloaded modules

Default:
"/_preload.ts"
extraData function

Extra data to pass to the pages

Description

This plugin allows to render pages on demand. It can be useful in some scenarios:

  • There are pages with dynamic content that must be generated at the request time.
  • The site is too big, with thousands of pages, so the build takes too much time.

Lume can generate pages on demand for these cases.

Note: This plugin only works with Deno Deploy.

Installation

Import this plugin in your _config.ts file to use it:

import lume from "lume/mod.ts";
import onDemand from "lume/plugins/on_demand.ts";

const site = lume();

site.use(onDemand(/* Options */));

export default site;

How does it work?

1. Mark the ondemand pages

First, you need to configure the pages that must be rendered on demand. This is done by setting the ondemand variable to true. For example, let's say we want to render the home page dynamically:

---
layout: layout.vto
title: This is a title
ondemand: true
---

<h1>{{ title }}</h1>

2. The _routes.json file

When the site is built, the pages with the ondemand variable are skipped and a _routes.json file is generated containing a map with the associated page file for each URL, for example:

{
  "/": "./index.vto"
}

3. The _preload.ts file

If your source folder contains any file with extensions .ts, .tsx, .js, .jsx or .mjs, the archive _preload.ts is also created. This file is important if your site is hosted on Deno Deploy.

Deno Deploy only can import dynamically files that are statically analyzable in the code. The _preload.ts file contains code that does nothing but Deno Deploy can analyze and prepare to execute on demand. You have more info in the Deno Deploy's Changelog.

4. The serve file

Finally, we have to configure a HTTP server with the onDemand middleware. Create the file serve.ts, that will be used by Deno Deploy with the following code:

import site from "./_config.ts";
import Server from "lume/core/server.ts";
import onDemand from "lume/middlewares/on_demand.ts";
import "./_preload.ts";

const server = new Server({
  port: 8000,
  root: site.dest(),
});

server.use(onDemand({ site }));

server.start();

console.log("Listening on http://localhost:8000");

The middleware needs an instance of our site in order to render the pages. We can import it from the _config.ts file. It also automatically loads the _routes.json file in order to know which file needs to be rendered for each URL.

And that's all! The _routes.json file is regenerated automatically by the build to ensure it's up to date with your changes.

Using extra data

If you want to use dynamic data, use the option extraData which accepts a function that must return an object with the extra data to be passed to the page. For example, let's say we want to pass the search parameters of the request's url:

import lume from "lume/mod.ts";
import onDemand from "lume/plugins/on_demand.ts";

site.use(onDemand({
  extraData(request: Request) {
    const searchParams = new URL(request.url).searchParams;
    const params = Object.fromEntries(searchParams.entries());

    return {
      params,
    };
  },
}));

export default site;

Now, the on-demand pages will have the params key with the search params values. For example, in a Nunjucks page:

---
layout: layout.vto
ondemand: true
url: /example/
---

Hello {{ params.name }}

The URL /example/?name=Óscar will return Hello Óscar.

See an example

You can see a live example of a site with two pages generated on demand. And the repository with the source code.