Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ConfigMaps items are generated with random order #4292

Open
narg95 opened this issue Nov 16, 2021 · 26 comments
Open

ConfigMaps items are generated with random order #4292

narg95 opened this issue Nov 16, 2021 · 26 comments
Labels
good first issue Denotes an issue ready for a new contributor, according to the "help wanted" guidelines. help wanted Denotes an issue that needs help from a contributor. Must meet "help wanted" guidelines. kind/feature Categorizes issue or PR as related to a new feature. triage/accepted Indicates an issue or PR is ready to be actively worked on.

Comments

@narg95
Copy link
Contributor

narg95 commented Nov 16, 2021

Description

kustomize build . produces configmap items in a random order and although it is somehow benign since it produces equivalent YAML files, it would be great if the build outputs are consistent and reproducible.

I tested it with kustomize v4.4.1

@k8s-ci-robot k8s-ci-robot added needs-kind Indicates a PR lacks a `kind/foo` label and requires one. needs-triage Indicates an issue or PR lacks a `triage/foo` label and requires one. labels Nov 16, 2021
@natasha41575
Copy link
Contributor

natasha41575 commented Nov 18, 2021

Would you mind providing a specific example with input, actual output, and expected output? This will help us write a test specifically targeted to the issue you're having.

To your point, however, we store the ConfigMap items in a Go map, which does not guarantee the order. What order are you expecting? We have two options if we want a deterministic order:

  1. Alphabetical, which would be pretty simple to implement - just sort the keys before adding them to the ConfigMap data.
  2. Preserve the input order. This will be a bigger change (though still small), we'd have to change the underlying data structure, and might have some extremely minor consequences on performance - but then again so would option 1.

I imagine users are expecting option 2, which we can and should do IMO.

@natasha41575 natasha41575 added the kind/feature Categorizes issue or PR as related to a new feature. label Nov 19, 2021
@k8s-ci-robot k8s-ci-robot removed the needs-kind Indicates a PR lacks a `kind/foo` label and requires one. label Nov 19, 2021
@natasha41575 natasha41575 added needs-kind Indicates a PR lacks a `kind/foo` label and requires one. triage/accepted Indicates an issue or PR is ready to be actively worked on. labels Nov 19, 2021
@k8s-ci-robot k8s-ci-robot removed the needs-triage Indicates an issue or PR lacks a `triage/foo` label and requires one. label Nov 19, 2021
@natasha41575 natasha41575 added help wanted Denotes an issue that needs help from a contributor. Must meet "help wanted" guidelines. needs-triage Indicates an issue or PR lacks a `triage/foo` label and requires one. good first issue Denotes an issue ready for a new contributor, according to the "help wanted" guidelines. and removed needs-kind Indicates a PR lacks a `kind/foo` label and requires one. needs-triage Indicates an issue or PR lacks a `triage/foo` label and requires one. labels Nov 19, 2021
@Shatakshi0805
Copy link

Hi, I would like to work on this.
I'm currently a beginner so could you please guide me a little so that I can get started to solve this issue

@natasha41575
Copy link
Contributor

It looks like the key-value pairs are being loaded into an unordered map here:

func makeValidatedDataMap(
ldr ifc.KvLoader, name string, sources types.KvPairSources) (map[string]string, error) {
pairs, err := ldr.Load(sources)
if err != nil {
return nil, errors.WrapPrefix(err, "loading KV pairs", 0)
}
knownKeys := make(map[string]string)
for _, p := range pairs {
// legal key: alphanumeric characters, '-', '_' or '.'
if err := ldr.Validator().ErrIfInvalidKey(p.Key); err != nil {
return nil, err
}
if _, ok := knownKeys[p.Key]; ok {
return nil, errors.Errorf(
"configmap %s illegally repeats the key `%s`", name, p.Key)
}
knownKeys[p.Key] = p.Value
}
return knownKeys, nil
}

We can have this function return something that keeps track of the order of key-value pairs, and preserve that order in the output.

@narg95
Copy link
Contributor Author

narg95 commented Dec 1, 2021

Hi @natasha41575 and @natasha41575, thanks for your answers.

Option 1. with alphabetical order will work for me. I am keeping in source control the kustomize build output and the non-deterministic behavior is giving false positives of changes.

To reproduce the issue build multiple times the following kustomize file, and you will get different outputs which are equivalent but with different order.

kustomize.yaml

configMapGenerator:
  - name: test
    literals:
      - EE10D2E63AF9F498=foo
      - FDC41072BFC0BC7F=foo
      - 32E0817AE855A845=foo
      - 0C6B462C403217C4=foo
      - 024233B4DFB484FC=foo
      - 6E7E3021F1821943=foo
      - DFD86A289C3857EF=foo

Outputs of running three times kustomize build . with the same file:

output 1

