From 4f01edb9230f58ff84b0dd892c931ec8ac9aad55 Mon Sep 17 00:00:00 2001 From: Brian Picciano Date: Tue, 13 Sep 2022 12:56:08 +0200 Subject: move src out of srv, clean up default.nix and Makefile --- src/http/tpl.go | 167 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 167 insertions(+) create mode 100644 src/http/tpl.go (limited to 'src/http/tpl.go') diff --git a/src/http/tpl.go b/src/http/tpl.go new file mode 100644 index 0000000..3e1a2ba --- /dev/null +++ b/src/http/tpl.go @@ -0,0 +1,167 @@ +package http + +import ( + "embed" + "fmt" + "html/template" + "io/fs" + "net/http" + "path/filepath" + "strings" + "time" + + "github.com/mediocregopher/blog.mediocregopher.com/srv/http/apiutil" +) + +//go:embed tpl +var tplFS embed.FS + +func mustReadTplFile(fileName string) string { + path := filepath.Join("tpl", fileName) + + b, err := fs.ReadFile(tplFS, path) + if err != nil { + panic(fmt.Errorf("reading file %q from tplFS: %w", path, err)) + } + + return string(b) +} + +func (a *api) blogURL(path string, abs bool) string { + // filepath.Join strips trailing slash, but we want to keep it + trailingSlash := strings.HasSuffix(path, "/") + + res := filepath.Join("/", path) + + if trailingSlash && res != "/" { + res += "/" + } + + if abs { + res = a.params.PublicURL.String() + res + } + + return res +} + +func (a *api) postURL(id string, abs bool) string { + path := filepath.Join("posts", id) + return a.blogURL(path, abs) +} + +func (a *api) postsURL(abs bool) string { + return a.blogURL("posts", abs) +} + +func (a *api) assetsURL(abs bool) string { + return a.blogURL("assets", abs) +} + +func (a *api) draftURL(id string, abs bool) string { + path := filepath.Join("drafts", id) + return a.blogURL(path, abs) +} + +func (a *api) draftsURL(abs bool) string { + return a.blogURL("drafts", abs) +} + +func (a *api) tplFuncs() template.FuncMap { + return template.FuncMap{ + "BlogURL": func(path string) string { + return a.blogURL(path, false) + }, + "StaticURL": func(path string) string { + path = filepath.Join("static", path) + return a.blogURL(path, false) + }, + "StaticInlineCSS": func(path string) (template.CSS, error) { + path = filepath.Join("static", path) + b, err := staticFS.ReadFile(path) + return template.CSS(b), err + }, + "PostURL": func(id string) string { + return a.postURL(id, false) + }, + "AssetURL": func(id string) string { + path := filepath.Join("assets", id) + return a.blogURL(path, false) + }, + "DraftURL": func(id string) string { + return a.draftURL(id, false) + }, + "DateTimeFormat": func(t time.Time) string { + return t.Format("2006-01-02") + }, + } +} + +func (a *api) parseTpl(name, tplBody string) (*template.Template, error) { + + tpl := template.New(name) + tpl = tpl.Funcs(a.tplFuncs()) + + var err error + + if tpl, err = tpl.Parse(tplBody); err != nil { + return nil, err + } + + return tpl, nil +} + +func (a *api) mustParseTpl(name string) *template.Template { + return template.Must(a.parseTpl(name, mustReadTplFile(name))) +} + +func (a *api) mustParseBasedTpl(name string) *template.Template { + tpl := a.mustParseTpl(name) + tpl = template.Must(tpl.New("base.html").Parse(mustReadTplFile("base.html"))) + return tpl +} + +type tplData struct { + Payload interface{} +} + +// executeTemplate expects to be the final action in an http.Handler +func executeTemplate( + rw http.ResponseWriter, r *http.Request, + tpl *template.Template, payload interface{}, +) { + + tplData := tplData{ + Payload: payload, + } + + if err := tpl.Execute(rw, tplData); err != nil { + apiutil.InternalServerError( + rw, r, fmt.Errorf("rendering template: %w", err), + ) + return + } +} + +func (a *api) executeRedirectTpl( + rw http.ResponseWriter, r *http.Request, url string, +) { + executeTemplate(rw, r, a.redirectTpl, struct { + URL string + }{ + URL: url, + }) +} + +func (a *api) renderDumbTplHandler(tplName string) http.Handler { + + tpl := a.mustParseBasedTpl(tplName) + + return http.HandlerFunc(func(rw http.ResponseWriter, r *http.Request) { + if err := tpl.Execute(rw, nil); err != nil { + apiutil.InternalServerError( + rw, r, fmt.Errorf("rendering %q: %w", tplName, err), + ) + return + } + }) +} -- cgit v1.2.3