package main import ( "fmt" "html/template" "io" "log" "mime" "net/http" "net/url" "os" "strings" "time" "git.sr.ht/~adnano/go-gemini" "git.sr.ht/~sircmpwn/getopt" ) var gemtextPage = template.Must(template. New("gemtext"). Funcs(template.FuncMap{ "heading": func(line gemini.Line) *GemtextHeading { switch l := line.(type) { case gemini.LineHeading1: return &GemtextHeading{1, string(l)} case gemini.LineHeading2: return &GemtextHeading{2, string(l)} case gemini.LineHeading3: return &GemtextHeading{3, string(l)} default: return nil } }, "link": func(line gemini.Line) *gemini.LineLink { switch l := line.(type) { case gemini.LineLink: return &l default: return nil } }, "li": func(line gemini.Line) *gemini.LineListItem { switch l := line.(type) { case gemini.LineListItem: return &l default: return nil } }, "pre_toggle_on": func(ctx *GemtextContext, line gemini.Line) *gemini.LinePreformattingToggle { switch l := line.(type) { case gemini.LinePreformattingToggle: if ctx.Pre%4 == 0 { ctx.Pre += 1 return &l } ctx.Pre += 1 return nil default: return nil } }, "pre_toggle_off": func(ctx *GemtextContext, line gemini.Line) *gemini.LinePreformattingToggle { switch l := line.(type) { case gemini.LinePreformattingToggle: if ctx.Pre%4 == 3 { ctx.Pre += 1 return &l } ctx.Pre += 1 return nil default: return nil } }, "pre": func(line gemini.Line) *gemini.LinePreformattedText { switch l := line.(type) { case gemini.LinePreformattedText: return &l default: return nil } }, "quote": func(line gemini.Line) *gemini.LineQuote { switch l := line.(type) { case gemini.LineQuote: return &l default: return nil } }, "text": func(line gemini.Line) *gemini.LineText { switch l := line.(type) { case gemini.LineText: return &l default: return nil } }, "url": func(ctx *GemtextContext, s string) template.URL { u, err := url.Parse(s) if err != nil { return template.URL("error") } u = ctx.URL.ResolveReference(u) if u.Scheme == "" || u.Scheme == "gemini" { if u.Host != ctx.Root.Host { u.Path = fmt.Sprintf("/x/%s%s", u.Host, u.Path) } u.Scheme = "" u.Host = "" } return template.URL(u.String()) }, "safeCSS": func(s string) template.CSS { return template.CSS(s) }, "safeURL": func(s string) template.URL { return template.URL(s) }, }). Parse(` {{- if .CSS }} {{- end }} {{.Title}}
{{ $ctx := . -}} {{- $isList := false -}} {{- range .Lines -}} {{- if and $isList (not (. | li)) }} {{- $isList = false -}} {{- end -}} {{- with . | heading }} {{- $isList = false -}} {{.Text}} {{- end -}} {{- with . | link }} {{- $isList = false -}}

{{if .Name}}{{.Name}}{{else}}{{.URL}}{{end}} {{- end -}} {{- with . | quote }} {{- $isList = false -}}

{{slice .String 1}}
{{- end -}} {{- with . | pre_toggle_on $ctx }}
{{- end -}} {{- with . | text }} {{- $isList = false }}

{{.}} {{- end -}} {{- with . | li }} {{- if not $isList }}

{{- end }}
Proxied content from {{.URL.String}} {{if .External}} (external content) {{end}}

Gemini request details:

Original URL
{{.URL.String}}
Status code
{{.Resp.Status}}
Meta
{{.Resp.Meta}}
Proxied by
kineto

Be advised that no attempt was made to verify the remote SSL certificate.

`)) // TODO: let user customize this const defaultCSS = `html { font-family: sans-serif; color: #080808; } body { max-width: 920px; margin: 0 auto; padding: 1rem 2rem; } blockquote { background-color: #eee; border-left: 3px solid #444; margin: 1rem -1rem 1rem calc(-1rem - 3px); padding: 1rem; } ul { margin-left: 0; padding: 0; } li { padding: 0; } li:not(:last-child) { margin-bottom: 0.5rem; } a { position: relative; } a:before { content: '⇒'; color: #999; text-decoration: none; font-weight: bold; position: absolute; left: -1.25rem; } pre { background-color: #eee; margin: 0 -1rem; padding: 1rem; overflow-x: auto; } details:not([open]) summary, details:not([open]) summary a { color: gray; } details summary a:before { display: none; } dl dt { font-weight: bold; } dl dt:not(:first-child) { margin-top: 0.5rem; } @media(prefers-color-scheme:dark) { html { background-color: #111; color: #eee; } blockquote { background-color: #000; } pre { background-color: #222; } a { color: #0087BD; } a:visited { color: #333399; } } ` type GemtextContext struct { CSS string External bool Lines []gemini.Line Pre int Resp *gemini.Response Title string URL *url.URL Root *url.URL } type GemtextHeading struct { Level int Text string } func proxyGemini(req gemini.Request, external bool, root *url.URL, w http.ResponseWriter, r *http.Request) { client := gemini.Client{ Timeout: 30 * time.Second, InsecureSkipTrust: true, } if h := (url.URL{Host: req.Host}); h.Port() == "" { req.Host += ":1965" } resp, err := client.Do(&req) if err != nil { w.WriteHeader(http.StatusBadGateway) w.Write([]byte(fmt.Sprintf("Gateway error: %v", err))) return } defer resp.Body.Close() switch resp.Status { case 20: break // OK case 30: case 31: w.WriteHeader(http.StatusNotImplemented) w.Write([]byte(fmt.Sprintf("This URL redirects to %s", resp.Meta))) return case 40: case 41: case 42: case 43: case 44: w.WriteHeader(http.StatusServiceUnavailable) w.Write([]byte(fmt.Sprintf("The remote server returned %d: %s", resp.Status, resp.Meta))) return case 50: case 51: w.WriteHeader(http.StatusNotFound) w.Write([]byte(fmt.Sprintf("The remote server returned %d: %s", resp.Status, resp.Meta))) return case 52: case 53: case 59: w.WriteHeader(http.StatusServiceUnavailable) w.Write([]byte(fmt.Sprintf("The remote server returned %d: %s", resp.Status, resp.Meta))) return default: w.WriteHeader(http.StatusNotImplemented) w.Write([]byte(fmt.Sprintf("Proxy does not understand Gemini response status %d", resp.Status))) return } // XXX: We could use the params I guess m, _, err := mime.ParseMediaType(resp.Meta) if err != nil { w.WriteHeader(http.StatusBadGateway) w.Write([]byte(fmt.Sprintf("Gateway error: %v", err))) return } if m != "text/gemini" { w.Header().Add("Content-Type", resp.Meta) io.Copy(w, resp.Body) return } w.Header().Add("Content-Type", "text/html") ctx := &GemtextContext{ CSS: defaultCSS, External: external, Resp: resp, Title: req.URL.Host + " " + req.URL.Path, URL: req.URL, Root: root, } var title bool gemini.ParseLines(resp.Body, func(line gemini.Line) { ctx.Lines = append(ctx.Lines, line) if !title { if h, ok := line.(gemini.LineHeading1); ok { ctx.Title = string(h) title = true } } }) err = gemtextPage.Execute(w, ctx) if err != nil { w.WriteHeader(http.StatusInternalServerError) w.Write([]byte(fmt.Sprintf("%v", err))) return } } func main() { var ( bind string = ":8080" ) opts, optind, err := getopt.Getopts(os.Args, "b:c:") if err != nil { log.Fatal(err) } for _, opt := range opts { switch opt.Option { case 'b': bind = opt.Value } } args := os.Args[optind:] if len(args) != 1 { log.Fatalf("Usage: %s ", os.Args[0]) } root, err := url.Parse(args[0]) if err != nil { log.Fatal(err) } http.Handle("/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { log.Printf("%s %s", r.Method, r.URL.Path) if r.Method != "GET" { w.WriteHeader(http.StatusMethodNotAllowed) w.Write([]byte("404 Not found")) return } if r.URL.Path == "/favicon.ico" { w.WriteHeader(http.StatusNotFound) w.Write([]byte("404 Not found")) return } req := gemini.Request{} req.URL = &url.URL{} req.URL.Scheme = root.Scheme req.URL.Host = root.Host req.URL.Path = r.URL.Path req.Host = root.Host proxyGemini(req, false, root, w, r) })) http.Handle("/x/", http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { if r.Method != "GET" { w.WriteHeader(http.StatusMethodNotAllowed) w.Write([]byte("404 Not found")) return } path := strings.SplitN(r.URL.Path, "/", 4) if len(path) != 4 { path = append(path, "") } req := gemini.Request{} req.URL, err = url.Parse(fmt.Sprintf("gemini://%s/%s", path[2], path[3])) if err != nil { w.WriteHeader(http.StatusInternalServerError) w.Write([]byte(fmt.Sprintf("Error: %v", err))) return } req.Host = path[2] log.Printf("%s (external) %s%s", r.Method, path[2], path[3]) proxyGemini(req, true, root, w, r) })) log.Printf("HTTP server listening on %s", bind) log.Fatal(http.ListenAndServe(bind, nil)) }