Compare commits
1 Commits
master
...
fix/monito
Author | SHA1 | Date | |
---|---|---|---|
|
c94fd93c2d |
106
.github/workflows/main.yml
vendored
106
.github/workflows/main.yml
vendored
@ -15,7 +15,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/setup-node@v1
|
- uses: actions/setup-node@v1
|
||||||
with:
|
with:
|
||||||
node-version: 12
|
node-version: 10
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
submodules: true
|
submodules: true
|
||||||
@ -48,7 +48,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/setup-node@v1
|
- uses: actions/setup-node@v1
|
||||||
with:
|
with:
|
||||||
node-version: 12
|
node-version: 10
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
submodules: true
|
submodules: true
|
||||||
@ -61,6 +61,31 @@ jobs:
|
|||||||
key: ${{ needs.initialize.outputs.cache_key }}
|
key: ${{ needs.initialize.outputs.cache_key }}
|
||||||
- name: yarn run ${{ matrix.task }}
|
- name: yarn run ${{ matrix.task }}
|
||||||
run: ${{ steps.cache-repo.outputs.cache-hit }} && yarn run ${{ matrix.task }}
|
run: ${{ steps.cache-repo.outputs.cache-hit }} && yarn run ${{ matrix.task }}
|
||||||
|
ui-coverage:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- initialize
|
||||||
|
if: github.ref == 'refs/heads/master' || startsWith(github.ref, 'refs/tags')
|
||||||
|
steps:
|
||||||
|
- uses: actions/setup-node@v1
|
||||||
|
with:
|
||||||
|
node-version: 10
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
submodules: true
|
||||||
|
- uses: actions/cache@v2
|
||||||
|
id: cache-repo
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
**/node_modules
|
||||||
|
contracts/build
|
||||||
|
key: ${{ needs.initialize.outputs.cache_key }}
|
||||||
|
- name: yarn workspace ui run coverage
|
||||||
|
run: ${{ steps.cache-repo.outputs.cache-hit }} && yarn workspace ui run coverage
|
||||||
|
- uses: coverallsapp/github-action@master
|
||||||
|
with:
|
||||||
|
github-token: ${{ github.token }}
|
||||||
|
path-to-lcov: ./ui/coverage/lcov.info
|
||||||
build-e2e-images:
|
build-e2e-images:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
@ -70,19 +95,21 @@ jobs:
|
|||||||
- name: Evaluate e2e docker images tags
|
- name: Evaluate e2e docker images tags
|
||||||
run: |
|
run: |
|
||||||
git submodule status > submodule.status
|
git submodule status > submodule.status
|
||||||
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e', 'e2e-commons') }}" >> $GITHUB_ENV
|
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e') }}" >> $GITHUB_ENV
|
||||||
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
||||||
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
||||||
|
echo "UI_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'ui') }}" >> $GITHUB_ENV
|
||||||
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
||||||
- name: Rebuild and push updated images
|
- name: Rebuild and push updated images
|
||||||
run: |
|
run: |
|
||||||
function check_if_image_exists() {
|
function check_if_image_exists() {
|
||||||
curl -fsSlL "https://${{ github.actor }}:${{ github.token }}@${DOCKER_REGISTRY}/v2/${DOCKER_REPO}/tokenbridge-e2e-$1/manifests/$2" > /dev/null
|
curl -fsSlL -H 'Authorization: bearer ${{ github.token }}' "https://${DOCKER_REGISTRY}/v2/${DOCKER_REPO}/tokenbridge-e2e-$1/manifests/$2" > /dev/null
|
||||||
}
|
}
|
||||||
updated=()
|
updated=()
|
||||||
if ! check_if_image_exists e2e ${E2E_TAG}; then updated+=("e2e"); fi
|
if ! check_if_image_exists e2e ${E2E_TAG}; then updated+=("e2e"); fi
|
||||||
if ! check_if_image_exists oracle ${ORACLE_TAG}; then updated+=("oracle-amb"); fi
|
if ! check_if_image_exists oracle ${ORACLE_TAG}; then updated+=("oracle"); fi
|
||||||
if ! check_if_image_exists monitor ${MONITOR_TAG}; then updated+=("monitor-amb"); fi
|
if ! check_if_image_exists monitor ${MONITOR_TAG}; then updated+=("monitor"); fi
|
||||||
|
if ! check_if_image_exists ui ${UI_TAG}; then updated+=("ui"); fi
|
||||||
if ! check_if_image_exists alm ${ALM_TAG}; then updated+=("alm"); fi
|
if ! check_if_image_exists alm ${ALM_TAG}; then updated+=("alm"); fi
|
||||||
if [ ${#updated[@]} -gt 0 ]; then
|
if [ ${#updated[@]} -gt 0 ]; then
|
||||||
echo "Updated services: ${updated[@]}"
|
echo "Updated services: ${updated[@]}"
|
||||||
@ -104,7 +131,7 @@ jobs:
|
|||||||
- name: Rebuild and push molecule runner e2e image
|
- name: Rebuild and push molecule runner e2e image
|
||||||
run: |
|
run: |
|
||||||
function check_if_image_exists() {
|
function check_if_image_exists() {
|
||||||
curl -fsSlL "https://${{ github.actor }}:${{ github.token }}@${DOCKER_REGISTRY}/v2/${DOCKER_REPO}/tokenbridge-e2e-$1/manifests/$2" > /dev/null
|
curl -fsSlL -H 'Authorization: bearer ${{ github.token }}' "https://${DOCKER_REGISTRY}/v2/${DOCKER_REPO}/tokenbridge-e2e-$1/manifests/$2" > /dev/null
|
||||||
}
|
}
|
||||||
if check_if_image_exists molecule_runner ${MOLECULE_RUNNER_TAG}; then
|
if check_if_image_exists molecule_runner ${MOLECULE_RUNNER_TAG}; then
|
||||||
echo "Image already exists"
|
echo "Image already exists"
|
||||||
@ -122,10 +149,12 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
task: [oracle-e2e, monitor-e2e, alm-e2e]
|
task: [oracle-e2e, monitor-e2e, alm-e2e, 'ui-e2e:ci']
|
||||||
include:
|
include:
|
||||||
- task: alm-e2e
|
- task: alm-e2e
|
||||||
use-cache: true
|
use-cache: true
|
||||||
|
- task: 'ui-e2e:ci'
|
||||||
|
use-cache: true
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
@ -133,9 +162,10 @@ jobs:
|
|||||||
- name: Evaluate e2e docker images tags
|
- name: Evaluate e2e docker images tags
|
||||||
run: |
|
run: |
|
||||||
git submodule status > submodule.status
|
git submodule status > submodule.status
|
||||||
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e', 'e2e-commons') }}" >> $GITHUB_ENV
|
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e') }}" >> $GITHUB_ENV
|
||||||
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
||||||
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
||||||
|
echo "UI_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'ui') }}" >> $GITHUB_ENV
|
||||||
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
||||||
- if: ${{ matrix.use-cache }}
|
- if: ${{ matrix.use-cache }}
|
||||||
uses: actions/cache@v2
|
uses: actions/cache@v2
|
||||||
@ -149,12 +179,6 @@ jobs:
|
|||||||
run: docker login ${DOCKER_REGISTRY} -u ${{ github.actor }} -p ${{ github.token }}
|
run: docker login ${DOCKER_REGISTRY} -u ${{ github.actor }} -p ${{ github.token }}
|
||||||
- name: yarn run ${{ matrix.task }}
|
- name: yarn run ${{ matrix.task }}
|
||||||
run: ${{ !matrix.use-cache || steps.cache-repo.outputs.cache-hit }} && yarn run ${{ matrix.task }}
|
run: ${{ !matrix.use-cache || steps.cache-repo.outputs.cache-hit }} && yarn run ${{ matrix.task }}
|
||||||
- name: Upload logs
|
|
||||||
if: always()
|
|
||||||
uses: actions/upload-artifact@v2
|
|
||||||
with:
|
|
||||||
name: logs-${{ matrix.task }}
|
|
||||||
path: e2e-commons/logs
|
|
||||||
deployment:
|
deployment:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs:
|
needs:
|
||||||
@ -163,7 +187,7 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
task: [oracle, monitor, multiple, repo]
|
task: [oracle, ui, monitor, multiple, repo]
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
@ -182,7 +206,20 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
task: [amb, erc-to-native]
|
task: [amb, erc-to-erc, erc-to-native, native-to-erc, amb-stake-erc-to-erc]
|
||||||
|
include:
|
||||||
|
- task: erc-to-erc
|
||||||
|
ui-e2e-grep: 'ERC TO ERC'
|
||||||
|
ui-config: 'e2e-commons/components-envs/ui-erc20.env'
|
||||||
|
- task: erc-to-native
|
||||||
|
ui-e2e-grep: 'ERC TO NATIVE'
|
||||||
|
ui-config: 'e2e-commons/components-envs/ui-erc20-native.env'
|
||||||
|
- task: native-to-erc
|
||||||
|
ui-e2e-grep: 'NATIVE TO ERC'
|
||||||
|
ui-config: 'e2e-commons/components-envs/ui.env'
|
||||||
|
- task: amb-stake-erc-to-erc
|
||||||
|
ui-e2e-grep: 'AMB-STAKE-ERC-TO-ERC'
|
||||||
|
ui-config: 'e2e-commons/components-envs/ui-amb-stake-erc20-erc20.env'
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
with:
|
with:
|
||||||
@ -190,9 +227,10 @@ jobs:
|
|||||||
- name: Evaluate e2e docker images tags
|
- name: Evaluate e2e docker images tags
|
||||||
run: |
|
run: |
|
||||||
git submodule status > submodule.status
|
git submodule status > submodule.status
|
||||||
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e', 'e2e-commons') }}" >> $GITHUB_ENV
|
echo "E2E_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'Dockerfile.e2e', 'commons', 'oracle-e2e', 'monitor-e2e') }}" >> $GITHUB_ENV
|
||||||
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
echo "ORACLE_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'oracle') }}" >> $GITHUB_ENV
|
||||||
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
echo "MONITOR_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'monitor') }}" >> $GITHUB_ENV
|
||||||
|
echo "UI_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'ui') }}" >> $GITHUB_ENV
|
||||||
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
echo "ALM_TAG=${{ hashFiles('yarn.lock', 'package.json', 'submodule.status', 'commons', 'alm') }}" >> $GITHUB_ENV
|
||||||
echo "MOLECULE_RUNNER_TAG=${{ hashFiles('./deployment-e2e/Dockerfile') }}" >> $GITHUB_ENV
|
echo "MOLECULE_RUNNER_TAG=${{ hashFiles('./deployment-e2e/Dockerfile') }}" >> $GITHUB_ENV
|
||||||
- uses: actions/cache@v2
|
- uses: actions/cache@v2
|
||||||
@ -205,23 +243,29 @@ jobs:
|
|||||||
- name: Login to docker registry
|
- name: Login to docker registry
|
||||||
run: docker login ${DOCKER_REGISTRY} -u ${{ github.actor }} -p ${{ github.token }}
|
run: docker login ${DOCKER_REGISTRY} -u ${{ github.actor }} -p ${{ github.token }}
|
||||||
- name: Deploy contracts
|
- name: Deploy contracts
|
||||||
run: ${{ steps.cache-repo.outputs.cache-hit }} && e2e-commons/up.sh deploy generate-amb-tx blocks
|
run: ${{ steps.cache-repo.outputs.cache-hit }} && e2e-commons/up.sh deploy blocks
|
||||||
- name: Pull e2e oracle image
|
- name: Pull e2e oracle image
|
||||||
run: |
|
run: |
|
||||||
docker-compose -f ./e2e-commons/docker-compose.yml pull oracle-amb
|
docker-compose -f ./e2e-commons/docker-compose.yml pull oracle
|
||||||
docker tag ${DOCKER_IMAGE_BASE}/tokenbridge-e2e-oracle:${ORACLE_TAG} poanetwork/tokenbridge-oracle:latest
|
docker tag ${DOCKER_IMAGE_BASE}/tokenbridge-e2e-oracle:${ORACLE_TAG} poanetwork/tokenbridge-oracle:latest
|
||||||
- name: Deploy oracle
|
- if: ${{ matrix.ui-e2e-grep }}
|
||||||
|
name: Pull e2e ui image
|
||||||
|
run: |
|
||||||
|
docker-compose -f ./e2e-commons/docker-compose.yml pull ui
|
||||||
|
docker build \
|
||||||
|
--build-arg DOCKER_IMAGE_BASE=${DOCKER_IMAGE_BASE} \
|
||||||
|
--build-arg UI_TAG=${UI_TAG} \
|
||||||
|
--build-arg DOT_ENV_PATH=${{ matrix.ui-config }} \
|
||||||
|
-f ./e2e-commons/Dockerfile.ui \
|
||||||
|
-t ui_ui:latest \
|
||||||
|
.
|
||||||
|
- name: Deploy oracle and ui
|
||||||
run: deployment-e2e/molecule.sh ultimate-${{ matrix.task }}
|
run: deployment-e2e/molecule.sh ultimate-${{ matrix.task }}
|
||||||
- name: Reset docker socket permissions
|
- name: Reset docker socket permissions
|
||||||
run: sudo chown -R $USER:docker /var/run/docker.sock
|
run: sudo chown -R $USER:docker /var/run/docker.sock
|
||||||
|
- name: Run ui e2e tests
|
||||||
|
if: ${{ matrix.ui-e2e-grep }}
|
||||||
|
run: cd ui-e2e && xvfb-run yarn mocha -g "${{ matrix.ui-e2e-grep }}" -b ./test.js
|
||||||
- name: Run oracle e2e tests
|
- name: Run oracle e2e tests
|
||||||
run: docker-compose -f ./e2e-commons/docker-compose.yml run -e ULTIMATE=true e2e yarn workspace oracle-e2e run ${{ matrix.task }}
|
if: ${{ !matrix.ui-e2e-grep }}
|
||||||
- name: Save logs
|
run: docker-compose -f ./e2e-commons/docker-compose.yml run e2e yarn workspace oracle-e2e run ${{ matrix.task }}
|
||||||
if: always()
|
|
||||||
run: e2e-commons/down.sh
|
|
||||||
- name: Upload logs
|
|
||||||
if: always()
|
|
||||||
uses: actions/upload-artifact@v2
|
|
||||||
with:
|
|
||||||
name: logs-ultimate-${{ matrix.task }}
|
|
||||||
path: e2e-commons/logs
|
|
||||||
|
@ -8,11 +8,11 @@ COMMON_HOME_RPC_URL | The HTTPS URL(s) used to communicate to the RPC nodes in t
|
|||||||
COMMON_FOREIGN_RPC_URL | The HTTPS URL(s) used to communicate to the RPC nodes in the Foreign network. Several URLs can be specified, delimited by spaces. If the connection to one of these nodes is lost the next URL is used for connection. | URL(s)
|
COMMON_FOREIGN_RPC_URL | The HTTPS URL(s) used to communicate to the RPC nodes in the Foreign network. Several URLs can be specified, delimited by spaces. If the connection to one of these nodes is lost the next URL is used for connection. | URL(s)
|
||||||
COMMON_HOME_BRIDGE_ADDRESS | The address of the bridge contract address in the Home network. It is used to listen to events from and send validators' transactions to the Home network. | hexidecimal beginning with "0x"
|
COMMON_HOME_BRIDGE_ADDRESS | The address of the bridge contract address in the Home network. It is used to listen to events from and send validators' transactions to the Home network. | hexidecimal beginning with "0x"
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS | The address of the bridge contract address in the Foreign network. It is used to listen to events from and send validators' transactions to the Foreign network. | hexidecimal beginning with "0x"
|
COMMON_FOREIGN_BRIDGE_ADDRESS | The address of the bridge contract address in the Foreign network. It is used to listen to events from and send validators' transactions to the Foreign network. | hexidecimal beginning with "0x"
|
||||||
COMMON_HOME_GAS_PRICE_SUPPLIER_URL | The URL used to get a JSON response from the gas price prediction oracle for the Home network. The gas price provided by the oracle is used to send the validator's transactions to the RPC node. Since it is assumed that the Home network has a predefined gas price (e.g. the gas price in the Core of POA.Network is `1 GWei`), the gas price oracle parameter can be omitted for such networks. Set to `eip1559-gas-estimation` if you want to use EIP1559 RPC-based gas estimation. | URL
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL | The URL used to get a JSON response from the gas price prediction oracle for the Home network. The gas price provided by the oracle is used to send the validator's transactions to the RPC node. Since it is assumed that the Home network has a predefined gas price (e.g. the gas price in the Core of POA.Network is `1 GWei`), the gas price oracle parameter can be omitted for such networks. | URL
|
||||||
COMMON_HOME_GAS_PRICE_SPEED_TYPE | Assuming the gas price oracle responds with the following JSON structure: `{"fast": 20.0, "block_time": 12.834, "health": true, "standard": 6.0, "block_number": 6470469, "instant": 71.0, "slow": 1.889}`, this parameter specifies the desirable transaction speed. The speed type can be omitted when `COMMON_HOME_GAS_PRICE_SUPPLIER_URL` is not used. | `instant` / `fast` / `standard` / `slow`
|
COMMON_HOME_GAS_PRICE_SPEED_TYPE | Assuming the gas price oracle responds with the following JSON structure: `{"fast": 20.0, "block_time": 12.834, "health": true, "standard": 6.0, "block_number": 6470469, "instant": 71.0, "slow": 1.889}`, this parameter specifies the desirable transaction speed. The speed type can be omitted when `COMMON_HOME_GAS_PRICE_SUPPLIER_URL` is not used. | `instant` / `fast` / `standard` / `slow`
|
||||||
COMMON_HOME_GAS_PRICE_FALLBACK | The gas price (in Wei) that is used if both the oracle and the fall back gas price specified in the Home Bridge contract are not available. | integer
|
COMMON_HOME_GAS_PRICE_FALLBACK | The gas price (in Wei) that is used if both the oracle and the fall back gas price specified in the Home Bridge contract are not available. | integer
|
||||||
COMMON_HOME_GAS_PRICE_FACTOR | A value that will multiply the gas price of the oracle to convert it to gwei. If the oracle API returns gas prices in gwei then this can be set to `1`. Also, it could be used to intentionally pay more gas than suggested by the oracle to guarantee the transaction verification. E.g. `1.25` or `1.5`. | integer
|
COMMON_HOME_GAS_PRICE_FACTOR | A value that will multiply the gas price of the oracle to convert it to gwei. If the oracle API returns gas prices in gwei then this can be set to `1`. Also, it could be used to intentionally pay more gas than suggested by the oracle to guarantee the transaction verification. E.g. `1.25` or `1.5`. | integer
|
||||||
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL | The URL used to get a JSON response from the gas price prediction oracle for the Foreign network. The provided gas price is used to send the validator's transactions to the RPC node. If the Foreign network is Ethereum Foundation mainnet, the oracle URL can be: https://gasprice.poa.network. Otherwise this parameter can be omitted. Set to `gas-price-oracle` if you want to use npm `gas-price-oracle` package for retrieving gas price from multiple sources. Set to `eip1559-gas-estimation` if you want to use EIP1559 RPC-based gas estimation. | URL
|
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL | The URL used to get a JSON response from the gas price prediction oracle for the Foreign network. The provided gas price is used to send the validator's transactions to the RPC node. If the Foreign network is Ethereum Foundation mainnet, the oracle URL can be: https://gasprice.poa.network. Otherwise this parameter can be omitted. Set to `gas-price-oracle` if you want to use npm `gas-price-oracle` package for retrieving gas price from multiple sources. | URL
|
||||||
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE | Assuming the gas price oracle responds with the following JSON structure: `{"fast": 20.0, "block_time": 12.834, "health": true, "standard": 6.0, "block_number": 6470469, "instant": 71.0, "slow": 1.889}`, this parameter specifies the desirable transaction speed. The speed type can be omitted when `COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL`is not used. | `instant` / `fast` / `standard` / `slow`
|
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE | Assuming the gas price oracle responds with the following JSON structure: `{"fast": 20.0, "block_time": 12.834, "health": true, "standard": 6.0, "block_number": 6470469, "instant": 71.0, "slow": 1.889}`, this parameter specifies the desirable transaction speed. The speed type can be omitted when `COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL`is not used. | `instant` / `fast` / `standard` / `slow`
|
||||||
COMMON_FOREIGN_GAS_PRICE_FALLBACK | The gas price (in Wei) used if both the oracle and fall back gas price specified in the Foreign Bridge contract are not available. | integer
|
COMMON_FOREIGN_GAS_PRICE_FALLBACK | The gas price (in Wei) used if both the oracle and fall back gas price specified in the Foreign Bridge contract are not available. | integer
|
||||||
COMMON_FOREIGN_GAS_PRICE_FACTOR | A value that will multiply the gas price of the oracle to convert it to gwei. If the oracle API returns gas prices in gwei then this can be set to `1`. Also, it could be used to intentionally pay more gas than suggested by the oracle to guarantee the transaction verification. E.g. `1.25` or `1.5`. | integer
|
COMMON_FOREIGN_GAS_PRICE_FACTOR | A value that will multiply the gas price of the oracle to convert it to gwei. If the oracle API returns gas prices in gwei then this can be set to `1`. Also, it could be used to intentionally pay more gas than suggested by the oracle to guarantee the transaction verification. E.g. `1.25` or `1.5`. | integer
|
||||||
@ -22,7 +22,7 @@ COMMON_FOREIGN_GAS_PRICE_FACTOR | A value that will multiply the gas price of th
|
|||||||
|
|
||||||
name | description | value
|
name | description | value
|
||||||
--- | --- | ---
|
--- | --- | ---
|
||||||
ORACLE_BRIDGE_MODE | The bridge mode. The bridge starts listening to a different set of events based on this parameter. | ERC_TO_NATIVE / ARBITRARY_MESSAGE
|
ORACLE_BRIDGE_MODE | The bridge mode. The bridge starts listening to a different set of events based on this parameter. | NATIVE_TO_ERC / ERC_TO_ERC / ERC_TO_NATIVE / ARBITRARY_MESSAGE
|
||||||
ORACLE_ALLOW_HTTP_FOR_RPC | **Only use in test environments - must be omitted in production environments.**. If this parameter is specified and set to `yes`, RPC URLs can be specified in form of HTTP links. A warning that the connection is insecure will be written to the logs. | `yes` / `no`
|
ORACLE_ALLOW_HTTP_FOR_RPC | **Only use in test environments - must be omitted in production environments.**. If this parameter is specified and set to `yes`, RPC URLs can be specified in form of HTTP links. A warning that the connection is insecure will be written to the logs. | `yes` / `no`
|
||||||
ORACLE_HOME_RPC_POLLING_INTERVAL | The interval in milliseconds used to request the RPC node in the Home network for new blocks. The interval should match the average production time for a new block. | integer
|
ORACLE_HOME_RPC_POLLING_INTERVAL | The interval in milliseconds used to request the RPC node in the Home network for new blocks. The interval should match the average production time for a new block. | integer
|
||||||
ORACLE_FOREIGN_RPC_POLLING_INTERVAL | The interval in milliseconds used to request the RPC node in the Foreign network for new blocks. The interval should match the average production time for a new block. | integer
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL | The interval in milliseconds used to request the RPC node in the Foreign network for new blocks. The interval should match the average production time for a new block. | integer
|
||||||
@ -44,23 +44,29 @@ ORACLE_ALWAYS_RELAY_SIGNATURES | If set to `true`, the oracle will always relay
|
|||||||
ORACLE_RPC_REQUEST_TIMEOUT | Timeout in milliseconds for a single RPC request. Default value is `ORACLE_*_RPC_POLLING_INTERVAL * 2`. | integer
|
ORACLE_RPC_REQUEST_TIMEOUT | Timeout in milliseconds for a single RPC request. Default value is `ORACLE_*_RPC_POLLING_INTERVAL * 2`. | integer
|
||||||
ORACLE_HOME_TX_RESEND_INTERVAL | Interval in milliseconds for automatic resending of stuck transactions for Home sender service. Defaults to 20 minutes. | integer
|
ORACLE_HOME_TX_RESEND_INTERVAL | Interval in milliseconds for automatic resending of stuck transactions for Home sender service. Defaults to 20 minutes. | integer
|
||||||
ORACLE_FOREIGN_TX_RESEND_INTERVAL | Interval in milliseconds for automatic resending of stuck transactions for Foreign sender service. Defaults to 20 minutes. | integer
|
ORACLE_FOREIGN_TX_RESEND_INTERVAL | Interval in milliseconds for automatic resending of stuck transactions for Foreign sender service. Defaults to 20 minutes. | integer
|
||||||
ORACLE_SHUTDOWN_SERVICE_URL | Optional external URL to some other service/monitor/configuration manager that controls the remote shutdown process. GET request should return `application/json` message with the following schema: `{ shutdown: true/false }`. | URL
|
|
||||||
ORACLE_SHUTDOWN_SERVICE_POLLING_INTERVAL | Optional interval in milliseconds used to request the side RPC node or external shutdown service. Default is 120000. | integer
|
|
||||||
ORACLE_SIDE_RPC_URL | Optional HTTPS URL(s) for communication with the external shutdown service or side RPC nodes, used for shutdown manager activities. Several URLs can be specified, delimited by spaces. If the connection to one of these nodes is lost the next URL is used for connection. | URL(s)
|
## UI configuration
|
||||||
ORACLE_FOREIGN_ARCHIVE_RPC_URL | Optional HTTPS URL(s) for communication with the archive nodes on the foreign network. Only used in AMB bridge mode for async information request processing. Several URLs can be specified, delimited by spaces. If the connection to one of these nodes is lost the next URL is used for connection. | URL(s)
|
|
||||||
ORACLE_SHUTDOWN_CONTRACT_ADDRESS | Optional contract address in the side chain accessible through `ORACLE_SIDE_RPC_URL`, where the method passed in `ORACLE_SHUTDOWN_CONTRACT_METHOD` is implemented. | `address`
|
name | description | value
|
||||||
ORACLE_SHUTDOWN_CONTRACT_METHOD | Method signature to be used in the side chain to identify the current shutdown status. Method should return boolean. Default value is `isShutdown()`. | `function signature`
|
--- | --- | ---
|
||||||
ORACLE_FOREIGN_RPC_BLOCK_POLLING_LIMIT | Max length for the block range used in `eth_getLogs` requests for polling contract events for the Foreign chain. Infinite, if not provided. | `integer`
|
UI_TITLE | The title for the bridge UI page. `%c` will be replaced by the name of the network. | string
|
||||||
ORACLE_HOME_RPC_BLOCK_POLLING_LIMIT | Max length for the block range used in `eth_getLogs` requests for polling contract events for the Home chain. Infinite, if not provided. | `integer`
|
UI_OG_TITLE | The meta title for the deployed bridge page. | string
|
||||||
ORACLE_JSONRPC_ERROR_CODES | Override default JSON rpc error codes that can trigger RPC fallback to the next URL from the list (or a retry in case of a single RPC URL). Default is `-32603,-32002,-32005`. Should be a comma-separated list of negative integers. | `string`
|
UI_DESCRIPTION | The meta description for the deployed bridge page. | string
|
||||||
ORACLE_HOME_EVENTS_REPROCESSING | If set to `true`, home events happened in the past will be refetched and processed once again, to ensure that nothing was missed on the first pass. | `bool`
|
UI_NATIVE_TOKEN_DISPLAY_NAME | name of the home native coin | string
|
||||||
ORACLE_HOME_EVENTS_REPROCESSING_BATCH_SIZE | Batch size for one `eth_getLogs` request when reprocessing old logs in the home chain. Defaults to `1000` | `integer`
|
UI_HOME_NETWORK_DISPLAY_NAME | name to be displayed for home network | string
|
||||||
ORACLE_HOME_EVENTS_REPROCESSING_BLOCK_DELAY | Block confirmations number, after which old logs are being reprocessed in the home chain. Defaults to `500` | `integer`
|
UI_FOREIGN_NETWORK_DISPLAY_NAME | name to be displayed for foreign network | string
|
||||||
ORACLE_HOME_RPC_SYNC_STATE_CHECK_INTERVAL | Interval for checking JSON RPC sync state, by requesting the latest block number. Oracle will switch to the fallback JSON RPC in case sync process is stuck | `integer`
|
UI_HOME_WITHOUT_EVENTS | `true` if home network doesn't support events | true/false
|
||||||
ORACLE_FOREIGN_EVENTS_REPROCESSING | If set to `true`, foreign events happened in the past will be refetched and processed once again, to ensure that nothing was missed on the first pass. | `bool`
|
UI_FOREIGN_WITHOUT_EVENTS | `true` if foreign network doesn't support events | true/false
|
||||||
ORACLE_FOREIGN_EVENTS_REPROCESSING_BATCH_SIZE | Batch size for one `eth_getLogs` request when reprocessing old logs in the foreign chain. Defaults to `500` | `integer`
|
UI_HOME_EXPLORER_TX_TEMPLATE | template link to transaction on home explorer. `%s` will be replaced by transaction hash | URL template
|
||||||
ORACLE_FOREIGN_EVENTS_REPROCESSING_BLOCK_DELAY | Block confirmations number, after which old logs are being reprocessed in the foreign chain. Defaults to `250` | `integer`
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE | template link to transaction on foreign explorer. `%s` will be replaced by transaction hash | URL template
|
||||||
ORACLE_FOREIGN_RPC_SYNC_STATE_CHECK_INTERVAL | Interval for checking JSON RPC sync state, by requesting the latest block number. Oracle will switch to the fallback JSON RPC in case sync process is stuck | `integer`
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE | template link to address on home explorer. `%s` will be replaced by address | URL template
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE | template link to address on foreign explorer. `%s` will be replaced by address | URL template
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL | An interval in milliseconds used to get the updated gas price value either from the oracle or from the Home Bridge contract. | integer
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL | An interval in milliseconds used to get the updated gas price value either from the oracle or from the Foreign Bridge contract. | integer
|
||||||
|
UI_PORT | The port for the UI app. | integer
|
||||||
|
UI_STYLES | The set of styles to render the bridge UI page. | core/classic/stake
|
||||||
|
UI_PUBLIC_URL | The public url for the deployed bridge page | string
|
||||||
|
|
||||||
|
|
||||||
## Monitor configuration
|
## Monitor configuration
|
||||||
@ -80,3 +86,5 @@ MONITOR_HOME_TO_FOREIGN_BLOCK_LIST | File with a list of addresses, separated by
|
|||||||
MONITOR_HOME_TO_FOREIGN_CHECK_SENDER | If set to `true`, instructs the oracle to do an extra check for transaction origin in the block/allowance list. `false` by default. | `true` / `false`
|
MONITOR_HOME_TO_FOREIGN_CHECK_SENDER | If set to `true`, instructs the oracle to do an extra check for transaction origin in the block/allowance list. `false` by default. | `true` / `false`
|
||||||
MONITOR_HOME_VALIDATORS_BALANCE_ENABLE | If set, defines the list of home validator addresses for which balance should be checked. | `string`
|
MONITOR_HOME_VALIDATORS_BALANCE_ENABLE | If set, defines the list of home validator addresses for which balance should be checked. | `string`
|
||||||
MONITOR_FOREIGN_VALIDATORS_BALANCE_ENABLE | If set, defines the list of foreign validator addresses for which balance should be checked. | `string`
|
MONITOR_FOREIGN_VALIDATORS_BALANCE_ENABLE | If set, defines the list of foreign validator addresses for which balance should be checked. | `string`
|
||||||
|
MONITOR_HOME_EXPLORER_API | The HTTPS URL of the Home network explorer API. If set, may be used as a fallback in case of the RPC node failure. | URL
|
||||||
|
MONITOR_FOREIGN_EXPLORER_API | The HTTPS URL of the Foreign network explorer API. If set, may be used as a fallback in case of the RPC node failure. | URL
|
@ -1,4 +1,4 @@
|
|||||||
FROM node:12 as contracts
|
FROM node:10 as contracts
|
||||||
|
|
||||||
WORKDIR /mono
|
WORKDIR /mono
|
||||||
|
|
||||||
@ -11,7 +11,7 @@ COPY ./contracts/truffle-config.js ./
|
|||||||
COPY ./contracts/contracts ./contracts
|
COPY ./contracts/contracts ./contracts
|
||||||
RUN npm run compile
|
RUN npm run compile
|
||||||
|
|
||||||
FROM node:12
|
FROM node:10
|
||||||
|
|
||||||
WORKDIR /mono
|
WORKDIR /mono
|
||||||
COPY package.json .
|
COPY package.json .
|
||||||
@ -19,7 +19,6 @@ COPY --from=contracts /mono/contracts/build ./contracts/build
|
|||||||
COPY commons/package.json ./commons/
|
COPY commons/package.json ./commons/
|
||||||
COPY oracle-e2e/package.json ./oracle-e2e/
|
COPY oracle-e2e/package.json ./oracle-e2e/
|
||||||
COPY monitor-e2e/package.json ./monitor-e2e/
|
COPY monitor-e2e/package.json ./monitor-e2e/
|
||||||
COPY oracle/src/utils/constants.js ./oracle/src/utils/constants.js
|
|
||||||
|
|
||||||
COPY yarn.lock .
|
COPY yarn.lock .
|
||||||
RUN NOYARNPOSTINSTALL=1 yarn install --frozen-lockfile --production
|
RUN NOYARNPOSTINSTALL=1 yarn install --frozen-lockfile --production
|
||||||
|
10
README.md
10
README.md
@ -19,11 +19,13 @@ Sub-repositories maintained within this monorepo are listed below.
|
|||||||
|
|
||||||
| Sub-repository | Description |
|
| Sub-repository | Description |
|
||||||
| --- | --- |
|
| --- | --- |
|
||||||
| [Oracle](oracle/README.md) | Responsible for listening to bridge related events and authorizing asset transfers. |
|
| [Oracle](oracle/README.md) | Oracle responsible for listening to bridge related events and authorizing asset transfers. |
|
||||||
|
| [UI](ui/README.md) | DApp interface to transfer tokens and coins between chains. |
|
||||||
| [Monitor](monitor/README.md) | Tool for checking balances and unprocessed events in bridged networks. |
|
| [Monitor](monitor/README.md) | Tool for checking balances and unprocessed events in bridged networks. |
|
||||||
| [Deployment](deployment/README.md) | Ansible playbooks for deploying cross-chain bridges. |
|
| [Deployment](deployment/README.md) | Ansible playbooks for deploying cross-chain bridges. |
|
||||||
| [Oracle-E2E](oracle-e2e/README.md) | End to end tests for the Oracle |
|
| [Oracle-E2E](oracle-e2e/README.md) | End to end tests for the Oracle |
|
||||||
| [Monitor-E2E](monitor-e2e/README.md) | End to end tests for the Monitor |
|
| [Monitor-E2E](monitor-e2e/README.md) | End to end tests for the Monitor |
|
||||||
|
| [UI-E2E](ui-e2e/README.md) | End to end tests for the UI |
|
||||||
| [Deployment-E2E](deployment-e2e/README.md) | End to end tests for the Deployment |
|
| [Deployment-E2E](deployment-e2e/README.md) | End to end tests for the Deployment |
|
||||||
| [Commons](commons/README.md) | Interfaces, constants and utilities shared between the sub-repositories |
|
| [Commons](commons/README.md) | Interfaces, constants and utilities shared between the sub-repositories |
|
||||||
| [E2E-Commons](e2e-commons/README.md) | Common utilities and configuration used in end to end tests |
|
| [E2E-Commons](e2e-commons/README.md) | Common utilities and configuration used in end to end tests |
|
||||||
@ -54,6 +56,8 @@ Additionally there are [Smart Contracts](https://github.com/poanetwork/tokenbrid
|
|||||||
|
|
||||||
The POA TokenBridge provides four operational modes:
|
The POA TokenBridge provides four operational modes:
|
||||||
|
|
||||||
|
- [x] `Native-to-ERC20` **Coins** on a Home network can be converted to ERC20-compatible **tokens** on a Foreign network. Coins are locked on the Home side and the corresponding amount of ERC20 tokens are minted on the Foreign side. When the operation is reversed, tokens are burnt on the Foreign side and unlocked in the Home network. **More Information: [POA-to-POA20 Bridge](https://medium.com/poa-network/introducing-poa-bridge-and-poa20-55d8b78058ac)**
|
||||||
|
- [x] `ERC20-to-ERC20` ERC20-compatible tokens on the Foreign network are locked and minted as ERC20-compatible tokens (ERC677 tokens) on the Home network. When transferred from Home to Foreign, they are burnt on the Home side and unlocked in the Foreign network. This can be considered a form of atomic swap when a user swaps the token "X" in network "A" to the token "Y" in network "B". **More Information: [ERC20-to-ERC20](https://medium.com/poa-network/introducing-the-erc20-to-erc20-tokenbridge-ce266cc1a2d0)**
|
||||||
- [x] `ERC20-to-Native`: Pre-existing **tokens** in the Foreign network are locked and **coins** are minted in the `Home` network. In this mode, the Home network consensus engine invokes [Parity's Block Reward contract](https://wiki.parity.io/Block-Reward-Contract.html) to mint coins per the bridge contract request. **More Information: [xDai Chain](https://medium.com/poa-network/poa-network-partners-with-makerdao-on-xdai-chain-the-first-ever-usd-stable-blockchain-65a078c41e6a)**
|
- [x] `ERC20-to-Native`: Pre-existing **tokens** in the Foreign network are locked and **coins** are minted in the `Home` network. In this mode, the Home network consensus engine invokes [Parity's Block Reward contract](https://wiki.parity.io/Block-Reward-Contract.html) to mint coins per the bridge contract request. **More Information: [xDai Chain](https://medium.com/poa-network/poa-network-partners-with-makerdao-on-xdai-chain-the-first-ever-usd-stable-blockchain-65a078c41e6a)**
|
||||||
- [x] `Arbitrary-Message`: Transfer arbitrary data between two networks as so the data could be interpreted as an arbitrary contract method invocation.
|
- [x] `Arbitrary-Message`: Transfer arbitrary data between two networks as so the data could be interpreted as an arbitrary contract method invocation.
|
||||||
|
|
||||||
@ -64,7 +68,7 @@ Clone the repository:
|
|||||||
git clone https://github.com/poanetwork/tokenbridge
|
git clone https://github.com/poanetwork/tokenbridge
|
||||||
```
|
```
|
||||||
|
|
||||||
If there is no need to build docker images for the TokenBridge components (oracle, monitor), initialize submodules, install dependencies, compile the Smart Contracts:
|
If there is no need to build docker images for the TokenBridge components (oracle, monitor, UI), initialize submodules, install dependencies, compile the Smart Contracts:
|
||||||
```
|
```
|
||||||
yarn initialize
|
yarn initialize
|
||||||
```
|
```
|
||||||
@ -87,7 +91,7 @@ Running tests for all projects:
|
|||||||
yarn test
|
yarn test
|
||||||
```
|
```
|
||||||
|
|
||||||
Additionally there are end-to-end tests for [Oracle](oracle-e2e/README.md) and [Monitor](monitor-e2e/README.md).
|
Additionally there are end-to-end tests for [Oracle](oracle-e2e/README.md) and [UI](ui-e2e/README.md).
|
||||||
|
|
||||||
For details on building, running and developing please refer to respective READMEs in sub-repositories.
|
For details on building, running and developing please refer to respective READMEs in sub-repositories.
|
||||||
|
|
||||||
|
@ -19,6 +19,6 @@
|
|||||||
"eslint-plugin-jest": "^23.18.0"
|
"eslint-plugin-jest": "^23.18.0"
|
||||||
},
|
},
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">= 12.22"
|
"node": ">= 10.18"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
cd $(dirname $0)
|
cd $(dirname $0)
|
||||||
|
|
||||||
../e2e-commons/up.sh deploy generate-amb-tx blocks alm alm-e2e
|
../e2e-commons/up.sh deploy blocks alm alm-e2e
|
||||||
|
|
||||||
# run oracle amb e2e tests to generate transactions for alm
|
# run oracle amb e2e tests to generate transactions for alm
|
||||||
docker-compose -f ../e2e-commons/docker-compose.yml run e2e yarn workspace oracle-e2e run alm
|
docker-compose -f ../e2e-commons/docker-compose.yml run e2e yarn workspace oracle-e2e run alm
|
||||||
|
@ -6,8 +6,8 @@ jest.setTimeout(60000)
|
|||||||
const statusText = 'Success'
|
const statusText = 'Success'
|
||||||
const statusSelector = 'label[data-id="status"]'
|
const statusSelector = 'label[data-id="status"]'
|
||||||
|
|
||||||
const homeToForeignTxURL = 'http://localhost:3004/77/0x295efbe6ae98937ef35d939376c9bd752b4dc6f6899a9d5ddd6a57cea3d76c89'
|
const homeToForeignTxURL = 'http://localhost:3004/77/0x58e7d63368335b9591d4dbb43889084f698fcee93ab7656fd7a39d8c66bc4b60'
|
||||||
const foreignToHomeTxURL = 'http://localhost:3004/42/0x7262f7dbe6c30599edded2137fbbe93c271b37f5c54dd27f713f0cf510e3b4dd'
|
const foreignToHomeTxURL = 'http://localhost:3004/42/0x592bf28fc896419d2838f71cd0388775814b692688f1ecd5b1519081566b994a'
|
||||||
|
|
||||||
describe('ALM', () => {
|
describe('ALM', () => {
|
||||||
let browser
|
let browser
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
FROM node:12 as contracts
|
FROM node:10 as contracts
|
||||||
|
|
||||||
WORKDIR /mono
|
WORKDIR /mono
|
||||||
|
|
||||||
|
@ -123,24 +123,6 @@ const abi: AbiItem[] = [
|
|||||||
stateMutability: 'nonpayable',
|
stateMutability: 'nonpayable',
|
||||||
type: 'function'
|
type: 'function'
|
||||||
},
|
},
|
||||||
{
|
|
||||||
constant: false,
|
|
||||||
inputs: [
|
|
||||||
{
|
|
||||||
name: '_data',
|
|
||||||
type: 'bytes'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: '_signatures',
|
|
||||||
type: 'bytes'
|
|
||||||
}
|
|
||||||
],
|
|
||||||
name: 'safeExecuteSignaturesWithAutoGasLimit',
|
|
||||||
outputs: [],
|
|
||||||
payable: false,
|
|
||||||
stateMutability: 'nonpayable',
|
|
||||||
type: 'function'
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
constant: true,
|
constant: true,
|
||||||
inputs: [
|
inputs: [
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import React, { useEffect, useState } from 'react'
|
import React from 'react'
|
||||||
import { TransactionReceipt } from 'web3-eth'
|
import { TransactionReceipt } from 'web3-eth'
|
||||||
import { useMessageConfirmations } from '../hooks/useMessageConfirmations'
|
import { useMessageConfirmations } from '../hooks/useMessageConfirmations'
|
||||||
import { MessageObject } from '../utils/web3'
|
import { MessageObject } from '../utils/web3'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
import { CONFIRMATIONS_STATUS, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
import { CONFIRMATIONS_STATUS } from '../config/constants'
|
||||||
import { CONFIRMATIONS_STATUS_LABEL, CONFIRMATIONS_STATUS_LABEL_HOME } from '../config/descriptions'
|
import { CONFIRMATIONS_STATUS_LABEL, CONFIRMATIONS_STATUS_LABEL_HOME } from '../config/descriptions'
|
||||||
import { SimpleLoading } from './commons/Loading'
|
import { SimpleLoading } from './commons/Loading'
|
||||||
import { ValidatorsConfirmations } from './ValidatorsConfirmations'
|
import { ValidatorsConfirmations } from './ValidatorsConfirmations'
|
||||||
@ -54,9 +54,7 @@ export const ConfirmationsContainer = ({
|
|||||||
home: { name: homeName },
|
home: { name: homeName },
|
||||||
foreign: { name: foreignName }
|
foreign: { name: foreignName }
|
||||||
} = useStateProvider()
|
} = useStateProvider()
|
||||||
const src = useValidatorContract(fromHome, receipt ? receipt.blockNumber : 0)
|
const { requiredSignatures, validatorList } = useValidatorContract({ fromHome, receipt })
|
||||||
const [executionBlockNumber, setExecutionBlockNumber] = useState(0)
|
|
||||||
const dst = useValidatorContract(!fromHome, executionBlockNumber || 'latest')
|
|
||||||
const { blockConfirmations } = useBlockConfirmations({ fromHome, receipt })
|
const { blockConfirmations } = useBlockConfirmations({ fromHome, receipt })
|
||||||
const {
|
const {
|
||||||
confirmations,
|
confirmations,
|
||||||
@ -73,21 +71,11 @@ export const ConfirmationsContainer = ({
|
|||||||
fromHome,
|
fromHome,
|
||||||
homeStartBlock,
|
homeStartBlock,
|
||||||
foreignStartBlock,
|
foreignStartBlock,
|
||||||
requiredSignatures: src.requiredSignatures,
|
requiredSignatures,
|
||||||
validatorList: src.validatorList,
|
validatorList,
|
||||||
targetValidatorList: dst.validatorList,
|
|
||||||
blockConfirmations
|
blockConfirmations
|
||||||
})
|
})
|
||||||
|
|
||||||
useEffect(
|
|
||||||
() => {
|
|
||||||
if (executionBlockNumber || executionData.status !== VALIDATOR_CONFIRMATION_STATUS.EXECUTION_SUCCESS) return
|
|
||||||
|
|
||||||
setExecutionBlockNumber(executionData.blockNumber)
|
|
||||||
},
|
|
||||||
[executionData.status, executionBlockNumber, executionData.blockNumber]
|
|
||||||
)
|
|
||||||
|
|
||||||
const statusLabel = fromHome ? CONFIRMATIONS_STATUS_LABEL_HOME : CONFIRMATIONS_STATUS_LABEL
|
const statusLabel = fromHome ? CONFIRMATIONS_STATUS_LABEL_HOME : CONFIRMATIONS_STATUS_LABEL
|
||||||
|
|
||||||
const parseDescription = () => {
|
const parseDescription = () => {
|
||||||
@ -126,22 +114,20 @@ export const ConfirmationsContainer = ({
|
|||||||
</MultiLine>
|
</MultiLine>
|
||||||
</StatusDescription>
|
</StatusDescription>
|
||||||
<ValidatorsConfirmations
|
<ValidatorsConfirmations
|
||||||
confirmations={fromHome ? confirmations.filter(c => dst.validatorList.includes(c.validator)) : confirmations}
|
confirmations={confirmations}
|
||||||
requiredSignatures={dst.requiredSignatures}
|
requiredSignatures={requiredSignatures}
|
||||||
validatorList={dst.validatorList}
|
validatorList={validatorList}
|
||||||
waitingBlocksResolved={waitingBlocksResolved}
|
waitingBlocksResolved={waitingBlocksResolved}
|
||||||
/>
|
/>
|
||||||
{signatureCollected && (
|
{signatureCollected && (
|
||||||
<ExecutionConfirmation
|
<ExecutionConfirmation
|
||||||
message={message}
|
messageData={message.data}
|
||||||
executionData={executionData}
|
executionData={executionData}
|
||||||
isHome={!fromHome}
|
isHome={!fromHome}
|
||||||
confirmations={confirmations}
|
signatureCollected={signatureCollected}
|
||||||
setExecutionData={setExecutionData}
|
setExecutionData={setExecutionData}
|
||||||
executionEventsFetched={executionEventsFetched}
|
executionEventsFetched={executionEventsFetched}
|
||||||
setPendingExecution={setPendingExecution}
|
setPendingExecution={setPendingExecution}
|
||||||
dstRequiredSignatures={dst.requiredSignatures}
|
|
||||||
dstValidatorList={dst.validatorList}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</StyledConfirmationContainer>
|
</StyledConfirmationContainer>
|
||||||
|
@ -1,50 +1,38 @@
|
|||||||
import React, { useEffect, useState } from 'react'
|
import React from 'react'
|
||||||
import { formatTimestamp, formatTxHash, getExplorerTxUrl } from '../utils/networks'
|
import { formatTimestamp, formatTxHash, getExplorerTxUrl } from '../utils/networks'
|
||||||
import { useWindowWidth } from '@react-hook/window-size'
|
import { useWindowWidth } from '@react-hook/window-size'
|
||||||
import { SEARCHING_TX, VALIDATOR_CONFIRMATION_STATUS, ALM_HOME_TO_FOREIGN_MANUAL_EXECUTION } from '../config/constants'
|
import { SEARCHING_TX, VALIDATOR_CONFIRMATION_STATUS, ALM_HOME_TO_FOREIGN_MANUAL_EXECUTION } from '../config/constants'
|
||||||
import { SimpleLoading } from './commons/Loading'
|
import { SimpleLoading } from './commons/Loading'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
import { ConfirmationParam, ExecutionData } from '../hooks/useMessageConfirmations'
|
import { ExecutionData } from '../hooks/useMessageConfirmations'
|
||||||
import { GreyLabel, RedLabel, SuccessLabel } from './commons/Labels'
|
import { GreyLabel, RedLabel, SuccessLabel } from './commons/Labels'
|
||||||
import { ExplorerTxLink } from './commons/ExplorerTxLink'
|
import { ExplorerTxLink } from './commons/ExplorerTxLink'
|
||||||
import { Thead, AgeTd, StatusTd } from './commons/Table'
|
import { Thead, AgeTd, StatusTd } from './commons/Table'
|
||||||
import { ManualExecutionButton } from './ManualExecutionButton'
|
import { ManualExecutionButton } from './ManualExecutionButton'
|
||||||
import { useStateProvider } from '../state/StateProvider'
|
|
||||||
import { matchesRule, MessageObject, WarnRule } from '../utils/web3'
|
|
||||||
import { WarningAlert } from './commons/WarningAlert'
|
|
||||||
import { ErrorAlert } from './commons/ErrorAlert'
|
|
||||||
|
|
||||||
const StyledExecutionConfirmation = styled.div`
|
const StyledExecutionConfirmation = styled.div`
|
||||||
margin-top: 30px;
|
margin-top: 30px;
|
||||||
`
|
`
|
||||||
|
|
||||||
export interface ExecutionConfirmationParams {
|
export interface ExecutionConfirmationParams {
|
||||||
message: MessageObject
|
messageData: string
|
||||||
executionData: ExecutionData
|
executionData: ExecutionData
|
||||||
setExecutionData: Function
|
setExecutionData: Function
|
||||||
confirmations: ConfirmationParam[]
|
signatureCollected: boolean | string[]
|
||||||
isHome: boolean
|
isHome: boolean
|
||||||
executionEventsFetched: boolean
|
executionEventsFetched: boolean
|
||||||
setPendingExecution: Function
|
setPendingExecution: Function
|
||||||
dstRequiredSignatures: number
|
|
||||||
dstValidatorList: string[]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ExecutionConfirmation = ({
|
export const ExecutionConfirmation = ({
|
||||||
message,
|
messageData,
|
||||||
executionData,
|
executionData,
|
||||||
setExecutionData,
|
setExecutionData,
|
||||||
confirmations,
|
signatureCollected,
|
||||||
isHome,
|
isHome,
|
||||||
executionEventsFetched,
|
executionEventsFetched,
|
||||||
setPendingExecution,
|
setPendingExecution
|
||||||
dstRequiredSignatures,
|
|
||||||
dstValidatorList
|
|
||||||
}: ExecutionConfirmationParams) => {
|
}: ExecutionConfirmationParams) => {
|
||||||
const { foreign } = useStateProvider()
|
|
||||||
const [safeExecutionAvailable, setSafeExecutionAvailable] = useState(false)
|
|
||||||
const [error, setError] = useState('')
|
|
||||||
const [warning, setWarning] = useState('')
|
|
||||||
const availableManualExecution =
|
const availableManualExecution =
|
||||||
!isHome &&
|
!isHome &&
|
||||||
(executionData.status === VALIDATOR_CONFIRMATION_STATUS.WAITING ||
|
(executionData.status === VALIDATOR_CONFIRMATION_STATUS.WAITING ||
|
||||||
@ -60,43 +48,9 @@ export const ExecutionConfirmation = ({
|
|||||||
const formattedValidator =
|
const formattedValidator =
|
||||||
windowWidth < 850 && executionData.validator ? formatTxHash(executionData.validator) : executionData.validator
|
windowWidth < 850 && executionData.validator ? formatTxHash(executionData.validator) : executionData.validator
|
||||||
|
|
||||||
useEffect(
|
|
||||||
() => {
|
|
||||||
if (!availableManualExecution || !foreign.bridgeContract) return
|
|
||||||
|
|
||||||
const p = foreign.bridgeContract.methods.getBridgeInterfacesVersion().call()
|
|
||||||
p.then(({ major, minor }: any) => {
|
|
||||||
major = parseInt(major, 10)
|
|
||||||
minor = parseInt(minor, 10)
|
|
||||||
if (major < 5 || (major === 5 && minor < 7)) return
|
|
||||||
|
|
||||||
setSafeExecutionAvailable(true)
|
|
||||||
})
|
|
||||||
},
|
|
||||||
[availableManualExecution, foreign.bridgeContract]
|
|
||||||
)
|
|
||||||
|
|
||||||
useEffect(
|
|
||||||
() => {
|
|
||||||
if (!message.data || !executionData || !availableManualExecution) return
|
|
||||||
|
|
||||||
try {
|
|
||||||
const fileName = 'warnRules'
|
|
||||||
const rules: WarnRule[] = require(`../snapshots/${fileName}.json`)
|
|
||||||
for (let rule of rules) {
|
|
||||||
if (matchesRule(rule, message)) {
|
|
||||||
setWarning(rule.message)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (e) {}
|
|
||||||
},
|
|
||||||
[availableManualExecution, executionData, message, message.data, setWarning]
|
|
||||||
)
|
|
||||||
|
|
||||||
const getExecutionStatusElement = (validatorStatus = '') => {
|
const getExecutionStatusElement = (validatorStatus = '') => {
|
||||||
switch (validatorStatus) {
|
switch (validatorStatus) {
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.EXECUTION_SUCCESS:
|
case VALIDATOR_CONFIRMATION_STATUS.SUCCESS:
|
||||||
return <SuccessLabel>{validatorStatus}</SuccessLabel>
|
return <SuccessLabel>{validatorStatus}</SuccessLabel>
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.FAILED:
|
case VALIDATOR_CONFIRMATION_STATUS.FAILED:
|
||||||
return <RedLabel>{validatorStatus}</RedLabel>
|
return <RedLabel>{validatorStatus}</RedLabel>
|
||||||
@ -114,8 +68,6 @@ export const ExecutionConfirmation = ({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<StyledExecutionConfirmation>
|
<StyledExecutionConfirmation>
|
||||||
{error && <ErrorAlert onClick={() => setError('')} error={error} />}
|
|
||||||
{warning && <WarningAlert onClick={() => setWarning('')} error={warning} />}
|
|
||||||
<table>
|
<table>
|
||||||
<Thead>
|
<Thead>
|
||||||
<tr>
|
<tr>
|
||||||
@ -153,14 +105,10 @@ export const ExecutionConfirmation = ({
|
|||||||
{availableManualExecution && (
|
{availableManualExecution && (
|
||||||
<td>
|
<td>
|
||||||
<ManualExecutionButton
|
<ManualExecutionButton
|
||||||
safeExecutionAvailable={safeExecutionAvailable}
|
messageData={messageData}
|
||||||
messageData={message.data}
|
|
||||||
setExecutionData={setExecutionData}
|
setExecutionData={setExecutionData}
|
||||||
confirmations={confirmations}
|
signatureCollected={signatureCollected as string[]}
|
||||||
setPendingExecution={setPendingExecution}
|
setPendingExecution={setPendingExecution}
|
||||||
setError={setError}
|
|
||||||
requiredSignatures={dstRequiredSignatures}
|
|
||||||
validatorList={dstValidatorList}
|
|
||||||
/>
|
/>
|
||||||
</td>
|
</td>
|
||||||
)}
|
)}
|
||||||
|
@ -8,6 +8,7 @@ import { TransactionReceipt } from 'web3-eth'
|
|||||||
import { InfoAlert } from './commons/InfoAlert'
|
import { InfoAlert } from './commons/InfoAlert'
|
||||||
import { ExplorerTxLink } from './commons/ExplorerTxLink'
|
import { ExplorerTxLink } from './commons/ExplorerTxLink'
|
||||||
import { FOREIGN_NETWORK_NAME, HOME_NETWORK_NAME } from '../config/constants'
|
import { FOREIGN_NETWORK_NAME, HOME_NETWORK_NAME } from '../config/constants'
|
||||||
|
import { ErrorAlert } from './commons/ErrorAlert'
|
||||||
|
|
||||||
const StyledMainPage = styled.div`
|
const StyledMainPage = styled.div`
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@ -51,7 +52,7 @@ export interface FormSubmitParams {
|
|||||||
|
|
||||||
export const MainPage = () => {
|
export const MainPage = () => {
|
||||||
const history = useHistory()
|
const history = useHistory()
|
||||||
const { home, foreign } = useStateProvider()
|
const { home, foreign, error, setError } = useStateProvider()
|
||||||
const [networkName, setNetworkName] = useState('')
|
const [networkName, setNetworkName] = useState('')
|
||||||
const [receipt, setReceipt] = useState<Maybe<TransactionReceipt>>(null)
|
const [receipt, setReceipt] = useState<Maybe<TransactionReceipt>>(null)
|
||||||
const [showInfoAlert, setShowInfoAlert] = useState(false)
|
const [showInfoAlert, setShowInfoAlert] = useState(false)
|
||||||
@ -131,6 +132,7 @@ export const MainPage = () => {
|
|||||||
</AlertP>
|
</AlertP>
|
||||||
</InfoAlert>
|
</InfoAlert>
|
||||||
)}
|
)}
|
||||||
|
{error && <ErrorAlert onClick={() => setError('')} error={error} />}
|
||||||
<Route exact path={['/']} children={<Form onSubmit={onFormSubmit} />} />
|
<Route exact path={['/']} children={<Form onSubmit={onFormSubmit} />} />
|
||||||
<Route
|
<Route
|
||||||
path={['/:chainId/:txHash/:messageIdParam', '/:chainId/:txHash']}
|
path={['/:chainId/:txHash/:messageIdParam', '/:chainId/:txHash']}
|
||||||
|
@ -14,92 +14,32 @@ import { useStateProvider } from '../state/StateProvider'
|
|||||||
import { signatureToVRS, packSignatures } from '../utils/signatures'
|
import { signatureToVRS, packSignatures } from '../utils/signatures'
|
||||||
import { getSuccessExecutionData } from '../utils/getFinalizationEvent'
|
import { getSuccessExecutionData } from '../utils/getFinalizationEvent'
|
||||||
import { TransactionReceipt } from 'web3-eth'
|
import { TransactionReceipt } from 'web3-eth'
|
||||||
import { ConfirmationParam } from '../hooks/useMessageConfirmations'
|
|
||||||
|
|
||||||
const ActionButton = styled.button`
|
const StyledButton = styled.button`
|
||||||
color: var(--button-color);
|
color: var(--button-color);
|
||||||
border-color: var(--font-color);
|
border-color: var(--font-color);
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
min-width: 120px;
|
|
||||||
padding: 1rem;
|
|
||||||
&:focus {
|
&:focus {
|
||||||
outline: var(--button-color);
|
outline: var(--button-color);
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
|
||||||
interface ManualExecutionButtonParams {
|
interface ManualExecutionButtonParams {
|
||||||
safeExecutionAvailable: boolean
|
|
||||||
messageData: string
|
messageData: string
|
||||||
setExecutionData: Function
|
setExecutionData: Function
|
||||||
confirmations: ConfirmationParam[]
|
signatureCollected: string[]
|
||||||
setPendingExecution: Function
|
setPendingExecution: Function
|
||||||
setError: Function
|
|
||||||
requiredSignatures: number
|
|
||||||
validatorList: string[]
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ManualExecutionButton = ({
|
export const ManualExecutionButton = ({
|
||||||
safeExecutionAvailable,
|
|
||||||
messageData,
|
messageData,
|
||||||
setExecutionData,
|
setExecutionData,
|
||||||
confirmations,
|
signatureCollected,
|
||||||
setPendingExecution,
|
setPendingExecution
|
||||||
setError,
|
|
||||||
requiredSignatures,
|
|
||||||
validatorList
|
|
||||||
}: ManualExecutionButtonParams) => {
|
}: ManualExecutionButtonParams) => {
|
||||||
const { foreign } = useStateProvider()
|
const { foreign, setError } = useStateProvider()
|
||||||
const { library, activate, account, active } = useWeb3React()
|
const { library, activate, account, active } = useWeb3React()
|
||||||
const [manualExecution, setManualExecution] = useState(false)
|
const [manualExecution, setManualExecution] = useState(false)
|
||||||
const [allowFailures, setAllowFailures] = useState(false)
|
|
||||||
const [ready, setReady] = useState(false)
|
|
||||||
const [title, setTitle] = useState('Loading')
|
|
||||||
const [validSignatures, setValidSignatures] = useState<string[]>([])
|
|
||||||
|
|
||||||
useEffect(
|
|
||||||
() => {
|
|
||||||
if (
|
|
||||||
!foreign.bridgeContract ||
|
|
||||||
!foreign.web3 ||
|
|
||||||
!confirmations ||
|
|
||||||
!confirmations.length ||
|
|
||||||
!requiredSignatures ||
|
|
||||||
!validatorList ||
|
|
||||||
!validatorList.length
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
const signatures = []
|
|
||||||
for (let i = 0; i < confirmations.length && signatures.length < requiredSignatures; i++) {
|
|
||||||
const sig = confirmations[i].signature
|
|
||||||
if (!sig) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
const { v, r, s } = signatureToVRS(sig)
|
|
||||||
const signer = foreign.web3.eth.accounts.recover(messageData, `0x${v}`, `0x${r}`, `0x${s}`)
|
|
||||||
if (validatorList.includes(signer)) {
|
|
||||||
signatures.push(sig)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (signatures.length >= requiredSignatures) {
|
|
||||||
setValidSignatures(signatures.slice(0, requiredSignatures))
|
|
||||||
setTitle('Execute')
|
|
||||||
setReady(true)
|
|
||||||
} else {
|
|
||||||
setTitle('Unavailable')
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[
|
|
||||||
foreign.bridgeContract,
|
|
||||||
foreign.web3,
|
|
||||||
validatorList,
|
|
||||||
requiredSignatures,
|
|
||||||
messageData,
|
|
||||||
setValidSignatures,
|
|
||||||
confirmations
|
|
||||||
]
|
|
||||||
)
|
|
||||||
|
|
||||||
useEffect(
|
useEffect(
|
||||||
() => {
|
() => {
|
||||||
@ -127,16 +67,12 @@ export const ManualExecutionButton = ({
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!library || !foreign.bridgeContract || !foreign.web3 || !validSignatures || !validSignatures.length) return
|
if (!library || !foreign.bridgeContract || !signatureCollected || !signatureCollected.length) return
|
||||||
|
|
||||||
const signatures = packSignatures(validSignatures.map(signatureToVRS))
|
const signatures = packSignatures(signatureCollected.map(signatureToVRS))
|
||||||
const messageId = messageData.slice(0, 66)
|
const messageId = messageData.slice(0, 66)
|
||||||
const bridge = foreign.bridgeContract
|
const bridge = foreign.bridgeContract
|
||||||
const executeMethod =
|
const data = bridge.methods.executeSignatures(messageData, signatures).encodeABI()
|
||||||
safeExecutionAvailable && !allowFailures
|
|
||||||
? bridge.methods.safeExecuteSignaturesWithAutoGasLimit
|
|
||||||
: bridge.methods.executeSignatures
|
|
||||||
const data = executeMethod(messageData, signatures).encodeABI()
|
|
||||||
setManualExecution(false)
|
setManualExecution(false)
|
||||||
|
|
||||||
library.eth
|
library.eth
|
||||||
@ -194,38 +130,17 @@ export const ManualExecutionButton = ({
|
|||||||
foreign.bridgeContract,
|
foreign.bridgeContract,
|
||||||
setError,
|
setError,
|
||||||
messageData,
|
messageData,
|
||||||
|
signatureCollected,
|
||||||
setExecutionData,
|
setExecutionData,
|
||||||
setPendingExecution,
|
setPendingExecution
|
||||||
safeExecutionAvailable,
|
|
||||||
allowFailures,
|
|
||||||
foreign.web3,
|
|
||||||
validSignatures
|
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div className="is-center">
|
||||||
<div className="is-center">
|
<StyledButton className="button outline" onClick={() => setManualExecution(true)}>
|
||||||
<ActionButton disabled={!ready} className="button outline" onClick={() => setManualExecution(true)}>
|
Execute
|
||||||
{title}
|
</StyledButton>
|
||||||
</ActionButton>
|
|
||||||
</div>
|
|
||||||
{safeExecutionAvailable && (
|
|
||||||
<div
|
|
||||||
title="Allow executed message to fail and record its failure on-chain without reverting the whole transaction.
|
|
||||||
Use fixed gas limit for execution."
|
|
||||||
className="is-center"
|
|
||||||
style={{ paddingTop: 10 }}
|
|
||||||
>
|
|
||||||
<input
|
|
||||||
type="checkbox"
|
|
||||||
id="allow-failures"
|
|
||||||
checked={allowFailures}
|
|
||||||
onChange={e => setAllowFailures(e.target.checked)}
|
|
||||||
/>
|
|
||||||
<label htmlFor="allow-failures">Unsafe mode</label>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import React from 'react'
|
import React from 'react'
|
||||||
import { formatTimestamp, formatTxHash, getExplorerTxUrl } from '../utils/networks'
|
import { formatTimestamp, formatTxHash, getExplorerTxUrl } from '../utils/networks'
|
||||||
import { useWindowWidth } from '@react-hook/window-size'
|
import { useWindowWidth } from '@react-hook/window-size'
|
||||||
import { RECENT_AGE, SEARCHING_TX, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
import { SEARCHING_TX, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
||||||
import { SimpleLoading } from './commons/Loading'
|
import { SimpleLoading } from './commons/Loading'
|
||||||
import styled from 'styled-components'
|
import styled from 'styled-components'
|
||||||
import { ConfirmationParam } from '../hooks/useMessageConfirmations'
|
import { ConfirmationParam } from '../hooks/useMessageConfirmations'
|
||||||
@ -31,9 +31,7 @@ export const ValidatorsConfirmations = ({
|
|||||||
const getValidatorStatusElement = (validatorStatus = '') => {
|
const getValidatorStatusElement = (validatorStatus = '') => {
|
||||||
switch (validatorStatus) {
|
switch (validatorStatus) {
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.SUCCESS:
|
case VALIDATOR_CONFIRMATION_STATUS.SUCCESS:
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.MANUAL:
|
return <SuccessLabel>{validatorStatus}</SuccessLabel>
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID:
|
|
||||||
return <SuccessLabel>{VALIDATOR_CONFIRMATION_STATUS.SUCCESS}</SuccessLabel>
|
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.FAILED:
|
case VALIDATOR_CONFIRMATION_STATUS.FAILED:
|
||||||
return <RedLabel>{validatorStatus}</RedLabel>
|
return <RedLabel>{validatorStatus}</RedLabel>
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.PENDING:
|
case VALIDATOR_CONFIRMATION_STATUS.PENDING:
|
||||||
@ -60,28 +58,26 @@ export const ValidatorsConfirmations = ({
|
|||||||
</tr>
|
</tr>
|
||||||
</Thead>
|
</Thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{confirmations.map((confirmation, i) => {
|
{validatorList.map((validator, i) => {
|
||||||
const displayedStatus = confirmation.status
|
const filteredConfirmation = confirmations.filter(c => c.validator === validator)
|
||||||
const explorerLink = getExplorerTxUrl(confirmation.txHash, true)
|
const confirmation = filteredConfirmation.length > 0 ? filteredConfirmation[0] : null
|
||||||
let elementIfNoTimestamp: any = <SimpleLoading />
|
const displayedStatus = confirmation && confirmation.status ? confirmation.status : ''
|
||||||
switch (displayedStatus) {
|
const explorerLink = confirmation && confirmation.txHash ? getExplorerTxUrl(confirmation.txHash, true) : ''
|
||||||
case '':
|
const elementIfNoTimestamp =
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.UNDEFINED:
|
displayedStatus !== VALIDATOR_CONFIRMATION_STATUS.WAITING &&
|
||||||
if (waitingBlocksResolved) {
|
displayedStatus !== VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED ? (
|
||||||
elementIfNoTimestamp = SEARCHING_TX
|
(displayedStatus === VALIDATOR_CONFIRMATION_STATUS.UNDEFINED || displayedStatus === '') &&
|
||||||
}
|
waitingBlocksResolved ? (
|
||||||
break
|
SEARCHING_TX
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.WAITING:
|
) : (
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED:
|
<SimpleLoading />
|
||||||
elementIfNoTimestamp = ''
|
)
|
||||||
break
|
) : (
|
||||||
case VALIDATOR_CONFIRMATION_STATUS.MANUAL:
|
''
|
||||||
elementIfNoTimestamp = RECENT_AGE
|
)
|
||||||
break
|
|
||||||
}
|
|
||||||
return (
|
return (
|
||||||
<tr key={i}>
|
<tr key={i}>
|
||||||
<td>{windowWidth < 850 ? formatTxHash(confirmation.validator) : confirmation.validator}</td>
|
<td>{windowWidth < 850 ? formatTxHash(validator) : validator}</td>
|
||||||
<StatusTd className="text-center">{getValidatorStatusElement(displayedStatus)}</StatusTd>
|
<StatusTd className="text-center">{getValidatorStatusElement(displayedStatus)}</StatusTd>
|
||||||
<AgeTd className="text-center">
|
<AgeTd className="text-center">
|
||||||
{confirmation && confirmation.timestamp > 0 ? (
|
{confirmation && confirmation.timestamp > 0 ? (
|
||||||
@ -98,7 +94,7 @@ export const ValidatorsConfirmations = ({
|
|||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
<RequiredConfirmations>
|
<RequiredConfirmations>
|
||||||
At least <strong>{requiredSignatures}</strong> of <strong>{validatorList.length}</strong> confirmations required
|
{requiredSignatures} of {validatorList.length} confirmations required
|
||||||
</RequiredConfirmations>
|
</RequiredConfirmations>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
@ -33,7 +33,7 @@ export const ErrorAlert = ({ onClick, error }: { onClick: () => void; error: str
|
|||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<div className="row is-center">
|
<div className="row is-center">
|
||||||
<StyledErrorAlert className="col-12 is-vertical-align row">
|
<StyledErrorAlert className="col-10 is-vertical-align row">
|
||||||
<InfoIcon color="var(--failed-color)" />
|
<InfoIcon color="var(--failed-color)" />
|
||||||
<TextContainer className="col-10">
|
<TextContainer className="col-10">
|
||||||
{text}
|
{text}
|
||||||
|
@ -1,34 +0,0 @@
|
|||||||
import React from 'react'
|
|
||||||
import styled from 'styled-components'
|
|
||||||
import { InfoIcon } from './InfoIcon'
|
|
||||||
import { CloseIcon } from './CloseIcon'
|
|
||||||
|
|
||||||
const StyledErrorAlert = styled.div`
|
|
||||||
border: 1px solid var(--warning-color);
|
|
||||||
border-radius: 4px;
|
|
||||||
margin-bottom: 20px;
|
|
||||||
padding-top: 10px;
|
|
||||||
`
|
|
||||||
|
|
||||||
const CloseIconContainer = styled.div`
|
|
||||||
cursor: pointer;
|
|
||||||
`
|
|
||||||
|
|
||||||
const TextContainer = styled.div`
|
|
||||||
white-space: pre-wrap;
|
|
||||||
flex-direction: column;
|
|
||||||
`
|
|
||||||
|
|
||||||
export const WarningAlert = ({ onClick, error }: { onClick: () => void; error: string }) => {
|
|
||||||
return (
|
|
||||||
<div className="row is-center">
|
|
||||||
<StyledErrorAlert className="col-12 is-vertical-align row">
|
|
||||||
<InfoIcon color="var(--warning-color)" />
|
|
||||||
<TextContainer className="col-10">{error}</TextContainer>
|
|
||||||
<CloseIconContainer className="col-1 is-vertical-align is-center" onClick={onClick}>
|
|
||||||
<CloseIcon color="var(--warning-color)" />
|
|
||||||
</CloseIconContainer>
|
|
||||||
</StyledErrorAlert>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
@ -54,19 +54,14 @@ export const CONFIRMATIONS_STATUS = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const VALIDATOR_CONFIRMATION_STATUS = {
|
export const VALIDATOR_CONFIRMATION_STATUS = {
|
||||||
SUCCESS: 'Confirmed',
|
SUCCESS: 'Success',
|
||||||
MANUAL: 'Manual',
|
|
||||||
EXECUTION_SUCCESS: 'Executed',
|
|
||||||
FAILED: 'Failed',
|
FAILED: 'Failed',
|
||||||
FAILED_VALID: 'Failed valid',
|
|
||||||
PENDING: 'Pending',
|
PENDING: 'Pending',
|
||||||
WAITING: 'Waiting',
|
WAITING: 'Waiting',
|
||||||
NOT_REQUIRED: 'Not required',
|
NOT_REQUIRED: 'Not required',
|
||||||
UNDEFINED: 'UNDEFINED'
|
UNDEFINED: 'UNDEFINED'
|
||||||
}
|
}
|
||||||
|
|
||||||
export const RECENT_AGE = 'Recent'
|
|
||||||
|
|
||||||
export const SEARCHING_TX = 'Searching Transaction...'
|
export const SEARCHING_TX = 'Searching Transaction...'
|
||||||
|
|
||||||
export const INCORRECT_CHAIN_ERROR = `Incorrect chain chosen. Switch to ${FOREIGN_NETWORK_NAME} in the wallet.`
|
export const INCORRECT_CHAIN_ERROR = `Incorrect chain chosen. Switch to ${FOREIGN_NETWORK_NAME} in the wallet.`
|
||||||
|
@ -29,16 +29,17 @@ export interface useMessageConfirmationsParams {
|
|||||||
foreignStartBlock: Maybe<number>
|
foreignStartBlock: Maybe<number>
|
||||||
requiredSignatures: number
|
requiredSignatures: number
|
||||||
validatorList: string[]
|
validatorList: string[]
|
||||||
targetValidatorList: string[]
|
|
||||||
blockConfirmations: number
|
blockConfirmations: number
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ConfirmationParam {
|
export interface BasicConfirmationParam {
|
||||||
validator: string
|
validator: string
|
||||||
status: string
|
status: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ConfirmationParam extends BasicConfirmationParam {
|
||||||
txHash: string
|
txHash: string
|
||||||
timestamp: number
|
timestamp: number
|
||||||
signature?: string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ExecutionData {
|
export interface ExecutionData {
|
||||||
@ -47,7 +48,6 @@ export interface ExecutionData {
|
|||||||
txHash: string
|
txHash: string
|
||||||
timestamp: number
|
timestamp: number
|
||||||
executionResult: boolean
|
executionResult: boolean
|
||||||
blockNumber: number
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useMessageConfirmations = ({
|
export const useMessageConfirmations = ({
|
||||||
@ -58,7 +58,6 @@ export const useMessageConfirmations = ({
|
|||||||
foreignStartBlock,
|
foreignStartBlock,
|
||||||
requiredSignatures,
|
requiredSignatures,
|
||||||
validatorList,
|
validatorList,
|
||||||
targetValidatorList,
|
|
||||||
blockConfirmations
|
blockConfirmations
|
||||||
}: useMessageConfirmationsParams) => {
|
}: useMessageConfirmationsParams) => {
|
||||||
const { home, foreign } = useStateProvider()
|
const { home, foreign } = useStateProvider()
|
||||||
@ -66,7 +65,7 @@ export const useMessageConfirmations = ({
|
|||||||
const [status, setStatus] = useState(CONFIRMATIONS_STATUS.UNDEFINED)
|
const [status, setStatus] = useState(CONFIRMATIONS_STATUS.UNDEFINED)
|
||||||
const [waitingBlocks, setWaitingBlocks] = useState(false)
|
const [waitingBlocks, setWaitingBlocks] = useState(false)
|
||||||
const [waitingBlocksResolved, setWaitingBlocksResolved] = useState(false)
|
const [waitingBlocksResolved, setWaitingBlocksResolved] = useState(false)
|
||||||
const [signatureCollected, setSignatureCollected] = useState(false)
|
const [signatureCollected, setSignatureCollected] = useState<boolean | string[]>(false)
|
||||||
const [executionEventsFetched, setExecutionEventsFetched] = useState(false)
|
const [executionEventsFetched, setExecutionEventsFetched] = useState(false)
|
||||||
const [collectedSignaturesEvent, setCollectedSignaturesEvent] = useState<Maybe<EventData>>(null)
|
const [collectedSignaturesEvent, setCollectedSignaturesEvent] = useState<Maybe<EventData>>(null)
|
||||||
const [executionData, setExecutionData] = useState<ExecutionData>({
|
const [executionData, setExecutionData] = useState<ExecutionData>({
|
||||||
@ -74,8 +73,7 @@ export const useMessageConfirmations = ({
|
|||||||
validator: '',
|
validator: '',
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0,
|
timestamp: 0,
|
||||||
executionResult: false,
|
executionResult: false
|
||||||
blockNumber: 0
|
|
||||||
})
|
})
|
||||||
const [waitingBlocksForExecution, setWaitingBlocksForExecution] = useState(false)
|
const [waitingBlocksForExecution, setWaitingBlocksForExecution] = useState(false)
|
||||||
const [waitingBlocksForExecutionResolved, setWaitingBlocksForExecutionResolved] = useState(false)
|
const [waitingBlocksForExecutionResolved, setWaitingBlocksForExecutionResolved] = useState(false)
|
||||||
@ -142,9 +140,10 @@ export const useMessageConfirmations = ({
|
|||||||
// The collected signature event is only fetched once the signatures are collected on tx from home to foreign, to calculate if
|
// The collected signature event is only fetched once the signatures are collected on tx from home to foreign, to calculate if
|
||||||
// the execution tx on the foreign network is waiting for block confirmations
|
// the execution tx on the foreign network is waiting for block confirmations
|
||||||
// This is executed if the message is in Home to Foreign direction only
|
// This is executed if the message is in Home to Foreign direction only
|
||||||
|
const hasCollectedSignatures = !!signatureCollected // true or string[]
|
||||||
useEffect(
|
useEffect(
|
||||||
() => {
|
() => {
|
||||||
if (!fromHome || !receipt || !home.web3 || !home.bridgeContract || !signatureCollected) return
|
if (!fromHome || !receipt || !home.web3 || !home.bridgeContract || !hasCollectedSignatures) return
|
||||||
|
|
||||||
let timeoutId: number
|
let timeoutId: number
|
||||||
let isCancelled = false
|
let isCancelled = false
|
||||||
@ -180,7 +179,7 @@ export const useMessageConfirmations = ({
|
|||||||
isCancelled = true
|
isCancelled = true
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[fromHome, home.bridgeContract, home.web3, message.data, receipt, signatureCollected]
|
[fromHome, home.bridgeContract, home.web3, message.data, receipt, hasCollectedSignatures]
|
||||||
)
|
)
|
||||||
|
|
||||||
// Check if the responsible validator is waiting for block confirmations to execute the message on foreign network
|
// Check if the responsible validator is waiting for block confirmations to execute the message on foreign network
|
||||||
@ -253,35 +252,6 @@ export const useMessageConfirmations = ({
|
|||||||
let timeoutId: number
|
let timeoutId: number
|
||||||
let isCancelled = false
|
let isCancelled = false
|
||||||
|
|
||||||
if (fromHome) {
|
|
||||||
if (!targetValidatorList || !targetValidatorList.length) return
|
|
||||||
const msgHash = home.web3.utils.sha3(message.data)!
|
|
||||||
const allValidators = [...validatorList, ...targetValidatorList].filter((v, i, s) => s.indexOf(v) === i)
|
|
||||||
const manualConfirmations = []
|
|
||||||
for (let i = 0; i < allValidators.length; i++) {
|
|
||||||
try {
|
|
||||||
const overrideSignatures: {
|
|
||||||
[key: string]: string
|
|
||||||
} = require(`../snapshots/signatures_${allValidators[i]}.json`)
|
|
||||||
if (overrideSignatures[msgHash]) {
|
|
||||||
console.log(`Adding manual signature from ${allValidators[i]}`)
|
|
||||||
manualConfirmations.push({
|
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.MANUAL,
|
|
||||||
validator: allValidators[i],
|
|
||||||
timestamp: 0,
|
|
||||||
txHash: '',
|
|
||||||
signature: overrideSignatures[msgHash]
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
console.log(`No manual signature from ${allValidators[i]} was found`)
|
|
||||||
}
|
|
||||||
} catch (e) {
|
|
||||||
console.log(`Signatures overrides are not present for ${allValidators[i]}`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
setConfirmations(manualConfirmations)
|
|
||||||
}
|
|
||||||
|
|
||||||
getConfirmationsForTx(
|
getConfirmationsForTx(
|
||||||
message.data,
|
message.data,
|
||||||
home.web3,
|
home.web3,
|
||||||
@ -314,8 +284,7 @@ export const useMessageConfirmations = ({
|
|||||||
home.bridgeContract,
|
home.bridgeContract,
|
||||||
requiredSignatures,
|
requiredSignatures,
|
||||||
waitingBlocksResolved,
|
waitingBlocksResolved,
|
||||||
homeStartBlock,
|
homeStartBlock
|
||||||
targetValidatorList
|
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -374,10 +343,7 @@ export const useMessageConfirmations = ({
|
|||||||
// Sets the message status based in the collected information
|
// Sets the message status based in the collected information
|
||||||
useEffect(
|
useEffect(
|
||||||
() => {
|
() => {
|
||||||
if (
|
if (executionData.status === VALIDATOR_CONFIRMATION_STATUS.SUCCESS && existsConfirmation(confirmations)) {
|
||||||
executionData.status === VALIDATOR_CONFIRMATION_STATUS.EXECUTION_SUCCESS &&
|
|
||||||
existsConfirmation(confirmations)
|
|
||||||
) {
|
|
||||||
const newStatus = executionData.executionResult
|
const newStatus = executionData.executionResult
|
||||||
? CONFIRMATIONS_STATUS.SUCCESS
|
? CONFIRMATIONS_STATUS.SUCCESS
|
||||||
: CONFIRMATIONS_STATUS.SUCCESS_MESSAGE_FAILED
|
: CONFIRMATIONS_STATUS.SUCCESS_MESSAGE_FAILED
|
||||||
|
@ -4,13 +4,19 @@ import Web3 from 'web3'
|
|||||||
import { getRequiredSignatures, getValidatorAddress, getValidatorList } from '../utils/contract'
|
import { getRequiredSignatures, getValidatorAddress, getValidatorList } from '../utils/contract'
|
||||||
import { BRIDGE_VALIDATORS_ABI } from '../abis'
|
import { BRIDGE_VALIDATORS_ABI } from '../abis'
|
||||||
import { useStateProvider } from '../state/StateProvider'
|
import { useStateProvider } from '../state/StateProvider'
|
||||||
|
import { TransactionReceipt } from 'web3-eth'
|
||||||
import { foreignSnapshotProvider, homeSnapshotProvider, SnapshotProvider } from '../services/SnapshotProvider'
|
import { foreignSnapshotProvider, homeSnapshotProvider, SnapshotProvider } from '../services/SnapshotProvider'
|
||||||
import { FOREIGN_EXPLORER_API, HOME_EXPLORER_API } from '../config/constants'
|
import { FOREIGN_EXPLORER_API, HOME_EXPLORER_API } from '../config/constants'
|
||||||
|
|
||||||
export const useValidatorContract = (isHome: boolean, blockNumber: number | 'latest') => {
|
export interface useValidatorContractParams {
|
||||||
|
fromHome: boolean
|
||||||
|
receipt: Maybe<TransactionReceipt>
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useValidatorContract = ({ receipt, fromHome }: useValidatorContractParams) => {
|
||||||
const [validatorContract, setValidatorContract] = useState<Maybe<Contract>>(null)
|
const [validatorContract, setValidatorContract] = useState<Maybe<Contract>>(null)
|
||||||
const [requiredSignatures, setRequiredSignatures] = useState(0)
|
const [requiredSignatures, setRequiredSignatures] = useState(0)
|
||||||
const [validatorList, setValidatorList] = useState<string[]>([])
|
const [validatorList, setValidatorList] = useState([])
|
||||||
|
|
||||||
const { home, foreign } = useStateProvider()
|
const { home, foreign } = useStateProvider()
|
||||||
|
|
||||||
@ -23,34 +29,34 @@ export const useValidatorContract = (isHome: boolean, blockNumber: number | 'lat
|
|||||||
|
|
||||||
const callRequiredSignatures = async (
|
const callRequiredSignatures = async (
|
||||||
contract: Maybe<Contract>,
|
contract: Maybe<Contract>,
|
||||||
blockNumber: number | 'latest',
|
receipt: TransactionReceipt,
|
||||||
setResult: Function,
|
setResult: Function,
|
||||||
snapshotProvider: SnapshotProvider,
|
snapshotProvider: SnapshotProvider,
|
||||||
web3: Web3,
|
web3: Web3,
|
||||||
api: string
|
api: string
|
||||||
) => {
|
) => {
|
||||||
if (!contract) return
|
if (!contract) return
|
||||||
const result = await getRequiredSignatures(contract, blockNumber, snapshotProvider, web3, api)
|
const result = await getRequiredSignatures(contract, receipt.blockNumber, snapshotProvider, web3, api)
|
||||||
setResult(result)
|
setResult(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
const callValidatorList = async (
|
const callValidatorList = async (
|
||||||
contract: Maybe<Contract>,
|
contract: Maybe<Contract>,
|
||||||
blockNumber: number | 'latest',
|
receipt: TransactionReceipt,
|
||||||
setResult: Function,
|
setResult: Function,
|
||||||
snapshotProvider: SnapshotProvider,
|
snapshotProvider: SnapshotProvider,
|
||||||
web3: Web3,
|
web3: Web3,
|
||||||
api: string
|
api: string
|
||||||
) => {
|
) => {
|
||||||
if (!contract) return
|
if (!contract) return
|
||||||
const result = await getValidatorList(contract, blockNumber, snapshotProvider, web3, api)
|
const result = await getValidatorList(contract, receipt.blockNumber, snapshotProvider, web3, api)
|
||||||
setResult(result)
|
setResult(result)
|
||||||
}
|
}
|
||||||
|
|
||||||
const web3 = isHome ? home.web3 : foreign.web3
|
const web3 = fromHome ? home.web3 : foreign.web3
|
||||||
const api = isHome ? HOME_EXPLORER_API : FOREIGN_EXPLORER_API
|
const api = fromHome ? HOME_EXPLORER_API : FOREIGN_EXPLORER_API
|
||||||
const bridgeContract = isHome ? home.bridgeContract : foreign.bridgeContract
|
const bridgeContract = fromHome ? home.bridgeContract : foreign.bridgeContract
|
||||||
const snapshotProvider = isHome ? homeSnapshotProvider : foreignSnapshotProvider
|
const snapshotProvider = fromHome ? homeSnapshotProvider : foreignSnapshotProvider
|
||||||
|
|
||||||
useEffect(
|
useEffect(
|
||||||
() => {
|
() => {
|
||||||
@ -62,11 +68,11 @@ export const useValidatorContract = (isHome: boolean, blockNumber: number | 'lat
|
|||||||
|
|
||||||
useEffect(
|
useEffect(
|
||||||
() => {
|
() => {
|
||||||
if (!web3 || !blockNumber) return
|
if (!web3 || !receipt) return
|
||||||
callRequiredSignatures(validatorContract, blockNumber, setRequiredSignatures, snapshotProvider, web3, api)
|
callRequiredSignatures(validatorContract, receipt, setRequiredSignatures, snapshotProvider, web3, api)
|
||||||
callValidatorList(validatorContract, blockNumber, setValidatorList, snapshotProvider, web3, api)
|
callValidatorList(validatorContract, receipt, setValidatorList, snapshotProvider, web3, api)
|
||||||
},
|
},
|
||||||
[validatorContract, blockNumber, web3, snapshotProvider, api]
|
[validatorContract, receipt, web3, snapshotProvider, api]
|
||||||
)
|
)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import React, { createContext, ReactNode } from 'react'
|
import React, { createContext, ReactNode, useState } from 'react'
|
||||||
import { useNetwork } from '../hooks/useNetwork'
|
import { useNetwork } from '../hooks/useNetwork'
|
||||||
import {
|
import {
|
||||||
HOME_RPC_URL,
|
HOME_RPC_URL,
|
||||||
@ -25,6 +25,8 @@ export interface StateContext {
|
|||||||
home: BaseNetworkParams
|
home: BaseNetworkParams
|
||||||
foreign: BaseNetworkParams
|
foreign: BaseNetworkParams
|
||||||
loading: boolean
|
loading: boolean
|
||||||
|
error: string
|
||||||
|
setError: Function
|
||||||
}
|
}
|
||||||
|
|
||||||
const initialState = {
|
const initialState = {
|
||||||
@ -42,7 +44,9 @@ const initialState = {
|
|||||||
bridgeAddress: FOREIGN_BRIDGE_ADDRESS,
|
bridgeAddress: FOREIGN_BRIDGE_ADDRESS,
|
||||||
bridgeContract: null
|
bridgeContract: null
|
||||||
},
|
},
|
||||||
loading: true
|
loading: true,
|
||||||
|
error: '',
|
||||||
|
setError: () => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
const StateContext = createContext<StateContext>(initialState)
|
const StateContext = createContext<StateContext>(initialState)
|
||||||
@ -54,6 +58,7 @@ export const StateProvider = ({ children }: { children: ReactNode }) => {
|
|||||||
homeWeb3: homeNetwork.web3,
|
homeWeb3: homeNetwork.web3,
|
||||||
foreignWeb3: foreignNetwork.web3
|
foreignWeb3: foreignNetwork.web3
|
||||||
})
|
})
|
||||||
|
const [error, setError] = useState('')
|
||||||
|
|
||||||
const value = {
|
const value = {
|
||||||
home: {
|
home: {
|
||||||
@ -68,7 +73,9 @@ export const StateProvider = ({ children }: { children: ReactNode }) => {
|
|||||||
bridgeContract: foreignBridge,
|
bridgeContract: foreignBridge,
|
||||||
...foreignNetwork
|
...foreignNetwork
|
||||||
},
|
},
|
||||||
loading: homeNetwork.loading || foreignNetwork.loading
|
loading: homeNetwork.loading || foreignNetwork.loading,
|
||||||
|
error,
|
||||||
|
setError
|
||||||
}
|
}
|
||||||
|
|
||||||
return <StateContext.Provider value={value}>{children}</StateContext.Provider>
|
return <StateContext.Provider value={value}>{children}</StateContext.Provider>
|
||||||
|
@ -28,7 +28,5 @@ export const GlobalStyle = createGlobalStyle<{ theme: ThemeType }>`
|
|||||||
--not-required-bg-color: ${props => props.theme.notRequired.backgroundColor};
|
--not-required-bg-color: ${props => props.theme.notRequired.backgroundColor};
|
||||||
--failed-color: ${props => props.theme.failed.textColor};
|
--failed-color: ${props => props.theme.failed.textColor};
|
||||||
--failed-bg-color: ${props => props.theme.failed.backgroundColor};
|
--failed-bg-color: ${props => props.theme.failed.backgroundColor};
|
||||||
--warning-color: ${props => props.theme.warning.textColor};
|
|
||||||
--warning-bg-color: ${props => props.theme.warning.backgroundColor};
|
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
@ -17,10 +17,6 @@ const theme = {
|
|||||||
failed: {
|
failed: {
|
||||||
textColor: '#de4437',
|
textColor: '#de4437',
|
||||||
backgroundColor: 'rgba(222,68,55,.1)'
|
backgroundColor: 'rgba(222,68,55,.1)'
|
||||||
},
|
|
||||||
warning: {
|
|
||||||
textColor: '#ffa758',
|
|
||||||
backgroundColor: 'rgba(222,68,55,.1)'
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
export default theme
|
export default theme
|
||||||
|
@ -14,40 +14,37 @@ const messageData = '0x123456'
|
|||||||
const OTHER_HASH = 'aabbccdd'
|
const OTHER_HASH = 'aabbccdd'
|
||||||
const bridgeAddress = '0xFe446bEF1DbF7AFE24E81e05BC8B271C1BA9a560'
|
const bridgeAddress = '0xFe446bEF1DbF7AFE24E81e05BC8B271C1BA9a560'
|
||||||
const otherAddress = '0xD4075FB57fCf038bFc702c915Ef9592534bED5c1'
|
const otherAddress = '0xD4075FB57fCf038bFc702c915Ef9592534bED5c1'
|
||||||
const validator1 = '0x45b96809336A8b714BFbdAB3E4B5e0fe5d839908'
|
|
||||||
const validator2 = '0xAe8bFfc8BBc6AAa9E21ED1E4e4957fe798BEA25f'
|
|
||||||
const validator3 = '0x285A6eB779be4db94dA65e2F3518B1c5F0f71244'
|
|
||||||
|
|
||||||
describe('getFailedTransactions', () => {
|
describe('getFailedTransactions', () => {
|
||||||
test('should only return failed transactions', async () => {
|
test('should only return failed transactions', async () => {
|
||||||
const to = otherAddress
|
const to = otherAddress
|
||||||
const transactions = [
|
const transactions = [
|
||||||
{ isError: '0', to, from: validator1 },
|
{ isError: '0', to },
|
||||||
{ isError: '1', to, from: validator1 },
|
{ isError: '1', to },
|
||||||
{ isError: '0', to, from: validator2 },
|
{ isError: '0', to },
|
||||||
{ isError: '1', to, from: validator2 },
|
{ isError: '1', to },
|
||||||
{ isError: '1', to, from: validator3 }
|
{ isError: '1', to }
|
||||||
]
|
]
|
||||||
|
|
||||||
const fetchAccountTransactions = jest.fn().mockImplementation(() => transactions)
|
const fetchAccountTransactions = jest.fn().mockImplementation(() => transactions)
|
||||||
const result = await getFailedTransactions(validator1, to, 0, 1, '', fetchAccountTransactions)
|
const result = await getFailedTransactions('', to, 0, 1, '', fetchAccountTransactions)
|
||||||
expect(result.length).toEqual(1)
|
expect(result.length).toEqual(3)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
describe('getSuccessTransactions', () => {
|
describe('getSuccessTransactions', () => {
|
||||||
test('should only return success transactions', async () => {
|
test('should only return success transactions', async () => {
|
||||||
const to = otherAddress
|
const to = otherAddress
|
||||||
const transactions = [
|
const transactions = [
|
||||||
{ isError: '0', to, from: validator1 },
|
{ isError: '0', to },
|
||||||
{ isError: '1', to, from: validator1 },
|
{ isError: '1', to },
|
||||||
{ isError: '0', to, from: validator2 },
|
{ isError: '0', to },
|
||||||
{ isError: '1', to, from: validator2 },
|
{ isError: '1', to },
|
||||||
{ isError: '1', to, from: validator3 }
|
{ isError: '1', to }
|
||||||
]
|
]
|
||||||
|
|
||||||
const fetchAccountTransactions = jest.fn().mockImplementation(() => transactions)
|
const fetchAccountTransactions = jest.fn().mockImplementation(() => transactions)
|
||||||
const result = await getSuccessTransactions(validator1, to, 0, 1, '', fetchAccountTransactions)
|
const result = await getSuccessTransactions('', to, 0, 1, '', fetchAccountTransactions)
|
||||||
expect(result.length).toEqual(1)
|
expect(result.length).toEqual(2)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
describe('filterValidatorSignatureTransaction', () => {
|
describe('filterValidatorSignatureTransaction', () => {
|
||||||
|
@ -5,7 +5,7 @@ import Web3 from 'web3'
|
|||||||
import { Contract } from 'web3-eth-contract'
|
import { Contract } from 'web3-eth-contract'
|
||||||
import { APIPendingTransaction, APITransaction } from '../explorer'
|
import { APIPendingTransaction, APITransaction } from '../explorer'
|
||||||
import { VALIDATOR_CONFIRMATION_STATUS } from '../../config/constants'
|
import { VALIDATOR_CONFIRMATION_STATUS } from '../../config/constants'
|
||||||
import { ConfirmationParam } from '../../hooks/useMessageConfirmations'
|
import { BasicConfirmationParam } from '../../hooks/useMessageConfirmations'
|
||||||
|
|
||||||
jest.mock('../validatorConfirmationHelpers')
|
jest.mock('../validatorConfirmationHelpers')
|
||||||
|
|
||||||
@ -18,9 +18,6 @@ const messageData = '0x111111111'
|
|||||||
const web3 = {
|
const web3 = {
|
||||||
utils: {
|
utils: {
|
||||||
soliditySha3Raw: (data: string) => `0xaaaa${data.replace('0x', '')}`
|
soliditySha3Raw: (data: string) => `0xaaaa${data.replace('0x', '')}`
|
||||||
},
|
|
||||||
eth: {
|
|
||||||
accounts: new Web3().eth.accounts
|
|
||||||
}
|
}
|
||||||
} as Web3
|
} as Web3
|
||||||
const validator1 = '0x45b96809336A8b714BFbdAB3E4B5e0fe5d839908'
|
const validator1 = '0x45b96809336A8b714BFbdAB3E4B5e0fe5d839908'
|
||||||
@ -28,7 +25,7 @@ const validator2 = '0xAe8bFfc8BBc6AAa9E21ED1E4e4957fe798BEA25f'
|
|||||||
const validator3 = '0x285A6eB779be4db94dA65e2F3518B1c5F0f71244'
|
const validator3 = '0x285A6eB779be4db94dA65e2F3518B1c5F0f71244'
|
||||||
const validatorList = [validator1, validator2, validator3]
|
const validatorList = [validator1, validator2, validator3]
|
||||||
const signature =
|
const signature =
|
||||||
'0x6f5b74905669999f1abdb52e1e215506907e1849aac7b31854da458b33a5954e15b165007c3703cfd16e61ca46a96a56727ed11fa47be359d3834515accd016e1b'
|
'0x519d704bceed17423daa79c20531cc34fc27a4be6e53fc5069a8023019188ca4519d704bceed17423daa79c20531cc34fc27a4be6e53fc5069a8023019188ca4'
|
||||||
const bridgeContract = {
|
const bridgeContract = {
|
||||||
methods: {
|
methods: {
|
||||||
signature: () => ({
|
signature: () => ({
|
||||||
@ -64,19 +61,19 @@ describe('getConfirmationsForTx', () => {
|
|||||||
validator,
|
validator,
|
||||||
status: validator !== validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
status: validator !== validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0
|
timestamp: 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0
|
timestamp: 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -113,8 +110,9 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect(setResult).toBeCalledTimes(2)
|
expect(setResult).toBeCalledTimes(2)
|
||||||
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
||||||
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
||||||
expect(setSignatureCollected).toBeCalledTimes(1)
|
expect(setSignatureCollected).toBeCalledTimes(2)
|
||||||
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
||||||
|
expect(setSignatureCollected.mock.calls[1][0]).toEqual([signature, signature])
|
||||||
|
|
||||||
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
||||||
expect(setFailedConfirmations).toBeCalledTimes(1)
|
expect(setFailedConfirmations).toBeCalledTimes(1)
|
||||||
@ -137,7 +135,7 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED, txHash: '', timestamp: 0 }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@ -146,19 +144,19 @@ describe('getConfirmationsForTx', () => {
|
|||||||
validator,
|
validator,
|
||||||
status: validator === validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
status: validator === validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0
|
timestamp: 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0
|
timestamp: 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -210,19 +208,19 @@ describe('getConfirmationsForTx', () => {
|
|||||||
validator,
|
validator,
|
||||||
status: validator !== validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
status: validator !== validator3 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: validatorData.validator !== validator3 ? '0x123' : '',
|
txHash: validatorData.validator !== validator3 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator !== validator3 ? 123 : 0
|
timestamp: validatorData.validator !== validator3 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
timestamp: 0
|
timestamp: 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -259,8 +257,9 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect(setResult).toBeCalledTimes(3)
|
expect(setResult).toBeCalledTimes(3)
|
||||||
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
||||||
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
||||||
expect(setSignatureCollected).toBeCalledTimes(1)
|
expect(setSignatureCollected).toBeCalledTimes(2)
|
||||||
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
||||||
|
expect(setSignatureCollected.mock.calls[1][0]).toEqual([signature, signature])
|
||||||
|
|
||||||
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
||||||
expect(setFailedConfirmations).toBeCalledTimes(1)
|
expect(setFailedConfirmations).toBeCalledTimes(1)
|
||||||
@ -282,16 +281,16 @@ describe('getConfirmationsForTx', () => {
|
|||||||
)
|
)
|
||||||
expect(res2).toEqual(
|
expect(res2).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res3).toEqual(
|
expect(res3).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED, txHash: '', timestamp: 0 }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@ -305,22 +304,22 @@ describe('getConfirmationsForTx', () => {
|
|||||||
? VALIDATOR_CONFIRMATION_STATUS.SUCCESS
|
? VALIDATOR_CONFIRMATION_STATUS.SUCCESS
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: validatorData.validator !== validator3 && validatorData.validator !== validator4 ? '0x123' : '',
|
txHash: validatorData.validator !== validator3 && validatorData.validator !== validator4 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator !== validator3 && validatorData.validator !== validator4 ? 123 : 0
|
timestamp: validatorData.validator !== validator3 && validatorData.validator !== validator4 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status:
|
status:
|
||||||
validatorData.validator === validator3
|
validatorData.validator === validator3
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
? VALIDATOR_CONFIRMATION_STATUS.FAILED
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: validatorData.validator === validator3 ? '0x123' : '',
|
txHash: validatorData.validator === validator3 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator3 ? 123 : 0
|
timestamp: validatorData.validator === validator3 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -357,8 +356,9 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect(setResult).toBeCalledTimes(4)
|
expect(setResult).toBeCalledTimes(4)
|
||||||
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
expect(getValidatorConfirmation).toBeCalledTimes(1)
|
||||||
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
expect(getSuccessExecutionTransaction).toBeCalledTimes(1)
|
||||||
expect(setSignatureCollected).toBeCalledTimes(1)
|
expect(setSignatureCollected).toBeCalledTimes(2)
|
||||||
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
expect(setSignatureCollected.mock.calls[0][0]).toEqual(true)
|
||||||
|
expect(setSignatureCollected.mock.calls[1][0]).toEqual([signature, signature])
|
||||||
|
|
||||||
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
||||||
expect(setFailedConfirmations).toBeCalledTimes(1)
|
expect(setFailedConfirmations).toBeCalledTimes(1)
|
||||||
@ -382,26 +382,26 @@ describe('getConfirmationsForTx', () => {
|
|||||||
)
|
)
|
||||||
expect(res2).toEqual(
|
expect(res2).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res3).toEqual(
|
expect(res3).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID, txHash: '0x123', timestamp: 123 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res4).toEqual(
|
expect(res4).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID, txHash: '0x123', timestamp: 123 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED, txHash: '', timestamp: 0 }
|
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@ -414,22 +414,22 @@ describe('getConfirmationsForTx', () => {
|
|||||||
validator,
|
validator,
|
||||||
status: validator === validator1 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
status: validator === validator1 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: validatorData.validator === validator1 ? '0x123' : '',
|
txHash: validatorData.validator === validator1 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator1 ? 123 : 0
|
timestamp: validatorData.validator === validator1 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status:
|
status:
|
||||||
validatorData.validator === validator2
|
validatorData.validator === validator2
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
? VALIDATOR_CONFIRMATION_STATUS.FAILED
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: validatorData.validator === validator2 ? '0x123' : '',
|
txHash: validatorData.validator === validator2 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator2 ? 123 : 0
|
timestamp: validatorData.validator === validator2 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status:
|
status:
|
||||||
validatorData.validator === validator3
|
validatorData.validator === validator3
|
||||||
@ -492,22 +492,22 @@ describe('getConfirmationsForTx', () => {
|
|||||||
)
|
)
|
||||||
expect(res2).toEqual(
|
expect(res2).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res3).toEqual(
|
expect(res3).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res4).toEqual(
|
expect(res4).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID, txHash: '0x123', timestamp: 123 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
@ -521,13 +521,13 @@ describe('getConfirmationsForTx', () => {
|
|||||||
validator,
|
validator,
|
||||||
status: validator === validator1 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
status: validator === validator1 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getSuccessExecutionTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: validatorData.validator === validator1 ? '0x123' : '',
|
txHash: validatorData.validator === validator1 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator1 ? 123 : 0
|
timestamp: validatorData.validator === validator1 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status:
|
status:
|
||||||
validatorData.validator !== validator1
|
validatorData.validator !== validator1
|
||||||
@ -536,7 +536,7 @@ describe('getConfirmationsForTx', () => {
|
|||||||
txHash: validatorData.validator !== validator1 ? '0x123' : '',
|
txHash: validatorData.validator !== validator1 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator !== validator1 ? 123 : 0
|
timestamp: validatorData.validator !== validator1 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
getValidatorPendingTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -596,9 +596,9 @@ describe('getConfirmationsForTx', () => {
|
|||||||
)
|
)
|
||||||
expect(res2).toEqual(
|
expect(res2).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 }
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res3).toEqual(
|
expect(res3).toEqual(
|
||||||
@ -610,13 +610,9 @@ describe('getConfirmationsForTx', () => {
|
|||||||
)
|
)
|
||||||
})
|
})
|
||||||
test('should remove pending state after transaction mined', async () => {
|
test('should remove pending state after transaction mined', async () => {
|
||||||
const validator4 = '0x9d2dC11C342F4eF3C5491A048D0f0eBCd2D8f7C3'
|
// Validator1 success
|
||||||
const validatorList = [validator1, validator2, validator3, validator4]
|
// Validator2 failed
|
||||||
|
// Validator3 Pending
|
||||||
// Validator1 success (ts=100)
|
|
||||||
// Validator2 failed (ts=200)
|
|
||||||
// Validator3 Pending (ts=300)
|
|
||||||
// Validator4 Excess confirmation (Failed) (ts=400)
|
|
||||||
|
|
||||||
getValidatorConfirmation
|
getValidatorConfirmation
|
||||||
.mockImplementationOnce(() => async (validator: string) => ({
|
.mockImplementationOnce(() => async (validator: string) => ({
|
||||||
@ -627,57 +623,41 @@ describe('getConfirmationsForTx', () => {
|
|||||||
.mockImplementation(() => async (validator: string) => ({
|
.mockImplementation(() => async (validator: string) => ({
|
||||||
validator,
|
validator,
|
||||||
status:
|
status:
|
||||||
validator === validator1 || validator === validator3
|
validator !== validator2 ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.SUCCESS
|
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
|
||||||
}))
|
}))
|
||||||
getSuccessExecutionTransaction
|
getSuccessExecutionTransaction
|
||||||
.mockImplementationOnce(() => async (validatorData: ConfirmationParam) => ({
|
.mockImplementationOnce(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash: validatorData.validator === validator1 ? '0x100' : '',
|
txHash: validatorData.validator === validator1 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator1 ? 100 : 0
|
timestamp: validatorData.validator === validator1 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash:
|
txHash: validatorData.validator !== validator2 ? '0x123' : '',
|
||||||
validatorData.validator === validator1 ? '0x100' : validatorData.validator === validator3 ? '0x300' : '',
|
timestamp: validatorData.validator !== validator2 ? 123 : 0
|
||||||
timestamp: validatorData.validator === validator1 ? 100 : validatorData.validator === validator3 ? 300 : ''
|
|
||||||
}))
|
|
||||||
getValidatorFailedTransaction
|
|
||||||
.mockImplementationOnce(() => async (validatorData: ConfirmationParam) => ({
|
|
||||||
validator: validatorData.validator,
|
|
||||||
status:
|
|
||||||
validatorData.validator === validator2
|
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.FAILED
|
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
|
||||||
txHash: validatorData.validator === validator2 ? '0x200' : '',
|
|
||||||
timestamp: validatorData.validator === validator2 ? 200 : 0
|
|
||||||
}))
|
|
||||||
.mockImplementation(() => async (validatorData: ConfirmationParam) => ({
|
|
||||||
validator: validatorData.validator,
|
|
||||||
status:
|
|
||||||
validatorData.validator === validator2 || validatorData.validator === validator4
|
|
||||||
? validatorData.validator === validator2
|
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.FAILED
|
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
|
||||||
txHash:
|
|
||||||
validatorData.validator === validator2 ? '0x200' : validatorData.validator === validator4 ? '0x400' : '',
|
|
||||||
timestamp: validatorData.validator === validator2 ? 200 : validatorData.validator === validator4 ? 400 : ''
|
|
||||||
}))
|
}))
|
||||||
|
getValidatorFailedTransaction.mockImplementation(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
|
validator: validatorData.validator,
|
||||||
|
status:
|
||||||
|
validatorData.validator === validator2
|
||||||
|
? VALIDATOR_CONFIRMATION_STATUS.FAILED
|
||||||
|
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
|
txHash: validatorData.validator === validator2 ? '0x123' : '',
|
||||||
|
timestamp: validatorData.validator === validator2 ? 123 : 0
|
||||||
|
}))
|
||||||
getValidatorPendingTransaction
|
getValidatorPendingTransaction
|
||||||
.mockImplementationOnce(() => async (validatorData: ConfirmationParam) => ({
|
.mockImplementationOnce(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status:
|
status:
|
||||||
validatorData.validator === validator3
|
validatorData.validator === validator3
|
||||||
? VALIDATOR_CONFIRMATION_STATUS.PENDING
|
? VALIDATOR_CONFIRMATION_STATUS.PENDING
|
||||||
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: validatorData.validator === validator3 ? '0x300' : '',
|
txHash: validatorData.validator === validator3 ? '0x123' : '',
|
||||||
timestamp: validatorData.validator === validator3 ? 300 : 0
|
timestamp: validatorData.validator === validator3 ? 123 : 0
|
||||||
}))
|
}))
|
||||||
.mockImplementationOnce(() => async (validatorData: ConfirmationParam) => ({
|
.mockImplementationOnce(() => async (validatorData: BasicConfirmationParam) => ({
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
||||||
txHash: '',
|
txHash: '',
|
||||||
@ -718,7 +698,7 @@ describe('getConfirmationsForTx', () => {
|
|||||||
|
|
||||||
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
expect(getValidatorFailedTransaction).toBeCalledTimes(1)
|
||||||
expect(setFailedConfirmations).toBeCalledTimes(1)
|
expect(setFailedConfirmations).toBeCalledTimes(1)
|
||||||
expect(setFailedConfirmations.mock.calls[0][0]).toEqual(true)
|
expect(setFailedConfirmations.mock.calls[0][0]).toEqual(false)
|
||||||
|
|
||||||
expect(getValidatorPendingTransaction).toBeCalledTimes(1)
|
expect(getValidatorPendingTransaction).toBeCalledTimes(1)
|
||||||
expect(setPendingConfirmations).toBeCalledTimes(1)
|
expect(setPendingConfirmations).toBeCalledTimes(1)
|
||||||
@ -732,32 +712,28 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res2).toEqual(
|
expect(res2).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res3).toEqual(
|
expect(res3).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x300', timestamp: 300 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res4).toEqual(
|
expect(res4).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x200', timestamp: 200 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x300', timestamp: 300 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.PENDING, txHash: '0x123', timestamp: 123 }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -785,13 +761,14 @@ describe('getConfirmationsForTx', () => {
|
|||||||
expect(setResult).toBeCalledTimes(7)
|
expect(setResult).toBeCalledTimes(7)
|
||||||
expect(getValidatorConfirmation).toBeCalledTimes(2)
|
expect(getValidatorConfirmation).toBeCalledTimes(2)
|
||||||
expect(getSuccessExecutionTransaction).toBeCalledTimes(2)
|
expect(getSuccessExecutionTransaction).toBeCalledTimes(2)
|
||||||
expect(setSignatureCollected).toBeCalledTimes(2)
|
expect(setSignatureCollected).toBeCalledTimes(3)
|
||||||
expect(setSignatureCollected.mock.calls[0][0]).toEqual(false)
|
expect(setSignatureCollected.mock.calls[0][0]).toEqual(false)
|
||||||
expect(setSignatureCollected.mock.calls[1][0]).toEqual(true)
|
expect(setSignatureCollected.mock.calls[1][0]).toEqual(true)
|
||||||
|
expect(setSignatureCollected.mock.calls[2][0]).toEqual([signature, signature])
|
||||||
|
|
||||||
expect(getValidatorFailedTransaction).toBeCalledTimes(2)
|
expect(getValidatorFailedTransaction).toBeCalledTimes(2)
|
||||||
expect(setFailedConfirmations).toBeCalledTimes(2)
|
expect(setFailedConfirmations).toBeCalledTimes(2)
|
||||||
expect(setFailedConfirmations.mock.calls[0][0]).toEqual(true)
|
expect(setFailedConfirmations.mock.calls[0][0]).toEqual(false)
|
||||||
expect(setFailedConfirmations.mock.calls[1][0]).toEqual(false)
|
expect(setFailedConfirmations.mock.calls[1][0]).toEqual(false)
|
||||||
|
|
||||||
expect(getValidatorPendingTransaction).toBeCalledTimes(1)
|
expect(getValidatorPendingTransaction).toBeCalledTimes(1)
|
||||||
@ -804,26 +781,23 @@ describe('getConfirmationsForTx', () => {
|
|||||||
const res7 = setResult.mock.calls[6][0](res6)
|
const res7 = setResult.mock.calls[6][0](res6)
|
||||||
expect(res5).toEqual(
|
expect(res5).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x200', timestamp: 200 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x300', timestamp: 300 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res6).toEqual(
|
expect(res6).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x200', timestamp: 200 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x300', timestamp: 300 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
expect(res7).toEqual(
|
expect(res7).toEqual(
|
||||||
expect.arrayContaining([
|
expect.arrayContaining([
|
||||||
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x100', timestamp: 100 },
|
{ validator: validator1, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x200', timestamp: 200 },
|
{ validator: validator2, status: VALIDATOR_CONFIRMATION_STATUS.FAILED, txHash: '0x123', timestamp: 123 },
|
||||||
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x300', timestamp: 300 },
|
{ validator: validator3, status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS, txHash: '0x123', timestamp: 123 }
|
||||||
{ validator: validator4, status: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID, txHash: '0x400', timestamp: 400 }
|
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
@ -84,11 +84,10 @@ describe('getFinalizationEvent', () => {
|
|||||||
expect(setResult).toBeCalledTimes(1)
|
expect(setResult).toBeCalledTimes(1)
|
||||||
expect(setResult.mock.calls[0][0]).toEqual({
|
expect(setResult.mock.calls[0][0]).toEqual({
|
||||||
validator: validator1,
|
validator: validator1,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.EXECUTION_SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
txHash,
|
txHash,
|
||||||
timestamp,
|
timestamp,
|
||||||
executionResult: true,
|
executionResult: true
|
||||||
blockNumber: 5523145
|
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(getFailedExecution).toBeCalledTimes(0)
|
expect(getFailedExecution).toBeCalledTimes(0)
|
||||||
@ -238,8 +237,7 @@ describe('getFinalizationEvent', () => {
|
|||||||
status: VALIDATOR_CONFIRMATION_STATUS.PENDING,
|
status: VALIDATOR_CONFIRMATION_STATUS.PENDING,
|
||||||
txHash,
|
txHash,
|
||||||
timestamp: expect.any(Number),
|
timestamp: expect.any(Number),
|
||||||
executionResult: false,
|
executionResult: false
|
||||||
blockNumber: 0
|
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(getFailedExecution).toBeCalledTimes(0)
|
expect(getFailedExecution).toBeCalledTimes(0)
|
||||||
@ -296,8 +294,7 @@ describe('getFinalizationEvent', () => {
|
|||||||
status: VALIDATOR_CONFIRMATION_STATUS.FAILED,
|
status: VALIDATOR_CONFIRMATION_STATUS.FAILED,
|
||||||
txHash,
|
txHash,
|
||||||
timestamp: expect.any(Number),
|
timestamp: expect.any(Number),
|
||||||
executionResult: false,
|
executionResult: false
|
||||||
blockNumber: expect.any(Number)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(getFailedExecution).toBeCalledTimes(1)
|
expect(getFailedExecution).toBeCalledTimes(1)
|
||||||
|
@ -22,16 +22,6 @@ export const getRequiredBlockConfirmations = async (
|
|||||||
web3: Web3 | null = null,
|
web3: Web3 | null = null,
|
||||||
api: string = ''
|
api: string = ''
|
||||||
) => {
|
) => {
|
||||||
let blockConfirmations
|
|
||||||
|
|
||||||
try {
|
|
||||||
blockConfirmations = await contract.methods.requiredBlockConfirmations().call()
|
|
||||||
} catch {}
|
|
||||||
|
|
||||||
if (blockConfirmations) {
|
|
||||||
return parseInt(blockConfirmations)
|
|
||||||
}
|
|
||||||
|
|
||||||
const eventsFromSnapshot = snapshotProvider.requiredBlockConfirmationEvents(blockNumber)
|
const eventsFromSnapshot = snapshotProvider.requiredBlockConfirmationEvents(blockNumber)
|
||||||
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
||||||
|
|
||||||
@ -45,10 +35,16 @@ export const getRequiredBlockConfirmations = async (
|
|||||||
|
|
||||||
const events = [...eventsFromSnapshot, ...contractEvents]
|
const events = [...eventsFromSnapshot, ...contractEvents]
|
||||||
|
|
||||||
// Use the value from last event before the transaction
|
let blockConfirmations
|
||||||
const event = events[events.length - 1]
|
if (events.length > 0) {
|
||||||
blockConfirmations = event.returnValues.requiredBlockConfirmations
|
// Use the value from last event before the transaction
|
||||||
|
const event = events[events.length - 1]
|
||||||
|
blockConfirmations = event.returnValues.requiredBlockConfirmations
|
||||||
|
} else {
|
||||||
|
// This is a special case where RequiredBlockConfirmationChanged was not emitted during initialization in early versions of AMB
|
||||||
|
// of Sokol - Kovan. In this case the current value is used.
|
||||||
|
blockConfirmations = await contract.methods.requiredBlockConfirmations().call()
|
||||||
|
}
|
||||||
return parseInt(blockConfirmations)
|
return parseInt(blockConfirmations)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -56,25 +52,11 @@ export const getValidatorAddress = (contract: Contract) => contract.methods.vali
|
|||||||
|
|
||||||
export const getRequiredSignatures = async (
|
export const getRequiredSignatures = async (
|
||||||
contract: Contract,
|
contract: Contract,
|
||||||
blockNumber: number | 'latest',
|
blockNumber: number,
|
||||||
snapshotProvider: SnapshotProvider,
|
snapshotProvider: SnapshotProvider,
|
||||||
web3: Web3 | null = null,
|
web3: Web3 | null = null,
|
||||||
api: string = ''
|
api: string = ''
|
||||||
) => {
|
) => {
|
||||||
let requiredSignatures
|
|
||||||
|
|
||||||
try {
|
|
||||||
requiredSignatures = await contract.methods.requiredSignatures().call()
|
|
||||||
} catch {}
|
|
||||||
|
|
||||||
if (requiredSignatures) {
|
|
||||||
return parseInt(requiredSignatures)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (blockNumber === 'latest') {
|
|
||||||
return contract.methods.requiredSignatures().call()
|
|
||||||
}
|
|
||||||
|
|
||||||
const eventsFromSnapshot = snapshotProvider.requiredSignaturesEvents(blockNumber)
|
const eventsFromSnapshot = snapshotProvider.requiredSignaturesEvents(blockNumber)
|
||||||
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
||||||
|
|
||||||
@ -90,25 +72,17 @@ export const getRequiredSignatures = async (
|
|||||||
|
|
||||||
// Use the value form last event before the transaction
|
// Use the value form last event before the transaction
|
||||||
const event = events[events.length - 1]
|
const event = events[events.length - 1]
|
||||||
;({ requiredSignatures } = event.returnValues)
|
const { requiredSignatures } = event.returnValues
|
||||||
return parseInt(requiredSignatures)
|
return parseInt(requiredSignatures)
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getValidatorList = async (
|
export const getValidatorList = async (
|
||||||
contract: Contract,
|
contract: Contract,
|
||||||
blockNumber: number | 'latest',
|
blockNumber: number,
|
||||||
snapshotProvider: SnapshotProvider,
|
snapshotProvider: SnapshotProvider,
|
||||||
web3: Web3 | null = null,
|
web3: Web3 | null = null,
|
||||||
api: string = ''
|
api: string = ''
|
||||||
) => {
|
) => {
|
||||||
try {
|
|
||||||
const currentList = await contract.methods.validatorList().call()
|
|
||||||
|
|
||||||
if (currentList) {
|
|
||||||
return currentList
|
|
||||||
}
|
|
||||||
} catch {}
|
|
||||||
|
|
||||||
const addedEventsFromSnapshot = snapshotProvider.validatorAddedEvents(blockNumber)
|
const addedEventsFromSnapshot = snapshotProvider.validatorAddedEvents(blockNumber)
|
||||||
const removedEventsFromSnapshot = snapshotProvider.validatorRemovedEvents(blockNumber)
|
const removedEventsFromSnapshot = snapshotProvider.validatorRemovedEvents(blockNumber)
|
||||||
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
const snapshotBlockNumber = snapshotProvider.snapshotBlockNumber()
|
||||||
|
@ -12,7 +12,6 @@ import Web3 from 'web3'
|
|||||||
import { Contract } from 'web3-eth-contract'
|
import { Contract } from 'web3-eth-contract'
|
||||||
|
|
||||||
export interface APITransaction {
|
export interface APITransaction {
|
||||||
from: string
|
|
||||||
timeStamp: string
|
timeStamp: string
|
||||||
isError: string
|
isError: string
|
||||||
input: string
|
input: string
|
||||||
@ -55,7 +54,7 @@ export const fetchAccountTransactions = async ({ account, startBlock, endBlock,
|
|||||||
url.searchParams.append('module', 'account')
|
url.searchParams.append('module', 'account')
|
||||||
url.searchParams.append('action', 'txlist')
|
url.searchParams.append('action', 'txlist')
|
||||||
url.searchParams.append('address', account)
|
url.searchParams.append('address', account)
|
||||||
url.searchParams.append('filterby', 'to')
|
url.searchParams.append('filterby', 'from')
|
||||||
url.searchParams.append('startblock', startBlock.toString())
|
url.searchParams.append('startblock', startBlock.toString())
|
||||||
url.searchParams.append('endblock', endBlock.toString())
|
url.searchParams.append('endblock', endBlock.toString())
|
||||||
|
|
||||||
@ -65,7 +64,7 @@ export const fetchAccountTransactions = async ({ account, startBlock, endBlock,
|
|||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
|
|
||||||
return result.result || []
|
return result.result
|
||||||
}
|
}
|
||||||
|
|
||||||
export const fetchPendingTransactions = async ({
|
export const fetchPendingTransactions = async ({
|
||||||
@ -181,9 +180,7 @@ export const getLogs = async (
|
|||||||
if (topics[i] !== null) {
|
if (topics[i] !== null) {
|
||||||
url.searchParams.append(`topic${i}`, topics[i] as string)
|
url.searchParams.append(`topic${i}`, topics[i] as string)
|
||||||
for (let j = 0; j < i; j++) {
|
for (let j = 0; j < i; j++) {
|
||||||
if (topics[j] !== null) {
|
url.searchParams.append(`topic${j}_${i}_opr`, 'and')
|
||||||
url.searchParams.append(`topic${j}_${i}_opr`, 'and')
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -197,7 +194,7 @@ export const getLogs = async (
|
|||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
|
||||||
const filterSender = (from: string) => (tx: APITransaction) => tx.from.toLowerCase() === from.toLowerCase()
|
const filterReceiver = (to: string) => (tx: APITransaction) => tx.to.toLowerCase() === to.toLowerCase()
|
||||||
|
|
||||||
export const getFailedTransactions = async (
|
export const getFailedTransactions = async (
|
||||||
account: string,
|
account: string,
|
||||||
@ -207,9 +204,9 @@ export const getFailedTransactions = async (
|
|||||||
api: string,
|
api: string,
|
||||||
getAccountTransactionsMethod = getAccountTransactions
|
getAccountTransactionsMethod = getAccountTransactions
|
||||||
): Promise<APITransaction[]> => {
|
): Promise<APITransaction[]> => {
|
||||||
const transactions = await getAccountTransactionsMethod({ account: to, startBlock, endBlock, api })
|
const transactions = await getAccountTransactionsMethod({ account, startBlock, endBlock, api })
|
||||||
|
|
||||||
return transactions.filter(t => t.isError !== '0').filter(filterSender(account))
|
return transactions.filter(t => t.isError !== '0').filter(filterReceiver(to))
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getSuccessTransactions = async (
|
export const getSuccessTransactions = async (
|
||||||
@ -220,9 +217,9 @@ export const getSuccessTransactions = async (
|
|||||||
api: string,
|
api: string,
|
||||||
getAccountTransactionsMethod = getAccountTransactions
|
getAccountTransactionsMethod = getAccountTransactions
|
||||||
): Promise<APITransaction[]> => {
|
): Promise<APITransaction[]> => {
|
||||||
const transactions = await getAccountTransactionsMethod({ account: to, startBlock, endBlock, api })
|
const transactions = await getAccountTransactionsMethod({ account, startBlock, endBlock, api })
|
||||||
|
|
||||||
return transactions.filter(t => t.isError === '0').filter(filterSender(account))
|
return transactions.filter(t => t.isError === '0').filter(filterReceiver(to))
|
||||||
}
|
}
|
||||||
|
|
||||||
export const filterValidatorSignatureTransaction = (
|
export const filterValidatorSignatureTransaction = (
|
||||||
|
@ -2,37 +2,26 @@ import Web3 from 'web3'
|
|||||||
import { Contract } from 'web3-eth-contract'
|
import { Contract } from 'web3-eth-contract'
|
||||||
import { HOME_RPC_POLLING_INTERVAL, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
import { HOME_RPC_POLLING_INTERVAL, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
||||||
import { GetTransactionParams, APITransaction, APIPendingTransaction, GetPendingTransactionParams } from './explorer'
|
import { GetTransactionParams, APITransaction, APIPendingTransaction, GetPendingTransactionParams } from './explorer'
|
||||||
|
import { getAffirmationsSigned, getMessagesSigned } from './contract'
|
||||||
import {
|
import {
|
||||||
getValidatorConfirmation,
|
getValidatorConfirmation,
|
||||||
getValidatorFailedTransaction,
|
getValidatorFailedTransaction,
|
||||||
getValidatorPendingTransaction,
|
getValidatorPendingTransaction,
|
||||||
getSuccessExecutionTransaction
|
getSuccessExecutionTransaction
|
||||||
} from './validatorConfirmationHelpers'
|
} from './validatorConfirmationHelpers'
|
||||||
import { ConfirmationParam } from '../hooks/useMessageConfirmations'
|
import { BasicConfirmationParam, ConfirmationParam } from '../hooks/useMessageConfirmations'
|
||||||
import { signatureToVRS } from './signatures'
|
|
||||||
|
|
||||||
const mergeConfirmations = (oldConfirmations: ConfirmationParam[], newConfirmations: ConfirmationParam[]) => {
|
const mergeConfirmations = (oldConfirmations: BasicConfirmationParam[], newConfirmations: BasicConfirmationParam[]) => {
|
||||||
const confirmations = [...oldConfirmations]
|
const confirmations = [...oldConfirmations]
|
||||||
newConfirmations.forEach(validatorData => {
|
newConfirmations.forEach(validatorData => {
|
||||||
const index = confirmations.findIndex(e => e.validator === validatorData.validator)
|
const index = confirmations.findIndex(e => e.validator === validatorData.validator)
|
||||||
if (index === -1) {
|
|
||||||
confirmations.push(validatorData)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
const currentStatus = confirmations[index].status
|
const currentStatus = confirmations[index].status
|
||||||
const newStatus = validatorData.status
|
const newStatus = validatorData.status
|
||||||
if (
|
if (
|
||||||
validatorData.txHash ||
|
(validatorData as ConfirmationParam).txHash ||
|
||||||
!!validatorData.signature ||
|
|
||||||
(newStatus !== currentStatus && newStatus !== VALIDATOR_CONFIRMATION_STATUS.UNDEFINED)
|
(newStatus !== currentStatus && newStatus !== VALIDATOR_CONFIRMATION_STATUS.UNDEFINED)
|
||||||
) {
|
) {
|
||||||
confirmations[index] = {
|
confirmations[index] = validatorData
|
||||||
status: validatorData.status,
|
|
||||||
validator: validatorData.validator,
|
|
||||||
timestamp: confirmations[index].timestamp || validatorData.timestamp,
|
|
||||||
txHash: confirmations[index].txHash || validatorData.txHash,
|
|
||||||
signature: confirmations[index].signature || validatorData.signature
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return confirmations
|
return confirmations
|
||||||
@ -56,17 +45,19 @@ export const getConfirmationsForTx = async (
|
|||||||
setPendingConfirmations: Function,
|
setPendingConfirmations: Function,
|
||||||
getSuccessTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
getSuccessTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
||||||
) => {
|
) => {
|
||||||
|
const confirmationContractMethod = fromHome ? getMessagesSigned : getAffirmationsSigned
|
||||||
|
|
||||||
const hashMsg = web3.utils.soliditySha3Raw(messageData)
|
const hashMsg = web3.utils.soliditySha3Raw(messageData)
|
||||||
let validatorConfirmations = await Promise.all(
|
let validatorConfirmations = await Promise.all(
|
||||||
validatorList.map(getValidatorConfirmation(web3, hashMsg, bridgeContract, fromHome))
|
validatorList.map(getValidatorConfirmation(web3, hashMsg, bridgeContract, confirmationContractMethod))
|
||||||
)
|
)
|
||||||
|
|
||||||
const updateConfirmations = (confirmations: ConfirmationParam[]) => {
|
const updateConfirmations = (confirmations: BasicConfirmationParam[]) => {
|
||||||
if (confirmations.length === 0) {
|
if (confirmations.length === 0) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
validatorConfirmations = mergeConfirmations(validatorConfirmations, confirmations)
|
validatorConfirmations = mergeConfirmations(validatorConfirmations, confirmations)
|
||||||
setResult((currentConfirmations: ConfirmationParam[]) => {
|
setResult((currentConfirmations: BasicConfirmationParam[]) => {
|
||||||
if (currentConfirmations && currentConfirmations.length) {
|
if (currentConfirmations && currentConfirmations.length) {
|
||||||
return mergeConfirmations(currentConfirmations, confirmations)
|
return mergeConfirmations(currentConfirmations, confirmations)
|
||||||
}
|
}
|
||||||
@ -76,37 +67,11 @@ export const getConfirmationsForTx = async (
|
|||||||
|
|
||||||
const successConfirmations = validatorConfirmations.filter(c => c.status === VALIDATOR_CONFIRMATION_STATUS.SUCCESS)
|
const successConfirmations = validatorConfirmations.filter(c => c.status === VALIDATOR_CONFIRMATION_STATUS.SUCCESS)
|
||||||
const notSuccessConfirmations = validatorConfirmations.filter(c => c.status !== VALIDATOR_CONFIRMATION_STATUS.SUCCESS)
|
const notSuccessConfirmations = validatorConfirmations.filter(c => c.status !== VALIDATOR_CONFIRMATION_STATUS.SUCCESS)
|
||||||
const hasEnoughSignatures = successConfirmations.length >= requiredSignatures
|
const hasEnoughSignatures = successConfirmations.length === requiredSignatures
|
||||||
|
|
||||||
updateConfirmations(validatorConfirmations)
|
updateConfirmations(validatorConfirmations)
|
||||||
setSignatureCollected(hasEnoughSignatures)
|
setSignatureCollected(hasEnoughSignatures)
|
||||||
|
|
||||||
if (hasEnoughSignatures) {
|
|
||||||
setPendingConfirmations(false)
|
|
||||||
if (fromHome) {
|
|
||||||
// fetch collected signatures for possible manual processing
|
|
||||||
const signatures = await Promise.all(
|
|
||||||
Array.from(Array(requiredSignatures).keys()).map(i => bridgeContract.methods.signature(hashMsg, i).call())
|
|
||||||
)
|
|
||||||
const confirmations = signatures.flatMap(sig => {
|
|
||||||
const { v, r, s } = signatureToVRS(sig)
|
|
||||||
const address = web3.eth.accounts.recover(messageData, `0x${v}`, `0x${r}`, `0x${s}`)
|
|
||||||
return successConfirmations.filter(c => c.validator === address).map(c => ({ ...c, signature: sig }))
|
|
||||||
})
|
|
||||||
updateConfirmations(confirmations)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// get transactions from success signatures
|
|
||||||
const successConfirmationWithData = await Promise.all(
|
|
||||||
successConfirmations.map(
|
|
||||||
getSuccessExecutionTransaction(web3, bridgeContract, fromHome, messageData, startBlock, getSuccessTransactions)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
const successConfirmationWithTxFound = successConfirmationWithData.filter(v => v.txHash !== '')
|
|
||||||
updateConfirmations(successConfirmationWithTxFound)
|
|
||||||
|
|
||||||
// If signatures not collected, look for pending transactions
|
// If signatures not collected, look for pending transactions
|
||||||
if (!hasEnoughSignatures) {
|
if (!hasEnoughSignatures) {
|
||||||
// Check if confirmation is pending
|
// Check if confirmation is pending
|
||||||
@ -119,6 +84,8 @@ export const getConfirmationsForTx = async (
|
|||||||
)
|
)
|
||||||
updateConfirmations(validatorPendingConfirmations)
|
updateConfirmations(validatorPendingConfirmations)
|
||||||
setPendingConfirmations(validatorPendingConfirmations.length > 0)
|
setPendingConfirmations(validatorPendingConfirmations.length > 0)
|
||||||
|
} else {
|
||||||
|
setPendingConfirmations(false)
|
||||||
}
|
}
|
||||||
|
|
||||||
const undefinedConfirmations = validatorConfirmations.filter(
|
const undefinedConfirmations = validatorConfirmations.filter(
|
||||||
@ -128,27 +95,13 @@ export const getConfirmationsForTx = async (
|
|||||||
// Check if confirmation failed
|
// Check if confirmation failed
|
||||||
const validatorFailedConfirmationsChecks = await Promise.all(
|
const validatorFailedConfirmationsChecks = await Promise.all(
|
||||||
undefinedConfirmations.map(
|
undefinedConfirmations.map(
|
||||||
getValidatorFailedTransaction(web3, bridgeContract, messageData, startBlock, getFailedTransactions)
|
getValidatorFailedTransaction(bridgeContract, messageData, startBlock, getFailedTransactions)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
let validatorFailedConfirmations = validatorFailedConfirmationsChecks.filter(
|
const validatorFailedConfirmations = validatorFailedConfirmationsChecks.filter(
|
||||||
c => c.status === VALIDATOR_CONFIRMATION_STATUS.FAILED || c.status === VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
c => c.status === VALIDATOR_CONFIRMATION_STATUS.FAILED
|
||||||
)
|
|
||||||
if (hasEnoughSignatures && !fromHome) {
|
|
||||||
const lastTS = Math.max(...successConfirmationWithTxFound.map(c => c.timestamp || 0))
|
|
||||||
validatorFailedConfirmations = validatorFailedConfirmations.map(
|
|
||||||
c =>
|
|
||||||
c.timestamp < lastTS
|
|
||||||
? c
|
|
||||||
: {
|
|
||||||
...c,
|
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
setFailedConfirmations(
|
|
||||||
!hasEnoughSignatures && validatorFailedConfirmations.some(c => c.status === VALIDATOR_CONFIRMATION_STATUS.FAILED)
|
|
||||||
)
|
)
|
||||||
|
setFailedConfirmations(validatorFailedConfirmations.length > validatorList.length - requiredSignatures)
|
||||||
updateConfirmations(validatorFailedConfirmations)
|
updateConfirmations(validatorFailedConfirmations)
|
||||||
|
|
||||||
const missingConfirmations = validatorConfirmations.filter(
|
const missingConfirmations = validatorConfirmations.filter(
|
||||||
@ -159,13 +112,30 @@ export const getConfirmationsForTx = async (
|
|||||||
// If signatures collected, it should set other signatures not found as not required
|
// If signatures collected, it should set other signatures not found as not required
|
||||||
const notRequiredConfirmations = missingConfirmations.map(c => ({
|
const notRequiredConfirmations = missingConfirmations.map(c => ({
|
||||||
validator: c.validator,
|
validator: c.validator,
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED,
|
status: VALIDATOR_CONFIRMATION_STATUS.NOT_REQUIRED
|
||||||
timestamp: 0,
|
|
||||||
txHash: ''
|
|
||||||
}))
|
}))
|
||||||
updateConfirmations(notRequiredConfirmations)
|
updateConfirmations(notRequiredConfirmations)
|
||||||
|
|
||||||
|
if (fromHome) {
|
||||||
|
// fetch collected signatures for possible manual processing
|
||||||
|
setSignatureCollected(
|
||||||
|
await Promise.all(
|
||||||
|
Array.from(Array(requiredSignatures).keys()).map(i => bridgeContract.methods.signature(hashMsg, i).call())
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// get transactions from success signatures
|
||||||
|
const successConfirmationWithData = await Promise.all(
|
||||||
|
successConfirmations.map(
|
||||||
|
getSuccessExecutionTransaction(web3, bridgeContract, fromHome, messageData, startBlock, getSuccessTransactions)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
const successConfirmationWithTxFound = successConfirmationWithData.filter(v => v.txHash !== '')
|
||||||
|
updateConfirmations(successConfirmationWithTxFound)
|
||||||
|
|
||||||
// retry if not all signatures are collected and some confirmations are still missing
|
// retry if not all signatures are collected and some confirmations are still missing
|
||||||
// or some success transactions were not fetched successfully
|
// or some success transactions were not fetched successfully
|
||||||
if (
|
if (
|
||||||
|
@ -59,12 +59,11 @@ export const getSuccessExecutionData = async (
|
|||||||
const validatorAddress = web3.utils.toChecksumAddress(txReceipt.from)
|
const validatorAddress = web3.utils.toChecksumAddress(txReceipt.from)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.EXECUTION_SUCCESS,
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
||||||
validator: validatorAddress,
|
validator: validatorAddress,
|
||||||
txHash: event.transactionHash,
|
txHash: event.transactionHash,
|
||||||
timestamp: blockTimestamp,
|
timestamp: blockTimestamp,
|
||||||
executionResult: event.returnValues.status,
|
executionResult: event.returnValues.status
|
||||||
blockNumber: event.blockNumber
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null
|
return null
|
||||||
@ -116,8 +115,7 @@ export const getFinalizationEvent = async (
|
|||||||
validator: validator,
|
validator: validator,
|
||||||
txHash: pendingTx.hash,
|
txHash: pendingTx.hash,
|
||||||
timestamp: nowTimestamp,
|
timestamp: nowTimestamp,
|
||||||
executionResult: false,
|
executionResult: false
|
||||||
blockNumber: 0
|
|
||||||
})
|
})
|
||||||
setPendingExecution(true)
|
setPendingExecution(true)
|
||||||
} else {
|
} else {
|
||||||
@ -146,8 +144,7 @@ export const getFinalizationEvent = async (
|
|||||||
validator: validator,
|
validator: validator,
|
||||||
txHash: failedTx.hash,
|
txHash: failedTx.hash,
|
||||||
timestamp,
|
timestamp,
|
||||||
executionResult: false,
|
executionResult: false
|
||||||
blockNumber: parseInt(failedTx.blockNumber)
|
|
||||||
})
|
})
|
||||||
setFailedExecution(true)
|
setFailedExecution(true)
|
||||||
}
|
}
|
||||||
|
@ -1,45 +1,38 @@
|
|||||||
import Web3 from 'web3'
|
import Web3 from 'web3'
|
||||||
import { Contract } from 'web3-eth-contract'
|
import { Contract } from 'web3-eth-contract'
|
||||||
import { ConfirmationParam } from '../hooks/useMessageConfirmations'
|
import { BasicConfirmationParam, ConfirmationParam } from '../hooks/useMessageConfirmations'
|
||||||
import validatorsCache from '../services/ValidatorsCache'
|
import validatorsCache from '../services/ValidatorsCache'
|
||||||
import { CACHE_KEY_FAILED, CACHE_KEY_SUCCESS, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
import { CACHE_KEY_FAILED, CACHE_KEY_SUCCESS, VALIDATOR_CONFIRMATION_STATUS } from '../config/constants'
|
||||||
import { APIPendingTransaction, APITransaction, GetTransactionParams, GetPendingTransactionParams } from './explorer'
|
import { APIPendingTransaction, APITransaction, GetTransactionParams, GetPendingTransactionParams } from './explorer'
|
||||||
import { homeBlockNumberProvider } from '../services/BlockNumberProvider'
|
import { homeBlockNumberProvider } from '../services/BlockNumberProvider'
|
||||||
import { getAffirmationsSigned, getMessagesSigned } from './contract'
|
|
||||||
|
|
||||||
export const getValidatorConfirmation = (
|
export const getValidatorConfirmation = (
|
||||||
web3: Web3,
|
web3: Web3,
|
||||||
hashMsg: string,
|
hashMsg: string,
|
||||||
bridgeContract: Contract,
|
bridgeContract: Contract,
|
||||||
fromHome: boolean
|
confirmationContractMethod: Function
|
||||||
) => async (validator: string): Promise<ConfirmationParam> => {
|
) => async (validator: string): Promise<BasicConfirmationParam> => {
|
||||||
const hashSenderMsg = web3.utils.soliditySha3Raw(validator, hashMsg)
|
const hashSenderMsg = web3.utils.soliditySha3Raw(validator, hashMsg)
|
||||||
|
|
||||||
const fromCache = validatorsCache.getData(hashSenderMsg)
|
const signatureFromCache = validatorsCache.get(hashSenderMsg)
|
||||||
if (fromCache) {
|
if (signatureFromCache) {
|
||||||
return fromCache
|
return {
|
||||||
|
validator,
|
||||||
|
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const confirmationContractMethod = fromHome ? getMessagesSigned : getAffirmationsSigned
|
|
||||||
const confirmed = await confirmationContractMethod(bridgeContract, hashSenderMsg)
|
const confirmed = await confirmationContractMethod(bridgeContract, hashSenderMsg)
|
||||||
|
const status = confirmed ? VALIDATOR_CONFIRMATION_STATUS.SUCCESS : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
|
|
||||||
// If validator confirmed signature, we cache the result to avoid doing future requests for a result that won't change
|
// If validator confirmed signature, we cache the result to avoid doing future requests for a result that won't change
|
||||||
if (confirmed) {
|
if (confirmed) {
|
||||||
const confirmation: ConfirmationParam = {
|
validatorsCache.set(hashSenderMsg, confirmed)
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.SUCCESS,
|
|
||||||
validator,
|
|
||||||
timestamp: 0,
|
|
||||||
txHash: ''
|
|
||||||
}
|
|
||||||
validatorsCache.setData(hashSenderMsg, confirmation)
|
|
||||||
return confirmation
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
|
||||||
validator,
|
validator,
|
||||||
timestamp: 0,
|
status
|
||||||
txHash: ''
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -50,7 +43,7 @@ export const getSuccessExecutionTransaction = (
|
|||||||
messageData: string,
|
messageData: string,
|
||||||
startBlock: number,
|
startBlock: number,
|
||||||
getSuccessTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
getSuccessTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
||||||
) => async (validatorData: ConfirmationParam): Promise<ConfirmationParam> => {
|
) => async (validatorData: BasicConfirmationParam): Promise<ConfirmationParam> => {
|
||||||
const { validator } = validatorData
|
const { validator } = validatorData
|
||||||
const validatorCacheKey = `${CACHE_KEY_SUCCESS}${validatorData.validator}-${messageData}`
|
const validatorCacheKey = `${CACHE_KEY_SUCCESS}${validatorData.validator}-${messageData}`
|
||||||
const fromCache = validatorsCache.getData(validatorCacheKey)
|
const fromCache = validatorsCache.getData(validatorCacheKey)
|
||||||
@ -94,12 +87,11 @@ export const getSuccessExecutionTransaction = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const getValidatorFailedTransaction = (
|
export const getValidatorFailedTransaction = (
|
||||||
web3: Web3,
|
|
||||||
bridgeContract: Contract,
|
bridgeContract: Contract,
|
||||||
messageData: string,
|
messageData: string,
|
||||||
startBlock: number,
|
startBlock: number,
|
||||||
getFailedTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
getFailedTransactions: (args: GetTransactionParams) => Promise<APITransaction[]>
|
||||||
) => async (validatorData: ConfirmationParam): Promise<ConfirmationParam> => {
|
) => async (validatorData: BasicConfirmationParam): Promise<ConfirmationParam> => {
|
||||||
const validatorCacheKey = `${CACHE_KEY_FAILED}${validatorData.validator}-${messageData}`
|
const validatorCacheKey = `${CACHE_KEY_FAILED}${validatorData.validator}-${messageData}`
|
||||||
const failedFromCache = validatorsCache.getData(validatorCacheKey)
|
const failedFromCache = validatorsCache.getData(validatorCacheKey)
|
||||||
|
|
||||||
@ -114,33 +106,30 @@ export const getValidatorFailedTransaction = (
|
|||||||
startBlock,
|
startBlock,
|
||||||
endBlock: homeBlockNumberProvider.get() || 0
|
endBlock: homeBlockNumberProvider.get() || 0
|
||||||
})
|
})
|
||||||
|
const newStatus =
|
||||||
|
failedTransactions.length > 0 ? VALIDATOR_CONFIRMATION_STATUS.FAILED : VALIDATOR_CONFIRMATION_STATUS.UNDEFINED
|
||||||
|
|
||||||
|
let txHashTimestamp = 0
|
||||||
|
let txHash = ''
|
||||||
// If validator signature failed, we cache the result to avoid doing future requests for a result that won't change
|
// If validator signature failed, we cache the result to avoid doing future requests for a result that won't change
|
||||||
if (failedTransactions.length > 0) {
|
if (failedTransactions.length > 0) {
|
||||||
const failedTx = failedTransactions[0]
|
const failedTx = failedTransactions[0]
|
||||||
const confirmation: ConfirmationParam = {
|
txHashTimestamp = parseInt(failedTx.timeStamp)
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.FAILED,
|
txHash = failedTx.hash
|
||||||
validator: validatorData.validator,
|
|
||||||
txHash: failedTx.hash,
|
|
||||||
timestamp: parseInt(failedTx.timeStamp)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (failedTx.input && failedTx.input.length > 10) {
|
validatorsCache.setData(validatorCacheKey, {
|
||||||
try {
|
validator: validatorData.validator,
|
||||||
const res = web3.eth.abi.decodeParameters(['bytes', 'bytes'], `0x${failedTx.input.slice(10)}`)
|
status: newStatus,
|
||||||
confirmation.signature = res[0]
|
txHash,
|
||||||
confirmation.status = VALIDATOR_CONFIRMATION_STATUS.FAILED_VALID
|
timestamp: txHashTimestamp
|
||||||
console.log(`Adding manual signature from failed message from ${validatorData.validator}`)
|
})
|
||||||
} catch {}
|
|
||||||
}
|
|
||||||
validatorsCache.setData(validatorCacheKey, confirmation)
|
|
||||||
return confirmation
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: VALIDATOR_CONFIRMATION_STATUS.UNDEFINED,
|
|
||||||
validator: validatorData.validator,
|
validator: validatorData.validator,
|
||||||
txHash: '',
|
status: newStatus,
|
||||||
timestamp: 0
|
txHash,
|
||||||
|
timestamp: txHashTimestamp
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -148,7 +137,7 @@ export const getValidatorPendingTransaction = (
|
|||||||
bridgeContract: Contract,
|
bridgeContract: Contract,
|
||||||
messageData: string,
|
messageData: string,
|
||||||
getPendingTransactions: (args: GetPendingTransactionParams) => Promise<APIPendingTransaction[]>
|
getPendingTransactions: (args: GetPendingTransactionParams) => Promise<APIPendingTransaction[]>
|
||||||
) => async (validatorData: ConfirmationParam): Promise<ConfirmationParam> => {
|
) => async (validatorData: BasicConfirmationParam): Promise<ConfirmationParam> => {
|
||||||
const failedTransactions = await getPendingTransactions({
|
const failedTransactions = await getPendingTransactions({
|
||||||
account: validatorData.validator,
|
account: validatorData.validator,
|
||||||
to: bridgeContract.options.address,
|
to: bridgeContract.options.address,
|
||||||
|
@ -10,37 +10,6 @@ import { SnapshotProvider } from '../services/SnapshotProvider'
|
|||||||
export interface MessageObject {
|
export interface MessageObject {
|
||||||
id: string
|
id: string
|
||||||
data: string
|
data: string
|
||||||
sender?: string
|
|
||||||
executor?: string
|
|
||||||
obToken?: string
|
|
||||||
obReceiver?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface WarnRule {
|
|
||||||
message: string
|
|
||||||
sender?: string
|
|
||||||
executor?: string
|
|
||||||
obToken?: string
|
|
||||||
obReceiver?: string
|
|
||||||
}
|
|
||||||
|
|
||||||
export const matchesRule = (rule: WarnRule, msg: MessageObject) => {
|
|
||||||
if (!msg.executor || !msg.sender) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if (!!rule.executor && rule.executor.toLowerCase() !== msg.executor.toLowerCase()) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if (!!rule.sender && rule.sender.toLowerCase() !== msg.sender.toLowerCase()) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if (!!rule.obToken && (!msg.obToken || rule.obToken.toLowerCase() !== msg.obToken.toLowerCase())) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if (!!rule.obReceiver && (!msg.obReceiver || rule.obReceiver.toLowerCase() !== msg.obReceiver.toLowerCase())) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const rawGetWeb3 = (url: string) => new Web3(new Web3.providers.HttpProvider(url))
|
const rawGetWeb3 = (url: string) => new Web3(new Web3.providers.HttpProvider(url))
|
||||||
@ -57,33 +26,15 @@ export const filterEventsByAbi = (
|
|||||||
const eventHash = web3.eth.abi.encodeEventSignature(eventAbi)
|
const eventHash = web3.eth.abi.encodeEventSignature(eventAbi)
|
||||||
const events = txReceipt.logs.filter(e => e.address === bridgeAddress && e.topics[0] === eventHash)
|
const events = txReceipt.logs.filter(e => e.address === bridgeAddress && e.topics[0] === eventHash)
|
||||||
|
|
||||||
if (!eventAbi || !eventAbi.inputs || !eventAbi.inputs.length) {
|
|
||||||
return []
|
|
||||||
}
|
|
||||||
const inputs = eventAbi.inputs
|
|
||||||
return events.map(e => {
|
return events.map(e => {
|
||||||
const { messageId, encodedData } = web3.eth.abi.decodeLog(inputs, e.data, [e.topics[1]])
|
let decodedLogs: { [p: string]: string } = {
|
||||||
let sender, executor, obToken, obReceiver
|
messageId: '',
|
||||||
if (encodedData.length >= 160) {
|
encodedData: ''
|
||||||
sender = `0x${encodedData.slice(66, 106)}`
|
|
||||||
executor = `0x${encodedData.slice(106, 146)}`
|
|
||||||
const dataOffset =
|
|
||||||
160 + (parseInt(encodedData.slice(154, 156), 16) + parseInt(encodedData.slice(156, 158), 16)) * 2 + 8
|
|
||||||
if (encodedData.length >= dataOffset + 64) {
|
|
||||||
obToken = `0x${encodedData.slice(dataOffset + 24, dataOffset + 64)}`
|
|
||||||
}
|
|
||||||
if (encodedData.length >= dataOffset + 128) {
|
|
||||||
obReceiver = `0x${encodedData.slice(dataOffset + 88, dataOffset + 128)}`
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return {
|
if (eventAbi && eventAbi.inputs && eventAbi.inputs.length) {
|
||||||
id: messageId || '',
|
decodedLogs = web3.eth.abi.decodeLog(eventAbi.inputs, e.data, [e.topics[1]])
|
||||||
data: encodedData || '',
|
|
||||||
sender,
|
|
||||||
executor,
|
|
||||||
obToken,
|
|
||||||
obReceiver
|
|
||||||
}
|
}
|
||||||
|
return { id: decodedLogs.messageId, data: decodedLogs.encodedData }
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binary file not shown.
Binary file not shown.
@ -1,7 +1,13 @@
|
|||||||
|
const HOME_NATIVE_TO_ERC_ABI = require('../contracts/build/contracts/HomeBridgeNativeToErc').abi
|
||||||
|
const FOREIGN_NATIVE_TO_ERC_ABI = require('../contracts/build/contracts/ForeignBridgeNativeToErc').abi
|
||||||
|
const HOME_ERC_TO_ERC_ABI = require('../contracts/build/contracts/HomeBridgeErcToErc').abi
|
||||||
|
const FOREIGN_ERC_TO_ERC_ABI = require('../contracts/build/contracts/ForeignBridgeErc677ToErc677').abi
|
||||||
const HOME_ERC_TO_NATIVE_ABI = require('../contracts/build/contracts/HomeBridgeErcToNative').abi
|
const HOME_ERC_TO_NATIVE_ABI = require('../contracts/build/contracts/HomeBridgeErcToNative').abi
|
||||||
const FOREIGN_ERC_TO_NATIVE_ABI = require('../contracts/build/contracts/XDaiForeignBridge.json').abi
|
const FOREIGN_ERC_TO_NATIVE_ABI = require('../contracts/build/contracts/ForeignBridgeErcToNative').abi
|
||||||
const ERC20_ABI = require('../contracts/build/contracts/ERC20').abi
|
const ERC20_ABI = require('../contracts/build/contracts/ERC20').abi
|
||||||
const BLOCK_REWARD_ABI = require('../contracts/build/contracts/BlockRewardMock').abi
|
const ERC677_ABI = require('../contracts/build/contracts/ERC677').abi
|
||||||
|
const ERC677_BRIDGE_TOKEN_ABI = require('../contracts/build/contracts/ERC677BridgeToken').abi
|
||||||
|
const BLOCK_REWARD_ABI = require('../contracts/build/contracts/BlockReward').abi
|
||||||
const BRIDGE_VALIDATORS_ABI = require('../contracts/build/contracts/BridgeValidators').abi
|
const BRIDGE_VALIDATORS_ABI = require('../contracts/build/contracts/BridgeValidators').abi
|
||||||
const REWARDABLE_VALIDATORS_ABI = require('../contracts/build/contracts/RewardableValidators').abi
|
const REWARDABLE_VALIDATORS_ABI = require('../contracts/build/contracts/RewardableValidators').abi
|
||||||
const HOME_AMB_ABI = require('../contracts/build/contracts/HomeAMB').abi
|
const HOME_AMB_ABI = require('../contracts/build/contracts/HomeAMB').abi
|
||||||
@ -9,13 +15,52 @@ const FOREIGN_AMB_ABI = require('../contracts/build/contracts/ForeignAMB').abi
|
|||||||
const BOX_ABI = require('../contracts/build/contracts/Box').abi
|
const BOX_ABI = require('../contracts/build/contracts/Box').abi
|
||||||
const HOME_AMB_ERC_TO_ERC_ABI = require('../contracts/build/contracts/HomeAMBErc677ToErc677').abi
|
const HOME_AMB_ERC_TO_ERC_ABI = require('../contracts/build/contracts/HomeAMBErc677ToErc677').abi
|
||||||
const FOREIGN_AMB_ERC_TO_ERC_ABI = require('../contracts/build/contracts/ForeignAMBErc677ToErc677').abi
|
const FOREIGN_AMB_ERC_TO_ERC_ABI = require('../contracts/build/contracts/ForeignAMBErc677ToErc677').abi
|
||||||
|
const HOME_STAKE_ERC_TO_ERC_ABI = require('../contracts/build/contracts/HomeStakeTokenMediator').abi
|
||||||
|
const FOREIGN_STAKE_ERC_TO_ERC_ABI = require('../contracts/build/contracts/ForeignStakeTokenMediator').abi
|
||||||
|
|
||||||
|
const { HOME_V1_ABI, FOREIGN_V1_ABI } = require('./v1Abis')
|
||||||
const { BRIDGE_MODES } = require('./constants')
|
const { BRIDGE_MODES } = require('./constants')
|
||||||
|
|
||||||
|
const ERC20_BYTES32_ABI = [
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'name',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'bytes32'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'symbol',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'bytes32'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
const OLD_AMB_USER_REQUEST_FOR_SIGNATURE_ABI = [
|
const OLD_AMB_USER_REQUEST_FOR_SIGNATURE_ABI = [
|
||||||
{
|
{
|
||||||
anonymous: false,
|
anonymous: false,
|
||||||
inputs: [
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: true,
|
||||||
|
name: 'messageId',
|
||||||
|
type: 'bytes32'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
indexed: false,
|
indexed: false,
|
||||||
name: 'encodedData',
|
name: 'encodedData',
|
||||||
@ -31,6 +76,11 @@ const OLD_AMB_USER_REQUEST_FOR_AFFIRMATION_ABI = [
|
|||||||
{
|
{
|
||||||
anonymous: false,
|
anonymous: false,
|
||||||
inputs: [
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: true,
|
||||||
|
name: 'messageId',
|
||||||
|
type: 'bytes32'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
indexed: false,
|
indexed: false,
|
||||||
name: 'encodedData',
|
name: 'encodedData',
|
||||||
@ -45,15 +95,27 @@ const OLD_AMB_USER_REQUEST_FOR_AFFIRMATION_ABI = [
|
|||||||
function getBridgeABIs(bridgeMode) {
|
function getBridgeABIs(bridgeMode) {
|
||||||
let HOME_ABI = null
|
let HOME_ABI = null
|
||||||
let FOREIGN_ABI = null
|
let FOREIGN_ABI = null
|
||||||
if (bridgeMode === BRIDGE_MODES.ERC_TO_NATIVE) {
|
if (bridgeMode === BRIDGE_MODES.NATIVE_TO_ERC) {
|
||||||
|
HOME_ABI = HOME_NATIVE_TO_ERC_ABI
|
||||||
|
FOREIGN_ABI = FOREIGN_NATIVE_TO_ERC_ABI
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.ERC_TO_ERC) {
|
||||||
|
HOME_ABI = HOME_ERC_TO_ERC_ABI
|
||||||
|
FOREIGN_ABI = FOREIGN_ERC_TO_ERC_ABI
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.ERC_TO_NATIVE) {
|
||||||
HOME_ABI = HOME_ERC_TO_NATIVE_ABI
|
HOME_ABI = HOME_ERC_TO_NATIVE_ABI
|
||||||
FOREIGN_ABI = FOREIGN_ERC_TO_NATIVE_ABI
|
FOREIGN_ABI = FOREIGN_ERC_TO_NATIVE_ABI
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.NATIVE_TO_ERC_V1) {
|
||||||
|
HOME_ABI = HOME_V1_ABI
|
||||||
|
FOREIGN_ABI = FOREIGN_V1_ABI
|
||||||
} else if (bridgeMode === BRIDGE_MODES.ARBITRARY_MESSAGE) {
|
} else if (bridgeMode === BRIDGE_MODES.ARBITRARY_MESSAGE) {
|
||||||
HOME_ABI = HOME_AMB_ABI
|
HOME_ABI = HOME_AMB_ABI
|
||||||
FOREIGN_ABI = FOREIGN_AMB_ABI
|
FOREIGN_ABI = FOREIGN_AMB_ABI
|
||||||
} else if (bridgeMode === BRIDGE_MODES.AMB_ERC_TO_ERC) {
|
} else if (bridgeMode === BRIDGE_MODES.AMB_ERC_TO_ERC) {
|
||||||
HOME_ABI = HOME_AMB_ERC_TO_ERC_ABI
|
HOME_ABI = HOME_AMB_ERC_TO_ERC_ABI
|
||||||
FOREIGN_ABI = FOREIGN_AMB_ERC_TO_ERC_ABI
|
FOREIGN_ABI = FOREIGN_AMB_ERC_TO_ERC_ABI
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.STAKE_AMB_ERC_TO_ERC) {
|
||||||
|
HOME_ABI = HOME_STAKE_ERC_TO_ERC_ABI
|
||||||
|
FOREIGN_ABI = FOREIGN_STAKE_ERC_TO_ERC_ABI
|
||||||
} else {
|
} else {
|
||||||
throw new Error(`Unrecognized bridge mode: ${bridgeMode}`)
|
throw new Error(`Unrecognized bridge mode: ${bridgeMode}`)
|
||||||
}
|
}
|
||||||
@ -63,15 +125,26 @@ function getBridgeABIs(bridgeMode) {
|
|||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
getBridgeABIs,
|
getBridgeABIs,
|
||||||
|
HOME_NATIVE_TO_ERC_ABI,
|
||||||
|
FOREIGN_NATIVE_TO_ERC_ABI,
|
||||||
|
HOME_ERC_TO_ERC_ABI,
|
||||||
|
FOREIGN_ERC_TO_ERC_ABI,
|
||||||
HOME_ERC_TO_NATIVE_ABI,
|
HOME_ERC_TO_NATIVE_ABI,
|
||||||
FOREIGN_ERC_TO_NATIVE_ABI,
|
FOREIGN_ERC_TO_NATIVE_ABI,
|
||||||
ERC20_ABI,
|
ERC20_ABI,
|
||||||
|
ERC677_ABI,
|
||||||
|
ERC677_BRIDGE_TOKEN_ABI,
|
||||||
BLOCK_REWARD_ABI,
|
BLOCK_REWARD_ABI,
|
||||||
BRIDGE_VALIDATORS_ABI,
|
BRIDGE_VALIDATORS_ABI,
|
||||||
REWARDABLE_VALIDATORS_ABI,
|
REWARDABLE_VALIDATORS_ABI,
|
||||||
|
HOME_V1_ABI,
|
||||||
|
FOREIGN_V1_ABI,
|
||||||
|
ERC20_BYTES32_ABI,
|
||||||
HOME_AMB_ABI,
|
HOME_AMB_ABI,
|
||||||
FOREIGN_AMB_ABI,
|
FOREIGN_AMB_ABI,
|
||||||
OLD_AMB_USER_REQUEST_FOR_AFFIRMATION_ABI,
|
OLD_AMB_USER_REQUEST_FOR_AFFIRMATION_ABI,
|
||||||
OLD_AMB_USER_REQUEST_FOR_SIGNATURE_ABI,
|
OLD_AMB_USER_REQUEST_FOR_SIGNATURE_ABI,
|
||||||
BOX_ABI
|
BOX_ABI,
|
||||||
|
HOME_STAKE_ERC_TO_ERC_ABI,
|
||||||
|
FOREIGN_STAKE_ERC_TO_ERC_ABI
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,30 @@
|
|||||||
const BRIDGE_MODES = {
|
const BRIDGE_MODES = {
|
||||||
|
NATIVE_TO_ERC: 'NATIVE_TO_ERC',
|
||||||
|
ERC_TO_ERC: 'ERC_TO_ERC',
|
||||||
ERC_TO_NATIVE: 'ERC_TO_NATIVE',
|
ERC_TO_NATIVE: 'ERC_TO_NATIVE',
|
||||||
|
NATIVE_TO_ERC_V1: 'NATIVE_TO_ERC_V1',
|
||||||
ARBITRARY_MESSAGE: 'ARBITRARY_MESSAGE',
|
ARBITRARY_MESSAGE: 'ARBITRARY_MESSAGE',
|
||||||
AMB_ERC_TO_ERC: 'AMB_ERC_TO_ERC'
|
AMB_ERC_TO_ERC: 'AMB_ERC_TO_ERC',
|
||||||
|
STAKE_AMB_ERC_TO_ERC: 'STAKE_AMB_ERC_TO_ERC'
|
||||||
|
}
|
||||||
|
|
||||||
|
const ERC_TYPES = {
|
||||||
|
ERC20: 'ERC20',
|
||||||
|
ERC677: 'ERC677'
|
||||||
|
}
|
||||||
|
|
||||||
|
const FEE_MANAGER_MODE = {
|
||||||
|
ONE_DIRECTION: 'ONE_DIRECTION',
|
||||||
|
BOTH_DIRECTIONS: 'BOTH_DIRECTIONS',
|
||||||
|
ONE_DIRECTION_STAKE: 'ONE_DIRECTION_STAKE',
|
||||||
|
UNDEFINED: 'UNDEFINED'
|
||||||
}
|
}
|
||||||
|
|
||||||
const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'
|
const ZERO_ADDRESS = '0x0000000000000000000000000000000000000000'
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
BRIDGE_MODES,
|
BRIDGE_MODES,
|
||||||
|
ERC_TYPES,
|
||||||
|
FEE_MANAGER_MODE,
|
||||||
ZERO_ADDRESS
|
ZERO_ADDRESS
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
const { soliditySha3 } = require('web3-utils')
|
|
||||||
|
|
||||||
function strip0x(input) {
|
function strip0x(input) {
|
||||||
return input.replace(/^0x/, '')
|
return input.replace(/^0x/, '')
|
||||||
}
|
}
|
||||||
@ -41,35 +39,8 @@ const normalizeAMBMessageEvent = e => {
|
|||||||
return parseAMBMessage(msgData)
|
return parseAMBMessage(msgData)
|
||||||
}
|
}
|
||||||
|
|
||||||
const ambInformationSignatures = [
|
|
||||||
'eth_call(address,bytes)',
|
|
||||||
'eth_call(address,bytes,uint256)',
|
|
||||||
'eth_call(address,address,uint256,bytes)',
|
|
||||||
'eth_blockNumber()',
|
|
||||||
'eth_getBlockByNumber()',
|
|
||||||
'eth_getBlockByNumber(uint256)',
|
|
||||||
'eth_getBlockByHash(bytes32)',
|
|
||||||
'eth_getBalance(address)',
|
|
||||||
'eth_getBalance(address,uint256)',
|
|
||||||
'eth_getTransactionCount(address)',
|
|
||||||
'eth_getTransactionCount(address,uint256)',
|
|
||||||
'eth_getTransactionByHash(bytes32)',
|
|
||||||
'eth_getTransactionReceipt(bytes32)',
|
|
||||||
'eth_getStorageAt(address,bytes32)',
|
|
||||||
'eth_getStorageAt(address,bytes32,uint256)'
|
|
||||||
]
|
|
||||||
const ambInformationSelectors = Object.fromEntries(ambInformationSignatures.map(sig => [soliditySha3(sig), sig]))
|
|
||||||
const normalizeAMBInfoRequest = e => ({
|
|
||||||
messageId: e.returnValues.messageId,
|
|
||||||
sender: e.returnValues.sender,
|
|
||||||
requestSelector: ambInformationSelectors[e.returnValues.requestSelector] || 'unknown',
|
|
||||||
data: e.returnValues.data
|
|
||||||
})
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
strip0x,
|
strip0x,
|
||||||
parseAMBMessage,
|
parseAMBMessage,
|
||||||
normalizeAMBMessageEvent,
|
normalizeAMBMessageEvent
|
||||||
ambInformationSignatures,
|
|
||||||
normalizeAMBInfoRequest
|
|
||||||
}
|
}
|
||||||
|
@ -8,10 +8,8 @@
|
|||||||
"test": "NODE_ENV=test mocha"
|
"test": "NODE_ENV=test mocha"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@mycrypto/gas-estimation": "^1.1.0",
|
|
||||||
"gas-price-oracle": "^0.1.5",
|
"gas-price-oracle": "^0.1.5",
|
||||||
"web3-utils": "^1.3.0",
|
"web3-utils": "1.0.0-beta.34"
|
||||||
"node-fetch": "^2.1.2"
|
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"bn-chai": "^1.0.1",
|
"bn-chai": "^1.0.1",
|
||||||
|
@ -1,8 +1,12 @@
|
|||||||
const { expect } = require('chai')
|
const { expect } = require('chai')
|
||||||
const { BRIDGE_MODES } = require('../constants')
|
const { BRIDGE_MODES, ERC_TYPES } = require('../constants')
|
||||||
|
|
||||||
describe('constants', () => {
|
describe('constants', () => {
|
||||||
it('should contain correct number of bridge types', () => {
|
it('should contain correct number of bridge types', () => {
|
||||||
expect(Object.keys(BRIDGE_MODES).length).to.be.equal(3)
|
expect(Object.keys(BRIDGE_MODES).length).to.be.equal(7)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should contain correct number of erc types', () => {
|
||||||
|
expect(Object.keys(ERC_TYPES).length).to.be.equal(2)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
159
commons/test/getTokenType.js
Normal file
159
commons/test/getTokenType.js
Normal file
@ -0,0 +1,159 @@
|
|||||||
|
const { expect } = require('chai')
|
||||||
|
const { getTokenType, ERC_TYPES } = require('..')
|
||||||
|
|
||||||
|
describe('getTokenType', () => {
|
||||||
|
it('should return ERC677 if bridgeContract is equal to bridgeAddress', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.resolve(bridgeAddress)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC677)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC20 if bridgeContract is not equal to bridgeAddress', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.resolve('0xBFCb120F7B1de491262CA4D9D8Eba70438b6896E')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC20)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC20 if bridgeContract is not present', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.reject()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC20)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC20 if bridgeContract and isBridge are not present', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.reject()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isBridge: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.reject()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC20)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC677 if isBridge returns true', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.reject()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isBridge: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.resolve(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC677)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC677 if isBridge returns true and bridgeContract not present', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
isBridge: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.resolve(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC677)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return ERC20 if isBridge returns false', async () => {
|
||||||
|
// Given
|
||||||
|
const bridgeAddress = '0xCecBE80Ed3548dE11D7d2D922a36576eA40C4c26'
|
||||||
|
const contract = {
|
||||||
|
methods: {
|
||||||
|
bridgeContract: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.reject()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isBridge: () => {
|
||||||
|
return {
|
||||||
|
call: () => Promise.resolve(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// When
|
||||||
|
const type = await getTokenType(contract, bridgeAddress)
|
||||||
|
|
||||||
|
// Then
|
||||||
|
expect(type).to.equal(ERC_TYPES.ERC20)
|
||||||
|
})
|
||||||
|
})
|
161
commons/utils.js
161
commons/utils.js
@ -1,18 +1,22 @@
|
|||||||
const { toWei, toBN, BN } = require('web3-utils')
|
const { toWei, toBN } = require('web3-utils')
|
||||||
const { GasPriceOracle } = require('gas-price-oracle')
|
const { GasPriceOracle } = require('gas-price-oracle')
|
||||||
const { estimateFees } = require('@mycrypto/gas-estimation')
|
const { BRIDGE_MODES, FEE_MANAGER_MODE, ERC_TYPES } = require('./constants')
|
||||||
const fetch = require('node-fetch')
|
|
||||||
const { BRIDGE_MODES } = require('./constants')
|
|
||||||
const { REWARDABLE_VALIDATORS_ABI } = require('./abis')
|
const { REWARDABLE_VALIDATORS_ABI } = require('./abis')
|
||||||
|
|
||||||
const gasPriceOracle = new GasPriceOracle()
|
const gasPriceOracle = new GasPriceOracle()
|
||||||
|
|
||||||
function decodeBridgeMode(bridgeModeHash) {
|
function decodeBridgeMode(bridgeModeHash) {
|
||||||
switch (bridgeModeHash) {
|
switch (bridgeModeHash) {
|
||||||
|
case '0x92a8d7fe':
|
||||||
|
return BRIDGE_MODES.NATIVE_TO_ERC
|
||||||
|
case '0xba4690f5':
|
||||||
|
return BRIDGE_MODES.ERC_TO_ERC
|
||||||
case '0x18762d46':
|
case '0x18762d46':
|
||||||
return BRIDGE_MODES.ERC_TO_NATIVE
|
return BRIDGE_MODES.ERC_TO_NATIVE
|
||||||
case '0x2544fbb9':
|
case '0x2544fbb9':
|
||||||
return BRIDGE_MODES.ARBITRARY_MESSAGE
|
return BRIDGE_MODES.ARBITRARY_MESSAGE
|
||||||
|
case '0x16ea01e9':
|
||||||
|
return BRIDGE_MODES.STAKE_AMB_ERC_TO_ERC
|
||||||
case '0x76595b56':
|
case '0x76595b56':
|
||||||
return BRIDGE_MODES.AMB_ERC_TO_ERC
|
return BRIDGE_MODES.AMB_ERC_TO_ERC
|
||||||
default:
|
default:
|
||||||
@ -20,9 +24,80 @@ function decodeBridgeMode(bridgeModeHash) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const decodeFeeManagerMode = managerModeHash => {
|
||||||
|
switch (managerModeHash) {
|
||||||
|
case '0xf2aed8f7':
|
||||||
|
return FEE_MANAGER_MODE.ONE_DIRECTION
|
||||||
|
case '0xd7de965f':
|
||||||
|
return FEE_MANAGER_MODE.BOTH_DIRECTIONS
|
||||||
|
default:
|
||||||
|
throw new Error(`Unrecognized fee manager mode hash: '${managerModeHash}'`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
async function getBridgeMode(contract) {
|
async function getBridgeMode(contract) {
|
||||||
const bridgeModeHash = await contract.methods.getBridgeMode().call()
|
try {
|
||||||
return decodeBridgeMode(bridgeModeHash)
|
const bridgeModeHash = await contract.methods.getBridgeMode().call()
|
||||||
|
return decodeBridgeMode(bridgeModeHash)
|
||||||
|
} catch (e) {
|
||||||
|
return BRIDGE_MODES.NATIVE_TO_ERC_V1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const getTokenType = async (bridgeTokenContract, bridgeAddress) => {
|
||||||
|
try {
|
||||||
|
const resultBridgeAddress = await bridgeTokenContract.methods.bridgeContract().call()
|
||||||
|
if (resultBridgeAddress === bridgeAddress) {
|
||||||
|
return ERC_TYPES.ERC677
|
||||||
|
} else {
|
||||||
|
return ERC_TYPES.ERC20
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
try {
|
||||||
|
const isBridge = await bridgeTokenContract.methods.isBridge(bridgeAddress).call()
|
||||||
|
if (isBridge) {
|
||||||
|
return ERC_TYPES.ERC677
|
||||||
|
} else {
|
||||||
|
return ERC_TYPES.ERC20
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
return ERC_TYPES.ERC20
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const isErcToErcMode = bridgeMode => {
|
||||||
|
return (
|
||||||
|
bridgeMode === BRIDGE_MODES.ERC_TO_ERC ||
|
||||||
|
bridgeMode === BRIDGE_MODES.AMB_ERC_TO_ERC ||
|
||||||
|
bridgeMode === BRIDGE_MODES.STAKE_AMB_ERC_TO_ERC
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
const isMediatorMode = bridgeMode => {
|
||||||
|
return bridgeMode === BRIDGE_MODES.AMB_ERC_TO_ERC || bridgeMode === BRIDGE_MODES.STAKE_AMB_ERC_TO_ERC
|
||||||
|
}
|
||||||
|
|
||||||
|
const getUnit = bridgeMode => {
|
||||||
|
let unitHome = null
|
||||||
|
let unitForeign = null
|
||||||
|
if (bridgeMode === BRIDGE_MODES.NATIVE_TO_ERC) {
|
||||||
|
unitHome = 'Native coins'
|
||||||
|
unitForeign = 'Tokens'
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.ERC_TO_ERC) {
|
||||||
|
unitHome = 'Tokens'
|
||||||
|
unitForeign = 'Tokens'
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.ERC_TO_NATIVE) {
|
||||||
|
unitHome = 'Native coins'
|
||||||
|
unitForeign = 'Tokens'
|
||||||
|
} else if (bridgeMode === BRIDGE_MODES.STAKE_AMB_ERC_TO_ERC) {
|
||||||
|
unitHome = 'Tokens'
|
||||||
|
unitForeign = 'Tokens'
|
||||||
|
} else {
|
||||||
|
throw new Error(`Unrecognized bridge mode: ${bridgeMode}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return { unitHome, unitForeign }
|
||||||
}
|
}
|
||||||
|
|
||||||
const parseValidatorEvent = event => {
|
const parseValidatorEvent = event => {
|
||||||
@ -75,8 +150,11 @@ const tryCall = async (method, fallbackValue) => {
|
|||||||
|
|
||||||
const getDeployedAtBlock = async contract => tryCall(contract.methods.deployedAtBlock(), 0)
|
const getDeployedAtBlock = async contract => tryCall(contract.methods.deployedAtBlock(), 0)
|
||||||
|
|
||||||
const getPastEventsOrSplit = async (contract, { event, fromBlock, toBlock, options }) => {
|
const getPastEvents = async (
|
||||||
let events = []
|
contract,
|
||||||
|
{ event = 'allEvents', fromBlock = toBN(0), toBlock = 'latest', options = {} }
|
||||||
|
) => {
|
||||||
|
let events
|
||||||
try {
|
try {
|
||||||
events = await contract.getPastEvents(event, {
|
events = await contract.getPastEvents(event, {
|
||||||
...options,
|
...options,
|
||||||
@ -84,19 +162,19 @@ const getPastEventsOrSplit = async (contract, { event, fromBlock, toBlock, optio
|
|||||||
toBlock
|
toBlock
|
||||||
})
|
})
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (e.message.includes('query returned more than') || e.message.toLowerCase().includes('timeout')) {
|
if (e.message.includes('query returned more than') && toBlock !== 'latest') {
|
||||||
const middle = toBN(fromBlock)
|
const middle = toBN(fromBlock)
|
||||||
.add(toBN(toBlock))
|
.add(toBN(toBlock))
|
||||||
.divRound(toBN(2))
|
.divRound(toBN(2))
|
||||||
const middlePlusOne = middle.add(toBN(1))
|
const middlePlusOne = middle.add(toBN(1))
|
||||||
|
|
||||||
const firstHalfEvents = await getPastEventsOrSplit(contract, {
|
const firstHalfEvents = await getPastEvents(contract, {
|
||||||
options,
|
options,
|
||||||
event,
|
event,
|
||||||
fromBlock,
|
fromBlock,
|
||||||
toBlock: middle
|
toBlock: middle
|
||||||
})
|
})
|
||||||
const secondHalfEvents = await getPastEventsOrSplit(contract, {
|
const secondHalfEvents = await getPastEvents(contract, {
|
||||||
options,
|
options,
|
||||||
event,
|
event,
|
||||||
fromBlock: middlePlusOne,
|
fromBlock: middlePlusOne,
|
||||||
@ -110,31 +188,6 @@ const getPastEventsOrSplit = async (contract, { event, fromBlock, toBlock, optio
|
|||||||
return events
|
return events
|
||||||
}
|
}
|
||||||
|
|
||||||
const getPastEvents = async (
|
|
||||||
contract,
|
|
||||||
{ event = 'allEvents', fromBlock = toBN(0), toBlock = 'latest', options = {} }
|
|
||||||
) => {
|
|
||||||
if (toBlock === 'latest') {
|
|
||||||
return contract.getPastEvents(event, {
|
|
||||||
...options,
|
|
||||||
fromBlock,
|
|
||||||
toBlock
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
const batchSize = 1000000
|
|
||||||
const to = toBN(toBlock)
|
|
||||||
const events = []
|
|
||||||
|
|
||||||
for (let from = toBN(fromBlock); from.lte(to); from = from.addn(batchSize + 1)) {
|
|
||||||
const opts = { event, fromBlock: from, toBlock: BN.min(to, from.addn(batchSize)), options }
|
|
||||||
const batch = await getPastEventsOrSplit(contract, opts)
|
|
||||||
events.push(batch)
|
|
||||||
}
|
|
||||||
|
|
||||||
return [].concat(...events)
|
|
||||||
}
|
|
||||||
|
|
||||||
const getValidatorList = async (address, eth, options) => {
|
const getValidatorList = async (address, eth, options) => {
|
||||||
options.logger && options.logger.debug && options.logger.debug('getting validatorList')
|
options.logger && options.logger.debug && options.logger.debug('getting validatorList')
|
||||||
|
|
||||||
@ -177,27 +230,20 @@ const gasPriceWithinLimits = (gasPrice, limits) => {
|
|||||||
const normalizeGasPrice = (oracleGasPrice, factor, limits = null) => {
|
const normalizeGasPrice = (oracleGasPrice, factor, limits = null) => {
|
||||||
let gasPrice = oracleGasPrice * factor
|
let gasPrice = oracleGasPrice * factor
|
||||||
gasPrice = gasPriceWithinLimits(gasPrice, limits)
|
gasPrice = gasPriceWithinLimits(gasPrice, limits)
|
||||||
return toWei(gasPrice.toFixed(2).toString(), 'gwei')
|
return toBN(toWei(gasPrice.toFixed(2).toString(), 'gwei'))
|
||||||
}
|
}
|
||||||
|
|
||||||
const gasPriceFromSupplier = async (web3, url, options = {}) => {
|
// fetchFn has to be supplied (instead of just url to oracle),
|
||||||
|
// because this utility function is shared between Browser and Node,
|
||||||
|
// we use built-in 'fetch' on browser side, and `node-fetch` package in Node.
|
||||||
|
const gasPriceFromSupplier = async (fetchFn, options = {}) => {
|
||||||
try {
|
try {
|
||||||
let json
|
let json
|
||||||
if (url === 'eip1559-gas-estimation') {
|
if (fetchFn) {
|
||||||
const { maxFeePerGas, maxPriorityFeePerGas } = await estimateFees(web3)
|
const response = await fetchFn()
|
||||||
const res = { maxFeePerGas: maxFeePerGas.toString(10), maxPriorityFeePerGas: maxPriorityFeePerGas.toString(10) }
|
|
||||||
options.logger &&
|
|
||||||
options.logger.debug &&
|
|
||||||
options.logger.debug(res, 'Gas price updated using eip1559-gas-estimation')
|
|
||||||
return res
|
|
||||||
}
|
|
||||||
if (url === 'gas-price-oracle') {
|
|
||||||
json = await gasPriceOracle.fetchGasPricesOffChain()
|
|
||||||
} else if (url) {
|
|
||||||
const response = await fetch(url, { timeout: 2000 })
|
|
||||||
json = await response.json()
|
json = await response.json()
|
||||||
} else {
|
} else {
|
||||||
return null
|
json = await gasPriceOracle.fetchGasPricesOffChain()
|
||||||
}
|
}
|
||||||
const oracleGasPrice = json[options.speedType]
|
const oracleGasPrice = json[options.speedType]
|
||||||
|
|
||||||
@ -214,7 +260,7 @@ const gasPriceFromSupplier = async (web3, url, options = {}) => {
|
|||||||
options.logger.debug &&
|
options.logger.debug &&
|
||||||
options.logger.debug({ oracleGasPrice, normalizedGasPrice }, 'Gas price updated using the API')
|
options.logger.debug({ oracleGasPrice, normalizedGasPrice }, 'Gas price updated using the API')
|
||||||
|
|
||||||
return { gasPrice: normalizedGasPrice }
|
return normalizedGasPrice
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
options.logger && options.logger.error && options.logger.error(`Gas Price API is not available. ${e.message}`)
|
options.logger && options.logger.error && options.logger.error(`Gas Price API is not available. ${e.message}`)
|
||||||
}
|
}
|
||||||
@ -223,11 +269,11 @@ const gasPriceFromSupplier = async (web3, url, options = {}) => {
|
|||||||
|
|
||||||
const gasPriceFromContract = async (bridgeContract, options = {}) => {
|
const gasPriceFromContract = async (bridgeContract, options = {}) => {
|
||||||
try {
|
try {
|
||||||
const gasPrice = (await bridgeContract.methods.gasPrice().call()).toString()
|
const gasPrice = await bridgeContract.methods.gasPrice().call()
|
||||||
options.logger &&
|
options.logger &&
|
||||||
options.logger.debug &&
|
options.logger.debug &&
|
||||||
options.logger.debug({ gasPrice }, 'Gas price updated using the contracts')
|
options.logger.debug({ gasPrice }, 'Gas price updated using the contracts')
|
||||||
return { gasPrice }
|
return gasPrice
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
options.logger &&
|
options.logger &&
|
||||||
options.logger.error &&
|
options.logger.error &&
|
||||||
@ -238,7 +284,10 @@ const gasPriceFromContract = async (bridgeContract, options = {}) => {
|
|||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
decodeBridgeMode,
|
decodeBridgeMode,
|
||||||
|
decodeFeeManagerMode,
|
||||||
getBridgeMode,
|
getBridgeMode,
|
||||||
|
getTokenType,
|
||||||
|
getUnit,
|
||||||
parseValidatorEvent,
|
parseValidatorEvent,
|
||||||
processValidatorsEvents,
|
processValidatorsEvents,
|
||||||
getValidatorList,
|
getValidatorList,
|
||||||
@ -247,5 +296,7 @@ module.exports = {
|
|||||||
normalizeGasPrice,
|
normalizeGasPrice,
|
||||||
gasPriceFromSupplier,
|
gasPriceFromSupplier,
|
||||||
gasPriceFromContract,
|
gasPriceFromContract,
|
||||||
gasPriceWithinLimits
|
gasPriceWithinLimits,
|
||||||
|
isErcToErcMode,
|
||||||
|
isMediatorMode
|
||||||
}
|
}
|
||||||
|
191
commons/v1Abis.js
Normal file
191
commons/v1Abis.js
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
const homeV1Abi = [
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'validatorContract',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'address'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
anonymous: false,
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'recipient',
|
||||||
|
type: 'address'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'value',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
name: 'Deposit',
|
||||||
|
type: 'event'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
anonymous: false,
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'recipient',
|
||||||
|
type: 'address'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'value',
|
||||||
|
type: 'uint256'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'transactionHash',
|
||||||
|
type: 'bytes32'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
name: 'Withdraw',
|
||||||
|
type: 'event'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'deployedAtBlock',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'requiredBlockConfirmations',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
const foreignViAbi = [
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'validatorContract',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'address'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
anonymous: false,
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'recipient',
|
||||||
|
type: 'address'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'value',
|
||||||
|
type: 'uint256'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'transactionHash',
|
||||||
|
type: 'bytes32'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
name: 'Deposit',
|
||||||
|
type: 'event'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
anonymous: false,
|
||||||
|
inputs: [
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'recipient',
|
||||||
|
type: 'address'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'value',
|
||||||
|
type: 'uint256'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
indexed: false,
|
||||||
|
name: 'homeGasPrice',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
name: 'Withdraw',
|
||||||
|
type: 'event'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'deployedAtBlock',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'erc677token',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'address'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
constant: true,
|
||||||
|
inputs: [],
|
||||||
|
name: 'requiredBlockConfirmations',
|
||||||
|
outputs: [
|
||||||
|
{
|
||||||
|
name: '',
|
||||||
|
type: 'uint256'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
payable: false,
|
||||||
|
stateMutability: 'view',
|
||||||
|
type: 'function'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
HOME_V1_ABI: homeV1Abi,
|
||||||
|
FOREIGN_V1_ABI: foreignViAbi
|
||||||
|
}
|
@ -1 +1 @@
|
|||||||
Subproject commit 908a48107919d4ab127f9af07d44d47eac91547e
|
Subproject commit 835742dfd8f1c869d4e7b61582155d250d6cf094
|
@ -16,7 +16,7 @@ Alternatively, if there are no changes except the playbooks, you can use the `ma
|
|||||||
./molecule.sh <scenario_name>
|
./molecule.sh <scenario_name>
|
||||||
```
|
```
|
||||||
|
|
||||||
In this case `master` branch will be used as a codebase for Monitor, Oracle and Contracts deployed by your local playbook.
|
In this case `master` branch will be used as a codebase for Monitor, UI, Oracle and Contracts deployed by your local playbook.
|
||||||
|
|
||||||
## Run the tests
|
## Run the tests
|
||||||
|
|
||||||
@ -29,6 +29,7 @@ Available scenarios:
|
|||||||
Scenario | Description
|
Scenario | Description
|
||||||
--- | ---
|
--- | ---
|
||||||
oracle | Deploys and checks standalone Oracle on Ubuntu host
|
oracle | Deploys and checks standalone Oracle on Ubuntu host
|
||||||
|
ui | Deploys and checks standalone UI on Ubuntu host
|
||||||
|
|
||||||
## Ultimate E2E tests
|
## Ultimate E2E tests
|
||||||
|
|
||||||
|
@ -16,6 +16,7 @@ platforms:
|
|||||||
children:
|
children:
|
||||||
- oracle
|
- oracle
|
||||||
- monitor
|
- monitor
|
||||||
|
- ui
|
||||||
image: ubuntu:16.04
|
image: ubuntu:16.04
|
||||||
privileged: true
|
privileged: true
|
||||||
network_mode: host
|
network_mode: host
|
||||||
@ -34,7 +35,7 @@ provisioner:
|
|||||||
inventory:
|
inventory:
|
||||||
host_vars:
|
host_vars:
|
||||||
multiple-host:
|
multiple-host:
|
||||||
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "6c48435bd464a53ed66ed62127c4dba8af75cf1a99a8ebe2680599948fbfbc6d"
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
MONITOR_PORT: 3003
|
MONITOR_PORT: 3003
|
||||||
syslog_server_port: "udp://127.0.0.1:514"
|
syslog_server_port: "udp://127.0.0.1:514"
|
||||||
verifier:
|
verifier:
|
||||||
|
@ -8,6 +8,7 @@ testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner(
|
|||||||
|
|
||||||
@pytest.mark.parametrize("service", [
|
@pytest.mark.parametrize("service", [
|
||||||
("poabridge"),
|
("poabridge"),
|
||||||
|
("tokenbridge-ui"),
|
||||||
("tokenbridge-monitor")
|
("tokenbridge-monitor")
|
||||||
])
|
])
|
||||||
def test_services(host, service):
|
def test_services(host, service):
|
||||||
@ -23,7 +24,7 @@ def test_services(host, service):
|
|||||||
("oracle_bridge_affirmation_1"),
|
("oracle_bridge_affirmation_1"),
|
||||||
("oracle_bridge_senderhome_1"),
|
("oracle_bridge_senderhome_1"),
|
||||||
("oracle_bridge_senderforeign_1"),
|
("oracle_bridge_senderforeign_1"),
|
||||||
("oracle_bridge_shutdown_1"),
|
("ui_ui_1"),
|
||||||
("monitor_monitor_1")
|
("monitor_monitor_1")
|
||||||
])
|
])
|
||||||
def test_docker_containers(host, name):
|
def test_docker_containers(host, name):
|
||||||
|
@ -33,7 +33,7 @@ provisioner:
|
|||||||
inventory:
|
inventory:
|
||||||
host_vars:
|
host_vars:
|
||||||
oracle-host:
|
oracle-host:
|
||||||
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "6c48435bd464a53ed66ed62127c4dba8af75cf1a99a8ebe2680599948fbfbc6d"
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
syslog_server_port: "udp://127.0.0.1:514"
|
syslog_server_port: "udp://127.0.0.1:514"
|
||||||
verifier:
|
verifier:
|
||||||
name: testinfra
|
name: testinfra
|
||||||
|
@ -14,7 +14,6 @@ testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner(
|
|||||||
("oracle_bridge_affirmation_1"),
|
("oracle_bridge_affirmation_1"),
|
||||||
("oracle_bridge_senderhome_1"),
|
("oracle_bridge_senderhome_1"),
|
||||||
("oracle_bridge_senderforeign_1"),
|
("oracle_bridge_senderforeign_1"),
|
||||||
("oracle_bridge_shutdown_1"),
|
|
||||||
])
|
])
|
||||||
def test_docker_containers(host, name):
|
def test_docker_containers(host, name):
|
||||||
container = host.docker(name)
|
container = host.docker(name)
|
||||||
|
@ -15,6 +15,7 @@ testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner(
|
|||||||
("/home/poadocker/bridge/contracts"),
|
("/home/poadocker/bridge/contracts"),
|
||||||
("/home/poadocker/bridge/oracle"),
|
("/home/poadocker/bridge/oracle"),
|
||||||
("/home/poadocker/bridge/monitor"),
|
("/home/poadocker/bridge/monitor"),
|
||||||
|
("/home/poadocker/bridge/ui"),
|
||||||
("/home/poadocker/bridge/parity")
|
("/home/poadocker/bridge/parity")
|
||||||
])
|
])
|
||||||
def test_existing_folders(host, path):
|
def test_existing_folders(host, path):
|
||||||
@ -27,7 +28,8 @@ def test_existing_folders(host, path):
|
|||||||
("/home/poadocker/bridge/commons/package.json"),
|
("/home/poadocker/bridge/commons/package.json"),
|
||||||
("/home/poadocker/bridge/contracts/package.json"),
|
("/home/poadocker/bridge/contracts/package.json"),
|
||||||
("/home/poadocker/bridge/oracle/package.json"),
|
("/home/poadocker/bridge/oracle/package.json"),
|
||||||
("/home/poadocker/bridge/monitor/package.json")
|
("/home/poadocker/bridge/monitor/package.json"),
|
||||||
|
("/home/poadocker/bridge/ui/package.json")
|
||||||
])
|
])
|
||||||
def test_existing_package_json(host, path):
|
def test_existing_package_json(host, path):
|
||||||
assert host.file(path).exists
|
assert host.file(path).exists
|
||||||
@ -39,7 +41,8 @@ def test_existing_package_json(host, path):
|
|||||||
("/home/poadocker/bridge/contracts/Dockerfile"),
|
("/home/poadocker/bridge/contracts/Dockerfile"),
|
||||||
("/home/poadocker/bridge/parity/Dockerfile"),
|
("/home/poadocker/bridge/parity/Dockerfile"),
|
||||||
("/home/poadocker/bridge/oracle/Dockerfile"),
|
("/home/poadocker/bridge/oracle/Dockerfile"),
|
||||||
("/home/poadocker/bridge/monitor/Dockerfile")
|
("/home/poadocker/bridge/monitor/Dockerfile"),
|
||||||
|
("/home/poadocker/bridge/ui/Dockerfile")
|
||||||
])
|
])
|
||||||
def test_existing_docker_files(host, path):
|
def test_existing_docker_files(host, path):
|
||||||
assert host.file(path).exists
|
assert host.file(path).exists
|
||||||
|
14
deployment-e2e/molecule/ui/Dockerfile.j2
Normal file
14
deployment-e2e/molecule/ui/Dockerfile.j2
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Molecule managed
|
||||||
|
|
||||||
|
{% if item.registry is defined %}
|
||||||
|
FROM {{ item.registry.url }}/{{ item.image }}
|
||||||
|
{% else %}
|
||||||
|
FROM {{ item.image }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
RUN if [ $(command -v apt-get) ]; then apt-get update && apt-get install -y python sudo bash ca-certificates && apt-get clean; \
|
||||||
|
elif [ $(command -v dnf) ]; then dnf makecache && dnf --assumeyes install python sudo python-devel python*-dnf bash && dnf clean all; \
|
||||||
|
elif [ $(command -v yum) ]; then yum makecache fast && yum install -y python sudo yum-plugin-ovl bash && sed -i 's/plugins=0/plugins=1/g' /etc/yum.conf && yum clean all; \
|
||||||
|
elif [ $(command -v zypper) ]; then zypper refresh && zypper install -y python sudo bash python-xml && zypper clean -a; \
|
||||||
|
elif [ $(command -v apk) ]; then apk update && apk add --no-cache python sudo bash ca-certificates; \
|
||||||
|
elif [ $(command -v xbps-install) ]; then xbps-install -Syu && xbps-install -y python sudo bash ca-certificates && xbps-remove -O; fi
|
55
deployment-e2e/molecule/ui/molecule.yml
Normal file
55
deployment-e2e/molecule/ui/molecule.yml
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
---
|
||||||
|
dependency:
|
||||||
|
name: galaxy
|
||||||
|
driver:
|
||||||
|
name: docker
|
||||||
|
lint:
|
||||||
|
name: yamllint
|
||||||
|
enabled: True
|
||||||
|
options:
|
||||||
|
config-data:
|
||||||
|
ignore: ../../hosts.yml
|
||||||
|
platforms:
|
||||||
|
- name: ui-host
|
||||||
|
groups:
|
||||||
|
- example
|
||||||
|
children:
|
||||||
|
- ui
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
provisioner:
|
||||||
|
name: ansible
|
||||||
|
lint:
|
||||||
|
name: ansible-lint
|
||||||
|
enabled: True
|
||||||
|
options:
|
||||||
|
r: ["bug"]
|
||||||
|
playbooks:
|
||||||
|
prepare: ../prepare.yml
|
||||||
|
converge: ../../../deployment/site.yml
|
||||||
|
inventory:
|
||||||
|
host_vars:
|
||||||
|
ui-host:
|
||||||
|
syslog_server_port: "udp://127.0.0.1:514"
|
||||||
|
verifier:
|
||||||
|
name: testinfra
|
||||||
|
lint:
|
||||||
|
name: flake8
|
||||||
|
additional_files_or_dirs:
|
||||||
|
- ../../tests/*
|
||||||
|
scenario:
|
||||||
|
name: ui
|
||||||
|
test_sequence:
|
||||||
|
- lint
|
||||||
|
- cleanup
|
||||||
|
- destroy
|
||||||
|
- dependency
|
||||||
|
- syntax
|
||||||
|
- create
|
||||||
|
- prepare
|
||||||
|
- converge
|
||||||
|
- verify
|
||||||
|
- destroy
|
48
deployment-e2e/molecule/ui/tests/test_ui.py
Normal file
48
deployment-e2e/molecule/ui/tests/test_ui.py
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
import os
|
||||||
|
import pytest
|
||||||
|
import testinfra.utils.ansible_runner
|
||||||
|
|
||||||
|
testinfra_hosts = testinfra.utils.ansible_runner.AnsibleRunner(
|
||||||
|
os.environ['MOLECULE_INVENTORY_FILE']).get_hosts('ui')
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("name", [
|
||||||
|
("ui_ui_1")
|
||||||
|
])
|
||||||
|
def test_docker_containers(host, name):
|
||||||
|
container = host.docker(name)
|
||||||
|
assert container.is_running
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("service", [
|
||||||
|
("tokenbridge-ui"),
|
||||||
|
("rsyslog")
|
||||||
|
])
|
||||||
|
def test_services(host, service):
|
||||||
|
assert host.service(service).is_enabled
|
||||||
|
assert host.service(service).is_running
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("filename", [
|
||||||
|
("/etc/rsyslog.d/32-ui-docker.conf"),
|
||||||
|
("/etc/rsyslog.d/37-ui-remote-logging.conf")
|
||||||
|
])
|
||||||
|
def test_logging(host, filename):
|
||||||
|
assert host.file(filename).exists
|
||||||
|
assert host.file(filename).mode == 0o0644
|
||||||
|
|
||||||
|
|
||||||
|
def test_index_page_title(host):
|
||||||
|
assert host.run_test(
|
||||||
|
'curl -s http://localhost:3001 | '
|
||||||
|
'grep "<title>" | '
|
||||||
|
'grep -q "TokenBridge UI app"'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_index_page_error(host):
|
||||||
|
assert host.run_expect(
|
||||||
|
[1],
|
||||||
|
'curl -s http://localhost:3001 | '
|
||||||
|
'grep -i -q "error"'
|
||||||
|
)
|
@ -0,0 +1,14 @@
|
|||||||
|
# Molecule managed
|
||||||
|
|
||||||
|
{% if item.registry is defined %}
|
||||||
|
FROM {{ item.registry.url }}/{{ item.image }}
|
||||||
|
{% else %}
|
||||||
|
FROM {{ item.image }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
RUN if [ $(command -v apt-get) ]; then apt-get update && apt-get install -y python sudo bash ca-certificates && apt-get clean; \
|
||||||
|
elif [ $(command -v dnf) ]; then dnf makecache && dnf --assumeyes install python sudo python-devel python*-dnf bash && dnf clean all; \
|
||||||
|
elif [ $(command -v yum) ]; then yum makecache fast && yum install -y python sudo yum-plugin-ovl bash && sed -i 's/plugins=0/plugins=1/g' /etc/yum.conf && yum clean all; \
|
||||||
|
elif [ $(command -v zypper) ]; then zypper refresh && zypper install -y python sudo bash python-xml && zypper clean -a; \
|
||||||
|
elif [ $(command -v apk) ]; then apk update && apk add --no-cache python sudo bash ca-certificates; \
|
||||||
|
elif [ $(command -v xbps-install) ]; then xbps-install -Syu && xbps-install -y python sudo bash ca-certificates && xbps-remove -O; fi
|
@ -0,0 +1,52 @@
|
|||||||
|
---
|
||||||
|
driver:
|
||||||
|
name: docker
|
||||||
|
platforms:
|
||||||
|
- name: oracle-amb-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- amb
|
||||||
|
children:
|
||||||
|
- oracle
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- name: ui-amb-stake-erc-to-erc-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- amb-stake-erc-to-erc
|
||||||
|
children:
|
||||||
|
- ui
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
provisioner:
|
||||||
|
name: ansible
|
||||||
|
playbooks:
|
||||||
|
prepare: ../prepare.yml
|
||||||
|
converge: ../ultimate-commons/converge.yml
|
||||||
|
inventory:
|
||||||
|
host_vars:
|
||||||
|
oracle-amb-host:
|
||||||
|
ORACLE_VALIDATOR_ADDRESS: "0xaaB52d66283F7A1D5978bcFcB55721ACB467384b"
|
||||||
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
|
ui-amb-stake-erc-to-erc-host:
|
||||||
|
COMMON_HOME_RPC_URL: "http://localhost:8541"
|
||||||
|
COMMON_FOREIGN_RPC_URL: "http://localhost:8542"
|
||||||
|
verifier:
|
||||||
|
name: testinfra
|
||||||
|
lint:
|
||||||
|
name: flake8
|
||||||
|
scenario:
|
||||||
|
name: ultimate-amb-stake-erc-to-erc
|
||||||
|
test_sequence:
|
||||||
|
- cleanup
|
||||||
|
- destroy
|
||||||
|
- syntax
|
||||||
|
- create
|
||||||
|
- prepare
|
||||||
|
- converge
|
@ -9,15 +9,12 @@
|
|||||||
- oracle_net_db_bridge_request
|
- oracle_net_db_bridge_request
|
||||||
- oracle_net_db_bridge_collected
|
- oracle_net_db_bridge_collected
|
||||||
- oracle_net_db_bridge_affirmation
|
- oracle_net_db_bridge_affirmation
|
||||||
- oracle_net_db_bridge_information
|
|
||||||
- oracle_net_db_bridge_transfer
|
- oracle_net_db_bridge_transfer
|
||||||
- oracle_net_db_bridge_senderhome
|
- oracle_net_db_bridge_senderhome
|
||||||
- oracle_net_db_bridge_senderforeign
|
- oracle_net_db_bridge_senderforeign
|
||||||
- oracle_net_db_bridge_shutdown
|
|
||||||
- oracle_net_rabbit_bridge_request
|
- oracle_net_rabbit_bridge_request
|
||||||
- oracle_net_rabbit_bridge_collected
|
- oracle_net_rabbit_bridge_collected
|
||||||
- oracle_net_rabbit_bridge_affirmation
|
- oracle_net_rabbit_bridge_affirmation
|
||||||
- oracle_net_rabbit_bridge_information
|
|
||||||
- oracle_net_rabbit_bridge_transfer
|
- oracle_net_rabbit_bridge_transfer
|
||||||
- oracle_net_rabbit_bridge_senderhome
|
- oracle_net_rabbit_bridge_senderhome
|
||||||
- oracle_net_rabbit_bridge_senderforeign
|
- oracle_net_rabbit_bridge_senderforeign
|
||||||
|
14
deployment-e2e/molecule/ultimate-erc-to-erc/Dockerfile.j2
Normal file
14
deployment-e2e/molecule/ultimate-erc-to-erc/Dockerfile.j2
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Molecule managed
|
||||||
|
|
||||||
|
{% if item.registry is defined %}
|
||||||
|
FROM {{ item.registry.url }}/{{ item.image }}
|
||||||
|
{% else %}
|
||||||
|
FROM {{ item.image }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
RUN if [ $(command -v apt-get) ]; then apt-get update && apt-get install -y python sudo bash ca-certificates && apt-get clean; \
|
||||||
|
elif [ $(command -v dnf) ]; then dnf makecache && dnf --assumeyes install python sudo python-devel python*-dnf bash && dnf clean all; \
|
||||||
|
elif [ $(command -v yum) ]; then yum makecache fast && yum install -y python sudo yum-plugin-ovl bash && sed -i 's/plugins=0/plugins=1/g' /etc/yum.conf && yum clean all; \
|
||||||
|
elif [ $(command -v zypper) ]; then zypper refresh && zypper install -y python sudo bash python-xml && zypper clean -a; \
|
||||||
|
elif [ $(command -v apk) ]; then apk update && apk add --no-cache python sudo bash ca-certificates; \
|
||||||
|
elif [ $(command -v xbps-install) ]; then xbps-install -Syu && xbps-install -y python sudo bash ca-certificates && xbps-remove -O; fi
|
52
deployment-e2e/molecule/ultimate-erc-to-erc/molecule.yml
Normal file
52
deployment-e2e/molecule/ultimate-erc-to-erc/molecule.yml
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
---
|
||||||
|
driver:
|
||||||
|
name: docker
|
||||||
|
platforms:
|
||||||
|
- name: oracle-erc-to-erc-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- erc-to-erc
|
||||||
|
children:
|
||||||
|
- oracle
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- name: ui-erc-to-erc-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- erc-to-erc
|
||||||
|
children:
|
||||||
|
- ui
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
provisioner:
|
||||||
|
name: ansible
|
||||||
|
playbooks:
|
||||||
|
prepare: ../prepare.yml
|
||||||
|
converge: ../ultimate-commons/converge.yml
|
||||||
|
inventory:
|
||||||
|
host_vars:
|
||||||
|
oracle-erc-to-erc-host:
|
||||||
|
ORACLE_VALIDATOR_ADDRESS: "0xaaB52d66283F7A1D5978bcFcB55721ACB467384b"
|
||||||
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
|
ui-erc-to-erc-host:
|
||||||
|
COMMON_HOME_RPC_URL: "http://localhost:8541"
|
||||||
|
COMMON_FOREIGN_RPC_URL: "http://localhost:8542"
|
||||||
|
verifier:
|
||||||
|
name: testinfra
|
||||||
|
lint:
|
||||||
|
name: flake8
|
||||||
|
scenario:
|
||||||
|
name: ultimate-erc-to-erc
|
||||||
|
test_sequence:
|
||||||
|
- cleanup
|
||||||
|
- destroy
|
||||||
|
- syntax
|
||||||
|
- create
|
||||||
|
- prepare
|
||||||
|
- converge
|
@ -13,6 +13,17 @@ platforms:
|
|||||||
network_mode: host
|
network_mode: host
|
||||||
volumes:
|
volumes:
|
||||||
- /var/run/docker.sock:/var/run/docker.sock
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- name: ui-erc-to-native-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- erc-to-native
|
||||||
|
children:
|
||||||
|
- ui
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
provisioner:
|
provisioner:
|
||||||
name: ansible
|
name: ansible
|
||||||
playbooks:
|
playbooks:
|
||||||
@ -25,6 +36,9 @@ provisioner:
|
|||||||
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
ORACLE_HOME_START_BLOCK: 1
|
ORACLE_HOME_START_BLOCK: 1
|
||||||
ORACLE_FOREIGN_START_BLOCK: 1
|
ORACLE_FOREIGN_START_BLOCK: 1
|
||||||
|
ui-erc-to-native-host:
|
||||||
|
COMMON_HOME_RPC_URL: "http://localhost:8541"
|
||||||
|
COMMON_FOREIGN_RPC_URL: "http://localhost:8542"
|
||||||
verifier:
|
verifier:
|
||||||
name: testinfra
|
name: testinfra
|
||||||
lint:
|
lint:
|
||||||
|
14
deployment-e2e/molecule/ultimate-native-to-erc/Dockerfile.j2
Normal file
14
deployment-e2e/molecule/ultimate-native-to-erc/Dockerfile.j2
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
# Molecule managed
|
||||||
|
|
||||||
|
{% if item.registry is defined %}
|
||||||
|
FROM {{ item.registry.url }}/{{ item.image }}
|
||||||
|
{% else %}
|
||||||
|
FROM {{ item.image }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
RUN if [ $(command -v apt-get) ]; then apt-get update && apt-get install -y python sudo bash ca-certificates && apt-get clean; \
|
||||||
|
elif [ $(command -v dnf) ]; then dnf makecache && dnf --assumeyes install python sudo python-devel python*-dnf bash && dnf clean all; \
|
||||||
|
elif [ $(command -v yum) ]; then yum makecache fast && yum install -y python sudo yum-plugin-ovl bash && sed -i 's/plugins=0/plugins=1/g' /etc/yum.conf && yum clean all; \
|
||||||
|
elif [ $(command -v zypper) ]; then zypper refresh && zypper install -y python sudo bash python-xml && zypper clean -a; \
|
||||||
|
elif [ $(command -v apk) ]; then apk update && apk add --no-cache python sudo bash ca-certificates; \
|
||||||
|
elif [ $(command -v xbps-install) ]; then xbps-install -Syu && xbps-install -y python sudo bash ca-certificates && xbps-remove -O; fi
|
52
deployment-e2e/molecule/ultimate-native-to-erc/molecule.yml
Normal file
52
deployment-e2e/molecule/ultimate-native-to-erc/molecule.yml
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
---
|
||||||
|
driver:
|
||||||
|
name: docker
|
||||||
|
platforms:
|
||||||
|
- name: oracle-native-to-erc-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- native-to-erc
|
||||||
|
children:
|
||||||
|
- oracle
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- name: ui-native-to-erc-host
|
||||||
|
groups:
|
||||||
|
- ultimate
|
||||||
|
- native-to-erc
|
||||||
|
children:
|
||||||
|
- ui
|
||||||
|
image: ubuntu:16.04
|
||||||
|
privileged: true
|
||||||
|
network_mode: host
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
provisioner:
|
||||||
|
name: ansible
|
||||||
|
playbooks:
|
||||||
|
prepare: ../prepare.yml
|
||||||
|
converge: ../ultimate-commons/converge.yml
|
||||||
|
inventory:
|
||||||
|
host_vars:
|
||||||
|
oracle-native-to-erc-host:
|
||||||
|
ORACLE_VALIDATOR_ADDRESS: "0xaaB52d66283F7A1D5978bcFcB55721ACB467384b"
|
||||||
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "8e829f695aed89a154550f30262f1529582cc49dc30eff74a6b491359e0230f9"
|
||||||
|
ui-native-to-erc-host:
|
||||||
|
COMMON_HOME_RPC_URL: "http://localhost:8541"
|
||||||
|
COMMON_FOREIGN_RPC_URL: "http://localhost:8542"
|
||||||
|
verifier:
|
||||||
|
name: testinfra
|
||||||
|
lint:
|
||||||
|
name: flake8
|
||||||
|
scenario:
|
||||||
|
name: ultimate-native-to-erc
|
||||||
|
test_sequence:
|
||||||
|
- cleanup
|
||||||
|
- destroy
|
||||||
|
- syntax
|
||||||
|
- create
|
||||||
|
- prepare
|
||||||
|
- converge
|
@ -34,6 +34,14 @@ cp hosts.yml.example hosts.yml
|
|||||||
#syslog_server_port: "<protocol>://<ip>:<port>" # When this parameter is set all bridge logs will be redirected to <ip>:<port> address.
|
#syslog_server_port: "<protocol>://<ip>:<port>" # When this parameter is set all bridge logs will be redirected to <ip>:<port> address.
|
||||||
<host_ip_B>:
|
<host_ip_B>:
|
||||||
# (...)
|
# (...)
|
||||||
|
ui:
|
||||||
|
hosts:
|
||||||
|
<host_ip_B>:
|
||||||
|
ansible_user: <user>
|
||||||
|
#syslog_server_port: "<protocol>://<ip>:<port>"
|
||||||
|
<host_ip_C>:
|
||||||
|
ansible_user: <user>
|
||||||
|
#syslog_server_port: "<protocol>://<ip>:<port>"
|
||||||
monitor:
|
monitor:
|
||||||
hosts:
|
hosts:
|
||||||
<host_ip_B>:
|
<host_ip_B>:
|
||||||
@ -42,7 +50,18 @@ cp hosts.yml.example hosts.yml
|
|||||||
#monitor_cron_schedule: "*/4 * * * *" # When this parameter is set, it will overwrite default schedule for performing checks
|
#monitor_cron_schedule: "*/4 * * * *" # When this parameter is set, it will overwrite default schedule for performing checks
|
||||||
```
|
```
|
||||||
|
|
||||||
The config above would install the Oracle on `<host_ip_A>`, and both Oracle and Monitor on `<host_ip_B>`.
|
The config above would install the Oracle on `<host_ip_A>`, UI on `<host_ip_C>`, and both Oracle, UI and Monitor on `<host_ip_B>`.
|
||||||
|
|
||||||
|
Example config for installing only UI:
|
||||||
|
```yaml
|
||||||
|
<bridge_name>:
|
||||||
|
children:
|
||||||
|
oracle:
|
||||||
|
hosts:
|
||||||
|
ui:
|
||||||
|
hosts:
|
||||||
|
<host_ip>:
|
||||||
|
ansible_user: <user>
|
||||||
```
|
```
|
||||||
|
|
||||||
| Value | Description |
|
| Value | Description |
|
||||||
|
@ -44,6 +44,7 @@ The deployed components have the following services:
|
|||||||
Component | Service Name
|
Component | Service Name
|
||||||
--- | ---
|
--- | ---
|
||||||
Oracle | poabridge
|
Oracle | poabridge
|
||||||
|
UI | tokenbridge-ui
|
||||||
Monitor | tokenbridge-monitor
|
Monitor | tokenbridge-monitor
|
||||||
|
|
||||||
Use the default `SysVinit` commands to `start`, `stop`, `restart`, and `rebuild` the service and to check the `status` of the service.
|
Use the default `SysVinit` commands to `start`, `stop`, `restart`, and `rebuild` the service and to check the `status` of the service.
|
||||||
|
4
deployment/group_vars/amb-stake-erc-to-erc.yml
Normal file
4
deployment/group_vars/amb-stake-erc-to-erc.yml
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS: "0xc26Aa60Ff574f157616D3aEE70e08aAC129E1dFC"
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0xc26Aa60Ff574f157616D3aEE70e08aAC129E1dFC"
|
||||||
|
UI_PORT: 3003
|
@ -1,5 +1,5 @@
|
|||||||
---
|
---
|
||||||
ORACLE_BRIDGE_MODE: "ARBITRARY_MESSAGE"
|
ORACLE_BRIDGE_MODE: "ARBITRARY_MESSAGE"
|
||||||
COMMON_HOME_BRIDGE_ADDRESS: "0x8397be90BCF57b0B71219f555Fe121b22e5a994C"
|
COMMON_HOME_BRIDGE_ADDRESS: "0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0"
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x1feB40aD9420b186F019A717c37f5546165d411E"
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0"
|
||||||
MONITOR_PORT: 3013
|
MONITOR_PORT: 3013
|
||||||
|
@ -1,14 +1,19 @@
|
|||||||
---
|
---
|
||||||
## General settings
|
## General settings
|
||||||
ORACLE_BRIDGE_MODE: "ERC_TO_NATIVE"
|
ORACLE_BRIDGE_MODE: "ERC_TO_NATIVE"
|
||||||
|
UI_NATIVE_TOKEN_DISPLAY_NAME: "xDai"
|
||||||
|
|
||||||
## Home contract
|
## Home contract
|
||||||
COMMON_HOME_RPC_URL: "https://dai.poa.network"
|
COMMON_HOME_RPC_URL: "https://dai.poa.network"
|
||||||
|
UI_HOME_NETWORK_DISPLAY_NAME: "xDai chain"
|
||||||
|
UI_HOME_WITHOUT_EVENTS: false
|
||||||
COMMON_HOME_BRIDGE_ADDRESS: "0x7301CFA0e1756B71869E93d4e4Dca5c7d0eb0AA6"
|
COMMON_HOME_BRIDGE_ADDRESS: "0x7301CFA0e1756B71869E93d4e4Dca5c7d0eb0AA6"
|
||||||
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
||||||
|
|
||||||
## Foreign contract
|
## Foreign contract
|
||||||
COMMON_FOREIGN_RPC_URL: "https://mainnet.infura.io"
|
COMMON_FOREIGN_RPC_URL: "https://mainnet.infura.io"
|
||||||
|
UI_FOREIGN_NETWORK_DISPLAY_NAME: "Ethereum Mainnet"
|
||||||
|
UI_FOREIGN_WITHOUT_EVENTS: false
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x4aa42145Aa6Ebf72e164C9bBC74fbD3788045016"
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x4aa42145Aa6Ebf72e164C9bBC74fbD3788045016"
|
||||||
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 5000
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 5000
|
||||||
|
|
||||||
@ -25,6 +30,20 @@ COMMON_FOREIGN_GAS_PRICE_FALLBACK: 10000000000
|
|||||||
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
||||||
|
|
||||||
|
## UI
|
||||||
|
UI_TITLE: "TokenBridge UI app - %c"
|
||||||
|
UI_OG_TITLE: "POA Bridge UI"
|
||||||
|
UI_DESCRIPTION: "The TokenBridge serves as a method of transferring MakerDAO stable tokens between the Ethereum network to xDai chain in a quick and cost-efficient manner."
|
||||||
|
UI_PORT: 3001
|
||||||
|
UI_HOME_EXPLORER_TX_TEMPLATE: https://blockscout.com/poa/dai/tx/%s
|
||||||
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE: https://blockscout.com/eth/mainnet/tx/%s
|
||||||
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/poa/dai/address/%s
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/eth/mainnet/address/%s
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_STYLES: "core"
|
||||||
|
UI_PUBLIC_URL: "https://dai-bridge.poa.network"
|
||||||
|
|
||||||
## Monitor
|
## Monitor
|
||||||
MONITOR_BRIDGE_NAME: "xdai"
|
MONITOR_BRIDGE_NAME: "xdai"
|
||||||
MONITOR_PORT: 3003
|
MONITOR_PORT: 3003
|
||||||
|
6
deployment/group_vars/erc-to-erc.yml
Normal file
6
deployment/group_vars/erc-to-erc.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
ORACLE_BRIDGE_MODE: "ERC_TO_ERC"
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS: "0x1feB40aD9420b186F019A717c37f5546165d411E"
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x4a58D6d8D416a5fBCAcf3dC52eb8bE8948E25127"
|
||||||
|
UI_PORT: 3001
|
||||||
|
MONITOR_PORT: 3011
|
@ -1,5 +1,6 @@
|
|||||||
---
|
---
|
||||||
ORACLE_BRIDGE_MODE: "ERC_TO_NATIVE"
|
ORACLE_BRIDGE_MODE: "ERC_TO_NATIVE"
|
||||||
COMMON_HOME_BRIDGE_ADDRESS: "0x5118AC62AE912Dd5B51EEfF7338c4fcb0248Ba8c"
|
COMMON_HOME_BRIDGE_ADDRESS: "0x488Af810997eD1730cB3a3918cD83b3216E6eAda"
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x32198D570fffC7033641F8A9094FFDCaAEF42624"
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x488Af810997eD1730cB3a3918cD83b3216E6eAda"
|
||||||
|
UI_PORT: 3002
|
||||||
MONITOR_PORT: 3012
|
MONITOR_PORT: 3012
|
||||||
|
@ -1,17 +1,22 @@
|
|||||||
---
|
---
|
||||||
## General settings
|
## General settings
|
||||||
ORACLE_BRIDGE_MODE: "ARBITRARY_MESSAGE"
|
ORACLE_BRIDGE_MODE: "NATIVE_TO_ERC"
|
||||||
ORACLE_LOG_LEVEL: debug
|
ORACLE_LOG_LEVEL: debug
|
||||||
|
UI_NATIVE_TOKEN_DISPLAY_NAME: "POA"
|
||||||
|
|
||||||
## Home contract
|
## Home contract
|
||||||
COMMON_HOME_RPC_URL: "https://sokol.poa.network"
|
COMMON_HOME_RPC_URL: "https://sokol.poa.network"
|
||||||
COMMON_HOME_BRIDGE_ADDRESS: "0x59ba90A588ce732AB33FD32Aab1b58c21400A0f6"
|
UI_HOME_NETWORK_DISPLAY_NAME: "POA Sokol"
|
||||||
|
UI_HOME_WITHOUT_EVENTS: false
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS: "0x98aFdE294f1C46aA0a27Cc4049ED337F879d8976"
|
||||||
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
||||||
|
|
||||||
## Foreign contract
|
## Foreign contract
|
||||||
COMMON_FOREIGN_RPC_URL: "https://kovan.infura.io/v3/5d7bd94c50ed43fab1cb8e74f58678b0"
|
COMMON_FOREIGN_RPC_URL: "https://sokol.poa.network"
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS: "0xdA4a49a00F4fF4A5988b9AceE95f99e3b2c208b6"
|
UI_FOREIGN_NETWORK_DISPLAY_NAME: "Kovan"
|
||||||
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 5000
|
UI_FOREIGN_WITHOUT_EVENTS: false
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x5a584f4C30B36f282848dAc9a2b20E7BEF481981"
|
||||||
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 1000
|
||||||
|
|
||||||
## Home Gasprice
|
## Home Gasprice
|
||||||
COMMON_HOME_GAS_PRICE_SUPPLIER_URL: "https://gasprice.poa.network/"
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL: "https://gasprice.poa.network/"
|
||||||
@ -27,12 +32,26 @@ COMMON_FOREIGN_GAS_PRICE_FALLBACK: 1000000000 # in wei
|
|||||||
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
||||||
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
|
||||||
|
## UI
|
||||||
|
UI_TITLE: "TokenBridge UI app - %c"
|
||||||
|
UI_OG_TITLE: "POA Bridge UI"
|
||||||
|
UI_DESCRIPTION: "The POA cross-chain bridge serves as a method of transferring POA native tokens from the POA Network to the Ethereum network in a quick and cost-efficient manner."
|
||||||
|
UI_PORT: 3001
|
||||||
|
UI_HOME_EXPLORER_TX_TEMPLATE: https://blockscout.com/poa/sokol/tx/%s
|
||||||
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE: https://blockscout.com/eth/kovan/tx/%s
|
||||||
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/poa/sokol/address/%s
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/eth/kovan/address/%s
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_STYLES: "core"
|
||||||
|
UI_PUBLIC_URL: "http://localhost:3001"
|
||||||
|
|
||||||
## Monitor
|
## Monitor
|
||||||
MONITOR_BRIDGE_NAME: "bridge"
|
MONITOR_BRIDGE_NAME: "bridge"
|
||||||
MONITOR_PORT: 3003
|
MONITOR_PORT: 3003
|
||||||
MONITOR_CACHE_EVENTS: "false"
|
MONITOR_CACHE_EVENTS: "false"
|
||||||
MONITOR_HOME_START_BLOCK: 20821049
|
MONITOR_HOME_START_BLOCK: 0
|
||||||
MONITOR_FOREIGN_START_BLOCK: 24773297
|
MONITOR_FOREIGN_START_BLOCK: 0
|
||||||
MONITOR_VALIDATOR_HOME_TX_LIMIT: 300000
|
MONITOR_VALIDATOR_HOME_TX_LIMIT: 300000
|
||||||
MONITOR_VALIDATOR_FOREIGN_TX_LIMIT: 300000
|
MONITOR_VALIDATOR_FOREIGN_TX_LIMIT: 300000
|
||||||
MONITOR_TX_NUMBER_THRESHOLD: 100
|
MONITOR_TX_NUMBER_THRESHOLD: 100
|
||||||
|
6
deployment/group_vars/native-to-erc.yml
Normal file
6
deployment/group_vars/native-to-erc.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
ORACLE_BRIDGE_MODE: "NATIVE_TO_ERC"
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS: "0x32198D570fffC7033641F8A9094FFDCaAEF42624"
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x2B6871b9B02F73fa24F4864322CdC78604207769"
|
||||||
|
UI_PORT: 3000
|
||||||
|
MONITOR_PORT: 3010
|
@ -1,14 +1,19 @@
|
|||||||
---
|
---
|
||||||
## General settings
|
## General settings
|
||||||
|
UI_NATIVE_TOKEN_DISPLAY_NAME: "POA"
|
||||||
ORACLE_ALLOW_HTTP_FOR_RPC: yes
|
ORACLE_ALLOW_HTTP_FOR_RPC: yes
|
||||||
ORACLE_LOG_LEVEL: debug
|
ORACLE_LOG_LEVEL: debug
|
||||||
|
|
||||||
## Home contract
|
## Home contract
|
||||||
COMMON_HOME_RPC_URL: "http://parity1:8545"
|
COMMON_HOME_RPC_URL: "http://parity1:8545"
|
||||||
|
UI_HOME_NETWORK_DISPLAY_NAME: "POA Sokol"
|
||||||
|
UI_HOME_WITHOUT_EVENTS: false
|
||||||
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
ORACLE_HOME_RPC_POLLING_INTERVAL: 5000
|
||||||
|
|
||||||
## Foreign contract
|
## Foreign contract
|
||||||
COMMON_FOREIGN_RPC_URL: "http://parity2:8545"
|
COMMON_FOREIGN_RPC_URL: "http://parity2:8545"
|
||||||
|
UI_FOREIGN_NETWORK_DISPLAY_NAME: "Kovan"
|
||||||
|
UI_FOREIGN_WITHOUT_EVENTS: false
|
||||||
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 1000
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 1000
|
||||||
|
|
||||||
## Home Gasprice
|
## Home Gasprice
|
||||||
@ -25,6 +30,20 @@ COMMON_FOREIGN_GAS_PRICE_FALLBACK: 1000000000 # in wei
|
|||||||
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
||||||
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
|
||||||
|
|
||||||
|
#ui
|
||||||
|
UI_TITLE: "TokenBridge UI app - %c"
|
||||||
|
UI_OG_TITLE: "POA Bridge UI"
|
||||||
|
UI_DESCRIPTION: "The POA cross-chain bridge serves as a method of transferring POA native tokens from the POA Network to the Ethereum network in a quick and cost-efficient manner."
|
||||||
|
UI_HOME_EXPLORER_TX_TEMPLATE: https://blockscout.com/poa/sokol/tx/%s
|
||||||
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE: https://blockscout.com/eth/kovan/tx/%s
|
||||||
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/poa/sokol/address/%s
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/eth/kovan/address/%s
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_STYLES: "core"
|
||||||
|
UI_PUBLIC_URL: "http://localhost"
|
||||||
|
|
||||||
#monitor
|
#monitor
|
||||||
MONITOR_BRIDGE_NAME: "bridge"
|
MONITOR_BRIDGE_NAME: "bridge"
|
||||||
MONITOR_CACHE_EVENTS: "true"
|
MONITOR_CACHE_EVENTS: "true"
|
||||||
|
56
deployment/group_vars/wetc.yml
Normal file
56
deployment/group_vars/wetc.yml
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
---
|
||||||
|
## General settings
|
||||||
|
ORACLE_BRIDGE_MODE: "NATIVE_TO_ERC"
|
||||||
|
UI_NATIVE_TOKEN_DISPLAY_NAME: "ETC"
|
||||||
|
|
||||||
|
## Home contract
|
||||||
|
COMMON_HOME_RPC_URL: "https://www.ethercluster.com/etc"
|
||||||
|
UI_HOME_NETWORK_DISPLAY_NAME: "Ethereum Classic"
|
||||||
|
UI_HOME_WITHOUT_EVENTS: false
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS: "0x073081832B4Ecdce79d4D6753565c85Ba4b3BeA9"
|
||||||
|
ORACLE_HOME_RPC_POLLING_INTERVAL: 7000
|
||||||
|
|
||||||
|
## Foreign contract
|
||||||
|
COMMON_FOREIGN_RPC_URL: "https://mainnet.infura.io/"
|
||||||
|
UI_FOREIGN_NETWORK_DISPLAY_NAME: "Ethereum Mainnet"
|
||||||
|
UI_FOREIGN_WITHOUT_EVENTS: false
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS: "0x0cB781EE62F815bdD9CD4c2210aE8600d43e7040"
|
||||||
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL: 7000
|
||||||
|
|
||||||
|
## Home Gasprice
|
||||||
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL: "https://gasprice-etc.poa.network/"
|
||||||
|
COMMON_HOME_GAS_PRICE_SPEED_TYPE: "standard"
|
||||||
|
COMMON_HOME_GAS_PRICE_FALLBACK: 15000000000 # in wei
|
||||||
|
ORACLE_HOME_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
COMMON_HOME_GAS_PRICE_FACTOR: 1
|
||||||
|
|
||||||
|
## Foreign Gasprice
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL: "https://gasprice.poa.network/"
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE: "standard"
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FALLBACK: 10000000000 # in wei
|
||||||
|
ORACLE_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FACTOR: 1
|
||||||
|
|
||||||
|
## UI
|
||||||
|
UI_TITLE: "TokenBridge UI app - %c"
|
||||||
|
UI_OG_TITLE: "POA Bridge UI"
|
||||||
|
UI_DESCRIPTION: "The TokenBridge serves as a method of transferring native tokens from the Ethereum Classic Network to the Ethereum network in a quick and cost-efficient manner."
|
||||||
|
UI_PORT: 3001
|
||||||
|
UI_HOME_EXPLORER_TX_TEMPLATE: https://blockscout.com/etc/mainnet/tx/%s
|
||||||
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE: https://blockscout.com/eth/mainnet/tx/%s
|
||||||
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/etc/mainnet/address/%s
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE: https://blockscout.com/eth/mainnet/address/%s
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL: 600000
|
||||||
|
UI_STYLES: "classic"
|
||||||
|
UI_PUBLIC_URL: "https://wetc.app"
|
||||||
|
|
||||||
|
## Monitor
|
||||||
|
MONITOR_BRIDGE_NAME: "wetc"
|
||||||
|
MONITOR_PORT: 3003
|
||||||
|
MONITOR_CACHE_EVENTS: "true"
|
||||||
|
MONITOR_HOME_START_BLOCK: 7703292
|
||||||
|
MONITOR_FOREIGN_START_BLOCK: 7412459
|
||||||
|
MONITOR_VALIDATOR_HOME_TX_LIMIT: 300000
|
||||||
|
MONITOR_VALIDATOR_FOREIGN_TX_LIMIT: 300000
|
||||||
|
MONITOR_TX_NUMBER_THRESHOLD: 100
|
@ -7,6 +7,11 @@ sokol-kovan:
|
|||||||
ansible_user: ubuntu
|
ansible_user: ubuntu
|
||||||
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
ORACLE_VALIDATOR_ADDRESS_PRIVATE_KEY: "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
||||||
#syslog_server_port: "udp://127.0.0.1:514"
|
#syslog_server_port: "udp://127.0.0.1:514"
|
||||||
|
ui:
|
||||||
|
hosts:
|
||||||
|
127.0.0.1:
|
||||||
|
ansible_user: ubuntu
|
||||||
|
#syslog_server_port: "udp://127.0.0.1:514"
|
||||||
monitor:
|
monitor:
|
||||||
hosts:
|
hosts:
|
||||||
127.0.0.1:
|
127.0.0.1:
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
/var/log/docker/*/docker.log {
|
/var/log/docker/*/docker.log {
|
||||||
rotate 5
|
rotate 5
|
||||||
size 100M
|
size 1G
|
||||||
compress
|
compress
|
||||||
missingok
|
missingok
|
||||||
delaycompress
|
delaycompress
|
||||||
@ -8,7 +8,7 @@
|
|||||||
}
|
}
|
||||||
/var/log/docker/*.log {
|
/var/log/docker/*.log {
|
||||||
rotate 5
|
rotate 5
|
||||||
size 100M
|
size 1G
|
||||||
compress
|
compress
|
||||||
missingok
|
missingok
|
||||||
delaycompress
|
delaycompress
|
||||||
|
@ -3,11 +3,11 @@
|
|||||||
with_items:
|
with_items:
|
||||||
- docker-compose
|
- docker-compose
|
||||||
- docker-compose-transfer
|
- docker-compose-transfer
|
||||||
- docker-compose-amb
|
- docker-compose-erc-native
|
||||||
loop_control:
|
loop_control:
|
||||||
loop_var: file
|
loop_var: file
|
||||||
|
|
||||||
- name: Set the oracle's containers local logs configuration file
|
- name: Set the local container logs configuration file
|
||||||
template:
|
template:
|
||||||
src: 31-oracle-docker.conf.j2
|
src: 31-oracle-docker.conf.j2
|
||||||
dest: /etc/rsyslog.d/31-oracle-docker.conf
|
dest: /etc/rsyslog.d/31-oracle-docker.conf
|
||||||
@ -15,22 +15,6 @@
|
|||||||
group: root
|
group: root
|
||||||
mode: 0644
|
mode: 0644
|
||||||
|
|
||||||
- name: Set the redis container local logs configuration file
|
|
||||||
template:
|
|
||||||
src: 32-redis-docker.conf.j2
|
|
||||||
dest: /etc/rsyslog.d/32-redis-docker.conf
|
|
||||||
owner: root
|
|
||||||
group: root
|
|
||||||
mode: 0644
|
|
||||||
|
|
||||||
- name: Set the rabbit MQ container local logs configuration file
|
|
||||||
template:
|
|
||||||
src: 33-rabbit-docker.conf.j2
|
|
||||||
dest: /etc/rsyslog.d/33-rabbit-docker.conf
|
|
||||||
owner: root
|
|
||||||
group: root
|
|
||||||
mode: 0644
|
|
||||||
|
|
||||||
- name: Set the log configuration file to send container logs to remote server
|
- name: Set the log configuration file to send container logs to remote server
|
||||||
template:
|
template:
|
||||||
src: 36-oracle-remote-logging.conf.j2
|
src: 36-oracle-remote-logging.conf.j2
|
||||||
|
@ -27,13 +27,24 @@
|
|||||||
set_fact:
|
set_fact:
|
||||||
ORACLE_VALIDATOR_ADDRESS: "{{ VADDRESS.stdout }}"
|
ORACLE_VALIDATOR_ADDRESS: "{{ VADDRESS.stdout }}"
|
||||||
|
|
||||||
- name: Extend docker compose file for erc to native
|
- name: Get foreign erc type
|
||||||
set_fact: composefileoverride="-f docker-compose-transfer.yml"
|
become_user: "{{ compose_service_user }}"
|
||||||
when: ORACLE_BRIDGE_MODE == "ERC_TO_NATIVE"
|
shell: docker-compose run --rm --entrypoint "node scripts/initialChecks.js" bridge_affirmation
|
||||||
|
args:
|
||||||
|
chdir: "{{ bridge_path }}/oracle"
|
||||||
|
register: ERCTYPE
|
||||||
|
|
||||||
- name: Extend docker compose file for amb
|
- name: Set FOREIGN_ERC_TYPE variable
|
||||||
set_fact: composefileoverride="-f docker-compose-amb.yml"
|
set_fact:
|
||||||
when: ORACLE_BRIDGE_MODE == "ARBITRARY_MESSAGE"
|
FOREIGN_ERC_TYPE: "{{ (ERCTYPE.stdout).foreignERC | default('') }}"
|
||||||
|
|
||||||
|
- name: Extend docker compose file
|
||||||
|
set_fact: composefileoverride="-f docker-compose-transfer.yml"
|
||||||
|
when: ORACLE_BRIDGE_MODE == "ERC_TO_ERC" and FOREIGN_ERC_TYPE != "ERC677"
|
||||||
|
|
||||||
|
- name: Extend docker compose file for erc to native
|
||||||
|
set_fact: composefileoverride="-f docker-compose-erc-native.yml"
|
||||||
|
when: ORACLE_BRIDGE_MODE == "ERC_TO_NATIVE"
|
||||||
|
|
||||||
- name: Install .key config
|
- name: Install .key config
|
||||||
template:
|
template:
|
||||||
|
@ -20,4 +20,4 @@
|
|||||||
with_items:
|
with_items:
|
||||||
- docker-compose.yml
|
- docker-compose.yml
|
||||||
- docker-compose-transfer.yml
|
- docker-compose-transfer.yml
|
||||||
- docker-compose-amb.yml
|
- docker-compose-erc-native.yml
|
||||||
|
@ -11,16 +11,9 @@ ORACLE_HOME_RPC_POLLING_INTERVAL={{ ORACLE_HOME_RPC_POLLING_INTERVAL }}
|
|||||||
|
|
||||||
## Foreign contract
|
## Foreign contract
|
||||||
COMMON_FOREIGN_RPC_URL={{ COMMON_FOREIGN_RPC_URL }}
|
COMMON_FOREIGN_RPC_URL={{ COMMON_FOREIGN_RPC_URL }}
|
||||||
{% if ORACLE_FOREIGN_ARCHIVE_RPC_URL | default('') != '' %}
|
|
||||||
ORACLE_FOREIGN_ARCHIVE_RPC_URL={{ ORACLE_FOREIGN_ARCHIVE_RPC_URL }}
|
|
||||||
{% endif %}
|
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS={{ COMMON_FOREIGN_BRIDGE_ADDRESS }}
|
COMMON_FOREIGN_BRIDGE_ADDRESS={{ COMMON_FOREIGN_BRIDGE_ADDRESS }}
|
||||||
ORACLE_FOREIGN_RPC_POLLING_INTERVAL={{ ORACLE_FOREIGN_RPC_POLLING_INTERVAL }}
|
ORACLE_FOREIGN_RPC_POLLING_INTERVAL={{ ORACLE_FOREIGN_RPC_POLLING_INTERVAL }}
|
||||||
|
|
||||||
{% if ORACLE_TX_REDUNDANCY | default('') != '' %}
|
|
||||||
ORACLE_TX_REDUNDANCY={{ ORACLE_TX_REDUNDANCY }}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
## Gasprice
|
## Gasprice
|
||||||
{% if COMMON_HOME_GAS_PRICE_SUPPLIER_URL | default('') != '' %}
|
{% if COMMON_HOME_GAS_PRICE_SUPPLIER_URL | default('') != '' %}
|
||||||
COMMON_HOME_GAS_PRICE_SUPPLIER_URL={{ COMMON_HOME_GAS_PRICE_SUPPLIER_URL }}
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL={{ COMMON_HOME_GAS_PRICE_SUPPLIER_URL }}
|
||||||
@ -54,28 +47,8 @@ COMMON_FOREIGN_GAS_PRICE_FACTOR={{ COMMON_FOREIGN_GAS_PRICE_FACTOR }}
|
|||||||
ORACLE_ALLOW_HTTP_FOR_RPC={{ "yes" if ORACLE_ALLOW_HTTP_FOR_RPC else "no" }}
|
ORACLE_ALLOW_HTTP_FOR_RPC={{ "yes" if ORACLE_ALLOW_HTTP_FOR_RPC else "no" }}
|
||||||
ORACLE_QUEUE_URL={{ ORACLE_QUEUE_URL }}
|
ORACLE_QUEUE_URL={{ ORACLE_QUEUE_URL }}
|
||||||
ORACLE_REDIS_URL={{ ORACLE_REDIS_URL }}
|
ORACLE_REDIS_URL={{ ORACLE_REDIS_URL }}
|
||||||
{% if ORACLE_FOREIGN_TX_RESEND_INTERVAL | default('') != '' %}
|
{% if ORACLE_TX_REDUNDANCY | default('') != '' %}
|
||||||
ORACLE_FOREIGN_TX_RESEND_INTERVAL={{ ORACLE_FOREIGN_TX_RESEND_INTERVAL }}
|
ORACLE_TX_REDUNDANCY={{ ORACLE_TX_REDUNDANCY }}
|
||||||
{% endif %}
|
|
||||||
{% if ORACLE_HOME_TX_RESEND_INTERVAL | default('') != '' %}
|
|
||||||
ORACLE_HOME_TX_RESEND_INTERVAL={{ ORACLE_HOME_TX_RESEND_INTERVAL }}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
## Emergency shutdown configuration
|
|
||||||
{% if ORACLE_SHUTDOWN_SERVICE_URL | default('') != '' %}
|
|
||||||
ORACLE_SHUTDOWN_SERVICE_URL={{ ORACLE_SHUTDOWN_SERVICE_URL }}
|
|
||||||
{% endif %}
|
|
||||||
{% if ORACLE_SHUTDOWN_SERVICE_POLLING_INTERVAL | default('') != '' %}
|
|
||||||
ORACLE_SHUTDOWN_SERVICE_POLLING_INTERVAL={{ ORACLE_SHUTDOWN_SERVICE_POLLING_INTERVAL }}
|
|
||||||
{% endif %}
|
|
||||||
{% if ORACLE_SIDE_RPC_URL | default('') != '' %}
|
|
||||||
ORACLE_SIDE_RPC_URL={{ ORACLE_SIDE_RPC_URL }}
|
|
||||||
{% endif %}
|
|
||||||
{% if ORACLE_SHUTDOWN_CONTRACT_ADDRESS | default('') != '' %}
|
|
||||||
ORACLE_SHUTDOWN_CONTRACT_ADDRESS={{ ORACLE_SHUTDOWN_CONTRACT_ADDRESS }}
|
|
||||||
{% endif %}
|
|
||||||
{% if ORACLE_SHUTDOWN_CONTRACT_METHOD | default('') != '' %}
|
|
||||||
ORACLE_SHUTDOWN_CONTRACT_METHOD={{ ORACLE_SHUTDOWN_CONTRACT_METHOD }}
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
|
||||||
{% if ORACLE_HOME_START_BLOCK | default('') != '' %}
|
{% if ORACLE_HOME_START_BLOCK | default('') != '' %}
|
||||||
|
@ -1,11 +0,0 @@
|
|||||||
$FileCreateMode 0644
|
|
||||||
template(name="DockerLogFileName_Redis" type="list") {
|
|
||||||
constant(value="/var/log/docker/")
|
|
||||||
property(name="syslogtag" securepath="replace" regex.type="ERE" regex.submatch="1" regex.expression="oracle_(.*redis.*)\\/[a-zA-Z0-9]+\\[")
|
|
||||||
constant(value="/docker.log")
|
|
||||||
}
|
|
||||||
|
|
||||||
if $programname contains 'oracle' and $programname contains 'redis' then \
|
|
||||||
?DockerLogFileName_Redis
|
|
||||||
|
|
||||||
$FileCreateMode 0600
|
|
@ -1,11 +0,0 @@
|
|||||||
$FileCreateMode 0644
|
|
||||||
template(name="DockerLogFileName_Rabbit" type="list") {
|
|
||||||
constant(value="/var/log/docker/")
|
|
||||||
property(name="syslogtag" securepath="replace" regex.type="ERE" regex.submatch="1" regex.expression="oracle_(.*rabbit.*)\\/[a-zA-Z0-9]+\\[")
|
|
||||||
constant(value="/docker.log")
|
|
||||||
}
|
|
||||||
|
|
||||||
if $programname contains 'oracle' and $programname contains 'rabbit' then \
|
|
||||||
?DockerLogFileName_Rabbit
|
|
||||||
|
|
||||||
$FileCreateMode 0600
|
|
3
deployment/roles/ui/meta/main.yml
Normal file
3
deployment/roles/ui/meta/main.yml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
---
|
||||||
|
dependencies:
|
||||||
|
- role: common
|
6
deployment/roles/ui/tasks/jumpbox.yml
Normal file
6
deployment/roles/ui/tasks/jumpbox.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
- name: Build the containers
|
||||||
|
shell: docker-compose build
|
||||||
|
args:
|
||||||
|
chdir: "{{ bridge_path }}/ui"
|
||||||
|
when: skip_build is undefined
|
41
deployment/roles/ui/tasks/logging.yml
Normal file
41
deployment/roles/ui/tasks/logging.yml
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
---
|
||||||
|
- name: Slurp docker compose file
|
||||||
|
slurp:
|
||||||
|
src: "{{ bridge_path }}/ui/docker-compose.yml"
|
||||||
|
register: docker_compose_slurp
|
||||||
|
|
||||||
|
- name: Parse docker compose file
|
||||||
|
set_fact:
|
||||||
|
docker_compose_parsed: "{{ docker_compose_slurp['content'] | b64decode | from_yaml }}"
|
||||||
|
|
||||||
|
- name: Set logger to remote server
|
||||||
|
set_fact:
|
||||||
|
docker_compose_parsed: "{{ docker_compose_parsed |combine({'services': {item: {'logging': {'driver': 'syslog','options': {'tag': '{{.Name}}/{{.ID}}'}}}}}, recursive=True) }}"
|
||||||
|
with_items: "{{ docker_compose_parsed.services }}"
|
||||||
|
|
||||||
|
- name: Write new docker-compose file
|
||||||
|
copy:
|
||||||
|
content: "{{ docker_compose_parsed | to_yaml }}"
|
||||||
|
dest: "{{ bridge_path }}/ui/docker-compose.yml"
|
||||||
|
|
||||||
|
- name: Set the local container logs configuration file
|
||||||
|
template:
|
||||||
|
src: 32-ui-docker.conf.j2
|
||||||
|
dest: /etc/rsyslog.d/32-ui-docker.conf
|
||||||
|
owner: root
|
||||||
|
group: root
|
||||||
|
mode: 0644
|
||||||
|
|
||||||
|
- name: Set the log configuration file to send container logs to remote server
|
||||||
|
template:
|
||||||
|
src: 37-ui-remote-logging.conf.j2
|
||||||
|
dest: /etc/rsyslog.d/37-ui-remote-logging.conf
|
||||||
|
owner: root
|
||||||
|
group: root
|
||||||
|
mode: 0644
|
||||||
|
when: syslog_server_port is defined
|
||||||
|
|
||||||
|
- name: restart rsyslog
|
||||||
|
service:
|
||||||
|
name: rsyslog
|
||||||
|
state: restarted
|
5
deployment/roles/ui/tasks/main.yml
Normal file
5
deployment/roles/ui/tasks/main.yml
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
- include_tasks: pre_config.yml
|
||||||
|
- include_tasks: logging.yml
|
||||||
|
- include_tasks: jumpbox.yml
|
||||||
|
- include_tasks: servinstall.yml
|
7
deployment/roles/ui/tasks/pre_config.yml
Normal file
7
deployment/roles/ui/tasks/pre_config.yml
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
---
|
||||||
|
- name: Install .env config
|
||||||
|
template:
|
||||||
|
src: .env.j2
|
||||||
|
dest: "{{ bridge_path }}/ui/.env"
|
||||||
|
owner: "{{ compose_service_user }}"
|
||||||
|
mode: '0640'
|
20
deployment/roles/ui/tasks/servinstall.yml
Normal file
20
deployment/roles/ui/tasks/servinstall.yml
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
# This role creates a tokenbridge-ui service which is designed to manage docker-compose ui deployment.
|
||||||
|
# /etc/init.d/tokenbridge-ui start, status, stop, restart - does what the services usually do in such cases.
|
||||||
|
# /etc/init.d/tokenbridge-ui rebuild - builds a new ui deployment from scratch.
|
||||||
|
---
|
||||||
|
- name: "Set the service"
|
||||||
|
template:
|
||||||
|
src: tokenbridge-ui.j2
|
||||||
|
dest: "/etc/init.d/tokenbridge-ui"
|
||||||
|
owner: root
|
||||||
|
mode: 755
|
||||||
|
|
||||||
|
- name: "Enable the service"
|
||||||
|
service:
|
||||||
|
name: "tokenbridge-ui"
|
||||||
|
state: started
|
||||||
|
enabled: yes
|
||||||
|
use: service
|
||||||
|
|
||||||
|
- name: Start the service
|
||||||
|
shell: service tokenbridge-ui start
|
42
deployment/roles/ui/templates/.env.j2
Normal file
42
deployment/roles/ui/templates/.env.j2
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
COMMON_HOME_BRIDGE_ADDRESS={{ COMMON_HOME_BRIDGE_ADDRESS }}
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS={{ COMMON_FOREIGN_BRIDGE_ADDRESS }}
|
||||||
|
COMMON_FOREIGN_RPC_URL={{ COMMON_FOREIGN_RPC_URL }}
|
||||||
|
COMMON_HOME_RPC_URL={{ COMMON_HOME_RPC_URL }}
|
||||||
|
|
||||||
|
UI_NATIVE_TOKEN_DISPLAY_NAME={{ UI_NATIVE_TOKEN_DISPLAY_NAME }}
|
||||||
|
|
||||||
|
UI_HOME_NETWORK_DISPLAY_NAME={{ UI_HOME_NETWORK_DISPLAY_NAME }}
|
||||||
|
UI_FOREIGN_NETWORK_DISPLAY_NAME={{ UI_FOREIGN_NETWORK_DISPLAY_NAME }}
|
||||||
|
|
||||||
|
UI_HOME_WITHOUT_EVENTS={{ UI_HOME_WITHOUT_EVENTS }}
|
||||||
|
UI_FOREIGN_WITHOUT_EVENTS={{ UI_FOREIGN_WITHOUT_EVENTS }}
|
||||||
|
|
||||||
|
UI_HOME_EXPLORER_TX_TEMPLATE={{ UI_HOME_EXPLORER_TX_TEMPLATE }}
|
||||||
|
UI_FOREIGN_EXPLORER_TX_TEMPLATE={{ UI_FOREIGN_EXPLORER_TX_TEMPLATE }}
|
||||||
|
UI_HOME_EXPLORER_ADDRESS_TEMPLATE={{ UI_HOME_EXPLORER_ADDRESS_TEMPLATE }}
|
||||||
|
UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE={{ UI_FOREIGN_EXPLORER_ADDRESS_TEMPLATE }}
|
||||||
|
|
||||||
|
{% if COMMON_HOME_GAS_PRICE_SUPPLIER_URL | default('') != '' %}
|
||||||
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL={{ COMMON_HOME_GAS_PRICE_SUPPLIER_URL }}
|
||||||
|
COMMON_HOME_GAS_PRICE_SPEED_TYPE={{ COMMON_HOME_GAS_PRICE_SPEED_TYPE }}
|
||||||
|
{% endif %}
|
||||||
|
COMMON_HOME_GAS_PRICE_FALLBACK={{ COMMON_HOME_GAS_PRICE_FALLBACK }}
|
||||||
|
UI_HOME_GAS_PRICE_UPDATE_INTERVAL={{ UI_HOME_GAS_PRICE_UPDATE_INTERVAL }}
|
||||||
|
{% if COMMON_HOME_GAS_PRICE_FACTOR | default('') != '' %}
|
||||||
|
COMMON_HOME_GAS_PRICE_FACTOR={{ COMMON_HOME_GAS_PRICE_FACTOR }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL={{ COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL }}
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE={{ COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE }}
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FALLBACK={{ COMMON_FOREIGN_GAS_PRICE_FALLBACK }}
|
||||||
|
UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL={{ UI_FOREIGN_GAS_PRICE_UPDATE_INTERVAL }}
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FACTOR={{ COMMON_FOREIGN_GAS_PRICE_FACTOR }}
|
||||||
|
|
||||||
|
# Default
|
||||||
|
UI_TITLE={{ UI_TITLE }}
|
||||||
|
UI_OG_TITLE={{ UI_OG_TITLE }}
|
||||||
|
UI_DESCRIPTION={{ UI_DESCRIPTION }}
|
||||||
|
UI_PORT={{ UI_PORT }}
|
||||||
|
UI_PUBLIC_URL={{ UI_PUBLIC_URL }}
|
||||||
|
|
||||||
|
UI_STYLES={{ UI_STYLES }}
|
11
deployment/roles/ui/templates/32-ui-docker.conf.j2
Normal file
11
deployment/roles/ui/templates/32-ui-docker.conf.j2
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
$FileCreateMode 0644
|
||||||
|
template(name="DockerLogFileName_UI" type="list") {
|
||||||
|
constant(value="/var/log/docker/")
|
||||||
|
property(name="syslogtag" securepath="replace" regex.type="ERE" regex.submatch="1" regex.expression="ui_(.*)\\/[a-zA-Z0-9]+\\[")
|
||||||
|
constant(value="/docker.log")
|
||||||
|
}
|
||||||
|
|
||||||
|
if $programname startswith 'ui_' then \
|
||||||
|
?DockerLogFileName_UI
|
||||||
|
|
||||||
|
$FileCreateMode 0600
|
15
deployment/roles/ui/templates/37-ui-remote-logging.conf.j2
Normal file
15
deployment/roles/ui/templates/37-ui-remote-logging.conf.j2
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
if $programname startswith 'ui_' then {
|
||||||
|
action(
|
||||||
|
type="omfwd"
|
||||||
|
protocol="{{ syslog_server_port.split(":")[0] }}"
|
||||||
|
target="{{ (syslog_server_port.split(":")[1])[2:] }}"
|
||||||
|
port="{{ syslog_server_port.split(":")[2] }}"
|
||||||
|
template="RemoteForwardFormat"
|
||||||
|
queue.SpoolDirectory="/var/spool/rsyslog"
|
||||||
|
queue.FileName="remote"
|
||||||
|
queue.MaxDiskSpace="1g"
|
||||||
|
queue.SaveOnShutdown="on"
|
||||||
|
queue.Type="LinkedList"
|
||||||
|
ResendLastMSGOnReconnect="on"
|
||||||
|
)
|
||||||
|
}
|
76
deployment/roles/ui/templates/tokenbridge-ui.j2
Normal file
76
deployment/roles/ui/templates/tokenbridge-ui.j2
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
|
||||||
|
### BEGIN INIT INFO
|
||||||
|
# Provides: tokenbridge-ui
|
||||||
|
# Required-Start: $remote_fs $syslog
|
||||||
|
# Required-Stop: $remote_fs $syslog
|
||||||
|
# Default-Start: 2 3 4 5
|
||||||
|
# Default-Stop: 0 1 6
|
||||||
|
# Short-Description: Start daemon at boot time
|
||||||
|
# Description: Enable service provided by daemon.
|
||||||
|
### END INIT INFO
|
||||||
|
|
||||||
|
WORKDIR="{{ '/home/' + compose_service_user | default('poadocker') + '/' + bridge_path + '/ui' if bridge_path[:1] != "/" else bridge_path + '/ui' }}"
|
||||||
|
|
||||||
|
start(){
|
||||||
|
echo "Starting TokenBridge UI.."
|
||||||
|
cd $WORKDIR
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose down -v
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose rm -fv
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose up --detach
|
||||||
|
}
|
||||||
|
|
||||||
|
stop(){
|
||||||
|
echo "Stopping TokenBridge UI.."
|
||||||
|
cd $WORKDIR
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose down -v
|
||||||
|
sleep 2
|
||||||
|
}
|
||||||
|
|
||||||
|
status(){
|
||||||
|
echo "TokenBridge UI status:"
|
||||||
|
cd $WORKDIR
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose ps
|
||||||
|
}
|
||||||
|
|
||||||
|
rebuild(){
|
||||||
|
echo "Rebuild TokenBridge UI.."
|
||||||
|
cd $WORKDIR
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose down -v
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose rm -fv
|
||||||
|
sudo -u "{{ compose_service_user }}" /usr/local/bin/docker-compose up --detach --force-recreate --no-deps --build
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
case "$1" in
|
||||||
|
|
||||||
|
start)
|
||||||
|
start
|
||||||
|
;;
|
||||||
|
|
||||||
|
stop)
|
||||||
|
stop
|
||||||
|
;;
|
||||||
|
|
||||||
|
status)
|
||||||
|
status
|
||||||
|
;;
|
||||||
|
|
||||||
|
restart)
|
||||||
|
echo "Restarting TokenBridge UI.."
|
||||||
|
stop
|
||||||
|
start
|
||||||
|
;;
|
||||||
|
|
||||||
|
rebuild)
|
||||||
|
rebuild
|
||||||
|
;;
|
||||||
|
|
||||||
|
*)
|
||||||
|
echo $"Usage: $0 {start|stop|restart|rebuild|status}"
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit 0
|
@ -4,6 +4,11 @@
|
|||||||
become: true
|
become: true
|
||||||
roles:
|
roles:
|
||||||
- { role: oracle }
|
- { role: oracle }
|
||||||
|
- name: Install UI
|
||||||
|
hosts: ui
|
||||||
|
become: true
|
||||||
|
roles:
|
||||||
|
- { role: ui }
|
||||||
- name: Install Monitor
|
- name: Install Monitor
|
||||||
hosts: monitor
|
hosts: monitor
|
||||||
become: true
|
become: true
|
||||||
|
7
e2e-commons/Dockerfile.ui
Normal file
7
e2e-commons/Dockerfile.ui
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
ARG DOCKER_IMAGE_BASE
|
||||||
|
ARG UI_TAG
|
||||||
|
FROM ${DOCKER_IMAGE_BASE}/tokenbridge-e2e-ui:${UI_TAG}
|
||||||
|
|
||||||
|
ARG DOT_ENV_PATH
|
||||||
|
|
||||||
|
COPY ${DOT_ENV_PATH} ./.env
|
@ -24,10 +24,14 @@ Shut down and cleans up containers, networks, services, running scripts:
|
|||||||
| oracle | Launches Oracle containers |
|
| oracle | Launches Oracle containers |
|
||||||
| oracle-validator-2 | Launches Oracle containers for second validator |
|
| oracle-validator-2 | Launches Oracle containers for second validator |
|
||||||
| oracle-validator-3 | Launches Oracle containers for third validator |
|
| oracle-validator-3 | Launches Oracle containers for third validator |
|
||||||
|
| ui | Launches UI containers |
|
||||||
| blocks | Auto mines blocks |
|
| blocks | Auto mines blocks |
|
||||||
| monitor | Launches Monitor containers |
|
| monitor | Launches Monitor containers |
|
||||||
|
| native-to-erc | Creates infrastructure for ultimate e2e testing, for native-to-erc type of bridge |
|
||||||
| erc-to-native | Creates infrastructure for ultimate e2e testing, for erc-to-native type of bridge |
|
| erc-to-native | Creates infrastructure for ultimate e2e testing, for erc-to-native type of bridge |
|
||||||
|
| erc-to-erc | Creates infrastructure for ultimate e2e testing, for erc-to-erc type of bridge |
|
||||||
| amb | Creates infrastructure for ultimate e2e testing, for arbitrary message type of bridge |
|
| amb | Creates infrastructure for ultimate e2e testing, for arbitrary message type of bridge |
|
||||||
|
| ultimate-amb-stake-erc-to-erc | Creates infrastructure for ultimate e2e testing, for stake token bridge |
|
||||||
|
|
||||||
#### Ultimate e2e testing
|
#### Ultimate e2e testing
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@ Documentation regarding the Ultimate end-to-end tests.
|
|||||||
|
|
||||||
## Overview
|
## Overview
|
||||||
|
|
||||||
The ultimate e2e test scenario covers erc-to-native and amb types of bridges.
|
The ultimate e2e test scenario covers native-to-erc type of bridge.
|
||||||
It runs the e2e tests on components deployed using the deployment playbooks.
|
It runs the e2e tests on components deployed using the deployment playbooks.
|
||||||
|
|
||||||
|
|
||||||
@ -15,15 +15,13 @@ It runs the e2e tests on components deployed using the deployment playbooks.
|
|||||||
Run the Parity nodes, deploy the bridge contracts, deploy Oracle using the deployment playbook.
|
Run the Parity nodes, deploy the bridge contracts, deploy Oracle using the deployment playbook.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
./e2e-commons/up.sh deploy blocks
|
./up.sh deploy native-to-erc blocks
|
||||||
./deployment-e2e/molecule.sh ultimate-erc-to-native
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### 2. Run the E2E tests
|
### 2. Run the E2E tests
|
||||||
|
|
||||||
```bash
|
```
|
||||||
cd e2e-commons
|
cd ui-e2e; yarn mocha -g "NATIVE_TO_ERC" -b ./test.js
|
||||||
docker-compose run -e ULTIMATE=true e2e yarn workspace oracle-e2e run erc-to-native
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Diagram
|
## Diagram
|
||||||
|
@ -1,2 +1,2 @@
|
|||||||
0xc9e38bfdB9c635F0796ad83CC8705dc379F41c04
|
0xc9e38bfdB9c635F0796ad83CC8705dc379F41c04
|
||||||
0xF9698Eb93702dfdd0e2d802088d4c21822a8A977
|
0x612E8bd50A7b1F009F43f2b8679E9B8eD91eb5CE
|
||||||
|
@ -5,11 +5,11 @@ set -e # exit when any command fails
|
|||||||
docker-compose build e2e
|
docker-compose build e2e
|
||||||
while [ "$1" != "" ]; do
|
while [ "$1" != "" ]; do
|
||||||
if [ "$1" == "oracle" ]; then
|
if [ "$1" == "oracle" ]; then
|
||||||
docker-compose build oracle-amb
|
docker-compose build oracle
|
||||||
elif [ "$1" == "alm-e2e" ]; then
|
|
||||||
docker-compose build oracle-amb
|
|
||||||
elif [ "$1" == "monitor" ]; then
|
elif [ "$1" == "monitor" ]; then
|
||||||
docker-compose build monitor-amb
|
docker-compose build monitor
|
||||||
|
elif [ "$1" == "ui" ]; then
|
||||||
|
docker-compose build ui
|
||||||
elif [ "$1" == "alm" ]; then
|
elif [ "$1" == "alm" ]; then
|
||||||
docker-compose build alm
|
docker-compose build alm
|
||||||
fi
|
fi
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
COMMON_HOME_BRIDGE_ADDRESS=0x8397be90BCF57b0B71219f555Fe121b22e5a994C
|
COMMON_HOME_BRIDGE_ADDRESS=0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS=0x1feB40aD9420b186F019A717c37f5546165d411E
|
COMMON_FOREIGN_BRIDGE_ADDRESS=0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0
|
||||||
|
|
||||||
COMMON_HOME_RPC_URL=http://localhost:8541
|
COMMON_HOME_RPC_URL=http://localhost:8541
|
||||||
COMMON_FOREIGN_RPC_URL=http://localhost:8542
|
COMMON_FOREIGN_RPC_URL=http://localhost:8542
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
COMMON_HOME_RPC_URL=http://parity1:8545
|
COMMON_HOME_RPC_URL=http://parity1:8545
|
||||||
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
||||||
COMMON_HOME_BRIDGE_ADDRESS=0x8397be90BCF57b0B71219f555Fe121b22e5a994C
|
COMMON_HOME_BRIDGE_ADDRESS=0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS=0x1feB40aD9420b186F019A717c37f5546165d411E
|
COMMON_FOREIGN_BRIDGE_ADDRESS=0x0AEe1FCD12dDFab6265F7f8956e6E012A9Fe4Aa0
|
||||||
MONITOR_HOME_START_BLOCK=0
|
MONITOR_HOME_START_BLOCK=0
|
||||||
MONITOR_FOREIGN_START_BLOCK=0
|
MONITOR_FOREIGN_START_BLOCK=0
|
||||||
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
COMMON_HOME_RPC_URL=http://parity1:8545
|
COMMON_HOME_RPC_URL=http://parity1:8545
|
||||||
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
||||||
COMMON_HOME_BRIDGE_ADDRESS=0x5118AC62AE912Dd5B51EEfF7338c4fcb0248Ba8c
|
COMMON_HOME_BRIDGE_ADDRESS=0x488Af810997eD1730cB3a3918cD83b3216E6eAda
|
||||||
COMMON_FOREIGN_BRIDGE_ADDRESS=0x32198D570fffC7033641F8A9094FFDCaAEF42624
|
COMMON_FOREIGN_BRIDGE_ADDRESS=0x488Af810997eD1730cB3a3918cD83b3216E6eAda
|
||||||
MONITOR_HOME_START_BLOCK=0
|
MONITOR_HOME_START_BLOCK=0
|
||||||
MONITOR_FOREIGN_START_BLOCK=0
|
MONITOR_FOREIGN_START_BLOCK=0
|
||||||
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
||||||
|
20
e2e-commons/components-envs/monitor-erc20.env
Normal file
20
e2e-commons/components-envs/monitor-erc20.env
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
COMMON_HOME_RPC_URL=http://parity1:8545
|
||||||
|
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS=0x1feB40aD9420b186F019A717c37f5546165d411E
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS=0x4a58D6d8D416a5fBCAcf3dC52eb8bE8948E25127
|
||||||
|
MONITOR_HOME_START_BLOCK=0
|
||||||
|
MONITOR_FOREIGN_START_BLOCK=0
|
||||||
|
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
||||||
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL=https://gasprice.poa.network/
|
||||||
|
COMMON_HOME_GAS_PRICE_SPEED_TYPE=standard
|
||||||
|
COMMON_HOME_GAS_PRICE_FALLBACK=1000000000
|
||||||
|
COMMON_HOME_GAS_PRICE_FACTOR=1
|
||||||
|
MONITOR_VALIDATOR_FOREIGN_TX_LIMIT=300000
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL=https://gasprice.poa.network/
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE=standard
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FALLBACK=1000000000
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FACTOR=1
|
||||||
|
MONITOR_TX_NUMBER_THRESHOLD=100
|
||||||
|
MONITOR_PORT=3011
|
||||||
|
MONITOR_BRIDGE_NAME=bridge
|
||||||
|
MONITOR_CACHE_EVENTS=false
|
20
e2e-commons/components-envs/monitor.env
Normal file
20
e2e-commons/components-envs/monitor.env
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
COMMON_HOME_RPC_URL=http://parity1:8545
|
||||||
|
COMMON_FOREIGN_RPC_URL=http://parity2:8545
|
||||||
|
COMMON_HOME_BRIDGE_ADDRESS=0x32198D570fffC7033641F8A9094FFDCaAEF42624
|
||||||
|
COMMON_FOREIGN_BRIDGE_ADDRESS=0x2B6871b9B02F73fa24F4864322CdC78604207769
|
||||||
|
MONITOR_HOME_START_BLOCK=0
|
||||||
|
MONITOR_FOREIGN_START_BLOCK=0
|
||||||
|
MONITOR_VALIDATOR_HOME_TX_LIMIT=300000
|
||||||
|
COMMON_HOME_GAS_PRICE_SUPPLIER_URL=https://gasprice.poa.network/
|
||||||
|
COMMON_HOME_GAS_PRICE_SPEED_TYPE=standard
|
||||||
|
COMMON_HOME_GAS_PRICE_FALLBACK=1000000000
|
||||||
|
COMMON_HOME_GAS_PRICE_FACTOR=1
|
||||||
|
MONITOR_VALIDATOR_FOREIGN_TX_LIMIT=300000
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SUPPLIER_URL=https://gasprice.poa.network/
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_SPEED_TYPE=standard
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FALLBACK=1000000000
|
||||||
|
COMMON_FOREIGN_GAS_PRICE_FACTOR=1
|
||||||
|
MONITOR_TX_NUMBER_THRESHOLD=100
|
||||||
|
MONITOR_PORT=3010
|
||||||
|
MONITOR_BRIDGE_NAME=bridge
|
||||||
|
MONITOR_CACHE_EVENTS=false
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user