diff --git a/common/txmgr/types/tx.go b/common/txmgr/types/tx.go index b65f7edf6e5..f04047a36c1 100644 --- a/common/txmgr/types/tx.go +++ b/common/txmgr/types/tx.go @@ -159,6 +159,10 @@ type TxMeta[ADDR types.Hashable, TX_HASH types.Hashable] struct { MessageIDs []string `json:"MessageIDs,omitempty"` // SeqNumbers is used by CCIP for tx to committed sequence numbers correlation in logs SeqNumbers []uint64 `json:"SeqNumbers,omitempty"` + + // Dual Broadcast + DualBroadcast *bool `json:"DualBroadcast,omitempty"` + DualBroadcastParams *string `json:"DualBroadcastParams,omitempty"` } type TxAttempt[ diff --git a/core/services/job/job_orm_test.go b/core/services/job/job_orm_test.go index bc6ec0d6ea2..1ddb18a3d89 100644 --- a/core/services/job/job_orm_test.go +++ b/core/services/job/job_orm_test.go @@ -2014,7 +2014,7 @@ func mustInsertPipelineRun(t *testing.T, orm pipeline.ORM, j job.Job) pipeline.R return run } -func TestORM_CreateJob_OCR2_With_AdaptiveSend(t *testing.T) { +func TestORM_CreateJob_OCR2_With_DualTransmission(t *testing.T) { ctx := testutils.Context(t) customChainID := big.New(testutils.NewRandomEVMChainID()) @@ -2030,27 +2030,66 @@ func TestORM_CreateJob_OCR2_With_AdaptiveSend(t *testing.T) { db := pgtest.NewSqlxDB(t) keyStore := cltest.NewKeyStore(t, db) require.NoError(t, keyStore.OCR2().Add(ctx, cltest.DefaultOCR2Key)) - _, transmitterID := cltest.MustInsertRandomKey(t, keyStore.Eth()) + baseJobSpec := fmt.Sprintf(testspecs.OCR2EVMDualTransmissionSpecMinimalTemplate, transmitterID.String()) + lggr := logger.TestLogger(t) pipelineORM := pipeline.NewORM(db, lggr, config.JobPipeline().MaxSuccessfulRuns()) bridgesORM := bridges.NewORM(db) jobORM := NewTestORM(t, db, pipelineORM, bridgesORM, keyStore) - adaptiveSendKey := cltest.MustGenerateRandomKey(t) + // Enabled but no config set + enabledDualTransmissionSpec := ` + enableDualTransmission=true` - jb, err := ocr2validate.ValidatedOracleSpecToml(testutils.Context(t), config.OCR2(), config.Insecure(), testspecs.GetOCR2EVMWithAdaptiveSendSpecMinimal(cltest.DefaultOCR2Key.ID(), transmitterID.String(), adaptiveSendKey.EIP55Address.String()), nil) + jb, err := ocr2validate.ValidatedOracleSpecToml(testutils.Context(t), config.OCR2(), config.Insecure(), baseJobSpec+enabledDualTransmissionSpec, nil) + require.NoError(t, err) + require.ErrorContains(t, jobORM.CreateJob(ctx, &jb), "dual transmission is enabled but no dual transmission config present") + + // ContractAddress not set + emptyContractAddress := ` + enableDualTransmission=true + [relayConfig.dualTransmission] + contractAddress="" + ` + jb, err = ocr2validate.ValidatedOracleSpecToml(testutils.Context(t), config.OCR2(), config.Insecure(), baseJobSpec+emptyContractAddress, nil) + require.NoError(t, err) + require.ErrorContains(t, jobORM.CreateJob(ctx, &jb), "invalid contract address in dual transmission config") + + // Transmitter address not set + emptyTransmitterAddress := ` + enableDualTransmission=true + [relayConfig.dualTransmission] + contractAddress = '0x613a38AC1659769640aaE063C651F48E0250454C' + transmitterAddress = '' + ` + jb, err = ocr2validate.ValidatedOracleSpecToml(testutils.Context(t), config.OCR2(), config.Insecure(), baseJobSpec+emptyTransmitterAddress, nil) + require.NoError(t, err) + require.ErrorContains(t, jobORM.CreateJob(ctx, &jb), "invalid transmitter address in dual transmission config") + + dtTransmitterAddress := cltest.MustGenerateRandomKey(t) + completeDualTransmissionSpec := fmt.Sprintf(` + enableDualTransmission=true + [relayConfig.dualTransmission] + contractAddress = '0x613a38AC1659769640aaE063C651F48E0250454C' + transmitterAddress = '%s' + [relayConfig.dualTransmission.meta] + key1 = 'val1' + key2 = ['val2','val3'] + `, + dtTransmitterAddress.Address.String()) + + jb, err = ocr2validate.ValidatedOracleSpecToml(testutils.Context(t), config.OCR2(), config.Insecure(), baseJobSpec+completeDualTransmissionSpec, nil) require.NoError(t, err) - require.Equal(t, "arbitrary-value", jb.AdaptiveSendSpec.Metadata["arbitraryParam"]) - t.Run("unknown transmitter address", func(t *testing.T) { - require.ErrorContains(t, jobORM.CreateJob(ctx, &jb), "failed to validate AdaptiveSendSpec.TransmitterAddress: no EVM key matching") - }) + jb.OCR2OracleSpec.TransmitterID = null.StringFrom(transmitterID.String()) - t.Run("multiple jobs", func(t *testing.T) { - keyStore.Eth().XXXTestingOnlyAdd(ctx, adaptiveSendKey) - require.NoError(t, jobORM.CreateJob(ctx, &jb), "failed to validate AdaptiveSendSpec.TransmitterAddress: no EVM key matching") - }) + // Unknown transmitter address + require.ErrorContains(t, jobORM.CreateJob(ctx, &jb), "unknown dual transmission transmitterAddress: no EVM key matching:") + + // Should not error + keyStore.Eth().XXXTestingOnlyAdd(ctx, dtTransmitterAddress) + require.NoError(t, jobORM.CreateJob(ctx, &jb)) } diff --git a/core/services/job/models.go b/core/services/job/models.go index 5054abd08b5..231bf10fda0 100644 --- a/core/services/job/models.go +++ b/core/services/job/models.go @@ -185,7 +185,6 @@ type Job struct { CCIPSpecID *int32 CCIPSpec *CCIPSpec CCIPBootstrapSpecID *int32 - AdaptiveSendSpec *AdaptiveSendSpec `toml:"adaptiveSend"` JobSpecErrors []SpecError Type Type `toml:"type"` SchemaVersion uint32 `toml:"schemaVersion"` @@ -1061,26 +1060,3 @@ type CCIPSpec struct { // and RMN network info for offchain blessing. PluginConfig JSONConfig `toml:"pluginConfig"` } - -type AdaptiveSendSpec struct { - TransmitterAddress *evmtypes.EIP55Address `toml:"transmitterAddress"` - ContractAddress *evmtypes.EIP55Address `toml:"contractAddress"` - Delay time.Duration `toml:"delay"` - Metadata JSONConfig `toml:"metadata"` -} - -func (o *AdaptiveSendSpec) Validate() error { - if o.TransmitterAddress == nil { - return errors.New("no AdaptiveSendSpec.TransmitterAddress found") - } - - if o.ContractAddress == nil { - return errors.New("no AdaptiveSendSpec.ContractAddress found") - } - - if o.Delay.Seconds() <= 1 { - return errors.New("AdaptiveSendSpec.Delay not set or smaller than 1s") - } - - return nil -} diff --git a/core/services/job/models_test.go b/core/services/job/models_test.go index a21a4553219..5ef36ea9f48 100644 --- a/core/services/job/models_test.go +++ b/core/services/job/models_test.go @@ -11,8 +11,6 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/codec" "github.com/smartcontractkit/chainlink-common/pkg/types" pkgworkflows "github.com/smartcontractkit/chainlink-common/pkg/workflows" - "github.com/smartcontractkit/chainlink/v2/core/internal/cltest" - "github.com/smartcontractkit/chainlink/v2/core/internal/testutils" "github.com/smartcontractkit/chainlink/v2/core/services/job" "github.com/smartcontractkit/chainlink/v2/core/services/relay" @@ -352,62 +350,3 @@ func TestWorkflowSpec_Validate(t *testing.T) { require.NotEmpty(t, w.WorkflowID) }) } - -func TestAdaptiveSendConfig(t *testing.T) { - tests := []struct { - name string - shouldError bool - expectedErrorMessage string - config job.AdaptiveSendSpec - }{ - { - name: "AdaptiveSendSpec.TransmitterAddress not set", - shouldError: true, - expectedErrorMessage: "no AdaptiveSendSpec.TransmitterAddress found", - config: job.AdaptiveSendSpec{ - TransmitterAddress: nil, - ContractAddress: ptr(cltest.NewEIP55Address()), - Delay: time.Second * 30, - }, - }, - { - name: "AdaptiveSendSpec.ContractAddress not set", - shouldError: true, - expectedErrorMessage: "no AdaptiveSendSpec.ContractAddress found", - config: job.AdaptiveSendSpec{ - TransmitterAddress: ptr(cltest.NewEIP55Address()), - ContractAddress: nil, - Delay: time.Second * 30, - }, - }, - { - name: "AdaptiveSendSpec.Delay not set", - shouldError: true, - expectedErrorMessage: "AdaptiveSendSpec.Delay not set or smaller than 1s", - config: job.AdaptiveSendSpec{ - TransmitterAddress: ptr(cltest.NewEIP55Address()), - ContractAddress: ptr(cltest.NewEIP55Address()), - }, - }, - { - name: "AdaptiveSendSpec.Delay set to 50ms", - shouldError: true, - expectedErrorMessage: "AdaptiveSendSpec.Delay not set or smaller than 1s", - config: job.AdaptiveSendSpec{ - TransmitterAddress: ptr(cltest.NewEIP55Address()), - ContractAddress: ptr(cltest.NewEIP55Address()), - Delay: time.Millisecond * 50, - }, - }, - } - - for _, test := range tests { - t.Run(test.name, func(t *testing.T) { - if test.shouldError { - require.ErrorContains(t, test.config.Validate(), test.expectedErrorMessage) - } else { - require.NoError(t, test.config.Validate()) - } - }) - } -} diff --git a/core/services/job/orm.go b/core/services/job/orm.go index a86da5f7111..63fa2cb3b0d 100644 --- a/core/services/job/orm.go +++ b/core/services/job/orm.go @@ -20,7 +20,6 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/sqlutil" "github.com/smartcontractkit/chainlink-common/pkg/types" - "github.com/smartcontractkit/chainlink/v2/core/bridges" evmconfig "github.com/smartcontractkit/chainlink/v2/core/chains/evm/config" evmtypes "github.com/smartcontractkit/chainlink/v2/core/chains/evm/types" @@ -304,10 +303,29 @@ func (o *orm) CreateJob(ctx context.Context, jb *Job) error { } } - if jb.AdaptiveSendSpec != nil { - err = validateKeyStoreMatchForRelay(ctx, jb.OCR2OracleSpec.Relay, tx.keyStore, jb.AdaptiveSendSpec.TransmitterAddress.String()) - if err != nil { - return fmt.Errorf("failed to validate AdaptiveSendSpec.TransmitterAddress: %w", err) + if enableDualTransmission, ok := jb.OCR2OracleSpec.RelayConfig["enableDualTransmission"]; ok && enableDualTransmission != nil { + rawDualTransmissionConfig, ok := jb.OCR2OracleSpec.RelayConfig["dualTransmission"] + if !ok { + return errors.New("dual transmission is enabled but no dual transmission config present") + } + + dualTransmissionConfig, ok := rawDualTransmissionConfig.(map[string]interface{}) + if !ok { + return errors.New("invalid dual transmission config") + } + + dtContractAddress, ok := dualTransmissionConfig["contractAddress"].(string) + if !ok || !common.IsHexAddress(dtContractAddress) { + return errors.New("invalid contract address in dual transmission config") + } + + dtTransmitterAddress, ok := dualTransmissionConfig["transmitterAddress"].(string) + if !ok || !common.IsHexAddress(dtTransmitterAddress) { + return errors.New("invalid transmitter address in dual transmission config") + } + + if err = validateKeyStoreMatchForRelay(ctx, jb.OCR2OracleSpec.Relay, tx.keyStore, dtTransmitterAddress); err != nil { + return errors.Wrap(err, "unknown dual transmission transmitterAddress") } } diff --git a/core/services/ocr2/validate/validate.go b/core/services/ocr2/validate/validate.go index 40e2c11c3e7..c2f3e455232 100644 --- a/core/services/ocr2/validate/validate.go +++ b/core/services/ocr2/validate/validate.go @@ -73,9 +73,6 @@ func ValidatedOracleSpecToml(ctx context.Context, config OCR2Config, insConf Ins if err = validateTimingParameters(config, insConf, spec); err != nil { return jb, err } - if err = validateAdaptiveSendSpec(ctx, jb); err != nil { - return jb, err - } return jb, nil } @@ -380,10 +377,3 @@ func validateOCR2LLOSpec(jsonConfig job.JSONConfig) error { } return pkgerrors.Wrap(pluginConfig.Validate(), "LLO PluginConfig is invalid") } - -func validateAdaptiveSendSpec(ctx context.Context, spec job.Job) error { - if spec.AdaptiveSendSpec != nil { - return spec.AdaptiveSendSpec.Validate() - } - return nil -} diff --git a/core/services/ocrcommon/transmitter.go b/core/services/ocrcommon/transmitter.go index 5d2de45295f..8121f3778d2 100644 --- a/core/services/ocrcommon/transmitter.go +++ b/core/services/ocrcommon/transmitter.go @@ -2,7 +2,10 @@ package ocrcommon import ( "context" + errors2 "errors" + "fmt" "math/big" + "net/url" "slices" "github.com/ethereum/go-ethereum/common" @@ -11,6 +14,7 @@ import ( "github.com/smartcontractkit/chainlink/v2/common/txmgr/types" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/forwarders" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/txmgr" + types2 "github.com/smartcontractkit/chainlink/v2/core/services/relay/evm/types" ) type roundRobinKeystore interface { @@ -88,13 +92,14 @@ func NewOCR2FeedsTransmitter( checker txmgr.TransmitCheckerSpec, chainID *big.Int, keystore roundRobinKeystore, + dualTransmissionConfig *types2.DualTransmissionConfig, ) (Transmitter, error) { // Ensure that a keystore is provided. if keystore == nil { return nil, errors.New("nil keystore provided to transmitter") } - return &ocr2FeedsTransmitter{ + baseTransmitter := &ocr2FeedsTransmitter{ ocr2Aggregator: ocr2Aggregator, txManagerOCR2: txm, transmitter: transmitter{ @@ -107,7 +112,17 @@ func NewOCR2FeedsTransmitter( chainID: chainID, keystore: keystore, }, - }, nil + } + + if dualTransmissionConfig != nil { + return &ocr2FeedsDualTransmission{ + transmitter: *baseTransmitter, + secondaryContractAddress: dualTransmissionConfig.ContractAddress, + secondaryFromAddress: dualTransmissionConfig.TransmitterAddress, + secondaryMeta: dualTransmissionConfig.Meta, + }, nil + } + return baseTransmitter, nil } func (t *transmitter) CreateEthTransaction(ctx context.Context, toAddress common.Address, payload []byte, txMeta *txmgr.TxMeta) error { @@ -203,3 +218,57 @@ func (t *ocr2FeedsTransmitter) forwarderAddress(ctx context.Context, eoa, ocr2Ag return forwarderAddress, nil } + +type ocr2FeedsDualTransmission struct { + transmitter ocr2FeedsTransmitter + + secondaryContractAddress common.Address + secondaryFromAddress common.Address + secondaryMeta map[string][]string +} + +func (t *ocr2FeedsDualTransmission) CreateEthTransaction(ctx context.Context, toAddress common.Address, payload []byte, txMeta *txmgr.TxMeta) error { + // Primary transmission + errPrimary := t.transmitter.CreateEthTransaction(ctx, toAddress, payload, txMeta) + if errPrimary != nil { + errPrimary = fmt.Errorf("skipped primary transmission: %w", errPrimary) + } + + if txMeta == nil { + txMeta = &txmgr.TxMeta{} + } + + dualBroadcast := true + dualBroadcastParams := t.urlParams() + + txMeta.DualBroadcast = &dualBroadcast + txMeta.DualBroadcastParams = &dualBroadcastParams + + // Secondary transmission + _, errSecondary := t.transmitter.txm.CreateTransaction(ctx, txmgr.TxRequest{ + FromAddress: t.secondaryFromAddress, + ToAddress: t.secondaryContractAddress, + EncodedPayload: payload, + FeeLimit: t.transmitter.gasLimit, + Strategy: t.transmitter.strategy, + Checker: t.transmitter.checker, + Meta: txMeta, + }) + + errSecondary = errors.Wrap(errSecondary, "skipped secondary transmission") + return errors2.Join(errPrimary, errSecondary) +} + +func (t *ocr2FeedsDualTransmission) FromAddress(ctx context.Context) common.Address { + return t.transmitter.FromAddress(ctx) +} + +func (t *ocr2FeedsDualTransmission) urlParams() string { + values := url.Values{} + for k, v := range t.secondaryMeta { + for _, p := range v { + values.Add(k, p) + } + } + return values.Encode() +} diff --git a/core/services/ocrcommon/transmitter_test.go b/core/services/ocrcommon/transmitter_test.go index d6a07190800..5f434e59c62 100644 --- a/core/services/ocrcommon/transmitter_test.go +++ b/core/services/ocrcommon/transmitter_test.go @@ -5,16 +5,19 @@ import ( "testing" "github.com/ethereum/go-ethereum/common" + "github.com/stretchr/testify/assert" "github.com/stretchr/testify/mock" "github.com/stretchr/testify/require" commontxmmocks "github.com/smartcontractkit/chainlink/v2/common/txmgr/types/mocks" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/txmgr" txmmocks "github.com/smartcontractkit/chainlink/v2/core/chains/evm/txmgr/mocks" + "github.com/smartcontractkit/chainlink/v2/core/chains/evm/utils" "github.com/smartcontractkit/chainlink/v2/core/internal/cltest" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils/pgtest" "github.com/smartcontractkit/chainlink/v2/core/services/ocrcommon" + "github.com/smartcontractkit/chainlink/v2/core/services/relay/evm/types" ) func newMockTxStrategy(t *testing.T) *commontxmmocks.TxStrategy { @@ -169,3 +172,76 @@ func Test_DefaultTransmitter_Forwarding_Enabled_CreateEthTransaction_No_Keystore ) require.Error(t, err) } + +func Test_DualTransmitter(t *testing.T) { + t.Parallel() + + db := pgtest.NewSqlxDB(t) + ethKeyStore := cltest.NewKeyStore(t, db).Eth() + + _, fromAddress := cltest.MustInsertRandomKey(t, ethKeyStore) + _, secondaryFromAddress := cltest.MustInsertRandomKey(t, ethKeyStore) + + contractAddress := utils.RandomAddress() + secondaryContractAddress := utils.RandomAddress() + + gasLimit := uint64(1000) + chainID := big.NewInt(0) + effectiveTransmitterAddress := fromAddress + toAddress := testutils.NewAddress() + payload := []byte{1, 2, 3} + txm := txmmocks.NewMockEvmTxManager(t) + strategy := newMockTxStrategy(t) + dualTransmissionConfig := &types.DualTransmissionConfig{ + ContractAddress: secondaryContractAddress, + TransmitterAddress: secondaryFromAddress, + Meta: map[string][]string{ + "key1": {"value1"}, + "key2": {"value2", "value3"}, + "key3": {"value4", "value5", "value6"}, + }, + } + + transmitter, err := ocrcommon.NewOCR2FeedsTransmitter( + txm, + []common.Address{fromAddress}, + contractAddress, + gasLimit, + effectiveTransmitterAddress, + strategy, + txmgr.TransmitCheckerSpec{}, + chainID, + ethKeyStore, + dualTransmissionConfig, + ) + require.NoError(t, err) + + primaryTxConfirmed := false + secondaryTxConfirmed := false + + txm.On("CreateTransaction", mock.Anything, mock.MatchedBy(func(tx txmgr.TxRequest) bool { + switch tx.FromAddress { + case fromAddress: + // Primary transmission + assert.Equal(t, tx.ToAddress, toAddress, "unexpected primary toAddress") + assert.Nil(t, tx.Meta, "Meta should be empty") + primaryTxConfirmed = true + case secondaryFromAddress: + // Secondary transmission + assert.Equal(t, tx.ToAddress, secondaryContractAddress, "unexpected secondary toAddress") + assert.True(t, *tx.Meta.DualBroadcast, "DualBroadcast should be true") + assert.Equal(t, "key1=value1&key2=value2&key2=value3&key3=value4&key3=value5&key3=value6", *tx.Meta.DualBroadcastParams, "DualBroadcastParams not equal") + secondaryTxConfirmed = true + default: + // Should never be reached + return false + } + + return true + })).Twice().Return(txmgr.Tx{}, nil) + + require.NoError(t, transmitter.CreateEthTransaction(testutils.Context(t), toAddress, payload, nil)) + + require.True(t, primaryTxConfirmed) + require.True(t, secondaryTxConfirmed) +} diff --git a/core/services/relay/evm/evm.go b/core/services/relay/evm/evm.go index 7c070cc282d..db0fe90796b 100644 --- a/core/services/relay/evm/evm.go +++ b/core/services/relay/evm/evm.go @@ -806,6 +806,7 @@ func generateTransmitterFrom(ctx context.Context, rargs commontypes.RelayArgs, e checker, configWatcher.chain.ID(), ethKeystore, + relayConfig.DualTransmissionConfig, ) case commontypes.CCIPExecution: transmitter, err = cciptransmitter.NewTransmitterWithStatusChecker( diff --git a/core/services/relay/evm/types/types.go b/core/services/relay/evm/types/types.go index e1a61098be5..2b56aee6379 100644 --- a/core/services/relay/evm/types/types.go +++ b/core/services/relay/evm/types/types.go @@ -192,6 +192,12 @@ func (c LLOConfigMode) String() string { return string(c) } +type DualTransmissionConfig struct { + ContractAddress common.Address `json:"contractAddress" toml:"contractAddress"` + TransmitterAddress common.Address `json:"transmitterAddress" toml:"transmitterAddress"` + Meta map[string][]string `json:"meta" toml:"meta"` +} + type RelayConfig struct { ChainID *big.Big `json:"chainID"` FromBlock uint64 `json:"fromBlock"` @@ -215,6 +221,10 @@ type RelayConfig struct { // LLO-specific LLODONID uint32 `json:"lloDonID" toml:"lloDonID"` LLOConfigMode LLOConfigMode `json:"lloConfigMode" toml:"lloConfigMode"` + + // DualTransmission specific + EnableDualTransmission bool `json:"enableDualTransmission" toml:"enableDualTransmission"` + DualTransmissionConfig *DualTransmissionConfig `json:"dualTransmission" toml:"dualTransmission"` } var ErrBadRelayConfig = errors.New("bad relay config") diff --git a/core/testdata/testspecs/v2_specs.go b/core/testdata/testspecs/v2_specs.go index d3d72467a83..d519ace6479 100644 --- a/core/testdata/testspecs/v2_specs.go +++ b/core/testdata/testspecs/v2_specs.go @@ -951,42 +951,28 @@ targets: inputs: consensus_output: $(a-consensus.outputs) ` -var OCR2EVMSpecMinimalWithAdaptiveSendTemplate = ` +var OCR2EVMDualTransmissionSpecMinimalTemplate = ` type = "offchainreporting2" schemaVersion = 1 -name = "%s" +name = "test-job" +relay = "evm" contractID = "0x613a38AC1659769640aaE063C651F48E0250454C" p2pv2Bootstrappers = [] -ocrKeyBundleID = "%s" -relay = "evm" -pluginType = "median" transmitterID = "%s" - +pluginType = "median" observationSource = """ ds [type=http method=GET url="https://chain.link/ETH-USD"]; ds_parse [type=jsonparse path="data.price" separator="."]; ds_multiply [type=multiply times=100]; ds -> ds_parse -> ds_multiply; """ - [pluginConfig] juelsPerFeeCoinSource = """ - ds1 [type=http method=GET url="https://chain.link/jules" allowunrestrictednetworkaccess="true"]; - ds1_parse [type=jsonparse path="answer"]; - ds1_multiply [type=multiply times=1]; - ds1 -> ds1_parse -> ds1_multiply; + ds [type=http method=GET url="https://chain.link/ETH-USD"]; + ds_parse [type=jsonparse path="data.price" separator="."]; + ds_multiply [type=multiply times=100]; + ds -> ds_parse -> ds_multiply; """ [relayConfig] chainID = 0 - -[adaptiveSend] -transmitterAddress = '%s' -contractAddress = '0xF67D0290337bca0847005C7ffD1BC75BA9AAE6e4' -delay = '30s' -[adaptiveSend.metadata] -arbitraryParam = 'arbitrary-value' ` - -func GetOCR2EVMWithAdaptiveSendSpecMinimal(keyBundle, transmitterID, secondaryTransmitterAddress string) string { - return fmt.Sprintf(OCR2EVMSpecMinimalWithAdaptiveSendTemplate, uuid.New(), keyBundle, transmitterID, secondaryTransmitterAddress) -}