apiVersion: v1
data:
  024233B4DFB484FC: foo
  0C6B462C403217C4: foo
  6E7E3021F1821943: foo
  32E0817AE855A845: foo
  DFD86A289C3857EF: foo
  EE10D2E63AF9F498: foo
  FDC41072BFC0BC7F: foo
kind: ConfigMap
metadata:
  name: test-7mt6kd57dt

output 2

apiVersion: v1
data:
 6E7E3021F1821943: foo
 024233B4DFB484FC: foo
 0C6B462C403217C4: foo
 32E0817AE855A845: foo
 DFD86A289C3857EF: foo
 EE10D2E63AF9F498: foo
 FDC41072BFC0BC7F: foo
kind: ConfigMap
metadata:
 name: test-7mt6kd57dt

output 3

apiVersion: v1
data:
  6E7E3021F1821943: foo
  32E0817AE855A845: foo
  024233B4DFB484FC: foo
  0C6B462C403217C4: foo
  DFD86A289C3857EF: foo
  EE10D2E63AF9F498: foo
  FDC41072BFC0BC7F: foo
kind: ConfigMap
metadata:
  name: test-7mt6kd57dt

As you can see each output has different order of configmap items.
I tested it with kustomize v4.4.1

@natasha41575
Copy link
Contributor

@Shatakshi0805 If you would like to work on this, please assign the issue to yourself. You can do so by leaving a comment with just "/assign".

@balalnx16
Copy link

Hi @Shatakshi0805, Are you working on this ? If not can I pick this up ?

@balalnx16
Copy link

It looks like the key-value pairs are being loaded into an unordered map here:

func makeValidatedDataMap(
ldr ifc.KvLoader, name string, sources types.KvPairSources) (map[string]string, error) {
pairs, err := ldr.Load(sources)
if err != nil {
return nil, errors.WrapPrefix(err, "loading KV pairs", 0)
}
knownKeys := make(map[string]string)
for _, p := range pairs {
// legal key: alphanumeric characters, '-', '_' or '.'
if err := ldr.Validator().ErrIfInvalidKey(p.Key); err != nil {
return nil, err
}
if _, ok := knownKeys[p.Key]; ok {
return nil, errors.Errorf(
"configmap %s illegally repeats the key `%s`", name, p.Key)
}
knownKeys[p.Key] = p.Value
}
return knownKeys, nil
}

We can have this function return something that keeps track of the order of key-value pairs, and preserve that order in the output.

Hi @natasha41575, go maps can't preserve the order, if we want to retain order we need to change the data-structure, but this data is consumed by a different api at multiple places and looks like it can't be changed, can you provide guidance on how to proceed ?

Please find an example below

