Fossil

Serving via httpd on OpenBSD
Login

httpd is the default web server that is included in the base install on OpenBSD. It's minimal and lightweight but secure and capable, and provides a clean interface for setting up a Fossil server using FastCGI.

This article will detail the steps required to setup a TLS-enabled httpd configuration that serves multiple Fossil repositories out of a single directory within a chroot, and allow ssh access to create new repositories remotely.

NOTE: The following instructions assume an OpenBSD 6.7 installation.

Install Fossil

Use the OpenBSD package manager pkg_add to install Fossil, making sure to select the statically linked binary.

$ doas pkg_add fossil
quirks-3.325 signed on 2020-06-12T06:24:53Z
Ambiguous: choose package for fossil
      0: <None>
      1: fossil-2.10v0
      2: fossil-2.10v0-static
Your choice: 2
fossil-2.10v0-static: ok

This installs Fossil into the chroot. To facilitate local use, create a symbolic link of the fossil executable into /usr/local/bin.

$ doas ln -s /var/www/bin/fossil /usr/local/bin/fossil

As a privileged user, create the file /var/www/cgi-bin/scm with the following contents to make the CGI script that httpd will execute in response to fsl.domain.tld requests; all paths are relative to the /var/www chroot.

#!/bin/fossil
directory: /htdocs/fsl.domain.tld
notfound: https://domain.tld
repolist
errorlog: /logs/fossil.log

The directory directive instructs Fossil to serve all repositories found in /var/www/htdocs/fsl.domain.tld, while errorlog sets logging to be saved to /var/www/logs/fossil.log; create the repository directory and log file—making the latter owned by the www user, and the script executable.

$ doas mkdir /var/www/htdocs/fsl.domain.tld
$ doas touch /var/www/logs/fossil.log
$ doas chown www /var/www/logs/fossil.log
$ doas chmod 660 /var/www/logs/fossil.log
$ doas chmod 755 /var/www/cgi-bin/scm

Setup chroot

Fossil needs both /dev/random and /dev/null, which aren't accessible from within the chroot, so need to be constructed; /var, however, is mounted with the nodev option. Rather than removing this default setting, create a small memory filesystem and then mount it on to /var/www/dev with mount_mfs(8) so that the random and null device files can be created. In order to avoid necessitating a startup script to recreate the device files at boot, create a template of the needed /dev tree to automatically populate the memory filesystem.

$ doas mkdir /var/www/dev
$ doas install -d -g daemon /template/dev
$ cd /template/dev
$ doas /dev/MAKEDEV urandom
$ doas mknod -m 666 null c 2 2
$ doas mount_mfs -s 1M -P /template/dev /dev/sd0b /var/www/dev
$ ls -l
total 0
crw-rw-rw-  1 root  daemon    2,   2 Jun 20 08:56 null
lrwxr-xr-x  1 root  daemon         7 Jun 18 06:30 random@ -> urandom
crw-r--r--  1 root  wheel    45,   0 Jun 18 06:30 urandom

To make the mountable memory filesystem permanent, open /etc/fstab as a privileged user and add the following line to automate creation of the filesystem at startup:

swap /var/www/dev mfs rw,-s=1048576,-P=/template/dev 0 0

The same user that executes the fossil binary must have writable access to the repository directory that resides within the chroot; on OpenBSD this is www. In addition, grant repository directory ownership to the user who will push to, pull from, and create repositories.

$ doas chown -R user:www /var/www/htdocs/fsl.domain.tld
$ doas chmod 770 /var/www/htdocs/fsl.domain.tld

Configure httpd

On OpenBSD, httpd.conf(5) is the configuration file for httpd. To setup the server to serve all Fossil repositores within the directory specified in the CGI script, and automatically redirect standard HTTP requests to HTTPS—apart from Let's Encrypt challenges issued in response to acme-client(1) certificate requests—create /etc/httpd.conf as a privileged user with the following contents.

server "fsl.domain.tld" {
    listen on * port http
    root "/htdocs/fsl.domain.tld"
    location "/.well-known/acme-challenge/*" {
        root "/acme"
        request strip 2
    }
    location * {
        block return 301 "https://$HTTP_HOST$REQUEST_URI"
    }
    location  "/*" {
        fastcgi { param SCRIPT_FILENAME "/cgi-bin/scm" }
    }
}

server "fsl.domain.tld" {
    listen on * tls port https
    root "/htdocs/fsl.domain.tld"
    tls {
        certificate "/etc/ssl/domain.tld.fullchain.pem"
        key "/etc/ssl/private/domain.tld.key"
    }
    hsts {
        max-age 15768000
        preload
        subdomains
    }
    connection max request body 104857600
    location  "/*" {
        fastcgi { param SCRIPT_FILENAME "/cgi-bin/scm" }
    }
    location "/.well-known/acme-challenge/*" {
        root "/acme"
        request strip 2
    }
}

