2021-09-29 15:38:50 +00:00
|
|
|
package build
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/tarampampam/error-pages/internal/config"
|
|
|
|
"github.com/tarampampam/error-pages/internal/tpl"
|
|
|
|
"go.uber.org/zap"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewCommand creates `build` command.
|
2022-01-27 12:29:49 +00:00
|
|
|
func NewCommand(log *zap.Logger, configFile *string) *cobra.Command {
|
2021-09-29 15:38:50 +00:00
|
|
|
var (
|
|
|
|
generateIndex bool
|
2022-04-12 10:34:35 +00:00
|
|
|
disableL10n bool
|
2021-09-29 15:38:50 +00:00
|
|
|
cfg *config.Config
|
|
|
|
)
|
|
|
|
|
|
|
|
cmd := &cobra.Command{
|
|
|
|
Use: "build <output-directory>",
|
|
|
|
Aliases: []string{"b"},
|
|
|
|
Short: "Build the error pages",
|
|
|
|
Args: cobra.ExactArgs(1),
|
2022-01-27 12:29:49 +00:00
|
|
|
PreRunE: func(*cobra.Command, []string) (err error) {
|
2021-09-29 15:38:50 +00:00
|
|
|
if configFile == nil {
|
|
|
|
return errors.New("path to the config file is required for this command")
|
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if cfg, err = config.FromYamlFile(*configFile); err != nil {
|
2021-09-29 15:38:50 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
return
|
2021-09-29 15:38:50 +00:00
|
|
|
},
|
|
|
|
RunE: func(_ *cobra.Command, args []string) error {
|
|
|
|
if len(args) != 1 {
|
|
|
|
return errors.New("wrong arguments count")
|
|
|
|
}
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
return run(log, cfg, args[0], generateIndex, disableL10n)
|
2022-01-27 12:29:49 +00:00
|
|
|
},
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
cmd.Flags().BoolVarP(
|
|
|
|
&generateIndex,
|
|
|
|
"index", "i",
|
|
|
|
false,
|
|
|
|
"generate index page",
|
|
|
|
)
|
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
cmd.Flags().BoolVarP(
|
|
|
|
&disableL10n,
|
|
|
|
"disable-l10n", "",
|
|
|
|
false,
|
|
|
|
"disable error pages localization",
|
|
|
|
)
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
return cmd
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
const (
|
|
|
|
outHTMLFileExt = ".html"
|
|
|
|
outIndexFileName = "index"
|
|
|
|
outFilePerm = os.FileMode(0664)
|
|
|
|
outDirPerm = os.FileMode(0775)
|
|
|
|
)
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-04-12 10:34:35 +00:00
|
|
|
func run(log *zap.Logger, cfg *config.Config, outDirectoryPath string, generateIndex, disableL10n bool) error { //nolint:funlen,lll
|
2022-01-27 12:29:49 +00:00
|
|
|
if len(cfg.Templates) == 0 {
|
|
|
|
return errors.New("no loaded templates")
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
log.Info("output directory preparing", zap.String("path", outDirectoryPath))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if err := createDirectory(outDirectoryPath, outDirPerm); err != nil {
|
|
|
|
return errors.Wrap(err, "cannot prepare output directory")
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-31 08:43:40 +00:00
|
|
|
history, renderer := newBuildingHistory(), tpl.NewTemplateRenderer()
|
|
|
|
defer func() { _ = renderer.Close() }()
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
for _, template := range cfg.Templates {
|
|
|
|
log.Debug("template processing", zap.String("name", template.Name()))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
for _, page := range cfg.Pages {
|
|
|
|
if err := createDirectory(path.Join(outDirectoryPath, template.Name()), outDirPerm); err != nil {
|
2021-10-06 17:38:00 +00:00
|
|
|
return err
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
var (
|
|
|
|
fileName = page.Code() + outHTMLFileExt
|
|
|
|
filePath = path.Join(outDirectoryPath, template.Name(), fileName)
|
|
|
|
)
|
|
|
|
|
2022-01-31 08:43:40 +00:00
|
|
|
content, renderingErr := renderer.Render(template.Content(), tpl.Properties{
|
2022-01-27 12:29:49 +00:00
|
|
|
Code: page.Code(),
|
|
|
|
Message: page.Message(),
|
|
|
|
Description: page.Description(),
|
|
|
|
ShowRequestDetails: false,
|
2022-04-12 10:34:35 +00:00
|
|
|
L10nDisabled: disableL10n,
|
2022-01-27 12:29:49 +00:00
|
|
|
})
|
|
|
|
if renderingErr != nil {
|
|
|
|
return renderingErr
|
|
|
|
}
|
2021-10-15 06:06:10 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if err := os.WriteFile(filePath, content, outFilePerm); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
log.Debug("page rendered", zap.String("path", filePath))
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if generateIndex {
|
|
|
|
history.Append(
|
|
|
|
template.Name(),
|
|
|
|
page.Code(),
|
|
|
|
page.Message(),
|
|
|
|
path.Join(template.Name(), fileName),
|
|
|
|
)
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
2022-01-27 12:29:49 +00:00
|
|
|
}
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
if generateIndex {
|
|
|
|
var filepath = path.Join(outDirectoryPath, outIndexFileName+outHTMLFileExt)
|
|
|
|
|
|
|
|
log.Info("index file generation", zap.String("path", filepath))
|
|
|
|
|
|
|
|
if err := history.WriteIndexFile(filepath, outFilePerm); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
log.Info("job is done")
|
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
|
|
|
}
|
|
|
|
|
2022-01-27 12:29:49 +00:00
|
|
|
func createDirectory(path string, perm os.FileMode) error {
|
2021-09-29 15:38:50 +00:00
|
|
|
stat, err := os.Stat(path)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
2022-01-27 12:29:49 +00:00
|
|
|
return os.MkdirAll(path, perm)
|
2021-09-29 15:38:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !stat.IsDir() {
|
|
|
|
return errors.New("is not a directory")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|