3 b9231167 2021-02-08 op gmid is a fast Gemini server written with security in mind. I
4 b9231167 2021-02-08 op initially wrote it to serve static files, but it has grown into a
5 747b35d0 2021-02-08 op featureful server.
10 601bc1cc 2021-01-30 op (random order)
12 747b35d0 2021-02-08 op - reconfiguration: reload the running configuration without
14 86edc4f4 2021-02-04 op - sandboxed by default on OpenBSD, Linux and FreeBSD
15 290b5baa 2021-02-23 op - automatic redirect/error pages (see `block return`)
16 4d2ec6d7 2021-01-13 op - IRI support (RFC3987)
17 286c4f40 2021-01-27 op - punycode support
18 601bc1cc 2021-01-30 op - dual stack (IPv4 and IPv6)
19 601bc1cc 2021-01-30 op - automatic certificate generation for config-less mode
21 290b5baa 2021-02-23 op - low memory footprint
22 b9231167 2021-02-08 op - event-based asynchronous I/O model
23 b9220ca4 2021-01-11 op - small codebase, easily hackable
24 b9220ca4 2021-01-11 op - virtual hosts
25 286c4f40 2021-01-27 op - per-location rules
26 286c4f40 2021-01-27 op - optional directory listings
27 286c4f40 2021-01-27 op - configurable mime types
28 0b00962d 2021-01-25 op - chroot support
31 286c4f40 2021-01-27 op ## Internationalisation (IRIs, UNICODE, punycode, all that stuff)
33 286c4f40 2021-01-27 op Even thought the current Gemini specification doesn't mention anything
34 601bc1cc 2021-01-30 op in this regard, I do think these are important things and so I tried
35 601bc1cc 2021-01-30 op to implement them in the most user-friendly way I could think of.
37 601bc1cc 2021-01-30 op For starters, gmid has full support for IRI (RFC3987 —
38 22a742e4 2021-01-29 op Internationalized Resource Identifiers). IRIs are a superset of URIs,
39 286c4f40 2021-01-27 op so there aren't incompatibilities with URI-only clients.
41 601bc1cc 2021-01-30 op There is full support also for punycode. In theory, the user doesn't
42 286c4f40 2021-01-27 op even need to know that punycode is a thing. The hostname in the
43 601bc1cc 2021-01-30 op configuration file can (and must be) in the decoded form (e.g. `naïve`
44 601bc1cc 2021-01-30 op and not `xn--nave-6pa`), gmid will do the rest.
46 601bc1cc 2021-01-30 op The only missing piece is UNICODE normalisation of the IRI path: gmid
47 601bc1cc 2021-01-30 op doesn't do that (yet).
50 1487e11e 2021-02-06 op ## Configuration
52 1487e11e 2021-02-06 op gmid has a rich configuration file, heavily inspired by OpenBSD'
53 1487e11e 2021-02-06 op httpd. While you should definitely check the manpage because it
54 290b5baa 2021-02-23 op documents every option in depth, here's a small example of how a
55 290b5baa 2021-02-23 op configuration file looks like.
58 1487e11e 2021-02-06 op ipv6 on # enable ipv6
60 1487e11e 2021-02-06 op server "example.com" {
61 1487e11e 2021-02-06 op cert "/path/to/cert.pem"
62 1487e11e 2021-02-06 op key "/path/to/key.pem"
63 1487e11e 2021-02-06 op root "/var/gemini/example.com"
67 1487e11e 2021-02-06 op location "/files/*" {
71 1487e11e 2021-02-06 op location "/repo/*" {
72 1487e11e 2021-02-06 op # change the index file name
73 1487e11e 2021-02-06 op index "README.gmi"
81 b9231167 2021-02-08 op gmid depends on a POSIX libc, libevent2, OpenSSL/LibreSSL and libtls
82 b9231167 2021-02-08 op (provided either by LibreSSL or libretls). At build time, flex and
83 b9231167 2021-02-08 op yacc (or GNU bison) are also needed.
85 771d8f28 2021-01-17 op The build is as simple as
90 8f0da580 2021-01-21 op If the configure scripts fails to pick up something, please open an
91 f980545b 2021-01-21 op issue or notify me via email.
93 771d8f28 2021-01-17 op To install execute:
99 601bc1cc 2021-01-30 op If you have trouble installing LibreSSL or libretls, you can use
100 601bc1cc 2021-01-30 op Docker to build a `gmid` image with:
102 17b09e3c 2021-01-18 op docker build -t gmid .
104 17b09e3c 2021-01-18 op and then run it with something along the lines of
106 17b09e3c 2021-01-18 op docker run --rm -it -p 1965:1965 \
107 6c117838 2021-01-25 op -v /path/to/gmid.conf:...:ro \
108 17b09e3c 2021-01-18 op -v /path/to/docs:/var/gemini \
109 6c117838 2021-01-25 op gmid -c .../gmid.conf
111 601bc1cc 2021-01-30 op (ellipses used for brevity)
113 dd8cc7d3 2021-01-22 op ### Local libretls
115 8f0da580 2021-01-21 op This is **NOT** recommended, please try to port LibreSSL/LibreTLS to
116 8f0da580 2021-01-21 op your distribution of choice or use docker instead.
118 601bc1cc 2021-01-30 op However, it's possible to statically-link `gmid` to locally-installed
119 601bc1cc 2021-01-30 op libretls quite easily. (It's how I test gmid on Fedora, for instance)
121 04397b32 2021-01-21 op Let's say you have compiled and installed libretls in `$LIBRETLS`,
122 8f0da580 2021-01-21 op then you can build `gmid` with
124 04397b32 2021-01-21 op ./configure CFLAGS="-I$LIBRETLS/include" \
125 1606927e 2021-02-11 op LDFLAGS="$LIBRETLS/lib/libtls.a -lssl -lcrypto -lpthread -levent"
134 601bc1cc 2021-01-30 op to start the suite. Keep in mind that the regression tests will
135 601bc1cc 2021-01-30 op create files inside the `regress` directory and bind the 10965 port.
138 881a9dd9 2021-01-16 op ## Architecture/Security considerations
140 290b5baa 2021-02-23 op gmid is composed by four processes: the parent process, the logger,
141 290b5baa 2021-02-23 op the listener and the executor. The parent process is the only one
142 290b5baa 2021-02-23 op that doesn't drop privileges, but all it does is to wait for a SIGHUP
143 290b5baa 2021-02-23 op to reload the configuration and spawn a new generation of children
144 290b5baa 2021-02-23 op process. The logger processes gather the logs and prints 'em to
145 290b5baa 2021-02-23 op stderr or syslog (for the time being.) The listener process is the
146 290b5baa 2021-02-23 op only one that needs internet access and is sandboxed by default. The
147 290b5baa 2021-02-23 op executor process exists only to fork and execute CGI scripts.
149 601bc1cc 2021-01-30 op On OpenBSD, the listener runs with the `stdio recvfd rpath inet`
150 601bc1cc 2021-01-30 op pledges, while the executor has `stdio sendfd proc exec`; both have
151 290b5baa 2021-02-23 op unveiled only the served directories. The logger process has pledge
154 94be0bf0 2021-03-20 op On FreeBSD, the listener and logger process are sandboxed with `capsicum(4)`.
156 601bc1cc 2021-01-30 op On Linux, a `seccomp(2)` filter is installed in the listener to allow
157 601bc1cc 2021-01-30 op only certain syscalls, see [sandbox.c](sandbox.c) for more information
158 601bc1cc 2021-01-30 op on the BPF program.
160 ea58dab1 2021-01-17 op In any case, you are invited to run gmid inside some sort of
161 0b00962d 2021-01-25 op container/jail/chroot.