The default limit for HTTP messages in OpenBSD’s httpd server is 1 MiB. Fossil chunks its sync protocol such that this is not normally a problem, but when sending unversioned content, it uses a single message for the entire file. Therefore, if you will be storing files larger than this limit as unversioned content, you need to raise the limit as we’ve done above with the “connection max request body” setting, raising the limit to 100 MiB.

NOTE: If not already in possession of an HTTPS certificate, comment out the https server block and proceed to securing a free Let's Encrypt Certificate; otherwise skip to Start httpd.

Let's Encrypt Certificate

In order for httpd to serve HTTPS, secure a free certificate from Let's Encrypt using acme-client. Before issuing the request, however, ensure you have a zone record for the subdomain with your registrar or nameserver. Then open /etc/acme-client.conf as a privileged user to configure the request.

authority letsencrypt {
    api url "https://acme-v02.api.letsencrypt.org/directory"
    account key "/etc/acme/letsencrypt-privkey.pem"
}

authority letsencrypt-staging {
    api url "https://acme-staging.api.letsencrypt.org/directory"
    account key "/etc/acme/letsencrypt-staging-privkey.pem"
}

domain domain.tld {
    alternative names { www.domain.tld fsl.domain.tld }
    domain key "/etc/ssl/private/domain.tld.key"
    domain certificate "/etc/ssl/domain.tld.crt"
    domain full chain certificate "/etc/ssl/domain.tld.fullchain.pem"
    sign with letsencrypt
}

Start httpd with the new configuration file, and issue the certificate request.

$ doas rcctl start httpd
$ doas acme-client -vv domain.tld
acme-client: /etc/acme/letsencrypt-privkey.pem: account key exists (not creating)
acme-client: /etc/acme/letsencrypt-privkey.pem: loaded RSA account key
acme-client: /etc/ssl/private/domain.tld.key: generated RSA domain key
acme-client: https://acme-v01.api.letsencrypt.org/directory: directories
acme-client: acme-v01.api.letsencrypt.org: DNS: 172.65.32.248
...
N(Q????Z???j?j?>W#????b???? H????eb??T??*? DNosz(???n{L}???D???4[?B] (1174 bytes)
acme-client: /etc/ssl/domain.tld.crt: created
acme-client: /etc/ssl/domain.tld.fullchain.pem: created

A successful result will output the public certificate, full chain of trust, and private key into the /etc/ssl directory as specified in acme-client.conf.

$ doas ls -lR /etc/ssl
-r--r--r--   1 root  wheel   2.3K Mar  2 01:31:03 2018 domain.tld.crt
-r--r--r--   1 root  wheel   3.9K Mar  2 01:31:03 2018 domain.tld.fullchain.pem

/etc/ssl/private:
-r--------  1 root  wheel   3.2K Mar  2 01:31:03 2018 domain.tld.key

Make sure to reopen /etc/httpd.conf to uncomment the second server block responsible for serving HTTPS requests before proceeding.

Start httpd

With httpd configured to serve Fossil repositories out of /var/www/htdocs/fsl.domain.tld, and the certificates and key in place, enable and start slowcgi—OpenBSD's FastCGI wrapper server that will execute the above Fossil CGI script—before checking that the syntax of the httpd.conf configuration file is correct, and (re)starting the server (if still running from requesting a Let's Encrypt certificate).

$ doas rcctl enable slowcgi
$ doas rcctl start slowcgi
slowcgi(ok)
$ doas httpd -vnf /etc/httpd.conf
configuration OK
$ doas rcctl start httpd
httpd(ok)

Configure Client

To facilitate creating new repositories and pushing them to the server, add the following function to your ~/.cshrc or ~/.zprofile or the config file for whichever shell you are using on your development box.

finit() {
    fossil init $1.fossil && \
    chmod 664 $1.fossil && \
    fossil open $1.fossil && \
    fossil user password $USER $PASSWD && \
    fossil remote-url https://$USER:$PASSWD@fsl.domain.tld/$1 && \
    rsync --perms $1.fossil $USER@fsl.domain.tld:/var/www/htdocs/fsl.domain.tld/ >/dev/null && \
    chmod 644 $1.fossil && \
    fossil ui
}

This enables a new repository to be made with finit repo, which will create the fossil repository file repo.fossil in the current working directory; by default, the repository user is set to the environment variable $USER. It then opens the repository and sets the user password to the $PASSWD environment variable (which you can either set with export PASSWD 'password' on the command line or add to a secured shell environment file), and the remote-url to https://fsl.domain.tld/repo with the credentials of $USER who is authenticated with $PASSWD. Finally, it rsync's the file to the server before opening the local repository in your browser where you can adjust settings such as anonymous user access, and set pertinent repository details. Thereafter, you can add files with fossil add, and commit with fossil ci -m 'commit message' where Fossil, by default, will push to the remote-url. It's suggested you read the Fossil documentation; with a sane and consistent development model, the system is much more efficient and cohesive than git—so the learning curve is not steep at all.

Return to the top-level Fossil server article.