package main import ( "context" "flag" "fmt" "google.golang.org/grpc/credentials/insecure" "log" "regexp" "strings" "time" "git.sr.ht/~adnano/go-gemini" "git.sr.ht/~adnano/go-gemini/certificate" "github.com/flosch/pongo2/v6" "google.golang.org/grpc" ) var ( hostname string port string fileSrvHost string fileSrvPort string certificatePath string key = "WgiPGpt5wM6SVEWo5iqI" ) func init() { flag.StringVar(&hostname, "hostname", "", "capsule hostname") flag.StringVar(&port, "port", "1965", "capsule port") flag.StringVar(&certificatePath, "certificatePath", "", "capsule certificate path") flag.StringVar(&fileSrvHost, "fileSrvHost", "", "file push server host") flag.StringVar(&fileSrvPort, "fileSrvPort", "", "file push server port") flag.StringVar(&key, "key", "", "secret key to communicate admin") flag.Parse() } func main() { certificates := &certificate.Store{} certificates.Register(hostname) if err := certificates.Load(certificatePath); err != nil { panic(err) } mux := &gemini.Mux{} mux.HandleFunc("/favicon.txt", processFavicon) mux.HandleFunc("/rebuild", processRebuild) // security // feed mux.HandleFunc("/", process) server := &gemini.Server{ Addr: ":" + port, Handler: mux, ReadTimeout: 30 * time.Second, WriteTimeout: 1 * time.Minute, GetCertificate: certificates.Get, } ctx := context.Background() if err := server.ListenAndServe(ctx); err != nil { log.Fatal(err) } } func processRebuild(ctx context.Context, w gemini.ResponseWriter, r *gemini.Request) { q, err := gemini.QueryUnescape(r.URL.RawQuery) if err != nil || q != key { w.WriteHeader(gemini.StatusServerUnavailable, "Don't wake up the dragons :-*") } conn, err := grpc.Dial(fileSrvHost+":"+fileSrvPort, grpc.WithTransportCredentials(insecure.NewCredentials())) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") log.Fatal(err) return } defer conn.Close() client := NewTreeManagerClient(conn) ts := TreeSecret{Key: key} rsp, err := client.RebuildTree(context.Background(), &ts) if rsp.Success { _, err = w.Write([]byte(":-)")) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") } } else { w.WriteHeader(gemini.StatusNotFound, "Out of space") } } func processFavicon(_ context.Context, w gemini.ResponseWriter, _ *gemini.Request) { w.SetMediaType("text/plain") _, err := w.Write([]byte("\U0001F31B")) if err != nil { w.WriteHeader(gemini.StatusNotFound, "Out of space") } } func process(_ context.Context, w gemini.ResponseWriter, r *gemini.Request) { log.Println("-> " + r.URL.Path) lang := regexp.MustCompile(`^/(sgs|en)`).FindString(r.URL.Path) if lang != "" { lang = lang[1:] } conn, err := grpc.Dial(fileSrvHost+":"+fileSrvPort, grpc.WithTransportCredentials(insecure.NewCredentials())) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") log.Fatal(err) return } defer conn.Close() client := NewTreeManagerClient(conn) switch { case "/" == r.URL.Path: w.WriteHeader(gemini.StatusPermanentRedirect, "/sgs") case regexp.MustCompile(`^/(sgs|en)/?$`).MatchString(r.URL.Path): renderIndex(lang, w, client) case regexp.MustCompile(`^/(sgs|en)/a/?$`).MatchString(r.URL.Path): renderAbout(lang, w) case regexp.MustCompile(`^/(sgs|en)/s/?$`).MatchString(r.URL.Path): renderSearch(lang, w, r, client) case regexp.MustCompile(`^/(sgs|en)/f/?$`).MatchString(r.URL.Path): renderAllFiles(lang, w, client) case regexp.MustCompile(`^/(sgs|en)/f/([\p{L}\d_+.]+/)+[\d\w]+/[\p{L}\d_+.]+.gmi$`).MatchString(r.URL.Path): renderFile(lang, w, r, client) case regexp.MustCompile(`^/(sgs|en)/f/[\p{L}\d_+.]+(/[\p{L}\d_+.]+)*/?$`).MatchString(r.URL.Path): renderCategory(lang, w, r, client) case regexp.MustCompile(`^/(sgs|en)/t/[\p{L}\d_+.]+/?$`).MatchString(r.URL.Path): renderTag(lang, w, r, client) // case abruozdelee default: w.WriteHeader(gemini.StatusNotFound, "Out of space") } } func renderIndex(lang string, w gemini.ResponseWriter, client TreeManagerClient) { langFilter := TreeRequest_Filter{Key: "lang", Value: lang} filters := []*TreeRequest_Filter{&langFilter} path := "" tr := TreeRequest{Path: &path, Filter: filters} tree, err := client.GetSummery(context.Background(), &tr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile(fmt.Sprintf("templates/%s/index.gmi", lang))) page, err := tpl.Execute(pongo2.Context{"tree": tree, "lang": lang, "lastFiles": GetLastFiles(tree.Files)}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } func renderAbout(lang string, w gemini.ResponseWriter) { w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile(fmt.Sprintf("templates/%s/about.gmi", lang))) page, err := tpl.Execute(pongo2.Context{}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") } } func renderSearch(lang string, w gemini.ResponseWriter, r *gemini.Request, client TreeManagerClient) { q, err := gemini.QueryUnescape(r.URL.RawQuery) if err != nil || q == "" { searchStr := "Input searching tag, created date or word in description" if lang == "sgs" { searchStr = "Ivesk ėiškuoma žīma, sokūrėma data arba žuodi ėš aprašīma" } w.WriteHeader(gemini.StatusInput, searchStr) return } descFilter := TreeRequest_Filter{Key: "description", Value: q} createdFilter := TreeRequest_Filter{Key: "created", Value: q} tagFilter := TreeRequest_Filter{Key: "tag", Value: q} filters := []*TreeRequest_Filter{&descFilter, &createdFilter, &tagFilter} path := "" tr := TreeRequest{Path: &path, Filter: filters} tree, err := client.GetSummery(context.Background(), &tr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile("templates/search.gmi")) page, err := tpl.Execute(pongo2.Context{"lang": lang, "tree": tree, "q": q}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } func renderCategory(lang string, w gemini.ResponseWriter, r *gemini.Request, client TreeManagerClient) { urlParts := strings.Split(strings.Trim(r.URL.Path, "/"), "/") path := "/" + strings.Join(urlParts[2:], "/") tr := TreeRequest{Path: &path, Filter: []*TreeRequest_Filter{}} tree, err := client.GetSummery(context.Background(), &tr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } indexFile, err := tree.GetIndexFile() var file *FileContent if err == nil { fr := FileRequest{Id: indexFile.Id} file, err = client.GetFile(context.Background(), &fr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile("templates/category.gmi")) page, err := tpl.Execute(pongo2.Context{"lang": lang, "tree": tree, "path": path, "indexFile": file}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } func renderFile(lang string, w gemini.ResponseWriter, r *gemini.Request, client TreeManagerClient) { urlParts := strings.Split(r.URL.Path, "/") id := urlParts[len(urlParts)-2] fr := FileRequest{Id: id} file, err := client.GetFile(context.Background(), &fr) if err != nil { log.Fatalf("client.GetSummery failed: %v", err) w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile("templates/page.gmi")) page, err := tpl.Execute(pongo2.Context{"lang": lang, "file": file}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } func renderAllFiles(lang string, w gemini.ResponseWriter, client TreeManagerClient) { langFilter := TreeRequest_Filter{Key: "lang", Value: lang} filters := []*TreeRequest_Filter{&langFilter} path := "" tr := TreeRequest{Path: &path, Filter: filters} tree, err := client.GetSummery(context.Background(), &tr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile("templates/all_texts.gmi")) page, err := tpl.Execute(pongo2.Context{"lang": lang, "tree": tree}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } } func renderTag(lang string, w gemini.ResponseWriter, r *gemini.Request, client TreeManagerClient) { urlParts := strings.Split(strings.Trim(r.URL.Path, "/"), "/") tag := urlParts[len(urlParts)-1] tagFilter := TreeRequest_Filter{Key: "tag", Value: tag} filters := []*TreeRequest_Filter{&tagFilter} path := "" tr := TreeRequest{Path: &path, Filter: filters} tree, err := client.GetSummery(context.Background(), &tr) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } w.SetMediaType("text/gemini") tpl := pongo2.Must(pongo2.FromFile("templates/tag.gmi")) page, err := tpl.Execute(pongo2.Context{"lang": lang, "tree": tree, "tag": tag}) if err != nil { log.Fatalf("template failed: %v", err) return } _, err = w.Write([]byte(page)) if err != nil { w.WriteHeader(gemini.StatusTemporaryFailure, "Internal server error") return } }