diff --git a/.changeset/blue-pumpkins-sniff.md b/.changeset/blue-pumpkins-sniff.md new file mode 100644 index 00000000000..0a7576f328e --- /dev/null +++ b/.changeset/blue-pumpkins-sniff.md @@ -0,0 +1,5 @@ +--- +"chainlink": minor +--- + +#internal move workflow validation to common repo diff --git a/.changeset/early-shoes-sit.md b/.changeset/early-shoes-sit.md new file mode 100644 index 00000000000..189fe37e1fc --- /dev/null +++ b/.changeset/early-shoes-sit.md @@ -0,0 +1,10 @@ +--- +"chainlink": patch +--- + +Fixed CPU usage issues caused by inefficiencies in HeadTracker. + +HeadTracker's support of finality tags caused a drastic increase in the number of tracked blocks on the Arbitrum chain (from 50 to 12,000), which has led to a 30% increase in CPU usage. + +The fix improves the data structure for tracking blocks and makes lookup more efficient. BenchmarkHeadTracker_Backfill shows 40x time reduction. +#bugfix diff --git a/.changeset/hungry-carpets-flow.md b/.changeset/hungry-carpets-flow.md new file mode 100644 index 00000000000..19835b99c17 --- /dev/null +++ b/.changeset/hungry-carpets-flow.md @@ -0,0 +1,5 @@ +--- +"chainlink": minor +--- + +Added a mechanism to validate forwarders for OCR2 and fallback to EOA if necessary #added diff --git a/.github/actions/setup-create-base64-config-live-testnets/action.yml b/.github/actions/setup-create-base64-config-live-testnets/action.yml index 0dac6e9d55b..64fc134b46e 100644 --- a/.github/actions/setup-create-base64-config-live-testnets/action.yml +++ b/.github/actions/setup-create-base64-config-live-testnets/action.yml @@ -89,6 +89,11 @@ runs: pyroscope_enabled=false fi + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi + cat << EOF > config.toml [Common] chainlink_node_funding=0.5 @@ -118,7 +123,7 @@ runs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token [Network] selected_networks=["$NETWORK"] diff --git a/.github/actions/setup-create-base64-config/action.yml b/.github/actions/setup-create-base64-config/action.yml index 46de20b6cd6..d54554278d1 100644 --- a/.github/actions/setup-create-base64-config/action.yml +++ b/.github/actions/setup-create-base64-config/action.yml @@ -111,6 +111,11 @@ runs: else execution_layer="geth" fi + + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi cat << EOF > config.toml [Network] @@ -142,7 +147,7 @@ runs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token [PrivateEthereumNetwork] execution_layer="$execution_layer" diff --git a/.github/actions/setup-create-base64-upgrade-config/action.yml b/.github/actions/setup-create-base64-upgrade-config/action.yml index a7cfcafde4c..c2d0bc19f35 100644 --- a/.github/actions/setup-create-base64-upgrade-config/action.yml +++ b/.github/actions/setup-create-base64-upgrade-config/action.yml @@ -83,6 +83,11 @@ runs: log_targets=$(convert_to_toml_array "$LOGSTREAM_LOG_TARGETS") + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi + cat << EOF > config.toml [Network] selected_networks=$selected_networks @@ -112,7 +117,7 @@ runs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token EOF BASE64_CONFIG_OVERRIDE=$(cat config.toml | base64 -w 0) diff --git a/.github/actions/setup-merge-base64-config/action.yml b/.github/actions/setup-merge-base64-config/action.yml index 48ca96bf948..b734873c1f6 100644 --- a/.github/actions/setup-merge-base64-config/action.yml +++ b/.github/actions/setup-merge-base64-config/action.yml @@ -45,6 +45,11 @@ runs: echo "NETWORKS=$NETWORKS" >> $GITHUB_ENV fi + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi + # use Loki config from GH secrets and merge it with base64 input cat << EOF > config.toml [Logging.Loki] @@ -57,7 +62,7 @@ runs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token EOF echo "$decoded_toml" >> final_config.toml diff --git a/.github/workflows/automation-ondemand-tests.yml b/.github/workflows/automation-ondemand-tests.yml index c232a6ba9e6..710cd72e1b4 100644 --- a/.github/workflows/automation-ondemand-tests.yml +++ b/.github/workflows/automation-ondemand-tests.yml @@ -273,6 +273,11 @@ jobs: log_targets=$(convert_to_toml_array "$LOGSTREAM_LOG_TARGETS") + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi + cat << EOF > config.toml [Network] selected_networks=$selected_networks @@ -300,7 +305,7 @@ jobs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token [Pyroscope] enabled=$pyroscope_enabled diff --git a/.github/workflows/client-compatibility-tests.yml b/.github/workflows/client-compatibility-tests.yml index 6b327824fb6..43e366c478c 100644 --- a/.github/workflows/client-compatibility-tests.yml +++ b/.github/workflows/client-compatibility-tests.yml @@ -195,6 +195,11 @@ jobs: pyroscope_enabled=false fi + grafana_bearer_token="" + if [ -n "$GRAFANA_BEARER_TOKEN" ]; then + grafana_bearer_token="bearer_token_secret=\"$GRAFANA_BEARER_TOKEN\"" + fi + cat << EOF > config.toml [Network] selected_networks=$selected_networks @@ -212,7 +217,7 @@ jobs: [Logging.Grafana] base_url="$GRAFANA_URL" dashboard_url="$GRAFANA_DASHBOARD_URL" - bearer_token_secret="$GRAFANA_BEARER_TOKEN" + $grafana_bearer_token [PrivateEthereumNetwork] ethereum_version="eth2" diff --git a/.github/workflows/on-demand-keeper-smoke-tests.yml b/.github/workflows/on-demand-keeper-smoke-tests.yml new file mode 100644 index 00000000000..6fee8bfc505 --- /dev/null +++ b/.github/workflows/on-demand-keeper-smoke-tests.yml @@ -0,0 +1,292 @@ +name: On Demand Keeper Smoke Tests +run-name: On Demand Keeper Smoke Tests ${{ inputs.distinct_run_name && inputs.distinct_run_name || '' }} +on: + workflow_dispatch: + inputs: + distinct_run_name: + description: 'A unique identifier for this run, only use from other repos' + required: false + type: string + +# Only run 1 of this workflow at a time per PR +concurrency: + group: on-demand-keeper-smoke-tests-${{ github.ref }}-${{ inputs.distinct_run_name }} + cancel-in-progress: true + +env: + # for run-test variables and environment + ENV_JOB_IMAGE: ${{ secrets.QA_AWS_ACCOUNT_NUMBER }}.dkr.ecr.${{ secrets.QA_AWS_REGION }}.amazonaws.com/chainlink-tests:${{ inputs.evm-ref || github.sha }} + CHAINLINK_IMAGE: ${{ secrets.QA_AWS_ACCOUNT_NUMBER }}.dkr.ecr.${{ secrets.QA_AWS_REGION }}.amazonaws.com/chainlink + TEST_SUITE: smoke + TEST_ARGS: -test.timeout 12m + INTERNAL_DOCKER_REPO: ${{ secrets.QA_AWS_ACCOUNT_NUMBER }}.dkr.ecr.${{ secrets.QA_AWS_REGION }}.amazonaws.com + MOD_CACHE_VERSION: 2 + COLLECTION_ID: chainlink-e2e-tests + +jobs: + build-chainlink: + environment: integration + permissions: + id-token: write + contents: read + strategy: + matrix: + image: + - name: "" + dockerfile: core/chainlink.Dockerfile + tag-suffix: "" + name: Build Chainlink Image ${{ matrix.image.name }} + runs-on: ubuntu22.04-16cores-64GB + needs: [changes] + steps: + - name: Collect Metrics + id: collect-gha-metrics + uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 + with: + id: ${{ env.COLLECTION_ID }}-build-chainlink + org-id: ${{ secrets.GRAFANA_INTERNAL_TENANT_ID }} + basic-auth: ${{ secrets.GRAFANA_INTERNAL_BASIC_AUTH }} + hostname: ${{ secrets.GRAFANA_INTERNAL_HOST }} + this-job-name: Build Chainlink Image ${{ matrix.image.name }} + continue-on-error: true + - name: Checkout the repo + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref || github.event.pull_request.head.sha || github.event.merge_group.head_sha }} + - name: Build Chainlink Image + uses: ./.github/actions/build-chainlink-image + with: + tag_suffix: ${{ matrix.image.tag-suffix }} + dockerfile: ${{ matrix.image.dockerfile }} + git_commit_sha: ${{ inputs.evm-ref || github.sha }} + AWS_REGION: ${{ secrets.QA_AWS_REGION }} + AWS_ROLE_TO_ASSUME: ${{ secrets.QA_AWS_ROLE_TO_ASSUME }} + dep_evm_sha: ${{ inputs.evm-ref }} + + compare-tests: + runs-on: ubuntu-latest + name: Build Automation Test List + outputs: + automation-matrix: ${{ env.AUTOMATION_JOB_MATRIX_JSON }} + steps: + - name: Checkout the repo + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref }} + - name: Compare Test Lists + run: | + cd ./integration-tests + ./scripts/compareTestList.sh ./smoke/keeper_test.go + - name: Build Test Matrix Lists + id: build-test-matrix-list + run: | + cd ./integration-tests + KEEPER_JOB_MATRIX_JSON=$(./scripts/buildTestMatrixList.sh ./smoke/keeper_test.go keeper ubuntu-latest 1) + echo "AUTOMATION_JOB_MATRIX_JSON=${KEEPER_JOB_MATRIX_JSON}" >> $GITHUB_ENV + + eth-smoke-tests-matrix-automation: + if: ${{ !contains(join(github.event.pull_request.labels.*.name, ' '), 'skip-smoke-tests') }} + environment: integration + permissions: + checks: write + pull-requests: write + id-token: write + contents: read + needs: + [build-chainlink, compare-tests] + env: + SELECTED_NETWORKS: SIMULATED,SIMULATED_1,SIMULATED_2 + CHAINLINK_COMMIT_SHA: ${{ inputs.evm-ref || github.sha }} + CHAINLINK_ENV_USER: ${{ github.actor }} + TEST_LOG_LEVEL: debug + strategy: + fail-fast: false + matrix: + product: ${{fromJson(needs.compare-tests.outputs.automation-matrix)}} + runs-on: ${{ matrix.product.os }} + name: ETH Smoke Tests ${{ matrix.product.name }} + steps: + - name: Collect Metrics + id: collect-gha-metrics + uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 + with: + id: ${{ env.COLLECTION_ID }}-matrix-${{ matrix.product.name }} + basic-auth: ${{ secrets.GRAFANA_INTERNAL_BASIC_AUTH }} + hostname: ${{ secrets.GRAFANA_INTERNAL_HOST }} + org-id: ${{ secrets.GRAFANA_INTERNAL_TENANT_ID }} + this-job-name: ETH Smoke Tests ${{ matrix.product.name }} + test-results-file: '{"testType":"go","filePath":"/tmp/gotest.log"}' + continue-on-error: true + - name: Checkout the repo + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref || github.event.pull_request.head.sha || github.event.merge_group.head_sha }} + - name: Build Go Test Command + id: build-go-test-command + run: | + # if the matrix.product.run is set, use it for a different command + if [ "${{ matrix.product.run }}" != "" ]; then + echo "run_command=${{ matrix.product.run }} ./smoke/${{ matrix.product.file }}_test.go" >> "$GITHUB_OUTPUT" + else + echo "run_command=./smoke/${{ matrix.product.name }}_test.go" >> "$GITHUB_OUTPUT" + fi + - name: Prepare Base64 TOML override + uses: ./.github/actions/setup-create-base64-config + with: + runId: ${{ github.run_id }} + testLogCollect: ${{ vars.TEST_LOG_COLLECT }} + selectedNetworks: ${{ env.SELECTED_NETWORKS }} + chainlinkImage: ${{ env.CHAINLINK_IMAGE }} + chainlinkVersion: ${{ inputs.evm-ref || github.sha }} + pyroscopeServer: ${{ matrix.product.pyroscope_env == '' && '' || !startsWith(github.ref, 'refs/tags/') && '' || secrets.QA_PYROSCOPE_INSTANCE }} # Avoid sending blank envs https://github.com/orgs/community/discussions/25725 + pyroscopeEnvironment: ${{ matrix.product.pyroscope_env }} + pyroscopeKey: ${{ secrets.QA_PYROSCOPE_KEY }} + lokiEndpoint: https://${{ secrets.GRAFANA_INTERNAL_HOST }}/loki/api/v1/push + lokiTenantId: ${{ secrets.GRAFANA_INTERNAL_TENANT_ID }} + lokiBasicAuth: ${{ secrets.GRAFANA_INTERNAL_BASIC_AUTH }} + logstreamLogTargets: ${{ vars.LOGSTREAM_LOG_TARGETS }} + grafanaUrl: ${{ vars.GRAFANA_URL }} + grafanaDashboardUrl: "/d/ddf75041-1e39-42af-aa46-361fe4c36e9e/ci-e2e-tests-logs" + + ## Run this step when changes that require tests to be run are made + - name: Run Tests + uses: smartcontractkit/chainlink-github-actions/chainlink-testing-framework/run-tests@519851800779323566b7b7c22cc21bff95dbb639 + with: + test_command_to_run: cd ./integration-tests && go test -timeout 30m -count=1 -json -test.parallel=${{ matrix.product.nodes }} ${{ steps.build-go-test-command.outputs.run_command }} 2>&1 | tee /tmp/gotest.log | gotestloghelper -ci -singlepackage + test_download_vendor_packages_command: cd ./integration-tests && go mod download + cl_repo: ${{ env.CHAINLINK_IMAGE }} + cl_image_tag: ${{ inputs.evm-ref || github.sha }} + aws_registries: ${{ secrets.QA_AWS_ACCOUNT_NUMBER }} + artifacts_name: ${{ matrix.product.name }}-test-logs + artifacts_location: | + ./integration-tests/smoke/logs/ + /tmp/gotest.log + publish_check_name: ${{ matrix.product.name }} + token: ${{ secrets.GITHUB_TOKEN }} + go_mod_path: ./integration-tests/go.mod + cache_key_id: core-e2e-${{ env.MOD_CACHE_VERSION }} + cache_restore_only: "true" + QA_AWS_REGION: ${{ secrets.QA_AWS_REGION }} + QA_AWS_ROLE_TO_ASSUME: ${{ secrets.QA_AWS_ROLE_TO_ASSUME }} + QA_KUBECONFIG: "" + should_tidy: "false" + go_coverage_src_dir: /var/tmp/go-coverage + go_coverage_dest_dir: ${{ github.workspace }}/.covdata + + - name: Upload Coverage Data + uses: actions/upload-artifact@65462800fd760344b1a7b4382951275a0abb4808 # v4.3.3 + with: + name: cl-node-coverage-data-${{ matrix.product.name }} + path: .covdata + retention-days: 1 + + - name: Print failed test summary + if: always() + uses: smartcontractkit/chainlink-github-actions/chainlink-testing-framework/show-test-summary@5dd916d08c03cb5f9a97304f4f174820421bb946 # v2.3.11 + + ### Used to check the required checks box when the matrix completes + eth-smoke-tests: + if: always() + runs-on: ubuntu-latest + name: ETH Smoke Tests + needs: [eth-smoke-tests-matrix-automation] + steps: + - name: Check smoke test matrix status + if: needs.eth-smoke-tests-matrix-automation.result != 'success' + run: | + echo "Automation: ${{ needs.eth-smoke-tests-matrix-automation.result }}" + exit 1 + - name: Collect Metrics + if: always() + id: collect-gha-metrics + uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 + with: + id: ${{ env.COLLECTION_ID }}-matrix-results + org-id: ${{ secrets.GRAFANA_INTERNAL_TENANT_ID }} + basic-auth: ${{ secrets.GRAFANA_INTERNAL_BASIC_AUTH }} + hostname: ${{ secrets.GRAFANA_INTERNAL_HOST }} + this-job-name: ETH Smoke Tests + matrix-aggregator-status: ${{ needs.eth-smoke-tests-matrix.result }} + continue-on-error: true + + cleanup: + name: Clean up integration environment deployments + if: always() + needs: [eth-smoke-tests] + runs-on: ubuntu-latest + steps: + - name: Checkout repo + if: ${{ github.event_name == 'pull_request' }} + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref }} + + - name: 🧼 Clean up Environment + if: ${{ github.event_name == 'pull_request' }} + uses: ./.github/actions/delete-deployments + with: + environment: integration + ref: ${{ github.head_ref }} # See https://github.com/github/docs/issues/15319#issuecomment-1476705663 + + - name: Collect Metrics + if: ${{ github.event_name == 'pull_request' }} + id: collect-gha-metrics + uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 + with: + id: ${{ env.COLLECTION_ID }}-env-cleanup + org-id: ${{ secrets.GRAFANA_INTERNAL_TENANT_ID }} + basic-auth: ${{ secrets.GRAFANA_INTERNAL_BASIC_AUTH }} + hostname: ${{ secrets.GRAFANA_INTERNAL_HOST }} + this-job-name: Clean up integration environment deployments + continue-on-error: true + + show-coverage: + name: Show Chainlink Node Go Coverage + if: always() + needs: [cleanup] + runs-on: ubuntu-latest + steps: + - name: Checkout the repo + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref || github.event.pull_request.head.sha || github.event.merge_group.head_sha }} + - name: Download All Artifacts + uses: actions/download-artifact@9c19ed7fe5d278cd354c7dfd5d3b88589c7e2395 # v4.1.6 + with: + path: cl-node-coverage-data + pattern: cl-node-coverage-data-* + merge-multiple: true + - name: Show Coverage + run: go run ./integration-tests/scripts/show_coverage.go "${{ github.workspace }}/cl-node-coverage-data/*/merged" + + # Run the setup if the matrix finishes but this time save the cache if we have a cache hit miss + # this will also only run if both of the matrix jobs pass + eth-smoke-go-mod-cache: + + environment: integration + needs: [eth-smoke-tests] + runs-on: ubuntu-latest + name: ETH Smoke Tests Go Mod Cache + continue-on-error: true + steps: + - name: Checkout the repo + uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 + with: + repository: smartcontractkit/chainlink + ref: ${{ inputs.cl_ref || github.event.pull_request.head.sha || github.event.merge_group.head_sha }} + - name: Run Setup + uses: smartcontractkit/chainlink-github-actions/chainlink-testing-framework/setup-go@5dd916d08c03cb5f9a97304f4f174820421bb946 # v2.3.11 + with: + test_download_vendor_packages_command: | + cd ./integration-tests + go mod download + # force download of test dependencies + go test -run=NonExistentTest ./smoke/... || echo "ignore expected test failure" + go_mod_path: ./integration-tests/go.mod + cache_key_id: core-e2e-${{ env.MOD_CACHE_VERSION }} + cache_restore_only: "false" \ No newline at end of file diff --git a/.github/workflows/solidity.yml b/.github/workflows/solidity.yml index 22ed53e72bc..6183109097d 100644 --- a/.github/workflows/solidity.yml +++ b/.github/workflows/solidity.yml @@ -114,22 +114,19 @@ jobs: run: working-directory: contracts needs: [changes] + if: needs.changes.outputs.changes == 'true' name: Solidity Lint runs-on: ubuntu-latest steps: - name: Checkout the repo uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 - name: Setup NodeJS - if: needs.changes.outputs.changes == 'true' uses: ./.github/actions/setup-nodejs - name: Run pnpm lint - if: needs.changes.outputs.changes == 'true' run: pnpm lint - name: Run solhint - if: needs.changes.outputs.changes == 'true' run: pnpm solhint - name: Collect Metrics - if: needs.changes.outputs.changes == 'true' id: collect-gha-metrics uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 with: @@ -145,19 +142,17 @@ jobs: run: working-directory: contracts needs: [changes] + if: needs.changes.outputs.changes == 'true' name: Prettier Formatting runs-on: ubuntu-latest steps: - name: Checkout the repo uses: actions/checkout@9bb56186c3b09b4f86b1c65136769dd318469633 # v4.1.2 - name: Setup NodeJS - if: needs.changes.outputs.changes == 'true' uses: ./.github/actions/setup-nodejs - name: Run prettier check - if: needs.changes.outputs.changes == 'true' run: pnpm prettier:check - name: Collect Metrics - if: needs.changes.outputs.changes == 'true' id: collect-gha-metrics uses: smartcontractkit/push-gha-metrics-action@dea9b546553cb4ca936607c2267a09c004e4ab3f # v3.0.0 with: @@ -172,8 +167,8 @@ jobs: name: Publish Beta NPM environment: publish-contracts needs: [tag-check, changes, lint, prettier, native-compile, prepublish-test] - runs-on: ubuntu-latest if: needs.tag-check.outputs.is-pre-release == 'true' + runs-on: ubuntu-latest steps: - name: Checkout the repo uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 @@ -209,10 +204,10 @@ jobs: name: Publish Prod NPM environment: publish-contracts needs: [tag-check, changes, lint, prettier, native-compile, prepublish-test] + if: needs.tag-check.outputs.is-release == 'true' runs-on: ubuntu-latest permissions: contents: write - if: needs.tag-check.outputs.is-release == 'true' steps: - name: Checkout the repo uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 diff --git a/common/txmgr/mocks/tx_manager.go b/common/txmgr/mocks/tx_manager.go index 935e7313817..a3e8c489314 100644 --- a/common/txmgr/mocks/tx_manager.go +++ b/common/txmgr/mocks/tx_manager.go @@ -301,6 +301,34 @@ func (_m *TxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) GetFor return r0, r1 } +// GetForwarderForEOAOCR2Feeds provides a mock function with given fields: eoa, ocr2AggregatorID +func (_m *TxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) GetForwarderForEOAOCR2Feeds(eoa ADDR, ocr2AggregatorID ADDR) (ADDR, error) { + ret := _m.Called(eoa, ocr2AggregatorID) + + if len(ret) == 0 { + panic("no return value specified for GetForwarderForEOAOCR2Feeds") + } + + var r0 ADDR + var r1 error + if rf, ok := ret.Get(0).(func(ADDR, ADDR) (ADDR, error)); ok { + return rf(eoa, ocr2AggregatorID) + } + if rf, ok := ret.Get(0).(func(ADDR, ADDR) ADDR); ok { + r0 = rf(eoa, ocr2AggregatorID) + } else { + r0 = ret.Get(0).(ADDR) + } + + if rf, ok := ret.Get(1).(func(ADDR, ADDR) error); ok { + r1 = rf(eoa, ocr2AggregatorID) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + // HealthReport provides a mock function with given fields: func (_m *TxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) HealthReport() map[string]error { ret := _m.Called() diff --git a/common/txmgr/txmgr.go b/common/txmgr/txmgr.go index 4d4eabe5c40..1c8b59a55cc 100644 --- a/common/txmgr/txmgr.go +++ b/common/txmgr/txmgr.go @@ -47,6 +47,7 @@ type TxManager[ Trigger(addr ADDR) CreateTransaction(ctx context.Context, txRequest txmgrtypes.TxRequest[ADDR, TX_HASH]) (etx txmgrtypes.Tx[CHAIN_ID, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE], err error) GetForwarderForEOA(eoa ADDR) (forwarder ADDR, err error) + GetForwarderForEOAOCR2Feeds(eoa, ocr2AggregatorID ADDR) (forwarder ADDR, err error) RegisterResumeCallback(fn ResumeCallback) SendNativeToken(ctx context.Context, chainID CHAIN_ID, from, to ADDR, value big.Int, gasLimit uint64) (etx txmgrtypes.Tx[CHAIN_ID, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE], err error) Reset(addr ADDR, abandon bool) error @@ -553,6 +554,15 @@ func (b *Txm[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, R, SEQ, FEE]) GetForward return } +// GetForwarderForEOAOCR2Feeds calls forwarderMgr to get a proper forwarder for a given EOA and checks if its set as a transmitter on the OCR2Aggregator contract. +func (b *Txm[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, R, SEQ, FEE]) GetForwarderForEOAOCR2Feeds(eoa, ocr2Aggregator ADDR) (forwarder ADDR, err error) { + if !b.txConfig.ForwardersEnabled() { + return forwarder, fmt.Errorf("forwarding is not enabled, to enable set Transactions.ForwardersEnabled =true") + } + forwarder, err = b.fwdMgr.ForwarderForOCR2Feeds(eoa, ocr2Aggregator) + return +} + func (b *Txm[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, R, SEQ, FEE]) checkEnabled(ctx context.Context, addr ADDR) error { if err := b.keyStore.CheckEnabled(ctx, addr, b.chainID); err != nil { return fmt.Errorf("cannot send transaction from %s on chain ID %s: %w", addr, b.chainID.String(), err) @@ -649,6 +659,10 @@ func (n *NullTxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) Cre func (n *NullTxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) GetForwarderForEOA(addr ADDR) (fwdr ADDR, err error) { return fwdr, err } +func (n *NullTxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) GetForwarderForEOAOCR2Feeds(_, _ ADDR) (fwdr ADDR, err error) { + return fwdr, err +} + func (n *NullTxManager[CHAIN_ID, HEAD, ADDR, TX_HASH, BLOCK_HASH, SEQ, FEE]) Reset(addr ADDR, abandon bool) error { return nil } diff --git a/common/txmgr/types/forwarder_manager.go b/common/txmgr/types/forwarder_manager.go index 4d70b730004..3e51ffb1524 100644 --- a/common/txmgr/types/forwarder_manager.go +++ b/common/txmgr/types/forwarder_manager.go @@ -9,6 +9,7 @@ import ( type ForwarderManager[ADDR types.Hashable] interface { services.Service ForwarderFor(addr ADDR) (forwarder ADDR, err error) + ForwarderForOCR2Feeds(eoa, ocr2Aggregator ADDR) (forwarder ADDR, err error) // Converts payload to be forwarder-friendly ConvertPayload(dest ADDR, origPayload []byte) ([]byte, error) } diff --git a/common/txmgr/types/mocks/forwarder_manager.go b/common/txmgr/types/mocks/forwarder_manager.go index fe40e7bb5e2..1021e776e9d 100644 --- a/common/txmgr/types/mocks/forwarder_manager.go +++ b/common/txmgr/types/mocks/forwarder_manager.go @@ -91,6 +91,34 @@ func (_m *ForwarderManager[ADDR]) ForwarderFor(addr ADDR) (ADDR, error) { return r0, r1 } +// ForwarderForOCR2Feeds provides a mock function with given fields: eoa, ocr2Aggregator +func (_m *ForwarderManager[ADDR]) ForwarderForOCR2Feeds(eoa ADDR, ocr2Aggregator ADDR) (ADDR, error) { + ret := _m.Called(eoa, ocr2Aggregator) + + if len(ret) == 0 { + panic("no return value specified for ForwarderForOCR2Feeds") + } + + var r0 ADDR + var r1 error + if rf, ok := ret.Get(0).(func(ADDR, ADDR) (ADDR, error)); ok { + return rf(eoa, ocr2Aggregator) + } + if rf, ok := ret.Get(0).(func(ADDR, ADDR) ADDR); ok { + r0 = rf(eoa, ocr2Aggregator) + } else { + r0 = ret.Get(0).(ADDR) + } + + if rf, ok := ret.Get(1).(func(ADDR, ADDR) error); ok { + r1 = rf(eoa, ocr2Aggregator) + } else { + r1 = ret.Error(1) + } + + return r0, r1 +} + // HealthReport provides a mock function with given fields: func (_m *ForwarderManager[ADDR]) HealthReport() map[string]error { ret := _m.Called() diff --git a/contracts/.changeset/seven-apes-drop.md b/contracts/.changeset/seven-apes-drop.md new file mode 100644 index 00000000000..2882975f663 --- /dev/null +++ b/contracts/.changeset/seven-apes-drop.md @@ -0,0 +1,5 @@ +--- +"@chainlink/contracts": patch +--- + +increase solhint max-warnings to 2 (from 0) to allow workflow to pass diff --git a/contracts/package.json b/contracts/package.json index 306f8284b5c..0101a202d4f 100644 --- a/contracts/package.json +++ b/contracts/package.json @@ -18,7 +18,7 @@ "prepublishOnly": "pnpm compile && ./scripts/prepublish_generate_abi_folder", "publish-beta": "pnpm publish --tag beta", "publish-prod": "pnpm publish --tag latest", - "solhint": "solhint --max-warnings 0 \"./src/v0.8/**/*.sol\"" + "solhint": "solhint --max-warnings 2 \"./src/v0.8/**/*.sol\"" }, "files": [ "src/v0.8", diff --git a/core/capabilities/streams/codec.go b/core/capabilities/streams/codec.go index c04d7ee6a0c..48befd602ac 100644 --- a/core/capabilities/streams/codec.go +++ b/core/capabilities/streams/codec.go @@ -1,21 +1,21 @@ package streams import ( - "github.com/smartcontractkit/chainlink-common/pkg/capabilities/mercury" + "github.com/smartcontractkit/chainlink-common/pkg/capabilities/datastreams" "github.com/smartcontractkit/chainlink-common/pkg/values" ) type Codec struct { } -func (c Codec) Unwrap(raw values.Value) ([]mercury.FeedReport, error) { - dest := []mercury.FeedReport{} +func (c Codec) Unwrap(raw values.Value) ([]datastreams.FeedReport, error) { + dest := []datastreams.FeedReport{} err := raw.UnwrapTo(&dest) // TODO (KS-196): validate reports return dest, err } -func (c Codec) Wrap(reports []mercury.FeedReport) (values.Value, error) { +func (c Codec) Wrap(reports []datastreams.FeedReport) (values.Value, error) { return values.Wrap(reports) } diff --git a/core/capabilities/syncer.go b/core/capabilities/syncer.go index 67e069c831d..98a77dc458a 100644 --- a/core/capabilities/syncer.go +++ b/core/capabilities/syncer.go @@ -8,7 +8,7 @@ import ( "time" "github.com/smartcontractkit/chainlink-common/pkg/capabilities" - "github.com/smartcontractkit/chainlink-common/pkg/capabilities/mercury" + "github.com/smartcontractkit/chainlink-common/pkg/capabilities/datastreams" "github.com/smartcontractkit/chainlink-common/pkg/capabilities/triggers" "github.com/smartcontractkit/chainlink-common/pkg/services" "github.com/smartcontractkit/chainlink-common/pkg/types/core" @@ -97,7 +97,7 @@ func (s *registrySyncer) Start(ctx context.Context) error { return err } // NOTE: temporary hard-coded capabilities - capId := "mercury-trigger" + capId := "streams-trigger" triggerInfo := capabilities.CapabilityInfo{ ID: capId, CapabilityType: capabilities.CapabilityTypeTrigger, @@ -233,7 +233,7 @@ func (m *mockMercuryDataProducer) loop() { prices[i].Add(prices[i], big.NewInt(1)) } - reports := []mercury.FeedReport{ + reports := []datastreams.FeedReport{ { FeedID: "0x1111111111111111111100000000000000000000000000000000000000000000", FullReport: []byte{0x11, 0xaa, 0xbb, 0xcc}, diff --git a/core/chains/evm/forwarders/forwarder_manager.go b/core/chains/evm/forwarders/forwarder_manager.go index 68015229307..9505cdfbbbf 100644 --- a/core/chains/evm/forwarders/forwarder_manager.go +++ b/core/chains/evm/forwarders/forwarder_manager.go @@ -2,6 +2,7 @@ package forwarders import ( "context" + "slices" "sync" "time" @@ -9,6 +10,7 @@ import ( "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core/types" pkgerrors "github.com/pkg/errors" + "github.com/smartcontractkit/libocr/gethwrappers2/ocr2aggregator" "github.com/smartcontractkit/chainlink-common/pkg/logger" "github.com/smartcontractkit/chainlink-common/pkg/services" @@ -131,6 +133,42 @@ func (f *FwdMgr) ForwarderFor(addr common.Address) (forwarder common.Address, er return common.Address{}, pkgerrors.Errorf("Cannot find forwarder for given EOA") } +func (f *FwdMgr) ForwarderForOCR2Feeds(eoa, ocr2Aggregator common.Address) (forwarder common.Address, err error) { + fwdrs, err := f.ORM.FindForwardersByChain(f.ctx, big.Big(*f.evmClient.ConfiguredChainID())) + if err != nil { + return common.Address{}, err + } + + offchainAggregator, err := ocr2aggregator.NewOCR2Aggregator(ocr2Aggregator, f.evmClient) + if err != nil { + return common.Address{}, err + } + + transmitters, err := offchainAggregator.GetTransmitters(&bind.CallOpts{Context: f.ctx}) + if err != nil { + return common.Address{}, pkgerrors.Errorf("failed to get ocr2 aggregator transmitters: %s", err.Error()) + } + + for _, fwdr := range fwdrs { + if !slices.Contains(transmitters, fwdr.Address) { + f.logger.Criticalw("Forwarder is not set as a transmitter", "forwarder", fwdr.Address, "ocr2Aggregator", ocr2Aggregator, "err", err) + continue + } + + eoas, err := f.getContractSenders(fwdr.Address) + if err != nil { + f.logger.Errorw("Failed to get forwarder senders", "forwarder", fwdr.Address, "err", err) + continue + } + for _, addr := range eoas { + if addr == eoa { + return fwdr.Address, nil + } + } + } + return common.Address{}, pkgerrors.Errorf("Cannot find forwarder for given EOA") +} + func (f *FwdMgr) ConvertPayload(dest common.Address, origPayload []byte) ([]byte, error) { databytes, err := f.getForwardedPayload(dest, origPayload) if err != nil { diff --git a/core/chains/evm/forwarders/forwarder_manager_test.go b/core/chains/evm/forwarders/forwarder_manager_test.go index 3a515e7ab39..993efacac4a 100644 --- a/core/chains/evm/forwarders/forwarder_manager_test.go +++ b/core/chains/evm/forwarders/forwarder_manager_test.go @@ -2,19 +2,23 @@ package forwarders_test import ( "math/big" + "slices" "testing" "time" - "github.com/smartcontractkit/chainlink-common/pkg/sqlutil" - + "github.com/ethereum/go-ethereum/accounts/abi/bind" "github.com/ethereum/go-ethereum/accounts/abi/bind/backends" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + "github.com/smartcontractkit/libocr/gethwrappers2/testocr2aggregator" + "github.com/smartcontractkit/chainlink-common/pkg/logger" + "github.com/smartcontractkit/chainlink-common/pkg/sqlutil" "github.com/smartcontractkit/chainlink-common/pkg/utils" + "github.com/smartcontractkit/chainlink/v2/core/services/ocr2/testhelpers" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/client" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/forwarders" @@ -150,3 +154,105 @@ func TestFwdMgr_AccountUnauthorizedToForward_SkipsForwarding(t *testing.T) { err = fwdMgr.Close() require.NoError(t, err) } + +func TestFwdMgr_InvalidForwarderForOCR2FeedsStates(t *testing.T) { + lggr := logger.Test(t) + db := pgtest.NewSqlxDB(t) + ctx := testutils.Context(t) + cfg := configtest.NewTestGeneralConfig(t) + evmcfg := evmtest.NewChainScopedConfig(t, cfg) + owner := testutils.MustNewSimTransactor(t) + ec := backends.NewSimulatedBackend(map[common.Address]core.GenesisAccount{ + owner.From: { + Balance: big.NewInt(0).Mul(big.NewInt(10), big.NewInt(1e18)), + }, + }, 10e6) + t.Cleanup(func() { ec.Close() }) + linkAddr := common.HexToAddress("0x01BE23585060835E02B77ef475b0Cc51aA1e0709") + operatorAddr, _, _, err := operator_wrapper.DeployOperator(owner, ec, linkAddr, owner.From) + require.NoError(t, err) + + forwarderAddr, _, forwarder, err := authorized_forwarder.DeployAuthorizedForwarder(owner, ec, linkAddr, owner.From, operatorAddr, []byte{}) + require.NoError(t, err) + ec.Commit() + + accessAddress, _, _, err := testocr2aggregator.DeploySimpleWriteAccessController(owner, ec) + require.NoError(t, err, "failed to deploy test access controller contract") + ocr2Address, _, ocr2, err := testocr2aggregator.DeployOCR2Aggregator( + owner, + ec, + linkAddr, + big.NewInt(0), + big.NewInt(10), + accessAddress, + accessAddress, + 9, + "TEST", + ) + require.NoError(t, err, "failed to deploy ocr2 test aggregator") + ec.Commit() + + evmClient := client.NewSimulatedBackendClient(t, ec, testutils.FixtureChainID) + lpOpts := logpoller.Opts{ + PollPeriod: 100 * time.Millisecond, + FinalityDepth: 2, + BackfillBatchSize: 3, + RpcBatchSize: 2, + KeepFinalizedBlocksDepth: 1000, + } + lp := logpoller.NewLogPoller(logpoller.NewORM(testutils.FixtureChainID, db, lggr), evmClient, lggr, lpOpts) + fwdMgr := forwarders.NewFwdMgr(db, evmClient, lp, lggr, evmcfg.EVM()) + fwdMgr.ORM = forwarders.NewORM(db) + + _, err = fwdMgr.ORM.CreateForwarder(ctx, forwarderAddr, ubig.Big(*testutils.FixtureChainID)) + require.NoError(t, err) + lst, err := fwdMgr.ORM.FindForwardersByChain(ctx, ubig.Big(*testutils.FixtureChainID)) + require.NoError(t, err) + require.Equal(t, len(lst), 1) + require.Equal(t, lst[0].Address, forwarderAddr) + + fwdMgr = forwarders.NewFwdMgr(db, evmClient, lp, lggr, evmcfg.EVM()) + require.NoError(t, fwdMgr.Start(testutils.Context(t))) + // cannot find forwarder because it isn't authorized nor added as a transmitter + addr, err := fwdMgr.ForwarderForOCR2Feeds(owner.From, ocr2Address) + require.ErrorContains(t, err, "Cannot find forwarder for given EOA") + require.True(t, utils.IsZero(addr)) + + _, err = forwarder.SetAuthorizedSenders(owner, []common.Address{owner.From}) + require.NoError(t, err) + ec.Commit() + + authorizedSenders, err := forwarder.GetAuthorizedSenders(&bind.CallOpts{Context: ctx}) + require.NoError(t, err) + require.Equal(t, owner.From, authorizedSenders[0]) + + // cannot find forwarder because it isn't added as a transmitter + addr, err = fwdMgr.ForwarderForOCR2Feeds(owner.From, ocr2Address) + require.ErrorContains(t, err, "Cannot find forwarder for given EOA") + require.True(t, utils.IsZero(addr)) + + onchainConfig, err := testhelpers.GenerateDefaultOCR2OnchainConfig(big.NewInt(0), big.NewInt(10)) + require.NoError(t, err) + + _, err = ocr2.SetConfig(owner, + []common.Address{testutils.NewAddress(), testutils.NewAddress(), testutils.NewAddress(), testutils.NewAddress()}, + []common.Address{forwarderAddr, testutils.NewAddress(), testutils.NewAddress(), testutils.NewAddress()}, + 1, + onchainConfig, + 0, + []byte{}) + require.NoError(t, err) + ec.Commit() + + transmitters, err := ocr2.GetTransmitters(&bind.CallOpts{Context: ctx}) + require.NoError(t, err) + require.True(t, slices.Contains(transmitters, forwarderAddr)) + + // create new fwd to have an empty cache that has to fetch authorized forwarders from log poller + fwdMgr = forwarders.NewFwdMgr(db, evmClient, lp, lggr, evmcfg.EVM()) + require.NoError(t, fwdMgr.Start(testutils.Context(t))) + addr, err = fwdMgr.ForwarderForOCR2Feeds(owner.From, ocr2Address) + require.NoError(t, err, "forwarder should be valid and found because it is both authorized and set as a transmitter") + require.Equal(t, forwarderAddr, addr) + require.NoError(t, fwdMgr.Close()) +} diff --git a/core/chains/evm/headtracker/head_tracker_test.go b/core/chains/evm/headtracker/head_tracker_test.go index b8bdb1f5703..bf2b984b548 100644 --- a/core/chains/evm/headtracker/head_tracker_test.go +++ b/core/chains/evm/headtracker/head_tracker_test.go @@ -31,6 +31,7 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/utils/mailbox/mailboxtest" htmocks "github.com/smartcontractkit/chainlink/v2/common/headtracker/mocks" + evmclient "github.com/smartcontractkit/chainlink/v2/core/chains/evm/client" evmclimocks "github.com/smartcontractkit/chainlink/v2/core/chains/evm/client/mocks" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/headtracker" httypes "github.com/smartcontractkit/chainlink/v2/core/chains/evm/headtracker/types" @@ -983,7 +984,46 @@ func TestHeadTracker_Backfill(t *testing.T) { }) } -func createHeadTracker(t *testing.T, ethClient *evmclimocks.Client, config headtracker.Config, htConfig headtracker.HeadTrackerConfig, orm headtracker.ORM) *headTrackerUniverse { +// BenchmarkHeadTracker_Backfill - benchmarks HeadTracker's Backfill with focus on efficiency after initial +// backfill on start up +func BenchmarkHeadTracker_Backfill(b *testing.B) { + cfg := configtest.NewGeneralConfig(b, nil) + + evmcfg := evmtest.NewChainScopedConfig(b, cfg) + db := pgtest.NewSqlxDB(b) + chainID := big.NewInt(evmclient.NullClientChainID) + orm := headtracker.NewORM(*chainID, db) + ethClient := evmclimocks.NewClient(b) + ethClient.On("ConfiguredChainID").Return(chainID) + ht := createHeadTracker(b, ethClient, evmcfg.EVM(), evmcfg.EVM().HeadTracker(), orm) + ctx := tests.Context(b) + makeHash := func(n int64) gethCommon.Hash { + return gethCommon.BigToHash(big.NewInt(n)) + } + const finalityDepth = 12000 // observed value on Arbitrum + makeBlock := func(n int64) *evmtypes.Head { + return &evmtypes.Head{Number: n, Hash: makeHash(n), ParentHash: makeHash(n - 1)} + } + latest := makeBlock(finalityDepth) + finalized := makeBlock(1) + ethClient.On("HeadByHash", mock.Anything, mock.Anything).Return(func(_ context.Context, hash gethCommon.Hash) (*evmtypes.Head, error) { + number := hash.Big().Int64() + return makeBlock(number), nil + }) + // run initial backfill to populate the database + err := ht.headTracker.Backfill(ctx, latest, finalized) + require.NoError(b, err) + b.ResetTimer() + // focus benchmark on processing of a new latest block + for i := 0; i < b.N; i++ { + latest = makeBlock(int64(finalityDepth + i)) + finalized = makeBlock(int64(i + 1)) + err := ht.headTracker.Backfill(ctx, latest, finalized) + require.NoError(b, err) + } +} + +func createHeadTracker(t testing.TB, ethClient *evmclimocks.Client, config headtracker.Config, htConfig headtracker.HeadTrackerConfig, orm headtracker.ORM) *headTrackerUniverse { lggr, ob := logger.TestObserved(t, zap.DebugLevel) hb := headtracker.NewHeadBroadcaster(lggr) hs := headtracker.NewHeadSaver(lggr, orm, config, htConfig) diff --git a/core/chains/evm/headtracker/heads.go b/core/chains/evm/headtracker/heads.go index 1edfb3e3788..a61e55dcd28 100644 --- a/core/chains/evm/headtracker/heads.go +++ b/core/chains/evm/headtracker/heads.go @@ -26,8 +26,9 @@ type Heads interface { } type heads struct { - heads []*evmtypes.Head - mu sync.RWMutex + heads []*evmtypes.Head + headsMap map[common.Hash]*evmtypes.Head + mu sync.RWMutex } func NewHeads() Heads { @@ -48,12 +49,11 @@ func (h *heads) HeadByHash(hash common.Hash) *evmtypes.Head { h.mu.RLock() defer h.mu.RUnlock() - for _, head := range h.heads { - if head.Hash == hash { - return head - } + if h.headsMap == nil { + return nil } - return nil + + return h.headsMap[hash] } func (h *heads) Count() int { @@ -74,26 +74,23 @@ func (h *heads) MarkFinalized(finalized common.Hash, minBlockToKeep int64) bool } // deep copy to avoid race on head.Parent - h.heads = deepCopy(h.heads, minBlockToKeep) + h.heads, h.headsMap = deepCopy(h.heads, minBlockToKeep) - head := h.heads[0] - foundFinalized := false - for head != nil { - if head.Hash == finalized { - foundFinalized = true - } - - // we might see finalized to move back in chain due to request to lagging RPC, - // we should not override the flag in such cases - head.IsFinalized = head.IsFinalized || foundFinalized - head = head.Parent + finalizedHead, ok := h.headsMap[finalized] + if !ok { + return false + } + for finalizedHead != nil { + finalizedHead.IsFinalized = true + finalizedHead = finalizedHead.Parent } - return foundFinalized + return true } -func deepCopy(oldHeads []*evmtypes.Head, minBlockToKeep int64) []*evmtypes.Head { +func deepCopy(oldHeads []*evmtypes.Head, minBlockToKeep int64) ([]*evmtypes.Head, map[common.Hash]*evmtypes.Head) { headsMap := make(map[common.Hash]*evmtypes.Head, len(oldHeads)) + heads := make([]*evmtypes.Head, 0, len(headsMap)) for _, head := range oldHeads { if head.Hash == head.ParentHash { // shouldn't happen but it is untrusted input @@ -111,18 +108,11 @@ func deepCopy(oldHeads []*evmtypes.Head, minBlockToKeep int64) []*evmtypes.Head // prefer head that was already in heads as it might have been marked as finalized on previous run if _, ok := headsMap[head.Hash]; !ok { headsMap[head.Hash] = &headCopy + heads = append(heads, &headCopy) } } - heads := make([]*evmtypes.Head, 0, len(headsMap)) - // unsorted unique heads - { - for _, head := range headsMap { - heads = append(heads, head) - } - } - - // sort the heads + // sort the heads as original slice might be out of order sort.SliceStable(heads, func(i, j int) bool { // sorting from the highest number to lowest return heads[i].Number > heads[j].Number @@ -137,7 +127,7 @@ func deepCopy(oldHeads []*evmtypes.Head, minBlockToKeep int64) []*evmtypes.Head } } - return heads + return heads, headsMap } func (h *heads) AddHeads(newHeads ...*evmtypes.Head) { @@ -145,5 +135,5 @@ func (h *heads) AddHeads(newHeads ...*evmtypes.Head) { defer h.mu.Unlock() // deep copy to avoid race on head.Parent - h.heads = deepCopy(append(h.heads, newHeads...), 0) + h.heads, h.headsMap = deepCopy(append(h.heads, newHeads...), 0) } diff --git a/core/scripts/go.mod b/core/scripts/go.mod index bc59f086ca9..0595d5f065c 100644 --- a/core/scripts/go.mod +++ b/core/scripts/go.mod @@ -21,7 +21,7 @@ require ( github.com/prometheus/client_golang v1.17.0 github.com/shopspring/decimal v1.3.1 github.com/smartcontractkit/chainlink-automation v1.0.3 - github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 + github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 github.com/smartcontractkit/chainlink-vrf v0.0.0-20240222010609-cd67d123c772 github.com/smartcontractkit/chainlink/v2 v2.0.0-00010101000000-000000000000 github.com/smartcontractkit/libocr v0.0.0-20240419185742-fd3cab206b2c diff --git a/core/scripts/go.sum b/core/scripts/go.sum index fee9295d617..e15c02cd9be 100644 --- a/core/scripts/go.sum +++ b/core/scripts/go.sum @@ -1185,8 +1185,8 @@ github.com/smartcontractkit/chain-selectors v1.0.10 h1:t9kJeE6B6G+hKD0GYR4kGJSCq github.com/smartcontractkit/chain-selectors v1.0.10/go.mod h1:d4Hi+E1zqjy9HqMkjBE5q1vcG9VGgxf5VxiRHfzi2kE= github.com/smartcontractkit/chainlink-automation v1.0.3 h1:h/ijT0NiyV06VxYVgcNfsE3+8OEzT3Q0Z9au0z1BPWs= github.com/smartcontractkit/chainlink-automation v1.0.3/go.mod h1:RjboV0Qd7YP+To+OrzHGXaxUxoSONveCoAK2TQ1INLU= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 h1:IwJKWZHPBJbbh4oI3BGX8VNT3c/ChNiPZ/XI4iq6c0E= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10/go.mod h1:sj0pjL+METqeYL9ibp0T8SXquymlaQsofa6bdfLgXX8= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 h1:Pr8/CdiTNnzRwpYc2z7NpHYbw3Dpl1eqiqt9/J/Bcqc= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28/go.mod h1:s+68EchlrXqHKRW3JJgZLEARvzMSKRI5+cE5Zx7pVJA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69 h1:Sec/GpBpUVaTEax1kSHlTvkzF/+d3w5roAQXaj5+SLA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69/go.mod h1:ZQKf+0OLzCLYIisH/OdOIQuFRI6bDuw+jPBTATyHfFM= github.com/smartcontractkit/chainlink-data-streams v0.0.0-20240220203239-09be0ea34540 h1:xFSv8561jsLtF6gYZr/zW2z5qUUAkcFkApin2mnbYTo= diff --git a/core/services/ocr2/delegate.go b/core/services/ocr2/delegate.go index 74e7f9e496f..8ea43582126 100644 --- a/core/services/ocr2/delegate.go +++ b/core/services/ocr2/delegate.go @@ -496,14 +496,22 @@ func GetEVMEffectiveTransmitterID(jb *job.Job, chain legacyevm.Chain, lggr logge if chain == nil { return "", fmt.Errorf("job forwarding requires non-nil chain") } - effectiveTransmitterID, err := chain.TxManager().GetForwarderForEOA(common.HexToAddress(spec.TransmitterID.String)) + + var err error + var effectiveTransmitterID common.Address + // Median forwarders need special handling because of OCR2Aggregator transmitters whitelist. + if spec.PluginType == types.Median { + effectiveTransmitterID, err = chain.TxManager().GetForwarderForEOAOCR2Feeds(common.HexToAddress(spec.TransmitterID.String), common.HexToAddress(spec.ContractID)) + } else { + effectiveTransmitterID, err = chain.TxManager().GetForwarderForEOA(common.HexToAddress(spec.TransmitterID.String)) + } + if err == nil { return effectiveTransmitterID.String(), nil } else if !spec.TransmitterID.Valid { return "", errors.New("failed to get forwarder address and transmitterID is not set") } lggr.Warnw("Skipping forwarding for job, will fallback to default behavior", "job", jb.Name, "err", err) - // this shouldn't happen unless behaviour above was changed } return spec.TransmitterID.String, nil diff --git a/core/services/ocr2/delegate_test.go b/core/services/ocr2/delegate_test.go index 720ad308348..bc5c2df2bbe 100644 --- a/core/services/ocr2/delegate_test.go +++ b/core/services/ocr2/delegate_test.go @@ -67,10 +67,17 @@ func TestGetEVMEffectiveTransmitterID(t *testing.T) { jb.OCR2OracleSpec.RelayConfig["sendingKeys"] = tc.sendingKeys jb.ForwardingAllowed = tc.forwardingEnabled + args := []interface{}{tc.getForwarderForEOAArg} + getForwarderMethodName := "GetForwarderForEOA" + if tc.pluginType == types.Median { + getForwarderMethodName = "GetForwarderForEOAOCR2Feeds" + args = append(args, common.HexToAddress(jb.OCR2OracleSpec.ContractID)) + } + if tc.forwardingEnabled && tc.getForwarderForEOAErr { - txManager.Mock.On("GetForwarderForEOA", tc.getForwarderForEOAArg).Return(common.HexToAddress("0x0"), errors.New("random error")).Once() + txManager.Mock.On(getForwarderMethodName, args...).Return(common.HexToAddress("0x0"), errors.New("random error")).Once() } else if tc.forwardingEnabled { - txManager.Mock.On("GetForwarderForEOA", tc.getForwarderForEOAArg).Return(common.HexToAddress(tc.expectedTransmitterID), nil).Once() + txManager.Mock.On(getForwarderMethodName, args...).Return(common.HexToAddress(tc.expectedTransmitterID), nil).Once() } } diff --git a/core/services/pipeline/task.divide_test.go b/core/services/pipeline/task.divide_test.go index 111087d2e3c..3c2f57c7a07 100644 --- a/core/services/pipeline/task.divide_test.go +++ b/core/services/pipeline/task.divide_test.go @@ -198,6 +198,7 @@ func TestDivideTask_Overflow(t *testing.T) { } func TestDivide_Example(t *testing.T) { + testutils.SkipFlakey(t, "BCF-3236") t.Parallel() dag := ` diff --git a/core/services/relay/evm/mercury/transmitter.go b/core/services/relay/evm/mercury/transmitter.go index 2d54b4b1f17..8ff90021a72 100644 --- a/core/services/relay/evm/mercury/transmitter.go +++ b/core/services/relay/evm/mercury/transmitter.go @@ -24,7 +24,7 @@ import ( "github.com/smartcontractkit/libocr/offchainreporting2plus/chains/evmutil" ocrtypes "github.com/smartcontractkit/libocr/offchainreporting2plus/types" - capMercury "github.com/smartcontractkit/chainlink-common/pkg/capabilities/mercury" + capStreams "github.com/smartcontractkit/chainlink-common/pkg/capabilities/datastreams" "github.com/smartcontractkit/chainlink-common/pkg/capabilities/triggers" commonconfig "github.com/smartcontractkit/chainlink-common/pkg/config" "github.com/smartcontractkit/chainlink-common/pkg/services" @@ -388,27 +388,28 @@ func (mt *mercuryTransmitter) HealthReport() map[string]error { return report } -func (mt *mercuryTransmitter) sendToTrigger(report ocrtypes.Report, rs [][32]byte, ss [][32]byte, vs [32]byte) error { - var rsUnsized [][]byte - var ssUnsized [][]byte - for idx := range rs { - rsUnsized = append(rsUnsized, rs[idx][:]) - ssUnsized = append(ssUnsized, ss[idx][:]) +func (mt *mercuryTransmitter) sendToTrigger(report ocrtypes.Report, signatures []ocrtypes.AttributedOnchainSignature) error { + rawSignatures := [][]byte{} + for _, sig := range signatures { + rawSignatures = append(rawSignatures, sig.Signature) } - converted := capMercury.FeedReport{ + converted := capStreams.FeedReport{ FeedID: mt.feedID.Hex(), FullReport: report, - Rs: rsUnsized, - Ss: ssUnsized, - Vs: vs[:], + Signatures: rawSignatures, // NOTE: Skipping fields derived from FullReport, they will be filled out at a later stage // after decoding and validating signatures. } - return mt.triggerCapability.ProcessReport([]capMercury.FeedReport{converted}) + return mt.triggerCapability.ProcessReport([]capStreams.FeedReport{converted}) } // Transmit sends the report to the on-chain smart contract's Transmit method. func (mt *mercuryTransmitter) Transmit(ctx context.Context, reportCtx ocrtypes.ReportContext, report ocrtypes.Report, signatures []ocrtypes.AttributedOnchainSignature) error { + if mt.triggerCapability != nil { + // Acting as a Capability - send report to trigger service and exit. + return mt.sendToTrigger(report, signatures) + } + var rs [][32]byte var ss [][32]byte var vs [32]byte @@ -423,11 +424,6 @@ func (mt *mercuryTransmitter) Transmit(ctx context.Context, reportCtx ocrtypes.R } rawReportCtx := evmutil.RawReportContext(reportCtx) - if mt.triggerCapability != nil { - // Acting as a Capability - send report to trigger service and exit. - return mt.sendToTrigger(report, rs, ss, vs) - } - payload, err := PayloadTypes.Pack(rawReportCtx, []byte(report), rs, ss, vs) if err != nil { return pkgerrors.Wrap(err, "abi.Pack failed") diff --git a/core/services/workflows/engine.go b/core/services/workflows/engine.go index 2b497057ada..447339b5e7f 100644 --- a/core/services/workflows/engine.go +++ b/core/services/workflows/engine.go @@ -14,6 +14,7 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/services" "github.com/smartcontractkit/chainlink-common/pkg/types/core" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" "github.com/smartcontractkit/chainlink/v2/core/logger" p2ptypes "github.com/smartcontractkit/chainlink/v2/core/services/p2p/types" "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" @@ -29,6 +30,11 @@ type donInfo struct { PeerID func() *p2ptypes.PeerID } +type stepRequest struct { + stepRef string + state store.WorkflowExecution +} + // Engine handles the lifecycle of a single workflow and its executions. type Engine struct { services.StateMachine @@ -103,11 +109,11 @@ func (e *Engine) resolveWorkflowCapabilities(ctx context.Context) error { // - fetching the capability // - register the capability to this workflow // - initializing the step's executionStrategy - capabilityRegistrationErr := e.workflow.walkDo(keywordTrigger, func(s *step) error { + capabilityRegistrationErr := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { // The graph contains a dummy step for triggers, but // we handle triggers separately since there might be more than one // trigger registered to a workflow. - if s.Ref == keywordTrigger { + if s.Ref == workflows.KeywordTrigger { return nil } @@ -441,13 +447,13 @@ func (e *Engine) startExecution(ctx context.Context, executionID string, event v e.logger.Debugw("executing on a trigger event", "event", event, "executionID", executionID) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: event, }, Status: store.StatusCompleted, ExecutionID: executionID, - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: e.workflow.id, @@ -463,7 +469,7 @@ func (e *Engine) startExecution(ctx context.Context, executionID string, event v // Find the tasks we need to fire when a trigger has fired and enqueue them. // This consists of a) nodes without a dependency and b) nodes which depend // on a trigger - triggerDependents, err := e.workflow.dependents(keywordTrigger) + triggerDependents, err := e.workflow.dependents(workflows.KeywordTrigger) if err != nil { return err } @@ -492,7 +498,7 @@ func (e *Engine) handleStepUpdate(ctx context.Context, stepUpdate store.Workflow // we've completed the workflow. if len(stepDependents) == 0 { workflowCompleted := true - err := e.workflow.walkDo(keywordTrigger, func(s *step) error { + err := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { step, ok := state.Steps[s.Ref] // The step is missing from the state, // which means it hasn't been processed yet. @@ -543,7 +549,7 @@ func (e *Engine) handleStepUpdate(ctx context.Context, stepUpdate store.Workflow func (e *Engine) queueIfReady(state store.WorkflowExecution, step *step) { // Check if all dependencies are completed for the current step var waitingOnDependencies bool - for _, dr := range step.dependencies { + for _, dr := range step.Vertex.Dependencies { stepState, ok := state.Steps[dr] if !ok { waitingOnDependencies = true @@ -701,8 +707,8 @@ func (e *Engine) Close() error { close(e.stopCh) e.wg.Wait() - err := e.workflow.walkDo(keywordTrigger, func(s *step) error { - if s.Ref == keywordTrigger { + err := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { + if s.Ref == workflows.KeywordTrigger { return nil } diff --git a/core/services/workflows/engine_test.go b/core/services/workflows/engine_test.go index 1ad7a3c2ae2..6abd241e66c 100644 --- a/core/services/workflows/engine_test.go +++ b/core/services/workflows/engine_test.go @@ -13,6 +13,7 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/capabilities" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" coreCap "github.com/smartcontractkit/chainlink/v2/core/capabilities" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils/pgtest" @@ -547,13 +548,13 @@ func TestEngine_ResumesPendingExecutions(t *testing.T) { dbstore := store.NewDBStore(pgtest.NewSqlxDB(t), clockwork.NewFakeClock()) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: resp, }, Status: store.StatusCompleted, ExecutionID: "", - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: "", @@ -602,13 +603,13 @@ func TestEngine_TimesOutOldExecutions(t *testing.T) { dbstore := store.NewDBStore(pgtest.NewSqlxDB(t), clock) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: resp, }, Status: store.StatusCompleted, ExecutionID: "", - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: "", diff --git a/core/services/workflows/models.go b/core/services/workflows/models.go index dadadc8ba0e..ac157e04b40 100644 --- a/core/services/workflows/models.go +++ b/core/services/workflows/models.go @@ -1,49 +1,15 @@ package workflows import ( - "errors" "fmt" "github.com/dominikbraun/graph" "github.com/smartcontractkit/chainlink-common/pkg/capabilities" "github.com/smartcontractkit/chainlink-common/pkg/values" - "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) -type stepRequest struct { - stepRef string - state store.WorkflowExecution -} - -// StepDefinition is the parsed representation of a step in a workflow. -// -// Within the workflow spec, they are called "Capability Properties". -type StepDefinition struct { - ID string `json:"id" jsonschema:"required"` - Ref string `json:"ref,omitempty" jsonschema:"pattern=^[a-z0-9_]+$"` - Inputs map[string]any `json:"inputs,omitempty"` - Config map[string]any `json:"config" jsonschema:"required"` - - CapabilityType capabilities.CapabilityType `json:"-"` -} - -// WorkflowSpec is the parsed representation of a workflow. -type WorkflowSpec struct { - Triggers []StepDefinition `json:"triggers" jsonschema:"required"` - Actions []StepDefinition `json:"actions,omitempty"` - Consensus []StepDefinition `json:"consensus" jsonschema:"required"` - Targets []StepDefinition `json:"targets" jsonschema:"required"` -} - -func (w *WorkflowSpec) Steps() []StepDefinition { - s := []StepDefinition{} - s = append(s, w.Actions...) - s = append(s, w.Consensus...) - s = append(s, w.Targets...) - return s -} - // workflow is a directed graph of nodes, where each node is a step. // // triggers are special steps that are stored separately, they're @@ -55,7 +21,7 @@ type workflow struct { triggers []*triggerCapability - spec *WorkflowSpec + spec *workflows.WorkflowSpec } func (w *workflow) walkDo(start string, do func(s *step) error) error { @@ -108,144 +74,29 @@ func (w *workflow) dependents(start string) ([]*step, error) { // step wraps a Vertex with additional context for execution that is mutated by the engine type step struct { - Vertex + workflows.Vertex capability capabilities.CallbackCapability config *values.Map executionStrategy executionStrategy } -type Vertex struct { - StepDefinition - dependencies []string -} - -// DependencyGraph is an intermediate representation of a workflow wherein all the graph -// vertices are represented and validated. It is a static representation of the workflow dependencies. -type DependencyGraph struct { - ID string - graph.Graph[string, *Vertex] - - Triggers []*StepDefinition - - Spec *WorkflowSpec -} - -// VID is an identifier for a Vertex that can be used to uniquely identify it in a graph. -// it represents the notion `hash` in the graph package AddVertex method. -// we refrain from naming it `hash` to avoid confusion with the hash function. -func (v *Vertex) VID() string { - return v.Ref -} - type triggerCapability struct { - StepDefinition + workflows.StepDefinition trigger capabilities.TriggerCapability config *values.Map } -const ( - keywordTrigger = "trigger" -) - func Parse(yamlWorkflow string) (*workflow, error) { - wf2, err := ParseDepedencyGraph(yamlWorkflow) + wf2, err := workflows.ParseDependencyGraph(yamlWorkflow) if err != nil { return nil, err } return createWorkflow(wf2) } -func ParseDepedencyGraph(yamlWorkflow string) (*DependencyGraph, error) { - spec, err := ParseWorkflowSpecYaml(yamlWorkflow) - if err != nil { - return nil, err - } - - // Construct and validate the graph. We instantiate an - // empty graph with just one starting entry: `trigger`. - // This provides the starting point for our graph and - // points to all dependent steps. - // Note: all triggers are represented by a single step called - // `trigger`. This is because for workflows with multiple triggers - // only one trigger will have started the workflow. - stepHash := func(s *Vertex) string { - return s.VID() - } - g := graph.New( - stepHash, - graph.PreventCycles(), - graph.Directed(), - ) - err = g.AddVertex(&Vertex{ - StepDefinition: StepDefinition{Ref: keywordTrigger}, - }) - if err != nil { - return nil, err - } - - // Next, let's populate the other entries in the graph. - for _, s := range spec.Steps() { - // TODO: The workflow format spec doesn't always require a `Ref` - // to be provided (triggers and targets don't have a `Ref` for example). - // To handle this, we default the `Ref` to the type, but ideally we - // should find a better long-term way to handle this. - if s.Ref == "" { - s.Ref = s.ID - } - - innerErr := g.AddVertex(&Vertex{StepDefinition: s}) - if innerErr != nil { - return nil, fmt.Errorf("cannot add vertex %s: %w", s.Ref, innerErr) - } - } - - stepRefs, err := g.AdjacencyMap() - if err != nil { - return nil, err - } - - // Next, let's iterate over the steps and populate - // any edges. - for stepRef := range stepRefs { - step, innerErr := g.Vertex(stepRef) - if innerErr != nil { - return nil, innerErr - } - - refs, innerErr := findRefs(step.Inputs) - if innerErr != nil { - return nil, innerErr - } - step.dependencies = refs - - if stepRef != keywordTrigger && len(refs) == 0 { - return nil, errors.New("all non-trigger steps must have a dependent ref") - } - - for _, r := range refs { - innerErr = g.AddEdge(r, step.Ref) - if innerErr != nil { - return nil, innerErr - } - } - } - - triggerSteps := []*StepDefinition{} - for _, t := range spec.Triggers { - tt := t - triggerSteps = append(triggerSteps, &tt) - } - wf := &DependencyGraph{ - Spec: &spec, - Graph: g, - Triggers: triggerSteps, - } - return wf, err -} - // createWorkflow converts a StaticWorkflow to an executable workflow // by adding metadata to the vertices that is owned by the workflow runtime. -func createWorkflow(wf2 *DependencyGraph) (*workflow, error) { +func createWorkflow(wf2 *workflows.DependencyGraph) (*workflow, error) { out := &workflow{ id: wf2.ID, triggers: []*triggerCapability{}, diff --git a/core/services/workflows/models_test.go b/core/services/workflows/models_test.go index 0964b13d277..6bc74ab109a 100644 --- a/core/services/workflows/models_test.go +++ b/core/services/workflows/models_test.go @@ -5,6 +5,8 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) func TestParse_Graph(t *testing.T) { @@ -41,7 +43,7 @@ targets: consensus_output: $(a-consensus.outputs) `, graph: map[string]map[string]struct{}{ - keywordTrigger: { + workflows.KeywordTrigger: { "an-action": struct{}{}, "a-consensus": struct{}{}, }, @@ -175,7 +177,7 @@ targets: consensus_output: $(a-consensus.outputs) `, graph: map[string]map[string]struct{}{ - keywordTrigger: { + workflows.KeywordTrigger: { "an-action": struct{}{}, }, "an-action": { @@ -240,3 +242,13 @@ targets: }) } } + +func TestParsesIntsCorrectly(t *testing.T) { + wf, err := Parse(hardcodedWorkflow) + require.NoError(t, err) + + n, err := wf.Vertex("evm_median") + require.NoError(t, err) + + assert.Equal(t, int64(3600), n.Config["aggregation_config"].(map[string]any)["0x1111111111111111111100000000000000000000000000000000000000000000"].(map[string]any)["heartbeat"]) +} diff --git a/core/services/workflows/models_yaml.go b/core/services/workflows/models_yaml.go deleted file mode 100644 index 90d3f109c06..00000000000 --- a/core/services/workflows/models_yaml.go +++ /dev/null @@ -1,342 +0,0 @@ -package workflows - -import ( - "bytes" - "encoding/json" - "fmt" - "slices" - "strings" - - "github.com/invopop/jsonschema" - "github.com/shopspring/decimal" - "sigs.k8s.io/yaml" - - "github.com/smartcontractkit/chainlink-common/pkg/capabilities" -) - -func GenerateJsonSchema() ([]byte, error) { - schema := jsonschema.Reflect(&workflowSpecYaml{}) - - return json.MarshalIndent(schema, "", " ") -} - -func ParseWorkflowSpecYaml(data string) (WorkflowSpec, error) { - w := workflowSpecYaml{} - err := yaml.Unmarshal([]byte(data), &w) - - return w.toWorkflowSpec(), err -} - -// workflowSpecYaml is the YAML representation of a workflow spec. -// -// It allows for multiple ways of defining a workflow spec, which we later -// convert to a single representation, `workflowSpec`. -type workflowSpecYaml struct { - // Triggers define a starting condition for the workflow, based on specific events or conditions. - Triggers []stepDefinitionYaml `json:"triggers" jsonschema:"required"` - // Actions represent a discrete operation within the workflow, potentially transforming input data. - Actions []stepDefinitionYaml `json:"actions,omitempty"` - // Consensus encapsulates the logic for aggregating and validating the results from various nodes. - Consensus []stepDefinitionYaml `json:"consensus" jsonschema:"required"` - // Targets represents the final step of the workflow, delivering the processed data to a specified location. - Targets []stepDefinitionYaml `json:"targets" jsonschema:"required"` -} - -// toWorkflowSpec converts a workflowSpecYaml to a workflowSpec. -// -// We support multiple ways of defining a workflow spec yaml, -// but internally we want to work with a single representation. -func (w workflowSpecYaml) toWorkflowSpec() WorkflowSpec { - triggers := make([]StepDefinition, 0, len(w.Triggers)) - for _, t := range w.Triggers { - sd := t.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeTrigger - triggers = append(triggers, sd) - } - - actions := make([]StepDefinition, 0, len(w.Actions)) - for _, a := range w.Actions { - sd := a.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeAction - actions = append(actions, sd) - } - - consensus := make([]StepDefinition, 0, len(w.Consensus)) - for _, c := range w.Consensus { - sd := c.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeConsensus - consensus = append(consensus, sd) - } - - targets := make([]StepDefinition, 0, len(w.Targets)) - for _, t := range w.Targets { - sd := t.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeTarget - targets = append(targets, sd) - } - - return WorkflowSpec{ - Triggers: triggers, - Actions: actions, - Consensus: consensus, - Targets: targets, - } -} - -type mapping map[string]any - -func (m *mapping) UnmarshalJSON(b []byte) error { - mp := map[string]any{} - - d := json.NewDecoder(bytes.NewReader(b)) - d.UseNumber() - - err := d.Decode(&mp) - if err != nil { - return err - } - - nm, err := convertNumbers(mp) - if err != nil { - return err - } - - *m = (mapping)(nm) - return err -} - -func convertNumber(el any) (any, error) { - switch elv := el.(type) { - case json.Number: - if strings.Contains(elv.String(), ".") { - f, err := elv.Float64() - if err == nil { - return decimal.NewFromFloat(f), nil - } - } - - return elv.Int64() - default: - return el, nil - } -} - -func convertNumbers(m map[string]any) (map[string]any, error) { - nm := map[string]any{} - for k, v := range m { - switch tv := v.(type) { - case map[string]any: - cm, err := convertNumbers(tv) - if err != nil { - return nil, err - } - - nm[k] = cm - case []any: - na := make([]any, len(tv)) - for i, v := range tv { - cv, err := convertNumber(v) - if err != nil { - return nil, err - } - - na[i] = cv - } - - nm[k] = na - default: - cv, err := convertNumber(v) - if err != nil { - return nil, err - } - - nm[k] = cv - } - } - - return nm, nil -} - -func (m mapping) MarshalJSON() ([]byte, error) { - return json.Marshal(map[string]any(m)) -} - -// stepDefinitionYaml is the YAML representation of a step in a workflow. -// -// It allows for multiple ways of defining a step, which we later -// convert to a single representation, `stepDefinition`. -type stepDefinitionYaml struct { - // A universally unique name for a capability will be defined under the “id” property. The uniqueness will, eventually, be enforced in the Capability Registry. - // - // Semver must be used to specify the version of the Capability at the end of the id field. Capability versions must be immutable. - // - // Initially, we will require major versions. This will ease upgrades early on while we develop the infrastructure. - // - // Eventually, we might support minor version and specific version pins. This will allow workflow authors to have flexibility when selecting the version, and node operators will be able to determine when they should update their capabilities. - // - // There are two ways to specify an id - using a string as a fully qualified ID or a structured table. When using a table, labels are ordered alphanumerically and joined into a string following a - // {name}:{label1_key}_{label1_value}:{label2_key}_{label2_value}@{version} - // pattern. - // - // The “id” supports [a-z0-9_-:] characters followed by an @ and [semver regex] at the end. - // - // Validation must throw an error if: - // - // Unsupported characters are used. - // (For Keystone only.) More specific than a major version is specified. - // - // Example (string) - // id: read_chain:chain_ethereum:network_mainnet@1 - // - // Example (table) - // - // id: - // name: read_chain - // version: 1 - // labels: - // chain: ethereum - // network: mainnet - // - // [semver regex]: https://semver.org/#is-there-a-suggested-regular-expression-regex-to-check-a-semver-string - ID stepDefinitionID `json:"id" jsonschema:"required"` - - // Actions and Consensus capabilities have a required “ref” property that must be unique within a Workflow file (not universally) This property enables referencing outputs and is required because Actions and Consensus always need to be referenced in the following phases. Triggers can optionally specify if they need to be referenced. - // - // The “ref” supports [a-z0-9_] characters. - // - // Validation must throw an error if: - // - Unsupported characters are used. - // - The same “ref” appears in the workflow multiple times. - // - “ref” is used on a Target capability. - // - “ref” has a circular reference. - // - // NOTE: Should introduce a custom validator to cover trigger case - Ref string `json:"ref,omitempty" jsonschema:"pattern=^[a-z0-9_-]+$"` - - // Capabilities can specify an additional optional ”inputs” property. It allows specifying a dependency on the result of one or more other capabilities. These are always runtime values that cannot be provided upfront. It takes a map of the argument name internal to the capability and an explicit reference to the values. - // - // References are specified using the [id].[ref].[path_to_value] pattern. - // - // The interpolation of “inputs” is allowed - // - // Validation must throw an error if: - // - Input reference cannot be resolved. - // - Input is defined on triggers - // NOTE: Should introduce a custom validator to cover trigger case - Inputs mapping `json:"inputs,omitempty"` - - // The configuration of a Capability will be done using the “config” property. Each capability is responsible for defining an external interface used during setup. This interface may be unique or identical, meaning multiple Capabilities might use the same configuration properties. - // - // The interpolation of “inputs” - // - // Interpolation of self inputs is allowed from within the “config” property. - // - // Example - // targets: - // - id: write_polygon_mainnet@1 - // inputs: - // report: - // - consensus.evm_median.outputs.report - // config: - // address: "0xaabbcc" - // method: "updateFeedValues(report bytes, role uint8)" - // params: [$(inputs.report), 1] - Config mapping `json:"config" jsonschema:"required"` -} - -// toStepDefinition converts a stepDefinitionYaml to a stepDefinition. -// -// `stepDefinition` is the converged representation of a step in a workflow. -func (s stepDefinitionYaml) toStepDefinition() StepDefinition { - return StepDefinition{ - Ref: s.Ref, - ID: s.ID.String(), - Inputs: s.Inputs, - Config: s.Config, - } -} - -// stepDefinitionID represents both the string and table representations of the "id" field in a stepDefinition. -type stepDefinitionID struct { - idStr string - idTable *stepDefinitionTableID -} - -func (s stepDefinitionID) String() string { - if s.idStr != "" { - return s.idStr - } - - return s.idTable.String() -} - -func (s *stepDefinitionID) UnmarshalJSON(data []byte) error { - // Unmarshal the JSON data into a map to determine if it's a string or a table - var m string - err := json.Unmarshal(data, &m) - if err == nil { - s.idStr = m - return nil - } - - // If the JSON data is a table, unmarshal it into a stepDefinitionTableID - var table stepDefinitionTableID - err = json.Unmarshal(data, &table) - if err != nil { - return err - } - s.idTable = &table - return nil -} - -func (s *stepDefinitionID) MarshalJSON() ([]byte, error) { - if s.idStr != "" { - return json.Marshal(s.idStr) - } - - return json.Marshal(s.idTable) -} - -// JSONSchema returns the JSON schema for a stepDefinitionID. -// -// The schema is a oneOf schema that allows either a string or a table. -func (stepDefinitionID) JSONSchema() *jsonschema.Schema { - reflector := jsonschema.Reflector{DoNotReference: true, ExpandedStruct: true} - tableSchema := reflector.Reflect(&stepDefinitionTableID{}) - stringSchema := &jsonschema.Schema{ - ID: "string", - Pattern: "^[a-z0-9_\\-:]+@(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$", - } - - return &jsonschema.Schema{ - Title: "id", - OneOf: []*jsonschema.Schema{ - stringSchema, - tableSchema, - }, - } -} - -// stepDefinitionTableID is the structured representation of a stepDefinitionID. -type stepDefinitionTableID struct { - Name string `json:"name"` - Version string `json:"version" jsonschema:"pattern=(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$"` - Labels map[string]string `json:"labels"` -} - -// String returns the string representation of a stepDefinitionTableID. -// -// It follows the format: -// -// {name}:{label1_key}_{label1_value}:{label2_key}_{label2_value}@{version} -// -// where labels are ordered alphanumerically. -func (s stepDefinitionTableID) String() string { - labels := make([]string, 0, len(s.Labels)) - for k, v := range s.Labels { - labels = append(labels, fmt.Sprintf("%s_%s", k, v)) - } - slices.Sort(labels) - - return fmt.Sprintf("%s:%s@%s", s.Name, strings.Join(labels, ":"), s.Version) -} diff --git a/core/services/workflows/models_yaml_test.go b/core/services/workflows/models_yaml_test.go deleted file mode 100644 index 5fa326dda5d..00000000000 --- a/core/services/workflows/models_yaml_test.go +++ /dev/null @@ -1,259 +0,0 @@ -package workflows - -import ( - "encoding/json" - "fmt" - "os" - "testing" - - "github.com/google/go-cmp/cmp" - "github.com/santhosh-tekuri/jsonschema/v5" - "github.com/shopspring/decimal" - "github.com/stretchr/testify/assert" - "github.com/stretchr/testify/require" - - "sigs.k8s.io/yaml" -) - -var fixtureDir = "./testdata/fixtures/workflows/" - -// yamlFixtureReaderObj reads a yaml fixture file and returns the parsed object -func yamlFixtureReaderObj(t *testing.T, testCase string) func(name string) any { - testFixtureReader := yamlFixtureReaderBytes(t, testCase) - - return func(name string) any { - testFileBytes := testFixtureReader(name) - - var testFileYaml any - err := yaml.Unmarshal(testFileBytes, &testFileYaml) - require.NoError(t, err) - - return testFileYaml - } -} - -// yamlFixtureReaderBytes reads a yaml fixture file and returns the bytes -func yamlFixtureReaderBytes(t *testing.T, testCase string) func(name string) []byte { - return func(name string) []byte { - testFileBytes, err := os.ReadFile(fmt.Sprintf(fixtureDir+"%s/%s.yaml", testCase, name)) - require.NoError(t, err) - - return testFileBytes - } -} - -var transformJSON = cmp.FilterValues(func(x, y []byte) bool { - return json.Valid(x) && json.Valid(y) -}, cmp.Transformer("ParseJSON", func(in []byte) (out interface{}) { - if err := json.Unmarshal(in, &out); err != nil { - panic(err) // should never occur given previous filter to ensure valid JSON - } - return out -})) - -func TestWorkflowSpecMarshalling(t *testing.T) { - t.Parallel() - fixtureReader := yamlFixtureReaderBytes(t, "marshalling") - - t.Run("Type coercion", func(t *testing.T) { - workflowBytes := fixtureReader("workflow_1") - - spec := workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, &spec) - require.NoError(t, err) - - // Test that our workflowSpec still keeps all of the original data - var rawSpec interface{} - err = yaml.Unmarshal(workflowBytes, &rawSpec) - require.NoError(t, err) - - workflowspecJson, err := json.MarshalIndent(spec, "", " ") - require.NoError(t, err) - rawWorkflowSpecJson, err := json.MarshalIndent(rawSpec, "", " ") - require.NoError(t, err) - - if diff := cmp.Diff(rawWorkflowSpecJson, workflowspecJson, transformJSON); diff != "" { - t.Errorf("ParseWorkflowWorkflowSpecFromString() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - - // Spot check some fields - consensusConfig := spec.Consensus[0].Config - v, ok := consensusConfig["aggregation_config"] - require.True(t, ok, "expected aggregation_config to be present in consensus config") - - // the type of the keys present in v should be string rather than a number - // this is because JSON keys are always strings - _, ok = v.(map[string]any) - require.True(t, ok, "expected map[string]interface{} but got %T", v) - - // Make sure we dont have any weird type coercion with possible boolean values - booleanCoercions, ok := spec.Triggers[0].Config["boolean_coercion"].(map[string]any) - require.True(t, ok, "expected boolean_coercion to be present in triggers config") - - // check bools - bools, ok := booleanCoercions["bools"] - require.True(t, ok, "expected bools to be present in boolean_coercions") - for _, v := range bools.([]interface{}) { - _, ok = v.(bool) - require.True(t, ok, "expected bool but got %T", v) - } - - // check strings - strings, ok := booleanCoercions["strings"] - require.True(t, ok, "expected strings to be present in boolean_coercions") - for _, v := range strings.([]interface{}) { - _, ok = v.(string) - require.True(t, ok, "expected string but got %T", v) - } - - // check numbers - numbers, ok := booleanCoercions["numbers"] - require.True(t, ok, "expected numbers to be present in boolean_coercions") - for _, v := range numbers.([]interface{}) { - _, ok = v.(int64) - require.True(t, ok, "expected int64 but got %T", v) - } - }) - - t.Run("Table and string capability id", func(t *testing.T) { - workflowBytes := fixtureReader("workflow_2") - - spec := workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, &spec) - require.NoError(t, err) - - // Test that our workflowSpec still keeps all of the original data - var rawSpec interface{} - err = yaml.Unmarshal(workflowBytes, &rawSpec) - require.NoError(t, err) - - workflowspecJson, err := json.MarshalIndent(spec, "", " ") - require.NoError(t, err) - rawWorkflowSpecJson, err := json.MarshalIndent(rawSpec, "", " ") - require.NoError(t, err) - - if diff := cmp.Diff(rawWorkflowSpecJson, workflowspecJson, transformJSON); diff != "" { - t.Errorf("ParseWorkflowWorkflowSpecFromString() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) - - t.Run("Yaml spec to spec", func(t *testing.T) { - expectedSpecPath := fixtureDir + "marshalling/" + "workflow_2_spec.json" - workflowBytes := fixtureReader("workflow_2") - - workflowYaml := &workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, workflowYaml) - require.NoError(t, err) - - workflowSpec := workflowYaml.toWorkflowSpec() - workflowSpecBytes, err := json.MarshalIndent(workflowSpec, "", " ") - require.NoError(t, err) - - // change this to update golden file - shouldUpdateWorkflowSpec := false - if shouldUpdateWorkflowSpec { - err = os.WriteFile(expectedSpecPath, workflowSpecBytes, 0600) - require.NoError(t, err) - } - - expectedSpecBytes, err := os.ReadFile(expectedSpecPath) - require.NoError(t, err) - diff := cmp.Diff(expectedSpecBytes, workflowSpecBytes, transformJSON) - if diff != "" { - t.Errorf("WorkflowYamlSpecToWorkflowSpec() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) -} - -func TestJsonSchema(t *testing.T) { - t.Parallel() - t.Run("GenerateJsonSchema", func(t *testing.T) { - expectedSchemaPath := fixtureDir + "workflow_schema.json" - generatedSchema, err := GenerateJsonSchema() - require.NoError(t, err) - - // change this to update golden file - shouldUpdateSchema := false - if shouldUpdateSchema { - err = os.WriteFile(expectedSchemaPath, generatedSchema, 0600) - require.NoError(t, err) - } - - expectedSchema, err := os.ReadFile(expectedSchemaPath) - require.NoError(t, err) - diff := cmp.Diff(expectedSchema, generatedSchema, transformJSON) - if diff != "" { - t.Errorf("GenerateJsonSchema() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) - - t.Run("ValidateJsonSchema", func(t *testing.T) { - generatedSchema, err := GenerateJsonSchema() - require.NoError(t, err) - - // test version regex - // for keystone, we should support major versions only along with prereleases and build metadata - t.Run("version", func(t *testing.T) { - readVersionFixture := yamlFixtureReaderObj(t, "versioning") - failingFixture1 := readVersionFixture("failing_1") - failingFixture2 := readVersionFixture("failing_2") - passingFixture1 := readVersionFixture("passing_1") - jsonSchema, err := jsonschema.CompileString("github.com/smartcontractkit/chainlink", string(generatedSchema)) - require.NoError(t, err) - - err = jsonSchema.Validate(failingFixture1) - require.Error(t, err) - - err = jsonSchema.Validate(failingFixture2) - require.Error(t, err) - - err = jsonSchema.Validate(passingFixture1) - require.NoError(t, err) - }) - - // test ref regex - t.Run("ref", func(t *testing.T) { - readRefFixture := yamlFixtureReaderObj(t, "references") - failingFixture1 := readRefFixture("failing_1") - passingFixture1 := readRefFixture("passing_1") - jsonSchema, err := jsonschema.CompileString("github.com/smartcontractkit/chainlink", string(generatedSchema)) - require.NoError(t, err) - - err = jsonSchema.Validate(failingFixture1) - require.Error(t, err) - - err = jsonSchema.Validate(passingFixture1) - require.NoError(t, err) - }) - }) -} - -func TestParsesIntsCorrectly(t *testing.T) { - wf, err := Parse(hardcodedWorkflow) - require.NoError(t, err) - - n, err := wf.Vertex("evm_median") - require.NoError(t, err) - - assert.Equal(t, int64(3600), n.Config["aggregation_config"].(map[string]any)["0x1111111111111111111100000000000000000000000000000000000000000000"].(map[string]any)["heartbeat"]) -} - -func TestMappingCustomType(t *testing.T) { - m := mapping(map[string]any{}) - data := ` -{ - "foo": 100, - "bar": 100.00, - "baz": { "gnat": 11.10 } -}` - - err := m.UnmarshalJSON([]byte(data)) - require.NoError(t, err) - assert.Equal(t, int64(100), m["foo"], m) - assert.Equal(t, decimal.NewFromFloat(100.00), m["bar"], m) - assert.Equal(t, decimal.NewFromFloat(11.10), m["baz"].(map[string]any)["gnat"], m) -} diff --git a/core/services/workflows/state.go b/core/services/workflows/state.go index 4026a59be0b..218022eae36 100644 --- a/core/services/workflows/state.go +++ b/core/services/workflows/state.go @@ -2,13 +2,13 @@ package workflows import ( "fmt" - "regexp" "strconv" "strings" "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) // copyState returns a deep copy of the input executionState @@ -118,19 +118,15 @@ func interpolateKey(key string, state store.WorkflowExecution) (any, error) { return val, nil } -var ( - interpolationTokenRe = regexp.MustCompile(`^\$\((\S+)\)$`) -) - // findAndInterpolateAllKeys takes an `input` any value, and recursively // identifies any values that should be replaced from `state`. // // A value `v` should be replaced if it is wrapped as follows: `$(v)`. func findAndInterpolateAllKeys(input any, state store.WorkflowExecution) (any, error) { - return deepMap( + return workflows.DeepMap( input, func(el string) (any, error) { - matches := interpolationTokenRe.FindStringSubmatch(el) + matches := workflows.InterpolationTokenRe.FindStringSubmatch(el) if len(matches) < 2 { return el, nil } @@ -140,92 +136,3 @@ func findAndInterpolateAllKeys(input any, state store.WorkflowExecution) (any, e }, ) } - -// findRefs takes an `inputs` map and returns a list of all the step references -// contained within it. -func findRefs(inputs map[string]any) ([]string, error) { - refs := []string{} - _, err := deepMap( - inputs, - // This function is called for each string in the map - // for each string, we iterate over each match of the interpolation token - // - if there are no matches, return no reference - // - if there is one match, return the reference - // - if there are multiple matches (in the case of a multi-part state reference), return just the step ref - func(el string) (any, error) { - matches := interpolationTokenRe.FindStringSubmatch(el) - if len(matches) < 2 { - return el, nil - } - - m := matches[1] - parts := strings.Split(m, ".") - if len(parts) < 1 { - return nil, fmt.Errorf("invalid ref %s", m) - } - - refs = append(refs, parts[0]) - return el, nil - }, - ) - return refs, err -} - -// deepMap recursively applies a transformation function -// over each string within: -// -// - a map[string]any -// - a []any -// - a string -func deepMap(input any, transform func(el string) (any, error)) (any, error) { - // in the case of a string, simply apply the transformation - // in the case of a map, recurse and apply the transformation to each value - // in the case of a list, recurse and apply the transformation to each element - switch tv := input.(type) { - case string: - nv, err := transform(tv) - if err != nil { - return nil, err - } - - return nv, nil - case mapping: - // coerce mapping to map[string]any - mp := map[string]any(tv) - - nm := map[string]any{} - for k, v := range mp { - nv, err := deepMap(v, transform) - if err != nil { - return nil, err - } - - nm[k] = nv - } - return nm, nil - case map[string]any: - nm := map[string]any{} - for k, v := range tv { - nv, err := deepMap(v, transform) - if err != nil { - return nil, err - } - - nm[k] = nv - } - return nm, nil - case []any: - a := []any{} - for _, el := range tv { - ne, err := deepMap(el, transform) - if err != nil { - return nil, err - } - - a = append(a, ne) - } - return a, nil - } - - return nil, fmt.Errorf("cannot traverse item %+v of type %T", input, input) -} diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml deleted file mode 100644 index 9a9870af875..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml +++ /dev/null @@ -1,88 +0,0 @@ - triggers: - - id: mercury-trigger@1 - ref: report_data - config: - boolean_coercion: - bools: - - y - - n - - yes - - no - - Y - - N - - YES - - NO - - No - - Yes - - TRUE - - FALSE - - True - - False - - true - - false - strings: - - TruE - - FalsE - - "true" - - "false" - - "TRUE" - - "FALSE" - - t - - f - - "T" - - "F" - - "t" - - "f" - - "1" - - "0" - - "yes" - - "no" - - "y" - - "n" - - "YES" - - "NO" - - "Y" - - "N" - numbers: - - 1 - - 0 - feed_ids: - - 123 # ETHUSD - - 456 # LINKUSD - - 789 # USDBTC - - # no actions - - consensus: - - id: offchain_reporting@1 - inputs: - observations: - - triggers.report_data.outputs - config: - aggregation_method: data_feeds_2_0 - aggregation_config: - 123: # ETHUSD - deviation: "0.005" - heartbeat: 24h - test: - 456: # LINKUSD - deviation: "0.001" - heartbeat: 24h - 789: # USDBTC - deviation: "0.002" - heartbeat: 6h - encoder: EVM - encoder_config: - abi: "mercury_reports bytes[]" - - targets: - - id: write_polygon_mainnet@1 - inputs: - report: - - consensus.evm_median.outputs.report - config: - address: "0xaabbcc" - method: "updateFeedValues(report bytes, role uint8)" - params: [$(inputs.report), 1] - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml deleted file mode 100644 index be40a91daa0..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml +++ /dev/null @@ -1,28 +0,0 @@ - triggers: - - id: on_mercury_report@1 - ref: report_data - config: {} - - # no actions - - consensus: - - id: - name: trigger_test - version: "2" - labels: - chain: ethereum - aaShouldBeFirst: "true" - network: mainnet - config: {} - inputs: - observations: - - triggers.report_data.outputs - - targets: - - id: write_polygon_mainnet@1 - config: {} - inputs: - report: - - consensus.evm_median.outputs.report - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json deleted file mode 100644 index 000fa469218..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json +++ /dev/null @@ -1,31 +0,0 @@ -{ - "triggers": [ - { - "id": "on_mercury_report@1", - "ref": "report_data", - "config": {} - } - ], - "consensus": [ - { - "id": "trigger_test:aaShouldBeFirst_true:chain_ethereum:network_mainnet@2", - "inputs": { - "observations": [ - "triggers.report_data.outputs" - ] - }, - "config": {} - } - ], - "targets": [ - { - "id": "write_polygon_mainnet@1", - "inputs": { - "report": [ - "consensus.evm_median.outputs.report" - ] - }, - "config": {} - } - ] -} \ No newline at end of file diff --git a/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml deleted file mode 100644 index b3c984e9892..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ -triggers: -- id: trigger_test@1 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting=1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml deleted file mode 100644 index cb2f424e981..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ -triggers: -- id: trigger_test@1 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml deleted file mode 100644 index 2e41eeb9898..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml +++ /dev/null @@ -1,16 +0,0 @@ -# Should fail since version is more specific than major -triggers: - - id: trigger_test@1.0 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml deleted file mode 100644 index 36cd5b68b6b..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml +++ /dev/null @@ -1,17 +0,0 @@ - -# Should fail since version is more specific than major -triggers: - - id: trigger_test@1.0.0 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml deleted file mode 100644 index 4579c2899b9..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ - triggers: - - id: trigger_test@1 - config: {} - - consensus: - - id: offchain_reporting@1-beta.1 - ref: offchain_reporting_1 - config: {} - - targets: - - id: write_polygon_mainnet@1-alpha+sha246er3 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json b/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json deleted file mode 100644 index f9f9fd88646..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json +++ /dev/null @@ -1,103 +0,0 @@ -{ - "$schema": "https://json-schema.org/draft/2020-12/schema", - "$id": "https://github.com/smartcontractkit/chainlink/v2/core/services/workflows/workflow-spec-yaml", - "$ref": "#/$defs/workflowSpecYaml", - "$defs": { - "mapping": { - "type": "object" - }, - "stepDefinitionID": { - "oneOf": [ - { - "$id": "string", - "pattern": "^[a-z0-9_\\-:]+@(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" - }, - { - "$schema": "https://json-schema.org/draft/2020-12/schema", - "$id": "https://github.com/smartcontractkit/chainlink/v2/core/services/workflows/step-definition-table-id", - "properties": { - "name": { - "type": "string" - }, - "version": { - "type": "string", - "pattern": "(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" - }, - "labels": { - "additionalProperties": { - "type": "string" - }, - "type": "object" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "name", - "version", - "labels" - ] - } - ], - "title": "id" - }, - "stepDefinitionYaml": { - "properties": { - "id": { - "$ref": "#/$defs/stepDefinitionID" - }, - "ref": { - "type": "string", - "pattern": "^[a-z0-9_-]+$" - }, - "inputs": { - "$ref": "#/$defs/mapping" - }, - "config": { - "$ref": "#/$defs/mapping" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "id", - "config" - ] - }, - "workflowSpecYaml": { - "properties": { - "triggers": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "actions": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "consensus": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "targets": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "triggers", - "consensus", - "targets" - ] - } - } -} \ No newline at end of file diff --git a/go.mod b/go.mod index 9220ad3e63d..8ba80e67eba 100644 --- a/go.mod +++ b/go.mod @@ -72,7 +72,7 @@ require ( github.com/shopspring/decimal v1.3.1 github.com/smartcontractkit/chain-selectors v1.0.10 github.com/smartcontractkit/chainlink-automation v1.0.3 - github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 + github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69 github.com/smartcontractkit/chainlink-data-streams v0.0.0-20240220203239-09be0ea34540 github.com/smartcontractkit/chainlink-feeds v0.0.0-20240422130241-13c17a91b2ab @@ -112,7 +112,6 @@ require ( google.golang.org/protobuf v1.33.0 gopkg.in/guregu/null.v4 v4.0.0 gopkg.in/natefinch/lumberjack.v2 v2.2.1 - sigs.k8s.io/yaml v1.4.0 ) require ( @@ -210,7 +209,7 @@ require ( github.com/golang/protobuf v1.5.4 // indirect github.com/golang/snappy v0.0.5-0.20220116011046-fa5810519dcb // indirect github.com/google/btree v1.1.2 // indirect - github.com/google/go-cmp v0.6.0 + github.com/google/go-cmp v0.6.0 // indirect github.com/google/go-tpm v0.9.0 // indirect github.com/google/gofuzz v1.2.0 // indirect github.com/gorilla/context v1.1.1 // indirect @@ -237,7 +236,7 @@ require ( github.com/huin/goupnp v1.3.0 // indirect github.com/imdario/mergo v0.3.16 // indirect github.com/inconshreveable/mousetrap v1.1.0 // indirect - github.com/invopop/jsonschema v0.12.0 + github.com/invopop/jsonschema v0.12.0 // indirect github.com/jackc/chunkreader/v2 v2.0.1 // indirect github.com/jackc/pgio v1.0.0 // indirect github.com/jackc/pgpassfile v1.0.0 // indirect @@ -279,7 +278,7 @@ require ( github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 // indirect github.com/rivo/uniseg v0.4.4 // indirect github.com/russross/blackfriday/v2 v2.1.0 // indirect - github.com/santhosh-tekuri/jsonschema/v5 v5.3.1 + github.com/santhosh-tekuri/jsonschema/v5 v5.3.1 // indirect github.com/sasha-s/go-deadlock v0.3.1 // indirect github.com/sethvargo/go-retry v0.2.4 // indirect github.com/shirou/gopsutil v3.21.11+incompatible // indirect @@ -337,6 +336,7 @@ require ( gopkg.in/yaml.v3 v3.0.1 // indirect pgregory.net/rapid v0.5.5 // indirect rsc.io/tmplfunc v0.0.3 // indirect + sigs.k8s.io/yaml v1.4.0 // indirect ) replace ( diff --git a/go.sum b/go.sum index 3d8d24b481b..70edbdfb066 100644 --- a/go.sum +++ b/go.sum @@ -1171,8 +1171,8 @@ github.com/smartcontractkit/chain-selectors v1.0.10 h1:t9kJeE6B6G+hKD0GYR4kGJSCq github.com/smartcontractkit/chain-selectors v1.0.10/go.mod h1:d4Hi+E1zqjy9HqMkjBE5q1vcG9VGgxf5VxiRHfzi2kE= github.com/smartcontractkit/chainlink-automation v1.0.3 h1:h/ijT0NiyV06VxYVgcNfsE3+8OEzT3Q0Z9au0z1BPWs= github.com/smartcontractkit/chainlink-automation v1.0.3/go.mod h1:RjboV0Qd7YP+To+OrzHGXaxUxoSONveCoAK2TQ1INLU= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 h1:IwJKWZHPBJbbh4oI3BGX8VNT3c/ChNiPZ/XI4iq6c0E= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10/go.mod h1:sj0pjL+METqeYL9ibp0T8SXquymlaQsofa6bdfLgXX8= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 h1:Pr8/CdiTNnzRwpYc2z7NpHYbw3Dpl1eqiqt9/J/Bcqc= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28/go.mod h1:s+68EchlrXqHKRW3JJgZLEARvzMSKRI5+cE5Zx7pVJA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69 h1:Sec/GpBpUVaTEax1kSHlTvkzF/+d3w5roAQXaj5+SLA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69/go.mod h1:ZQKf+0OLzCLYIisH/OdOIQuFRI6bDuw+jPBTATyHfFM= github.com/smartcontractkit/chainlink-data-streams v0.0.0-20240220203239-09be0ea34540 h1:xFSv8561jsLtF6gYZr/zW2z5qUUAkcFkApin2mnbYTo= diff --git a/integration-tests/docker/test_env/test_env_builder.go b/integration-tests/docker/test_env/test_env_builder.go index cdb310114ee..ad3b75a3ca4 100644 --- a/integration-tests/docker/test_env/test_env_builder.go +++ b/integration-tests/docker/test_env/test_env_builder.go @@ -23,12 +23,12 @@ import ( "github.com/smartcontractkit/chainlink-testing-framework/networks" "github.com/smartcontractkit/chainlink-testing-framework/testreporters" "github.com/smartcontractkit/chainlink-testing-framework/utils/osutil" - evmcfg "github.com/smartcontractkit/chainlink/v2/core/chains/evm/config/toml" "github.com/smartcontractkit/chainlink/v2/core/services/chainlink" actions_seth "github.com/smartcontractkit/chainlink/integration-tests/actions/seth" "github.com/smartcontractkit/chainlink/integration-tests/contracts" "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" + "github.com/smartcontractkit/chainlink/integration-tests/utils" ) type CleanUpType string @@ -48,7 +48,6 @@ type ChainlinkNodeLogScannerSettings struct { type CLTestEnvBuilder struct { hasLogStream bool hasKillgrave bool - hasForwarders bool hasSeth bool hasEVMClient bool clNodeConfig *chainlink.Config @@ -60,10 +59,9 @@ type CLTestEnvBuilder struct { l zerolog.Logger t *testing.T te *CLClusterTestEnv - isNonEVM bool + isEVM bool cleanUpType CleanUpType cleanUpCustomFn func() - chainOptionsFn []ChainOption evmNetworkOption []EVMNetworkOption privateEthereumNetworks []*ctf_config.EthereumNetworkConfig testConfig ctf_config.GlobalTestConfig @@ -75,6 +73,7 @@ type CLTestEnvBuilder struct { var DefaultAllowedMessages = []testreporters.AllowedLogMessage{ testreporters.NewAllowedLogMessage("Failed to get LINK balance", "Happens only when we deploy LINK token for test purposes. Harmless.", zapcore.ErrorLevel, testreporters.WarnAboutAllowedMsgs_No), + testreporters.NewAllowedLogMessage("Error stopping job service", "It's a known issue with lifecycle. There's ongoing work that will fix it.", zapcore.DPanicLevel, testreporters.WarnAboutAllowedMsgs_No), } var DefaultChainlinkNodeLogScannerSettings = ChainlinkNodeLogScannerSettings{ @@ -97,6 +96,7 @@ func NewCLTestEnvBuilder() *CLTestEnvBuilder { l: log.Logger, hasLogStream: true, hasEVMClient: true, + isEVM: true, chainlinkNodeLogScannerSettings: &DefaultChainlinkNodeLogScannerSettings, } } @@ -170,11 +170,6 @@ func (b *CLTestEnvBuilder) WithCLNodeOptions(opt ...ClNodeOption) *CLTestEnvBuil return b } -func (b *CLTestEnvBuilder) WithForwarders() *CLTestEnvBuilder { - b.hasForwarders = true - return b -} - func (b *CLTestEnvBuilder) WithFunding(eth *big.Float) *CLTestEnvBuilder { b.ETHFunds = eth return b @@ -186,6 +181,12 @@ func (b *CLTestEnvBuilder) WithSeth() *CLTestEnvBuilder { return b } +func (b *CLTestEnvBuilder) WithoutEvmClients() *CLTestEnvBuilder { + b.hasSeth = false + b.hasEVMClient = false + return b +} + func (b *CLTestEnvBuilder) WithPrivateEthereumNetwork(en ctf_config.EthereumNetworkConfig) *CLTestEnvBuilder { b.privateEthereumNetworks = append(b.privateEthereumNetworks, &en) return b @@ -196,6 +197,7 @@ func (b *CLTestEnvBuilder) WithPrivateEthereumNetworks(ens []*ctf_config.Ethereu return b } +// Deprecated: Use TOML instead func (b *CLTestEnvBuilder) WithCLNodeConfig(cfg *chainlink.Config) *CLTestEnvBuilder { b.clNodeConfig = cfg return b @@ -213,7 +215,7 @@ func (b *CLTestEnvBuilder) WithMockAdapter() *CLTestEnvBuilder { // WithNonEVM sets the test environment to not use EVM when built. func (b *CLTestEnvBuilder) WithNonEVM() *CLTestEnvBuilder { - b.isNonEVM = true + b.isEVM = false return b } @@ -233,18 +235,12 @@ func (b *CLTestEnvBuilder) WithCustomCleanup(customFn func()) *CLTestEnvBuilder return b } -type ChainOption = func(*evmcfg.Chain) *evmcfg.Chain - -func (b *CLTestEnvBuilder) WithChainOptions(opts ...ChainOption) *CLTestEnvBuilder { - b.chainOptionsFn = make([]ChainOption, 0) - b.chainOptionsFn = append(b.chainOptionsFn, opts...) - - return b -} - type EVMNetworkOption = func(*blockchain.EVMNetwork) *blockchain.EVMNetwork -func (b *CLTestEnvBuilder) EVMNetworkOptions(opts ...EVMNetworkOption) *CLTestEnvBuilder { +// WithEVMNetworkOptions sets the options for the EVM network. This is especially useful for simulated networks, which +// by usually use default options, so if we want to change any of them before the configuration is passed to evm client +// or Chainlnik node, we can do it here. +func (b *CLTestEnvBuilder) WithEVMNetworkOptions(opts ...EVMNetworkOption) *CLTestEnvBuilder { b.evmNetworkOption = make([]EVMNetworkOption, 0) b.evmNetworkOption = append(b.evmNetworkOption, opts...) @@ -284,7 +280,7 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { } // this clean up has to be added as the FIRST one, because cleanup functions are executed in reverse order (LIFO) - if b.t != nil { + if b.t != nil && b.cleanUpType == CleanUpTypeStandard { b.t.Cleanup(func() { b.l.Info().Msg("Shutting down LogStream") logPath, err := osutil.GetAbsoluteFolderPath("logs") @@ -343,6 +339,8 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { scanClNodeLogs() } }) + } else { + b.l.Warn().Msg("LogStream won't be cleaned up, because test instance is not set or cleanup type is not standard") } } @@ -380,7 +378,7 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { } if b.te.LogStream != nil { - // this is not the cleanest way to do this, but when we originally build ethereum networks, we don't have the logstream reference + // this is not the cleanest way to do this, but when we originally build ethereum networks, we don't have the logstream reference, // so we need to rebuild them here and pass logstream to them for i := range b.privateEthereumNetworks { builder := test_env.NewEthereumNetworkBuilder() @@ -430,9 +428,26 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { b.te.rpcProviders[networkConfig.ChainID] = &rpcProvider b.te.EVMNetworks = append(b.te.EVMNetworks, &networkConfig) + } + dereferrencedEvms := make([]blockchain.EVMNetwork, 0) + for _, en := range b.te.EVMNetworks { + dereferrencedEvms = append(dereferrencedEvms, *en) } - err = b.te.StartClCluster(b.clNodeConfig, b.clNodesCount, b.secretsConfig, b.testConfig, b.clNodesOpts...) + + nodeConfigInToml := b.testConfig.GetNodeConfig() + + nodeConfig, _, err := node.BuildChainlinkNodeConfig( + dereferrencedEvms, + nodeConfigInToml.BaseConfigTOML, + nodeConfigInToml.CommonChainConfigTOML, + nodeConfigInToml.ChainConfigTOMLByChainID, + ) + if err != nil { + return nil, err + } + + err = b.te.StartClCluster(nodeConfig, b.clNodesCount, b.secretsConfig, b.testConfig, b.clNodesOpts...) if err != nil { return nil, err } @@ -472,18 +487,19 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { b.te.rpcProviders[networkConfig.ChainID] = &rpcProvider b.te.isSimulatedNetwork = false } + b.te.EVMNetworks = append(b.te.EVMNetworks, &networkConfig) } if !b.hasSeth && !b.hasEVMClient { - return nil, errors.New("you need to specify, which evm client to use: Seth or EVMClient") + log.Debug().Msg("No EVM client or SETH client specified, not starting any clients") } if b.hasSeth && b.hasEVMClient { return nil, errors.New("you can't use both Seth and EMVClient at the same time") } - if !b.isNonEVM { + if b.isEVM { if b.evmNetworkOption != nil && len(b.evmNetworkOption) > 0 { for _, fn := range b.evmNetworkOption { fn(&networkConfig) @@ -513,12 +529,21 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { if b.hasSeth { b.te.sethClients = make(map[int64]*seth.Client) - seth, err := actions_seth.GetChainClient(b.testConfig, networkConfig) + readSethCfg := b.testConfig.GetSethConfig() + sethCfg, err := utils.MergeSethAndEvmNetworkConfigs(networkConfig, *readSethCfg) + if err != nil { + return nil, err + } + err = utils.ValidateSethNetworkConfig(sethCfg.Network) + if err != nil { + return nil, err + } + sethClient, err := actions_seth.GetChainClient(b.testConfig, networkConfig) if err != nil { return nil, err } - b.te.sethClients[networkConfig.ChainID] = seth + b.te.sethClients[networkConfig.ChainID] = sethClient } } @@ -526,44 +551,33 @@ func (b *CLTestEnvBuilder) Build() (*CLClusterTestEnv, error) { // Start Chainlink Nodes if b.clNodesCount > 0 { - var cfg *chainlink.Config - if b.clNodeConfig != nil { - cfg = b.clNodeConfig - } else { - cfg = node.NewConfig(node.NewBaseConfig(), - node.WithOCR1(), - node.WithP2Pv2(), - ) - } - - if !b.isNonEVM { - var httpUrls []string - var wsUrls []string - rpcProvider, ok := b.te.rpcProviders[networkConfig.ChainID] - if !ok { - return nil, fmt.Errorf("rpc provider for chain %d not found", networkConfig.ChainID) - } - if networkConfig.Simulated { - httpUrls = rpcProvider.PrivateHttpUrls() - wsUrls = rpcProvider.PrivateWsUrsl() - } else { - httpUrls = networkConfig.HTTPURLs - wsUrls = networkConfig.URLs + dereferrencedEvms := make([]blockchain.EVMNetwork, 0) + for _, en := range b.te.EVMNetworks { + network := *en + if en.Simulated { + if rpcs, ok := b.te.rpcProviders[network.ChainID]; ok { + network.HTTPURLs = rpcs.PrivateHttpUrls() + network.URLs = rpcs.PrivateWsUrsl() + } else { + return nil, fmt.Errorf("rpc provider for chain %d not found", network.ChainID) + } } + dereferrencedEvms = append(dereferrencedEvms, network) + } - node.SetChainConfig(cfg, wsUrls, httpUrls, networkConfig, b.hasForwarders) + nodeConfigInToml := b.testConfig.GetNodeConfig() - if b.chainOptionsFn != nil && len(b.chainOptionsFn) > 0 { - for _, fn := range b.chainOptionsFn { - for _, evmCfg := range cfg.EVM { - chainCfg := evmCfg.Chain - fn(&chainCfg) - } - } - } + nodeConfig, _, err := node.BuildChainlinkNodeConfig( + dereferrencedEvms, + nodeConfigInToml.BaseConfigTOML, + nodeConfigInToml.CommonChainConfigTOML, + nodeConfigInToml.ChainConfigTOMLByChainID, + ) + if err != nil { + return nil, err } - err := b.te.StartClCluster(cfg, b.clNodesCount, b.secretsConfig, b.testConfig, b.clNodesOpts...) + err = b.te.StartClCluster(nodeConfig, b.clNodesCount, b.secretsConfig, b.testConfig, b.clNodesOpts...) if err != nil { return nil, err } diff --git a/integration-tests/go.mod b/integration-tests/go.mod index 610e7067ec2..9355e8007bc 100644 --- a/integration-tests/go.mod +++ b/integration-tests/go.mod @@ -27,8 +27,8 @@ require ( github.com/shopspring/decimal v1.3.1 github.com/slack-go/slack v0.12.2 github.com/smartcontractkit/chainlink-automation v1.0.3 - github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 - github.com/smartcontractkit/chainlink-testing-framework v1.28.12 + github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 + github.com/smartcontractkit/chainlink-testing-framework v1.28.14 github.com/smartcontractkit/chainlink-vrf v0.0.0-20231120191722-fef03814f868 github.com/smartcontractkit/chainlink/v2 v2.0.0-00010101000000-000000000000 github.com/smartcontractkit/libocr v0.0.0-20240419185742-fd3cab206b2c diff --git a/integration-tests/go.sum b/integration-tests/go.sum index 63f81dd8252..5fc82cbff5d 100644 --- a/integration-tests/go.sum +++ b/integration-tests/go.sum @@ -1512,8 +1512,8 @@ github.com/smartcontractkit/chain-selectors v1.0.10 h1:t9kJeE6B6G+hKD0GYR4kGJSCq github.com/smartcontractkit/chain-selectors v1.0.10/go.mod h1:d4Hi+E1zqjy9HqMkjBE5q1vcG9VGgxf5VxiRHfzi2kE= github.com/smartcontractkit/chainlink-automation v1.0.3 h1:h/ijT0NiyV06VxYVgcNfsE3+8OEzT3Q0Z9au0z1BPWs= github.com/smartcontractkit/chainlink-automation v1.0.3/go.mod h1:RjboV0Qd7YP+To+OrzHGXaxUxoSONveCoAK2TQ1INLU= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 h1:IwJKWZHPBJbbh4oI3BGX8VNT3c/ChNiPZ/XI4iq6c0E= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10/go.mod h1:sj0pjL+METqeYL9ibp0T8SXquymlaQsofa6bdfLgXX8= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 h1:Pr8/CdiTNnzRwpYc2z7NpHYbw3Dpl1eqiqt9/J/Bcqc= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28/go.mod h1:s+68EchlrXqHKRW3JJgZLEARvzMSKRI5+cE5Zx7pVJA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69 h1:Sec/GpBpUVaTEax1kSHlTvkzF/+d3w5roAQXaj5+SLA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69/go.mod h1:ZQKf+0OLzCLYIisH/OdOIQuFRI6bDuw+jPBTATyHfFM= github.com/smartcontractkit/chainlink-data-streams v0.0.0-20240220203239-09be0ea34540 h1:xFSv8561jsLtF6gYZr/zW2z5qUUAkcFkApin2mnbYTo= @@ -1524,8 +1524,8 @@ github.com/smartcontractkit/chainlink-solana v1.0.3-0.20240510181707-46b1311a5a8 github.com/smartcontractkit/chainlink-solana v1.0.3-0.20240510181707-46b1311a5a83/go.mod h1:RdAtOeBUWq2zByw2kEbwPlXaPIb7YlaDOmnn+nVUBJI= github.com/smartcontractkit/chainlink-starknet/relayer v0.0.1-beta-test.0.20240508155030-1024f2b55c69 h1:ssh/w3oXWu+C6bE88GuFRC1+0Bx/4ihsbc80XMLrl2k= github.com/smartcontractkit/chainlink-starknet/relayer v0.0.1-beta-test.0.20240508155030-1024f2b55c69/go.mod h1:VsfjhvWgjxqWja4q+FlXEtX5lu8BSxn10xRo6gi948g= -github.com/smartcontractkit/chainlink-testing-framework v1.28.12 h1:15ssos9DvWekvj6JjmiPjTYsj/uw12HvTWlm1FHdYaA= -github.com/smartcontractkit/chainlink-testing-framework v1.28.12/go.mod h1:x1zDOz8zcLjEvs9fNA9y/DMguLam/2+CJdpxX0+rM8A= +github.com/smartcontractkit/chainlink-testing-framework v1.28.14 h1:LChhbd/dJWf+yainZ/mjbfu6XP8IbGQ9S64MDTg+tt4= +github.com/smartcontractkit/chainlink-testing-framework v1.28.14/go.mod h1:x1zDOz8zcLjEvs9fNA9y/DMguLam/2+CJdpxX0+rM8A= github.com/smartcontractkit/chainlink-testing-framework/grafana v0.0.0-20240328204215-ac91f55f1449 h1:fX/xmGm1GBsD1ZZnooNT+eWA0hiTAqFlHzOC5CY4dy8= github.com/smartcontractkit/chainlink-testing-framework/grafana v0.0.0-20240328204215-ac91f55f1449/go.mod h1:DC8sQMyTlI/44UCTL8QWFwb0bYNoXCfjwCv2hMivYZU= github.com/smartcontractkit/chainlink-vrf v0.0.0-20231120191722-fef03814f868 h1:FFdvEzlYwcuVHkdZ8YnZR/XomeMGbz5E2F2HZI3I3w8= diff --git a/integration-tests/load/go.mod b/integration-tests/load/go.mod index 575986038e2..1e51191034c 100644 --- a/integration-tests/load/go.mod +++ b/integration-tests/load/go.mod @@ -16,8 +16,8 @@ require ( github.com/rs/zerolog v1.30.0 github.com/slack-go/slack v0.12.2 github.com/smartcontractkit/chainlink-automation v1.0.3 - github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 - github.com/smartcontractkit/chainlink-testing-framework v1.28.12 + github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 + github.com/smartcontractkit/chainlink-testing-framework v1.28.14 github.com/smartcontractkit/chainlink/integration-tests v0.0.0-20240214231432-4ad5eb95178c github.com/smartcontractkit/chainlink/v2 v2.9.0-beta0.0.20240216210048-da02459ddad8 github.com/smartcontractkit/libocr v0.0.0-20240419185742-fd3cab206b2c diff --git a/integration-tests/load/go.sum b/integration-tests/load/go.sum index a045ea3fd34..9542f355788 100644 --- a/integration-tests/load/go.sum +++ b/integration-tests/load/go.sum @@ -1502,8 +1502,8 @@ github.com/smartcontractkit/chain-selectors v1.0.10 h1:t9kJeE6B6G+hKD0GYR4kGJSCq github.com/smartcontractkit/chain-selectors v1.0.10/go.mod h1:d4Hi+E1zqjy9HqMkjBE5q1vcG9VGgxf5VxiRHfzi2kE= github.com/smartcontractkit/chainlink-automation v1.0.3 h1:h/ijT0NiyV06VxYVgcNfsE3+8OEzT3Q0Z9au0z1BPWs= github.com/smartcontractkit/chainlink-automation v1.0.3/go.mod h1:RjboV0Qd7YP+To+OrzHGXaxUxoSONveCoAK2TQ1INLU= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10 h1:IwJKWZHPBJbbh4oI3BGX8VNT3c/ChNiPZ/XI4iq6c0E= -github.com/smartcontractkit/chainlink-common v0.1.7-0.20240516150131-e1be553a9d10/go.mod h1:sj0pjL+METqeYL9ibp0T8SXquymlaQsofa6bdfLgXX8= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28 h1:Pr8/CdiTNnzRwpYc2z7NpHYbw3Dpl1eqiqt9/J/Bcqc= +github.com/smartcontractkit/chainlink-common v0.1.7-0.20240517134904-f4446b816a28/go.mod h1:s+68EchlrXqHKRW3JJgZLEARvzMSKRI5+cE5Zx7pVJA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69 h1:Sec/GpBpUVaTEax1kSHlTvkzF/+d3w5roAQXaj5+SLA= github.com/smartcontractkit/chainlink-cosmos v0.4.1-0.20240508101745-af1ed7bc8a69/go.mod h1:ZQKf+0OLzCLYIisH/OdOIQuFRI6bDuw+jPBTATyHfFM= github.com/smartcontractkit/chainlink-data-streams v0.0.0-20240220203239-09be0ea34540 h1:xFSv8561jsLtF6gYZr/zW2z5qUUAkcFkApin2mnbYTo= @@ -1514,8 +1514,8 @@ github.com/smartcontractkit/chainlink-solana v1.0.3-0.20240510181707-46b1311a5a8 github.com/smartcontractkit/chainlink-solana v1.0.3-0.20240510181707-46b1311a5a83/go.mod h1:RdAtOeBUWq2zByw2kEbwPlXaPIb7YlaDOmnn+nVUBJI= github.com/smartcontractkit/chainlink-starknet/relayer v0.0.1-beta-test.0.20240508155030-1024f2b55c69 h1:ssh/w3oXWu+C6bE88GuFRC1+0Bx/4ihsbc80XMLrl2k= github.com/smartcontractkit/chainlink-starknet/relayer v0.0.1-beta-test.0.20240508155030-1024f2b55c69/go.mod h1:VsfjhvWgjxqWja4q+FlXEtX5lu8BSxn10xRo6gi948g= -github.com/smartcontractkit/chainlink-testing-framework v1.28.12 h1:15ssos9DvWekvj6JjmiPjTYsj/uw12HvTWlm1FHdYaA= -github.com/smartcontractkit/chainlink-testing-framework v1.28.12/go.mod h1:x1zDOz8zcLjEvs9fNA9y/DMguLam/2+CJdpxX0+rM8A= +github.com/smartcontractkit/chainlink-testing-framework v1.28.14 h1:LChhbd/dJWf+yainZ/mjbfu6XP8IbGQ9S64MDTg+tt4= +github.com/smartcontractkit/chainlink-testing-framework v1.28.14/go.mod h1:x1zDOz8zcLjEvs9fNA9y/DMguLam/2+CJdpxX0+rM8A= github.com/smartcontractkit/chainlink-testing-framework/grafana v0.0.0-20240328204215-ac91f55f1449 h1:fX/xmGm1GBsD1ZZnooNT+eWA0hiTAqFlHzOC5CY4dy8= github.com/smartcontractkit/chainlink-testing-framework/grafana v0.0.0-20240328204215-ac91f55f1449/go.mod h1:DC8sQMyTlI/44UCTL8QWFwb0bYNoXCfjwCv2hMivYZU= github.com/smartcontractkit/chainlink-vrf v0.0.0-20231120191722-fef03814f868 h1:FFdvEzlYwcuVHkdZ8YnZR/XomeMGbz5E2F2HZI3I3w8= diff --git a/integration-tests/smoke/automation_test.go b/integration-tests/smoke/automation_test.go index 2c5295326bc..cafeea67c19 100644 --- a/integration-tests/smoke/automation_test.go +++ b/integration-tests/smoke/automation_test.go @@ -20,11 +20,9 @@ import ( ocr3 "github.com/smartcontractkit/libocr/offchainreporting2plus/ocr3confighelper" ocr2keepers30config "github.com/smartcontractkit/chainlink-automation/pkg/v3/config" - commonconfig "github.com/smartcontractkit/chainlink-common/pkg/config" ctfTestEnv "github.com/smartcontractkit/chainlink-testing-framework/docker/test_env" "github.com/smartcontractkit/chainlink-testing-framework/logging" "github.com/smartcontractkit/chainlink-testing-framework/networks" - "github.com/smartcontractkit/chainlink-testing-framework/utils/ptr" "github.com/smartcontractkit/chainlink-testing-framework/utils/testcontext" "github.com/smartcontractkit/chainlink/integration-tests/actions" @@ -34,7 +32,6 @@ import ( "github.com/smartcontractkit/chainlink/integration-tests/docker/test_env" tc "github.com/smartcontractkit/chainlink/integration-tests/testconfig" "github.com/smartcontractkit/chainlink/integration-tests/types" - "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" ac "github.com/smartcontractkit/chainlink/v2/core/gethwrappers/generated/automation_compatible_utils" "github.com/smartcontractkit/chainlink/v2/core/services/ocr2/plugins/ocr2keeper/evmregistry/v21/core" "github.com/smartcontractkit/chainlink/v2/core/services/ocr2/plugins/ocr2keeper/evmregistry/v21/gasprice" @@ -1353,17 +1350,6 @@ func setupAutomationTestDocker( registryConfig.RegistryVersion = registryVersion network := networks.MustGetSelectedNetworkConfig(automationTestConfig.GetNetworkConfig())[0] - // build the node config - clNodeConfig := node.NewConfig(node.NewBaseConfig()) - syncInterval := *commonconfig.MustNewDuration(5 * time.Minute) - clNodeConfig.Feature.LogPoller = ptr.Ptr[bool](true) - clNodeConfig.OCR2.Enabled = ptr.Ptr[bool](true) - clNodeConfig.Keeper.TurnLookBack = ptr.Ptr[int64](int64(0)) - clNodeConfig.Keeper.Registry.SyncInterval = &syncInterval - clNodeConfig.Keeper.Registry.PerformGasOverhead = ptr.Ptr[uint32](uint32(150000)) - clNodeConfig.P2P.V2.AnnounceAddresses = &[]string{"0.0.0.0:6690"} - clNodeConfig.P2P.V2.ListenAddresses = &[]string{"0.0.0.0:6690"} - //launch the environment var env *test_env.CLClusterTestEnv var err error @@ -1375,14 +1361,13 @@ func setupAutomationTestDocker( require.NoError(t, err, "Error building ethereum network config") if isMercuryV02 || isMercuryV03 { - env, err = test_env.NewCLTestEnvBuilder(). + // start mock adapter only + mockAdapterEnv, err := test_env.NewCLTestEnvBuilder(). WithTestInstance(t). WithTestConfig(automationTestConfig). - WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). - WithFunding(big.NewFloat(*automationTestConfig.GetCommonConfig().ChainlinkNodeFunding)). - WithStandardCleanup(). - WithSeth(). + WithoutEvmClients(). + WithoutCleanup(). Build() require.NoError(t, err, "Error deploying test environment for Mercury") @@ -1392,28 +1377,24 @@ func setupAutomationTestDocker( URL = '%s' Username = 'node' Password = 'nodepass'` - secretsConfig = fmt.Sprintf(secretsConfig, env.MockAdapter.InternalEndpoint, env.MockAdapter.InternalEndpoint) - - rpcProvider, err := env.GetRpcProvider(network.ChainID) - require.NoError(t, err, "Error getting rpc provider") - - var httpUrls []string - var wsUrls []string - if network.Simulated { - httpUrls = []string{rpcProvider.PrivateHttpUrls()[0]} - wsUrls = []string{rpcProvider.PrivateWsUrsl()[0]} - } else { - httpUrls = network.HTTPURLs - wsUrls = network.URLs - } + secretsConfig = fmt.Sprintf(secretsConfig, mockAdapterEnv.MockAdapter.InternalEndpoint, mockAdapterEnv.MockAdapter.InternalEndpoint) - node.SetChainConfig(clNodeConfig, wsUrls, httpUrls, network, false) + builder, err := test_env.NewCLTestEnvBuilder().WithTestEnv(mockAdapterEnv) + require.NoError(t, err, "Error building test environment for Mercury") - err = env.StartClCluster(clNodeConfig, clNodesCount, secretsConfig, automationTestConfig) - require.NoError(t, err, "Error starting CL nodes test environment for Mercury") - err = env.FundChainlinkNodes(big.NewFloat(*automationTestConfig.GetCommonConfig().ChainlinkNodeFunding)) - require.NoError(t, err, "Error funding CL nodes") + env, err = builder. + WithTestInstance(t). + WithTestConfig(automationTestConfig). + WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). + WithSecretsConfig(secretsConfig). + WithCLNodes(clNodesCount). + WithFunding(big.NewFloat(*automationTestConfig.GetCommonConfig().ChainlinkNodeFunding)). + WithStandardCleanup(). + WithSeth(). + Build() + require.NoError(t, err, "Error deploying test environment for Mercury") + env.MockAdapter = mockAdapterEnv.MockAdapter } else { env, err = test_env.NewCLTestEnvBuilder(). WithTestInstance(t). @@ -1421,7 +1402,6 @@ func setupAutomationTestDocker( WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). WithCLNodes(clNodesCount). - WithCLNodeConfig(clNodeConfig). WithFunding(big.NewFloat(*automationTestConfig.GetCommonConfig().ChainlinkNodeFunding)). WithStandardCleanup(). WithSeth(). diff --git a/integration-tests/smoke/forwarder_ocr_test.go b/integration-tests/smoke/forwarder_ocr_test.go index 0446254362a..2cae2ec3f45 100644 --- a/integration-tests/smoke/forwarder_ocr_test.go +++ b/integration-tests/smoke/forwarder_ocr_test.go @@ -37,7 +37,6 @@ func TestForwarderOCRBasic(t *testing.T) { WithTestConfig(&config). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). - WithForwarders(). WithCLNodes(6). WithFunding(big.NewFloat(.1)). WithStandardCleanup(). diff --git a/integration-tests/smoke/forwarders_ocr2_test.go b/integration-tests/smoke/forwarders_ocr2_test.go index 9dd5d5c39a4..a8d0e1987fe 100644 --- a/integration-tests/smoke/forwarders_ocr2_test.go +++ b/integration-tests/smoke/forwarders_ocr2_test.go @@ -18,8 +18,6 @@ import ( actions_seth "github.com/smartcontractkit/chainlink/integration-tests/actions/seth" "github.com/smartcontractkit/chainlink/integration-tests/contracts" "github.com/smartcontractkit/chainlink/integration-tests/docker/test_env" - "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" - tc "github.com/smartcontractkit/chainlink/integration-tests/testconfig" ) @@ -40,11 +38,6 @@ func TestForwarderOCR2Basic(t *testing.T) { WithTestConfig(&config). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). - WithCLNodeConfig(node.NewConfig(node.NewBaseConfig(), - node.WithOCR2(), - node.WithP2Pv2(), - )). - WithForwarders(). WithCLNodes(6). WithFunding(big.NewFloat(.1)). WithStandardCleanup(). @@ -94,9 +87,6 @@ func TestForwarderOCR2Basic(t *testing.T) { ocrInstances, err := actions_seth.DeployOCRv2Contracts(l, sethClient, 1, common.HexToAddress(lt.Address()), transmitters, ocrOffchainOptions) require.NoError(t, err, "Error deploying OCRv2 contracts with forwarders") - err = actions.CreateOCRv2JobsLocal(ocrInstances, bootstrapNode, workerNodes, env.MockAdapter, "ocr2", 5, uint64(sethClient.ChainID), true, false) - require.NoError(t, err, "Error creating OCRv2 jobs with forwarders") - ocrv2Config, err := actions.BuildMedianOCR2ConfigLocal(workerNodes, ocrOffchainOptions) require.NoError(t, err, "Error building OCRv2 config") ocrv2Config.Transmitters = authorizedForwarders @@ -104,6 +94,9 @@ func TestForwarderOCR2Basic(t *testing.T) { err = actions_seth.ConfigureOCRv2AggregatorContracts(ocrv2Config, ocrInstances) require.NoError(t, err, "Error configuring OCRv2 aggregator contracts") + err = actions.CreateOCRv2JobsLocal(ocrInstances, bootstrapNode, workerNodes, env.MockAdapter, "ocr2", 5, uint64(sethClient.ChainID), true, false) + require.NoError(t, err, "Error creating OCRv2 jobs with forwarders") + err = actions_seth.WatchNewOCRRound(l, sethClient, 1, contracts.V2OffChainAgrregatorToOffChainAggregatorWithRounds(ocrInstances), time.Duration(10*time.Minute)) require.NoError(t, err, "error watching for new OCRv2 round") diff --git a/integration-tests/smoke/keeper_test.go b/integration-tests/smoke/keeper_test.go index 00e45256cea..99817562c62 100644 --- a/integration-tests/smoke/keeper_test.go +++ b/integration-tests/smoke/keeper_test.go @@ -14,7 +14,6 @@ import ( "github.com/smartcontractkit/seth" "github.com/stretchr/testify/require" - commonconfig "github.com/smartcontractkit/chainlink-common/pkg/config" "github.com/smartcontractkit/chainlink-testing-framework/logging" "github.com/smartcontractkit/chainlink-testing-framework/networks" "github.com/smartcontractkit/chainlink-testing-framework/utils/testcontext" @@ -25,8 +24,6 @@ import ( "github.com/smartcontractkit/chainlink/integration-tests/contracts" "github.com/smartcontractkit/chainlink/integration-tests/contracts/ethereum" "github.com/smartcontractkit/chainlink/integration-tests/docker/test_env" - "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" - tc "github.com/smartcontractkit/chainlink/integration-tests/testconfig" ) @@ -1231,14 +1228,6 @@ func setupKeeperTest(l zerolog.Logger, t *testing.T, config *tc.TestConfig) ( contracts.LinkToken, *test_env.CLClusterTestEnv, ) { - clNodeConfig := node.NewConfig(node.NewBaseConfig(), node.WithP2Pv2()) - turnLookBack := int64(0) - syncInterval := *commonconfig.MustNewDuration(5 * time.Second) - performGasOverhead := uint32(150000) - clNodeConfig.Keeper.TurnLookBack = &turnLookBack - clNodeConfig.Keeper.Registry.SyncInterval = &syncInterval - clNodeConfig.Keeper.Registry.PerformGasOverhead = &performGasOverhead - privateNetwork, err := actions.EthereumNetworkConfigFromConfig(l, config) require.NoError(t, err, "Error building ethereum network config") @@ -1247,7 +1236,6 @@ func setupKeeperTest(l zerolog.Logger, t *testing.T, config *tc.TestConfig) ( WithTestConfig(config). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithCLNodes(5). - WithCLNodeConfig(clNodeConfig). WithFunding(big.NewFloat(.5)). WithStandardCleanup(). WithSeth(). diff --git a/integration-tests/smoke/log_poller_test.go b/integration-tests/smoke/log_poller_test.go index 04327471e7f..4ddffab30a6 100644 --- a/integration-tests/smoke/log_poller_test.go +++ b/integration-tests/smoke/log_poller_test.go @@ -211,7 +211,7 @@ func executeLogPollerReplay(t *testing.T, consistencyTimeout string) { eb, err := sethClient.Client.BlockNumber(testcontext.Get(t)) require.NoError(t, err, "Error getting latest block number") - endBlock, err := logpoller.GetEndBlockToWaitFor(int64(eb), sethClient.ChainID, cfg) + endBlock, err := logpoller.GetEndBlockToWaitFor(int64(eb), network, cfg) require.NoError(t, err, "Error getting end block to wait for") l.Info().Int64("Ending Block", endBlock).Int("Total logs emitted", totalLogsEmitted).Int64("Expected total logs emitted", expectedLogsEmitted).Str("Duration", fmt.Sprintf("%d sec", duration)).Str("LPS", fmt.Sprintf("%d/sec", totalLogsEmitted/duration)).Msg("FINISHED EVENT EMISSION") @@ -292,8 +292,6 @@ func prepareEnvironment(l zerolog.Logger, t *testing.T, testConfig *tc.TestConfi ethereum.RegistryVersion_2_1, logpoller.DefaultOCRRegistryConfig, upKeepsNeeded, - cfg.General.LogPollInterval.Duration, - *cfg.General.BackupLogPollerBlockDelay, *cfg.General.UseFinalityTag, testConfig, logScannerSettings, diff --git a/integration-tests/smoke/ocr2_test.go b/integration-tests/smoke/ocr2_test.go index d2df0c858c0..0e1a5c96633 100644 --- a/integration-tests/smoke/ocr2_test.go +++ b/integration-tests/smoke/ocr2_test.go @@ -4,6 +4,7 @@ import ( "fmt" "math/big" "net/http" + "strings" "testing" "time" @@ -13,6 +14,7 @@ import ( "github.com/stretchr/testify/require" "github.com/smartcontractkit/chainlink-testing-framework/logging" + "github.com/smartcontractkit/chainlink-testing-framework/logstream" "github.com/smartcontractkit/chainlink-testing-framework/networks" "github.com/smartcontractkit/chainlink-testing-framework/utils/testcontext" "github.com/smartcontractkit/chainlink/v2/core/config/env" @@ -22,20 +24,29 @@ import ( "github.com/smartcontractkit/chainlink/integration-tests/contracts" "github.com/smartcontractkit/chainlink/integration-tests/docker/test_env" tc "github.com/smartcontractkit/chainlink/integration-tests/testconfig" - "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" ) +type ocr2test struct { + name string + env map[string]string + chainReaderAndCodec bool +} + +func defaultTestData() ocr2test { + return ocr2test{ + name: "n/a", + env: make(map[string]string), + chainReaderAndCodec: false, + } +} + // Tests a basic OCRv2 median feed func TestOCRv2Basic(t *testing.T) { t.Parallel() noMedianPlugin := map[string]string{string(env.MedianPlugin.Cmd): ""} medianPlugin := map[string]string{string(env.MedianPlugin.Cmd): "chainlink-feeds"} - for _, test := range []struct { - name string - env map[string]string - chainReaderAndCodec bool - }{ + for _, test := range []ocr2test{ {"legacy", noMedianPlugin, false}, {"legacy-chain-reader", noMedianPlugin, true}, {"plugins", medianPlugin, false}, @@ -46,9 +57,9 @@ func TestOCRv2Basic(t *testing.T) { t.Parallel() l := logging.GetTestLogger(t) - env, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, l, 5) + testEnv, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, test, l, 5) - err := env.MockAdapter.SetAdapterBasedIntValuePath("ocr2", []string{http.MethodGet, http.MethodPost}, 10) + err := testEnv.MockAdapter.SetAdapterBasedIntValuePath("ocr2", []string{http.MethodGet, http.MethodPost}, 10) require.NoError(t, err) err = actions_seth.WatchNewOCRRound(l, sethClient, 2, contracts.V2OffChainAgrregatorToOffChainAggregatorWithRounds(aggregatorContracts), time.Minute*5) require.NoError(t, err) @@ -68,7 +79,7 @@ func TestOCRv2Request(t *testing.T) { t.Parallel() l := logging.GetTestLogger(t) - _, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, l, 5) + _, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, defaultTestData(), l, 5) // Keep the mockserver value the same and continually request new rounds for round := 2; round <= 4; round++ { @@ -90,7 +101,7 @@ func TestOCRv2JobReplacement(t *testing.T) { t.Parallel() l := logging.GetTestLogger(t) - env, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, l, 5) + env, aggregatorContracts, sethClient := prepareORCv2SmokeTestEnv(t, defaultTestData(), l, 5) nodeClients := env.ClCluster.NodeAPIs() bootstrapNode, workerNodes := nodeClients[0], nodeClients[1:] @@ -126,7 +137,7 @@ func TestOCRv2JobReplacement(t *testing.T) { ) } -func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult int) (*test_env.CLClusterTestEnv, []contracts.OffchainAggregatorV2, *seth.Client) { +func prepareORCv2SmokeTestEnv(t *testing.T, testData ocr2test, l zerolog.Logger, firstRoundResult int) (*test_env.CLClusterTestEnv, []contracts.OffchainAggregatorV2, *seth.Client) { config, err := tc.GetConfig("Smoke", tc.OCR2) if err != nil { t.Fatal(err) @@ -135,17 +146,15 @@ func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult i privateNetwork, err := actions.EthereumNetworkConfigFromConfig(l, &config) require.NoError(t, err, "Error building ethereum network config") - env, err := test_env.NewCLTestEnvBuilder(). + clNodeCount := 6 + + testEnv, err := test_env.NewCLTestEnvBuilder(). WithTestInstance(t). WithTestConfig(&config). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). - WithCLNodeConfig(node.NewConfig(node.NewBaseConfig(), - node.WithOCR2(), - node.WithP2Pv2(), - node.WithTracing(), - )). - WithCLNodes(6). + WithCLNodes(clNodeCount). + WithCLNodeOptions(test_env.WithNodeEnvVars(testData.env)). WithFunding(big.NewFloat(.1)). WithStandardCleanup(). WithSeth(). @@ -153,10 +162,10 @@ func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult i require.NoError(t, err) selectedNetwork := networks.MustGetSelectedNetworkConfig(config.Network)[0] - sethClient, err := env.GetSethClient(selectedNetwork.ChainID) + sethClient, err := testEnv.GetSethClient(selectedNetwork.ChainID) require.NoError(t, err, "Error getting seth client") - nodeClients := env.ClCluster.NodeAPIs() + nodeClients := testEnv.ClCluster.NodeAPIs() bootstrapNode, workerNodes := nodeClients[0], nodeClients[1:] linkContract, err := contracts.DeployLinkTokenContract(l, sethClient) @@ -179,7 +188,7 @@ func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult i aggregatorContracts, err := actions_seth.DeployOCRv2Contracts(l, sethClient, 1, common.HexToAddress(linkContract.Address()), transmitters, ocrOffchainOptions) require.NoError(t, err, "Error deploying OCRv2 aggregator contracts") - err = actions.CreateOCRv2JobsLocal(aggregatorContracts, bootstrapNode, workerNodes, env.MockAdapter, "ocr2", 5, uint64(sethClient.ChainID), false, false) + err = actions.CreateOCRv2JobsLocal(aggregatorContracts, bootstrapNode, workerNodes, testEnv.MockAdapter, "ocr2", 5, uint64(sethClient.ChainID), false, testData.chainReaderAndCodec) require.NoError(t, err, "Error creating OCRv2 jobs") ocrv2Config, err := actions.BuildMedianOCR2ConfigLocal(workerNodes, ocrOffchainOptions) @@ -188,6 +197,8 @@ func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult i err = actions_seth.ConfigureOCRv2AggregatorContracts(ocrv2Config, aggregatorContracts) require.NoError(t, err, "Error configuring OCRv2 aggregator contracts") + assertCorrectNodeConfiguration(t, l, clNodeCount, testData, testEnv) + err = actions_seth.WatchNewOCRRound(l, sethClient, 1, contracts.V2OffChainAgrregatorToOffChainAggregatorWithRounds(aggregatorContracts), time.Minute*5) require.NoError(t, err, "Error watching for new OCR2 round") roundData, err := aggregatorContracts[0].GetRound(testcontext.Get(t), big.NewInt(1)) @@ -197,5 +208,37 @@ func prepareORCv2SmokeTestEnv(t *testing.T, l zerolog.Logger, firstRoundResult i roundData.Answer.Int64(), ) - return env, aggregatorContracts, sethClient + return testEnv, aggregatorContracts, sethClient +} + +func assertCorrectNodeConfiguration(t *testing.T, l zerolog.Logger, totalNodeCount int, testData ocr2test, testEnv *test_env.CLClusterTestEnv) { + expectedNodesWithConfiguration := totalNodeCount - 1 // minus bootstrap node + expectedPatterns := []string{} + + if testData.env[string(env.MedianPlugin.Cmd)] != "" { + expectedPatterns = append(expectedPatterns, "Registered loopp.*OCR2.*Median.*") + } + + if testData.chainReaderAndCodec { + expectedPatterns = append(expectedPatterns, "relayConfig\\.chainReader") + } else { + expectedPatterns = append(expectedPatterns, "ChainReader missing from RelayConfig; falling back to internal MedianContract") + } + + // make sure that nodes are correctly configured by scanning the logs + for _, pattern := range expectedPatterns { + l.Info().Msgf("Checking for pattern: '%s' in CL node logs", pattern) + var correctlyConfiguredNodes []string + for i := 1; i < len(testEnv.ClCluster.Nodes); i++ { + logProcessor, processFn, err := logstream.GetRegexMatchingProcessor(testEnv.LogStream, pattern) + require.NoError(t, err, "Error getting regex matching processor") + + count, err := logProcessor.ProcessContainerLogs(testEnv.ClCluster.Nodes[i].ContainerName, processFn) + require.NoError(t, err, "Error processing container logs") + if *count >= 1 { + correctlyConfiguredNodes = append(correctlyConfiguredNodes, testEnv.ClCluster.Nodes[i].ContainerName) + } + } + require.Equal(t, expectedNodesWithConfiguration, len(correctlyConfiguredNodes), "expected correct plugin config to be applied to %d cl-nodes, but only following ones had it: %s; regexp used: %s", expectedNodesWithConfiguration, strings.Join(correctlyConfiguredNodes, ", "), string(pattern)) + } } diff --git a/integration-tests/smoke/ocr2vrf_test.go b/integration-tests/smoke/ocr2vrf_test.go index bf4f5804e99..83b96828444 100644 --- a/integration-tests/smoke/ocr2vrf_test.go +++ b/integration-tests/smoke/ocr2vrf_test.go @@ -33,6 +33,7 @@ var ocr2vrfSmokeConfig *testconfig.TestConfig func TestOCR2VRFRedeemModel(t *testing.T) { t.Parallel() + // remember to add TOML config for Chainlink node before trying to run this test in future t.Skip("VRFv3 is on pause, skipping") l := logging.GetTestLogger(t) config, err := testconfig.GetConfig("Smoke", testconfig.OCR2) diff --git a/integration-tests/smoke/runlog_test.go b/integration-tests/smoke/runlog_test.go index b01c5a019b1..3e806fc690b 100644 --- a/integration-tests/smoke/runlog_test.go +++ b/integration-tests/smoke/runlog_test.go @@ -39,7 +39,7 @@ func TestRunLogBasic(t *testing.T) { WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithMockAdapter(). WithCLNodes(1). - WithFunding(big.NewFloat(.1)). + WithFunding(big.NewFloat(.5)). WithStandardCleanup(). WithSeth(). Build() diff --git a/integration-tests/smoke/vrf_test.go b/integration-tests/smoke/vrf_test.go index 70911e8de66..1cc7bf73d69 100644 --- a/integration-tests/smoke/vrf_test.go +++ b/integration-tests/smoke/vrf_test.go @@ -196,7 +196,7 @@ func prepareVRFtestEnv(t *testing.T, l zerolog.Logger) (*test_env.CLClusterTestE WithTestConfig(&config). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithCLNodes(1). - WithFunding(big.NewFloat(.1)). + WithFunding(big.NewFloat(.5)). WithStandardCleanup(). WithSeth(). Build() diff --git a/integration-tests/testconfig/automation/automation.toml b/integration-tests/testconfig/automation/automation.toml index 4df2cbebc52..86eb279de39 100644 --- a/integration-tests/testconfig/automation/automation.toml +++ b/integration-tests/testconfig/automation/automation.toml @@ -2,6 +2,51 @@ [Common] chainlink_node_funding = 0.5 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + # smoke test specific overrodes [Smoke.Automation.AutomationConfig] use_log_buffer_v1=false diff --git a/integration-tests/testconfig/automation/example.toml b/integration-tests/testconfig/automation/example.toml index 7a3d33951ba..56d84cceeef 100644 --- a/integration-tests/testconfig/automation/example.toml +++ b/integration-tests/testconfig/automation/example.toml @@ -69,6 +69,81 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Common [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/default.toml b/integration-tests/testconfig/default.toml index 6ca5d4aec28..07017bc140d 100644 --- a/integration-tests/testconfig/default.toml +++ b/integration-tests/testconfig/default.toml @@ -29,6 +29,64 @@ addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] [PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] Deneb=500 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true +DefaultTransactionQueueDepth = 0 + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + [Seth] # controls which transactions are decoded/traced. Possbile values are: none, all, reverted (default). # if transaction level doesn't match, then calling Decode() does nothing. It's advised to keep it set diff --git a/integration-tests/testconfig/forwarder_ocr/example.toml b/integration-tests/testconfig/forwarder_ocr/example.toml new file mode 100644 index 00000000000..469b765b1db --- /dev/null +++ b/integration-tests/testconfig/forwarder_ocr/example.toml @@ -0,0 +1,168 @@ +# Example of full config with all fields +# General part +[ChainlinkImage] +image="public.ecr.aws/chainlink/chainlink" +version="2.7.0" + +[Logging] +# if set to true will save logs even if test did not fail +test_log_collect=false + +[Logging.LogStream] +# supported targets: file, loki, in-memory. if empty no logs will be persistet +log_targets=["file"] +# context timeout for starting log producer and also time-frame for requesting logs +log_producer_timeout="10s" +# number of retries before log producer gives up and stops listening to logs +log_producer_retry_limit=10 + +[Logging.Loki] +tenant_id="tenant_id" +# full URL of Loki ingest endpoint +endpoint="https://loki.url/api/v3/push" +# currently only needed when using public instance +basic_auth_secret="loki-basic-auth" +# only needed for cloud grafana +bearer_token_secret="bearer_token" + +# LogStream will try to shorten Grafana URLs by default (if all 3 variables are set) +[Logging.Grafana] +# grafana url (trailing "/" will be stripped) +base_url="http://grafana.url" +# url of your grafana dashboard (prefix and suffix "/" are stirpped), example: /d/ad61652-2712-1722/my-dashboard +dashboard_url="/d/your-dashboard" +bearer_token_secret="my-awesome-token" + +# if you want to use polygon_mumbial +[Network] +selected_networks=["polygon_mumbai"] + +[Network.RpcHttpUrls] +polygon_mumbai = ["https://my-rpc-endpoint.io"] + +[Network.RpcWsUrls] +polygon_mumbai = ["https://my-rpc-endpoint.io"] + +[Network.WalletKeys] +polygon_mumbai = ["change-me-to-your-PK"] + +[PrivateEthereumNetwork] +# pos or pow +consensus_type="pos" +# only prysm supported currently +consensus_layer="prysm" +# geth, besu, nethermind or erigon +execution_layer="geth" +# if true after env started it will wait for at least 1 epoch to be finalised before continuing +wait_for_finalization=false + +[PrivateEthereumNetwork.EthereumChainConfig] +# duration of single slot, lower => faster block production, must be >= 4 +seconds_per_slot=12 +# numer of slots in epoch, lower => faster epoch finalisation, must be >= 4 +slots_per_epoch=6 +# extra genesis gelay, no need to modify, but it should be after all validators/beacon chain starts +genesis_delay=15 +# number of validators in the network +validator_count=8 +chain_id=1337 +# list of addresses to be prefunded in genesis +addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] + +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true +DefaultTransactionQueueDepth = 0 + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + +# load test specific configuration +[Load.OCR] +[Load.OCR.Common] +eth_funds = 3 + +[Load.OCR.Load] +test_duration = "3m" +rate_limit_unit_duration = "1m" +rate = 3 +verification_interval = "5s" +verification_timeout = "3m" +ea_change_interval = "5s" + +# soak test specific configuration +[Soak.Common] +chainlink_node_funding = 100 + +[Soak.OCR] +[Soak.OCR.Common] +test_duration="15m" + +[Soak.OCR.Soak] +ocr_version="1" +number_of_contracts=2 +time_between_rounds="1m" \ No newline at end of file diff --git a/integration-tests/testconfig/forwarder_ocr/forwarder_ocr.toml b/integration-tests/testconfig/forwarder_ocr/forwarder_ocr.toml new file mode 100644 index 00000000000..f38cb4f5c4b --- /dev/null +++ b/integration-tests/testconfig/forwarder_ocr/forwarder_ocr.toml @@ -0,0 +1,57 @@ +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true +DefaultTransactionQueueDepth = 0 + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +[NodeConfig.ChainConfigTOMLByChainID] +1337 = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[Transactions] +ForwardersEnabled = true + """ + + diff --git a/integration-tests/testconfig/forwarder_ocr2/example.toml b/integration-tests/testconfig/forwarder_ocr2/example.toml new file mode 100644 index 00000000000..2bd03827506 --- /dev/null +++ b/integration-tests/testconfig/forwarder_ocr2/example.toml @@ -0,0 +1,168 @@ +# Example of full config with all fields +# General part +[ChainlinkImage] +image="public.ecr.aws/chainlink/chainlink" +version="2.7.0" + +[Logging] +# if set to true will save logs even if test did not fail +test_log_collect=false + +[Logging.LogStream] +# supported targets: file, loki, in-memory. if empty no logs will be persistet +log_targets=["file"] +# context timeout for starting log producer and also time-frame for requesting logs +log_producer_timeout="10s" +# number of retries before log producer gives up and stops listening to logs +log_producer_retry_limit=10 + +[Logging.Loki] +tenant_id="tenant_id" +# full URL of Loki ingest endpoint +endpoint="https://loki.url/api/v3/push" +# currently only needed when using public instance +basic_auth_secret="loki-basic-auth" +# only needed for cloud grafana +bearer_token_secret="bearer_token" + +# LogStream will try to shorten Grafana URLs by default (if all 3 variables are set) +[Logging.Grafana] +# grafana url (trailing "/" will be stripped) +base_url="http://grafana.url" +# url of your grafana dashboard (prefix and suffix "/" are stirpped), example: /d/ad61652-2712-1722/my-dashboard +dashboard_url="/d/your-dashboard" +bearer_token_secret="my-awesome-token" + +# if you want to use polygon_mumbial +[Network] +selected_networks=["polygon_mumbai"] + +[Network.RpcHttpUrls] +polygon_mumbai = ["https://my-rpc-endpoint.io"] + +[Network.RpcWsUrls] +polygon_mumbai = ["https://my-rpc-endpoint.io"] + +[Network.WalletKeys] +polygon_mumbai = ["change-me-to-your-PK"] + +[PrivateEthereumNetwork] +# pos or pow +consensus_type="pos" +# only prysm supported currently +consensus_layer="prysm" +# geth, besu, nethermind or erigon +execution_layer="geth" +# if true after env started it will wait for at least 1 epoch to be finalised before continuing +wait_for_finalization=false + +[PrivateEthereumNetwork.EthereumChainConfig] +# duration of single slot, lower => faster block production, must be >= 4 +seconds_per_slot=12 +# numer of slots in epoch, lower => faster epoch finalisation, must be >= 4 +slots_per_epoch=6 +# extra genesis gelay, no need to modify, but it should be after all validators/beacon chain starts +genesis_delay=15 +# number of validators in the network +validator_count=8 +chain_id=1337 +# list of addresses to be prefunded in genesis +addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] + +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true +DefaultTransactionQueueDepth = 0 + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + +# load test specific configuration +[Load.OCR] +[Load.OCR.Common] +eth_funds = 3 + +[Load.OCR.Load] +test_duration = "3m" +rate_limit_unit_duration = "1m" +rate = 3 +verification_interval = "5s" +verification_timeout = "3m" +ea_change_interval = "5s" + +# soak test specific configuration +[Soak.Common] +chainlink_node_funding = 100 + +[Soak.OCR] +[Soak.OCR.Common] +test_duration="15m" + +[Soak.OCR.Soak] +ocr_version="1" +number_of_contracts=2 +time_between_rounds="1m" \ No newline at end of file diff --git a/integration-tests/testconfig/forwarder_ocr2/forwarder_ocr2.toml b/integration-tests/testconfig/forwarder_ocr2/forwarder_ocr2.toml new file mode 100644 index 00000000000..7ce658935f8 --- /dev/null +++ b/integration-tests/testconfig/forwarder_ocr2/forwarder_ocr2.toml @@ -0,0 +1,53 @@ +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true +DefaultTransactionQueueDepth = 0 + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +[NodeConfig.ChainConfigTOMLByChainID] +1337 = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[Transactions] +ForwardersEnabled = true + """ + + diff --git a/integration-tests/testconfig/functions/example.toml b/integration-tests/testconfig/functions/example.toml index b96b39b0409..7cd875d4c8d 100644 --- a/integration-tests/testconfig/functions/example.toml +++ b/integration-tests/testconfig/functions/example.toml @@ -69,6 +69,81 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Common [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/keeper/example.toml b/integration-tests/testconfig/keeper/example.toml index 0bda9982988..87cf2045feb 100644 --- a/integration-tests/testconfig/keeper/example.toml +++ b/integration-tests/testconfig/keeper/example.toml @@ -69,6 +69,73 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[Keeper] +TurnLookBack = 0 +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Product part [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/keeper/keeper.toml b/integration-tests/testconfig/keeper/keeper.toml index 228ea077bd3..b3b0869e7c5 100644 --- a/integration-tests/testconfig/keeper/keeper.toml +++ b/integration-tests/testconfig/keeper/keeper.toml @@ -2,6 +2,43 @@ [Common] chainlink_node_funding = 0.5 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[Keeper] +TurnLookBack = 0 +""" + [Keeper.Common] registry_to_test = "2_1" number_of_registries = 1 diff --git a/integration-tests/testconfig/log_poller/config.go b/integration-tests/testconfig/log_poller/config.go index 890c33f26c9..f6e3249432a 100644 --- a/integration-tests/testconfig/log_poller/config.go +++ b/integration-tests/testconfig/log_poller/config.go @@ -8,11 +8,6 @@ import ( "github.com/smartcontractkit/chainlink-testing-framework/blockchain" ) -const ( - ErrReadPerfConfig = "failed to read TOML config for performance tests" - ErrUnmarshalPerfConfig = "failed to unmarshal TOML config for performance tests" -) - type GeneratorType = string const ( @@ -90,13 +85,11 @@ func (l *LoopedConfig) Validate() error { } type General struct { - Generator *string `toml:"generator"` - EventsToEmit []abi.Event `toml:"-"` - Contracts *int `toml:"contracts"` - EventsPerTx *int `toml:"events_per_tx"` - UseFinalityTag *bool `toml:"use_finality_tag"` - BackupLogPollerBlockDelay *uint64 `toml:"backup_log_poller_block_delay"` - LogPollInterval *blockchain.StrDuration `toml:"log_poll_interval"` + Generator *string `toml:"generator"` + EventsToEmit []abi.Event `toml:"-"` + Contracts *int `toml:"contracts"` + EventsPerTx *int `toml:"events_per_tx"` + UseFinalityTag *bool `toml:"use_finality_tag"` } func (g *General) Validate() error { diff --git a/integration-tests/testconfig/log_poller/example.toml b/integration-tests/testconfig/log_poller/example.toml index 88922e3fedb..a41433b1d06 100644 --- a/integration-tests/testconfig/log_poller/example.toml +++ b/integration-tests/testconfig/log_poller/example.toml @@ -69,6 +69,88 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[Keeper] +TurnLookBack = 0 + +[Keeper.Registry] +PerformGasOverhead = 150000 +SyncInterval = '5m0s' + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Common [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/log_poller/log_poller.toml b/integration-tests/testconfig/log_poller/log_poller.toml index 0e87c0e1e01..2f34bf3863d 100644 --- a/integration-tests/testconfig/log_poller/log_poller.toml +++ b/integration-tests/testconfig/log_poller/log_poller.toml @@ -6,6 +6,68 @@ consensus_layer="prysm" seconds_per_slot=4 slots_per_epoch=2 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[Keeper] +TurnLookBack = 0 + +[Keeper.Registry] +PerformGasOverhead = 150000 +SyncInterval = '5m0s' + +[P2P] +[P2P.V2] +Enabled = true +ListenAddresses = ['0.0.0.0:6690'] +AnnounceAddresses = ['0.0.0.0:6690'] +DeltaDial = '500ms' +DeltaReconcile = '5s' +""" + +# Chainlnk node settings that will be used by all tests +# BackupLogPollerBlockDelay = 0 disables the backup log poller +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityTagEnabled = true +""" + [Seth] ephemeral_addresses_number = 50 @@ -16,9 +78,6 @@ generator = "looped" contracts = 2 events_per_tx = 4 use_finality_tag = true -log_poll_interval = "500ms" -# 0 disables backup poller -backup_log_poller_block_delay = 0 [LogPoller.Looped] execution_count = 100 @@ -26,6 +85,15 @@ min_emit_wait_time_ms = 200 max_emit_wait_time_ms = 500 # test-specific +[TestLogPollerFewFiltersFixedDepth.NodeConfig] +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityDepth = 10 +FinalityTagEnabled = false +""" [TestLogPollerFewFiltersFixedDepth.LogPoller.General] use_finality_tag = false @@ -36,6 +104,16 @@ events_per_tx = 3 [TestLogManyFiltersPollerFinalityTag.LogPoller.Looped] execution_count = 30 +[TestLogManyFiltersPollerFixedDepth.NodeConfig] +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityDepth = 10 +FinalityTagEnabled = false +""" + [TestLogManyFiltersPollerFixedDepth.LogPoller.General] use_finality_tag = false contracts = 300 @@ -50,6 +128,16 @@ execution_count = 30 experiment_count = 4 target_component = "chainlink" +[TestLogPollerWithChaosFixedDepth.NodeConfig] +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityDepth = 10 +FinalityTagEnabled = false +""" + [TestLogPollerWithChaosFixedDepth.LogPoller.General] execution_count = 30 use_finality_tag = false @@ -63,6 +151,16 @@ execution_count = 30 experiment_count = 4 target_component = "postgres" +[TestLogPollerWithChaosPostgresFixedDepth.NodeConfig] +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityDepth = 10 +FinalityTagEnabled = false +""" + [TestLogPollerWithChaosPostgresFixedDepth.LogPoller.General] execution_count = 30 use_finality_tag = false @@ -70,6 +168,15 @@ use_finality_tag = false experiment_count = 4 target_component = "postgres" +[TestLogPollerReplayFixedDepth.NodeConfig] +CommonChainConfigTOML = """ +AutoCreateKey = true +MinContractPayment = 0 +LogPollInterval="500ms" +BackupLogPollerBlockDelay = 0 +FinalityDepth = 10 +FinalityTagEnabled = false +""" [TestLogPollerReplayFixedDepth.LogPoller.General] use_finality_tag = false diff --git a/integration-tests/testconfig/ocr/example.toml b/integration-tests/testconfig/ocr/example.toml index b8eb891ba33..3355ede75f7 100644 --- a/integration-tests/testconfig/ocr/example.toml +++ b/integration-tests/testconfig/ocr/example.toml @@ -69,6 +69,77 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # load test specific configuration [Load.OCR] [Load.OCR.Common] diff --git a/integration-tests/testconfig/ocr/ocr.toml b/integration-tests/testconfig/ocr/ocr.toml index 67d7d4588a0..5bc9b975648 100644 --- a/integration-tests/testconfig/ocr/ocr.toml +++ b/integration-tests/testconfig/ocr/ocr.toml @@ -2,6 +2,47 @@ [Common] chainlink_node_funding = 0.5 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + # load test specific configuration [Load.OCR] [Load.OCR.Common] diff --git a/integration-tests/testconfig/ocr2/example.toml b/integration-tests/testconfig/ocr2/example.toml index b8eb891ba33..9f7fdc7ff92 100644 --- a/integration-tests/testconfig/ocr2/example.toml +++ b/integration-tests/testconfig/ocr2/example.toml @@ -69,6 +69,77 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # load test specific configuration [Load.OCR] [Load.OCR.Common] diff --git a/integration-tests/testconfig/ocr2/ocr2.toml b/integration-tests/testconfig/ocr2/ocr2.toml index 8d3c73ca761..1d33cdc1ec4 100644 --- a/integration-tests/testconfig/ocr2/ocr2.toml +++ b/integration-tests/testconfig/ocr2/ocr2.toml @@ -2,6 +2,47 @@ [Common] chainlink_node_funding = 0.5 +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR2] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + # load test specific configuration [Load.OCR] [Load.OCR.Common] diff --git a/integration-tests/testconfig/testconfig.go b/integration-tests/testconfig/testconfig.go index abeca8e6eb2..337c960204a 100644 --- a/integration-tests/testconfig/testconfig.go +++ b/integration-tests/testconfig/testconfig.go @@ -231,7 +231,6 @@ type Product string const ( Automation Product = "automation" Cron Product = "cron" - DirectRequest Product = "direct_request" Flux Product = "flux" ForwarderOcr Product = "forwarder_ocr" ForwarderOcr2 Product = "forwarder_ocr2" @@ -248,8 +247,6 @@ const ( VRFv2Plus Product = "vrfv2plus" ) -var TestTypesWithLoki = []string{"Load", "Soak", "Stress", "Spike", "Volume"} - const TestTypeEnvVarName = "TEST_TYPE" func GetConfigurationNameFromEnv() (string, error) { diff --git a/integration-tests/testconfig/vrf/vrf.toml b/integration-tests/testconfig/vrf/vrf.toml new file mode 100644 index 00000000000..5282dae8910 --- /dev/null +++ b/integration-tests/testconfig/vrf/vrf.toml @@ -0,0 +1,40 @@ +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" \ No newline at end of file diff --git a/integration-tests/testconfig/vrfv2/example.toml b/integration-tests/testconfig/vrfv2/example.toml index 8a8139b92d4..0351ae85142 100644 --- a/integration-tests/testconfig/vrfv2/example.toml +++ b/integration-tests/testconfig/vrfv2/example.toml @@ -69,6 +69,77 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Common [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/vrfv2/vrfv2.toml b/integration-tests/testconfig/vrfv2/vrfv2.toml index 59affd85f5a..56257b9c1a6 100644 --- a/integration-tests/testconfig/vrfv2/vrfv2.toml +++ b/integration-tests/testconfig/vrfv2/vrfv2.toml @@ -1,6 +1,47 @@ # default config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + [Common] -chainlink_node_funding = 0.1 +chainlink_node_funding = 0.5 [VRFv2] [VRFv2.General] diff --git a/integration-tests/testconfig/vrfv2plus/example.toml b/integration-tests/testconfig/vrfv2plus/example.toml index 76fe23a2e70..1aaf8951e20 100644 --- a/integration-tests/testconfig/vrfv2plus/example.toml +++ b/integration-tests/testconfig/vrfv2plus/example.toml @@ -69,6 +69,77 @@ chain_id=1337 # list of addresses to be prefunded in genesis addresses_to_fund=["0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266"] +[PrivateEthereumNetwork.EthereumChainConfig.HardForkEpochs] +# hardforks to be applied (fork_name = epoch) +Deneb=500 + +# Chainlink node TOML config +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + +# override config toml related to EVMNode configs for chainlink nodes; applicable to all EVM node configs in chainlink toml +CommonChainConfigTOML = """ +AutoCreateKey = true +FinalityDepth = 1 +MinContractPayment = 0 + +[GasEstimator] +PriceMax = '200 gwei' +LimitDefault = 6000000 +FeeCapDefault = '200 gwei' +""" + +# chainlink override config toml for EVMNode config specific to EVM chains with chain id as mentioned in the key +[NodeConfig.ChainConfigTOMLByChainID] +# applicable for arbitrum-goerli chain +421613 = """ +[GasEstimator] +PriceMax = '400 gwei' +LimitDefault = 100000000 +FeeCapDefault = '200 gwei' +BumpThreshold = 60 +BumpPercent = 20 +BumpMin = '100 gwei' +""" + # Common [Common] chainlink_node_funding = 0.5 diff --git a/integration-tests/testconfig/vrfv2plus/vrfv2plus.toml b/integration-tests/testconfig/vrfv2plus/vrfv2plus.toml index b420a1c1d88..e441e647949 100644 --- a/integration-tests/testconfig/vrfv2plus/vrfv2plus.toml +++ b/integration-tests/testconfig/vrfv2plus/vrfv2plus.toml @@ -1,6 +1,48 @@ # default config + +[NodeConfig] +BaseConfigTOML = """ +[Feature] +FeedsManager = true +LogPoller = true +UICSAKeys = true + +[Log] +Level = 'debug' +JSONConsole = true + +[Log.File] +MaxSize = '0b' + +[WebServer] +AllowOrigins = '*' +HTTPPort = 6688 +SecureCookies = false +HTTPWriteTimeout = '1m' +SessionTimeout = '999h0m0s' + +[WebServer.RateLimit] +Authenticated = 2000 +Unauthenticated = 1000 + +[WebServer.TLS] +HTTPSPort = 0 + +[Database] +MaxIdleConns = 20 +MaxOpenConns = 40 +MigrateOnStartup = true + +[OCR] +Enabled = true + +[P2P] +[P2P.V2] +ListenAddresses = ['0.0.0.0:6690'] +""" + [Common] -chainlink_node_funding = 0.1 +chainlink_node_funding = 0.5 [VRFv2Plus] [VRFv2Plus.General] diff --git a/integration-tests/types/config/node/core.go b/integration-tests/types/config/node/core.go index 23efdf13a8b..8c8d6d1c339 100644 --- a/integration-tests/types/config/node/core.go +++ b/integration-tests/types/config/node/core.go @@ -4,18 +4,19 @@ import ( "bytes" "fmt" "math/big" - "os" + "strconv" "time" "github.com/segmentio/ksuid" "go.uber.org/zap/zapcore" - commonassets "github.com/smartcontractkit/chainlink-common/pkg/assets" + corechainlink "github.com/smartcontractkit/chainlink/v2/core/services/chainlink" + "github.com/smartcontractkit/chainlink-common/pkg/config" commonconfig "github.com/smartcontractkit/chainlink-common/pkg/config" "github.com/smartcontractkit/chainlink-testing-framework/blockchain" "github.com/smartcontractkit/chainlink-testing-framework/utils/ptr" - it_utils "github.com/smartcontractkit/chainlink/integration-tests/utils" + itutils "github.com/smartcontractkit/chainlink/integration-tests/utils" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/assets" evmcfg "github.com/smartcontractkit/chainlink/v2/core/chains/evm/config/toml" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/types" @@ -73,16 +74,9 @@ func NewConfig(baseConf *chainlink.Config, opts ...NodeConfigOpt) *chainlink.Con return baseConf } -func NewConfigFromToml(tomlFile string, opts ...NodeConfigOpt) (*chainlink.Config, error) { - readFile, err := os.ReadFile(tomlFile) - if err != nil { - return nil, err - } +func NewConfigFromToml(tomlConfig []byte, opts ...NodeConfigOpt) (*chainlink.Config, error) { var cfg chainlink.Config - if err != nil { - return nil, err - } - err = config.DecodeTOML(bytes.NewReader(readFile), &cfg) + err := config.DecodeTOML(bytes.NewReader(tomlConfig), &cfg) if err != nil { return nil, err } @@ -92,116 +86,37 @@ func NewConfigFromToml(tomlFile string, opts ...NodeConfigOpt) (*chainlink.Confi return &cfg, nil } -func WithOCR1() NodeConfigOpt { - return func(c *chainlink.Config) { - c.OCR = toml.OCR{ - Enabled: ptr.Ptr(true), - } - } -} - -func WithOCR2() NodeConfigOpt { - return func(c *chainlink.Config) { - c.OCR2 = toml.OCR2{ - Enabled: ptr.Ptr(true), - } - } -} - -func WithP2Pv2() NodeConfigOpt { - return func(c *chainlink.Config) { - c.P2P.V2 = toml.P2PV2{ - ListenAddresses: &[]string{"0.0.0.0:6690"}, +func WithPrivateEVMs(networks []blockchain.EVMNetwork, commonChainConfig *evmcfg.Chain, chainSpecificConfig map[int64]evmcfg.Chain) NodeConfigOpt { + var evmConfigs []*evmcfg.EVMConfig + for _, network := range networks { + var evmNodes []*evmcfg.Node + for i := range network.URLs { + evmNodes = append(evmNodes, &evmcfg.Node{ + Name: ptr.Ptr(fmt.Sprintf("%s-%d", network.Name, i)), + WSURL: itutils.MustURL(network.URLs[i]), + HTTPURL: itutils.MustURL(network.HTTPURLs[i]), + }) } - } -} - -func WithTracing() NodeConfigOpt { - return func(c *chainlink.Config) { - c.Tracing = toml.Tracing{ - Enabled: ptr.Ptr(true), - CollectorTarget: ptr.Ptr("otel-collector:4317"), - // ksortable unique id - NodeID: ptr.Ptr(ksuid.New().String()), - SamplingRatio: ptr.Ptr(1.0), - Mode: ptr.Ptr("unencrypted"), - Attributes: map[string]string{ - "env": "smoke", - }, + evmConfig := &evmcfg.EVMConfig{ + ChainID: ubig.New(big.NewInt(network.ChainID)), + Nodes: evmNodes, + Chain: evmcfg.Chain{}, } - } -} - -func SetChainConfig( - cfg *chainlink.Config, - wsUrls, - httpUrls []string, - chain blockchain.EVMNetwork, - forwarders bool, -) { - if cfg.EVM == nil { - var nodes []*evmcfg.Node - for i := range wsUrls { - node := evmcfg.Node{ - Name: ptr.Ptr(fmt.Sprintf("node_%d_%s", i, chain.Name)), - WSURL: it_utils.MustURL(wsUrls[i]), - HTTPURL: it_utils.MustURL(httpUrls[i]), - SendOnly: ptr.Ptr(false), - } - - nodes = append(nodes, &node) + if commonChainConfig != nil { + evmConfig.Chain = *commonChainConfig } - var chainConfig evmcfg.Chain - if chain.Simulated { - chainConfig = evmcfg.Chain{ - AutoCreateKey: ptr.Ptr(true), - FinalityDepth: ptr.Ptr[uint32](1), - MinContractPayment: commonassets.NewLinkFromJuels(0), + if chainSpecificConfig != nil { + if overriddenChainCfg, ok := chainSpecificConfig[network.ChainID]; ok { + evmConfig.Chain = overriddenChainCfg } } - cfg.EVM = evmcfg.EVMConfigs{ - { - ChainID: ubig.New(big.NewInt(chain.ChainID)), - Chain: chainConfig, - Nodes: nodes, - }, + if evmConfig.Chain.FinalityDepth == nil && network.FinalityDepth > 0 { + evmConfig.Chain.FinalityDepth = ptr.Ptr(uint32(network.FinalityDepth)) } - if forwarders { - cfg.EVM[0].Transactions = evmcfg.Transactions{ - ForwardersEnabled: ptr.Ptr(true), - } + if evmConfig.Chain.FinalityTagEnabled == nil && network.FinalityTag { + evmConfig.Chain.FinalityTagEnabled = ptr.Ptr(network.FinalityTag) } - } -} - -func WithPrivateEVMs(networks []blockchain.EVMNetwork) NodeConfigOpt { - var evmConfigs []*evmcfg.EVMConfig - for _, network := range networks { - evmConfigs = append(evmConfigs, &evmcfg.EVMConfig{ - ChainID: ubig.New(big.NewInt(network.ChainID)), - Chain: evmcfg.Chain{ - AutoCreateKey: ptr.Ptr(true), - FinalityDepth: ptr.Ptr[uint32](50), - MinContractPayment: commonassets.NewLinkFromJuels(0), - LogPollInterval: commonconfig.MustNewDuration(1 * time.Second), - HeadTracker: evmcfg.HeadTracker{ - HistoryDepth: ptr.Ptr(uint32(100)), - }, - GasEstimator: evmcfg.GasEstimator{ - LimitDefault: ptr.Ptr(uint64(6000000)), - PriceMax: assets.GWei(200), - FeeCapDefault: assets.GWei(200), - }, - }, - Nodes: []*evmcfg.Node{ - { - Name: ptr.Ptr(network.Name), - WSURL: it_utils.MustURL(network.URLs[0]), - HTTPURL: it_utils.MustURL(network.HTTPURLs[0]), - SendOnly: ptr.Ptr(false), - }, - }, - }) + evmConfigs = append(evmConfigs, evmConfig) } return func(c *chainlink.Config) { c.EVM = evmConfigs @@ -237,3 +152,46 @@ func WithLogPollInterval(interval time.Duration) NodeConfigOpt { c.EVM[0].Chain.LogPollInterval = commonconfig.MustNewDuration(interval) } } + +func BuildChainlinkNodeConfig(nets []blockchain.EVMNetwork, nodeConfig, commonChain string, configByChain map[string]string) (*corechainlink.Config, string, error) { + var tomlCfg *corechainlink.Config + var err error + var commonChainConfig *evmcfg.Chain + if commonChain != "" { + err = config.DecodeTOML(bytes.NewReader([]byte(commonChain)), &commonChainConfig) + if err != nil { + return nil, "", err + } + } + configByChainMap := make(map[int64]evmcfg.Chain) + for k, v := range configByChain { + var chain evmcfg.Chain + err = config.DecodeTOML(bytes.NewReader([]byte(v)), &chain) + if err != nil { + return nil, "", err + } + chainId, err := strconv.ParseInt(k, 10, 64) + if err != nil { + return nil, "", err + } + configByChainMap[chainId] = chain + } + if nodeConfig == "" { + tomlCfg = NewConfig( + NewBaseConfig(), + WithPrivateEVMs(nets, commonChainConfig, configByChainMap)) + } else { + tomlCfg, err = NewConfigFromToml([]byte(nodeConfig), WithPrivateEVMs(nets, commonChainConfig, configByChainMap)) + if err != nil { + return nil, "", err + } + } + + // we need unique id for each node for OTEL tracing + if tomlCfg.Tracing.Enabled != nil && *tomlCfg.Tracing.Enabled { + tomlCfg.Tracing.NodeID = ptr.Ptr(ksuid.New().String()) + } + + tomlStr, err := tomlCfg.TOMLString() + return tomlCfg, tomlStr, err +} diff --git a/integration-tests/universal/log_poller/helpers.go b/integration-tests/universal/log_poller/helpers.go index d732ad4af5d..9de11c439c8 100644 --- a/integration-tests/universal/log_poller/helpers.go +++ b/integration-tests/universal/log_poller/helpers.go @@ -25,12 +25,10 @@ import ( "github.com/smartcontractkit/seth" "github.com/smartcontractkit/wasp" - commonconfig "github.com/smartcontractkit/chainlink-common/pkg/config" "github.com/smartcontractkit/chainlink-testing-framework/blockchain" ctf_test_env "github.com/smartcontractkit/chainlink-testing-framework/docker/test_env" "github.com/smartcontractkit/chainlink-testing-framework/logging" "github.com/smartcontractkit/chainlink-testing-framework/networks" - "github.com/smartcontractkit/chainlink-testing-framework/utils/ptr" "github.com/smartcontractkit/chainlink/integration-tests/actions" actions_seth "github.com/smartcontractkit/chainlink/integration-tests/actions/seth" @@ -40,8 +38,6 @@ import ( "github.com/smartcontractkit/chainlink/integration-tests/docker/test_env" tc "github.com/smartcontractkit/chainlink/integration-tests/testconfig" lp_config "github.com/smartcontractkit/chainlink/integration-tests/testconfig/log_poller" - "github.com/smartcontractkit/chainlink/integration-tests/types/config/node" - evmcfg "github.com/smartcontractkit/chainlink/v2/core/chains/evm/config/toml" "github.com/smartcontractkit/chainlink/v2/core/chains/evm/logpoller" cltypes "github.com/smartcontractkit/chainlink/v2/core/chains/evm/types" ac "github.com/smartcontractkit/chainlink/v2/core/gethwrappers/generated/automation_compatible_utils" @@ -1046,46 +1042,17 @@ func ExecuteChaosExperiment(l zerolog.Logger, testEnv *test_env.CLClusterTestEnv }() } -// GetFinalityDepth returns the finality depth for the provided chain ID -func GetFinalityDepth(chainId int64) (int64, error) { - var finalityDepth int64 - switch chainId { - // Ethereum Sepolia - case 11155111: - finalityDepth = 50 - // Polygon Mumbai - case 80001: - finalityDepth = 500 - // Simulated network - case 1337: - finalityDepth = 10 - default: - return 0, fmt.Errorf("no known finality depth for chain %d", chainId) - } - - return finalityDepth, nil -} - // GetEndBlockToWaitFor returns the end block to wait for based on chain id and finality tag provided in config -func GetEndBlockToWaitFor(endBlock, chainId int64, cfg *lp_config.Config) (int64, error) { +func GetEndBlockToWaitFor(endBlock int64, network blockchain.EVMNetwork, cfg *lp_config.Config) (int64, error) { if *cfg.General.UseFinalityTag { return endBlock + 1, nil } - finalityDepth, err := GetFinalityDepth(chainId) - if err != nil { - return 0, err - } - - return endBlock + finalityDepth, nil + return endBlock + int64(network.FinalityDepth), nil } const ( - automationDefaultUpkeepGasLimit = uint32(2500000) - automationDefaultLinkFunds = int64(9e18) - automationDefaultUpkeepsToDeploy = 10 - automationExpectedData = "abcdef" - defaultAmountOfUpkeeps = 2 + defaultAmountOfUpkeeps = 2 ) var ( @@ -1111,8 +1078,6 @@ func SetupLogPollerTestDocker( registryVersion ethereum.KeeperRegistryVersion, registryConfig contracts.KeeperRegistrySettings, upkeepsNeeded int, - lpPollingInterval time.Duration, - backupPollingInterval uint64, finalityTagEnabled bool, testConfig *tc.TestConfig, logScannerSettings test_env.ChainlinkNodeLogScannerSettings, @@ -1131,43 +1096,21 @@ func SetupLogPollerTestDocker( registryConfig.RegistryVersion = registryVersion network := networks.MustGetSelectedNetworkConfig(testConfig.Network)[0] - finalityDepth, err := GetFinalityDepth(network.ChainID) - require.NoError(t, err, "Error getting finality depth") - - // build the node config - clNodeConfig := node.NewConfig(node.NewBaseConfig()) - syncInterval := *commonconfig.MustNewDuration(5 * time.Minute) - clNodeConfig.Feature.LogPoller = ptr.Ptr[bool](true) - clNodeConfig.OCR2.Enabled = ptr.Ptr[bool](true) - clNodeConfig.Keeper.TurnLookBack = ptr.Ptr[int64](int64(0)) - clNodeConfig.Keeper.Registry.SyncInterval = &syncInterval - clNodeConfig.Keeper.Registry.PerformGasOverhead = ptr.Ptr[uint32](uint32(150000)) - clNodeConfig.P2P.V2.Enabled = ptr.Ptr[bool](true) - clNodeConfig.P2P.V2.AnnounceAddresses = &[]string{"0.0.0.0:6690"} - clNodeConfig.P2P.V2.ListenAddresses = &[]string{"0.0.0.0:6690"} - //launch the environment var env *test_env.CLClusterTestEnv chainlinkNodeFunding := 0.5 l.Debug().Msgf("Funding amount: %f", chainlinkNodeFunding) clNodesCount := 5 - var logPolllerSettingsFn = func(chain *evmcfg.Chain) *evmcfg.Chain { - chain.LogPollInterval = commonconfig.MustNewDuration(lpPollingInterval) - chain.FinalityDepth = ptr.Ptr[uint32](uint32(finalityDepth)) - chain.FinalityTagEnabled = ptr.Ptr[bool](finalityTagEnabled) - chain.BackupLogPollerBlockDelay = ptr.Ptr[uint64](backupPollingInterval) - return chain - } - - var evmNetworkSettingsFn = func(network *blockchain.EVMNetwork) *blockchain.EVMNetwork { - network.FinalityDepth = uint64(finalityDepth) + var evmNetworkExtraSettingsFn = func(network *blockchain.EVMNetwork) *blockchain.EVMNetwork { + // we need it, because by default finality depth is 0 for our simulated network + if network.Simulated && !finalityTagEnabled { + network.FinalityDepth = 10 + } network.FinalityTag = finalityTagEnabled return network } - evmNetworkSettingsFn(&network) - privateNetwork, err := actions.EthereumNetworkConfigFromConfig(l, testConfig) require.NoError(t, err, "Error building ethereum network config") @@ -1176,10 +1119,8 @@ func SetupLogPollerTestDocker( WithTestInstance(t). WithPrivateEthereumNetwork(privateNetwork.EthereumNetworkConfig). WithCLNodes(clNodesCount). - WithCLNodeConfig(clNodeConfig). WithFunding(big.NewFloat(chainlinkNodeFunding)). - WithChainOptions(logPolllerSettingsFn). - EVMNetworkOptions(evmNetworkSettingsFn). + WithEVMNetworkOptions(evmNetworkExtraSettingsFn). WithChainlinkNodeLogScanner(logScannerSettings). WithStandardCleanup(). WithSeth().