Skip to content

Commit 0310201

Browse files
committedNov 5, 2023
Add fix for showconfig command
This commit adds a fix where we create a copy of the top-level config map, instead of using it directly. Previously this situation caused an infinite loop in the map structure, so creating a copy prevents that from happening.
1 parent 77064ad commit 0310201

File tree

3 files changed

+46
-9
lines changed

3 files changed

+46
-9
lines changed
 

‎cmd/showconfig.go

+6-4
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,11 @@ func showConfig(
3737
if err != nil {
3838
return stackerr.NewStackErrf(err, "failed to unmarshal config")
3939
}
40+
log, err := logging.GetLogger(config.LogLevel)
41+
if err != nil {
42+
return fmt.Errorf("getting logger: %w", err)
43+
}
44+
ctx = log.WithContext(ctx)
4045
if err := config.Initialize(ctx); err != nil {
4146
return err
4247
}
@@ -48,10 +53,7 @@ func showConfig(
4853
if err != nil {
4954
return stackerr.NewStackErrf(err, "failed to marshal yaml")
5055
}
51-
log, err := logging.GetLogger(config.LogLevel)
52-
if err != nil {
53-
panic(err)
54-
}
56+
5557
log.Info().Msgf("Using config: %s", config.Config)
5658

5759
fmt.Fprintf(outputter, "%s", string(out))

‎pkg/config/config.go

+35-4
Original file line numberDiff line numberDiff line change
@@ -210,7 +210,11 @@ func (c *Config) getPackageConfigMap(ctx context.Context, packageName string) (m
210210
return configAsMap, nil
211211
}
212212

213-
return map[string]any{}, nil
213+
// Package is something other than map, so set its value to an
214+
// empty map.
215+
emptyMap := map[string]any{}
216+
packageSection[packageName] = emptyMap
217+
return emptyMap, nil
214218

215219
}
216220
func (c *Config) GetPackageConfig(ctx context.Context, packageName string) (*Config, error) {
@@ -659,6 +663,17 @@ func contains[T comparable](slice []T, elem T) bool {
659663
return false
660664
}
661665

666+
func deepCopyConfigMap(src map[string]any) map[string]any {
667+
new := map[string]any{}
668+
for key, val := range src {
669+
if contains([]string{"packages", "config", "interfaces"}, key) {
670+
continue
671+
}
672+
new[key] = val
673+
}
674+
return new
675+
}
676+
662677
// mergeInConfig takes care of merging inheritable configuration
663678
// in the config map. For example, it merges default config, then
664679
// package-level config, then interface-level config.
@@ -684,15 +699,28 @@ func (c *Config) mergeInConfig(ctx context.Context) error {
684699
pkgLog.Err(err).Msg("failed to get package config")
685700
return fmt.Errorf("failed to get package config: %w", err)
686701
}
702+
pkgLog.Trace().Msg("got package config map")
703+
687704
configSectionUntyped, configExists := packageConfig["config"]
688705
if !configExists {
689-
packageConfig["config"] = defaultCfg
690-
continue
706+
pkgLog.Trace().Msg("config section doesn't exist, setting it to a deepcopy of the top-level config")
707+
packageConfig["config"] = deepCopyConfigMap(defaultCfg)
691708
}
709+
710+
pkgLog.Trace().Msg("got config section for package")
692711
// Sometimes the config section may be provided, but it's nil.
693712
// We need to account for this fact.
694713
if configSectionUntyped == nil {
695-
configSectionUntyped = map[string]any{}
714+
pkgLog.Trace().Msg("config section is nil, converting to empty map")
715+
emptyMap := map[string]any{}
716+
717+
// We need to add this to the "global" config mapping so the change
718+
// gets persisted, and also into configSectionUntyped for the logic
719+
// further down.
720+
packageConfig["config"] = emptyMap
721+
configSectionUntyped = emptyMap
722+
} else {
723+
pkgLog.Trace().Msg("config section is not nil")
696724
}
697725

698726
configSectionTyped := configSectionUntyped.(map[string]any)
@@ -701,8 +729,11 @@ func (c *Config) mergeInConfig(ctx context.Context) error {
701729
if contains([]string{"packages", "config"}, key) {
702730
continue
703731
}
732+
keyValLog := pkgLog.With().Str("key", key).Str("value", fmt.Sprintf("%v", value)).Logger()
733+
704734
_, keyExists := configSectionTyped[key]
705735
if !keyExists {
736+
keyValLog.Trace().Msg("setting key to value")
706737
configSectionTyped[key] = value
707738
}
708739
}

‎pkg/config/config_test.go

+5-1
Original file line numberDiff line numberDiff line change
@@ -1204,6 +1204,7 @@ with-expecter: false
12041204
}
12051205
for _, tt := range tests {
12061206
t.Run(tt.name, func(t *testing.T) {
1207+
ctx := context.Background()
12071208
tmpdir := pathlib.NewPath(t.TempDir())
12081209
cfg := tmpdir.Join("config.yaml")
12091210
require.NoError(t, cfg.WriteFile([]byte(tt.cfgYaml)))
@@ -1218,7 +1219,10 @@ with-expecter: false
12181219
t.Errorf("Config.Initialize() error = %v, wantErr %v", err, tt.wantErr)
12191220
}
12201221

1221-
cfgAsStr, err := yaml.Marshal(c._cfgAsMap)
1222+
cfgAsMap, err := c.CfgAsMap(ctx)
1223+
require.NoError(t, err)
1224+
1225+
cfgAsStr, err := yaml.Marshal(cfgAsMap)
12221226
require.NoError(t, err)
12231227

12241228
if tt.wantCfgMap != "" && !reflect.DeepEqual(string(cfgAsStr), tt.wantCfgMap) {

0 commit comments

Comments
 (0)
Please sign in to comment.