Skip to content

Commit

Permalink
Add new custom calldata DA oracle
Browse files Browse the repository at this point in the history
  • Loading branch information
ogtownsend committed Oct 15, 2024
1 parent f0c928b commit f13e0d7
Show file tree
Hide file tree
Showing 8 changed files with 251 additions and 8 deletions.
5 changes: 5 additions & 0 deletions .changeset/metal-forks-arrive.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"chainlink": patch
---

Adds new custom calldata DA oracle #wip
6 changes: 4 additions & 2 deletions core/chains/evm/config/chain_scoped_gas_estimator.go
Original file line number Diff line number Diff line change
Expand Up @@ -139,8 +139,10 @@ func (d *daOracleConfig) OracleAddress() *types.EIP55Address {

// CustomGasPriceCalldata returns the calldata for a custom gas price API.
func (d *daOracleConfig) CustomGasPriceCalldata() string {
// TODO: CCIP-3710 update once custom calldata oracle is added
return ""
if d.c.OracleType == oracletype.CustomCalldata {
return d.c.CustomGasPriceCalldata
}
panic("CustomGasPriceCalldata is only supported on the CustomCalldata oracle type")
}

type limitJobTypeConfig struct {
Expand Down
9 changes: 5 additions & 4 deletions core/chains/evm/config/oracletype/oracletype.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,15 @@ package oracletype
type OracleType string

const (
OPStack = OracleType("opstack")
Arbitrum = OracleType("arbitrum")
ZKSync = OracleType("zksync")
OPStack = OracleType("opstack")
Arbitrum = OracleType("arbitrum")
ZKSync = OracleType("zksync")
CustomCalldata = OracleType("custom_calldata")
)

func (o OracleType) IsValid() bool {
switch o {
case "", OPStack, Arbitrum, ZKSync:
case "", OPStack, Arbitrum, ZKSync, CustomCalldata:
return true
}
return false
Expand Down
164 changes: 164 additions & 0 deletions core/chains/evm/gas/rollups/custom_calldata_da_oracle.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
package rollups

import (
"context"
"encoding/hex"
"fmt"
"math/big"
"strings"
"sync"
"time"

"github.com/ethereum/go-ethereum"

"github.com/smartcontractkit/chainlink-common/pkg/logger"
"github.com/smartcontractkit/chainlink-common/pkg/services"

"github.com/smartcontractkit/chainlink/v2/core/chains/evm/assets"
evmclient "github.com/smartcontractkit/chainlink/v2/core/chains/evm/client"
evmconfig "github.com/smartcontractkit/chainlink/v2/core/chains/evm/config"
"github.com/smartcontractkit/chainlink/v2/core/chains/evm/config/oracletype"
)

type customCalldataDAOracle struct {
services.StateMachine
client l1OracleClient
pollPeriod time.Duration
logger logger.SugaredLogger

daOracleConfig evmconfig.DAOracle
daGasPriceMu sync.RWMutex
daGasPrice priceEntry

chInitialized chan struct{}
chStop services.StopChan
chDone chan struct{}
}

// NewCustomCalldataDAOracle creates a new custom calldata DA oracle. The CustomCalldataDAOracle fetches gas price from
// whatever function is specified in the DAOracle's CustomGasPriceCalldata field. This allows for more flexibility when
// chains have custom DA gas calculation methods.
func NewCustomCalldataDAOracle(lggr logger.Logger, ethClient l1OracleClient, daOracleConfig evmconfig.DAOracle) (*customCalldataDAOracle, error) {
if daOracleConfig.OracleType() != oracletype.CustomCalldata {
return nil, fmt.Errorf("expected CustomCalldata oracle type, got %s", daOracleConfig.OracleType())
}
return &customCalldataDAOracle{
client: ethClient,
pollPeriod: PollPeriod,
logger: logger.Sugared(logger.Named(lggr, fmt.Sprintf("CustomCalldataDAOracle(%s)", daOracleConfig.OracleType()))),

daOracleConfig: daOracleConfig,

chInitialized: make(chan struct{}),
chStop: make(chan struct{}),
chDone: make(chan struct{}),
}, nil
}

func (o *customCalldataDAOracle) Name() string {
return o.logger.Name()
}

func (o *customCalldataDAOracle) Start(_ context.Context) error {
return o.StartOnce(o.Name(), func() error {
go o.run()
<-o.chInitialized
return nil
})
}

func (o *customCalldataDAOracle) Close() error {
return o.StopOnce(o.Name(), func() error {
close(o.chStop)
<-o.chDone
return nil
})
}

func (o *customCalldataDAOracle) HealthReport() map[string]error {
return map[string]error{o.Name(): o.Healthy()}
}

func (o *customCalldataDAOracle) run() {
defer close(o.chDone)

o.refresh()
close(o.chInitialized)

t := services.TickerConfig{
Initial: o.pollPeriod,
JitterPct: services.DefaultJitter,
}.NewTicker(o.pollPeriod)
defer t.Stop()

for {
select {
case <-o.chStop:
return
case <-t.C:
o.refresh()
}
}
}

func (o *customCalldataDAOracle) refresh() {
err := o.refreshWithError()
if err != nil {
o.logger.Criticalw("Failed to refresh gas price", "err", err)
o.SvcErrBuffer.Append(err)
}
}

func (o *customCalldataDAOracle) refreshWithError() error {
ctx, cancel := o.chStop.CtxCancel(evmclient.ContextWithDefaultTimeout())
defer cancel()

price, err := o.getCustomCalldataGasPrice(ctx)
if err != nil {
return err
}

o.daGasPriceMu.Lock()
defer o.daGasPriceMu.Unlock()
o.daGasPrice = priceEntry{price: assets.NewWei(price), timestamp: time.Now()}
return nil
}

func (o *customCalldataDAOracle) GasPrice(_ context.Context) (daGasPrice *assets.Wei, err error) {
var timestamp time.Time
ok := o.IfStarted(func() {
o.daGasPriceMu.RLock()
daGasPrice = o.daGasPrice.price
timestamp = o.daGasPrice.timestamp
o.daGasPriceMu.RUnlock()
})
if !ok {
return daGasPrice, fmt.Errorf("DAGasOracle is not started; cannot estimate gas")
}
if daGasPrice == nil {
return daGasPrice, fmt.Errorf("failed to get DA gas price; gas price not set")
}
// Validate the price has been updated within the pollPeriod * 2
// Allowing double the poll period before declaring the price stale to give ample time for the refresh to process
if time.Since(timestamp) > o.pollPeriod*2 {
return daGasPrice, fmt.Errorf("gas price is stale")
}
return
}

func (o *customCalldataDAOracle) getCustomCalldataGasPrice(ctx context.Context) (*big.Int, error) {
daOracleAddress := o.daOracleConfig.OracleAddress().Address()
calldata := strings.TrimPrefix(o.daOracleConfig.CustomGasPriceCalldata(), "0x")
calldataBytes, err := hex.DecodeString(calldata)
if err != nil {
return nil, fmt.Errorf("failed to decode custom fee method calldata: %w", err)
}
b, err := o.client.CallContract(ctx, ethereum.CallMsg{
To: &daOracleAddress,
Data: calldataBytes,
}, nil)
if err != nil {
return nil, fmt.Errorf("custom fee method call failed: %w", err)
}
return new(big.Int).SetBytes(b), nil
}
69 changes: 69 additions & 0 deletions core/chains/evm/gas/rollups/custom_calldata_da_oracle_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
package rollups

import (
"math/big"
"testing"

"github.com/ethereum/go-ethereum"
"github.com/ethereum/go-ethereum/common/hexutil"

"github.com/smartcontractkit/chainlink-common/pkg/utils/tests"

"github.com/smartcontractkit/chainlink/v2/core/chains/evm/config/oracletype"
"github.com/smartcontractkit/chainlink/v2/core/chains/evm/gas/rollups/mocks"
"github.com/smartcontractkit/chainlink/v2/core/chains/evm/utils"

"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"

"github.com/smartcontractkit/chainlink-common/pkg/logger"
)

func TestCustomCalldataDAOracle_NewCustomCalldata(t *testing.T) {
oracleAddress := utils.RandomAddress().String()
t.Parallel()

t.Run("panics if oracle type is not CustomCalldata", func(t *testing.T) {
ethClient := mocks.NewL1OracleClient(t)
daOracleConfig := CreateTestDAOracle(t, oracletype.OPStack, oracleAddress, "")
_, err := NewCustomCalldataDAOracle(logger.Test(t), ethClient, daOracleConfig)
require.Error(t, err)
})
}

func TestCustomCalldataDAOracle_getCustomCalldataGasPrice(t *testing.T) {
oracleAddress := utils.RandomAddress().String()
t.Parallel()

t.Run("correctly fetches gas price if chain has custom calldata", func(t *testing.T) {
ethClient := mocks.NewL1OracleClient(t)
expectedPriceHex := "0x0000000000000000000000000000000000000000000000000000000000000032"

daOracleConfig := CreateTestDAOracle(t, oracletype.CustomCalldata, oracleAddress, "0x0000000000000000000000000000000000001234")
oracle, err := NewCustomCalldataDAOracle(logger.Test(t), ethClient, daOracleConfig)
require.NoError(t, err)

ethClient.On("CallContract", mock.Anything, mock.IsType(ethereum.CallMsg{}), mock.IsType(&big.Int{})).Run(func(args mock.Arguments) {
callMsg := args.Get(1).(ethereum.CallMsg)
blockNumber := args.Get(2).(*big.Int)
require.NotNil(t, callMsg.To)
require.Equal(t, oracleAddress, callMsg.To.String())
require.Nil(t, blockNumber)
}).Return(hexutil.MustDecode(expectedPriceHex), nil).Once()

price, err := oracle.getCustomCalldataGasPrice(tests.Context(t))
require.NoError(t, err)
require.Equal(t, big.NewInt(50), price)
})

t.Run("throws error if custom calldata fails to decode", func(t *testing.T) {
ethClient := mocks.NewL1OracleClient(t)

daOracleConfig := CreateTestDAOracle(t, oracletype.CustomCalldata, oracleAddress, "0xblahblahblah")
oracle, err := NewCustomCalldataDAOracle(logger.Test(t), ethClient, daOracleConfig)
require.NoError(t, err)

_, err = oracle.getCustomCalldataGasPrice(tests.Context(t))
require.Error(t, err)
})
}
2 changes: 2 additions & 0 deletions core/chains/evm/gas/rollups/l1_oracle.go
Original file line number Diff line number Diff line change
Expand Up @@ -63,6 +63,8 @@ func NewL1GasOracle(lggr logger.Logger, ethClient l1OracleClient, chainType chai
l1Oracle, err = NewArbitrumL1GasOracle(lggr, ethClient)
case oracletype.ZKSync:
l1Oracle = NewZkSyncL1GasOracle(lggr, ethClient)
case oracletype.CustomCalldata:
l1Oracle, err = NewCustomCalldataDAOracle(lggr, ethClient, daOracle)
default:
err = fmt.Errorf("unsupported DA oracle type %s. Going forward all chain configs should specify an oracle type", daOracle.OracleType())
}
Expand Down
2 changes: 1 addition & 1 deletion core/config/docs/chains-evm.toml
Original file line number Diff line number Diff line change
Expand Up @@ -264,7 +264,7 @@ TipCapDefault = '1 wei' # Default
TipCapMin = '1 wei' # Default

[EVM.GasEstimator.DAOracle]
# OracleType refers to the oracle family this config belongs to. Currently the available oracle types are: 'opstack', 'arbitrum', and 'zksync'.
# OracleType refers to the oracle family this config belongs to. Currently the available oracle types are: 'opstack', 'arbitrum', 'zksync', and 'custom_calldata'.
OracleType = 'opstack' # Example
# OracleAddress is the address of the oracle contract.
OracleAddress = '0x420000000000000000000000000000000000000F' # Example
Expand Down
2 changes: 1 addition & 1 deletion docs/CONFIG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9321,7 +9321,7 @@ CustomGasPriceCalldata = '' # Default
```toml
OracleType = 'opstack' # Example
```
OracleType refers to the oracle family this config belongs to. Currently the available oracle types are: 'opstack', 'arbitrum', and 'zksync'.
OracleType refers to the oracle family this config belongs to. Currently the available oracle types are: 'opstack', 'arbitrum', 'zksync', and 'custom_calldata'.

### OracleAddress
```toml
Expand Down

0 comments on commit f13e0d7

Please sign in to comment.