Skip to content

Commit

Permalink
fix: don't check the "done" state when processing a Cancelled event (#…
Browse files Browse the repository at this point in the history
…121)

The RBACTimelock contract does not mark an operation as `done` when it's
cancelled. So the `isDone(operationID)` check we had in the "Cancelled"
event handler actually prevented the timelock worker service from
properly removing the operation from the scheduler.

This PR simply removes the `isDone()` check and adds an integration test
to verify that we're properly de-scheduling the operation.
  • Loading branch information
gustavogama-cll authored Dec 27, 2024
1 parent 25d40b5 commit 3676476
Show file tree
Hide file tree
Showing 4 changed files with 74 additions and 10 deletions.
2 changes: 1 addition & 1 deletion pkg/timelock/scheduler.go
Original file line number Diff line number Diff line change
Expand Up @@ -132,7 +132,7 @@ func (tw *scheduler) addToScheduler(op *contracts.RBACTimelockCallScheduled) {

// delFromScheduler deletes an operation safely from the store.
func (tw *scheduler) delFromScheduler(op operationKey) {
tw.logger.Debugf("de-scheduling operation: %v", op)
tw.logger.Debugf("de-scheduling operation: %x", op)
tw.del <- op
}

Expand Down
8 changes: 3 additions & 5 deletions pkg/timelock/timelock.go
Original file line number Diff line number Diff line change
Expand Up @@ -488,11 +488,9 @@ func (tw *Worker) handleLog(ctx context.Context, log types.Log) error {
return err
}

if isDone(ctx, tw.contract, cs.Id) {
tw.logger.With(fieldTXHash, fmt.Sprintf("%x", cs.Raw.TxHash[:])).
With(fieldBlockNumber, cs.Raw.BlockNumber).Infof("%s received, cancelling operation", eventCancelled)
tw.scheduler.delFromScheduler(cs.Id)
}
tw.logger.With(fieldTXHash, fmt.Sprintf("%x", cs.Raw.TxHash[:])).
With(fieldBlockNumber, cs.Raw.BlockNumber).Infof("%s received, cancelling operation", eventCancelled)
tw.scheduler.delFromScheduler(cs.Id)
default:
tw.logger.With("event", event.Name).Info("discarding event")
}
Expand Down
21 changes: 21 additions & 0 deletions tests/integration/onchain_ops.go
Original file line number Diff line number Diff line change
Expand Up @@ -141,6 +141,27 @@ func ExecuteBatch(
return transaction, receipt
}

func CancelBatch(
t *testing.T,
ctx context.Context, transactor *bind.TransactOpts, backend Backend,
timelockContract *contracts.RBACTimelock, operationID [32]byte,
) (
*types.Transaction, *types.Receipt,
) {
t.Helper()

transaction, err := timelockContract.Cancel(transactor, operationID)
require.NoError(t, err)

backend.Commit()
receipt, err := bind.WaitMined(ctx, backend, transaction)
require.NoError(t, err)
require.Equal(t, types.ReceiptStatusSuccessful, receipt.Status)
t.Logf("cancel batch transaction: %v", transaction.Hash())

return transaction, receipt
}

func SendTransaction(
t *testing.T, ctx context.Context, transactor *bind.TransactOpts, backend Backend,
account TestAccount, chainID *big.Int, value *big.Int, to common.Address, data []byte,
Expand Down
53 changes: 49 additions & 4 deletions tests/integration/timelock_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -110,8 +110,8 @@ func (s *integrationTestSuite) TestTimelockWorkerDryRun() {
assert: func(t *testing.T, logs *observer.ObservedLogs) {
t.Helper()
s.Require().EventuallyWithT(func(t *assert.CollectT) {
assert.Equal(t, logs.FilterMessage("CallScheduled received").Len(), 1)
assert.Equal(t, logs.FilterMessage("nop.addToScheduler").Len(), 1)
assertLogMessage(t, logs, "CallScheduled received")
assertLogMessage(t, logs, "nop.addToScheduler")
}, 2*time.Second, 100*time.Millisecond)
},
},
Expand All @@ -121,8 +121,8 @@ func (s *integrationTestSuite) TestTimelockWorkerDryRun() {
assert: func(t *testing.T, logs *observer.ObservedLogs) {
t.Helper()
s.Require().EventuallyWithT(func(t *assert.CollectT) {
assert.Equal(t, logs.FilterMessage("scheduling operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237").Len(), 1)
assert.Equal(t, logs.FilterMessage("scheduled operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237").Len(), 1)
assertLogMessage(t, logs, "scheduling operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237")
assertLogMessage(t, logs, "scheduled operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237")
}, 2*time.Second, 100*time.Millisecond)
},
},
Expand All @@ -148,6 +148,47 @@ func (s *integrationTestSuite) TestTimelockWorkerDryRun() {
}
}

func (s *integrationTestSuite) TestTimelockWorkerCancelledEvent() {
// --- arrange ---
ctx, cancel := context.WithCancel(s.Ctx)
defer cancel()

account := NewTestAccount(s.T())
_, err := s.GethContainer.CreateAccount(ctx, account.HexAddress, account.HexPrivateKey, 1)
s.Require().NoError(err)
s.Logf("new account created: %v", account)

gethURL := s.GethContainer.HTTPConnStr(s.T(), ctx)
backend := NewRPCBackend(s.T(), ctx, gethURL)
transactor := s.KeyedTransactor(account.PrivateKey, nil)
logger, logs := timelockTests.NewTestLogger()

timelockAddress, _, _, timelockContract := DeployTimelock(s.T(), ctx, transactor, backend,
account.Address, big.NewInt(1))
callProxyAddress, _, _, _ := DeployCallProxy(s.T(), ctx, transactor, backend, timelockAddress)

go runTimelockWorker(s.T(), ctx, gethURL, timelockAddress.String(), callProxyAddress.String(),
account.HexPrivateKey, big.NewInt(0), int64(1), int64(1), false, logger)

calls := []contracts.RBACTimelockCall{{
Target: common.HexToAddress("0x000000000000000000000000000000000000000"),
Value: big.NewInt(1),
Data: hexutil.MustDecode("0x0123456789abcdef"),
}}
predecessor := common.Hash{}
salt := common.Hash{}
operationID := common.HexToHash("371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237")

// --- act ---
ScheduleBatch(s.T(), ctx, transactor, backend, timelockContract, calls, predecessor, salt, big.NewInt(1))
CancelBatch(s.T(), ctx, transactor, backend, timelockContract, operationID)

// --- assert ---
assertLogMessage(s.T(), logs, "Cancelled received, cancelling operation")
assertLogMessage(s.T(), logs, "de-scheduling operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237")
assertLogMessage(s.T(), logs, "de-scheduled operation: 371141ec10c0cc52996bed94240931136172d0b46bdc4bceaea1ef76675c1237")
}

// ----- helpers -----

func runTimelockWorker(
Expand All @@ -164,3 +205,7 @@ func runTimelockWorker(
err = timelockWorker.Listen(ctx)
require.NoError(t, err)
}

func assertLogMessage(t assert.TestingT, logs *observer.ObservedLogs, message string) {
assert.Equal(t, logs.FilterMessage(message).Len(), 1)
}

0 comments on commit 3676476

Please sign in to comment.