error-pages/internal/cli/shared/flags.go

135 lines
3.8 KiB
Go
Raw Normal View History

2023-01-29 10:54:56 +00:00
package shared
import (
2024-06-21 13:13:27 +00:00
"fmt"
"net"
2024-06-21 23:32:10 +00:00
"os"
"strings"
2023-01-29 10:54:56 +00:00
2024-06-21 13:13:27 +00:00
"github.com/urfave/cli/v3"
2024-06-29 12:34:03 +00:00
"gh.tarampamp.am/error-pages/internal/config"
2023-01-29 10:54:56 +00:00
)
2024-06-21 23:32:10 +00:00
// Note: Don't use pointers for flags, because they have own state which is not thread-safe.
// https://github.com/urfave/cli/issues/1926
2024-06-21 13:13:27 +00:00
var ListenAddrFlag = cli.StringFlag{
Name: "listen",
Aliases: []string{"l"},
Usage: "IP (v4 or v6) address to listen on",
Value: "0.0.0.0", // bind to all interfaces by default
Sources: cli.EnvVars("LISTEN_ADDR"),
OnlyOnce: true,
Config: cli.StringConfig{TrimSpace: true},
Validator: func(ip string) error {
if ip == "" {
return fmt.Errorf("missing IP address")
}
if net.ParseIP(ip) == nil {
return fmt.Errorf("wrong IP address [%s] for listening", ip)
}
2023-01-29 10:54:56 +00:00
2024-06-21 13:13:27 +00:00
return nil
},
2023-01-29 10:54:56 +00:00
}
2024-06-21 13:13:27 +00:00
var ListenPortFlag = cli.UintFlag{
Name: "port",
Aliases: []string{"p"},
Usage: "TCP port number",
Value: 8080, // default port number
Sources: cli.EnvVars("LISTEN_PORT"),
OnlyOnce: true,
Validator: func(port uint64) error {
if port == 0 || port > 65535 {
return fmt.Errorf("wrong TCP port number [%d]", port)
}
return nil
},
2023-01-29 10:54:56 +00:00
}
2024-06-21 23:32:10 +00:00
2024-06-26 10:52:21 +00:00
var AddTemplatesFlag = cli.StringSliceFlag{
2024-06-21 23:32:10 +00:00
Name: "add-template",
Usage: "to add a new template, provide the path to the file using this flag (the filename without the extension " +
"will be used as the template name)",
Config: cli.StringConfig{TrimSpace: true},
Validator: func(paths []string) error {
for _, path := range paths {
if path == "" {
return fmt.Errorf("missing template path")
}
if stat, err := os.Stat(path); err != nil || stat.IsDir() {
return fmt.Errorf("wrong template path [%s]", path)
}
}
return nil
},
}
2024-06-26 10:52:21 +00:00
var DisableTemplateNamesFlag = cli.StringSliceFlag{
Name: "disable-template",
2024-06-29 15:37:24 +00:00
Usage: "disable the specified template by its name (useful to disable the built-in templates and use only custom ones)",
2024-06-26 10:52:21 +00:00
Config: cli.StringConfig{TrimSpace: true},
}
var AddHTTPCodesFlag = cli.StringMapFlag{
2024-06-29 15:11:21 +00:00
Name: "add-http-code",
Aliases: []string{"add-code"},
2024-06-21 23:32:10 +00:00
Usage: "to add a new HTTP status code, provide the code and its message/description using this flag (the format " +
2024-06-29 15:37:24 +00:00
"should be '%code%=%message%/%description%'; the code may contain a wildcard '*' to cover multiple codes at " +
"once, for example, '4**' will cover all 4xx codes unless a more specific code is described previously)",
2024-06-21 23:32:10 +00:00
Config: cli.StringConfig{TrimSpace: true},
Validator: func(codes map[string]string) error {
for code, msgAndDesc := range codes {
if code == "" {
return fmt.Errorf("missing HTTP code")
} else if len(code) != 3 {
return fmt.Errorf("wrong HTTP code [%s]: it should be 3 characters long", code)
}
if parts := strings.SplitN(msgAndDesc, "/", 3); len(parts) < 1 || len(parts) > 2 {
return fmt.Errorf("wrong message/description format for HTTP code [%s]: %s", code, msgAndDesc)
} else if parts[0] == "" {
return fmt.Errorf("missing message for HTTP code [%s]", code)
}
}
return nil
},
}
2024-06-29 12:34:03 +00:00
// ParseHTTPCodes converts a map of HTTP status codes and their messages/descriptions into a map of codes and
// descriptions. Should be used together with [AddHTTPCodesFlag].
func ParseHTTPCodes(codes map[string]string) map[string]config.CodeDescription {
var result = make(map[string]config.CodeDescription, len(codes))
for code, msgAndDesc := range codes {
var (
parts = strings.SplitN(msgAndDesc, "/", 2)
desc config.CodeDescription
)
desc.Message = strings.TrimSpace(parts[0])
if len(parts) > 1 {
desc.Description = strings.TrimSpace(parts[1])
}
result[code] = desc
}
return result
}
var DisableL10nFlag = cli.BoolFlag{
Name: "disable-l10n",
Usage: "disable localization of error pages (if the template supports localization)",
Sources: cli.EnvVars("DISABLE_L10N"),
OnlyOnce: true,
}