Skip to content

Commit a8919cc

Browse files
authored
Merge pull request #97 from andkononykhin/issue-94-load-testing
[issue 94] initial load testing framework
2 parents 4af9fe4 + e848046 commit a8919cc

17 files changed

+865
-14
lines changed

.gitignore

+10-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,16 @@
11
settings.json
22
.idea
3-
build
43
localnet
54
rest-server.out
65
.DS_Store
76
.vscode
7+
8+
# vim
9+
*.swp
10+
11+
# go
12+
build
13+
vendor
14+
15+
# python
16+
__pycache__

Makefile

+16-3
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ ldflags = -X github.com/cosmos/cosmos-sdk/version.Name=DcLedger \
77
-X github.com/cosmos/cosmos-sdk/version.ServerName=dcld \
88
-X github.com/cosmos/cosmos-sdk/version.ClientName=dclcli \
99
-X github.com/cosmos/cosmos-sdk/version.Version=$(VERSION) \
10-
-X github.com/cosmos/cosmos-sdk/version.Commit=$(COMMIT)
10+
-X github.com/cosmos/cosmos-sdk/version.Commit=$(COMMIT)
1111

1212
BUILD_FLAGS := -ldflags '$(ldflags)'
1313
OUTPUT_DIR ?= build
@@ -17,7 +17,7 @@ LOCALNET_DIR ?= localnet
1717
LICENSE_TYPE = "apache"
1818
COPYRIGHT_YEAR = "2020"
1919
COPYRIGHT_HOLDER = "DSR Corporation"
20-
LICENSED_FILES = $(shell find . -type f -not -wholename '*/.*')
20+
LICENSED_FILES = $(shell find . -type f -not -path '*/.*' -not -name '*.md' -not -name 'requirements.txt')
2121

2222
all: install
2323

@@ -62,7 +62,20 @@ localnet_start:
6262
localnet_stop:
6363
docker-compose down
6464

65+
localnet_export: localnet_stop
66+
docker-compose run node0 dcld export --for-zero-height >genesis.export.node0.json
67+
docker-compose run node1 dcld export --for-zero-height >genesis.export.node1.json
68+
docker-compose run node2 dcld export --for-zero-height >genesis.export.node2.json
69+
docker-compose run node3 dcld export --for-zero-height >genesis.export.node3.json
70+
71+
72+
localnet_reset: localnet_stop
73+
docker-compose run node0 dcld unsafe-reset-all
74+
docker-compose run node1 dcld unsafe-reset-all
75+
docker-compose run node2 dcld unsafe-reset-all
76+
docker-compose run node3 dcld unsafe-reset-all
77+
6578
localnet_clean: localnet_stop
6679
rm -rf $(LOCALNET_DIR)
6780

68-
.PHONY: all build install test lint clean image localnet_init localnet_start localnet_stop localnet_clean license license-check
81+
.PHONY: all build install test lint clean image localnet_init localnet_start localnet_stop localnet_clean localnet_export localnet_reset license license-check

bench/README.md

