aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGabriel A. Giovanini <mail@gabrielgio.me>2024-05-05 19:02:43 +0200
committerGabriel A. Giovanini <mail@gabrielgio.me>2024-05-05 19:02:43 +0200
commitf5b5f72f7eaf14692b8036c3698037c647b2423a (patch)
treef74eb37277771fd1dceec702d1f3820fb553dc2a
parent129c73757036a8073467a2046bc73e95918f1ca1 (diff)
downloadcerrado-f5b5f72f7eaf14692b8036c3698037c647b2423a.tar.gz
cerrado-f5b5f72f7eaf14692b8036c3698037c647b2423a.tar.bz2
cerrado-f5b5f72f7eaf14692b8036c3698037c647b2423a.zip
feat: Add qtc templating
-rw-r--r--.gitignore1
-rw-r--r--Makefile13
-rw-r--r--go.mod3
-rw-r--r--go.sum21
-rw-r--r--main.go8
-rw-r--r--static/static.go6
-rw-r--r--templates/base.qtpl44
-rw-r--r--templates/base.qtpl.go217
-rw-r--r--templates/helloworld.qtpl16
-rw-r--r--templates/helloworld.qtpl.go131
10 files changed, 454 insertions, 6 deletions
diff --git a/.gitignore b/.gitignore
index e660fd9..776d7a2 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1 +1,2 @@
bin/
+static/*.css
diff --git a/Makefile b/Makefile
index c947b68..38db3a9 100644
--- a/Makefile
+++ b/Makefile
@@ -1,12 +1,19 @@
-build: sass
+.ONESHELL:
+
+build: sass tmpl
go build -o bin/cerrado
-run: sass
+run: sass tmpl
go run .
test:
go test -v --tags=unit ./...
sass:
+ @make -p static
sassc \
- -I scss scss/main.scss bin/main.css
+ -I scss scss/main.scss static/main.css
+
+tmpl:
+ cd ./templates
+ qtc *
diff --git a/go.mod b/go.mod
index bfbe03a..b8b36b4 100644
--- a/go.mod
+++ b/go.mod
@@ -5,5 +5,8 @@ go 1.22.2
require (
git.sr.ht/~emersion/go-scfg v0.0.0-20240128091534-2ae16e782082
github.com/google/go-cmp v0.6.0
+ github.com/valyala/quicktemplate v1.7.0
golang.org/x/sync v0.7.0
)
+
+require github.com/valyala/bytebufferpool v1.0.0 // indirect
diff --git a/go.sum b/go.sum
index a982044..0ba6fdb 100644
--- a/go.sum
+++ b/go.sum
@@ -1,8 +1,29 @@
git.sr.ht/~emersion/go-scfg v0.0.0-20240128091534-2ae16e782082 h1:9Udx5fm4vRtmgDIBjy2ef5QioHbzpw5oHabbhpAUyEw=
git.sr.ht/~emersion/go-scfg v0.0.0-20240128091534-2ae16e782082/go.mod h1:ybgvEJTIx5XbaspSviB3KNa6OdPmAZqDoSud7z8fFlw=
+github.com/andybalholm/brotli v1.0.2/go.mod h1:loMXtMfwqflxFJPmdbJO0a3KNoPuLBgiu3qAvBg8x/Y=
+github.com/andybalholm/brotli v1.0.3/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
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/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q=
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
+github.com/klauspost/compress v1.13.4/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg=
+github.com/klauspost/compress v1.13.5/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
+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.30.0/go.mod h1:2rsYD01CKFrjjsvFxx75KlEUNpWNBY9JWD3K/7o2Cus=
+github.com/valyala/quicktemplate v1.7.0 h1:LUPTJmlVcb46OOUY3IeD9DojFpAVbsG+5WFTcjMJzCM=
+github.com/valyala/quicktemplate v1.7.0/go.mod h1:sqKJnoaOF88V07vkO+9FL8fb9uZg/VPSJnLYn+LmLk8=
+github.com/valyala/tcplisten v1.0.0/go.mod h1:T0xQ8SeCZGxckz9qRXTfG43PvQ/mcWh7FwZEA7Ioqkc=
+golang.org/x/crypto v0.0.0-20210513164829-c07d793c2f9a/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8=
+golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
+golang.org/x/net v0.0.0-20210510120150-4163338589ed/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/sync v0.7.0 h1:YsImfSBoP9QPYL0xyKJPq0gcaJdG3rInoqxTWbfQu9M=
golang.org/x/sync v0.7.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
+golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
+golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
+golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
diff --git a/main.go b/main.go
index ba441fe..55f866e 100644
--- a/main.go
+++ b/main.go
@@ -12,6 +12,7 @@ import (
"git.gabrielgio.me/cerrado/pkg/config"
"git.gabrielgio.me/cerrado/pkg/worker"
+ "git.gabrielgio.me/cerrado/templates"
)
func main() {
@@ -51,10 +52,11 @@ func run(ctx context.Context) error {
return
}
- if _, err := w.Write(b); err != nil {
- slog.Error("Error handling index", "error", err)
- return
+ hello := &templates.HelloPage{
+ Body: string(b),
}
+
+ templates.WritePageTemplate(w, hello)
})
serverTask := worker.NewServerTask(&http.Server{Handler: mux, Addr: "0.0.0.0:8080"})
diff --git a/static/static.go b/static/static.go
new file mode 100644
index 0000000..ada0434
--- /dev/null
+++ b/static/static.go
@@ -0,0 +1,6 @@
+package static
+
+import "embed"
+
+//go:embed *
+var Static embed.FS
diff --git a/templates/base.qtpl b/templates/base.qtpl
new file mode 100644
index 0000000..1683981
--- /dev/null
+++ b/templates/base.qtpl
@@ -0,0 +1,44 @@
+This is a base page template. All the other template pages implement this interface.
+
+{% import "strconv" %}
+
+{% interface
+Page {
+ Title()
+ Content()
+ Script()
+}
+
+%}
+
+{% code func FromUInttoString(u *uint) string {
+ if u != nil {
+ return strconv.FormatUint(uint64(*u), 10)
+ }
+ return ""
+ }
+%}
+
+
+Page prints a page implementing Page interface.
+{% func PageTemplate(p Page) %}
+<html lang="en">
+ <head>
+ <meta charset="utf-8">
+ <title>img | {%= p.Title() %}</title>
+ <link rel="stylesheet" href="/static/main.css">
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
+ </head>
+ <body>
+ <div>
+ {%= p.Content() %}
+ </div>
+ </body>
+ {%= p.Script() %}
+</html>
+{% endfunc %}
+
+{% code type BasePage struct {} %}
+{% func (p *BasePage) Title() %}Empty{% endfunc %}
+{% func (p *BasePage) Body() %}HelloWorld{% endfunc %}
+{% func (p *BasePage) Script() %}{% endfunc %}
diff --git a/templates/base.qtpl.go b/templates/base.qtpl.go
new file mode 100644
index 0000000..8d93fa8
--- /dev/null
+++ b/templates/base.qtpl.go
@@ -0,0 +1,217 @@
+// Code generated by qtc from "base.qtpl". DO NOT EDIT.
+// See https://github.com/valyala/quicktemplate for details.
+
+// This is a base page template. All the other template pages implement this interface.
+//
+
+//line base.qtpl:3
+package templates
+
+//line base.qtpl:3
+import "strconv"
+
+//line base.qtpl:5
+import (
+ qtio422016 "io"
+
+ qt422016 "github.com/valyala/quicktemplate"
+)
+
+//line base.qtpl:5
+var (
+ _ = qtio422016.Copy
+ _ = qt422016.AcquireByteBuffer
+)
+
+//line base.qtpl:6
+type Page interface {
+//line base.qtpl:6
+ Title() string
+//line base.qtpl:6
+ StreamTitle(qw422016 *qt422016.Writer)
+//line base.qtpl:6
+ WriteTitle(qq422016 qtio422016.Writer)
+//line base.qtpl:6
+ Content() string
+//line base.qtpl:6
+ StreamContent(qw422016 *qt422016.Writer)
+//line base.qtpl:6
+ WriteContent(qq422016 qtio422016.Writer)
+//line base.qtpl:6
+ Script() string
+//line base.qtpl:6
+ StreamScript(qw422016 *qt422016.Writer)
+//line base.qtpl:6
+ WriteScript(qq422016 qtio422016.Writer)
+//line base.qtpl:6
+}
+
+//line base.qtpl:14
+func FromUInttoString(u *uint) string {
+ if u != nil {
+ return strconv.FormatUint(uint64(*u), 10)
+ }
+ return ""
+}
+
+// Page prints a page implementing Page interface.
+
+//line base.qtpl:24
+func StreamPageTemplate(qw422016 *qt422016.Writer, p Page) {
+//line base.qtpl:24
+ qw422016.N().S(`
+<html lang="en">
+ <head>
+ <meta charset="utf-8">
+ <title>img | `)
+//line base.qtpl:28
+ p.StreamTitle(qw422016)
+//line base.qtpl:28
+ qw422016.N().S(`</title>
+ <link rel="stylesheet" href="/static/main.css">
+ <meta name="viewport" content="width=device-width, initial-scale=1" />
+ </head>
+ <body>
+ <div>
+ `)
+//line base.qtpl:34
+ p.StreamContent(qw422016)
+//line base.qtpl:34
+ qw422016.N().S(`
+ </div>
+ </body>
+ `)
+//line base.qtpl:37
+ p.StreamScript(qw422016)
+//line base.qtpl:37
+ qw422016.N().S(`
+</html>
+`)
+//line base.qtpl:39
+}
+
+//line base.qtpl:39
+func WritePageTemplate(qq422016 qtio422016.Writer, p Page) {
+//line base.qtpl:39
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line base.qtpl:39
+ StreamPageTemplate(qw422016, p)
+//line base.qtpl:39
+ qt422016.ReleaseWriter(qw422016)
+//line base.qtpl:39
+}
+
+//line base.qtpl:39
+func PageTemplate(p Page) string {
+//line base.qtpl:39
+ qb422016 := qt422016.AcquireByteBuffer()
+//line base.qtpl:39
+ WritePageTemplate(qb422016, p)
+//line base.qtpl:39
+ qs422016 := string(qb422016.B)
+//line base.qtpl:39
+ qt422016.ReleaseByteBuffer(qb422016)
+//line base.qtpl:39
+ return qs422016
+//line base.qtpl:39
+}
+
+//line base.qtpl:41
+type BasePage struct{}
+
+//line base.qtpl:42
+func (p *BasePage) StreamTitle(qw422016 *qt422016.Writer) {
+//line base.qtpl:42
+ qw422016.N().S(`Empty`)
+//line base.qtpl:42
+}
+
+//line base.qtpl:42
+func (p *BasePage) WriteTitle(qq422016 qtio422016.Writer) {
+//line base.qtpl:42
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line base.qtpl:42
+ p.StreamTitle(qw422016)
+//line base.qtpl:42
+ qt422016.ReleaseWriter(qw422016)
+//line base.qtpl:42
+}
+
+//line base.qtpl:42
+func (p *BasePage) Title() string {
+//line base.qtpl:42
+ qb422016 := qt422016.AcquireByteBuffer()
+//line base.qtpl:42
+ p.WriteTitle(qb422016)
+//line base.qtpl:42
+ qs422016 := string(qb422016.B)
+//line base.qtpl:42
+ qt422016.ReleaseByteBuffer(qb422016)
+//line base.qtpl:42
+ return qs422016
+//line base.qtpl:42
+}
+
+//line base.qtpl:43
+func (p *BasePage) StreamBody(qw422016 *qt422016.Writer) {
+//line base.qtpl:43
+ qw422016.N().S(`HelloWorld`)
+//line base.qtpl:43
+}
+
+//line base.qtpl:43
+func (p *BasePage) WriteBody(qq422016 qtio422016.Writer) {
+//line base.qtpl:43
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line base.qtpl:43
+ p.StreamBody(qw422016)
+//line base.qtpl:43
+ qt422016.ReleaseWriter(qw422016)
+//line base.qtpl:43
+}
+
+//line base.qtpl:43
+func (p *BasePage) Body() string {
+//line base.qtpl:43
+ qb422016 := qt422016.AcquireByteBuffer()
+//line base.qtpl:43
+ p.WriteBody(qb422016)
+//line base.qtpl:43
+ qs422016 := string(qb422016.B)
+//line base.qtpl:43
+ qt422016.ReleaseByteBuffer(qb422016)
+//line base.qtpl:43
+ return qs422016
+//line base.qtpl:43
+}
+
+//line base.qtpl:44
+func (p *BasePage) StreamScript(qw422016 *qt422016.Writer) {
+//line base.qtpl:44
+}
+
+//line base.qtpl:44
+func (p *BasePage) WriteScript(qq422016 qtio422016.Writer) {
+//line base.qtpl:44
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line base.qtpl:44
+ p.StreamScript(qw422016)
+//line base.qtpl:44
+ qt422016.ReleaseWriter(qw422016)
+//line base.qtpl:44
+}
+
+//line base.qtpl:44
+func (p *BasePage) Script() string {
+//line base.qtpl:44
+ qb422016 := qt422016.AcquireByteBuffer()
+//line base.qtpl:44
+ p.WriteScript(qb422016)
+//line base.qtpl:44
+ qs422016 := string(qb422016.B)
+//line base.qtpl:44
+ qt422016.ReleaseByteBuffer(qb422016)
+//line base.qtpl:44
+ return qs422016
+//line base.qtpl:44
+}
diff --git a/templates/helloworld.qtpl b/templates/helloworld.qtpl
new file mode 100644
index 0000000..02c0968
--- /dev/null
+++ b/templates/helloworld.qtpl
@@ -0,0 +1,16 @@
+{% code
+type HelloPage struct {
+ Body string
+}
+%}
+
+{% func (p *HelloPage) Title() %}Hello{% endfunc %}
+
+{% func (p *HelloPage) Content() %}
+HelloWorld
+
+{%s p.Body %}
+{% endfunc %}
+
+{% func (p *HelloPage) Script() %}
+{% endfunc %}
diff --git a/templates/helloworld.qtpl.go b/templates/helloworld.qtpl.go
new file mode 100644
index 0000000..9ef78f5
--- /dev/null
+++ b/templates/helloworld.qtpl.go
@@ -0,0 +1,131 @@
+// Code generated by qtc from "helloworld.qtpl". DO NOT EDIT.
+// See https://github.com/valyala/quicktemplate for details.
+
+//line helloworld.qtpl:1
+package templates
+
+//line helloworld.qtpl:1
+import (
+ qtio422016 "io"
+
+ qt422016 "github.com/valyala/quicktemplate"
+)
+
+//line helloworld.qtpl:1
+var (
+ _ = qtio422016.Copy
+ _ = qt422016.AcquireByteBuffer
+)
+
+//line helloworld.qtpl:2
+type HelloPage struct {
+ Body string
+}
+
+//line helloworld.qtpl:7
+func (p *HelloPage) StreamTitle(qw422016 *qt422016.Writer) {
+//line helloworld.qtpl:7
+ qw422016.N().S(`Hello`)
+//line helloworld.qtpl:7
+}
+
+//line helloworld.qtpl:7
+func (p *HelloPage) WriteTitle(qq422016 qtio422016.Writer) {
+//line helloworld.qtpl:7
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line helloworld.qtpl:7
+ p.StreamTitle(qw422016)
+//line helloworld.qtpl:7
+ qt422016.ReleaseWriter(qw422016)
+//line helloworld.qtpl:7
+}
+
+//line helloworld.qtpl:7
+func (p *HelloPage) Title() string {
+//line helloworld.qtpl:7
+ qb422016 := qt422016.AcquireByteBuffer()
+//line helloworld.qtpl:7
+ p.WriteTitle(qb422016)
+//line helloworld.qtpl:7
+ qs422016 := string(qb422016.B)
+//line helloworld.qtpl:7
+ qt422016.ReleaseByteBuffer(qb422016)
+//line helloworld.qtpl:7
+ return qs422016
+//line helloworld.qtpl:7
+}
+
+//line helloworld.qtpl:9
+func (p *HelloPage) StreamContent(qw422016 *qt422016.Writer) {
+//line helloworld.qtpl:9
+ qw422016.N().S(`
+HelloWorld
+
+`)
+//line helloworld.qtpl:12
+ qw422016.E().S(p.Body)
+//line helloworld.qtpl:12
+ qw422016.N().S(`
+`)
+//line helloworld.qtpl:13
+}
+
+//line helloworld.qtpl:13
+func (p *HelloPage) WriteContent(qq422016 qtio422016.Writer) {
+//line helloworld.qtpl:13
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line helloworld.qtpl:13
+ p.StreamContent(qw422016)
+//line helloworld.qtpl:13
+ qt422016.ReleaseWriter(qw422016)
+//line helloworld.qtpl:13
+}
+
+//line helloworld.qtpl:13
+func (p *HelloPage) Content() string {
+//line helloworld.qtpl:13
+ qb422016 := qt422016.AcquireByteBuffer()
+//line helloworld.qtpl:13
+ p.WriteContent(qb422016)
+//line helloworld.qtpl:13
+ qs422016 := string(qb422016.B)
+//line helloworld.qtpl:13
+ qt422016.ReleaseByteBuffer(qb422016)
+//line helloworld.qtpl:13
+ return qs422016
+//line helloworld.qtpl:13
+}
+
+//line helloworld.qtpl:15
+func (p *HelloPage) StreamScript(qw422016 *qt422016.Writer) {
+//line helloworld.qtpl:15
+ qw422016.N().S(`
+`)
+//line helloworld.qtpl:16
+}
+
+//line helloworld.qtpl:16
+func (p *HelloPage) WriteScript(qq422016 qtio422016.Writer) {
+//line helloworld.qtpl:16
+ qw422016 := qt422016.AcquireWriter(qq422016)
+//line helloworld.qtpl:16
+ p.StreamScript(qw422016)
+//line helloworld.qtpl:16
+ qt422016.ReleaseWriter(qw422016)
+//line helloworld.qtpl:16
+}
+
+//line helloworld.qtpl:16
+func (p *HelloPage) Script() string {
+//line helloworld.qtpl:16
+ qb422016 := qt422016.AcquireByteBuffer()
+//line helloworld.qtpl:16
+ p.WriteScript(qb422016)
+//line helloworld.qtpl:16
+ qs422016 := string(qb422016.B)
+//line helloworld.qtpl:16
+ qt422016.ReleaseByteBuffer(qb422016)
+//line helloworld.qtpl:16
+ return qs422016
+//line helloworld.qtpl:16
+}