From 30c921e57bab03793b259448320922b1e09981b1 Mon Sep 17 00:00:00 2001 From: AKP Date: Sun, 5 Nov 2023 10:52:41 +0000 Subject: [PATCH] Is it finished?? --- go.mod | 14 ++- go.sum | 29 ++++++ ui/internal/httpcore/httpcore.go | 158 +++++++++++++++++++++++++++++++ ui/main.go | 18 +--- 4 files changed, 202 insertions(+), 17 deletions(-) create mode 100644 ui/internal/httpcore/httpcore.go diff --git a/go.mod b/go.mod index 4634570..e26fc70 100644 --- a/go.mod +++ b/go.mod @@ -14,14 +14,26 @@ require ( ) require ( + github.com/andybalholm/brotli v1.0.5 // indirect github.com/andybalholm/cascadia v1.3.1 // indirect + github.com/gofiber/fiber/v2 v2.50.0 // indirect + github.com/google/uuid v1.3.1 // indirect github.com/jinzhu/inflection v1.0.0 // indirect + github.com/klauspost/compress v1.16.7 // indirect github.com/kr/text v0.2.0 // indirect + github.com/maragudk/gomponents v0.20.1 // indirect + github.com/mattn/go-colorable v0.1.13 // indirect + github.com/mattn/go-isatty v0.0.19 // indirect + github.com/mattn/go-runewidth v0.0.15 // indirect + github.com/rivo/uniseg v0.2.0 // indirect github.com/surge/glog v0.0.0-20141108051140-2578deb2b95c // indirect github.com/tmthrgd/go-hex v0.0.0-20190904060850-447a3041c3bc // indirect + github.com/valyala/bytebufferpool v1.0.0 // indirect + github.com/valyala/fasthttp v1.50.0 // indirect + github.com/valyala/tcplisten v1.0.0 // indirect github.com/vmihailenco/msgpack/v5 v5.3.5 // indirect github.com/vmihailenco/tagparser/v2 v2.0.0 // indirect golang.org/x/net v0.15.0 // indirect - golang.org/x/sys v0.12.0 // indirect + golang.org/x/sys v0.13.0 // indirect gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index 0e6f5c3..aa6cd51 100644 --- a/go.sum +++ b/go.sum @@ -2,6 +2,8 @@ git.tdpain.net/pkg/cfger v0.1.0 h1:Yhs2DaFIdbcSrbyywhsoxrHPevDEEBEKbqJqrUa3eso= git.tdpain.net/pkg/cfger v0.1.0/go.mod h1:Kq5/hsUnYSYM2BVGFtXMlYEDIsIYiZTz4MUIlqZeX0k= github.com/PuerkitoBio/goquery v1.8.1 h1:uQxhNlArOIdbrH1tr0UXwdVFgDcZDrZVdcpygAcwmWM= github.com/PuerkitoBio/goquery v1.8.1/go.mod h1:Q8ICL1kNUJ2sXGoAhPGUdYDJvgQgHzJsnnd3H7Ho5jQ= +github.com/andybalholm/brotli v1.0.5 h1:8uQZIdzKmjc/iuPu7O2ioW48L81FgatrcpfFmiq/cCs= +github.com/andybalholm/brotli v1.0.5/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig= github.com/andybalholm/cascadia v1.3.1 h1:nhxRkql1kdYCc8Snf7D5/D3spOX+dBgjA6u8x004T2c= github.com/andybalholm/cascadia v1.3.1/go.mod h1:R4bJ1UQfqADjvDa4P6HZHLh/3OxWWEqc0Sk8XGwHqvA= github.com/bwmarrin/snowflake v0.3.0 h1:xm67bEhkKh6ij1790JB83OujPR5CzNe8QuQqAgISZN0= @@ -12,16 +14,33 @@ github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ3 github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/gofiber/fiber/v2 v2.50.0 h1:ia0JaB+uw3GpNSCR5nvC5dsaxXjRU5OEu36aytx+zGw= +github.com/gofiber/fiber/v2 v2.50.0/go.mod h1:21eytvay9Is7S6z+OgPi7c7n4++tnClWmhpimVHMimw= +github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4= +github.com/google/uuid v1.3.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/jinzhu/inflection v1.0.0 h1:K317FqzuhWc8YvSVlFMCCUb36O/S9MCKRDI7QkRKD/E= github.com/jinzhu/inflection v1.0.0/go.mod h1:h+uFLlag+Qp1Va5pdKtLDYj+kHp5pxUVkryuEj+Srlc= +github.com/klauspost/compress v1.16.7 h1:2mk3MPGNzKyxErAw8YaohYh69+pa4sIQSC0fPGCFR9I= +github.com/klauspost/compress v1.16.7/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/maragudk/gomponents v0.20.1 h1:TeJY1fXEcfUvzmvjeUgxol42dvkYMggK1c0V67crWWs= +github.com/maragudk/gomponents v0.20.1/go.mod h1:nHkNnZL6ODgMBeJhrZjkMHVvNdoYsfmpKB2/hjdQ0Hg= +github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA= +github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg= +github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA= +github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y= +github.com/mattn/go-runewidth v0.0.15 h1:UNAjwbU9l54TA3KzvqLGxwWjHmMgBUVhBiTjelZgg3U= +github.com/mattn/go-runewidth v0.0.15/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= github.com/mattn/go-sqlite3 v1.14.17 h1:mCRHCLDUBXgpKAqIKsaAaAsrAlbkeomtRFKXh2L6YIM= github.com/mattn/go-sqlite3 v1.14.17/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/rivo/uniseg v0.2.0 h1:S1pD9weZBuJdFmowNwbpi7BJ8TNftyUImj/0WQi72jY= +github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= @@ -34,6 +53,12 @@ github.com/uptrace/bun v1.1.16 h1:cn9cgEMFwcyYRsQLfxCRMUxyK1WaHwOVrR3TvzEFZ/A= github.com/uptrace/bun v1.1.16/go.mod h1:7HnsMRRvpLFUcquJxp22JO8PsWKpFQO/gNXqqsuGWg8= github.com/uptrace/bun/dialect/sqlitedialect v1.1.16 h1:gbc9BP/e4sNOB9VBj+Si46dpOz2oktmZPidkda92GYY= github.com/uptrace/bun/dialect/sqlitedialect v1.1.16/go.mod h1:YNezpK7fIn5Wa2WGmTCZ/nEyiswcXmuT4iNWADeL1x4= +github.com/valyala/bytebufferpool v1.0.0 h1:GqA5TC/0021Y/b9FG4Oi9Mr3q7XYx6KllzawFIhcdPw= +github.com/valyala/bytebufferpool v1.0.0/go.mod h1:6bBcMArwyJ5K/AmCkWv1jt77kVWyCJ6HpOuEn7z0Csc= +github.com/valyala/fasthttp v1.50.0 h1:H7fweIlBm0rXLs2q0XbalvJ6r0CUPFWK3/bB4N13e9M= +github.com/valyala/fasthttp v1.50.0/go.mod h1:k2zXd82h/7UZc3VOdJ2WaUqt1uZ/XpXAfE9i+HBC3lA= +github.com/valyala/tcplisten v1.0.0 h1:rBHj/Xf+E1tRGZyWIWwJDiRY0zc1Js+CV5DqwacVSA8= +github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc= github.com/vmihailenco/msgpack/v5 v5.3.5 h1:5gO0H1iULLWGhs2H5tbAHIZTV8/cYafcFOr9znI5mJU= github.com/vmihailenco/msgpack/v5 v5.3.5/go.mod h1:7xyJ9e+0+9SaZT0Wt1RGleJXzli6Q/V5KbhBonMG9jc= github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g= @@ -59,9 +84,13 @@ golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o= golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.13.0 h1:Af8nKPmuFypiUBjVoU9V20FiaFXOcuZI21p0ycVYYGE= +golang.org/x/sys v0.13.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= diff --git a/ui/internal/httpcore/httpcore.go b/ui/internal/httpcore/httpcore.go new file mode 100644 index 0000000..0673407 --- /dev/null +++ b/ui/internal/httpcore/httpcore.go @@ -0,0 +1,158 @@ +package httpcore + +import ( + "bytes" + "git.tdpain.net/codemicro/hn84/ui/internal/config" + "git.tdpain.net/codemicro/hn84/ui/internal/search" + "github.com/gofiber/fiber/v2" + g "github.com/maragudk/gomponents" + "github.com/maragudk/gomponents/components" + "github.com/maragudk/gomponents/html" + "github.com/uptrace/bun" + "os" + "path" +) + +type endpoints struct { + DB *bun.DB +} + +func ListenAndServe(db *bun.DB) error { + app := fiber.New() + + e := &endpoints{db} + + app.Get("/", e.index) + app.Get("/search", e.search) + + return app.Listen("127.0.0.1:8080") +} + +func renderPage(ctx *fiber.Ctx, p g.Node) error { + b := new(bytes.Buffer) + if err := p.Render(b); err != nil { + return err + } + ctx.Type("html") + return ctx.Send(b.Bytes()) +} + +func (*endpoints) index(ctx *fiber.Ctx) error { + return renderPage(ctx, basePage("Rummage", + html.Div(g.Attr("class", "row"), + html.Div(g.Attr("class", "col-4 mx-auto text-center"), + html.Img(g.Attr("class", "pb-3"), g.Attr("src", ""), g.Attr("alt", "Clipart box of stuff")), + ), + ), + searchBox(""), + )) +} + +func (e *endpoints) search(ctx *fiber.Ctx) error { + queryStr := ctx.Query("q") + if queryStr == "" { + return ctx.Redirect("/") + } + + queryTokens := search.PlaintextToTokens(queryStr) + results, err := search.DoSearch(e.DB, queryTokens) + if err != nil { + return err + } + + var resultNodes []g.Node + + conf := config.Get() + + for i, res := range results { + plaintext, err := os.ReadFile(path.Join(conf.CrawlDataDir, res.Document.ID+".txt")) + + var text g.Node + + if err == nil { + middleToken := res.Tokens[len(res.Tokens)/2] + startPos := middleToken.Start - 50 + endPos := 100 - (middleToken.End - middleToken.Start) + middleToken.End + if endPos >= len(plaintext) { + endPos = len(plaintext) - 1 + } + + x := make([]g.Node, endPos-startPos) + + for _, tok := range res.Tokens { + if tok.Start >= startPos && tok.End <= endPos { + for i, ch := range plaintext[tok.Start : tok.End+1] { + x[(tok.Start-startPos)+i] = html.B(g.Text(string(ch))) + } + } + } + + for i, b := range x { + if b == nil { + x[i] = g.Text(string(plaintext[startPos+i])) + } + } + + text = g.Group(x) + } + + class := "mt-4 p-1 border-radius-1" + if i%2 == 0 { + class += " bg-primary-subtle" + } + + resultNodes = append(resultNodes, html.Div( + g.Attr("class", class), + html.H4(unstyledLink(res.Document.Title, res.Document.URL)), + html.Span(g.Attr("class", "text-success"), html.A(g.Text(res.Document.URL), g.Attr("href", res.Document.URL), g.Attr("class", "text-decoration-none text-success"))), + html.Br(), + html.Span(g.Attr("class", "text-secondary fst-italic text-sm"), html.A(text, g.Attr("href", res.Document.URL), g.Attr("class", "text-decoration-none text-secondary"))), + )) + } + + return renderPage(ctx, basePage("Rummage search results", + searchBox(queryStr), + g.Group(resultNodes), + )) +} + +func unstyledLink(text string, href string) g.Node { + return html.A(g.Text(text), g.Attr("href", href), g.Attr("class", "text-decoration-none")) +} + +func basePage(title string, content ...g.Node) g.Node { + return components.HTML5(components.HTML5Props{ + Title: title, + Language: "en-GB", + Head: []g.Node{ + html.Link(g.Attr("href", "https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css"), g.Attr("rel", "stylesheet")), + }, + Body: []g.Node{ + html.Nav(g.Attr("class", "navbar bg-dark border-bottom border-body"), g.Attr("data-bs-theme", "dark"), + html.Div(g.Attr("class", "container-fluid"), html.A(g.Attr("class", "navbar-brand"), g.Attr("href", "/"), g.Text("Rummage"))), + ), + html.Div(g.Attr("class", "pt-4 container"), g.Group(content)), + html.Script(g.Attr("defer", "true"), g.Attr("src", "https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js"))}, + }) +} + +func searchBox(query string) g.Node { + return html.FormEl( + g.Attr("action", "/search"), + html.Div( + g.Attr("class", "input-group input-group-lg"), + html.Input( + g.Attr("type", "text"), + g.Attr("class", "form-control"), + g.Attr("placeholder", "Enter your query..."), + g.Attr("name", "q"), + g.If(query != "", g.Attr("value", query)), + ), + html.Button( + g.Attr("class", "btn btn-outline-primary"), + g.Attr("type", "submit"), + g.Text("Submit"), + ), + ), + ) +} diff --git a/ui/main.go b/ui/main.go index e30f4d9..1ce02a1 100644 --- a/ui/main.go +++ b/ui/main.go @@ -1,10 +1,9 @@ package main import ( - "fmt" "git.tdpain.net/codemicro/hn84/ui/internal/config" "git.tdpain.net/codemicro/hn84/ui/internal/database" - "git.tdpain.net/codemicro/hn84/ui/internal/search" + "git.tdpain.net/codemicro/hn84/ui/internal/httpcore" "git.tdpain.net/codemicro/hn84/util" "log/slog" ) @@ -21,18 +20,5 @@ func run() error { return util.Wrap("setup database", err) } - query := search.PlaintextToTokens("reading list") - - matches, err := search.DoSearch(db, query) - if err != nil { - return util.Wrap("run search", err) - } - - fmt.Println(query) - - for _, m := range matches { - fmt.Println(m.Document.Title, m.Ranking) - } - - return nil + return httpcore.ListenAndServe(db) }