From 587b83e1711213cabd4991ed4678dfc53b677838 Mon Sep 17 00:00:00 2001 From: Joseph Phillips Date: Wed, 11 Sep 2024 12:12:55 +0200 Subject: [PATCH 1/5] chore: replace error equality checks with errors.Is --- worker/uniter/relation/resolver.go | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/worker/uniter/relation/resolver.go b/worker/uniter/relation/resolver.go index 41d37c59ebf..cf95355ffff 100644 --- a/worker/uniter/relation/resolver.go +++ b/worker/uniter/relation/resolver.go @@ -55,7 +55,7 @@ func (r *relationsResolver) NextOp(localState resolver.LocalState, remoteState r if r.logger.IsTraceEnabled() { r.logger.Tracef("relation resolver next op for new remote relations %# v", pretty.Formatter(remoteState.Relations)) defer func() { - if err == resolver.ErrNoOperation { + if errors.Is(err, resolver.ErrNoOperation) { r.logger.Tracef("no relation operation to run") } }() @@ -99,7 +99,7 @@ func (r *relationsResolver) NextOp(localState resolver.LocalState, remoteState r relState = NewState(relationId) } hInfo, err := r.nextHookForRelation(relState, relationSnapshot, remoteBroken) - if err == resolver.ErrNoOperation { + if errors.Is(err, resolver.ErrNoOperation) { continue } return opFactory.NewRunHook(hInfo) @@ -343,7 +343,7 @@ func (r *createdRelationsResolver) NextOp( if r.logger.IsTraceEnabled() { r.logger.Tracef("create relation resolver next op for new remote relations %# v", pretty.Formatter(remoteState.Relations)) defer func() { - if err == resolver.ErrNoOperation { + if errors.Is(err, resolver.ErrNoOperation) { r.logger.Tracef("no create relation operation to run") } }() @@ -369,7 +369,7 @@ func (r *createdRelationsResolver) NextOp( hook, err := r.nextHookForRelation(relationId) if err != nil { - if err == resolver.ErrNoOperation { + if errors.Is(err, resolver.ErrNoOperation) { continue } From 50223ef4d2a58aabb9174427ad34b56551451d6b Mon Sep 17 00:00:00 2001 From: Joseph Phillips Date: Wed, 11 Sep 2024 12:36:33 +0200 Subject: [PATCH 2/5] refactor: extract processRelationSnapshot method Extracts the body of the loop for each relation in NextOp, so that we can reuse it. --- worker/uniter/relation/resolver.go | 75 ++++++++++++++++++------------ 1 file changed, 45 insertions(+), 30 deletions(-) diff --git a/worker/uniter/relation/resolver.go b/worker/uniter/relation/resolver.go index cf95355ffff..df8db8e9672 100644 --- a/worker/uniter/relation/resolver.go +++ b/worker/uniter/relation/resolver.go @@ -51,7 +51,9 @@ type relationsResolver struct { } // NextOp implements resolver.Resolver. -func (r *relationsResolver) NextOp(localState resolver.LocalState, remoteState remotestate.Snapshot, opFactory operation.Factory) (_ operation.Operation, err error) { +func (r *relationsResolver) NextOp( + localState resolver.LocalState, remoteState remotestate.Snapshot, opFactory operation.Factory, +) (_ operation.Operation, err error) { if r.logger.IsTraceEnabled() { r.logger.Tracef("relation resolver next op for new remote relations %# v", pretty.Formatter(remoteState.Relations)) defer func() { @@ -72,42 +74,55 @@ func (r *relationsResolver) NextOp(localState resolver.LocalState, remoteState r return nil, errors.Trace(err) } - // Check whether we need to fire a hook for any of the relations - for relationId, relationSnapshot := range remoteState.Relations { - if !r.stateTracker.IsKnown(relationId) { - r.logger.Tracef("unknown relation %d resolving next op", relationId) - continue - } else if isImplicit, _ := r.stateTracker.IsImplicit(relationId); isImplicit { - continue - } - - // If either the unit or the relation are Dying, or the - // relation becomes suspended, then the relation should be - // broken. - var remoteBroken bool - if remoteState.Life == life.Dying || relationSnapshot.Life == life.Dying || relationSnapshot.Suspended { - relationSnapshot = remotestate.RelationSnapshot{} - remoteBroken = true - // TODO(axw) if relation is implicit, leave scope & remove. - } - - // Examine local/remote states and figure out if a hook needs - // to be fired for this relation. - relState, err := r.stateTracker.State(relationId) + // Check whether we need to fire a hook for any of the relations. + for relationID, relationSnapshot := range remoteState.Relations { + op, err := r.processRelationSnapshot(relationID, relationSnapshot, remoteState, opFactory) if err != nil { - // - relState = NewState(relationId) - } - hInfo, err := r.nextHookForRelation(relState, relationSnapshot, remoteBroken) - if errors.Is(err, resolver.ErrNoOperation) { - continue + if errors.Is(err, resolver.ErrNoOperation) { + continue + } + return nil, errors.Trace(err) } - return opFactory.NewRunHook(hInfo) + return op, nil } return nil, resolver.ErrNoOperation } +func (r *relationsResolver) processRelationSnapshot( + relationID int, + relationSnapshot remotestate.RelationSnapshot, + remoteState remotestate.Snapshot, + opFactory operation.Factory, +) (operation.Operation, error) { + if !r.stateTracker.IsKnown(relationID) { + r.logger.Infof("unknown relation %d resolving next op", relationID) + return nil, resolver.ErrNoOperation + } else if isImplicit, _ := r.stateTracker.IsImplicit(relationID); isImplicit { + return nil, resolver.ErrNoOperation + } + + // If either the unit or the relation are Dying, or the relation + // becomes suspended, then the relation should be broken. + var remoteBroken bool + if remoteState.Life == life.Dying || relationSnapshot.Life == life.Dying || relationSnapshot.Suspended { + relationSnapshot = remotestate.RelationSnapshot{} + remoteBroken = true + } + + // Examine local/remote states and figure out if a hook needs + // to be fired for this relation. + relState, err := r.stateTracker.State(relationID) + if err != nil { + relState = NewState(relationID) + } + hInfo, err := r.nextHookForRelation(relState, relationSnapshot, remoteBroken) + if err != nil { + return nil, errors.Trace(err) + } + return opFactory.NewRunHook(hInfo) +} + // maybeDestroySubordinates checks whether the remote state indicates that the // unit is dying and ensures that any related subordinates are properly // destroyed. From 0078ddf0c38641148c649b921b8ce2cdaffd8577 Mon Sep 17 00:00:00 2001 From: Joseph Phillips Date: Wed, 11 Sep 2024 13:42:20 +0200 Subject: [PATCH 3/5] fix: spelling for newRelationStateTracker --- worker/uniter/relation/resolver_test.go | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/worker/uniter/relation/resolver_test.go b/worker/uniter/relation/resolver_test.go index 5c01f592cda..1d23f9f2a1c 100644 --- a/worker/uniter/relation/resolver_test.go +++ b/worker/uniter/relation/resolver_test.go @@ -136,7 +136,7 @@ func assertNumCalls(c *gc.C, numCalls *int32, expected int32) { c.Assert(v, gc.Equals, expected) } -func (s *relationResolverSuite) newRelationStateTracer(c *gc.C, apiCaller base.APICaller, unitTag names.UnitTag) relation.RelationStateTracker { +func (s *relationResolverSuite) newRelationStateTracker(c *gc.C, apiCaller base.APICaller, unitTag names.UnitTag) relation.RelationStateTracker { abort := make(chan struct{}) st := uniter.NewState(apiCaller, unitTag) u, err := st.Unit(unitTag) @@ -166,7 +166,7 @@ func (s *relationResolverSuite) setupRelations(c *gc.C) relation.RelationStateTr uniterAPICall("State", unitEntity, unitStateResults, nil), uniterAPICall("RelationsStatus", unitEntity, params.RelationUnitStatusResults{Results: []params.RelationUnitStatusResult{{RelationResults: []params.RelationUnitStatus{}}}}, nil), ) - r := s.newRelationStateTracer(c, apiCaller, unitTag) + r := s.newRelationStateTracker(c, apiCaller, unitTag) assertNumCalls(c, &numCalls, 4) return r } @@ -231,7 +231,7 @@ func (s *relationResolverSuite) assertNewRelationsWithExistingRelations(c *gc.C, ) } apiCaller := mockAPICaller(c, &numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, unitTag) + r := s.newRelationStateTracker(c, apiCaller, unitTag) assertNumCalls(c, &numCalls, int32(len(apiCalls))) info := r.GetInfo() @@ -268,7 +268,7 @@ func (s *relationResolverSuite) TestNextOpNothing(c *gc.C) { uniterAPICall("State", unitEntity, unitStateResults, nil), uniterAPICall("RelationsStatus", unitEntity, params.RelationUnitStatusResults{Results: []params.RelationUnitStatusResult{{RelationResults: []params.RelationUnitStatus{}}}}, nil), ) - r := s.newRelationStateTracer(c, apiCaller, unitTag) + r := s.newRelationStateTracker(c, apiCaller, unitTag) assertNumCalls(c, &numCalls, 4) localState := resolver.LocalState{ @@ -393,7 +393,7 @@ func (s *relationResolverSuite) assertHookRelationJoined(c *gc.C, numCalls *int3 unitTag := names.NewUnitTag("wordpress/0") apiCaller := mockAPICaller(c, numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, unitTag) + r := s.newRelationStateTracker(c, apiCaller, unitTag) assertNumCalls(c, numCalls, 4) localState := resolver.LocalState{ @@ -839,7 +839,7 @@ func (s *relationResolverSuite) TestImplicitRelationNoHooks(c *gc.C) { var numCalls int32 apiCaller := mockAPICaller(c, &numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, unitTag) + r := s.newRelationStateTracker(c, apiCaller, unitTag) localState := resolver.LocalState{ State: operation.State{ @@ -979,7 +979,7 @@ func (s *relationResolverSuite) TestSubSubPrincipalRelationDyingDestroysUnit(c * //apiCalls = append(apiCalls, uniterAPICall("State", nrpeUnitEntity, unitStateResults, nil)) apiCaller := mockAPICaller(c, &numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, nrpeUnitTag) + r := s.newRelationStateTracker(c, apiCaller, nrpeUnitTag) assertNumCalls(c, &numCalls, callsBeforeDestroy) // So now we have a relations object with two relations, one to @@ -1035,7 +1035,7 @@ func (s *relationResolverSuite) TestSubSubOtherRelationDyingNotDestroyed(c *gc.C apiCaller := mockAPICaller(c, &numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, nrpeUnitTag) + r := s.newRelationStateTracker(c, apiCaller, nrpeUnitTag) // TODO: Fix this test... // This test intermittently makes either 16 or 17 @@ -1154,7 +1154,7 @@ func (s *relationResolverSuite) TestPrincipalDyingDestroysSubordinates(c *gc.C) //apiCalls = append(apiCalls, uniterAPICall("State", nrpeUnitEntity, unitStateResults, nil)) apiCaller := mockAPICaller(c, &numCalls, apiCalls...) - r := s.newRelationStateTracer(c, apiCaller, nrpeUnitTag) + r := s.newRelationStateTracker(c, apiCaller, nrpeUnitTag) assertNumCalls(c, &numCalls, callsBeforeDestroy) // So now we have a relation between a principal (wordpress) and a From 9cab7d90f461227f5271e46396ccda27fb21fdd0 Mon Sep 17 00:00:00 2001 From: Joseph Phillips Date: Wed, 11 Sep 2024 14:16:03 +0200 Subject: [PATCH 4/5] fix: process peer relations last It was found that when removing applications, hooks for some relations rely on the application data populated by peer relations. When peer relations are departed/broken first, these hooks fail, blocking teardown. It is reasonable to process existing relations peer-last to prevent this situation. Hooks for relation creation and joining are unaffected. --- worker/uniter/relation/resolver.go | 27 +++++++++++++- worker/uniter/relation/resolver_test.go | 49 +++++++++++++++++-------- 2 files changed, 60 insertions(+), 16 deletions(-) diff --git a/worker/uniter/relation/resolver.go b/worker/uniter/relation/resolver.go index df8db8e9672..6d2021deea6 100644 --- a/worker/uniter/relation/resolver.go +++ b/worker/uniter/relation/resolver.go @@ -74,8 +74,33 @@ func (r *relationsResolver) NextOp( return nil, errors.Trace(err) } - // Check whether we need to fire a hook for any of the relations. + // Collect peer relations and defer their processing until after other + // relations. This is simpler than implementing a sort based on the type. + // Processing them last ensures that upon application removal, the hooks + // for other relations can rely on the peer relations still being present. + var peerRelations []int + + // Check whether we need to fire a hook for any of the non-peer relations. for relationID, relationSnapshot := range remoteState.Relations { + if isPeer, _ := r.stateTracker.IsPeerRelation(relationID); isPeer { + peerRelations = append(peerRelations, relationID) + continue + } + + op, err := r.processRelationSnapshot(relationID, relationSnapshot, remoteState, opFactory) + if err != nil { + if errors.Is(err, resolver.ErrNoOperation) { + continue + } + return nil, errors.Trace(err) + } + return op, nil + } + + // Process the deferred peer relations. + for _, relationID := range peerRelations { + relationSnapshot := remoteState.Relations[relationID] + op, err := r.processRelationSnapshot(relationID, relationSnapshot, remoteState, opFactory) if err != nil { if errors.Is(err, resolver.ErrNoOperation) { diff --git a/worker/uniter/relation/resolver_test.go b/worker/uniter/relation/resolver_test.go index 1d23f9f2a1c..e9fb03847ed 100644 --- a/worker/uniter/relation/resolver_test.go +++ b/worker/uniter/relation/resolver_test.go @@ -1359,7 +1359,8 @@ func (s *mockRelationResolverSuite) TestHookRelationJoined(c *gc.C) { s.expectIsKnown(1) s.expectIsImplicitFalse(1) s.expectStateUnknown(1) - s.expectIsPeerRelationFalse(1) + + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) @@ -1402,7 +1403,8 @@ func (s *mockRelationResolverSuite) TestHookRelationChangedApplication(c *gc.C) s.expectIsKnown(1) s.expectIsImplicitFalse(1) s.expectState(relationState) - s.expectIsPeerRelationFalse(1) + + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) @@ -1441,6 +1443,8 @@ func (s *mockRelationResolverSuite) TestHookRelationChangedSuspended(c *gc.C) { s.expectState(relationState) s.expectLocalUnitAndApplicationLife() + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(err, jc.ErrorIsNil) @@ -1478,6 +1482,8 @@ func (s *mockRelationResolverSuite) TestHookRelationDeparted(c *gc.C) { s.expectState(relationState) s.expectLocalUnitAndApplicationLife() + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(err, jc.ErrorIsNil) @@ -1495,22 +1501,35 @@ func (s *mockRelationResolverSuite) TestHookRelationBroken(c *gc.C) { 1: { Life: life.Dying, }, + 2: { + Life: life.Dying, + }, }, } - relationState := relation.State{ + + defer s.setupMocks(c).Finish() + + s.expectSyncScopes(remoteState) + + relationState1 := relation.State{ RelationId: 1, Members: map[string]int64{}, ApplicationMembers: map[string]int64{}, ChangedPending: "", } - defer s.setupMocks(c).Finish() - s.expectSyncScopes(remoteState) s.expectIsKnown(1) s.expectIsImplicitFalse(1) - s.expectState(relationState) - s.expectIsPeerRelationFalse(1) + s.expectState(relationState1) s.expectStateFound(1) s.expectRemoteApplication(1, "") + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) + + // The `Relations` map in the remote state snapshot (as with all Go maps) + // has random iteration order. This will sometimes be called + // (relation ID 2 first) and sometimes not (ID 1 first). The test here is + // that in all cases, the next operation is for ID 1 (non-peer) - it is + // always enqueued ahead of ID 2, which is a peer relation. + s.mockRelStTracker.EXPECT().IsPeerRelation(2).Return(true, nil).MaxTimes(1) relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) @@ -1543,10 +1562,11 @@ func (s *mockRelationResolverSuite) TestHookRelationBrokenWhenSuspended(c *gc.C) s.expectIsKnown(1) s.expectIsImplicitFalse(1) s.expectState(relationState) - s.expectIsPeerRelationFalse(1) s.expectStateFound(1) s.expectRemoteApplication(1, "") + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(err, jc.ErrorIsNil) @@ -1577,9 +1597,10 @@ func (s *mockRelationResolverSuite) TestHookRelationBrokenOnlyOnce(c *gc.C) { s.expectIsKnown(1) s.expectIsImplicitFalse(1) s.expectState(relationState) - s.expectIsPeerRelationFalse(1) s.expectStateFoundFalse(1) + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) _, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(errors.Cause(err), gc.Equals, resolver.ErrNoOperation) @@ -1606,6 +1627,8 @@ func (s *mockRelationResolverSuite) TestImplicitRelationNoHooks(c *gc.C) { s.expectIsKnown(1) s.expectIsImplicit(1) + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, s.mockSupDestroyer) _, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(errors.Cause(err), gc.Equals, resolver.ErrNoOperation) @@ -1643,13 +1666,14 @@ func (s *mockRelationResolverSuite) TestPrincipalDyingDestroysSubordinates(c *gc s.expectIsKnown(1) s.expectIsImplicitFalse(1) s.expectState(relationState) - s.expectIsPeerRelationFalse(1) s.expectHasContainerScope(1) s.expectStateFound(1) s.expectRemoteApplication(1, "") destroyer := mocks.NewMockSubordinateDestroyer(ctrl) destroyer.EXPECT().DestroyAllSubordinates().Return(nil) + s.mockRelStTracker.EXPECT().IsPeerRelation(1).Return(false, nil).Times(2) + relationsResolver := s.newRelationResolver(s.mockRelStTracker, destroyer) op, err := relationsResolver.NextOp(localState, remoteState, &mockOperations{}) c.Assert(err, jc.ErrorIsNil) @@ -1698,11 +1722,6 @@ func (s *mockRelationResolverSuite) expectState(st relation.State) { exp.State(st.RelationId).Return(&st, nil) } -func (s *mockRelationResolverSuite) expectIsPeerRelationFalse(id int) { - exp := s.mockRelStTracker.EXPECT() - exp.IsPeerRelation(id).Return(false, nil) -} - func (s *mockRelationResolverSuite) expectLocalUnitAndApplicationLife() { exp := s.mockRelStTracker.EXPECT() exp.LocalUnitAndApplicationLife().Return(life.Alive, life.Alive, nil) From 415278a006db20a36a51a2381dd3381d845c803a Mon Sep 17 00:00:00 2001 From: Joseph Phillips Date: Wed, 18 Sep 2024 16:38:57 +0200 Subject: [PATCH 5/5] chore: explicitly check for NotFound We shouldn't rely on the fact that a callee only returns NotFound to pun err != nil. Instead, explicitly handle NotFound and throw out for other error types. --- worker/uniter/relation/resolver.go | 5 +++++ worker/uniter/relation/resolver_test.go | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/worker/uniter/relation/resolver.go b/worker/uniter/relation/resolver.go index 6d2021deea6..2685d804657 100644 --- a/worker/uniter/relation/resolver.go +++ b/worker/uniter/relation/resolver.go @@ -114,6 +114,8 @@ func (r *relationsResolver) NextOp( return nil, resolver.ErrNoOperation } +// processRelationSnapshot reconciles the local and remote states for a +// single relation and determines what hoof (if any) should be fired. func (r *relationsResolver) processRelationSnapshot( relationID int, relationSnapshot remotestate.RelationSnapshot, @@ -139,6 +141,9 @@ func (r *relationsResolver) processRelationSnapshot( // to be fired for this relation. relState, err := r.stateTracker.State(relationID) if err != nil { + if !errors.Is(err, errors.NotFound) { + return nil, errors.Trace(err) + } relState = NewState(relationID) } hInfo, err := r.nextHookForRelation(relState, relationSnapshot, remoteBroken) diff --git a/worker/uniter/relation/resolver_test.go b/worker/uniter/relation/resolver_test.go index e9fb03847ed..6bc4628bbdc 100644 --- a/worker/uniter/relation/resolver_test.go +++ b/worker/uniter/relation/resolver_test.go @@ -1714,7 +1714,7 @@ func (s *mockRelationResolverSuite) expectIsImplicitFalse(id int) { func (s *mockRelationResolverSuite) expectStateUnknown(id int) { exp := s.mockRelStTracker.EXPECT() - exp.State(id).Return(nil, errors.Errorf("unknown relation: %d", id)) + exp.State(id).Return(nil, errors.NotFoundf("relation: %d", id)) } func (s *mockRelationResolverSuite) expectState(st relation.State) {