Set up a pipeline
This commit is contained in:
parent
016dbd0814
commit
ed825ce202
8 changed files with 508 additions and 0 deletions
4
.dagger/.gitattributes
vendored
Normal file
4
.dagger/.gitattributes
vendored
Normal file
|
|
@ -0,0 +1,4 @@
|
|||
/dagger.gen.go linguist-generated
|
||||
/internal/dagger/** linguist-generated
|
||||
/internal/querybuilder/** linguist-generated
|
||||
/internal/telemetry/** linguist-generated
|
||||
5
.dagger/.gitignore
vendored
Normal file
5
.dagger/.gitignore
vendored
Normal file
|
|
@ -0,0 +1,5 @@
|
|||
/dagger.gen.go
|
||||
/internal/dagger
|
||||
/internal/querybuilder
|
||||
/internal/telemetry
|
||||
/.env
|
||||
50
.dagger/go.mod
Normal file
50
.dagger/go.mod
Normal file
|
|
@ -0,0 +1,50 @@
|
|||
module dagger/blog
|
||||
|
||||
go 1.25.5
|
||||
|
||||
require (
|
||||
github.com/99designs/gqlgen v0.17.81
|
||||
github.com/Khan/genqlient v0.8.1
|
||||
github.com/vektah/gqlparser/v2 v2.5.30
|
||||
go.opentelemetry.io/otel v1.38.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.14.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.14.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.38.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.38.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.38.0
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.38.0
|
||||
go.opentelemetry.io/otel/log v0.14.0
|
||||
go.opentelemetry.io/otel/metric v1.38.0
|
||||
go.opentelemetry.io/otel/sdk v1.38.0
|
||||
go.opentelemetry.io/otel/sdk/log v0.14.0
|
||||
go.opentelemetry.io/otel/sdk/metric v1.38.0
|
||||
go.opentelemetry.io/otel/trace v1.38.0
|
||||
go.opentelemetry.io/proto/otlp v1.8.0
|
||||
golang.org/x/sync v0.17.0
|
||||
google.golang.org/grpc v1.76.0
|
||||
)
|
||||
|
||||
require (
|
||||
github.com/cenkalti/backoff/v5 v5.0.3 // indirect
|
||||
github.com/go-logr/logr v1.4.3 // indirect
|
||||
github.com/go-logr/stdr v1.2.2 // indirect
|
||||
github.com/google/uuid v1.6.0 // indirect
|
||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.27.2 // indirect
|
||||
github.com/sosodev/duration v1.3.1 // indirect
|
||||
go.opentelemetry.io/auto/sdk v1.1.0 // indirect
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.38.0 // indirect
|
||||
golang.org/x/net v0.44.0 // indirect
|
||||
golang.org/x/sys v0.36.0 // indirect
|
||||
golang.org/x/text v0.29.0 // indirect
|
||||
google.golang.org/genproto/googleapis/api v0.0.0-20250825161204-c5933d9347a5 // indirect
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250825161204-c5933d9347a5 // indirect
|
||||
google.golang.org/protobuf v1.36.9 // indirect
|
||||
)
|
||||
|
||||
replace go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc => go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.14.0
|
||||
|
||||
replace go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp => go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.14.0
|
||||
|
||||
replace go.opentelemetry.io/otel/log => go.opentelemetry.io/otel/log v0.14.0
|
||||
|
||||
replace go.opentelemetry.io/otel/sdk/log => go.opentelemetry.io/otel/sdk/log v0.14.0
|
||||
89
.dagger/go.sum
Normal file
89
.dagger/go.sum
Normal file
|
|
@ -0,0 +1,89 @@
|
|||
github.com/99designs/gqlgen v0.17.81 h1:kCkN/xVyRb5rEQpuwOHRTYq83i0IuTQg9vdIiwEerTs=
|
||||
github.com/99designs/gqlgen v0.17.81/go.mod h1:vgNcZlLwemsUhYim4dC1pvFP5FX0pr2Y+uYUoHFb1ig=
|
||||
github.com/Khan/genqlient v0.8.1 h1:wtOCc8N9rNynRLXN3k3CnfzheCUNKBcvXmVv5zt6WCs=
|
||||
github.com/Khan/genqlient v0.8.1/go.mod h1:R2G6DzjBvCbhjsEajfRjbWdVglSH/73kSivC9TLWVjU=
|
||||
github.com/andreyvit/diff v0.0.0-20170406064948-c7f18ee00883 h1:bvNMNQO63//z+xNgfBlViaCIJKLlCJ6/fmUseuG0wVQ=
|
||||
github.com/andreyvit/diff v0.0.0-20170406064948-c7f18ee00883/go.mod h1:rCTlJbsFo29Kk6CurOXKm700vrz8f0KW0JNfpkRJY/8=
|
||||
github.com/cenkalti/backoff/v5 v5.0.3 h1:ZN+IMa753KfX5hd8vVaMixjnqRZ3y8CuJKRKj1xcsSM=
|
||||
github.com/cenkalti/backoff/v5 v5.0.3/go.mod h1:rkhZdG3JZukswDf7f0cwqPNk4K0sa+F97BxZthm/crw=
|
||||
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/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
|
||||
github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI=
|
||||
github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
|
||||
github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag=
|
||||
github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE=
|
||||
github.com/golang/protobuf v1.5.4 h1:i7eJL8qZTpSEXOPTxNKhASYpMn+8e5Q6AdndVa1dWek=
|
||||
github.com/golang/protobuf v1.5.4/go.mod h1:lnTiLA8Wa4RWRcIUkrtSVa5nRhsEGBg48fD6rSs7xps=
|
||||
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
|
||||
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
|
||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.27.2 h1:8Tjv8EJ+pM1xP8mK6egEbD1OgnVTyacbefKhmbLhIhU=
|
||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.27.2/go.mod h1:pkJQ2tZHJ0aFOVEEot6oZmaVEZcRme73eIFmhiVuRWs=
|
||||
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/sergi/go-diff v1.3.1 h1:xkr+Oxo4BOQKmkn/B9eMK0g5Kg/983T9DqqPHwYqD+8=
|
||||
github.com/sergi/go-diff v1.3.1/go.mod h1:aMJSSKb2lpPvRNec0+w3fl7LP9IOFzdc9Pa4NFbPK1I=
|
||||
github.com/sosodev/duration v1.3.1 h1:qtHBDMQ6lvMQsL15g4aopM4HEfOaYuhWBw3NPTtlqq4=
|
||||
github.com/sosodev/duration v1.3.1/go.mod h1:RQIBBX0+fMLc/D9+Jb/fwvVmo0eZvDDEERAikUR6SDg=
|
||||
github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U=
|
||||
github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U=
|
||||
github.com/vektah/gqlparser/v2 v2.5.30 h1:EqLwGAFLIzt1wpx1IPpY67DwUujF1OfzgEyDsLrN6kE=
|
||||
github.com/vektah/gqlparser/v2 v2.5.30/go.mod h1:D1/VCZtV3LPnQrcPBeR/q5jkSQIPti0uYCP/RI0gIeo=
|
||||
go.opentelemetry.io/auto/sdk v1.1.0 h1:cH53jehLUN6UFLY71z+NDOiNJqDdPRaXzTel0sJySYA=
|
||||
go.opentelemetry.io/auto/sdk v1.1.0/go.mod h1:3wSPjt5PWp2RhlCcmmOial7AvC4DQqZb7a7wCow3W8A=
|
||||
go.opentelemetry.io/otel v1.38.0 h1:RkfdswUDRimDg0m2Az18RKOsnI8UDzppJAtj01/Ymk8=
|
||||
go.opentelemetry.io/otel v1.38.0/go.mod h1:zcmtmQ1+YmQM9wrNsTGV/q/uyusom3P8RxwExxkZhjM=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.14.0 h1:OMqPldHt79PqWKOMYIAQs3CxAi7RLgPxwfFSwr4ZxtM=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploggrpc v0.14.0/go.mod h1:1biG4qiqTxKiUCtoWDPpL3fB3KxVwCiGw81j3nKMuHE=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.14.0 h1:QQqYw3lkrzwVsoEX0w//EhH/TCnpRdEenKBOOEIMjWc=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlplog/otlploghttp v0.14.0/go.mod h1:gSVQcr17jk2ig4jqJ2DX30IdWH251JcNAecvrqTxH1s=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.38.0 h1:vl9obrcoWVKp/lwl8tRE33853I8Xru9HFbw/skNeLs8=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.38.0/go.mod h1:GAXRxmLJcVM3u22IjTg74zWBrRCKq8BnOqUVLodpcpw=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.38.0 h1:Oe2z/BCg5q7k4iXC3cqJxKYg0ieRiOqF0cecFYdPTwk=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.38.0/go.mod h1:ZQM5lAJpOsKnYagGg/zV2krVqTtaVdYdDkhMoX6Oalg=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.38.0 h1:GqRJVj7UmLjCVyVJ3ZFLdPRmhDUp2zFmQe3RHIOsw24=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.38.0/go.mod h1:ri3aaHSmCTVYu2AWv44YMauwAQc0aqI9gHKIcSbI1pU=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.38.0 h1:lwI4Dc5leUqENgGuQImwLo4WnuXFPetmPpkLi2IrX54=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.38.0/go.mod h1:Kz/oCE7z5wuyhPxsXDuaPteSWqjSBD5YaSdbxZYGbGk=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.38.0 h1:aTL7F04bJHUlztTsNGJ2l+6he8c+y/b//eR0jjjemT4=
|
||||
go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.38.0/go.mod h1:kldtb7jDTeol0l3ewcmd8SDvx3EmIE7lyvqbasU3QC4=
|
||||
go.opentelemetry.io/otel/log v0.14.0 h1:2rzJ+pOAZ8qmZ3DDHg73NEKzSZkhkGIua9gXtxNGgrM=
|
||||
go.opentelemetry.io/otel/log v0.14.0/go.mod h1:5jRG92fEAgx0SU/vFPxmJvhIuDU9E1SUnEQrMlJpOno=
|
||||
go.opentelemetry.io/otel/metric v1.38.0 h1:Kl6lzIYGAh5M159u9NgiRkmoMKjvbsKtYRwgfrA6WpA=
|
||||
go.opentelemetry.io/otel/metric v1.38.0/go.mod h1:kB5n/QoRM8YwmUahxvI3bO34eVtQf2i4utNVLr9gEmI=
|
||||
go.opentelemetry.io/otel/sdk v1.38.0 h1:l48sr5YbNf2hpCUj/FoGhW9yDkl+Ma+LrVl8qaM5b+E=
|
||||
go.opentelemetry.io/otel/sdk v1.38.0/go.mod h1:ghmNdGlVemJI3+ZB5iDEuk4bWA3GkTpW+DOoZMYBVVg=
|
||||
go.opentelemetry.io/otel/sdk/log v0.14.0 h1:JU/U3O7N6fsAXj0+CXz21Czg532dW2V4gG1HE/e8Zrg=
|
||||
go.opentelemetry.io/otel/sdk/log v0.14.0/go.mod h1:imQvII+0ZylXfKU7/wtOND8Hn4OpT3YUoIgqJVksUkM=
|
||||
go.opentelemetry.io/otel/sdk/log/logtest v0.14.0 h1:Ijbtz+JKXl8T2MngiwqBlPaHqc4YCaP/i13Qrow6gAM=
|
||||
go.opentelemetry.io/otel/sdk/log/logtest v0.14.0/go.mod h1:dCU8aEL6q+L9cYTqcVOk8rM9Tp8WdnHOPLiBgp0SGOA=
|
||||
go.opentelemetry.io/otel/sdk/metric v1.38.0 h1:aSH66iL0aZqo//xXzQLYozmWrXxyFkBJ6qT5wthqPoM=
|
||||
go.opentelemetry.io/otel/sdk/metric v1.38.0/go.mod h1:dg9PBnW9XdQ1Hd6ZnRz689CbtrUp0wMMs9iPcgT9EZA=
|
||||
go.opentelemetry.io/otel/trace v1.38.0 h1:Fxk5bKrDZJUH+AMyyIXGcFAPah0oRcT+LuNtJrmcNLE=
|
||||
go.opentelemetry.io/otel/trace v1.38.0/go.mod h1:j1P9ivuFsTceSWe1oY+EeW3sc+Pp42sO++GHkg4wwhs=
|
||||
go.opentelemetry.io/proto/otlp v1.8.0 h1:fRAZQDcAFHySxpJ1TwlA1cJ4tvcrw7nXl9xWWC8N5CE=
|
||||
go.opentelemetry.io/proto/otlp v1.8.0/go.mod h1:tIeYOeNBU4cvmPqpaji1P+KbB4Oloai8wN4rWzRrFF0=
|
||||
go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto=
|
||||
go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE=
|
||||
golang.org/x/net v0.44.0 h1:evd8IRDyfNBMBTTY5XRF1vaZlD+EmWx6x8PkhR04H/I=
|
||||
golang.org/x/net v0.44.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY=
|
||||
golang.org/x/sync v0.17.0 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug=
|
||||
golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
||||
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
|
||||
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
||||
golang.org/x/text v0.29.0 h1:1neNs90w9YzJ9BocxfsQNHKuAT4pkghyXc4nhZ6sJvk=
|
||||
golang.org/x/text v0.29.0/go.mod h1:7MhJOA9CD2qZyOKYazxdYMF85OwPdEr9jTtBpO7ydH4=
|
||||
gonum.org/v1/gonum v0.16.0 h1:5+ul4Swaf3ESvrOnidPp4GZbzf0mxVQpDCYUQE7OJfk=
|
||||
gonum.org/v1/gonum v0.16.0/go.mod h1:fef3am4MQ93R2HHpKnLk4/Tbh/s0+wqD5nfa6Pnwy4E=
|
||||
google.golang.org/genproto/googleapis/api v0.0.0-20250825161204-c5933d9347a5 h1:BIRfGDEjiHRrk0QKZe3Xv2ieMhtgRGeLcZQ0mIVn4EY=
|
||||
google.golang.org/genproto/googleapis/api v0.0.0-20250825161204-c5933d9347a5/go.mod h1:j3QtIyytwqGr1JUDtYXwtMXWPKsEa5LtzIFN1Wn5WvE=
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250825161204-c5933d9347a5 h1:eaY8u2EuxbRv7c3NiGK0/NedzVsCcV6hDuU5qPX5EGE=
|
||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20250825161204-c5933d9347a5/go.mod h1:M4/wBTSeyLxupu3W3tJtOgB14jILAS/XWPSSa3TAlJc=
|
||||
google.golang.org/grpc v1.76.0 h1:UnVkv1+uMLYXoIz6o7chp59WfQUYA2ex/BXQ9rHZu7A=
|
||||
google.golang.org/grpc v1.76.0/go.mod h1:Ju12QI8M6iQJtbcsV+awF5a4hfJMLi4X0JLo94ULZ6c=
|
||||
google.golang.org/protobuf v1.36.9 h1:w2gp2mA27hUeUzj9Ex9FBjsBm40zfaDtEWow293U7Iw=
|
||||
google.golang.org/protobuf v1.36.9/go.mod h1:fuxRtAxBytpl4zzqUh6/eyUujkJdNiuEkXntxiD/uRU=
|
||||
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
|
||||
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
|
||||
274
.dagger/main.go
Normal file
274
.dagger/main.go
Normal file
|
|
@ -0,0 +1,274 @@
|
|||
// A Dagger module for building and deploying a Zola static website
|
||||
//
|
||||
// This module provides modular functions to:
|
||||
// - Build the Zola website
|
||||
// - Create a Docker image serving the site with nginx
|
||||
// - Deploy to S3 (future)
|
||||
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"dagger/blog/internal/dagger"
|
||||
"fmt"
|
||||
"strings"
|
||||
)
|
||||
|
||||
const (
|
||||
// Zola version to use for building
|
||||
ZolaImage = "ghcr.io/getzola/zola:v0.21.0"
|
||||
// Nginx version to use for serving
|
||||
NginxImage = "nginx:1.27-alpine3.20-slim"
|
||||
// Build output directory
|
||||
BuildOutputDir = "public"
|
||||
// Lychee version for link checking
|
||||
LycheeImage = "lycheeverse/lychee:0.22-alpine"
|
||||
// htmltest version for HTML validation
|
||||
HtmltestImage = "wjdp/htmltest:v0.17.0"
|
||||
)
|
||||
|
||||
type Blog struct{}
|
||||
|
||||
// Build builds the Zola website and returns the public directory
|
||||
//
|
||||
// Args:
|
||||
// - source: The source directory containing the Zola site
|
||||
//
|
||||
// Returns:
|
||||
// - A Directory containing the built website
|
||||
func (m *Blog) Build(
|
||||
ctx context.Context,
|
||||
// The source directory containing the Zola site
|
||||
// +defaultPath="/"
|
||||
// +ignore=[".git", ".dagger", "public"]
|
||||
source *dagger.Directory,
|
||||
) *dagger.Directory {
|
||||
return dag.Container().
|
||||
From(ZolaImage).
|
||||
WithMountedDirectory("/site", source).
|
||||
WithWorkdir("/site").
|
||||
WithExec([]string{"zola", "build", "--output-dir", BuildOutputDir, "--minify"}).
|
||||
Directory(BuildOutputDir)
|
||||
}
|
||||
|
||||
// prepareNginx prepares an nginx container with configuration
|
||||
func (m *Blog) prepareNginx() *dagger.Container {
|
||||
nginxConfig := `server {
|
||||
listen 80;
|
||||
listen [::]:80;
|
||||
server_name _;
|
||||
|
||||
root /usr/share/nginx/html;
|
||||
index index.html;
|
||||
|
||||
# Enable gzip compression
|
||||
gzip on;
|
||||
gzip_vary on;
|
||||
gzip_min_length 1024;
|
||||
gzip_types text/plain text/css text/xml text/javascript application/javascript application/xml+rss application/json;
|
||||
|
||||
location / {
|
||||
try_files $uri $uri/ =404;
|
||||
}
|
||||
|
||||
# Cache static assets
|
||||
location ~* \.(jpg|jpeg|png|gif|ico|css|js|svg|woff|woff2|ttf|eot)$ {
|
||||
expires 1y;
|
||||
add_header Cache-Control "public, immutable";
|
||||
}
|
||||
}`
|
||||
|
||||
return dag.Container().
|
||||
From(NginxImage).
|
||||
WithNewFile("/etc/nginx/conf.d/default.conf", nginxConfig).
|
||||
WithExposedPort(80)
|
||||
}
|
||||
|
||||
// BuildContainer creates a Docker container image with nginx serving the built website
|
||||
//
|
||||
// Args:
|
||||
// - source: The source directory containing the Zola site
|
||||
//
|
||||
// Returns:
|
||||
// - A Container ready to serve the website
|
||||
func (m *Blog) BuildContainer(
|
||||
ctx context.Context,
|
||||
// The source directory containing the Zola site
|
||||
// +defaultPath="/"
|
||||
// +ignore=[".git", ".dagger", "public"]
|
||||
source *dagger.Directory,
|
||||
) *dagger.Container {
|
||||
builtSite := m.Build(ctx, source)
|
||||
nginxContainer := m.prepareNginx()
|
||||
|
||||
return nginxContainer.WithDirectory("/usr/share/nginx/html", builtSite)
|
||||
}
|
||||
|
||||
// getDefaultTag returns a tag based on the latest git commit date
|
||||
func (m *Blog) getDefaultTag(
|
||||
ctx context.Context,
|
||||
source *dagger.Directory,
|
||||
) (string, error) {
|
||||
// Get the commit date in YYYYMMDDHHMMSS format
|
||||
output, err := dag.Container().
|
||||
From("alpine/git:latest").
|
||||
WithMountedDirectory("/repo", source).
|
||||
WithWorkdir("/repo").
|
||||
WithExec([]string{"git", "log", "-1", "--format=%cd", "--date=format:%Y%m%d%H%M%S"}).
|
||||
Stdout(ctx)
|
||||
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return strings.TrimSpace(output), nil
|
||||
}
|
||||
|
||||
// Publish publishes the container to a registry
|
||||
//
|
||||
// Args:
|
||||
// - source: The source directory containing the Zola site
|
||||
// - registry: The registry address (e.g., "ghcr.io/username/blog")
|
||||
// - tag: The tag to use (defaults to git commit date in YYYYMMDDHHMMSS format)
|
||||
//
|
||||
// Returns:
|
||||
// - The published image reference
|
||||
func (m *Blog) Publish(
|
||||
ctx context.Context,
|
||||
// The source directory containing the Zola site
|
||||
// +defaultPath="/"
|
||||
// +ignore=[".git", ".dagger", "public"]
|
||||
source *dagger.Directory,
|
||||
// The registry address
|
||||
registry string,
|
||||
// +optional
|
||||
tag string,
|
||||
) (string, error) {
|
||||
if tag == "" {
|
||||
defaultTag, err := m.getDefaultTag(ctx, source)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
tag = defaultTag
|
||||
}
|
||||
|
||||
container := m.BuildContainer(ctx, source)
|
||||
address := registry + ":" + tag
|
||||
|
||||
return container.Publish(ctx, address)
|
||||
}
|
||||
|
||||
// checkLinks validates all links in the built site
|
||||
func (m *Blog) checkLinks(
|
||||
ctx context.Context,
|
||||
builtSite *dagger.Directory,
|
||||
) error {
|
||||
_, err := dag.Container().
|
||||
From(LycheeImage).
|
||||
WithMountedDirectory("/site", builtSite).
|
||||
WithWorkdir("/site").
|
||||
WithExec([]string{
|
||||
"lychee",
|
||||
"--base-url", "https://enoent.fr",
|
||||
"--remap", "https://enoent\\.fr/ file:///site/",
|
||||
"--offline",
|
||||
"--no-progress",
|
||||
"/site",
|
||||
}).
|
||||
Sync(ctx)
|
||||
|
||||
return err
|
||||
}
|
||||
|
||||
// checkHTML validates HTML structure
|
||||
func (m *Blog) checkHTML(
|
||||
ctx context.Context,
|
||||
builtSite *dagger.Directory,
|
||||
) error {
|
||||
// htmltest configuration
|
||||
// Enable comprehensive HTML validation
|
||||
htmltestConfig := `DirectoryPath: /site
|
||||
CheckDoctype: true
|
||||
CheckAnchors: true
|
||||
CheckLinks: true
|
||||
CheckImages: true
|
||||
CheckScripts: true
|
||||
CheckFavicon: false
|
||||
CheckMetaRefresh: true
|
||||
CheckMetaDescription: true
|
||||
EnforceHTTPS: false
|
||||
IgnoreInternalEmptyHash: true
|
||||
IgnoreDirectoryMissingTrailingSlash: true
|
||||
CheckExternal: false
|
||||
IgnoreURLs:
|
||||
- "^https://enoent\\.fr/"`
|
||||
|
||||
_, err := dag.Container().
|
||||
From(HtmltestImage).
|
||||
WithMountedDirectory("/site", builtSite).
|
||||
WithWorkdir("/test").
|
||||
WithNewFile(".htmltest.yml", htmltestConfig).
|
||||
WithExec([]string{"htmltest"}).
|
||||
Sync(ctx)
|
||||
|
||||
return err
|
||||
}
|
||||
|
||||
// Check runs all quality checks on the built site
|
||||
//
|
||||
// Args:
|
||||
// - source: The source directory containing the Zola site
|
||||
//
|
||||
// Returns:
|
||||
// - An error if any check fails
|
||||
func (m *Blog) Check(
|
||||
ctx context.Context,
|
||||
// The source directory containing the Zola site
|
||||
// +defaultPath="/"
|
||||
// +ignore=[".git", ".dagger", "public"]
|
||||
source *dagger.Directory,
|
||||
) error {
|
||||
// Build the site first
|
||||
builtSite := m.Build(ctx, source)
|
||||
|
||||
type checkResult struct {
|
||||
name string
|
||||
err error
|
||||
}
|
||||
|
||||
checks := []struct {
|
||||
name string
|
||||
fn func(context.Context, *dagger.Directory) error
|
||||
}{
|
||||
{"Link Check", m.checkLinks},
|
||||
{"HTML Validation", m.checkHTML},
|
||||
}
|
||||
|
||||
results := make(chan checkResult, len(checks))
|
||||
|
||||
for _, check := range checks {
|
||||
check := check
|
||||
go func() {
|
||||
results <- checkResult{
|
||||
name: check.name,
|
||||
err: check.fn(ctx, builtSite),
|
||||
}
|
||||
}()
|
||||
}
|
||||
|
||||
// Collect results
|
||||
var failures []string
|
||||
for i := 0; i < len(checks); i++ {
|
||||
result := <-results
|
||||
if result.err != nil {
|
||||
failures = append(failures, fmt.Sprintf("%s failed: %v", result.name, result.err))
|
||||
}
|
||||
}
|
||||
|
||||
// Return combined error if any checks failed
|
||||
if len(failures) > 0 {
|
||||
return fmt.Errorf("checks failed:\n%s", strings.Join(failures, "\n"))
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue