datastar/backends/go/site/router.go

156 lines
3.7 KiB
Go

package site
import (
"bytes"
"context"
"embed"
"errors"
"fmt"
"io"
"net/http"
"github.com/a-h/templ"
"github.com/alecthomas/chroma"
"github.com/alecthomas/chroma/formatters/html"
"github.com/alecthomas/chroma/lexers"
"github.com/alecthomas/chroma/styles"
"github.com/benbjohnson/hashfs"
"github.com/delaneyj/toolbelt"
"github.com/go-chi/chi/v5"
"github.com/go-chi/chi/v5/middleware"
"github.com/gomarkdown/markdown/ast"
mdhtml "github.com/gomarkdown/markdown/html"
)
//go:embed static/*
var staticFS embed.FS
var (
staticSys = hashfs.NewFS(staticFS)
highlightCSS templ.Component
mdRenderer func() *mdhtml.Renderer
)
func staticPath(path string) string {
return "/" + staticSys.HashName("static/"+path)
}
func RunBlocking(port int) toolbelt.CtxErrFunc {
upsertIIfeBuildSize()
return func(ctx context.Context) error {
b, err := staticFS.ReadFile("static/library/package.json")
if err != nil {
return fmt.Errorf("error reading package.json: %w", err)
}
packageJSON, err = UnmarshalPackageJSON(b)
if err != nil {
return fmt.Errorf("error unmarshaling package.json: %w", err)
}
router := chi.NewRouter()
router.Use(
middleware.Logger,
middleware.Recoverer,
// toolbelt.CompressMiddleware(),
)
setupRoutes(router)
srv := &http.Server{
Addr: fmt.Sprintf(":%d", port),
Handler: router,
}
go func() {
<-ctx.Done()
srv.Shutdown(context.Background())
}()
return srv.ListenAndServe()
}
}
func setupRoutes(router chi.Router) error {
defer router.Handle("/static/*", hashfs.FileServer(staticSys))
// defer router.Get("/hotreload", func(w http.ResponseWriter, r *http.Request) {
// sse := datastar.NewSSE(w, r)
// <-r.Context().Done()
// sse.Send("reload", datastar.WithSSERetry(250))
// })
htmlFormatter := html.New(html.WithClasses(true), html.TabWidth(2))
if htmlFormatter == nil {
return fmt.Errorf("couldn't create html formatter")
}
styleName := "nord"
highlightStyle := styles.Get(styleName)
if highlightStyle == nil {
return fmt.Errorf("couldn't find style %s", styleName)
}
highlightCSSBuffer := &bytes.Buffer{}
if err := htmlFormatter.WriteCSS(highlightCSSBuffer, highlightStyle); err != nil {
return fmt.Errorf("error writing highlight css: %w", err)
}
highlightCSS = templ.ComponentFunc(func(ctx context.Context, w io.Writer) error {
_, err := io.WriteString(w, fmt.Sprintf(`<style>%s</style>`, highlightCSSBuffer.String()))
return err
})
mdRenderer = func() *mdhtml.Renderer {
return mdhtml.NewRenderer(mdhtml.RendererOptions{
Flags: mdhtml.CommonFlags | mdhtml.HrefTargetBlank,
RenderNodeHook: func(w io.Writer, node ast.Node, entering bool) (ast.WalkStatus, bool) {
// based on https://github.com/alecthomas/chroma/blob/master/quick/quick.go
htmlHighlight := func(w io.Writer, source, lang, defaultLang string) error {
if lang == "" {
lang = defaultLang
}
l := lexers.Get(lang)
if l == nil {
l = lexers.Analyse(source)
}
if l == nil {
l = lexers.Fallback
}
l = chroma.Coalesce(l)
it, err := l.Tokenise(nil, source)
if err != nil {
return err
}
return htmlFormatter.Format(w, highlightStyle, it)
}
renderCode := func(w io.Writer, codeBlock *ast.CodeBlock) {
defaultLang := ""
lang := string(codeBlock.Info)
htmlHighlight(w, string(codeBlock.Literal), lang, defaultLang)
}
if code, ok := node.(*ast.CodeBlock); ok {
renderCode(w, code)
return ast.GoToNext, true
}
return ast.GoToNext, false
},
})
}
if err := errors.Join(
setupHome(router),
setupGuide(router),
setupReferenceRoutes(router),
setupExamples(router),
setupEssays(router),
); err != nil {
return fmt.Errorf("error setting up routes: %w", err)
}
return nil
}