diff options
author | Alan Pearce | 2024-06-23 17:22:09 +0200 |
---|---|---|
committer | Alan Pearce | 2024-06-23 19:23:20 +0200 |
commit | fe4a23b550b7450e6e24cd3c78dfd575122d361d (patch) | |
tree | 61e4f2158455ad8e7c05a7c8f219e909a3239263 /internal | |
parent | 3a7e71544045647fa3ac702a1f0c5ad14623cb98 (diff) | |
download | website-fe4a23b550b7450e6e24cd3c78dfd575122d361d.tar.lz website-fe4a23b550b7450e6e24cd3c78dfd575122d361d.tar.zst website-fe4a23b550b7450e6e24cd3c78dfd575122d361d.zip |
pre-compress generated files in production
Diffstat (limited to 'internal')
-rw-r--r-- | internal/builder/builder.go | 24 | ||||
-rw-r--r-- | internal/builder/files.go | 83 |
2 files changed, 98 insertions, 9 deletions
diff --git a/internal/builder/builder.go b/internal/builder/builder.go index 44ab402..4f305db 100644 --- a/internal/builder/builder.go +++ b/internal/builder/builder.go @@ -32,30 +32,34 @@ type Result struct { Hashes []string } +var compressFiles = false + func mkdirp(dirs ...string) error { err := os.MkdirAll(path.Join(dirs...), 0755) return errors.Wrap(err, "could not create directory") } -func outputToFile(output io.Reader, filename ...string) error { - // log.Debug("outputting file", "filename", path.Join(filename...)) - file, err := os.OpenFile(path.Join(filename...), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644) +func outputToFile(output io.Reader, pathParts ...string) error { + filename := path.Join(pathParts...) + // log.Debug("outputting file", "filename", filename) + file, err := openFileAndVariants(filename) if err != nil { return errors.WithMessage(err, "could not open output file") } defer file.Close() - if _, err := file.ReadFrom(output); err != nil { + if _, err := io.Copy(file, output); err != nil { return errors.WithMessage(err, "could not write output file") } return nil } -func renderToFile(component templ.Component, filename ...string) error { - // log.Debug("outputting file", "filename", path.Join(filename...)) - file, err := os.OpenFile(path.Join(filename...), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644) +func renderToFile(component templ.Component, pathParts ...string) error { + filename := path.Join(pathParts...) + // log.Debug("outputting file", "filename", filename) + file, err := openFileAndVariants(filename) if err != nil { return errors.WithMessage(err, "could not open output file") } @@ -68,9 +72,10 @@ func renderToFile(component templ.Component, filename ...string) error { return nil } -func writerToFile(writer io.WriterTo, filename ...string) error { +func writerToFile(writer io.WriterTo, pathParts ...string) error { + filename := path.Join(pathParts...) // log.Debug("outputting file", "filename", path.Join(filename...)) - file, err := os.OpenFile(path.Join(filename...), os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644) + file, err := openFileAndVariants(filename) if err != nil { return errors.WithMessage(err, "could not open output file") } @@ -252,6 +257,7 @@ func BuildSite(ioConfig IOConfig) (*Result, error) { return nil, errors.WithMessage(err, "could not get config") } config.InjectLiveReload = ioConfig.Development + compressFiles = !ioConfig.Development if ioConfig.BaseURL.URL != nil { config.BaseURL.URL, err = url.Parse(ioConfig.BaseURL.String()) diff --git a/internal/builder/files.go b/internal/builder/files.go new file mode 100644 index 0000000..57d4d2e --- /dev/null +++ b/internal/builder/files.go @@ -0,0 +1,83 @@ +package builder + +import ( + "compress/gzip" + "io" + "os" + + "github.com/andybalholm/brotli" +) + +const ( + gzipLevel = 6 + brotliLevel = 9 +) + +type MultiWriteCloser struct { + writers []io.WriteCloser + multiWriter io.Writer +} + +func (mw *MultiWriteCloser) Write(p []byte) (n int, err error) { + return mw.multiWriter.Write(p) +} + +func (mw *MultiWriteCloser) Close() error { + var lastErr error + for _, w := range mw.writers { + err := w.Close() + if err != nil { + lastErr = err + } + } + return lastErr +} + +func openFileWrite(filename string) (*os.File, error) { + return os.OpenFile(filename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0644) +} + +func openFileGz(filename string) (*gzip.Writer, error) { + filenameGz := filename + ".gz" + f, err := openFileWrite(filenameGz) + if err != nil { + return nil, err + } + return gzip.NewWriterLevel(f, gzipLevel) +} + +func openFileBrotli(filename string) (*brotli.Writer, error) { + filenameBrotli := filename + ".br" + f, err := openFileWrite(filenameBrotli) + if err != nil { + return nil, err + } + return brotli.NewWriterLevel(f, brotliLevel), nil +} + +func multiOpenFile(filename string) (*MultiWriteCloser, error) { + r, err := openFileWrite(filename) + if err != nil { + return nil, err + } + gz, err := openFileGz(filename) + if err != nil { + return nil, err + } + br, err := openFileBrotli(filename) + if err != nil { + return nil, err + } + return &MultiWriteCloser{ + writers: []io.WriteCloser{r, gz, br}, + multiWriter: io.MultiWriter(r, gz, br), + }, nil +} + +func openFileAndVariants(filename string) (io.WriteCloser, error) { + if compressFiles { + return multiOpenFile(filename) + } else { + return openFileWrite(filename) + } +} |