+134
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,134 @@
1+
# DCLedger Load Testing
2+
3+
DCLedger testing is implemented in python3 and based on [Locust](https://locust.io/) framework.
4+
5+
## Requirements
6+
7+
* python >= 3.7
8+
9+
## Installation
10+
11+
Run (consider to use virtual environment):
12+
13+
```bash
14+
pip3 install -r bench/requirements.txt
15+
```
16+
17+
**Optional** If you need to monitor server side metrics please install [Prometheus](https://prometheus.io/docs/prometheus/latest/getting_started/).
18+
19+
## Preparation
20+
21+
Each write transactions is signed and thus requires:
22+
23+
* an account with write permissions (e.g. Vendor account)
24+
* proper values for txn `sequence` which enforces txns ordering for an account
25+
26+
By that reason load test uses prepared load data which can be generated as follows:
27+
28+
* Initialize the pool and test accounts (**Warning** applicable to local in-docker pool only for now):
29+
30+
```bash
31+
sudo make localnet_clean
32+
make localnet_init
33+
34+
# ./gentestaccounts.sh [<NUM-USERS>]
35+
./gentestaccounts.sh
36+
37+
make localnet_start
38+
# Note: once started ledger may require some time to complete the initialization.
39+
```
40+
* Generate test transactions:
41+
42+
```bash
43+
# DCLBENCH_WRITE_USERS_COUNT=<NUM-USERS> DCLBENCH_WRITE_USERS_Q_COUNT=<NUM-REQ-PER-USER> python bench/generate.py bench/test.spec.yaml bench/txns
44+
python bench/generate.py bench/test.spec.yaml bench/txns
45+
```
46+
47+
Here the following (**optional**) inputs are considered:
48+
49+
* `NUM-USERS`: number of client accounts with write access (created as Vendors). Default: 10
50+
* `NUM-REQ-PER-USER`: number of write txns to perform per a user. Default: 1000
51+
52+
## Run
53+
54+
### (Optional) Launch Prometheus
55+
56+
```bash
57+
prometheus --config.file=bench/prometheus.yml
58+
```
59+
60+
And open <http://localhost:9090/> to query and monitor the server side metrics.
61+
62+
### Headless
63+
64+
```bash
65+
locust --headless
66+
```
67+
68+
### Web UI
69+
70+
```bash
71+
locust
72+
```
73+
74+
Then you can open <http://localhost:8089/> and launch the tests from the browser.
75+
76+
### Configuration
77+
78+
Run options (DCLedger custom ones):
79+
80+
* `--dcl-users`: number of users
81+
* `--dcl-spawn-rate` Rate to spawn users at (users per second)
82+
* `--dcl-hosts <comma-sepated-list>`: list of DCL nodes to target. Each user randomly picks one
83+
E.g. for local ledger `http://localhost:26657,http://localhost:26659,http://localhost:26661,http://localhost:26663` will specify all the nodes.
84+
* `--dcl-txn-file` path to a file with generated txns
85+
86+
Statistic options:
87+
88+
[Locust](https://locust.io/) provides the following options to present the results:
89+
90+
* `--csv <prefix>`: generates a set of stat files (summary, failures, exceptions and stats history) with the provided `<prefix>`
91+
* `--csv-full-history`: populates the stats history with more entries (including each specific request type)
92+
* `--html <path>`: generates an html report
93+
* Web UI also includes `Download Data` tab where the reports can be found.
94+
95+
More details can be found in:
96+
97+
* [locust.conf](../locust.conf): default values
98+
* `locust --help` (being in the project root)
99+
* [locust configuration](https://docs.locust.io/en/stable/configuration.html)
100+
* [locust stats](https://docs.locust.io/en/stable/retrieving-stats.html)
101+
102+
### Re-run
103+
104+
**Warning** applicable to local in-docker pool only for now
105+
106+
Next time when you run the test using the same data you will likely get many (all) failures since DCLedger
107+
will complain about already written data or wrong sequence numbers.
108+
109+
For that case you may consider to reset the ledger as follows:
110+
111+
```bash
112+
make localnet_reset localnet_start
113+
```
114+
115+
## FAQ
116+
117+
### locust may complain about ulimit values for open files
118+
119+
Please check the details [here](https://github.com/locustio/locust/wiki/Installation#increasing-maximum-number-of-open-files-limit).
120+
121+
Additional sources (linux):
122+
123+
* `man limits.conf`
124+
* [RedHat: How to set ulimit values](https://access.redhat.com/solutions/61334)
125+
126+
## ToDo
127+
128+
* explore the options to export test accounts to commit static data (accounts and test txns)
129+
* read requests loads
130+
* combined (and configured) loads: write + read
131+
* stat gathering and interpretation
132+
* non-local setups automation and targeting (e.g. AWS)
133+
* harden data generation scripts
134+
* consider different types of tx: async, sync (currently used), commit

bench/generate.py

+199
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,199 @@
1+
#!/usr/bin/env python
2+
# -*- coding: utf-8 -*-
3+
#
4+
# Copyright 2020 DSR Corporation
5+
#
6+
# Licensed under the Apache License, Version 2.0 (the "License");
7+
# you may not use this file except in compliance with the License.
8+
# You may obtain a copy of the License at
9+
#
10+
# http://www.apache.org/licenses/LICENSE-2.0
11+
#
12+
# Unless required by applicable law or agreed to in writing, software
13+
# distributed under the License is distributed on an "AS IS" BASIS,
14+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
# See the License for the specific language governing permissions and
16+
# limitations under the License.
17+
18+
import sys
19+
import os
20+
import yaml
21+
import json
22+
import subprocess
23+
import tempfile
24+
from struct import pack
25+
from pathlib import Path
26+
27+
from render import render
28+
29+
30+
DCLCLI = "dclcli"
31+
32+
DEF_ACCOUNT_N_START = 4
33+
DEF_SEQUENCE_START = 0
34+
35+
ACCOUNT_N_START_F = "account-number-start"
36+
SEQUENCE_START_F = "sequence-number-start"
37+
QUERIES_F = "q"
38+
39+
TEST_PASSWORD = "test1234"
40+
41+
MODEL_INFO_PREFIX = 1
42+
VENDOR_PRODUCTS_PREFIX = 2
43+
44+
45+
def pack_model_info_key(vid, pid):
46+
return pack('<bhh', MODEL_INFO_PREFIX, vid, pid)
47+
48+
49+
def run_shell_cmd(cmd, **kwargs):
50+
_kwargs = dict(
51+
check=True,
52+
universal_newlines=True,
53+
stdout=subprocess.PIPE,
54+
stderr=subprocess.PIPE
55+
)
56+
57+
_kwargs.update(kwargs)
58+
59+
if not _kwargs.get("shell") and type(cmd) is str:
60+
cmd = cmd.split()
61+
62+
try:
63+
return subprocess.run(cmd, **_kwargs)
64+
except (subprocess.CalledProcessError, FileNotFoundError) as exc:
65+
raise RuntimeError(f"command '{cmd}' failed: {exc.stderr}") from exc
66+
67+
68+
def run_for_json_res(cmd, **kwargs):
69+
return json.loads(run_shell_cmd(cmd, **kwargs).stdout)
70+
71+
72+
def to_cli_args(**kwargs):
73+
res = []
74+
for k, v in kwargs.items():
75+
k = "--{}".format(k.replace("_", "-"))
76+
res.extend([k, str(v)])
77+
return res
78+
79+
80+
def yaml_dump(
81+
data,
82+
stream=None,
83+
width=1,
84+
indent=4,
85+
default_flow_style=False,
86+
canonical=False,
87+
**kwargs
88+
):
89+
return yaml.safe_dump(
90+
data,
91+
stream,
92+
default_flow_style=default_flow_style,
93+
canonical=canonical,
94+
width=width,
95+
indent=indent,
96+
**kwargs
97+
)
98+
99+
100+
def resolve_users():
101+
return {u["name"]: u for u in run_for_json_res([DCLCLI, "keys", "list"])}
102+
103+
104+
def txn_generate(u_address, txn_t_cls, txn_t_cmd, **params):
105+
cmd = [DCLCLI, "tx", txn_t_cls, txn_t_cmd]
106+
params["from"] = u_address
107+
cmd += to_cli_args(**params)
108+
cmd.append("--generate-only")
109+
return run_shell_cmd(cmd).stdout
110+
111+
112+
def txn_sign(u_address, account, sequence, f_path):
113+
cmd = [DCLCLI, "tx", "sign"]
114+
params = {"from": u_address}
115+
cmd += to_cli_args(
116+
account_number=account, sequence=sequence, gas="auto", **params
117+
)
118+
cmd.extend(["--offline", f_path])
119+
cmd = f"echo '{TEST_PASSWORD}' | {' '.join(cmd)}"
120+
return run_shell_cmd(cmd, shell=True).stdout
121+
122+
123+
def txn_encode(f_path):
124+
cmd = [DCLCLI, "tx", "encode", f_path]
125+
return run_shell_cmd(cmd).stdout
126+
127+
128+
ENV_PREFIX = "DCLBENCH_"
129+
130+
131+
def main():
132+
render_ctx = {
133+
k.split(ENV_PREFIX)[1].lower(): v
134+
for k, v in os.environ.items()
135+
if k.startswith(ENV_PREFIX)
136+
}
137+
138+
# TODO argument parsing using argparse
139+
spec_yaml = render(sys.argv[1], ctx=render_ctx)
140+
spec = yaml.safe_load(spec_yaml)
141+
142+
try:
143+
out_file = Path(sys.argv[2]).resolve()
144+
except IndexError:
145+
out_file = None
146+
147+
account_n_start = spec["defaults"].get(
148+
ACCOUNT_N_START_F, DEF_ACCOUNT_N_START)
149+
sequence_start = spec["defaults"].get(
150+
SEQUENCE_START_F, DEF_SEQUENCE_START)
151+
152+
users = resolve_users()
153+
154+
res = {}
155+
156+
account_n = account_n_start
157+
with tempfile.TemporaryDirectory() as tmpdirname:
158+
159+
for user, u_data in spec["users"].items():
160+
res[user] = []
161+
tmp_file = (Path(tmpdirname) / user).resolve()
162+
163+
u_address = users[user]["address"]
164+
165+
sequence = sequence_start
166+
for q in u_data[QUERIES_F]:
167+
q_id, q_data = next(iter(q.items()))
168+
q_cls, q_t, q_cmd = q_id.split("/")
169+
170+
if q_cls == "tx":
171+
tmp_file.write_text(
172+
txn_generate(u_address, q_t, q_cmd, **q_data)
173+
)
174+
# XXX by some reason pipe to encode doesn't work
175+
tmp_file.write_text(
176+
txn_sign(u_address, account_n, sequence, str(tmp_file))
177+
)
178+
179+
txn_encoded = txn_encode(str(tmp_file))
180+
res[user].append(txn_encoded.strip().strip('"'))
181+
sequence += 1
182+
else:
183+
raise ValueError("Unexpected query class: {q_cls}")
184+
185+
if out_file:
186+
print(f"User {user}: done")
187+
188+
account_n += 1
189+
190+
# TODO optimize for big data
191+
if out_file is None:
192+
print(yaml_dump(res))
193+
else:
194+
with out_file.open('w') as fd:
195+
yaml_dump(res, fd)
196+
197+
198+
if __name__ == "__main__":
199+
main()

0 commit comments

Comments
 (0)