-
Notifications
You must be signed in to change notification settings - Fork 1.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
keystone end to end integration tests
- Loading branch information
Showing
19 changed files
with
2,177 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
--- | ||
"chainlink": patch | ||
--- | ||
|
||
#internal end to end test for streams capabilities |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
302 changes: 302 additions & 0 deletions
302
core/capabilities/integration_tests/keystone_contracts_setup.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,302 @@ | ||
package integration_tests | ||
|
||
import ( | ||
"context" | ||
"encoding/hex" | ||
"fmt" | ||
"log" | ||
"strings" | ||
"testing" | ||
"time" | ||
|
||
"github.com/ethereum/go-ethereum" | ||
"github.com/ethereum/go-ethereum/accounts/abi/bind/backends" | ||
"github.com/ethereum/go-ethereum/common" | ||
"github.com/ethereum/go-ethereum/core" | ||
"github.com/ethereum/go-ethereum/eth/ethconfig" | ||
"github.com/stretchr/testify/require" | ||
"google.golang.org/protobuf/proto" | ||
|
||
ragetypes "github.com/smartcontractkit/libocr/ragep2p/types" | ||
|
||
"github.com/smartcontractkit/chainlink-common/pkg/capabilities" | ||
|
||
"github.com/smartcontractkit/chainlink/v2/core/chains/evm/assets" | ||
"github.com/smartcontractkit/chainlink/v2/core/gethwrappers/keystone/generated/feeds_consumer" | ||
"github.com/smartcontractkit/chainlink/v2/core/gethwrappers/keystone/generated/forwarder" | ||
"github.com/smartcontractkit/chainlink/v2/core/internal/cltest" | ||
"github.com/smartcontractkit/chainlink/v2/core/internal/testutils" | ||
|
||
"github.com/ethereum/go-ethereum/accounts/abi/bind" | ||
|
||
remotetypes "github.com/smartcontractkit/chainlink/v2/core/capabilities/remote/types" | ||
kcr "github.com/smartcontractkit/chainlink/v2/core/gethwrappers/keystone/generated/capabilities_registry" | ||
) | ||
|
||
type peer struct { | ||
PeerID string | ||
Signer string | ||
} | ||
|
||
func peerIDToBytes(peerID string) ([32]byte, error) { | ||
var peerIDB ragetypes.PeerID | ||
err := peerIDB.UnmarshalText([]byte(peerID)) | ||
if err != nil { | ||
return [32]byte{}, err | ||
} | ||
|
||
return peerIDB, nil | ||
} | ||
|
||
func peers(ps []peer) ([][32]byte, error) { | ||
out := [][32]byte{} | ||
for _, p := range ps { | ||
b, err := peerIDToBytes(p.PeerID) | ||
if err != nil { | ||
return nil, err | ||
} | ||
|
||
out = append(out, b) | ||
} | ||
|
||
return out, nil | ||
} | ||
|
||
func peerToNode(nopID uint32, p peer) (kcr.CapabilitiesRegistryNodeParams, error) { | ||
peerIDB, err := peerIDToBytes(p.PeerID) | ||
if err != nil { | ||
return kcr.CapabilitiesRegistryNodeParams{}, fmt.Errorf("failed to convert peerID: %w", err) | ||
} | ||
|
||
sig := strings.TrimPrefix(p.Signer, "0x") | ||
signerB, err := hex.DecodeString(sig) | ||
if err != nil { | ||
return kcr.CapabilitiesRegistryNodeParams{}, fmt.Errorf("failed to convert signer: %w", err) | ||
} | ||
|
||
var sigb [32]byte | ||
copy(sigb[:], signerB) | ||
|
||
return kcr.CapabilitiesRegistryNodeParams{ | ||
NodeOperatorId: nopID, | ||
P2pId: peerIDB, | ||
Signer: sigb, | ||
}, nil | ||
} | ||
|
||
func setupCapabilitiesRegistryContract(ctx context.Context, t *testing.T, workflowDonPeers []peer, triggerDonPeers []peer, | ||
targetDonPeerIDs []peer, | ||
transactOpts *bind.TransactOpts, backend *ethBackend) common.Address { | ||
addr, _, reg, err := kcr.DeployCapabilitiesRegistry(transactOpts, backend) | ||
require.NoError(t, err) | ||
|
||
backend.Commit() | ||
|
||
streamsTrigger := kcr.CapabilitiesRegistryCapability{ | ||
LabelledName: "streams-trigger", | ||
Version: "1.0.0", | ||
CapabilityType: uint8(capabilities.CapabilityTypeTrigger), | ||
} | ||
sid, err := reg.GetHashedCapabilityId(&bind.CallOpts{}, streamsTrigger.LabelledName, streamsTrigger.Version) | ||
require.NoError(t, err) | ||
|
||
writeChain := kcr.CapabilitiesRegistryCapability{ | ||
LabelledName: "write_geth-testnet", | ||
Version: "1.0.0", | ||
CapabilityType: uint8(capabilities.CapabilityTypeTarget), | ||
} | ||
wid, err := reg.GetHashedCapabilityId(&bind.CallOpts{}, writeChain.LabelledName, writeChain.Version) | ||
if err != nil { | ||
log.Printf("failed to call GetHashedCapabilityId: %s", err) | ||
} | ||
|
||
ocr := kcr.CapabilitiesRegistryCapability{ | ||
LabelledName: "offchain_reporting", | ||
Version: "1.0.0", | ||
CapabilityType: uint8(capabilities.CapabilityTypeConsensus), | ||
} | ||
ocrid, err := reg.GetHashedCapabilityId(&bind.CallOpts{}, ocr.LabelledName, ocr.Version) | ||
require.NoError(t, err) | ||
|
||
_, err = reg.AddCapabilities(transactOpts, []kcr.CapabilitiesRegistryCapability{ | ||
streamsTrigger, | ||
writeChain, | ||
ocr, | ||
}) | ||
require.NoError(t, err) | ||
backend.Commit() | ||
|
||
_, err = reg.AddNodeOperators(transactOpts, []kcr.CapabilitiesRegistryNodeOperator{ | ||
{ | ||
Admin: transactOpts.From, | ||
Name: "TEST_NODE_OPERATOR", | ||
}, | ||
}) | ||
require.NoError(t, err) | ||
blockHash := backend.Commit() | ||
|
||
logs, err := backend.FilterLogs(ctx, ethereum.FilterQuery{ | ||
BlockHash: &blockHash, | ||
FromBlock: nil, | ||
ToBlock: nil, | ||
Addresses: nil, | ||
Topics: nil, | ||
}) | ||
|
||
require.NoError(t, err) | ||
|
||
recLog, err := reg.ParseNodeOperatorAdded(logs[0]) | ||
require.NoError(t, err) | ||
|
||
nopID := recLog.NodeOperatorId | ||
nodes := []kcr.CapabilitiesRegistryNodeParams{} | ||
for _, wfPeer := range workflowDonPeers { | ||
n, innerErr := peerToNode(nopID, wfPeer) | ||
require.NoError(t, innerErr) | ||
|
||
n.HashedCapabilityIds = [][32]byte{ocrid} | ||
nodes = append(nodes, n) | ||
} | ||
|
||
for _, triggerPeer := range triggerDonPeers { | ||
n, innerErr := peerToNode(nopID, triggerPeer) | ||
require.NoError(t, innerErr) | ||
|
||
n.HashedCapabilityIds = [][32]byte{sid} | ||
nodes = append(nodes, n) | ||
} | ||
|
||
for _, targetPeer := range targetDonPeerIDs { | ||
n, innerErr := peerToNode(nopID, targetPeer) | ||
require.NoError(t, innerErr) | ||
|
||
n.HashedCapabilityIds = [][32]byte{wid} | ||
nodes = append(nodes, n) | ||
} | ||
|
||
_, err = reg.AddNodes(transactOpts, nodes) | ||
require.NoError(t, err) | ||
|
||
// workflow DON | ||
ps, err := peers(workflowDonPeers) | ||
require.NoError(t, err) | ||
|
||
cfgs := []kcr.CapabilitiesRegistryCapabilityConfiguration{ | ||
{ | ||
CapabilityId: ocrid, | ||
}, | ||
} | ||
|
||
workflowDonF := uint8(2) | ||
_, err = reg.AddDON(transactOpts, ps, cfgs, false, true, workflowDonF) | ||
require.NoError(t, err) | ||
|
||
// trigger DON | ||
ps, err = peers(triggerDonPeers) | ||
require.NoError(t, err) | ||
|
||
triggerDonF := 1 | ||
config := &remotetypes.RemoteTriggerConfig{ | ||
RegistrationRefreshMs: 20000, | ||
RegistrationExpiryMs: 60000, | ||
// F + 1 | ||
MinResponsesToAggregate: uint32(triggerDonF) + 1, | ||
} | ||
configb, err := proto.Marshal(config) | ||
require.NoError(t, err) | ||
|
||
cfgs = []kcr.CapabilitiesRegistryCapabilityConfiguration{ | ||
{ | ||
CapabilityId: sid, | ||
Config: configb, | ||
}, | ||
} | ||
|
||
_, err = reg.AddDON(transactOpts, ps, cfgs, true, false, uint8(triggerDonF)) | ||
require.NoError(t, err) | ||
|
||
// target DON | ||
ps, err = peers(targetDonPeerIDs) | ||
require.NoError(t, err) | ||
|
||
cfgs = []kcr.CapabilitiesRegistryCapabilityConfiguration{ | ||
{ | ||
CapabilityId: wid, | ||
}, | ||
} | ||
|
||
targetDonF := uint8(1) | ||
_, err = reg.AddDON(transactOpts, ps, cfgs, true, false, targetDonF) | ||
require.NoError(t, err) | ||
|
||
backend.Commit() | ||
|
||
return addr | ||
} | ||
|
||
func setupForwarderContract(t *testing.T, workflowDonPeers []peer, workflowDonId uint32, | ||
configVersion uint32, f uint8, | ||
transactOpts *bind.TransactOpts, backend *ethBackend) (common.Address, *forwarder.KeystoneForwarder) { | ||
addr, _, fwd, err := forwarder.DeployKeystoneForwarder(transactOpts, backend) | ||
require.NoError(t, err) | ||
backend.Commit() | ||
|
||
var signers []common.Address | ||
for _, p := range workflowDonPeers { | ||
signers = append(signers, common.HexToAddress(p.Signer)) | ||
} | ||
|
||
_, err = fwd.SetConfig(transactOpts, workflowDonId, configVersion, f, signers) | ||
require.NoError(t, err) | ||
backend.Commit() | ||
|
||
return addr, fwd | ||
} | ||
|
||
func setupConsumerContract(t *testing.T, transactOpts *bind.TransactOpts, backend *ethBackend, | ||
forwarderAddress common.Address, workflowOwner string, workflowName string) (common.Address, *feeds_consumer.KeystoneFeedsConsumer) { | ||
addr, _, consumer, err := feeds_consumer.DeployKeystoneFeedsConsumer(transactOpts, backend) | ||
require.NoError(t, err) | ||
backend.Commit() | ||
|
||
var nameBytes [10]byte | ||
copy(nameBytes[:], workflowName) | ||
|
||
ownerAddr := common.HexToAddress(workflowOwner) | ||
|
||
_, err = consumer.SetConfig(transactOpts, []common.Address{forwarderAddress}, []common.Address{ownerAddr}, [][10]byte{nameBytes}) | ||
require.NoError(t, err) | ||
|
||
backend.Commit() | ||
|
||
return addr, consumer | ||
} | ||
|
||
type ethBackend struct { | ||
*backends.SimulatedBackend | ||
} | ||
|
||
func setupBlockchain(t *testing.T, initialEth int) (*ethBackend, *bind.TransactOpts) { | ||
transactOpts := testutils.MustNewSimTransactor(t) // config contract deployer and owner | ||
genesisData := core.GenesisAlloc{transactOpts.From: {Balance: assets.Ether(initialEth).ToInt()}} | ||
backend := cltest.NewSimulatedBackend(t, genesisData, uint32(ethconfig.Defaults.Miner.GasCeil)) | ||
backend.Commit() | ||
|
||
return ðBackend{backend}, transactOpts | ||
} | ||
|
||
func (b ethBackend) Start(ctx context.Context, blockTimeProcessingTime time.Duration) { | ||
go func() { | ||
ticker := time.NewTicker(blockTimeProcessingTime) | ||
defer ticker.Stop() | ||
|
||
for { | ||
select { | ||
case <-ctx.Done(): | ||
return | ||
case <-ticker.C: | ||
b.SimulatedBackend.Commit() | ||
} | ||
} | ||
}() | ||
} |
Oops, something went wrong.