Skip to content

dwellir-public/artillery-engine-substrate

Repository files navigation

Artillery Substrate Engine

npm version Publish Node.js Package

Stress test substrate based nodes with Artillery.io

The development work is sponsored by Kusama Treasury.

Introduction

This project is the continuation of RPC-perf toolkit. The RPC-perf project served as a good proof of concept but lacked comprehensive workload modelling.

We chose Artillery.io because of its maturity, ease of use, modularity and the capability to use polkadot.js client to generate load.

This Substrate Engine for Artillery makes it easy to script virtual user flows in yaml and stress test any substrate based node without developing any code.

How to use

Prerequisites

  • node.js version > 16
  • npm > 6

Installation:

  • Install artillery and substrate plugin
npm install -g artillery
npm install -g artillery-engine-substrate

Quickstart

  • Create a test script or copy example/script-basic.yml to get started
  • Run the scenarios
artillery run --output report.json script.yml
  • Generate Report
artillery report report.json

You can learn more about Artillery Test Scripts in the documentation.

Configuration

config:
  target: "wss://westend.my-node.xyz"
  phases:
    - duration: 3
      arrivalRate: 1
      name: Engine test phase
  engines:
   substrate: {}

config.target: The substrate node endpoint (websocket) to connect to.

config.phases: Learn more about load phases in artillery documentation.

config.engines: This initializes the artillery Substrate engine.

Define your scenario

Artillery lets you define multiple scenarios. Each user is randomly assigned a scenario and runs till all the steps in a scenario has ran.

scenarios:
  - engine: substrate
    name: my_scenario
    flow:
      - connect: "{{ target }}"
      - loop:
        - call: 
            method: api.rpc.chain.getHeader()
            saveTo: header
        - log: "Current hash: {{ header.hash }}"
        - call:
            method: api.rpc.chain.getBlock({{ header.hash }})
            saveTo: block
        - log: "Current Block Number: {{ block.block.header.number }}"
        count: 2

To make a call to a rpc method exposed by the node, you can add multiple call steps. Refer substrate json-rpc documentation to see common methods exposed by substrate nodes.

The response of the call can be accessed by variable data. Remember, this can get overwritten by the next call action.

You have the option to instead declare your own variable to save the response to. This can be useful if you want to keep a variable and use it in a action later down in the flow. It is achieved by expanding call action to declare a method and saveTo field.

If you want to log certain values, you can use log action to do so.

scenarios:
  - engine: substrate
    name: my_scenario
    flow:
      - connect: "{{ target }}"
      - call: api.rpc.chain.getHeader()
      - log: "Current header hash: {{ data.hash }}"
      - call: 
          method: api.rpc.chain.getBlock({{ data.hash }})
          saveTo: blockResponse
      - log: "Current Block Number: {{ blockResponse.block.header.number }}"

The actions can also be looped as shown below.

scenarios:
  - engine: substrate
    name: my_scenario
    flow:
      - connect: "{{ target }}"
      - loop:
        - call: api.rpc.chain.getHeader()
        ...
        count: 100

Advanced usage

It is generally not required to develop any code to run Test Scripts with artillery.

However, the plugin allows using custom javascript functions for complex actions that may not be possible to implement via the yaml test script.

Let's look at an example, consider the following multi query operation:

const [{ nonce: accountNonce }, now] = await Promise.all([
   userContext.api.query.system.account(ALICE),
   userContext.api.query.timestamp.now()
]);

Set config.processor with the path to the file with the custom function.

config:
  target: "..."
  processor: "./functions.js"

Define a scenario with your function

scenarios:
  - engine: substrate
    name: complex_call
    flow:
      - function: "someComplexCall"
      - log: "Account Nonce {{ accountNonce }}"
      - log: "Last block timestamp {{ now }}"

And finally define your function

module.exports = { someComplexCall };
async function someComplexCall(userContext, events, done) {
  const ACCOUNT = '5G********tQY';
  const [{ nonce: accountNonce }, now] = await Promise.all([
   userContext.api.query.system.account(ACCOUNT),
   userContext.api.query.timestamp.now()
  ]);

  userContext.vars.accountNonce = accountNonce;
  userContext.vars.now = now;
  return done();
}

Run the scenario and generate the report:

artillery run --output report.json my-scenario.yml
artillery report report.json

Running with Docker

In some cases you may need to test from systems without development tools. You can run and get reports with using Docker with:

docker run -v $(pwd)/example:/scripts dwellir/artillery-substrate run --output /scripts/report.json /scripts/script.yml
docker run -v $(pwd)/example:/scripts dwellir/artillery-substrate report /scripts/report.json

Contributing to the Artillery Substrate Engine

Bare in mind that Artillery is a rich ecosystem with multiple plugins. Review the existing plugins in case someone is already working on the required functionality.

If you are looking to contribute to this engine, you can fork the repository and send a Pull Request.

Developing

Please note that:

  • Code structure and nomenclature follows Artillery.io conventions.
  • The engine logic is in two files, index.js and util.js.
  • When adding a new functionality, please also add a test for it.
  • To run the tests, use npm test.

Few ideas about the improvements that can be made to the engine

License and Copyright

Artillery Substrate Engine is Open Source licensed under Apache-2.0.

©2022 Dwellir AB, Authors and Contributors.

About

Artillery engine for perf testing substrate based nodes

Resources

Stars

Watchers

Forks

Packages

No packages published

Contributors 4

  •  
  •  
  •  
  •