2023-03-12 16:00:57 +01:00
|
|
|
// GoToSocial
|
|
|
|
|
// Copyright (C) GoToSocial Authors admin@gotosocial.org
|
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2022-05-30 13:41:24 +01:00
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"flag"
|
|
|
|
|
"fmt"
|
2022-12-11 13:03:15 +00:00
|
|
|
"io"
|
2022-05-30 13:41:24 +01:00
|
|
|
"os"
|
|
|
|
|
"os/exec"
|
|
|
|
|
"reflect"
|
2025-05-31 17:30:57 +02:00
|
|
|
"slices"
|
2022-12-11 13:03:15 +00:00
|
|
|
"strings"
|
2025-05-06 15:51:45 +00:00
|
|
|
"time"
|
2022-05-30 13:41:24 +01:00
|
|
|
|
2025-04-26 15:34:10 +02:00
|
|
|
"code.superseriousbusiness.org/gotosocial/internal/config"
|
2022-05-30 13:41:24 +01:00
|
|
|
)
|
|
|
|
|
|
2023-03-12 16:00:57 +01:00
|
|
|
const license = `// GoToSocial
|
|
|
|
|
// Copyright (C) GoToSocial Authors admin@gotosocial.org
|
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2022-05-30 13:41:24 +01:00
|
|
|
|
|
|
|
|
`
|
|
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
var durationType = reflect.TypeOf(time.Duration(0))
|
|
|
|
|
var stringerType = reflect.TypeOf((*interface{ String() string })(nil)).Elem()
|
|
|
|
|
var stringersType = reflect.TypeOf((*interface{ Strings() []string })(nil)).Elem()
|
|
|
|
|
var flagSetType = reflect.TypeOf((*interface{ Set(string) error })(nil)).Elem()
|
|
|
|
|
|
2022-05-30 13:41:24 +01:00
|
|
|
func main() {
|
2022-12-11 13:03:15 +00:00
|
|
|
var out string
|
2022-05-30 13:41:24 +01:00
|
|
|
|
|
|
|
|
// Load runtime config flags
|
|
|
|
|
flag.StringVar(&out, "out", "", "Generated file output path")
|
|
|
|
|
flag.Parse()
|
|
|
|
|
|
|
|
|
|
// Open output file path
|
|
|
|
|
output, err := os.OpenFile(out, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, 0o644)
|
|
|
|
|
if err != nil {
|
|
|
|
|
panic(err)
|
|
|
|
|
}
|
|
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
configType := reflect.TypeOf(config.Configuration{})
|
|
|
|
|
|
|
|
|
|
// Parse our config type for usable fields.
|
|
|
|
|
fields := loadConfigFields(nil, nil, configType)
|
|
|
|
|
|
|
|
|
|
fprintf(output, "// THIS IS A GENERATED FILE, DO NOT EDIT BY HAND\n")
|
|
|
|
|
fprintf(output, license)
|
|
|
|
|
fprintf(output, "package config\n\n")
|
|
|
|
|
fprintf(output, "import (\n")
|
|
|
|
|
fprintf(output, "\t\"fmt\"\n")
|
|
|
|
|
fprintf(output, "\t\"time\"\n\n")
|
|
|
|
|
fprintf(output, "\t\"codeberg.org/gruf/go-bytesize\"\n")
|
|
|
|
|
fprintf(output, "\t\"code.superseriousbusiness.org/gotosocial/internal/language\"\n")
|
|
|
|
|
fprintf(output, "\t\"github.com/spf13/pflag\"\n")
|
|
|
|
|
fprintf(output, "\t\"github.com/spf13/cast\"\n")
|
|
|
|
|
fprintf(output, ")\n")
|
|
|
|
|
fprintf(output, "\n")
|
2025-06-10 15:43:31 +02:00
|
|
|
generateFlagConsts(output, fields)
|
2025-05-06 15:51:45 +00:00
|
|
|
generateFlagRegistering(output, fields)
|
|
|
|
|
generateMapMarshaler(output, fields)
|
|
|
|
|
generateMapUnmarshaler(output, fields)
|
|
|
|
|
generateGetSetters(output, fields)
|
|
|
|
|
generateMapFlattener(output, fields)
|
|
|
|
|
must(output.Close())
|
|
|
|
|
must(exec.Command("gofumpt", "-w", out).Run())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type ConfigField struct {
|
|
|
|
|
// Any CLI flag prefixes,
|
|
|
|
|
// i.e. with nested fields.
|
|
|
|
|
Prefixes []string
|
|
|
|
|
|
|
|
|
|
// The base CLI flag
|
|
|
|
|
// name of the field.
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Path to struct field
|
|
|
|
|
// in dot-separated form.
|
|
|
|
|
Path string
|
|
|
|
|
|
|
|
|
|
// Usage string.
|
|
|
|
|
Usage string
|
|
|
|
|
|
|
|
|
|
// The underlying Go type
|
|
|
|
|
// of the config field.
|
|
|
|
|
Type reflect.Type
|
|
|
|
|
|
|
|
|
|
// i.e. is this found in the configuration file?
|
|
|
|
|
// or just used in specific CLI commands? in the
|
|
|
|
|
// future we'll remove these from config struct.
|
|
|
|
|
Ephemeral bool
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Flag returns the combined "prefixes-name" CLI flag for config field.
|
|
|
|
|
func (f ConfigField) Flag() string {
|
|
|
|
|
flag := strings.Join(append(f.Prefixes, f.Name), "-")
|
|
|
|
|
flag = strings.ToLower(flag)
|
|
|
|
|
return flag
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// PossibleKeys returns a list of possible map key combinations
|
|
|
|
|
// that this config field may be found under. The combined "prefixes-name"
|
|
|
|
|
// will always be in the list, but also separates them out to account for
|
|
|
|
|
// possible nesting. This allows us to support both nested and un-nested
|
|
|
|
|
// configuration files, always prioritizing "prefixes-name" as its the CLI flag.
|
|
|
|
|
func (f ConfigField) PossibleKeys() [][]string {
|
|
|
|
|
if len(f.Prefixes) == 0 {
|
|
|
|
|
return [][]string{{f.Name}}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var keys [][]string
|
|
|
|
|
|
|
|
|
|
combined := f.Flag()
|
|
|
|
|
keys = append(keys, []string{combined})
|
|
|
|
|
|
|
|
|
|
basePrefix := strings.TrimSuffix(combined, "-"+f.Name)
|
|
|
|
|
keys = append(keys, []string{basePrefix, f.Name})
|
|
|
|
|
|
|
|
|
|
for i := len(f.Prefixes) - 1; i >= 0; i-- {
|
|
|
|
|
prefix := f.Prefixes[i]
|
|
|
|
|
|
|
|
|
|
basePrefix = strings.TrimSuffix(basePrefix, prefix)
|
|
|
|
|
basePrefix = strings.TrimSuffix(basePrefix, "-")
|
|
|
|
|
if len(basePrefix) == 0 {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var key []string
|
|
|
|
|
key = append(key, basePrefix)
|
|
|
|
|
key = append(key, f.Prefixes[i:]...)
|
|
|
|
|
key = append(key, f.Name)
|
|
|
|
|
keys = append(keys, key)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return keys
|
2022-12-11 13:03:15 +00:00
|
|
|
}
|
|
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
func loadConfigFields(pathPrefixes, flagPrefixes []string, t reflect.Type) []ConfigField {
|
|
|
|
|
var out []ConfigField
|
2022-12-11 13:03:15 +00:00
|
|
|
for i := 0; i < t.NumField(); i++ {
|
2025-05-06 15:51:45 +00:00
|
|
|
// Struct field at index.
|
2022-12-11 13:03:15 +00:00
|
|
|
field := t.Field(i)
|
|
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
// Get field's tagged name.
|
|
|
|
|
name := field.Tag.Get("name")
|
|
|
|
|
if name == "" || name == "-" {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2022-12-11 13:03:15 +00:00
|
|
|
if ft := field.Type; ft.Kind() == reflect.Struct {
|
2025-05-06 15:51:45 +00:00
|
|
|
// This is a nested struct, load nested fields.
|
|
|
|
|
pathPrefixes := append(pathPrefixes, field.Name)
|
|
|
|
|
flagPrefixes := append(flagPrefixes, name)
|
|
|
|
|
out = append(out, loadConfigFields(pathPrefixes, flagPrefixes, ft)...)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Get prefixed, period-separated, config variable struct "path".
|
|
|
|
|
fieldPath := strings.Join(append(pathPrefixes, field.Name), ".")
|
|
|
|
|
|
|
|
|
|
// Append prepared ConfigField.
|
|
|
|
|
out = append(out, ConfigField{
|
|
|
|
|
Prefixes: flagPrefixes,
|
|
|
|
|
Name: name,
|
|
|
|
|
Path: fieldPath,
|
|
|
|
|
Usage: field.Tag.Get("usage"),
|
|
|
|
|
Ephemeral: field.Tag.Get("ephemeral") == "yes",
|
|
|
|
|
Type: field.Type,
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
return out
|
|
|
|
|
}
|
|
|
|
|
|
2025-06-10 15:43:31 +02:00
|
|
|
func generateFlagConsts(out io.Writer, fields []ConfigField) {
|
|
|
|
|
fprintf(out, "const (\n")
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
name := strings.ReplaceAll(field.Path, ".", "")
|
|
|
|
|
fprintf(out, "\t%sFlag = \"%s\"\n", name, field.Flag())
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, ")\n\n")
|
|
|
|
|
}
|
2025-05-06 15:51:45 +00:00
|
|
|
|
|
|
|
|
func generateFlagRegistering(out io.Writer, fields []ConfigField) {
|
|
|
|
|
fprintf(out, "func (cfg *Configuration) RegisterFlags(flags *pflag.FlagSet) {\n")
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
if field.Ephemeral {
|
|
|
|
|
// Skip registering
|
|
|
|
|
// ephemeral flags.
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for easy cases of just regular primitive types.
|
|
|
|
|
if field.Type.Kind().String() == field.Type.String() {
|
|
|
|
|
typeName := field.Type.String()
|
|
|
|
|
typeName = strings.ToUpper(typeName[:1]) + typeName[1:]
|
|
|
|
|
fprintf(out, "\tflags.%s(\"%s\", cfg.%s, \"%s\")\n", typeName, field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for easy cases of just
|
|
|
|
|
// regular primitive slice types.
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
elem := field.Type.Elem()
|
|
|
|
|
if elem.Kind().String() == elem.String() {
|
|
|
|
|
typeName := elem.String()
|
|
|
|
|
typeName = strings.ToUpper(typeName[:1]) + typeName[1:]
|
|
|
|
|
fprintf(out, "\tflags.%sSlice(\"%s\", cfg.%s, \"%s\")\n", typeName, field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Durations should get set directly
|
|
|
|
|
// as their types as viper knows how
|
|
|
|
|
// to deal with this type directly.
|
|
|
|
|
if field.Type == durationType {
|
|
|
|
|
fprintf(out, "\tflags.Duration(\"%s\", cfg.%s, \"%s\")\n", field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
// Check if the field supports Stringers{}.
|
|
|
|
|
if field.Type.Implements(stringersType) {
|
|
|
|
|
fprintf(out, "\tflags.StringSlice(\"%s\", cfg.%s.Strings(), \"%s\")\n", field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Or the pointer type of the field value supports Stringers{}.
|
|
|
|
|
if ptr := reflect.PointerTo(field.Type); ptr.Implements(stringersType) {
|
|
|
|
|
fprintf(out, "\tflags.StringSlice(\"%s\", cfg.%s.Strings(), \"%s\")\n", field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(os.Stderr, "field %s doesn't implement %s!\n", field.Path, stringersType)
|
|
|
|
|
} else {
|
|
|
|
|
// Check if the field supports Stringer{}.
|
|
|
|
|
if field.Type.Implements(stringerType) {
|
|
|
|
|
fprintf(out, "\tflags.String(\"%s\", cfg.%s.String(), \"%s\")\n", field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Or the pointer type of the field value supports Stringer{}.
|
|
|
|
|
if ptr := reflect.PointerTo(field.Type); ptr.Implements(stringerType) {
|
|
|
|
|
fprintf(out, "\tflags.String(\"%s\", cfg.%s.String(), \"%s\")\n", field.Flag(), field.Path, field.Usage)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(os.Stderr, "field %s doesn't implement %s!\n", field.Path, stringerType)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func generateMapMarshaler(out io.Writer, fields []ConfigField) {
|
|
|
|
|
fprintf(out, "func (cfg *Configuration) MarshalMap() map[string]any {\n")
|
|
|
|
|
fprintf(out, "\tcfgmap := make(map[string]any, %d)\n", len(fields))
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
// Check for easy cases of just regular primitive types.
|
|
|
|
|
if field.Type.Kind().String() == field.Type.String() {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for easy cases of just
|
|
|
|
|
// regular primitive slice types.
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
elem := field.Type.Elem()
|
|
|
|
|
if elem.Kind().String() == elem.String() {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Durations should get set directly
|
|
|
|
|
// as their types as viper knows how
|
|
|
|
|
// to deal with this type directly.
|
|
|
|
|
if field.Type == durationType {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
// Either the field must support Stringers{}.
|
|
|
|
|
if field.Type.Implements(stringersType) {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s.Strings()\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Or the pointer type of the field value must support Stringers{}.
|
|
|
|
|
if ptr := reflect.PointerTo(field.Type); ptr.Implements(stringersType) {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s.Strings()\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(os.Stderr, "field %s doesn't implement %s!\n", field.Path, stringersType)
|
|
|
|
|
} else {
|
|
|
|
|
// Either the field must support Stringer{}.
|
|
|
|
|
if field.Type.Implements(stringerType) {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s.String()\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Or the pointer type of the field value must support Stringer{}.
|
|
|
|
|
if ptr := reflect.PointerTo(field.Type); ptr.Implements(stringerType) {
|
|
|
|
|
fprintf(out, "\tcfgmap[\"%s\"] = cfg.%s.String()\n", field.Flag(), field.Path)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(os.Stderr, "field %s doesn't implement %s!\n", field.Path, stringerType)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\treturn cfgmap")
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func generateMapUnmarshaler(out io.Writer, fields []ConfigField) {
|
|
|
|
|
fprintf(out, "func (cfg *Configuration) UnmarshalMap(cfgmap map[string]any) error {\n")
|
|
|
|
|
fprintf(out, "// VERY IMPORTANT FIRST STEP!\n")
|
|
|
|
|
fprintf(out, "// flatten to normalize map to\n")
|
|
|
|
|
fprintf(out, "// entirely un-nested key values\n")
|
|
|
|
|
fprintf(out, "flattenConfigMap(cfgmap)\n")
|
|
|
|
|
fprintf(out, "\n")
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
// Check for easy cases of just regular primitive types.
|
|
|
|
|
if field.Type.Kind().String() == field.Type.String() {
|
|
|
|
|
generateUnmarshalerPrimitive(out, field)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for easy cases of just
|
|
|
|
|
// regular primitive slice types.
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
elem := field.Type.Elem()
|
|
|
|
|
if elem.Kind().String() == elem.String() {
|
|
|
|
|
generateUnmarshalerPrimitive(out, field)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Durations should get set directly
|
|
|
|
|
// as their types as viper knows how
|
|
|
|
|
// to deal with this type directly.
|
|
|
|
|
if field.Type == durationType {
|
|
|
|
|
generateUnmarshalerPrimitive(out, field)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Either the field must support flag.Value{}.
|
|
|
|
|
if field.Type.Implements(flagSetType) {
|
|
|
|
|
generateUnmarshalerFlagType(out, field)
|
2022-12-11 13:03:15 +00:00
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
// Or the pointer type of the field value must support flag.Value{}.
|
|
|
|
|
if ptr := reflect.PointerTo(field.Type); ptr.Implements(flagSetType) {
|
|
|
|
|
generateUnmarshalerFlagType(out, field)
|
|
|
|
|
continue
|
|
|
|
|
}
|
2022-12-11 13:03:15 +00:00
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
fprintf(os.Stderr, "field %s doesn't implement %s!\n", field.Path, flagSetType)
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\treturn nil\n")
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
}
|
2022-12-11 13:03:15 +00:00
|
|
|
|
2025-05-06 15:51:45 +00:00
|
|
|
func generateUnmarshalerPrimitive(out io.Writer, field ConfigField) {
|
|
|
|
|
fprintf(out, "\t\tif ival, ok := cfgmap[\"%s\"]; ok {\n", field.Flag())
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
elem := field.Type.Elem()
|
|
|
|
|
typeName := elem.String()
|
|
|
|
|
if i := strings.IndexRune(typeName, '.'); i >= 0 {
|
|
|
|
|
typeName = typeName[i+1:]
|
|
|
|
|
}
|
|
|
|
|
typeName = strings.ToUpper(typeName[:1]) + typeName[1:]
|
|
|
|
|
fprintf(out, "\t\t\tvar err error\n")
|
|
|
|
|
// note we specifically handle slice types ourselves to split by comma
|
|
|
|
|
fprintf(out, "\t\t\tcfg.%s, err = to%sSlice(ival)\n", field.Path, typeName)
|
|
|
|
|
fprintf(out, "\t\t\tif err != nil {\n")
|
|
|
|
|
fprintf(out, "\t\t\t\treturn fmt.Errorf(\"error casting %%#v -> []%s for '%s': %%w\", ival, err)\n", elem.String(), field.Flag())
|
|
|
|
|
fprintf(out, "\t\t\t}\n")
|
|
|
|
|
} else {
|
|
|
|
|
typeName := field.Type.String()
|
|
|
|
|
if i := strings.IndexRune(typeName, '.'); i >= 0 {
|
|
|
|
|
typeName = typeName[i+1:]
|
|
|
|
|
}
|
|
|
|
|
typeName = strings.ToUpper(typeName[:1]) + typeName[1:]
|
|
|
|
|
fprintf(out, "\t\t\tvar err error\n")
|
|
|
|
|
fprintf(out, "\t\t\tcfg.%s, err = cast.To%sE(ival)\n", field.Path, typeName)
|
|
|
|
|
fprintf(out, "\t\t\tif err != nil {\n")
|
|
|
|
|
fprintf(out, "\t\t\t\treturn fmt.Errorf(\"error casting %%#v -> %s for '%s': %%w\", ival, err)\n", field.Type.String(), field.Flag())
|
|
|
|
|
fprintf(out, "\t\t\t}\n")
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\t}\n")
|
|
|
|
|
fprintf(out, "\n")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func generateUnmarshalerFlagType(out io.Writer, field ConfigField) {
|
|
|
|
|
fprintf(out, "\t\tif ival, ok := cfgmap[\"%s\"]; ok {\n", field.Flag())
|
|
|
|
|
if field.Type.Kind() == reflect.Slice {
|
|
|
|
|
// same as above re: slice types and splitting on comma
|
|
|
|
|
fprintf(out, "\t\tt, err := toStringSlice(ival)\n")
|
|
|
|
|
fprintf(out, "\t\tif err != nil {\n")
|
|
|
|
|
fprintf(out, "\t\t\treturn fmt.Errorf(\"error casting %%#v -> []string for '%s': %%w\", ival, err)\n", field.Flag())
|
|
|
|
|
fprintf(out, "\t\t}\n")
|
|
|
|
|
fprintf(out, "\t\tcfg.%s = %s{}\n", field.Path, strings.TrimPrefix(field.Type.String(), "config."))
|
|
|
|
|
fprintf(out, "\t\tfor _, in := range t {\n")
|
|
|
|
|
fprintf(out, "\t\t\tif err := cfg.%s.Set(in); err != nil {\n", field.Path)
|
|
|
|
|
fprintf(out, "\t\t\t\treturn fmt.Errorf(\"error parsing %%#v for '%s': %%w\", ival, err)\n", field.Flag())
|
|
|
|
|
fprintf(out, "\t\t\t}\n")
|
|
|
|
|
fprintf(out, "\t\t}\n")
|
|
|
|
|
} else {
|
|
|
|
|
fprintf(out, "\t\tt, err := cast.ToStringE(ival)\n")
|
|
|
|
|
fprintf(out, "\t\tif err != nil {\n")
|
|
|
|
|
fprintf(out, "\t\t\treturn fmt.Errorf(\"error casting %%#v -> string for '%s': %%w\", ival, err)\n", field.Flag())
|
|
|
|
|
fprintf(out, "\t\t}\n")
|
|
|
|
|
fprintf(out, "\t\tcfg.%s = %#v\n", field.Path, reflect.New(field.Type).Elem().Interface())
|
|
|
|
|
fprintf(out, "\t\tif err := cfg.%s.Set(t); err != nil {\n", field.Path)
|
|
|
|
|
fprintf(out, "\t\t\treturn fmt.Errorf(\"error parsing %%#v for '%s': %%w\", ival, err)\n", field.Flag())
|
|
|
|
|
fprintf(out, "\t\t}\n")
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\t}\n")
|
|
|
|
|
fprintf(out, "\n")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func generateGetSetters(out io.Writer, fields []ConfigField) {
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
// Get name from struct path, without periods.
|
|
|
|
|
name := strings.ReplaceAll(field.Path, ".", "")
|
2022-12-11 13:03:15 +00:00
|
|
|
|
2023-07-07 16:17:39 +02:00
|
|
|
// Get type without "config." prefix.
|
|
|
|
|
fieldType := strings.ReplaceAll(
|
|
|
|
|
field.Type.String(),
|
|
|
|
|
"config.", "",
|
|
|
|
|
)
|
|
|
|
|
|
2022-12-11 13:03:15 +00:00
|
|
|
// ConfigState structure helper methods
|
2025-05-06 15:51:45 +00:00
|
|
|
fprintf(out, "// Get%s safely fetches the Configuration value for state's '%s' field\n", name, field.Path)
|
|
|
|
|
fprintf(out, "func (st *ConfigState) Get%s() (v %s) {\n", name, fieldType)
|
|
|
|
|
fprintf(out, "\tst.mutex.RLock()\n")
|
|
|
|
|
fprintf(out, "\tv = st.config.%s\n", field.Path)
|
|
|
|
|
fprintf(out, "\tst.mutex.RUnlock()\n")
|
|
|
|
|
fprintf(out, "\treturn\n")
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
fprintf(out, "// Set%s safely sets the Configuration value for state's '%s' field\n", name, field.Path)
|
|
|
|
|
fprintf(out, "func (st *ConfigState) Set%s(v %s) {\n", name, fieldType)
|
|
|
|
|
fprintf(out, "\tst.mutex.Lock()\n")
|
|
|
|
|
fprintf(out, "\tdefer st.mutex.Unlock()\n")
|
|
|
|
|
fprintf(out, "\tst.config.%s = v\n", field.Path)
|
|
|
|
|
fprintf(out, "\tst.reloadToViper()\n")
|
|
|
|
|
fprintf(out, "}\n\n")
|
2022-05-30 13:41:24 +01:00
|
|
|
|
2022-12-11 13:03:15 +00:00
|
|
|
// Global ConfigState helper methods
|
2025-05-06 15:51:45 +00:00
|
|
|
fprintf(out, "// Get%s safely fetches the value for global configuration '%s' field\n", name, field.Path)
|
|
|
|
|
fprintf(out, "func Get%[1]s() %[2]s { return global.Get%[1]s() }\n\n", name, fieldType)
|
|
|
|
|
fprintf(out, "// Set%s safely sets the value for global configuration '%s' field\n", name, field.Path)
|
|
|
|
|
fprintf(out, "func Set%[1]s(v %[2]s) { global.Set%[1]s(v) }\n\n", name, fieldType)
|
|
|
|
|
}
|
2025-05-31 17:30:57 +02:00
|
|
|
|
|
|
|
|
// Separate out the config fields (from a clone!!!) to get only the 'mem-ratio' members.
|
|
|
|
|
memFields := slices.DeleteFunc(slices.Clone(fields), func(field ConfigField) bool {
|
|
|
|
|
return !strings.Contains(field.Path, "MemRatio")
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
fprintf(out, "// GetTotalOfMemRatios safely fetches the combined value for all the state's mem ratio fields\n")
|
|
|
|
|
fprintf(out, "func (st *ConfigState) GetTotalOfMemRatios() (total float64) {\n")
|
|
|
|
|
fprintf(out, "\tst.mutex.RLock()\n")
|
|
|
|
|
for _, field := range memFields {
|
|
|
|
|
fprintf(out, "\ttotal += st.config.%s\n", field.Path)
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\tst.mutex.RUnlock()\n")
|
|
|
|
|
fprintf(out, "\treturn\n")
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
|
|
|
|
|
fprintf(out, "// GetTotalOfMemRatios safely fetches the combined value for all the global state's mem ratio fields\n")
|
|
|
|
|
fprintf(out, "func GetTotalOfMemRatios() (total float64) { return global.GetTotalOfMemRatios() }\n\n")
|
2025-05-06 15:51:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func generateMapFlattener(out io.Writer, fields []ConfigField) {
|
|
|
|
|
fprintf(out, "func flattenConfigMap(cfgmap map[string]any) {\n")
|
|
|
|
|
fprintf(out, "\tnestedKeys := make(map[string]struct{})\n")
|
|
|
|
|
for _, field := range fields {
|
|
|
|
|
keys := field.PossibleKeys()
|
|
|
|
|
if len(keys) <= 1 {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\tfor _, key := range [][]string{\n")
|
|
|
|
|
for _, key := range keys[1:] {
|
|
|
|
|
fprintf(out, "\t\t{\"%s\"},\n", strings.Join(key, "\", \""))
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\t} {\n")
|
|
|
|
|
fprintf(out, "\t\tival, ok := mapGet(cfgmap, key...)\n")
|
|
|
|
|
fprintf(out, "\t\tif ok {\n")
|
|
|
|
|
fprintf(out, "\t\t\tcfgmap[\"%s\"] = ival\n", field.Flag())
|
|
|
|
|
fprintf(out, "\t\t\tnestedKeys[key[0]] = struct{}{}\n")
|
|
|
|
|
fprintf(out, "\t\t\tbreak\n")
|
|
|
|
|
fprintf(out, "\t\t}\n")
|
|
|
|
|
fprintf(out, "\t}\n\n")
|
|
|
|
|
}
|
|
|
|
|
fprintf(out, "\tfor key := range nestedKeys {\n")
|
|
|
|
|
fprintf(out, "\t\tdelete(cfgmap, key)\n")
|
|
|
|
|
fprintf(out, "\t}\n")
|
|
|
|
|
fprintf(out, "}\n\n")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func fprintf(out io.Writer, format string, args ...any) {
|
|
|
|
|
_, err := fmt.Fprintf(out, format, args...)
|
|
|
|
|
must(err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func must(err error) {
|
|
|
|
|
if err != nil {
|
|
|
|
|
panic(err)
|
2022-05-30 13:41:24 +01:00
|
|
|
}
|
|
|
|
|
}
|