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

deepequal check in helper.CreateOrUpdateWork() always return false, because resource.MarshalJSON() will add \n at the end of JSON #5938

Closed
zach593 opened this issue Dec 11, 2024 · 10 comments · Fixed by #5939 · May be fixed by #5940
Assignees
Labels
kind/bug Categorizes issue or PR as related to a bug.

Comments

@zach593
Copy link
Contributor

zach593 commented Dec 11, 2024

What happened:

deepequal check in helper.CreateOrUpdateWork() always return false, because resource.MarshalJSON() will add \n at the end of JSON, but the workloads read from work.Spec.Workload.Manifests never have this \n. So everytime this function is called, an update operation will occur.

Fortunately, the same thing doesn't happen to work-status-controller, rb/crb-status-controller. Although they are both using DeepEqual() on runtime.RawExtension, but the byte slices are obtained from resource interpret webhook.

What you expected to happen:

remove unecessary update operations.

How to reproduce it (as minimally and precisely as possible):

func Test(t *testing.T) {
	mgr, err := ctrl.NewManager(restConf, ctrl.Options{
		Scheme: gclient.NewSchema(),
	})
	if err != nil {
		panic(err)
	}
	go mgr.GetCache().Start(context.TODO())
	time.Sleep(time.Second)

	work := &workv1alpha1.Work{}
	err = mgr.GetClient().Get(context.TODO(), client.ObjectKey{Namespace: "example-ns", Name: "example-name"}, work)
	if err != nil {
		panic(err)
	}
	fmt.Println(work.Spec.Workload.Manifests[0].Raw)

	workload := &unstructured.Unstructured{}
	_ = json.Unmarshal(work.Spec.Workload.Manifests[0].Raw, workload)

	j0, _ := json.Marshal(workload)
	fmt.Println(j0)

	j1, _ := workload.MarshalJSON()
	fmt.Println(j1)
}

image

Anything else we need to know?:

@snowplayfire found this, I'm just providing some possible fixes here. Many thanks to her, I sincerely hope she has a happy holiday.

This is the root cause why binding controller can’t use DeepEqual() to skip the update operation when calling the ensureWork(). But for deepequal to work, we need another PR to copy part of the karmada webhook logic to the update operation, which I will complete as soon as possible.

For fixing this issue, one way is just need one line: replace resource.MarshalJSON() with json.Marshal(resource), it could work. #5939

But deepequal checks should be used in semantic scenarios, not to judge byte streams. Therefore, if we want to use semantic deepequal checking to fix it, we need to directly judge the unmarshaled workload read from work. #5940

Environment:

  • Karmada version: any
  • kubectl-karmada or karmadactl version (the result of kubectl-karmada version or karmadactl version):
  • Others:
@zhzhuang-zju
Copy link
Contributor

thanks @zach593 @snowplayfire
Follow the test file you provided, I have also reproduced this issue. workload.MarshalJSON() compared to json.Marshal does add \n to the end. And the evidence of this can be found in the code:

https://github.com/golang/go/blob/80a2982a801eaedc416d59801ac8fefcf1e4acf5/src/encoding/json/stream.go#L214-L221

@XiShanYongYe-Chang
Copy link
Member

Thank you very much, it was a great find.

I still have a doubt. Before the work is updated, the manifest file ends with 10. However, why does the manifest in the work geted by using the client not end with 10? Where is the 10 digested?

@XiShanYongYe-Chang
Copy link
Member

/assign @zach593

@zach593
Copy link
Contributor Author

zach593 commented Jan 1, 2025

I still have a doubt. Before the work is updated, the manifest file ends with 10. However, why does the manifest in the work geted by using the client not end with 10? Where is the 10 digested?

When the object is passed to the kube-apiserver, the entire object will be unmarshaled as unstructured.Unstructured, the runtime.RawExtension field will be parsed as map[string]any, and that \n will be omitted.

@XiShanYongYe-Chang
Copy link
Member

@zach593 Thank you for your answer.

@XiShanYongYe-Chang
Copy link
Member

Fortunately, the same thing doesn't happen to work-status-controller, rb/crb-status-controller. Although they are both using DeepEqual() on runtime.RawExtension, but the byte slices are obtained from resource interpret webhook.

Hi @zach593, can you help point out the location of the decision logic code for these controllers?

@XiShanYongYe-Chang
Copy link
Member

But for deepequal to work, we need another PR to copy part of the karmada webhook logic to the update operation, which I will complete as soon as possible.

I don't understand this clearly. Can you explain it a little bit more, maybe we can track it with a new issue?

@zach593
Copy link
Contributor Author

zach593 commented Jan 8, 2025

I don't understand this clearly. Can you explain it a little bit more, maybe we can track it with a new issue?

already got one: #6017

@RainbowMango
Copy link
Member

@snowplayfire found this, I'm just providing some possible fixes here. Many thanks to her, I sincerely hope she has a happy holiday.

Thanks to both of you. @zach593 and @snowplayfire.

@zach593
Copy link
Contributor Author

zach593 commented Jan 10, 2025

Hi @zach593, can you help point out the location of the decision logic code for these controllers?

@XiShanYongYe-Chang

In work status controller, the location is

statusRaw, err := c.ResourceInterpreter.ReflectStatus(clusterObj)

If the DefaultInterpreter is used, it will eventually call

func BuildStatusRawExtension(status interface{}) (*runtime.RawExtension, error) {
statusJSON, err := json.Marshal(status)
if err != nil {
klog.Errorf("Failed to marshal status. Error: %v.", statusJSON)
return nil, err
}
return &runtime.RawExtension{
Raw: statusJSON,
}, nil
}

If other interpreters are implemented and enabled, then obviously this will depend on how the interpreter is implemented.


As for rb/crb status controller,

func (c *RBStatusController) syncBindingStatus(ctx context.Context, binding *workv1alpha2.ResourceBinding) error {
err := helper.AggregateResourceBindingWorkStatus(ctx, c.Client, binding, c.EventRecorder)
if err != nil {
klog.Errorf("Failed to aggregate workStatus to resourceBinding(%s/%s), Error: %v",
binding.Namespace, binding.Name, err)
return err
}
err = updateResourceStatus(ctx, c.DynamicClient, c.RESTMapper, c.ResourceInterpreter, c.EventRecorder, binding.Spec.Resource, binding.Status)
if err != nil {
return err
}
return nil
}

The helper.AggregateResourceBindingWorkStatus() just passes the runtime.RawExtension from work to binding and don't know what's inside.

And the updateResourceStatus() part does not marshal unstructured.Unstructured to JSON and is irrelevant.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
kind/bug Categorizes issue or PR as related to a bug.
Projects
Status: No status
4 participants