Skip to content

Commit

Permalink
fix: Set operation state to nil if new operation has been initiated b…
Browse files Browse the repository at this point in the history
…y a direct object update (argoproj#20875)

Fixes argoproj#20875

Some systems like Kargo update the application operation directly to initiate a new sync, bypassing the server. However, they can't update the status operation state to become nil, resulting in discrepancy, see akuity/kargo#2985 and the discussion. Handle this case on the app controller side.

Let's cherry-pick to v2.13.

Signed-off-by: Andrii Korotkov <[email protected]>
  • Loading branch information
andrii-korotkov-verkada committed Nov 24, 2024
1 parent 6d8d32f commit b735204
Show file tree
Hide file tree
Showing 4 changed files with 83 additions and 0 deletions.
26 changes: 26 additions & 0 deletions controller/appcontroller.go
Original file line number Diff line number Diff line change
Expand Up @@ -1927,6 +1927,27 @@ func (ctrl *ApplicationController) persistAppStatus(orig *appv1.Application, new
return patchMs
}

// resetAppStatusOperationState updates the app status operation state to be nil. If no changes were made, it is a no-op
func (ctrl *ApplicationController) resetAppStatusOperationState(orig *appv1.Application) {
logCtx := getAppLog(orig).WithField("method", "resetAppStatusOperationState")
if orig.Status.OperationState == nil {
return
}
newStatus := orig.Status.DeepCopy()
newStatus.OperationState = nil
patch, _, err := createMergePatch(
&appv1.Application{Status: orig.Status},
&appv1.Application{Status: *newStatus})
if err != nil {
logCtx.Errorf("Error constructing app status patch: %v", err)
return
}
_, err = ctrl.PatchAppWithWriteBack(context.Background(), orig.Name, orig.Namespace, types.MergePatchType, patch, metav1.PatchOptions{})
if err != nil {
logCtx.Warnf("Error updating application: %v", err)
}
}

// autoSync will initiate a sync operation for an application configured with automated sync
func (ctrl *ApplicationController) autoSync(app *appv1.Application, syncStatus *appv1.SyncStatus, resources []appv1.ResourceStatus, revisionUpdated bool) (*appv1.ApplicationCondition, time.Duration) {
logCtx := getAppLog(app)
Expand Down Expand Up @@ -2318,6 +2339,11 @@ func (ctrl *ApplicationController) newApplicationInformerAndLister() (cache.Shar
jitter := time.Duration(float64(ctrl.statusRefreshJitter) * rand.Float64())
delay = &jitter
}
// Something updated application operation state bypassing the controller and
// operation state in status still reflects the old sync. Reset it.
if oldApp.Operation == nil && newApp.Operation != nil && newApp.Status.OperationState != nil {
ctrl.resetAppStatusOperationState(newApp)
}
}

ctrl.requestAppRefresh(newApp.QualifiedName(), compareWith, delay)
Expand Down
22 changes: 22 additions & 0 deletions server/application/application_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1972,6 +1972,28 @@ func TestAppJsonPatch(t *testing.T) {
app, err = appServer.Patch(ctx, &application.ApplicationPatchRequest{Name: &testApp.Name, Patch: ptr.To(`[{"op": "remove", "path": "/metadata/annotations/test.annotation"}]`)})
require.NoError(t, err)
assert.NotContains(t, app.Annotations, "test.annotation")

app, err = appServer.Patch(
ctx,
&application.ApplicationPatchRequest{
Name: &testApp.Name,
Patch: ptr.To(`[{
"op": "add",
"path": "/operation",
"value": {
"sync": {
"revisions": ["test"]
},
"initiatedBy": {
"username": "test"
}
}
}]`),
},
)
require.NoError(t, err)
// Can't patch the operation field directly
assert.Nil(t, app.Operation)
}

func TestAppMergePatch(t *testing.T) {
Expand Down
24 changes: 24 additions & 0 deletions test/e2e/app_management_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -3001,3 +3001,27 @@ func TestDeletionConfirmation(t *testing.T) {
When().ConfirmDeletion().
Then().Expect(DoesNotExist())
}

func TestNewOperationTriggeredByUpdatingAppObject(t *testing.T) {
Given(t).
Path(guestbookPath).
When().
CreateApp().
Sync().
Then().
Expect(OperationPhaseIs(OperationSucceeded)).
Expect(SyncStatusIs(SyncStatusCodeSynced)).
Expect(HealthIs(health.HealthStatusHealthy)).
When().
// Can't patch using PatchApp* since they ignore Operation changes
UpdateApp(func(app *Application) {
app.Operation = &Operation{
Sync: &SyncOperation{Revisions: []string{"test"}},
InitiatedBy: OperationInitiator{Username: "test"},
}
}).
Then().
And(func(app *Application) {
assert.Nil(t, app.Status.OperationState)
})
}
11 changes: 11 additions & 0 deletions test/e2e/fixture/app/actions.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package app

import (
"context"
"encoding/json"
"fmt"
"os"
Expand Down Expand Up @@ -296,6 +297,16 @@ func (a *Actions) DeclarativeWithCustomRepo(filename string, repoURL string) *Ac
return a
}

func (a *Actions) UpdateApp(update func(app *Application)) *Actions {
a.context.t.Helper()
app, err := fixture.AppClientset.ArgoprojV1alpha1().Applications(a.context.AppNamespace()).Get(context.Background(), a.context.AppName(), v1.GetOptions{})
errors.CheckError(err)
update(app)
_, err = fixture.AppClientset.ArgoprojV1alpha1().Applications(a.context.AppNamespace()).Update(context.Background(), app, v1.UpdateOptions{})
errors.CheckError(err)
return a
}

func (a *Actions) PatchApp(patch string) *Actions {
a.context.t.Helper()
a.runCli("app", "patch", a.context.AppQualifiedName(), "--patch", patch)
Expand Down

0 comments on commit b735204

Please sign in to comment.