go-gemini/examples/server.go

109 lines
2.2 KiB
Go
Raw Normal View History

2020-10-12 20:34:52 +00:00
// +build ignore
2020-09-21 21:23:51 +00:00
2020-12-18 05:47:30 +00:00
// This example illustrates a Gemini server.
2020-09-21 21:23:51 +00:00
package main
import (
2021-02-21 05:26:30 +00:00
"context"
2020-09-21 21:23:51 +00:00
"log"
2021-02-17 14:25:44 +00:00
"os"
"os/signal"
"time"
2020-10-28 18:59:45 +00:00
"git.sr.ht/~adnano/go-gemini"
2021-01-15 02:23:13 +00:00
"git.sr.ht/~adnano/go-gemini/certificate"
2020-09-21 21:23:51 +00:00
)
func main() {
2021-02-20 23:30:49 +00:00
certificates := &certificate.Store{}
2021-02-19 23:45:19 +00:00
certificates.Register("localhost")
if err := certificates.Load("/var/lib/gemini/certs"); err != nil {
log.Fatal(err)
}
2020-10-12 20:34:52 +00:00
2021-02-19 23:45:19 +00:00
mux := &gemini.ServeMux{}
mux.Handle("/", gemini.FileServer(os.DirFS("/var/www")))
server := &gemini.Server{
Handler: logMiddleware(mux),
2021-02-19 23:45:19 +00:00
ReadTimeout: 30 * time.Second,
WriteTimeout: 1 * time.Minute,
2021-02-23 13:43:41 +00:00
GetCertificate: certificates.Get,
2021-02-19 23:45:19 +00:00
}
2020-09-21 21:23:51 +00:00
// Listen for interrupt signal
c := make(chan os.Signal, 1)
signal.Notify(c, os.Interrupt)
errch := make(chan error)
go func() {
ctx := context.Background()
errch <- server.ListenAndServe(ctx)
}()
select {
case err := <-errch:
2020-10-12 04:13:24 +00:00
log.Fatal(err)
case <-c:
// Shutdown the server
log.Println("Shutting down...")
2021-02-24 15:27:28 +00:00
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
defer cancel()
err := server.Shutdown(ctx)
if err != nil {
log.Fatal(err)
}
2020-10-12 04:13:24 +00:00
}
2020-09-21 21:23:51 +00:00
}
func logMiddleware(h gemini.Handler) gemini.Handler {
return gemini.HandlerFunc(func(ctx context.Context, w gemini.ResponseWriter, r *gemini.Request) {
lw := &logResponseWriter{rw: w}
h.ServeGemini(ctx, lw, r)
host := r.TLS().ServerName
log.Printf("gemini: %s %q %d %d", host, r.URL, lw.status, lw.wrote)
})
}
type logResponseWriter struct {
rw gemini.ResponseWriter
status gemini.Status
meta string
mediatype string
wroteHeader bool
wrote int
}
func (w *logResponseWriter) SetMediaType(mediatype string) {
w.mediatype = mediatype
}
func (w *logResponseWriter) Write(b []byte) (int, error) {
if !w.wroteHeader {
w.WriteHeader(gemini.StatusSuccess, w.mediatype)
}
n, err := w.rw.Write(b)
w.wrote += n
return n, err
}
func (w *logResponseWriter) WriteHeader(status gemini.Status, meta string) {
if w.wroteHeader {
return
}
w.status = status
w.meta = meta
w.wroteHeader = true
w.rw.WriteHeader(status, meta)
w.wrote += len(meta) + 5
}
func (w *logResponseWriter) Flush() error {
return nil
}
func (w *logResponseWriter) Close() error {
return nil
}