Blame


1 b9220ca4 2021-01-11 op # gmid
2 3e4749f7 2020-10-02 op
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.
6 3e4749f7 2020-10-02 op
7 3e4749f7 2020-10-02 op
8 b9220ca4 2021-01-11 op ## Features
9 3e4749f7 2020-10-02 op
10 601bc1cc 2021-01-30 op (random order)
11 601bc1cc 2021-01-30 op
12 08ce6f52 2021-04-29 op - sandboxed by default on OpenBSD, Linux and FreeBSD
13 747b35d0 2021-02-08 op - reconfiguration: reload the running configuration without
14 747b35d0 2021-02-08 op interruption
15 290b5baa 2021-02-23 op - automatic redirect/error pages (see `block return`)
16 4d2ec6d7 2021-01-13 op - IRI support (RFC3987)
17 601bc1cc 2021-01-30 op - automatic certificate generation for config-less mode
18 a6c80955 2021-06-11 op - CGI and FastCGI support
19 08ce6f52 2021-04-29 op - virtual hosts
20 08ce6f52 2021-04-29 op - location rules
21 b9231167 2021-02-08 op - event-based asynchronous I/O model
22 08ce6f52 2021-04-29 op - low memory footprint
23 b9220ca4 2021-01-11 op - small codebase, easily hackable
24 3e4749f7 2020-10-02 op
25 fab952e1 2020-10-03 op
26 286c4f40 2021-01-27 op ## Internationalisation (IRIs, UNICODE, punycode, all that stuff)
27 286c4f40 2021-01-27 op
28 286c4f40 2021-01-27 op Even thought the current Gemini specification doesn't mention anything
29 601bc1cc 2021-01-30 op in this regard, I do think these are important things and so I tried
30 601bc1cc 2021-01-30 op to implement them in the most user-friendly way I could think of.
31 286c4f40 2021-01-27 op
32 601bc1cc 2021-01-30 op For starters, gmid has full support for IRI (RFC3987 —
33 22a742e4 2021-01-29 op Internationalized Resource Identifiers). IRIs are a superset of URIs,
34 286c4f40 2021-01-27 op so there aren't incompatibilities with URI-only clients.
35 286c4f40 2021-01-27 op
36 601bc1cc 2021-01-30 op There is full support also for punycode. In theory, the user doesn't
37 286c4f40 2021-01-27 op even need to know that punycode is a thing. The hostname in the
38 601bc1cc 2021-01-30 op configuration file can (and must be) in the decoded form (e.g. `naïve`
39 601bc1cc 2021-01-30 op and not `xn--nave-6pa`), gmid will do the rest.
40 286c4f40 2021-01-27 op
41 601bc1cc 2021-01-30 op The only missing piece is UNICODE normalisation of the IRI path: gmid
42 601bc1cc 2021-01-30 op doesn't do that (yet).
43 286c4f40 2021-01-27 op
44 286c4f40 2021-01-27 op
45 1487e11e 2021-02-06 op ## Configuration
46 1487e11e 2021-02-06 op
47 1487e11e 2021-02-06 op gmid has a rich configuration file, heavily inspired by OpenBSD'
48 08ce6f52 2021-04-29 op httpd, with every detail carefully documented in the manpage. Here's
49 08ce6f52 2021-04-29 op a minimal example of a config file:
50 1487e11e 2021-02-06 op
51 1487e11e 2021-02-06 op ```conf
52 08ce6f52 2021-04-29 op server "example.com" {
53 08ce6f52 2021-04-29 op cert "/path/to/cert.pem"
54 08ce6f52 2021-04-29 op key "/path/to/key.pem"
55 08ce6f52 2021-04-29 op root "/var/gemini/example.com"
56 08ce6f52 2021-04-29 op }
57 08ce6f52 2021-04-29 op ```
58 08ce6f52 2021-04-29 op
59 08ce6f52 2021-04-29 op and a slightly complex one
60 08ce6f52 2021-04-29 op
61 08ce6f52 2021-04-29 op ```conf
62 1487e11e 2021-02-06 op ipv6 on # enable ipv6
63 1487e11e 2021-02-06 op
64 1487e11e 2021-02-06 op server "example.com" {
65 08ce6f52 2021-04-29 op alias "foobar.com"
66 1487e11e 2021-02-06 op
67 08ce6f52 2021-04-29 op cert "/path/to/cert.pem"
68 7a4f7426 2021-04-29 op key "/path/to/key.pem"
69 08ce6f52 2021-04-29 op root "/var/gemini/example.com"
70 1487e11e 2021-02-06 op
71 08ce6f52 2021-04-29 op # lang for text/gemini files
72 08ce6f52 2021-04-29 op lang "it"
73 08ce6f52 2021-04-29 op
74 08ce6f52 2021-04-29 op # execute CGI scripts in /cgi/
75 8070ffa7 2021-04-29 op cgi "/cgi/*"
76 08ce6f52 2021-04-29 op
77 08ce6f52 2021-04-29 op # only for locations that matches /files/*
78 08ce6f52 2021-04-29 op location "/files/*" {
79 08ce6f52 2021-04-29 op # generate directory listings
80 08ce6f52 2021-04-29 op auto index on
81 08ce6f52 2021-04-29 op }
82 08ce6f52 2021-04-29 op
83 08ce6f52 2021-04-29 op location "/repo/*" {
84 08ce6f52 2021-04-29 op # change the index file name
85 08ce6f52 2021-04-29 op index "README.gmi"
86 08ce6f52 2021-04-29 op lang "en"
87 08ce6f52 2021-04-29 op }
88 1487e11e 2021-02-06 op }
89 1487e11e 2021-02-06 op ```
90 1487e11e 2021-02-06 op
91 1487e11e 2021-02-06 op
92 b9220ca4 2021-01-11 op ## Building
93 3e4749f7 2020-10-02 op
94 b9231167 2021-02-08 op gmid depends on a POSIX libc, libevent2, OpenSSL/LibreSSL and libtls
95 74f0778b 2021-06-16 op (provided either by LibreSSL or libretls). At build time, yacc (or
96 74f0778b 2021-06-16 op GNU bison) is also needed.
97 3e4749f7 2020-10-02 op
98 771d8f28 2021-01-17 op The build is as simple as
99 3e4749f7 2020-10-02 op
100 601bc1cc 2021-01-30 op ./configure
101 b9220ca4 2021-01-11 op make
102 3e4749f7 2020-10-02 op
103 2ddc9271 2021-03-29 op or `make static` to build a statically-linked executable.
104 2ddc9271 2021-03-29 op
105 8f0da580 2021-01-21 op If the configure scripts fails to pick up something, please open an
106 f980545b 2021-01-21 op issue or notify me via email.
107 8f0da580 2021-01-21 op
108 771d8f28 2021-01-17 op To install execute:
109 881a9dd9 2021-01-16 op
110 771d8f28 2021-01-17 op make install
111 771d8f28 2021-01-17 op
112 bb4be662 2021-04-25 op Please keep in mind that the master branch, from time to time, may be
113 bb4be662 2021-04-25 op accidentally broken on some platforms. gmid is developed primarily on
114 bb4be662 2021-04-25 op OpenBSD/amd64 and commits on the master branch don't get always tested
115 bb4be662 2021-04-25 op in other OSes. Before tagging a release however, a comprehensive
116 c79b63f5 2021-04-27 op testing on various platform is done to ensure that everything is
117 c79b63f5 2021-04-27 op working as intended.
118 bb4be662 2021-04-25 op
119 bb4be662 2021-04-25 op
120 601bc1cc 2021-01-30 op ### Docker
121 771d8f28 2021-01-17 op
122 601bc1cc 2021-01-30 op If you have trouble installing LibreSSL or libretls, you can use
123 601bc1cc 2021-01-30 op Docker to build a `gmid` image with:
124 601bc1cc 2021-01-30 op
125 17b09e3c 2021-01-18 op docker build -t gmid .
126 17b09e3c 2021-01-18 op
127 17b09e3c 2021-01-18 op and then run it with something along the lines of
128 17b09e3c 2021-01-18 op
129 17b09e3c 2021-01-18 op docker run --rm -it -p 1965:1965 \
130 0c7a1d3c 2021-06-04 op -v /path/to/gmid.conf:/etc/gmid.conf:ro \
131 17b09e3c 2021-01-18 op -v /path/to/docs:/var/gemini \
132 0c7a1d3c 2021-06-04 op gmid /bin/gmid -c /etc/gmid.conf
133 17b09e3c 2021-01-18 op
134 17b09e3c 2021-01-18 op
135 dd8cc7d3 2021-01-22 op ### Local libretls
136 17b09e3c 2021-01-18 op
137 8f0da580 2021-01-21 op This is **NOT** recommended, please try to port LibreSSL/LibreTLS to
138 8f0da580 2021-01-21 op your distribution of choice or use docker instead.
139 8f0da580 2021-01-21 op
140 601bc1cc 2021-01-30 op However, it's possible to statically-link `gmid` to locally-installed
141 601bc1cc 2021-01-30 op libretls quite easily. (It's how I test gmid on Fedora, for instance)
142 8f0da580 2021-01-21 op
143 04397b32 2021-01-21 op Let's say you have compiled and installed libretls in `$LIBRETLS`,
144 8f0da580 2021-01-21 op then you can build `gmid` with
145 8f0da580 2021-01-21 op
146 04397b32 2021-01-21 op ./configure CFLAGS="-I$LIBRETLS/include" \
147 1606927e 2021-02-11 op LDFLAGS="$LIBRETLS/lib/libtls.a -lssl -lcrypto -lpthread -levent"
148 8f0da580 2021-01-21 op make
149 8f0da580 2021-01-21 op
150 dd8cc7d3 2021-01-22 op ### Testing
151 dd8cc7d3 2021-01-22 op
152 31a4993a 2021-01-23 op Execute
153 dd8cc7d3 2021-01-22 op
154 dd8cc7d3 2021-01-22 op make regress
155 dd8cc7d3 2021-01-22 op
156 601bc1cc 2021-01-30 op to start the suite. Keep in mind that the regression tests will
157 601bc1cc 2021-01-30 op create files inside the `regress` directory and bind the 10965 port.
158 dd8cc7d3 2021-01-22 op
159 dd8cc7d3 2021-01-22 op
160 881a9dd9 2021-01-16 op ## Architecture/Security considerations
161 881a9dd9 2021-01-16 op
162 290b5baa 2021-02-23 op gmid is composed by four processes: the parent process, the logger,
163 290b5baa 2021-02-23 op the listener and the executor. The parent process is the only one
164 290b5baa 2021-02-23 op that doesn't drop privileges, but all it does is to wait for a SIGHUP
165 290b5baa 2021-02-23 op to reload the configuration and spawn a new generation of children
166 290b5baa 2021-02-23 op process. The logger processes gather the logs and prints 'em to
167 290b5baa 2021-02-23 op stderr or syslog (for the time being.) The listener process is the
168 290b5baa 2021-02-23 op only one that needs internet access and is sandboxed by default. The
169 290b5baa 2021-02-23 op executor process exists only to fork and execute CGI scripts.
170 881a9dd9 2021-01-16 op
171 601bc1cc 2021-01-30 op On OpenBSD, the listener runs with the `stdio recvfd rpath inet`
172 601bc1cc 2021-01-30 op pledges, while the executor has `stdio sendfd proc exec`; both have
173 290b5baa 2021-02-23 op unveiled only the served directories. The logger process has pledge
174 290b5baa 2021-02-23 op `stdio`.
175 881a9dd9 2021-01-16 op
176 94be0bf0 2021-03-20 op On FreeBSD, the listener and logger process are sandboxed with `capsicum(4)`.
177 d939d0f0 2021-01-17 op
178 601bc1cc 2021-01-30 op On Linux, a `seccomp(2)` filter is installed in the listener to allow
179 601bc1cc 2021-01-30 op only certain syscalls, see [sandbox.c](sandbox.c) for more information
180 601bc1cc 2021-01-30 op on the BPF program.
181 ea58dab1 2021-01-17 op
182 6957a8c2 2021-06-04 op In any case, it's advisable to run gmid inside some sort of
183 0b00962d 2021-01-25 op container/jail/chroot.