if err = rn.LoadMapIntoSecretData(m); err != nil {

@natasha41575
Copy link
Contributor

this data is consumed by a different api at multiple places and looks like it can't be changed

Why can't it be changed? You can also just return a string slice along with the map containing the keys in the desired order.

@balalnx16
Copy link

@natasha41575 yes you are right my bad, looks like the keys are getting sorted here, before being used further, so will the proposed solution work ?

func SortedMapKeys(m map[string]string) []string {
keys := make([]string, len(m))
i := 0
for k := range m {
keys[i] = k
i++
}
sort.Strings(keys)
return keys
}

@natasha41575
Copy link
Contributor

natasha41575 commented Dec 9, 2021

@balalnx16 That will give us the keys in alphabetical order, which would resolve @narg95's issue. However, I would prefer, if instead of having the keys in alphabetical order, if we can preserve the order of keys from the input. I was thinking we can do this by changing the signature of makeValidatedDataMap to something like

func makeValidatedDataMap( 
 	ldr ifc.KvLoader, name string, sources types.KvPairSources) (map[string]string, []string, error) 

where the returned list of strings contains the keys in the original order that they were added to the map. Do you see any reason why this might be more difficult or not possible?

@balalnx16
Copy link

@natasha41575 this is possible, but preserving keys in sorted order is already implemented and in place but is still not preserving the order when generating yaml for the same input, for multiple iterations of the test, that is the reason I'm wondering if this is the right place to do the change. Should we quickly meet to get this sorted ? I'm available on Kubernetes slack as "Balakumaran GR"

@natasha41575
Copy link
Contributor

preserving keys in sorted order is already implemented and in place but is still not preserving the order when generating yaml for the same input, for multiple iterations of the test, that is the reason I'm wondering if this is the right place to do the change

I think I'm not quite following. There is code to sort the map keys but it is just getting loaded into another (unordered) go map so the order can't be preserved. My suggestion is to use an additional data structure to store the keys in order. It may even be worth defining a DataMap type to return instead of a gomap that stores the keys in a sorted order.

If it is easier to just sort the keys into alphabetical order before outputting them that's fine by me, or if there is a better place to make the change that's also fine by me. Any alternative solution that resolves the issue should be fine, if you have questions about any specific implementation feel free to submit a PR and we can review it.

@DaAlbrecht
Copy link

As suggested by @natasha41575 I would like to work on this as my first issue. If I understood the current state of this issue correctly, option 2. Preserve the input order would be preferred?

@natasha41575
Copy link
Contributor

@DaAlbrecht yes, preserving input order would be preferred. But if that is too difficult, alphabetical is OK.

@DaAlbrecht
Copy link

/assign

@OsirisMedici
Copy link

/assign

@mowtschan
Copy link

Hi guys!

I tried that PR but unfortunately the issue still persists.

I created a new PR with the simple solution which will keep default order (sorted alphabetically).

please have a look!

Sascha

@narg95
Copy link
Contributor Author

narg95 commented Sep 10, 2022

Hi all,

first PR got closed by Author without merging, seems as if the issue was not fully fixed there, can somebody have a look at the new PR #4785 from @mowtschan? it seems to fix it.

@mowtschan
Copy link

Dear @KnVerey
Dear @yuwenma

could you a have a quick look to a very small(just 3 lines!) PR #4785

@gxwilkerson33
Copy link

gxwilkerson33 commented Jun 15, 2023

@natasha41575 @KnVerey

I took a run at this issue and found that the order is maintained (alphabetically currently which can be changed to original order easily) until the yaml.JSONToYAML(json) call in resource.go. Which return the yaml with some random order of the resources.

I guess a couple questions:

  • Do we have something in kyaml that solves this issue?
  • If not does it seem that we should add som functionality to kyaml to solve this issue when converting from JSON to YAML.
    Thanks!

@MaurGi
Copy link

MaurGi commented Jun 15, 2023

this is impactful - we cannot rely on calling kustomize edit add configmap idempotently - we call the same installation twice and it fails!

@OsirisMedici OsirisMedici removed their assignment Jun 17, 2023
@gxwilkerson33
Copy link

/assign

@stormqueen1990
Copy link
Member

I did some investigation on this issue and it would require kubernetes-sigs/yaml#76 to be done. I narrowed this issue down to the sorting in go-yaml v2 not being entirely stable. I noticed this happens when keys that can evaluate to integer values are present. I also noticed that this behaviour happens outside of Kustomize as well.

While experimenting, I saw that bumping go-yaml to v3 solves the problem in the sense that the order is stable although not alphabetical, but I didn't do very extensive testing to check if it'd still be the case when new items are added. This sample shows the issue happening, and if sigs.k8s.io/yaml gets swapped with gopkg.in/yaml.v3, it's possible to see that the issue goes away.

@natasha41575
Copy link
Contributor

Thank you for the investigation @stormqueen1990! I will ping Jordan to see if we can get kubernetes-sigs/yaml#76 merged soon.

@chetak123
Copy link
Member

I see the kubernetes-sigs/yaml#76 is merged now,
Is this issue still up for grab ?

@mowtschan
Copy link

I see the kubernetes-sigs/yaml#76 is merged now, Is this issue still up for grab ?

the issue is still exists 😢

I tried to fix it hier #4785

just put this simple unit test into api/resource/resource_test.go
and then multiple times run go test ./api/resource -run TestAsYAMLSorted

func TestAsYAMLSorted(t *testing.T) {
	given := factory.FromMap(
		map[string]interface{}{
			"configMapGenerator": map[string]interface{}{
				"literals": map[string]interface{}{
					"32E0817AE855A845": "foo",
					"024233B4DFB484FC": "foo",
					"EE10D2E63AF9F498": "foo",
					"6E7E3021F1821943": "foo",
					"FDC41072BFC0BC7F": "foo",
					"DFD86A289C3857EF": "foo",
					"0C6B462C403217C4": "foo",
				},
			},
		})
	expected := `configMapGenerator:
  literals:
    024233B4DFB484FC: foo
    0C6B462C403217C4: foo
    32E0817AE855A845: foo
    6E7E3021F1821943: foo
    DFD86A289C3857EF: foo
    EE10D2E63AF9F498: foo
    FDC41072BFC0BC7F: foo
`
	t.Logf("%v", given)

	yaml, err := given.AsYAML()
	if err != nil {
		t.Fatal(err)
	}
	if string(yaml) != expected {
		t.Fatalf("--- expected\n%s\n--- got\n%s\n", expected, string(yaml))
	}
}
``

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
good first issue Denotes an issue ready for a new contributor, according to the "help wanted" guidelines. help wanted Denotes an issue that needs help from a contributor. Must meet "help wanted" guidelines. kind/feature Categorizes issue or PR as related to a new feature. triage/accepted Indicates an issue or PR is ready to be actively worked on.
Projects
Development

Successfully merging a pull request may close this issue.