Smallweb 0.25

by Achille Lacoin

4 min read

Smallweb 0.25 is out! It includes a includes two really powerful features: pluggable authentication, and support for receiving emails. Let's get into it!

OpenID Connect Support

Smallweb now supports OpenID Connect! This means that you can use any OpenID Connect provider to authenticate your users. This includes popular providers like Google, GitHub, and Microsoft.

The easier way to test this out is to use the https://lastlogin.net provider. Just add the oidc.issuer key to your smallweb config, and whitelist your own email:

{
    "domain": "example.com",
    "oidc.issuer": "https://lastlogin.net",
    // emails in this list will be able to access all private apps
    "authorizedEmails": [
        "[email protected]"
    ],
    "apps": {
        "dashboard": {
            // all routes in this app will require authentication
            "private": true,
            // emails in this list will only be able to access this app
            "authorizedEmails": [
                "[email protected]"
            ]
        }
    }
}

The next time you go to https://dashboard.example.com, you will be redirected to the provider authorization page. After you log in (as one of the authorized emails), you will be redirected back to your app.

You can access the user information using the Remote-* headers.

export default {
    fetch: (req: Request) => {
        const email = req.headers.get("Remote-Email");

        return new Response(`Hello ${email}!`);
    }
}

The coolest part is that you can host your own OpenID Connect provider in smallweb thanks to a library like OpenAuth.

New email endpoint

You can now send email to smallweb apps ! Just start smallweb with the --smtp-addr flag:

smallweb up --smtp-addr :25

You'll then need to set a few records in your DNS, and your app will be accessible at <app>@<domain>.

To handle incoming emails, just declare a email entrypoint in your app:

import PostalMime from "npm:postal-mime";

export default {
    email: (msg: ReadableStream) => {
        const email = await PostalMime.parse(msg);

        console.log('Subject:', email.subject);
        console.log('HTML:', email.html);
        console.log('Text:', email.text);
    }
}

As you can see, smallweb does not parse the email content for you. As there is no web-standard Email object, I prefer to leverage external libraries. I had a good experience using postal-mime.

SSH keys support in .sops.yaml

You can now public ssh public keys in your .sops.yaml file (instead of age public keys).

# $SMALLWEB_DIR/.sops.yaml
creation_rules:
  - key_groups:
      - age:
          - ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIJW+GQk0KCvSreL+y3AZdtCu82+13E2eEled+sGRkIEv # laptop
          - ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIHpF8kTXaBHTZqKfmEsKTILSxQYpPenI7wyMK0qTNE8y # vps

More information in the docs.

Process stdin from the run entrypoint

You can now access input piped an app cli by using the input parameter in your run entrypoint. For example, here is a simple app that will prettify json passed through stdin:

// ~/smallweb/prettify/main.ts
import { toJson } from "jsr:@std/streams"

export default {
    run: async (_args: string[], input: ReadableStream) => {
        const json = await toJson(input);
        console.log(JSON.stringify(json, null, 2));
    }
}

Here is how it can be invoked:

cat example.json | smallweb run prettify

It also works when accessible app clis through ssh:

cat example.json | ssh [email protected]

Cron tasks are moved to the global config

Instead of declaring cron tasks in the app config:

// ~/smallweb/my-app/smallweb.json
{
    "crons": [
        {
            "schedule": "@hourly",
            "args": ["refresh"]
        }
    ]
}

Crons are now specified from the global config:

// ~/smallweb/.smallweb/config.json
{
    "apps": {
        "my-app": {
            "crons": [
                {
                    "schedule": "@hourly",
                    "args": ["refresh"]
                }
            ]
        }
    }
}

Reworked docker image

I've spent a lot of time working on the docker image to simplify using smallweb from your homelab. Here is a minimal example of a compose setup:

services:
  smallweb:
    image: ghcr.io/pomdtr/smallweb:latest
    restart: unless-stopped
    command: up --enable-crons --ssh-addr :2222
    ports:
      - 7777:7777
      - 2222:2222
    environment:
      - PUID=1000
      - PGID=1000
    volumes:
      - $HOME/smallweb:/smallweb
      - $HOME/.ssh/id_ed25519:/home/smallweb/.ssh/id_ed25519
      - deno_cache:/home/smallweb/.cache/deno

volumes:
    deno_cache:

Json logs on stdout

In addition to the opentelemetry support released in 0.24, a lot of you have been asking for a simple way to get access to the http / console logs of your smallweb app.

The smallweb up command will now dump these logs using json to stdout.

$ smallweb up
{"time":"2025-03-27T16:01:24.100825+01:00","level":"INFO","msg":"serving http","domain":"smallweb.localhost","dir":"/Users/pomdtr/Developer/pomdtr/smallweb/example"}
{"time":"2025-03-27T16:01:30.524557+01:00","level":"INFO","msg":"<-- GET /.well-known/oauth-authorization-server","logger":"console","app":"auth","stream":"stdout"}
{"time":"2025-03-27T16:01:30.525509+01:00","level":"INFO","msg":"--> GET /.well-known/oauth-authorization-server \u001b[32m200\u001b[0m 1ms","logger":"console","app":"auth","stream":"stdout"}
{"time":"2025-03-27T16:01:30.526423+01:00","level":"INFO","msg":"200: OK","logger":"http","request":{"time":"2025-03-27T15:01:30.215387Z","method":"GET","host":"auth.smallweb.localhost","path":"/.well-known/openid-configuration","query":"","ip":"[::1]:51965","referer":"","length":0},"response":{"time":"2025-03-27T15:01:30.526409Z","latency":311019625,"status":200,"length":256},"id":"00cf33b0-a1e2-4dea-98dd-26bd6e78b237"}
{"time":"2025-03-27T16:01:30.564595+01:00","level":"INFO","msg":"200: OK","logger":"http","request":{"time":"2025-03-27T15:01:30.196483Z","method":"GET","host":"hono.smallweb.localhost","path":"/","query":"","ip":"[::1]:51963","referer":"","length":0},"response":{"time":"2025-03-27T15:01:30.564581Z","latency":368094708,"status":200,"length":5},"id":"30cbd0c1-67ad-45a7-b124-1fe2fff7ecfd"}

You can then get these logs from journalctl or docker logs, and use tools like jq to process them.