go-gemini/examples/client.go

154 lines
3.0 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 client.
2020-09-21 21:23:51 +00:00
package main
import (
"bufio"
2020-11-01 02:50:42 +00:00
"crypto/x509"
2020-12-18 00:50:26 +00:00
"errors"
2020-09-21 21:23:51 +00:00
"fmt"
2020-10-27 23:16:55 +00:00
"io/ioutil"
"log"
"net/url"
2020-09-21 21:23:51 +00:00
"os"
2020-11-09 17:04:53 +00:00
"path/filepath"
2020-09-28 03:49:41 +00:00
"time"
2020-10-28 02:12:10 +00:00
"git.sr.ht/~adnano/go-gemini"
2021-01-10 21:44:32 +00:00
"git.sr.ht/~adnano/go-gemini/tofu"
2020-11-09 17:04:53 +00:00
"git.sr.ht/~adnano/go-xdg"
2020-09-21 21:23:51 +00:00
)
var (
2021-01-10 21:44:32 +00:00
hosts tofu.KnownHostsFile
scanner *bufio.Scanner
)
func init() {
// Load known hosts file
path := filepath.Join(xdg.DataHome(), "gemini", "known_hosts")
err := hosts.Load(path)
if err != nil {
log.Println(err)
}
scanner = bufio.NewScanner(os.Stdin)
}
const trustPrompt = `The certificate offered by %s is of unknown trust. Its fingerprint is:
%s
If you knew the fingerprint to expect in advance, verify that this matches.
Otherwise, this should be safe to trust.
[t]rust always; trust [o]nce; [a]bort
=> `
func trustCertificate(hostname string, cert *x509.Certificate) error {
2021-01-10 21:44:32 +00:00
fingerprint := tofu.NewFingerprint(cert.Raw, cert.NotAfter)
knownHost, ok := hosts.Lookup(hostname)
if ok && time.Now().Before(knownHost.Expires) {
2020-12-19 18:44:33 +00:00
// Check fingerprint
if knownHost.Hex == fingerprint.Hex {
return nil
}
return errors.New("error: fingerprint does not match!")
2020-12-18 00:50:26 +00:00
}
fmt.Printf(trustPrompt, hostname, fingerprint.Hex)
scanner.Scan()
switch scanner.Text() {
case "t":
hosts.Add(hostname, fingerprint)
hosts.Write(hostname, fingerprint)
return nil
case "o":
hosts.Add(hostname, fingerprint)
return nil
default:
return errors.New("certificate not trusted")
2020-12-18 00:50:26 +00:00
}
}
2020-12-18 00:50:26 +00:00
func getInput(prompt string, sensitive bool) (input string, ok bool) {
fmt.Printf("%s ", prompt)
scanner.Scan()
return scanner.Text(), true
}
2020-12-18 00:50:26 +00:00
func do(req *gemini.Request, via []*gemini.Request) (*gemini.Response, error) {
client := gemini.Client{
TrustCertificate: trustCertificate,
}
resp, err := client.Do(req)
if err != nil {
return resp, err
}
switch resp.Status.Class() {
case gemini.StatusClassInput:
input, ok := getInput(resp.Meta, resp.Status == gemini.StatusSensitiveInput)
if !ok {
break
}
req.URL.ForceQuery = true
req.URL.RawQuery = gemini.QueryEscape(input)
return do(req, via)
case gemini.StatusClassRedirect:
via = append(via, req)
if len(via) > 5 {
return resp, errors.New("too many redirects")
2020-12-18 00:50:26 +00:00
}
2020-09-21 21:23:51 +00:00
target, err := url.Parse(resp.Meta)
if err != nil {
return resp, err
2020-11-01 02:50:42 +00:00
}
target = req.URL.ResolveReference(target)
redirect := *req
redirect.URL = target
return do(&redirect, via)
2020-11-01 02:50:42 +00:00
}
return resp, err
}
func main() {
if len(os.Args) < 2 {
fmt.Printf("usage: %s <url> [host]\n", os.Args[0])
os.Exit(1)
}
2020-09-27 23:45:48 +00:00
2020-12-18 00:50:26 +00:00
// Do the request
2020-09-27 23:45:48 +00:00
url := os.Args[1]
2020-10-28 02:12:10 +00:00
req, err := gemini.NewRequest(url)
2020-09-27 23:45:48 +00:00
if err != nil {
fmt.Println(err)
os.Exit(1)
}
if len(os.Args) == 3 {
req.Host = os.Args[2]
}
resp, err := do(req, nil)
if err != nil {
fmt.Println(err)
os.Exit(1)
}
2020-12-18 00:50:26 +00:00
// Handle response
if resp.Status.Class() == gemini.StatusClassSuccess {
defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
log.Fatal(err)
}
fmt.Print(string(body))
} else {
2021-01-10 05:10:57 +00:00
fmt.Printf("%d %s\n", resp.Status, resp.Meta)
2020-12-18 00:50:26 +00:00
os.Exit(1)
2020-09-21 21:23:51 +00:00
}
}