2021-09-29 15:38:50 +00:00
|
|
|
package build
|
|
|
|
|
|
|
|
import (
|
2024-07-03 14:12:13 +00:00
|
|
|
"context"
|
|
|
|
_ "embed"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"html/template"
|
2021-09-29 15:38:50 +00:00
|
|
|
"os"
|
|
|
|
"path"
|
2024-07-03 14:12:13 +00:00
|
|
|
"path/filepath"
|
|
|
|
"slices"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
"github.com/urfave/cli/v3"
|
2023-01-29 10:54:56 +00:00
|
|
|
|
2023-02-23 17:49:45 +00:00
|
|
|
"gh.tarampamp.am/error-pages/internal/cli/shared"
|
|
|
|
"gh.tarampamp.am/error-pages/internal/config"
|
2024-07-03 14:12:13 +00:00
|
|
|
"gh.tarampamp.am/error-pages/internal/logger"
|
|
|
|
appTemplate "gh.tarampamp.am/error-pages/internal/template"
|
2021-09-29 15:38:50 +00:00
|
|
|
)
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
//go:embed index.html
|
|
|
|
var indexHtml string
|
|
|
|
|
2023-01-29 10:54:56 +00:00
|
|
|
type command struct {
|
|
|
|
c *cli.Command
|
2024-07-03 14:12:13 +00:00
|
|
|
|
|
|
|
opt struct {
|
|
|
|
createIndex bool
|
|
|
|
targetDirAbsPath string
|
|
|
|
}
|
2023-01-29 10:54:56 +00:00
|
|
|
}
|
|
|
|
|
2021-09-29 15:38:50 +00:00
|
|
|
// NewCommand creates `build` command.
|
2024-07-03 14:12:13 +00:00
|
|
|
func NewCommand(log *logger.Logger) *cli.Command { //nolint:funlen,gocognit
|
|
|
|
var (
|
|
|
|
cmd command
|
|
|
|
cfg = config.New()
|
|
|
|
|
2024-07-05 14:59:07 +00:00
|
|
|
addTplFlag = shared.AddTemplatesFlag
|
|
|
|
disableTplFlag = shared.DisableTemplateNamesFlag
|
|
|
|
addCodeFlag = shared.AddHTTPCodesFlag
|
|
|
|
disableL10nFlag = shared.DisableL10nFlag
|
|
|
|
disableMinificationFlag = shared.DisableMinificationFlag
|
|
|
|
createIndexFlag = cli.BoolFlag{
|
2024-07-03 14:12:13 +00:00
|
|
|
Name: "index",
|
|
|
|
Aliases: []string{"i"},
|
|
|
|
Usage: "Generate index.html file with links to all error pages",
|
|
|
|
Category: shared.CategoryBuild,
|
|
|
|
}
|
|
|
|
targetDirFlag = cli.StringFlag{
|
|
|
|
Name: "target-dir",
|
|
|
|
Aliases: []string{"out", "dir", "o"},
|
|
|
|
Usage: "Directory to put the built error pages into",
|
|
|
|
Value: ".", // current directory by default
|
|
|
|
Config: cli.StringConfig{TrimSpace: true},
|
|
|
|
Category: shared.CategoryBuild,
|
|
|
|
OnlyOnce: true,
|
|
|
|
Validator: func(dir string) error {
|
|
|
|
if dir == "" {
|
|
|
|
return errors.New("missing target directory")
|
|
|
|
}
|
|
|
|
|
|
|
|
if stat, err := os.Stat(dir); err != nil {
|
|
|
|
return fmt.Errorf("cannot access the target directory '%s': %w", dir, err)
|
|
|
|
} else if !stat.IsDir() {
|
|
|
|
return fmt.Errorf("'%s' is not a directory", dir)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
)
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
disableL10nFlag.Value = cfg.L10n.Disable // set the default value depending on the configuration
|
|
|
|
|
2023-01-29 10:54:56 +00:00
|
|
|
cmd.c = &cli.Command{
|
2024-07-03 14:12:13 +00:00
|
|
|
Name: "build",
|
|
|
|
Aliases: []string{"b"},
|
|
|
|
Usage: "Build the static error pages and put them into a specified directory",
|
|
|
|
Action: func(ctx context.Context, c *cli.Command) error {
|
|
|
|
cfg.L10n.Disable = c.Bool(disableL10nFlag.Name)
|
2024-07-05 14:59:07 +00:00
|
|
|
cfg.DisableMinification = c.Bool(disableMinificationFlag.Name)
|
2024-07-03 14:12:13 +00:00
|
|
|
cmd.opt.createIndex = c.Bool(createIndexFlag.Name)
|
|
|
|
cmd.opt.targetDirAbsPath, _ = filepath.Abs(c.String(targetDirFlag.Name)) // an error checked by [os.Stat] validator
|
|
|
|
|
|
|
|
// add templates from files to the configuration
|
|
|
|
if add := c.StringSlice(addTplFlag.Name); len(add) > 0 {
|
|
|
|
for _, templatePath := range add {
|
|
|
|
if addedName, err := cfg.Templates.AddFromFile(templatePath); err != nil {
|
|
|
|
return fmt.Errorf("cannot add template from file %s: %w", templatePath, err)
|
|
|
|
} else {
|
|
|
|
log.Info("Template added",
|
|
|
|
logger.String("name", addedName),
|
|
|
|
logger.String("path", templatePath),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
// disable templates specified by the user
|
|
|
|
if disable := c.StringSlice(disableTplFlag.Name); len(disable) > 0 {
|
|
|
|
for _, templateName := range disable {
|
|
|
|
if ok := cfg.Templates.Remove(templateName); ok {
|
|
|
|
log.Info("Template disabled", logger.String("name", templateName))
|
|
|
|
}
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
// add custom HTTP codes to the configuration
|
|
|
|
if add := c.StringMap(addCodeFlag.Name); len(add) > 0 {
|
|
|
|
for code, desc := range shared.ParseHTTPCodes(add) {
|
|
|
|
cfg.Codes[code] = desc
|
|
|
|
|
|
|
|
log.Info("HTTP code added",
|
|
|
|
logger.String("code", code),
|
|
|
|
logger.String("message", desc.Message),
|
|
|
|
logger.String("description", desc.Description),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(cfg.Templates) == 0 {
|
|
|
|
return errors.New("no templates specified")
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("Building error pages",
|
|
|
|
logger.String("targetDir", cmd.opt.targetDirAbsPath),
|
|
|
|
logger.Strings("templates", cfg.Templates.Names()...),
|
|
|
|
logger.Bool("index", cmd.opt.createIndex),
|
|
|
|
logger.Bool("l10n", !cfg.L10n.Disable),
|
|
|
|
)
|
|
|
|
|
|
|
|
return cmd.Run(ctx, log, &cfg)
|
2023-01-29 10:54:56 +00:00
|
|
|
},
|
2024-07-03 14:12:13 +00:00
|
|
|
Flags: []cli.Flag{
|
|
|
|
&addTplFlag,
|
|
|
|
&disableTplFlag,
|
|
|
|
&addCodeFlag,
|
|
|
|
&disableL10nFlag,
|
|
|
|
&createIndexFlag,
|
|
|
|
&targetDirFlag,
|
2024-07-05 14:59:07 +00:00
|
|
|
&disableMinificationFlag,
|
2022-01-27 12:29:49 +00:00
|
|
|
},
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2023-01-29 10:54:56 +00:00
|
|
|
return cmd.c
|
2022-01-27 12:29:49 +00:00
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-05 14:59:07 +00:00
|
|
|
func (cmd *command) Run( //nolint:funlen,gocognit
|
2024-07-03 14:12:13 +00:00
|
|
|
ctx context.Context,
|
|
|
|
log *logger.Logger,
|
|
|
|
cfg *config.Config,
|
|
|
|
) error {
|
|
|
|
type historyItem struct{ Code, Message, RelativePath string }
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
var history = make(map[string][]historyItem, len(cfg.Codes)*len(cfg.Templates)) // map[template_name]codes
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
for templateName, templateContent := range cfg.Templates {
|
|
|
|
log.Debug("Processing template", logger.String("name", templateName))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
for code, codeDescription := range cfg.Codes {
|
|
|
|
if err := createDirectory(filepath.Join(cmd.opt.targetDirAbsPath, templateName)); err != nil {
|
|
|
|
return fmt.Errorf("cannot create directory for template '%s': %w", templateName, err)
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
var codeAsUint, codeParsingErr = strconv.ParseUint(code, 10, 32)
|
|
|
|
if codeParsingErr != nil {
|
|
|
|
log.Warn("Cannot parse code", logger.String("code", code))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
continue
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
var outFilePath = path.Join(cmd.opt.targetDirAbsPath, templateName, code+".html")
|
2022-01-27 12:29:49 +00:00
|
|
|
|
2024-07-05 14:59:07 +00:00
|
|
|
if content, renderErr := appTemplate.Render(templateContent, appTemplate.Props{ //nolint:nestif
|
2024-07-03 14:12:13 +00:00
|
|
|
Code: uint16(codeAsUint),
|
|
|
|
Message: codeDescription.Message,
|
|
|
|
Description: codeDescription.Description,
|
|
|
|
L10nDisabled: cfg.L10n.Disable,
|
2022-01-27 12:29:49 +00:00
|
|
|
ShowRequestDetails: false,
|
2024-07-03 14:12:13 +00:00
|
|
|
}); renderErr == nil {
|
2024-07-05 14:59:07 +00:00
|
|
|
if !cfg.DisableMinification {
|
|
|
|
if mini, minErr := appTemplate.MiniHTML(content); minErr != nil {
|
|
|
|
log.Warn("Cannot minify the content", logger.Error(minErr))
|
|
|
|
} else {
|
|
|
|
content = mini
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
if err := os.WriteFile(outFilePath, []byte(content), os.FileMode(0664)); err != nil { //nolint:mnd
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return fmt.Errorf("cannot render template '%s': %w", templateName, renderErr)
|
2022-01-27 12:29:49 +00:00
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
log.Debug("Page built", logger.String("template", templateName), logger.String("code", code))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
history[templateName] = append(history[templateName], historyItem{
|
|
|
|
Code: code,
|
|
|
|
Message: codeDescription.Message,
|
|
|
|
RelativePath: "." + strings.TrimPrefix(outFilePath, cmd.opt.targetDirAbsPath), // to make it relative
|
|
|
|
})
|
2022-01-27 12:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
if cmd.opt.createIndex {
|
|
|
|
log.Debug("Creating the index file")
|
|
|
|
|
|
|
|
for name := range history {
|
|
|
|
slices.SortFunc(history[name], func(a, b historyItem) int { return strings.Compare(a.Code, b.Code) })
|
|
|
|
}
|
|
|
|
|
|
|
|
indexTpl, tplErr := template.New("index").Parse(indexHtml)
|
|
|
|
if tplErr != nil {
|
|
|
|
return tplErr
|
|
|
|
}
|
2022-01-27 12:29:49 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
var buf strings.Builder
|
2022-01-27 12:29:49 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
if err := indexTpl.Execute(&buf, history); err != nil {
|
2022-01-27 12:29:49 +00:00
|
|
|
return err
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
return os.WriteFile(
|
|
|
|
filepath.Join(cmd.opt.targetDirAbsPath, "index.html"),
|
|
|
|
[]byte(buf.String()),
|
|
|
|
os.FileMode(0664), //nolint:mnd
|
|
|
|
)
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
return nil
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2024-07-03 14:12:13 +00:00
|
|
|
func createDirectory(path string) error {
|
|
|
|
var stat, err = os.Stat(path)
|
2021-09-29 15:38:50 +00:00
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
2024-07-03 14:12:13 +00:00
|
|
|
return os.MkdirAll(path, os.FileMode(0775)) //nolint:mnd
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !stat.IsDir() {
|
|
|
|
return errors.New("is not a directory")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|