e2e/

directory
v0.0.0-...-3786997 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Feb 10, 2023 License: Apache-2.0

README

End-to-end Tests

Structure

e2e Package

The e2e package defines an integration testing suite used for full end-to-end testing functionality. This package is decoupled from depending on the Merlin codebase. It initializes the chains for testing via Docker files. As a result, the test suite may provide the desired Merlin version to Docker containers during the initialization. This design allows for the opportunity of testing chain upgrades in the future by providing an older Merlin version to the container, performing the chain upgrade, and running the latest test suite. When testing a normal upgrade, the e2e test suite submits an upgrade proposal at an upgrade height, ensures the upgrade happens at the desired height, and then checks that operations that worked before still work as intended. If testing a fork, the test suite instead starts the chain a few blocks before the set fork height and ensures the chain continues after the fork triggers the upgrade. Note that a regular upgrade and a fork upgrade are mutually exclusive.

The file e2e_setup_test.go defines the testing suite and contains the core bootstrapping logic that creates a testing environment via Docker containers. A testing network is created dynamically with 2 test validators.

The file e2e_test.go contains the actual end-to-end integration tests that utilize the testing suite.

Currently, there is a single IBC test in e2e_test.go.

Additionally, there is an ability to disable certain components of the e2e suite. This can be done by setting the environment variables. See "Environment variables" section below for more details.

How It Works

Conceptually, we can split the e2e setup into 2 parts:

  1. Chain Initialization

    The chain can either be initialized off of the current branch, or off the prior mainnet release and then upgraded to the current branch.

    If current, we run chain initialization off of the current Git branch by calling chain.Init(...) method in the configurer/current.go.

    If with the upgrade, the same chain.Init(...) function is run inside a Docker container of the previous Merlin version, inside configurer/upgrade.go. This is needed to initialize chain configs and the genesis of the previous version that we are upgrading from.

    The decision of what configuration type to use is decided by the Configurer. This is an interface that has CurrentBranchConfigurer and UpgradeConfigurer implementations. There is also a BaseConfigurer which is shared by the concrete implementations. However, the user of the configurer package does not need to know about this detail.

    When the desired configurer is created, the caller may configure the chain in the desired way as follows:

    conf, _ := configurer.New(..., < isIBCEnabled bool >, < isUpgradeEnabled bool >)
    
    conf.ConfigureChains()
    

    The caller (e2e setup logic), does not need to be concerned about what type of configurations is hapenning in the background. The appropriate logic is selected depending on what the values of the arguments to configurer.New(...) are.

    CurrentBranchConfigurer configures chains from current Git branch. UpgradeConfigurer takes older merlin versions and configures chains from them.

    The configurer constructor is using a factory design pattern to decide on what kind of configurer to return. Factory design pattern is used to decouple the client from the initialization details of the configurer. More on this can be found here

    The rules for deciding on the configurer type are as follows:

    • If only isIBCEnabled, we want to have 2 chains initialized at the current branch version of Merlin codebase

    • If only isUpgradeEnabled, that's invalid (we can decouple upgrade testing from IBC in a future PR)

    • If both isIBCEnabled and isUpgradeEnabled, we want 2 chain with IBC initialized at the previous Merlin version

    • If none are true, we only need one chain at the current branch version of the Merlin code

  2. Setting up e2e components

    Currently, there exist the following components:

    • Base logic
      • This is the most basic type of setup where a single chain is created
      • It simply spins up the desired number of validators on a chain.
    • IBC testing
      • 2 chains are created connected by Hermes relayer
      • Upgrade Testing
      • 2 chains of the older Merlin version are created, and connected by Hermes relayer
    • Upgrade testing
      • CLI commands are run to create an upgrade proposal and approve it
      • Old version containers are stopped and the upgrade binary is added
      • Current branch Merlin version is spun up to continue with testing
    • State Sync Testing (WIP)
      • An additional full node is created after a chain has started.
      • This node is meant to state sync with the rest of the system.

    This is done in configurer/setup.go via function decorator design pattern where we chain the desired setup components during configurer creation. Example

initialization Package

The initialization package introduces the logic necessary for initializing a chain by creating a genesis file and all required configuration files such as the app.toml. In addition, it starts chain initialization. This package directly depends on the Merlin codebase.

In addition, there is a Dockerfile init-e2e.Dockerfile. When executed, its container produces all files necessary for starting up a new chain. These resulting files can be mounted on a volume and propagated to our production merlin container to start the merlind service.

containers Package

Introduces an abstraction necessary for creating and managing Docker containers. Currently, validator containers are created with a name of the corresponding validator struct that is initialized in the initialization/chain package.

Running From Current Branch

