package main import ( "context" "embed" "html" "net/http" "os" "text/template" "time" "github.com/kelseyhightower/envconfig" "github.com/rs/cors" "github.com/rs/zerolog" ) type Settings struct { Port string `envconfig:"PORT" default:"2999"` DiskCachePath string `envconfig:"DISK_CACHE_PATH" default:"/tmp/njump-cache"` Domain string `envconfig:"DOMAIN" default:"njump.me"` } //go:embed static/* var static embed.FS //go:embed templates/* var templates embed.FS var ( s Settings tmpls *template.Template templateMapping map[string]string log = zerolog.New(os.Stderr).Output(zerolog.ConsoleWriter{Out: os.Stdout}).With().Timestamp().Logger() ) func updateArchives(ctx context.Context) { // do this so we don't run this every time we restart it locally time.Sleep(10 * time.Minute) for { select { case <-ctx.Done(): return default: loadNpubsArchive(ctx) loadRelaysArchive(ctx) } time.Sleep(24 * time.Hour) } } func main() { err := envconfig.Process("", &s) if err != nil { log.Fatal().Err(err).Msg("couldn't process envconfig") return } else { if canonicalHost := os.Getenv("CANONICAL_HOST"); canonicalHost != "" { s.Domain = canonicalHost } } // initialize disk cache defer cache.initialize()() // initialize the function to update the npubs/relays archive ctx := context.Background() go updateArchives(ctx) // initialize templates // use a mapping to expressly link the templates and share them between more kinds/types templateMapping = map[string]string{ "profile_sitemap": "sitemap.xml", "relay": "relay.html", "relay_sitemap": "sitemap.xml", "archive_sitemap": "sitemap.xml", } funcMap := template.FuncMap{ "s": func() Settings { return s }, "basicFormatting": func(input string) string { return basicFormatting(input, false, false) }, "previewNotesFormatting": previewNotesFormatting, "escapeString": html.EscapeString, "sanitizeXSS": sanitizeXSS, "trimProtocol": trimProtocol, "normalizeWebsiteURL": normalizeWebsiteURL, } tmpls = template.Must( template.New("tmpl"). Funcs(funcMap). ParseFS(templates, "templates/*"), ) // routes mux := http.NewServeMux() mux.Handle("/njump/static/", http.StripPrefix("/njump/", http.FileServer(http.FS(static)))) mux.HandleFunc("/relays-archive.xml", renderArchive) mux.HandleFunc("/npubs-archive.xml", renderArchive) mux.HandleFunc("/services/oembed", renderOEmbed) mux.HandleFunc("/relays-archive/", renderArchive) mux.HandleFunc("/npubs-archive/", renderArchive) mux.HandleFunc("/njump/image/", renderImage) mux.HandleFunc("/njump/proxy/", proxy) mux.HandleFunc("/favicon.ico", renderFavicon) mux.HandleFunc("/robots.txt", renderRobots) mux.HandleFunc("/try", renderTry) mux.HandleFunc("/", render) log.Print("listening at http://0.0.0.0:" + s.Port) if err := http.ListenAndServe("0.0.0.0:"+s.Port, cors.Default().Handler(mux)); err != nil { log.Fatal().Err(err).Msg("") } select {} }