To build chain initialization image

Please refer to tests/e2e/initialization/README.md

To build the debug Merlin image
    make docker-build-e2e-debug
Environment variables

Some tests take a long time to run. Sometimes, we would like to disable them locally or in CI. The following are the environment variables to disable certain components of e2e testing.

  • MERLIN_E2E_SKIP_UPGRADE - when true, skips the upgrade tests. If MERLIN_E2E_SKIP_IBC is true, this must also be set to true because upgrade tests require IBC logic.

  • MERLIN_E2E_SKIP_IBC - when true, skips the IBC tests.

  • MERLIN_E2E_SKIP_STATE_SYNC - when true, skips the state sync tests.

  • MERLIN_E2E_SKIP_CLEANUP - when true, avoids cleaning up the e2e Docker containers.

  • MERLIN_E2E_FORK_HEIGHT - when the above "IS_FORK" env variable is set to true, this is the string of the height in which the network should fork. This should match the ForkHeight set in constants.go

  • MERLIN_E2E_UPGRADE_VERSION - string of what version will be upgraded to (for example, "v10")

  • MERLIN_E2E_DEBUG_LOG - when true, prints debug logs from executing CLI commands via Docker containers. Set to true in CI by default.

VS Code Debug Configuration

This debug configuration helps to run e2e tests locally and skip the desired tests.

{
    "version": "0.2.0",
    "configurations": [
        {
            "name": "E2E: (make test-e2e-short)",
            "type": "go",
            "request": "launch",
            "mode": "test",
            "program": "${workspaceFolder}/tests/e2e",
            "args": [
                "-test.timeout",
                "30m",
                "-test.run",
                "IntegrationTestSuite",
                "-test.v"
            ],
            "buildFlags": "-tags e2e",
            "env": {
                "MERLIN_E2E": "True",
                "MERLIN_E2E_SKIP_IBC": "true",
                "MERLIN_E2E_SKIP_UPGRADE": "true",
                "MERLIN_E2E_SKIP_CLEANUP": "true",
                "MERLIN_E2E_SKIP_STATE_SYNC": "true",
                "MERLIN_E2E_UPGRADE_VERSION": "v13",
                "MERLIN_E2E_DEBUG_LOG": "true",
            },
            "preLaunchTask": "e2e-setup"
        }
    ]
}
Common Problems

Please note that if the tests are stopped mid-way, the e2e framework might fail to start again due to duplicated containers. Make sure that containers are removed before running the tests again: docker containers rm -f $(docker containers ls -a -q).

Additionally, Docker networks do not get auto-removed. Therefore, you can manually remove them by running docker network prune.

Good to know things about e2e

This section contains common "gotchas" that is sometimes very good to know when working with e2e.

  1. Order of executions of tests in e2e_test.go is alphabetic.

    If one test fails and breaks the chain, all subsequent tests might also fail. It is important to know about an order, because sometimes you might see a failed test in e2e_test.go, however, in reality, the core reason of failure lays in something else.

    Example: you see TestAddToExistingLock failing. It might be because of something failing in test, of course. However, you need to keep in mind that something potentially broke during the e2e setup and because of that, lexicographically first test (currently TestAddToExistingLock) fails.

    A way to deal with this problem: disable upgrade logic by setting MERLIN_E2E_SKIP_UPGRADE to false and see how test performs.

  2. Node update failure before running e2e_test.go

    Nodes can completely crash when upgrading for many reasons. To detect that, we check the node's Status as soon as the upgrade is complete. An active node should return a response and no other errors. If not, all test would not be ran.

    To fix that, check the docker container and see if you node is working. Or remove all related container then rerun e2e test

Debugging

This section contains information about debugging merlin's e2e tests.

  1. Executing commands on docker containers

    Currently, e2e emits a lot of useful logs into standard output. However, sometimes that might not be enough. In this case, it is a good practice to run some commands inside docker containers and inspect outputs. In order to do that, run:

        docker exec < container name/id > < command >
    

    This will execute the specified command and print the response to standard output. Example: docker exec mer-test-a-node-prune-nothing-snapshot merlind status will print a node status of mer-test-a-node-prune-nothing-snapshot container.

  2. Viewing docker container logs

    Another useful thing to do when debugging some low level error is inspecing container's logs. This can be done by running:

        docker logs < container name/id >
    

    Example: docker logs mer-test-a-node-prune-nothing-snapshot will print logs emitted by container mer-test-a-node-prune-nothing-snapshot to your standard output.

  3. Viewing docker container logs when run merlin's e2e tests in CI

    When a failure occurs in CI, we have persists these log files of all container running as artifacts logs.tgz.

Directories

Path Synopsis

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL