forked from TrueCloudLab/frostfs-testcases
Compare commits
40 commits
master
...
notsomaste
Author | SHA1 | Date | |
---|---|---|---|
85ea5de04c | |||
6e4558d792 | |||
791c461da1 | |||
|
b869e8751b | ||
69199d301b | |||
5083b5adad | |||
a2d272eaee | |||
6367e5e8ed | |||
4b59c1f327 | |||
6bf672c203 | |||
359bb6be90 | |||
|
5d7fca14ee | ||
7a11560000 | |||
|
37cc85028e | ||
79cb3c3698 | |||
146aaa371f | |||
d357241a61 | |||
dc6e521f59 | |||
29a8c1e252 | |||
32b010df67 | |||
c33857fd80 | |||
523e7ebac0 | |||
d438748b2a | |||
de6434934d | |||
25a9443209 | |||
c670893805 | |||
efc5f271bf | |||
294e0fbae3 | |||
ac987c49aa | |||
bd78c9f4c0 | |||
8aab12d5a8 | |||
33e20b0562 | |||
cd98d362e3 | |||
b9b8c4e1f8 | |||
e99cc7d05d | |||
071914b45c | |||
12d8b1b10b | |||
089ac609c3 | |||
adc63b0221 | |||
7d3fc5d15c |
105 changed files with 9646 additions and 2962 deletions
|
@ -5,98 +5,98 @@ hosts:
|
||||||
- name: s01
|
- name: s01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: s01
|
container_name: s01
|
||||||
config_path: ../frostfs-dev-env/services/storage/.storage.env
|
config_path: ../neofs-dev-env/services/storage/.storage.env
|
||||||
wallet_path: ../frostfs-dev-env/services/storage/wallet01.json
|
wallet_path: ../neofs-dev-env/services/storage/wallet01.json
|
||||||
local_config_path: ./TemporaryDir/empty-password.yml
|
local_config_path: ./TemporaryDir/empty-password.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/storage/wallet01.json
|
local_wallet_path: ../neofs-dev-env/services/storage/wallet01.json
|
||||||
wallet_password: ""
|
wallet_password: ""
|
||||||
volume_name: storage_storage_s01
|
volume_name: storage_storage_s01
|
||||||
endpoint_data0: s01.frostfs.devenv:8080
|
endpoint_data0: s01.neofs.devenv:8080
|
||||||
control_endpoint: s01.frostfs.devenv:8081
|
control_endpoint: s01.neofs.devenv:8081
|
||||||
un_locode: "RU MOW"
|
un_locode: "RU MOW"
|
||||||
- name: s02
|
- name: s02
|
||||||
attributes:
|
attributes:
|
||||||
container_name: s02
|
container_name: s02
|
||||||
config_path: ../frostfs-dev-env/services/storage/.storage.env
|
config_path: ../neofs-dev-env/services/storage/.storage.env
|
||||||
wallet_path: ../frostfs-dev-env/services/storage/wallet02.json
|
wallet_path: ../neofs-dev-env/services/storage/wallet02.json
|
||||||
local_config_path: ./TemporaryDir/empty-password.yml
|
local_config_path: ./TemporaryDir/empty-password.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/storage/wallet02.json
|
local_wallet_path: ../neofs-dev-env/services/storage/wallet02.json
|
||||||
wallet_password: ""
|
wallet_password: ""
|
||||||
volume_name: storage_storage_s02
|
volume_name: storage_storage_s02
|
||||||
endpoint_data0: s02.frostfs.devenv:8080
|
endpoint_data0: s02.neofs.devenv:8080
|
||||||
control_endpoint: s02.frostfs.devenv:8081
|
control_endpoint: s02.neofs.devenv:8081
|
||||||
un_locode: "RU LED"
|
un_locode: "RU LED"
|
||||||
- name: s03
|
- name: s03
|
||||||
attributes:
|
attributes:
|
||||||
container_name: s03
|
container_name: s03
|
||||||
config_path: ../frostfs-dev-env/services/storage/.storage.env
|
config_path: ../neofs-dev-env/services/storage/.storage.env
|
||||||
wallet_path: ../frostfs-dev-env/services/storage/wallet03.json
|
wallet_path: ../neofs-dev-env/services/storage/wallet03.json
|
||||||
local_config_path: ./TemporaryDir/empty-password.yml
|
local_config_path: ./TemporaryDir/empty-password.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/storage/wallet03.json
|
local_wallet_path: ../neofs-dev-env/services/storage/wallet03.json
|
||||||
wallet_password: ""
|
wallet_password: ""
|
||||||
volume_name: storage_storage_s03
|
volume_name: storage_storage_s03
|
||||||
endpoint_data0: s03.frostfs.devenv:8080
|
endpoint_data0: s03.neofs.devenv:8080
|
||||||
control_endpoint: s03.frostfs.devenv:8081
|
control_endpoint: s03.neofs.devenv:8081
|
||||||
un_locode: "SE STO"
|
un_locode: "SE STO"
|
||||||
- name: s04
|
- name: s04
|
||||||
attributes:
|
attributes:
|
||||||
container_name: s04
|
container_name: s04
|
||||||
config_path: ../frostfs-dev-env/services/storage/.storage.env
|
config_path: ../neofs-dev-env/services/storage/.storage.env
|
||||||
wallet_path: ../frostfs-dev-env/services/storage/wallet04.json
|
wallet_path: ../neofs-dev-env/services/storage/wallet04.json
|
||||||
local_config_path: ./TemporaryDir/empty-password.yml
|
local_config_path: ./TemporaryDir/empty-password.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/storage/wallet04.json
|
local_wallet_path: ../neofs-dev-env/services/storage/wallet04.json
|
||||||
wallet_password: ""
|
wallet_password: ""
|
||||||
volume_name: storage_storage_s04
|
volume_name: storage_storage_s04
|
||||||
endpoint_data0: s04.frostfs.devenv:8080
|
endpoint_data0: s04.neofs.devenv:8080
|
||||||
control_endpoint: s04.frostfs.devenv:8081
|
control_endpoint: s04.neofs.devenv:8081
|
||||||
un_locode: "FI HEL"
|
un_locode: "FI HEL"
|
||||||
- name: s3-gate01
|
- name: s3-gate01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: s3_gate
|
container_name: s3_gate
|
||||||
config_path: ../frostfs-dev-env/services/s3_gate/.s3.env
|
config_path: ../neofs-dev-env/services/s3_gate/.s3.env
|
||||||
wallet_path: ../frostfs-dev-env/services/s3_gate/wallet.json
|
wallet_path: ../neofs-dev-env/services/s3_gate/wallet.json
|
||||||
local_config_path: ./TemporaryDir/password-s3.yml
|
local_config_path: ./TemporaryDir/password-s3.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/s3_gate/wallet.json
|
local_wallet_path: ../neofs-dev-env/services/s3_gate/wallet.json
|
||||||
wallet_password: "s3"
|
wallet_password: "s3"
|
||||||
endpoint_data0: https://s3.frostfs.devenv:8080
|
endpoint_data0: https://s3.neofs.devenv:8080
|
||||||
- name: http-gate01
|
- name: http-gate01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: http_gate
|
container_name: http_gate
|
||||||
config_path: ../frostfs-dev-env/services/http_gate/.http.env
|
config_path: ../neofs-dev-env/services/http_gate/.http.env
|
||||||
wallet_path: ../frostfs-dev-env/services/http_gate/wallet.json
|
wallet_path: ../neofs-dev-env/services/http_gate/wallet.json
|
||||||
local_config_path: ./TemporaryDir/password-other.yml
|
local_config_path: ./TemporaryDir/password-other.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/http_gate/wallet.json
|
local_wallet_path: ../neofs-dev-env/services/http_gate/wallet.json
|
||||||
wallet_password: "one"
|
wallet_password: "one"
|
||||||
endpoint_data0: http://http.frostfs.devenv
|
endpoint_data0: http://http.neofs.devenv
|
||||||
- name: ir01
|
- name: ir01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: ir01
|
container_name: ir01
|
||||||
config_path: ../frostfs-dev-env/services/ir/.ir.env
|
config_path: ../neofs-dev-env/services/ir/.ir.env
|
||||||
wallet_path: ../frostfs-dev-env/services/ir/az.json
|
wallet_path: ../neofs-dev-env/services/ir/az.json
|
||||||
local_config_path: ./TemporaryDir/password-other.yml
|
local_config_path: ./TemporaryDir/password-other.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/ir/az.json
|
local_wallet_path: ../neofs-dev-env/services/ir/az.json
|
||||||
wallet_password: "one"
|
wallet_password: "one"
|
||||||
- name: morph-chain01
|
- name: morph-chain01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: morph_chain
|
container_name: morph_chain
|
||||||
config_path: ../frostfs-dev-env/services/morph_chain/protocol.privnet.yml
|
config_path: ../neofs-dev-env/services/morph_chain/protocol.privnet.yml
|
||||||
wallet_path: ../frostfs-dev-env/services/morph_chain/node-wallet.json
|
wallet_path: ../neofs-dev-env/services/morph_chain/node-wallet.json
|
||||||
local_config_path: ./TemporaryDir/password-other.yml
|
local_config_path: ./TemporaryDir/password-other.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/morph_chain/node-wallet.json
|
local_wallet_path: ../neofs-dev-env/services/morph_chain/node-wallet.json
|
||||||
wallet_password: "one"
|
wallet_password: "one"
|
||||||
endpoint_internal0: http://morph-chain.frostfs.devenv:30333
|
endpoint_internal0: http://morph-chain.neofs.devenv:30333
|
||||||
- name: main-chain01
|
- name: main-chain01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: main_chain
|
container_name: main_chain
|
||||||
config_path: ../frostfs-dev-env/services/chain/protocol.privnet.yml
|
config_path: ../neofs-dev-env/services/chain/protocol.privnet.yml
|
||||||
wallet_path: ../frostfs-dev-env/services/chain/node-wallet.json
|
wallet_path: ../neofs-dev-env/services/chain/node-wallet.json
|
||||||
local_config_path: ./TemporaryDir/password-other.yml
|
local_config_path: ./TemporaryDir/password-other.yml
|
||||||
local_wallet_path: ../frostfs-dev-env/services/chain/node-wallet.json
|
local_wallet_path: ../neofs-dev-env/services/chain/node-wallet.json
|
||||||
wallet_password: "one"
|
wallet_password: "one"
|
||||||
endpoint_internal0: http://main-chain.frostfs.devenv:30333
|
endpoint_internal0: http://main-chain.neofs.devenv:30333
|
||||||
- name: coredns01
|
- name: coredns01
|
||||||
attributes:
|
attributes:
|
||||||
container_name: coredns
|
container_name: coredns
|
||||||
clis:
|
clis:
|
||||||
- name: frostfs-cli
|
- name: neofs-cli
|
||||||
exec_path: frostfs-cli
|
exec_path: neofs-cli
|
||||||
|
|
2
.github/CODEOWNERS
vendored
2
.github/CODEOWNERS
vendored
|
@ -1 +1 @@
|
||||||
* @vdomnich-yadro @dansingjulia @yadro-vavdeev @alexchetaev @abereziny
|
* @vdomnich-yadro @dansingjulia @yadro-vavdeev @abereziny
|
||||||
|
|
8
.gitignore
vendored
8
.gitignore
vendored
|
@ -1,11 +1,5 @@
|
||||||
# ignore IDE files
|
# ignore IDE files
|
||||||
.vscode
|
.vscode
|
||||||
.idea
|
|
||||||
|
|
||||||
.DS_Store
|
|
||||||
|
|
||||||
venv_macos
|
|
||||||
|
|
||||||
|
|
||||||
# ignore test results
|
# ignore test results
|
||||||
**/log.html
|
**/log.html
|
||||||
|
@ -26,4 +20,4 @@ TemporaryDir/*
|
||||||
artifacts/*
|
artifacts/*
|
||||||
docs/*
|
docs/*
|
||||||
venv.*/*
|
venv.*/*
|
||||||
wallet_config.yml
|
/*wallet_config.yml
|
||||||
|
|
|
@ -3,9 +3,9 @@ repos:
|
||||||
rev: 22.8.0
|
rev: 22.8.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: black
|
- id: black
|
||||||
language_version: python3.10
|
language_version: python3.9
|
||||||
- repo: https://github.com/pycqa/isort
|
- repo: https://github.com/pycqa/isort
|
||||||
rev: 5.12.0
|
rev: 5.10.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: isort
|
- id: isort
|
||||||
name: isort (python)
|
name: isort (python)
|
||||||
|
|
|
@ -3,8 +3,8 @@
|
||||||
First, thank you for contributing! We love and encourage pull requests from
|
First, thank you for contributing! We love and encourage pull requests from
|
||||||
everyone. Please follow the guidelines:
|
everyone. Please follow the guidelines:
|
||||||
|
|
||||||
- Check the open [issues](https://github.com/TrueCloudLab/frostfs-testcases/issues) and
|
- Check the open [issues](https://github.com/nspcc-dev/neofs-testcases/issues) and
|
||||||
[pull requests](https://github.com/TrueCloudLab/frostfs-testcases/pulls) for existing
|
[pull requests](https://github.com/nspcc-dev/neofs-testcases/pulls) for existing
|
||||||
discussions.
|
discussions.
|
||||||
|
|
||||||
- Open an issue first, to discuss a new feature or enhancement.
|
- Open an issue first, to discuss a new feature or enhancement.
|
||||||
|
@ -22,13 +22,13 @@ everyone. Please follow the guidelines:
|
||||||
|
|
||||||
## Development Workflow
|
## Development Workflow
|
||||||
|
|
||||||
Start by forking the `frostfs-testcases` repository, make changes in a branch and then
|
Start by forking the `neofs-testcases` repository, make changes in a branch and then
|
||||||
send a pull request. We encourage pull requests to discuss code changes. Here
|
send a pull request. We encourage pull requests to discuss code changes. Here
|
||||||
are the steps in details:
|
are the steps in details:
|
||||||
|
|
||||||
### Set up your GitHub Repository
|
### Set up your GitHub Repository
|
||||||
|
|
||||||
Fork [FrosfFS testcases upstream](https://github.com/TrueCloudLab/frostfs-testcases/fork) source
|
Fork [NeoFS testcases upstream](https://github.com/nspcc-dev/neofs-testcases/fork) source
|
||||||
repository to your own personal repository. Copy the URL of your fork and clone it:
|
repository to your own personal repository. Copy the URL of your fork and clone it:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
|
@ -36,20 +36,33 @@ $ git clone <url of your fork>
|
||||||
```
|
```
|
||||||
|
|
||||||
### Set up git remote as ``upstream``
|
### Set up git remote as ``upstream``
|
||||||
```sh
|
|
||||||
$ cd frostfs-testcases
|
```shell
|
||||||
$ git remote add upstream https://github.com/TrueCloudLab/frostfs-testcases
|
$ cd neofs-testcases
|
||||||
|
$ git remote add upstream https://github.com/nspcc-dev/neofs-testcases
|
||||||
$ git fetch upstream
|
$ git fetch upstream
|
||||||
```
|
```
|
||||||
|
|
||||||
### Set up development environment
|
### Set up development environment
|
||||||
|
|
||||||
To setup development environment for `frosfs-testcases`, please, take the following steps:
|
To setup development environment for `neofs-testcases`, please, take the following steps:
|
||||||
1. Prepare virtualenv
|
1. Prepare virtualenv
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
$ make venv
|
$ virtualenv --python=python3.9 venv
|
||||||
$ source frostfs-testcases-3.10/bin/activate
|
$ source venv/bin/activate
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Install all dependencies:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
$ pip install -r requirements.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
3. Setup pre-commit hooks to run code formatters on staged files before you run a `git commit` command:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
$ pre-commit install
|
||||||
```
|
```
|
||||||
|
|
||||||
Optionally you might want to integrate code formatters with your code editor to apply formatters to code files as you go:
|
Optionally you might want to integrate code formatters with your code editor to apply formatters to code files as you go:
|
||||||
|
|
45
Makefile
45
Makefile
|
@ -1,29 +1,28 @@
|
||||||
SHELL := /bin/bash
|
#!/usr/bin/make -f
|
||||||
PYTHON_VERSION := 3.10
|
|
||||||
VENV_NAME = frostfs-testcases-${PYTHON_VERSION}
|
|
||||||
VENV_DIR := venv.${VENV_NAME}
|
|
||||||
|
|
||||||
current_dir := $(shell pwd)
|
.DEFAULT_GOAL := help
|
||||||
|
|
||||||
venv: create requirements paths precommit
|
SHELL ?= bash
|
||||||
@echo Ready
|
|
||||||
|
|
||||||
precommit:
|
VENVS = $(shell ls -1d venv/*/ | sort -u | xargs basename -a)
|
||||||
@echo Isntalling pre-commit hooks
|
|
||||||
. ${VENV_DIR}/bin/activate && pre-commit install
|
|
||||||
|
|
||||||
paths:
|
.PHONY: all
|
||||||
@echo Append paths for project
|
all: venvs
|
||||||
@echo Virtual environment: ${VENV_DIR}
|
|
||||||
@sudo rm -rf ${VENV_DIR}/lib/python${PYTHON_VERSION}/site-packages/_paths.pth
|
|
||||||
@sudo touch ${VENV_DIR}/lib/python${PYTHON_VERSION}/site-packages/_paths.pth
|
|
||||||
@echo ${current_dir} | sudo tee ${VENV_DIR}/lib/python${PYTHON_VERSION}/site-packages/_paths.pth
|
|
||||||
|
|
||||||
create:
|
include venv_template.mk
|
||||||
@echo Create virtual environment for
|
|
||||||
virtualenv --python=python${PYTHON_VERSION} --prompt=${VENV_NAME} ${VENV_DIR}
|
|
||||||
|
|
||||||
requirements:
|
.PHONY: venvs
|
||||||
@echo Isntalling pip requirements
|
venvs:
|
||||||
. ${VENV_DIR}/bin/activate && pip install -e ../frostfs-testlib
|
$(foreach venv,$(VENVS),venv.$(venv))
|
||||||
. ${VENV_DIR}/bin/activate && pip install -Ur pytest_tests/requirements.txt
|
|
||||||
|
$(foreach venv,$(VENVS),$(eval $(call VENV_template,$(venv))))
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -rf venv.*
|
||||||
|
|
||||||
|
pytest-local:
|
||||||
|
@echo "⇒ Run Pytest"
|
||||||
|
python -m pytest pytest_tests/testsuites/
|
||||||
|
|
||||||
|
help:
|
||||||
|
@echo "⇒ run Run testcases ${R}"
|
||||||
|
|
60
README.md
60
README.md
|
@ -2,58 +2,70 @@
|
||||||
|
|
||||||
Tests written with PyTest Framework are located under `pytest_tests/testsuites` directory.
|
Tests written with PyTest Framework are located under `pytest_tests/testsuites` directory.
|
||||||
|
|
||||||
These tests rely on resources and utility modules that have been originally developed for Pytest Framework.
|
These tests rely on resources and utility modules that have been originally developed for Robot Framework:
|
||||||
|
|
||||||
|
`robot/resources/files` - static files that are used in tests' commands.
|
||||||
|
|
||||||
|
`robot/resources/lib/` - common Python libraries that provide utility functions used as building blocks in tests.
|
||||||
|
|
||||||
|
`robot/variables/` - constants and configuration variables for tests.
|
||||||
|
|
||||||
## Testcases execution
|
## Testcases execution
|
||||||
|
|
||||||
### Initial preparation
|
### Initial preparation
|
||||||
|
|
||||||
1. Install frostfs-cli
|
1. Install neofs-cli
|
||||||
- `git clone git@github.com:TrueCloudLab/frostfs-node.git`
|
- `git clone git@github.com:nspcc-dev/neofs-node.git`
|
||||||
- `cd frostfs-node`
|
- `cd neofs-node`
|
||||||
- `make`
|
- `make`
|
||||||
- `sudo cp bin/frostfs-cli /usr/local/bin/frostfs-cli`
|
- `sudo cp bin/neofs-cli /usr/local/bin/neofs-cli`
|
||||||
|
|
||||||
2. Install frostfs-s3-authmate
|
2. Install neofs-authmate
|
||||||
- `git clone git@github.com:TrueCloudLab/frostfs-s3-gw.git`
|
- `git clone git@github.com:nspcc-dev/neofs-s3-gw.git`
|
||||||
- `cd frostfs-s3-gw`
|
- `cd neofs-s3-gw`
|
||||||
- `make`
|
- `make`
|
||||||
- `sudo cp bin/frostfs-s3-authmate /usr/local/bin/frostfs-s3-authmate`
|
- `sudo cp bin/neofs-authmate /usr/local/bin/neofs-authmate`
|
||||||
|
|
||||||
3. Install neo-go
|
3. Install neo-go
|
||||||
- `git clone git@github.com:nspcc-dev/neo-go.git`
|
- `git clone git@github.com:nspcc-dev/neo-go.git`
|
||||||
- `cd neo-go`
|
- `cd neo-go`
|
||||||
- `git checkout v0.101.0` (or the current version in the frostfs-dev-env)
|
- `git checkout v0.92.0` (or the current version in the neofs-dev-env)
|
||||||
- `make`
|
- `make`
|
||||||
- `sudo cp bin/neo-go /usr/local/bin/neo-go`
|
- `sudo cp bin/neo-go /usr/local/bin/neo-go`
|
||||||
or download binary from releases: https://github.com/nspcc-dev/neo-go/releases
|
or download binary from releases: https://github.com/nspcc-dev/neo-go/releases
|
||||||
|
|
||||||
4. Clone frostfs-dev-env
|
4. Clone neofs-dev-env
|
||||||
`git clone git@github.com:TrueCloudLab/frostfs-dev-env.git`
|
`git clone git@github.com:nspcc-dev/neofs-dev-env.git`
|
||||||
|
|
||||||
Note that we expect frostfs-dev-env to be located under
|
Note that we expect neofs-dev-env to be located under
|
||||||
the `<testcases_root_dir>/../frostfs-dev-env` directory. If you put this repo in any other place,
|
the `<testcases_root_dir>/../neofs-dev-env` directory. If you put this repo in any other place,
|
||||||
manually set the full path to frostfs-dev-env in the environment variable `DEVENV_PATH` at this step.
|
manually set the full path to neofs-dev-env in the environment variable `DEVENV_PATH` at this step.
|
||||||
|
|
||||||
5. Make sure you have installed all the following prerequisites on your machine
|
5. Make sure you have installed all of the following prerequisites on your machine
|
||||||
|
|
||||||
```
|
```
|
||||||
make
|
make
|
||||||
python3.10
|
python3.9
|
||||||
python3.10-dev
|
python3.9-dev
|
||||||
libssl-dev
|
libssl-dev
|
||||||
```
|
```
|
||||||
As we use frostfs-dev-env, you'll also need to install
|
As we use neofs-dev-env, you'll also need to install
|
||||||
[prerequisites](https://github.com/TrueCloudLab/frostfs-dev-env#prerequisites) of this repository.
|
[prerequisites](https://github.com/nspcc-dev/neofs-dev-env#prerequisites) of this repository.
|
||||||
|
|
||||||
6. Prepare virtualenv
|
6. Prepare virtualenv
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
$ make venv
|
$ make venv.local-pytest
|
||||||
$ source venv.frostfs-testcases-3.10/bin/activate
|
$ . venv.local-pytest/bin/activate
|
||||||
```
|
```
|
||||||
|
|
||||||
7. Optionally you might want to integrate code formatters with your code editor to apply formatters to code files as you go:
|
7. Setup pre-commit hooks to run code formatters on staged files before you run a `git commit` command:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
$ pre-commit install
|
||||||
|
```
|
||||||
|
|
||||||
|
Optionally you might want to integrate code formatters with your code editor to apply formatters to code files as you go:
|
||||||
* isort is supported by [PyCharm](https://plugins.jetbrains.com/plugin/15434-isortconnect), [VS Code](https://cereblanco.medium.com/setup-black-and-isort-in-vscode-514804590bf9). Plugins exist for other IDEs/editors as well.
|
* isort is supported by [PyCharm](https://plugins.jetbrains.com/plugin/15434-isortconnect), [VS Code](https://cereblanco.medium.com/setup-black-and-isort-in-vscode-514804590bf9). Plugins exist for other IDEs/editors as well.
|
||||||
* black can be integrated with multiple editors, please, instructions are available [here](https://black.readthedocs.io/en/stable/integrations/editors.html).
|
* black can be integrated with multiple editors, please, instructions are available [here](https://black.readthedocs.io/en/stable/integrations/editors.html).
|
||||||
|
|
||||||
|
@ -110,7 +122,7 @@ $ docker run -p 5050:5050 -e CHECK_RESULTS_EVERY_SECONDS=30 -e KEEP_HISTORY=1 \
|
||||||
|
|
||||||
Then, you can check the allure report in your browser [by this link](http://localhost:5050/allure-docker-service/projects/default/reports/latest/index.html?redirect=false)
|
Then, you can check the allure report in your browser [by this link](http://localhost:5050/allure-docker-service/projects/default/reports/latest/index.html?redirect=false)
|
||||||
|
|
||||||
NOTE: feel free to select a different location for `allure-reports` directory, there is no requirement to have it inside `frostfs-testcases`. For example, you can place it under `/tmp` path.
|
NOTE: feel free to select a different location for `allure-reports` directory, there is no requirement to have it inside `neofs-testcases`. For example, you can place it under `/tmp` path.
|
||||||
|
|
||||||
# Contributing
|
# Contributing
|
||||||
|
|
||||||
|
|
31
build_assets/activate.patch
Normal file
31
build_assets/activate.patch
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
diff -urN bin.orig/activate bin/activate
|
||||||
|
--- bin.orig/activate 2018-12-27 14:55:13.916461020 +0900
|
||||||
|
+++ bin/activate 2018-12-27 20:38:35.223248728 +0900
|
||||||
|
@@ -30,6 +30,15 @@
|
||||||
|
unset _OLD_VIRTUAL_PS1
|
||||||
|
fi
|
||||||
|
|
||||||
|
+ # Unset exported dev-env variables
|
||||||
|
+ pushd ${DEVENV_PATH} > /dev/null
|
||||||
|
+ unset `make env | awk -F= '{print $1}'`
|
||||||
|
+ popd > /dev/null
|
||||||
|
+
|
||||||
|
+ # Unset external env variables
|
||||||
|
+ declare -f env_deactivate > /dev/null && env_deactivate
|
||||||
|
+ declare -f venv_deactivate > /dev/null && venv_deactivate
|
||||||
|
+
|
||||||
|
unset VIRTUAL_ENV
|
||||||
|
if [ ! "${1-}" = "nondestructive" ] ; then
|
||||||
|
# Self destruct!
|
||||||
|
@@ -47,6 +56,11 @@
|
||||||
|
PATH="$VIRTUAL_ENV/bin:$PATH"
|
||||||
|
export PATH
|
||||||
|
|
||||||
|
+# Set external variables
|
||||||
|
+if [ -f ${VIRTUAL_ENV}/bin/environment.sh ] ; then
|
||||||
|
+ . ${VIRTUAL_ENV}/bin/environment.sh
|
||||||
|
+fi
|
||||||
|
+
|
||||||
|
# unset PYTHONHOME if set
|
||||||
|
if ! [ -z "${PYTHONHOME+_}" ] ; then
|
||||||
|
_OLD_VIRTUAL_PYTHONHOME="$PYTHONHOME"
|
|
@ -1,8 +1,8 @@
|
||||||
[tool.isort]
|
[tool.isort]
|
||||||
profile = "black"
|
profile = "black"
|
||||||
src_paths = ["pytest_tests"]
|
src_paths = ["neofs-keywords", "pytest_tests", "robot"]
|
||||||
line_length = 100
|
line_length = 100
|
||||||
|
|
||||||
[tool.black]
|
[tool.black]
|
||||||
line-length = 100
|
line-length = 100
|
||||||
target-version = ["py310"]
|
target-version = ["py39"]
|
||||||
|
|
595
pytest_tests/helpers/aws_cli_client.py
Normal file
595
pytest_tests/helpers/aws_cli_client.py
Normal file
|
@ -0,0 +1,595 @@
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
from datetime import datetime
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cli_helpers import _cmd_run
|
||||||
|
from common import ASSETS_DIR
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
REGULAR_TIMEOUT = 90
|
||||||
|
LONG_TIMEOUT = 240
|
||||||
|
|
||||||
|
|
||||||
|
class AwsCliClient:
|
||||||
|
# Flags that we use for all S3 commands: disable SSL verification (as we use self-signed
|
||||||
|
# certificate in devenv) and disable automatic pagination in CLI output
|
||||||
|
common_flags = "--no-verify-ssl --no-paginate"
|
||||||
|
s3gate_endpoint: str
|
||||||
|
|
||||||
|
def __init__(self, s3gate_endpoint) -> None:
|
||||||
|
self.s3gate_endpoint = s3gate_endpoint
|
||||||
|
|
||||||
|
def create_bucket(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
ObjectLockEnabledForBucket: Optional[bool] = None,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
GrantFullControl: Optional[str] = None,
|
||||||
|
GrantRead: Optional[str] = None,
|
||||||
|
GrantWrite: Optional[str] = None,
|
||||||
|
CreateBucketConfiguration: Optional[dict] = None,
|
||||||
|
):
|
||||||
|
if ObjectLockEnabledForBucket is None:
|
||||||
|
object_lock = ""
|
||||||
|
elif ObjectLockEnabledForBucket:
|
||||||
|
object_lock = " --object-lock-enabled-for-bucket"
|
||||||
|
else:
|
||||||
|
object_lock = " --no-object-lock-enabled-for-bucket"
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api create-bucket --bucket {Bucket} "
|
||||||
|
f"{object_lock} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
if GrantFullControl:
|
||||||
|
cmd += f" --grant-full-control {GrantFullControl}"
|
||||||
|
if GrantWrite:
|
||||||
|
cmd += f" --grant-write {GrantWrite}"
|
||||||
|
if GrantRead:
|
||||||
|
cmd += f" --grant-read {GrantRead}"
|
||||||
|
if CreateBucketConfiguration:
|
||||||
|
cmd += f" --create-bucket-configuration LocationConstraint={CreateBucketConfiguration['LocationConstraint']}"
|
||||||
|
_cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
|
||||||
|
def list_buckets(self) -> dict:
|
||||||
|
cmd = f"aws {self.common_flags} s3api list-buckets --endpoint {self.s3gate_endpoint}"
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_acl(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-acl --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_versioning(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-versioning --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_location(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-location --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_bucket_versioning(self, Bucket: str, VersioningConfiguration: dict) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-bucket-versioning --bucket {Bucket} "
|
||||||
|
f'--versioning-configuration Status={VersioningConfiguration.get("Status")} '
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def list_objects(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api list-objects --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def list_objects_v2(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api list-objects-v2 --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def list_object_versions(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api list-object-versions --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def copy_object(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
CopySource: str,
|
||||||
|
Key: str,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
MetadataDirective: Optional[str] = None,
|
||||||
|
Metadata: Optional[dict] = None,
|
||||||
|
TaggingDirective: Optional[str] = None,
|
||||||
|
Tagging: Optional[str] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api copy-object --copy-source {CopySource} "
|
||||||
|
f"--bucket {Bucket} --key {Key} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
if MetadataDirective:
|
||||||
|
cmd += f" --metadata-directive {MetadataDirective}"
|
||||||
|
if Metadata:
|
||||||
|
cmd += " --metadata "
|
||||||
|
for key, value in Metadata.items():
|
||||||
|
cmd += f" {key}={value}"
|
||||||
|
if TaggingDirective:
|
||||||
|
cmd += f" --tagging-directive {TaggingDirective}"
|
||||||
|
if Tagging:
|
||||||
|
cmd += f" --tagging {Tagging}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def head_bucket(self, Bucket: str) -> dict:
|
||||||
|
cmd = f"aws {self.common_flags} s3api head-bucket --bucket {Bucket} --endpoint {self.s3gate_endpoint}"
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object(
|
||||||
|
self,
|
||||||
|
Body: str,
|
||||||
|
Bucket: str,
|
||||||
|
Key: str,
|
||||||
|
Metadata: Optional[dict] = None,
|
||||||
|
Tagging: Optional[str] = None,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
ObjectLockMode: Optional[str] = None,
|
||||||
|
ObjectLockRetainUntilDate: Optional[datetime] = None,
|
||||||
|
ObjectLockLegalHoldStatus: Optional[str] = None,
|
||||||
|
GrantFullControl: Optional[str] = None,
|
||||||
|
GrantRead: Optional[str] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object --bucket {Bucket} --key {Key} "
|
||||||
|
f"--body {Body} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if Metadata:
|
||||||
|
cmd += f" --metadata"
|
||||||
|
for key, value in Metadata.items():
|
||||||
|
cmd += f" {key}={value}"
|
||||||
|
if Tagging:
|
||||||
|
cmd += f" --tagging '{Tagging}'"
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
if ObjectLockMode:
|
||||||
|
cmd += f" --object-lock-mode {ObjectLockMode}"
|
||||||
|
if ObjectLockRetainUntilDate:
|
||||||
|
cmd += f' --object-lock-retain-until-date "{ObjectLockRetainUntilDate}"'
|
||||||
|
if ObjectLockLegalHoldStatus:
|
||||||
|
cmd += f" --object-lock-legal-hold-status {ObjectLockLegalHoldStatus}"
|
||||||
|
if GrantFullControl:
|
||||||
|
cmd += f" --grant-full-control '{GrantFullControl}'"
|
||||||
|
if GrantRead:
|
||||||
|
cmd += f" --grant-read {GrantRead}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def head_object(self, Bucket: str, Key: str, VersionId: str = None) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api head-object --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_object(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
Key: str,
|
||||||
|
file_path: str,
|
||||||
|
VersionId: Optional[str] = None,
|
||||||
|
Range: Optional[str] = None,
|
||||||
|
) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-object --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} {file_path} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if Range:
|
||||||
|
cmd += f" --range {Range}"
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_object_acl(self, Bucket: str, Key: str, VersionId: Optional[str] = None) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-object-acl --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_acl(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
Key: str,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
GrantWrite: Optional[str] = None,
|
||||||
|
GrantRead: Optional[str] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-acl --bucket {Bucket} --key {Key} "
|
||||||
|
f" --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
if GrantWrite:
|
||||||
|
cmd += f" --grant-write {GrantWrite}"
|
||||||
|
if GrantRead:
|
||||||
|
cmd += f" --grant-read {GrantRead}"
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_bucket_acl(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
GrantWrite: Optional[str] = None,
|
||||||
|
GrantRead: Optional[str] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-bucket-acl --bucket {Bucket} "
|
||||||
|
f" --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
if GrantWrite:
|
||||||
|
cmd += f" --grant-write {GrantWrite}"
|
||||||
|
if GrantRead:
|
||||||
|
cmd += f" --grant-read {GrantRead}"
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_objects(self, Bucket: str, Delete: dict) -> dict:
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, "delete.json")
|
||||||
|
with open(file_path, "w") as out_file:
|
||||||
|
out_file.write(json.dumps(Delete))
|
||||||
|
logger.info(f"Input file for delete-objects: {json.dumps(Delete)}")
|
||||||
|
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api delete-objects --bucket {Bucket} "
|
||||||
|
f"--delete file://{file_path} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_object(self, Bucket: str, Key: str, VersionId: str = None) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api delete-object --bucket {Bucket} "
|
||||||
|
f"--key {Key} {version} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_object_attributes(
|
||||||
|
self,
|
||||||
|
bucket: str,
|
||||||
|
key: str,
|
||||||
|
*attributes: str,
|
||||||
|
version_id: str = None,
|
||||||
|
max_parts: int = None,
|
||||||
|
part_number: int = None,
|
||||||
|
) -> dict:
|
||||||
|
attrs = ",".join(attributes)
|
||||||
|
version = f" --version-id {version_id}" if version_id else ""
|
||||||
|
parts = f"--max-parts {max_parts}" if max_parts else ""
|
||||||
|
part_number = f"--part-number-marker {part_number}" if part_number else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-object-attributes --bucket {bucket} "
|
||||||
|
f"--key {key} {version} {parts} {part_number} --object-attributes {attrs} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_bucket(self, Bucket: str) -> dict:
|
||||||
|
cmd = f"aws {self.common_flags} s3api delete-bucket --bucket {Bucket} --endpoint {self.s3gate_endpoint}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_tagging(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-tagging --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_policy(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-policy --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_bucket_policy(self, Bucket: str, Policy: dict) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-bucket-policy --bucket {Bucket} "
|
||||||
|
f"--policy {json.dumps(Policy)} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_bucket_cors(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-bucket-cors --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_bucket_cors(self, Bucket: str, CORSConfiguration: dict) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-bucket-cors --bucket {Bucket} "
|
||||||
|
f"--cors-configuration '{json.dumps(CORSConfiguration)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_bucket_cors(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api delete-bucket-cors --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_bucket_tagging(self, Bucket: str, Tagging: dict) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-bucket-tagging --bucket {Bucket} "
|
||||||
|
f"--tagging '{json.dumps(Tagging)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_bucket_tagging(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api delete-bucket-tagging --bucket {Bucket} "
|
||||||
|
f"--endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_retention(
|
||||||
|
self, Bucket: str, Key: str, Retention: dict, VersionId: Optional[str] = None
|
||||||
|
) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-retention --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --retention '{json.dumps(Retention, indent=4, sort_keys=True, default=str)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_legal_hold(
|
||||||
|
self, Bucket: str, Key: str, LegalHold: dict, VersionId: Optional[str] = None
|
||||||
|
) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-legal-hold --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --legal-hold '{json.dumps(LegalHold)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_retention(
|
||||||
|
self,
|
||||||
|
Bucket: str,
|
||||||
|
Key: str,
|
||||||
|
Retention: dict,
|
||||||
|
VersionId: Optional[str] = None,
|
||||||
|
BypassGovernanceRetention: Optional[bool] = None,
|
||||||
|
) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-retention --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --retention '{json.dumps(Retention, indent=4, sort_keys=True, default=str)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if not BypassGovernanceRetention is None:
|
||||||
|
cmd += " --bypass-governance-retention"
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_legal_hold(
|
||||||
|
self, Bucket: str, Key: str, LegalHold: dict, VersionId: Optional[str] = None
|
||||||
|
) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-legal-hold --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --legal-hold '{json.dumps(LegalHold)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_tagging(self, Bucket: str, Key: str, Tagging: dict) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-tagging --bucket {Bucket} --key {Key} "
|
||||||
|
f"--tagging '{json.dumps(Tagging)}' --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_object_tagging(self, Bucket: str, Key: str, VersionId: Optional[str] = None) -> dict:
|
||||||
|
version = f" --version-id {VersionId}" if VersionId else ""
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-object-tagging --bucket {Bucket} --key {Key} "
|
||||||
|
f"{version} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, REGULAR_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def delete_object_tagging(self, Bucket: str, Key: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api delete-object-tagging --bucket {Bucket} "
|
||||||
|
f"--key {Key} --endpoint {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
@allure.step("Sync directory S3")
|
||||||
|
def sync(
|
||||||
|
self,
|
||||||
|
bucket_name: str,
|
||||||
|
dir_path: str,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
Metadata: Optional[dict] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3 sync {dir_path} s3://{bucket_name} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
if Metadata:
|
||||||
|
cmd += f" --metadata"
|
||||||
|
for key, value in Metadata.items():
|
||||||
|
cmd += f" {key}={value}"
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
@allure.step("CP directory S3")
|
||||||
|
def cp(
|
||||||
|
self,
|
||||||
|
bucket_name: str,
|
||||||
|
dir_path: str,
|
||||||
|
ACL: Optional[str] = None,
|
||||||
|
Metadata: Optional[dict] = None,
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3 cp {dir_path} s3://{bucket_name} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint} --recursive"
|
||||||
|
)
|
||||||
|
if Metadata:
|
||||||
|
cmd += f" --metadata"
|
||||||
|
for key, value in Metadata.items():
|
||||||
|
cmd += f" {key}={value}"
|
||||||
|
if ACL:
|
||||||
|
cmd += f" --acl {ACL}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def create_multipart_upload(self, Bucket: str, Key: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api create-multipart-upload --bucket {Bucket} "
|
||||||
|
f"--key {Key} --endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def list_multipart_uploads(self, Bucket: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api list-multipart-uploads --bucket {Bucket} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def abort_multipart_upload(self, Bucket: str, Key: str, UploadId: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api abort-multipart-upload --bucket {Bucket} "
|
||||||
|
f"--key {Key} --upload-id {UploadId} --endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def upload_part(self, UploadId: str, Bucket: str, Key: str, PartNumber: int, Body: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api upload-part --bucket {Bucket} --key {Key} "
|
||||||
|
f"--upload-id {UploadId} --part-number {PartNumber} --body {Body} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def upload_part_copy(
|
||||||
|
self, UploadId: str, Bucket: str, Key: str, PartNumber: int, CopySource: str
|
||||||
|
) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api upload-part-copy --bucket {Bucket} --key {Key} "
|
||||||
|
f"--upload-id {UploadId} --part-number {PartNumber} --copy-source {CopySource} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def list_parts(self, UploadId: str, Bucket: str, Key: str) -> dict:
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api list-parts --bucket {Bucket} --key {Key} "
|
||||||
|
f"--upload-id {UploadId} --endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def complete_multipart_upload(
|
||||||
|
self, Bucket: str, Key: str, UploadId: str, MultipartUpload: dict
|
||||||
|
) -> dict:
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, "parts.json")
|
||||||
|
with open(file_path, "w") as out_file:
|
||||||
|
out_file.write(json.dumps(MultipartUpload))
|
||||||
|
logger.info(f"Input file for complete-multipart-upload: {json.dumps(MultipartUpload)}")
|
||||||
|
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api complete-multipart-upload --bucket {Bucket} "
|
||||||
|
f"--key {Key} --upload-id {UploadId} --multipart-upload file://{file_path} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def put_object_lock_configuration(self, Bucket, ObjectLockConfiguration):
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api put-object-lock-configuration --bucket {Bucket} "
|
||||||
|
f"--object-lock-configuration '{json.dumps(ObjectLockConfiguration)}' --endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
def get_object_lock_configuration(self, Bucket):
|
||||||
|
cmd = (
|
||||||
|
f"aws {self.common_flags} s3api get-object-lock-configuration --bucket {Bucket} "
|
||||||
|
f"--endpoint-url {self.s3gate_endpoint}"
|
||||||
|
)
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
return self._to_json(output)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _to_json(output: str) -> dict:
|
||||||
|
json_output = {}
|
||||||
|
try:
|
||||||
|
json_output = json.loads(output)
|
||||||
|
except Exception:
|
||||||
|
if "{" not in output and "}" not in output:
|
||||||
|
logger.warning(f"Could not parse json from output {output}")
|
||||||
|
return json_output
|
||||||
|
json_output = json.loads(output[output.index("{") :])
|
||||||
|
|
||||||
|
return json_output
|
73
pytest_tests/helpers/binary_version_helper.py
Normal file
73
pytest_tests/helpers/binary_version_helper.py
Normal file
|
@ -0,0 +1,73 @@
|
||||||
|
import logging
|
||||||
|
import re
|
||||||
|
|
||||||
|
from common import NEOFS_ADM_EXEC, NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from neofs_testlib.cli import NeofsAdm, NeofsCli
|
||||||
|
from neofs_testlib.hosting import Hosting
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
def get_local_binaries_versions(shell: Shell) -> dict[str, str]:
|
||||||
|
versions = {}
|
||||||
|
|
||||||
|
for binary in ["neo-go", "neofs-authmate"]:
|
||||||
|
out = shell.exec(f"{binary} --version").stdout
|
||||||
|
versions[binary] = _parse_version(out)
|
||||||
|
|
||||||
|
neofs_cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
versions["neofs-cli"] = _parse_version(neofs_cli.version.get().stdout)
|
||||||
|
|
||||||
|
try:
|
||||||
|
neofs_adm = NeofsAdm(shell, NEOFS_ADM_EXEC)
|
||||||
|
versions["neofs-adm"] = _parse_version(neofs_adm.version.get().stdout)
|
||||||
|
except RuntimeError:
|
||||||
|
logger.info(f"neofs-adm not installed")
|
||||||
|
|
||||||
|
out = shell.exec("aws --version").stdout
|
||||||
|
out_lines = out.split("\n")
|
||||||
|
versions["AWS"] = out_lines[0] if out_lines else "Unknown"
|
||||||
|
|
||||||
|
return versions
|
||||||
|
|
||||||
|
|
||||||
|
def get_remote_binaries_versions(hosting: Hosting) -> dict[str, str]:
|
||||||
|
versions_by_host = {}
|
||||||
|
for host in hosting.hosts:
|
||||||
|
binary_path_by_name = {} # Maps binary name to executable path
|
||||||
|
for service_config in host.config.services:
|
||||||
|
exec_path = service_config.attributes.get("exec_path")
|
||||||
|
if exec_path:
|
||||||
|
binary_path_by_name[service_config.name] = exec_path
|
||||||
|
for cli_config in host.config.clis:
|
||||||
|
binary_path_by_name[cli_config.name] = cli_config.exec_path
|
||||||
|
|
||||||
|
shell = host.get_shell()
|
||||||
|
versions_at_host = {}
|
||||||
|
for binary_name, binary_path in binary_path_by_name.items():
|
||||||
|
try:
|
||||||
|
result = shell.exec(f"{binary_path} --version")
|
||||||
|
versions_at_host[binary_name] = _parse_version(result.stdout)
|
||||||
|
except Exception as exc:
|
||||||
|
logger.error(f"Cannot get version for {binary_path} because of\n{exc}")
|
||||||
|
versions_at_host[binary_name] = "Unknown"
|
||||||
|
versions_by_host[host.config.address] = versions_at_host
|
||||||
|
|
||||||
|
# Consolidate versions across all hosts
|
||||||
|
versions = {}
|
||||||
|
for host, binary_versions in versions_by_host.items():
|
||||||
|
for name, version in binary_versions.items():
|
||||||
|
captured_version = versions.get(name)
|
||||||
|
if captured_version:
|
||||||
|
assert (
|
||||||
|
captured_version == version
|
||||||
|
), f"Binary {name} has inconsistent version on host {host}"
|
||||||
|
else:
|
||||||
|
versions[name] = version
|
||||||
|
return versions
|
||||||
|
|
||||||
|
|
||||||
|
def _parse_version(version_output: str) -> str:
|
||||||
|
version = re.search(r"version[:\s]*v?(.+)", version_output, re.IGNORECASE)
|
||||||
|
return version.group(1).strip() if version else "Unknown"
|
369
pytest_tests/helpers/cluster.py
Normal file
369
pytest_tests/helpers/cluster.py
Normal file
|
@ -0,0 +1,369 @@
|
||||||
|
import random
|
||||||
|
import re
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
import data_formatters
|
||||||
|
import yaml
|
||||||
|
from neofs_testlib.blockchain import RPCClient
|
||||||
|
from neofs_testlib.hosting import Host, Hosting
|
||||||
|
from neofs_testlib.hosting.config import ServiceConfig
|
||||||
|
from test_control import wait_for_success
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class NodeBase:
|
||||||
|
"""
|
||||||
|
Represents a node of some underlying service
|
||||||
|
"""
|
||||||
|
|
||||||
|
id: str
|
||||||
|
name: str
|
||||||
|
host: Host
|
||||||
|
|
||||||
|
def __init__(self, id, name, host) -> None:
|
||||||
|
self.id = id
|
||||||
|
self.name = name
|
||||||
|
self.host = host
|
||||||
|
self.construct()
|
||||||
|
|
||||||
|
def construct(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return self.name == other.name
|
||||||
|
|
||||||
|
def __hash__(self):
|
||||||
|
return id(self.name)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.label
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return self.label
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
@wait_for_success(60, 1)
|
||||||
|
def start_service(self):
|
||||||
|
self.host.start_service(self.name)
|
||||||
|
|
||||||
|
@wait_for_success(60, 1)
|
||||||
|
def stop_service(self):
|
||||||
|
self.host.stop_service(self.name)
|
||||||
|
|
||||||
|
def get_wallet_password(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.WALLET_PASSWORD)
|
||||||
|
|
||||||
|
def get_wallet_path(self) -> str:
|
||||||
|
return self._get_attribute(
|
||||||
|
_ConfigAttributes.LOCAL_WALLET_PATH,
|
||||||
|
_ConfigAttributes.WALLET_PATH,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_remote_wallet_path(self) -> str:
|
||||||
|
"""
|
||||||
|
Returns node wallet file path located on remote host
|
||||||
|
"""
|
||||||
|
return self._get_attribute(
|
||||||
|
_ConfigAttributes.WALLET_PATH,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_remote_config_path(self) -> str:
|
||||||
|
"""
|
||||||
|
Returns node config file path located on remote host
|
||||||
|
"""
|
||||||
|
return self._get_attribute(
|
||||||
|
_ConfigAttributes.CONFIG_PATH,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_wallet_config_path(self):
|
||||||
|
return self._get_attribute(
|
||||||
|
_ConfigAttributes.LOCAL_WALLET_CONFIG,
|
||||||
|
_ConfigAttributes.WALLET_CONFIG,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_wallet_public_key(self):
|
||||||
|
storage_wallet_path = self.get_wallet_path()
|
||||||
|
storage_wallet_pass = self.get_wallet_password()
|
||||||
|
return data_formatters.get_wallet_public_key(storage_wallet_path, storage_wallet_pass)
|
||||||
|
|
||||||
|
def _get_attribute(self, attribute_name: str, default_attribute_name: str = None) -> list[str]:
|
||||||
|
config = self.host.get_service_config(self.name)
|
||||||
|
if default_attribute_name:
|
||||||
|
return config.attributes.get(
|
||||||
|
attribute_name, config.attributes.get(default_attribute_name)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
return config.attributes.get(attribute_name)
|
||||||
|
|
||||||
|
def _get_service_config(self) -> ServiceConfig:
|
||||||
|
return self.host.get_service_config(self.name)
|
||||||
|
|
||||||
|
|
||||||
|
class InnerRingNode(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents inner ring node in a cluster
|
||||||
|
|
||||||
|
Inner ring node is not always the same as physical host (or physical node, if you will):
|
||||||
|
It can be service running in a container or on physical host
|
||||||
|
For testing perspective, it's not relevant how it is actually running,
|
||||||
|
since neofs network will still treat it as "node"
|
||||||
|
"""
|
||||||
|
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class S3Gate(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents S3 gateway in a cluster
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.ENDPOINT_DATA)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return f"{self.name}: {self.get_endpoint()}"
|
||||||
|
|
||||||
|
|
||||||
|
class HTTPGate(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents HTTP gateway in a cluster
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.ENDPOINT_DATA)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return f"{self.name}: {self.get_endpoint()}"
|
||||||
|
|
||||||
|
|
||||||
|
class MorphChain(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents side-chain aka morph-chain consensus node in a cluster
|
||||||
|
|
||||||
|
Consensus node is not always the same as physical host (or physical node, if you will):
|
||||||
|
It can be service running in a container or on physical host
|
||||||
|
For testing perspective, it's not relevant how it is actually running,
|
||||||
|
since neofs network will still treat it as "node"
|
||||||
|
"""
|
||||||
|
|
||||||
|
rpc_client: RPCClient = None
|
||||||
|
|
||||||
|
def construct(self):
|
||||||
|
self.rpc_client = RPCClient(self.get_endpoint())
|
||||||
|
|
||||||
|
def get_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.ENDPOINT_INTERNAL)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return f"{self.name}: {self.get_endpoint()}"
|
||||||
|
|
||||||
|
|
||||||
|
class MainChain(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents main-chain consensus node in a cluster
|
||||||
|
|
||||||
|
Consensus node is not always the same as physical host:
|
||||||
|
It can be service running in a container or on physical host (or physical node, if you will):
|
||||||
|
For testing perspective, it's not relevant how it is actually running,
|
||||||
|
since neofs network will still treat it as "node"
|
||||||
|
"""
|
||||||
|
|
||||||
|
rpc_client: RPCClient = None
|
||||||
|
|
||||||
|
def construct(self):
|
||||||
|
self.rpc_client = RPCClient(self.get_endpoint())
|
||||||
|
|
||||||
|
def get_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.ENDPOINT_INTERNAL)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return f"{self.name}: {self.get_endpoint()}"
|
||||||
|
|
||||||
|
|
||||||
|
class StorageNode(NodeBase):
|
||||||
|
"""
|
||||||
|
Class represents storage node in a storage cluster
|
||||||
|
|
||||||
|
Storage node is not always the same as physical host:
|
||||||
|
It can be service running in a container or on physical host (or physical node, if you will):
|
||||||
|
For testing perspective, it's not relevant how it is actually running,
|
||||||
|
since neofs network will still treat it as "node"
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get_rpc_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.ENDPOINT_DATA)
|
||||||
|
|
||||||
|
def get_control_endpoint(self) -> str:
|
||||||
|
return self._get_attribute(_ConfigAttributes.CONTROL_ENDPOINT)
|
||||||
|
|
||||||
|
def get_un_locode(self):
|
||||||
|
return self._get_attribute(_ConfigAttributes.UN_LOCODE)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def label(self) -> str:
|
||||||
|
return f"{self.name}: {self.get_rpc_endpoint()}"
|
||||||
|
|
||||||
|
|
||||||
|
class Cluster:
|
||||||
|
"""
|
||||||
|
This class represents a Cluster object for the whole storage based on provided hosting
|
||||||
|
"""
|
||||||
|
|
||||||
|
default_rpc_endpoint: str
|
||||||
|
default_s3_gate_endpoint: str
|
||||||
|
default_http_gate_endpoint: str
|
||||||
|
|
||||||
|
def __init__(self, hosting: Hosting) -> None:
|
||||||
|
self._hosting = hosting
|
||||||
|
self.default_rpc_endpoint = self.storage_nodes[0].get_rpc_endpoint()
|
||||||
|
self.default_s3_gate_endpoint = self.s3gates[0].get_endpoint()
|
||||||
|
self.default_http_gate_endpoint = self.http_gates[0].get_endpoint()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hosts(self) -> list[Host]:
|
||||||
|
"""
|
||||||
|
Returns list of Hosts
|
||||||
|
"""
|
||||||
|
return self._hosting.hosts
|
||||||
|
|
||||||
|
@property
|
||||||
|
def hosting(self) -> Hosting:
|
||||||
|
return self._hosting
|
||||||
|
|
||||||
|
def _create_wallet_config(self, service: ServiceConfig) -> None:
|
||||||
|
wallet_path = service.attributes[_ConfigAttributes.LOCAL_WALLET_CONFIG]
|
||||||
|
wallet_password = service.attributes[_ConfigAttributes.WALLET_PASSWORD]
|
||||||
|
with open(wallet_path, "w") as file:
|
||||||
|
yaml.dump({"password": wallet_password}, file)
|
||||||
|
|
||||||
|
def create_wallet_configs(self, hosting: Hosting) -> None:
|
||||||
|
configs = hosting.find_service_configs(".*")
|
||||||
|
for config in configs:
|
||||||
|
if _ConfigAttributes.LOCAL_WALLET_CONFIG in config.attributes:
|
||||||
|
self._create_wallet_config(config)
|
||||||
|
|
||||||
|
def is_local_devevn(self) -> bool:
|
||||||
|
if len(self.hosting.hosts) == 1:
|
||||||
|
host = self.hosting.hosts[0]
|
||||||
|
if host.config.address == "localhost" and host.config.plugin_name == "docker":
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def storage_nodes(self) -> list[StorageNode]:
|
||||||
|
"""
|
||||||
|
Returns list of Storage Nodes (not physical nodes)
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.STORAGE)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def s3gates(self) -> list[S3Gate]:
|
||||||
|
"""
|
||||||
|
Returns list of S3 gates
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.S3_GATE)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def http_gates(self) -> list[S3Gate]:
|
||||||
|
"""
|
||||||
|
Returns list of HTTP gates
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.HTTP_GATE)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def morph_chain_nodes(self) -> list[MorphChain]:
|
||||||
|
"""
|
||||||
|
Returns list of morph-chain consensus nodes (not physical nodes)
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.MORPH_CHAIN)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def main_chain_nodes(self) -> list[MainChain]:
|
||||||
|
"""
|
||||||
|
Returns list of main-chain consensus nodes (not physical nodes)
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.MAIN_CHAIN)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ir_nodes(self) -> list[InnerRingNode]:
|
||||||
|
"""
|
||||||
|
Returns list of inner-ring nodes (not physical nodes)
|
||||||
|
"""
|
||||||
|
return self._get_nodes(_ServicesNames.INNER_RING)
|
||||||
|
|
||||||
|
def _get_nodes(self, service_name) -> list[StorageNode]:
|
||||||
|
configs = self.hosting.find_service_configs(f"{service_name}\d*$")
|
||||||
|
|
||||||
|
class_mapping: dict[str, Any] = {
|
||||||
|
_ServicesNames.STORAGE: StorageNode,
|
||||||
|
_ServicesNames.INNER_RING: InnerRingNode,
|
||||||
|
_ServicesNames.MORPH_CHAIN: MorphChain,
|
||||||
|
_ServicesNames.S3_GATE: S3Gate,
|
||||||
|
_ServicesNames.HTTP_GATE: HTTPGate,
|
||||||
|
_ServicesNames.MAIN_CHAIN: MainChain,
|
||||||
|
}
|
||||||
|
|
||||||
|
cls = class_mapping.get(service_name)
|
||||||
|
return [
|
||||||
|
cls(
|
||||||
|
self._get_id(config.name),
|
||||||
|
config.name,
|
||||||
|
self.hosting.get_host_by_service(config.name),
|
||||||
|
)
|
||||||
|
for config in configs
|
||||||
|
]
|
||||||
|
|
||||||
|
def _get_id(self, node_name) -> str:
|
||||||
|
pattern = "\d*$"
|
||||||
|
|
||||||
|
matches = re.search(pattern, node_name)
|
||||||
|
if matches:
|
||||||
|
return int(matches.group())
|
||||||
|
|
||||||
|
def get_random_storage_rpc_endpoint(self) -> str:
|
||||||
|
return random.choice(self.get_storage_rpc_endpoints())
|
||||||
|
|
||||||
|
def get_random_storage_rpc_endpoint_mgmt(self) -> str:
|
||||||
|
return random.choice(self.get_storage_rpc_endpoints_mgmt())
|
||||||
|
|
||||||
|
def get_storage_rpc_endpoints(self) -> list[str]:
|
||||||
|
nodes = self.storage_nodes
|
||||||
|
return [node.get_rpc_endpoint() for node in nodes]
|
||||||
|
|
||||||
|
def get_storage_rpc_endpoints_mgmt(self) -> list[str]:
|
||||||
|
nodes = self.storage_nodes
|
||||||
|
return [node.get_rpc_endpoint_mgmt() for node in nodes]
|
||||||
|
|
||||||
|
def get_morph_endpoints(self) -> list[str]:
|
||||||
|
nodes = self.morph_chain_nodes
|
||||||
|
return [node.get_endpoint() for node in nodes]
|
||||||
|
|
||||||
|
|
||||||
|
class _ServicesNames:
|
||||||
|
STORAGE = "s"
|
||||||
|
S3_GATE = "s3-gate"
|
||||||
|
HTTP_GATE = "http-gate"
|
||||||
|
MORPH_CHAIN = "morph-chain"
|
||||||
|
INNER_RING = "ir"
|
||||||
|
MAIN_CHAIN = "main-chain"
|
||||||
|
|
||||||
|
|
||||||
|
class _ConfigAttributes:
|
||||||
|
WALLET_PASSWORD = "wallet_password"
|
||||||
|
WALLET_PATH = "wallet_path"
|
||||||
|
WALLET_CONFIG = "wallet_config"
|
||||||
|
CONFIG_PATH = "config_path"
|
||||||
|
LOCAL_WALLET_PATH = "local_wallet_path"
|
||||||
|
LOCAL_WALLET_CONFIG = "local_config_path"
|
||||||
|
ENDPOINT_DATA = "endpoint_data0"
|
||||||
|
ENDPOINT_INTERNAL = "endpoint_internal0"
|
||||||
|
CONTROL_ENDPOINT = "control_endpoint"
|
||||||
|
UN_LOCODE = "un_locode"
|
87
pytest_tests/helpers/container.py
Normal file
87
pytest_tests/helpers/container.py
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import Cluster
|
||||||
|
from file_helper import generate_file, get_file_hash
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_verbs import put_object, put_object_to_random_node
|
||||||
|
from storage_object import StorageObjectInfo
|
||||||
|
from wallet import WalletFile
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class StorageContainerInfo:
|
||||||
|
id: str
|
||||||
|
wallet_file: WalletFile
|
||||||
|
|
||||||
|
|
||||||
|
class StorageContainer:
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
storage_container_info: StorageContainerInfo,
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
) -> None:
|
||||||
|
self.shell = shell
|
||||||
|
self.storage_container_info = storage_container_info
|
||||||
|
self.cluster = cluster
|
||||||
|
|
||||||
|
def get_id(self) -> str:
|
||||||
|
return self.storage_container_info.id
|
||||||
|
|
||||||
|
def get_wallet_path(self) -> str:
|
||||||
|
return self.storage_container_info.wallet_file.path
|
||||||
|
|
||||||
|
def get_wallet_config_path(self) -> str:
|
||||||
|
return self.storage_container_info.wallet_file.config_path
|
||||||
|
|
||||||
|
@allure.step("Generate new object and put in container")
|
||||||
|
def generate_object(
|
||||||
|
self,
|
||||||
|
size: int,
|
||||||
|
expire_at: Optional[int] = None,
|
||||||
|
bearer_token: Optional[str] = None,
|
||||||
|
endpoint: Optional[str] = None,
|
||||||
|
) -> StorageObjectInfo:
|
||||||
|
with allure.step(f"Generate object with size {size}"):
|
||||||
|
file_path = generate_file(size)
|
||||||
|
file_hash = get_file_hash(file_path)
|
||||||
|
|
||||||
|
container_id = self.get_id()
|
||||||
|
wallet_path = self.get_wallet_path()
|
||||||
|
wallet_config = self.get_wallet_config_path()
|
||||||
|
with allure.step(f"Put object with size {size} to container {container_id}"):
|
||||||
|
if endpoint:
|
||||||
|
object_id = put_object(
|
||||||
|
wallet=wallet_path,
|
||||||
|
path=file_path,
|
||||||
|
cid=container_id,
|
||||||
|
expire_at=expire_at,
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=endpoint,
|
||||||
|
bearer=bearer_token,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
object_id = put_object_to_random_node(
|
||||||
|
wallet=wallet_path,
|
||||||
|
path=file_path,
|
||||||
|
cid=container_id,
|
||||||
|
expire_at=expire_at,
|
||||||
|
shell=self.shell,
|
||||||
|
cluster=self.cluster,
|
||||||
|
bearer=bearer_token,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
|
||||||
|
storage_object = StorageObjectInfo(
|
||||||
|
container_id,
|
||||||
|
object_id,
|
||||||
|
size=size,
|
||||||
|
wallet_file_path=wallet_path,
|
||||||
|
file_path=file_path,
|
||||||
|
file_hash=file_hash,
|
||||||
|
)
|
||||||
|
|
||||||
|
return storage_object
|
40
pytest_tests/helpers/env_properties.py
Normal file
40
pytest_tests/helpers/env_properties.py
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import logging
|
||||||
|
import re
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from pytest import Config
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Read environment.properties")
|
||||||
|
def read_env_properties(config: Config) -> dict:
|
||||||
|
environment_dir = config.getoption("--alluredir")
|
||||||
|
if not environment_dir:
|
||||||
|
return None
|
||||||
|
|
||||||
|
file_path = f"{environment_dir}/environment.properties"
|
||||||
|
with open(file_path, "r") as file:
|
||||||
|
raw_content = file.read()
|
||||||
|
|
||||||
|
env_properties = {}
|
||||||
|
for line in raw_content.split("\n"):
|
||||||
|
m = re.match("(.*?)=(.*)", line)
|
||||||
|
if not m:
|
||||||
|
logger.warning(f"Could not parse env property from {line}")
|
||||||
|
continue
|
||||||
|
key, value = m.group(1), m.group(2)
|
||||||
|
env_properties[key] = value
|
||||||
|
return env_properties
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Update data in environment.properties")
|
||||||
|
def save_env_properties(config: Config, env_data: dict) -> None:
|
||||||
|
environment_dir = config.getoption("--alluredir")
|
||||||
|
if not environment_dir:
|
||||||
|
return None
|
||||||
|
|
||||||
|
file_path = f"{environment_dir}/environment.properties"
|
||||||
|
with open(file_path, "a+") as env_file:
|
||||||
|
for env, env_value in env_data.items():
|
||||||
|
env_file.write(f"{env}={env_value}\n")
|
167
pytest_tests/helpers/file_helper.py
Normal file
167
pytest_tests/helpers/file_helper.py
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
import hashlib
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from common import ASSETS_DIR
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
def generate_file(size: int) -> str:
|
||||||
|
"""Generates a binary file with the specified size in bytes.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
size: Size in bytes, can be declared as 6e+6 for example.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The path to the generated file.
|
||||||
|
"""
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
with open(file_path, "wb") as file:
|
||||||
|
file.write(os.urandom(size))
|
||||||
|
logger.info(f"File with size {size} bytes has been generated: {file_path}")
|
||||||
|
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
def generate_file_with_content(
|
||||||
|
size: int,
|
||||||
|
file_path: Optional[str] = None,
|
||||||
|
content: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
"""Creates a new file with specified content.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
file_path: Path to the file that should be created. If not specified, then random file
|
||||||
|
path will be generated.
|
||||||
|
content: Content that should be stored in the file. If not specified, then random binary
|
||||||
|
content will be generated.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Path to the generated file.
|
||||||
|
"""
|
||||||
|
mode = "w+"
|
||||||
|
if content is None:
|
||||||
|
content = os.urandom(size)
|
||||||
|
mode = "wb"
|
||||||
|
|
||||||
|
if not file_path:
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
else:
|
||||||
|
if not os.path.exists(os.path.dirname(file_path)):
|
||||||
|
os.makedirs(os.path.dirname(file_path))
|
||||||
|
|
||||||
|
with open(file_path, mode) as file:
|
||||||
|
file.write(content)
|
||||||
|
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get File Hash")
|
||||||
|
def get_file_hash(file_path: str, len: Optional[int] = None, offset: Optional[int] = None) -> str:
|
||||||
|
"""Generates hash for the specified file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
file_path: Path to the file to generate hash for.
|
||||||
|
len: How many bytes to read.
|
||||||
|
offset: Position to start reading from.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Hash of the file as hex-encoded string.
|
||||||
|
"""
|
||||||
|
file_hash = hashlib.sha256()
|
||||||
|
with open(file_path, "rb") as out:
|
||||||
|
if len and not offset:
|
||||||
|
file_hash.update(out.read(len))
|
||||||
|
elif len and offset:
|
||||||
|
out.seek(offset, 0)
|
||||||
|
file_hash.update(out.read(len))
|
||||||
|
elif offset and not len:
|
||||||
|
out.seek(offset, 0)
|
||||||
|
file_hash.update(out.read())
|
||||||
|
else:
|
||||||
|
file_hash.update(out.read())
|
||||||
|
return file_hash.hexdigest()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Concatenation set of files to one file")
|
||||||
|
def concat_files(file_paths: list, resulting_file_path: Optional[str] = None) -> str:
|
||||||
|
"""Concatenates several files into a single file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
file_paths: Paths to the files to concatenate.
|
||||||
|
resulting_file_name: Path to the file where concatenated content should be stored.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Path to the resulting file.
|
||||||
|
"""
|
||||||
|
if not resulting_file_path:
|
||||||
|
resulting_file_path = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
with open(resulting_file_path, "wb") as f:
|
||||||
|
for file in file_paths:
|
||||||
|
with open(file, "rb") as part_file:
|
||||||
|
f.write(part_file.read())
|
||||||
|
return resulting_file_path
|
||||||
|
|
||||||
|
|
||||||
|
def split_file(file_path: str, parts: int) -> list[str]:
|
||||||
|
"""Splits specified file into several specified number of parts.
|
||||||
|
|
||||||
|
Each part is saved under name `{original_file}_part_{i}`.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
file_path: Path to the file that should be split.
|
||||||
|
parts: Number of parts the file should be split into.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Paths to the part files.
|
||||||
|
"""
|
||||||
|
with open(file_path, "rb") as file:
|
||||||
|
content = file.read()
|
||||||
|
|
||||||
|
content_size = len(content)
|
||||||
|
chunk_size = int((content_size + parts) / parts)
|
||||||
|
|
||||||
|
part_id = 1
|
||||||
|
part_file_paths = []
|
||||||
|
for content_offset in range(0, content_size + 1, chunk_size):
|
||||||
|
part_file_name = f"{file_path}_part_{part_id}"
|
||||||
|
part_file_paths.append(part_file_name)
|
||||||
|
with open(part_file_name, "wb") as out_file:
|
||||||
|
out_file.write(content[content_offset : content_offset + chunk_size])
|
||||||
|
part_id += 1
|
||||||
|
|
||||||
|
return part_file_paths
|
||||||
|
|
||||||
|
|
||||||
|
def get_file_content(
|
||||||
|
file_path: str, content_len: Optional[int] = None, mode: str = "r", offset: Optional[int] = None
|
||||||
|
) -> Any:
|
||||||
|
"""Returns content of specified file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
file_path: Path to the file.
|
||||||
|
content_len: Limit of content length. If None, then entire file content is returned;
|
||||||
|
otherwise only the first content_len bytes of the content are returned.
|
||||||
|
mode: Mode of opening the file.
|
||||||
|
offset: Position to start reading from.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Content of the specified file.
|
||||||
|
"""
|
||||||
|
with open(file_path, mode) as file:
|
||||||
|
if content_len and not offset:
|
||||||
|
content = file.read(content_len)
|
||||||
|
elif content_len and offset:
|
||||||
|
file.seek(offset, 0)
|
||||||
|
content = file.read(content_len)
|
||||||
|
elif offset and not content_len:
|
||||||
|
file.seek(offset, 0)
|
||||||
|
content = file.read()
|
||||||
|
else:
|
||||||
|
content = file.read()
|
||||||
|
|
||||||
|
return content
|
36
pytest_tests/helpers/grpc_responses.py
Normal file
36
pytest_tests/helpers/grpc_responses.py
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
import re
|
||||||
|
|
||||||
|
# Regex patterns of status codes of Container service (https://github.com/nspcc-dev/neofs-spec/blob/98b154848116223e486ce8b43eaa35fec08b4a99/20-api-v2/container.md)
|
||||||
|
CONTAINER_NOT_FOUND = "code = 3072.*message = container not found"
|
||||||
|
|
||||||
|
|
||||||
|
# Regex patterns of status codes of Object service (https://github.com/nspcc-dev/neofs-spec/blob/98b154848116223e486ce8b43eaa35fec08b4a99/20-api-v2/object.md)
|
||||||
|
MALFORMED_REQUEST = "code = 1024.*message = malformed request"
|
||||||
|
OBJECT_ACCESS_DENIED = "code = 2048.*message = access to object operation denied"
|
||||||
|
OBJECT_NOT_FOUND = "code = 2049.*message = object not found"
|
||||||
|
OBJECT_ALREADY_REMOVED = "code = 2052.*message = object already removed"
|
||||||
|
SESSION_NOT_FOUND = "code = 4096.*message = session token not found"
|
||||||
|
OUT_OF_RANGE = "code = 2053.*message = out of range"
|
||||||
|
# TODO: Due to https://github.com/nspcc-dev/neofs-node/issues/2092 we have to check only codes until fixed
|
||||||
|
# OBJECT_IS_LOCKED = "code = 2050.*message = object is locked"
|
||||||
|
# LOCK_NON_REGULAR_OBJECT = "code = 2051.*message = ..." will be available once 2092 is fixed
|
||||||
|
OBJECT_IS_LOCKED = "code = 2050"
|
||||||
|
LOCK_NON_REGULAR_OBJECT = "code = 2051"
|
||||||
|
|
||||||
|
LIFETIME_REQUIRED = "either expiration epoch of a lifetime is required"
|
||||||
|
LOCK_OBJECT_REMOVAL = "lock object removal"
|
||||||
|
LOCK_OBJECT_EXPIRATION = "lock object expiration: {expiration_epoch}; current: {current_epoch}"
|
||||||
|
INVALID_RANGE_ZERO_LENGTH = "invalid '{range}' range: zero length"
|
||||||
|
INVALID_RANGE_OVERFLOW = "invalid '{range}' range: uint64 overflow"
|
||||||
|
INVALID_OFFSET_SPECIFIER = "invalid '{range}' range offset specifier"
|
||||||
|
INVALID_LENGTH_SPECIFIER = "invalid '{range}' range length specifier"
|
||||||
|
|
||||||
|
|
||||||
|
def error_matches_status(error: Exception, status_pattern: str) -> bool:
|
||||||
|
"""
|
||||||
|
Determines whether exception matches specified status pattern.
|
||||||
|
|
||||||
|
We use re.search to be consistent with pytest.raises.
|
||||||
|
"""
|
||||||
|
match = re.search(status_pattern, str(error))
|
||||||
|
return match is not None
|
|
@ -1,4 +1,4 @@
|
||||||
from frostfs_testlib.shell import Shell
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
|
||||||
class IpTablesHelper:
|
class IpTablesHelper:
|
||||||
|
|
236
pytest_tests/helpers/k6.py
Normal file
236
pytest_tests/helpers/k6.py
Normal file
|
@ -0,0 +1,236 @@
|
||||||
|
import re
|
||||||
|
from contextlib import contextmanager
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from time import sleep
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from remote_process import RemoteProcess
|
||||||
|
|
||||||
|
EXIT_RESULT_CODE = 0
|
||||||
|
LOAD_RESULTS_PATTERNS = {
|
||||||
|
"grpc": {
|
||||||
|
"write_ops": r"neofs_obj_put_total\W*\d*\W*(?P<write_ops>\d*\.\d*)",
|
||||||
|
"read_ops": r"neofs_obj_get_total\W*\d*\W*(?P<read_ops>\d*\.\d*)",
|
||||||
|
},
|
||||||
|
"s3": {
|
||||||
|
"write_ops": r"aws_obj_put_total\W*\d*\W*(?P<write_ops>\d*\.\d*)",
|
||||||
|
"read_ops": r"aws_obj_get_total\W*\d*\W*(?P<write_ops>\d*\.\d*)",
|
||||||
|
},
|
||||||
|
"http": {"total_ops": r"http_reqs\W*\d*\W*(?P<total_ops>\d*\.\d*)"},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class LoadParams:
|
||||||
|
load_type: str
|
||||||
|
endpoint: str
|
||||||
|
writers: Optional[int] = None
|
||||||
|
readers: Optional[int] = None
|
||||||
|
deleters: Optional[int] = None
|
||||||
|
clients: Optional[int] = None
|
||||||
|
containers_count: Optional[int] = None
|
||||||
|
out_file: Optional[str] = None
|
||||||
|
load_time: Optional[int] = None
|
||||||
|
obj_count: Optional[int] = None
|
||||||
|
obj_size: Optional[int] = None
|
||||||
|
registry_file: Optional[str] = None
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class LoadResults:
|
||||||
|
data_sent: float = 0.0
|
||||||
|
data_received: float = 0.0
|
||||||
|
read_ops: float = 0.0
|
||||||
|
write_ops: float = 0.0
|
||||||
|
total_ops: float = 0.0
|
||||||
|
|
||||||
|
|
||||||
|
class K6:
|
||||||
|
def __init__(self, load_params: LoadParams, shell: Shell):
|
||||||
|
|
||||||
|
self.load_params = load_params
|
||||||
|
self.shell = shell
|
||||||
|
|
||||||
|
self._k6_dir = None
|
||||||
|
self._k6_result = None
|
||||||
|
|
||||||
|
self._k6_process = None
|
||||||
|
self._k6_stop_attempts = 5
|
||||||
|
self._k6_stop_timeout = 15
|
||||||
|
|
||||||
|
@property
|
||||||
|
def process_dir(self) -> str:
|
||||||
|
return self._k6_process.process_dir
|
||||||
|
|
||||||
|
@property
|
||||||
|
def k6_dir(self) -> str:
|
||||||
|
if not self._k6_dir:
|
||||||
|
self._k6_dir = self.shell.exec(
|
||||||
|
r"sudo find . -name 'k6' -exec dirname {} \; -quit"
|
||||||
|
).stdout.strip("\n")
|
||||||
|
return self._k6_dir
|
||||||
|
|
||||||
|
@allure.step("Prepare containers and objects")
|
||||||
|
def prepare(self) -> str:
|
||||||
|
self._k6_dir = self.k6_dir
|
||||||
|
if self.load_params.load_type == "http" or self.load_params.load_type == "grpc":
|
||||||
|
command = (
|
||||||
|
f"{self.k6_dir}/scenarios/preset/preset_grpc.py "
|
||||||
|
f"--size {self.load_params.obj_size} "
|
||||||
|
f"--containers {self.load_params.containers_count} "
|
||||||
|
f"--out {self.k6_dir}/{self.load_params.load_type}_{self.load_params.out_file} "
|
||||||
|
f"--endpoint {self.load_params.endpoint.split(',')[0]} "
|
||||||
|
f"--preload_obj {self.load_params.obj_count} "
|
||||||
|
)
|
||||||
|
terminal = self.shell.exec(command)
|
||||||
|
return terminal.stdout.strip("\n")
|
||||||
|
elif self.load_params.load_type == "s3":
|
||||||
|
command = (
|
||||||
|
f"{self.k6_dir}/scenarios/preset/preset_s3.py --size {self.load_params.obj_size} "
|
||||||
|
f"--buckets {self.load_params.containers_count} "
|
||||||
|
f"--out {self.k6_dir}/{self.load_params.load_type}_{self.load_params.out_file} "
|
||||||
|
f"--endpoint {self.load_params.endpoint.split(',')[0]} "
|
||||||
|
f"--preload_obj {self.load_params.obj_count} "
|
||||||
|
f"--location load-1-1"
|
||||||
|
)
|
||||||
|
terminal = self.shell.exec(command)
|
||||||
|
return terminal.stdout.strip("\n")
|
||||||
|
else:
|
||||||
|
raise AssertionError("Wrong K6 load type")
|
||||||
|
|
||||||
|
@allure.step("Generate K6 command")
|
||||||
|
def _generate_env_variables(self, load_params: LoadParams, k6_dir: str) -> str:
|
||||||
|
env_vars = {
|
||||||
|
"DURATION": load_params.load_time or None,
|
||||||
|
"WRITE_OBJ_SIZE": load_params.obj_size or None,
|
||||||
|
"WRITERS": load_params.writers or 0,
|
||||||
|
"READERS": load_params.readers or 0,
|
||||||
|
"DELETERS": load_params.deleters or 0,
|
||||||
|
"REGISTRY_FILE": load_params.registry_file or None,
|
||||||
|
"CLIENTS": load_params.clients or None,
|
||||||
|
f"{self.load_params.load_type.upper()}_ENDPOINTS": self.load_params.endpoint,
|
||||||
|
"PREGEN_JSON": f"{self.k6_dir}/{self.load_params.load_type}_{self.load_params.out_file}"
|
||||||
|
if load_params.out_file
|
||||||
|
else None,
|
||||||
|
}
|
||||||
|
allure.attach(
|
||||||
|
"\n".join(f"{param}: {value}" for param, value in env_vars.items()),
|
||||||
|
"K6 ENV variables",
|
||||||
|
allure.attachment_type.TEXT,
|
||||||
|
)
|
||||||
|
return " ".join(
|
||||||
|
[f"-e {param}={value}" for param, value in env_vars.items() if value is not None]
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Start K6 on initiator")
|
||||||
|
def start(self) -> None:
|
||||||
|
|
||||||
|
self._k6_dir = self.k6_dir
|
||||||
|
command = (
|
||||||
|
f"{self.k6_dir}/k6 run {self._generate_env_variables(self.load_params, self.k6_dir)} "
|
||||||
|
f"{self.k6_dir}/scenarios/{self.load_params.load_type}.js"
|
||||||
|
)
|
||||||
|
self._k6_process = RemoteProcess.create(command, self.shell)
|
||||||
|
|
||||||
|
@allure.step("Wait until K6 is finished")
|
||||||
|
def wait_until_finished(self, timeout: int = 0, k6_should_be_running: bool = False) -> None:
|
||||||
|
if self._k6_process is None:
|
||||||
|
assert "No k6 instances were executed"
|
||||||
|
if k6_should_be_running:
|
||||||
|
assert self._k6_process.running(), "k6 should be running."
|
||||||
|
for __attempt in reversed(range(5)) if timeout else [0]:
|
||||||
|
if not self._k6_process.running():
|
||||||
|
return
|
||||||
|
if __attempt: # no sleep in last iteration
|
||||||
|
sleep(int(timeout / 5))
|
||||||
|
self._stop_k6()
|
||||||
|
raise TimeoutError(f"Expected K6 finished in {timeout} sec.")
|
||||||
|
|
||||||
|
@contextmanager
|
||||||
|
def start_context(
|
||||||
|
self, warm_up_time: int = 0, expected_finish: bool = False, expected_fail: bool = False
|
||||||
|
) -> None:
|
||||||
|
self.start()
|
||||||
|
sleep(warm_up_time)
|
||||||
|
try:
|
||||||
|
yield self
|
||||||
|
except Exception as err:
|
||||||
|
if self._k6_process.running():
|
||||||
|
self._kill_k6()
|
||||||
|
raise
|
||||||
|
|
||||||
|
if expected_fail:
|
||||||
|
self._kill_k6()
|
||||||
|
elif expected_finish:
|
||||||
|
if self._k6_process.running():
|
||||||
|
self._kill_k6()
|
||||||
|
raise AssertionError("K6 has not finished in expected time")
|
||||||
|
else:
|
||||||
|
self._k6_should_be_finished()
|
||||||
|
else:
|
||||||
|
self._stop_k6()
|
||||||
|
|
||||||
|
@allure.step("Get K6 results")
|
||||||
|
def get_k6_results(self) -> None:
|
||||||
|
self.__log_k6_output()
|
||||||
|
|
||||||
|
@allure.step("Assert K6 should be finished")
|
||||||
|
def _k6_should_be_finished(self) -> None:
|
||||||
|
k6_rc = self._k6_process.rc()
|
||||||
|
assert k6_rc == 0, f"K6 unexpectedly finished with RC {k6_rc}"
|
||||||
|
|
||||||
|
@allure.step("Terminate K6 on initiator")
|
||||||
|
def stop(self) -> None:
|
||||||
|
if not self._k6_process.running():
|
||||||
|
raise AssertionError("K6 unexpectedly finished")
|
||||||
|
|
||||||
|
self._stop_k6()
|
||||||
|
|
||||||
|
k6_rc = self._k6_process.rc()
|
||||||
|
assert k6_rc == EXIT_RESULT_CODE, f"Return code of K6 job should be 0, but {k6_rc}"
|
||||||
|
|
||||||
|
def check_k6_is_running(self) -> bool:
|
||||||
|
if self._k6_process:
|
||||||
|
return self._k6_process.running()
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_finished(self) -> bool:
|
||||||
|
return not self._k6_process.running()
|
||||||
|
|
||||||
|
def parsing_results(self) -> LoadResults:
|
||||||
|
output = self._k6_process.stdout(full=True).replace("\n", "")
|
||||||
|
metric_regex_map = {
|
||||||
|
"data_received": r"data_received\W*\d*.\d*.\w*\W*(?P<data_received>\d*)",
|
||||||
|
"data_sent": r"data_sent\W*\d*.\d*.\w*\W*(?P<data_sent>\d*)",
|
||||||
|
}
|
||||||
|
metric_regex_map.update(LOAD_RESULTS_PATTERNS[self.load_params.load_type])
|
||||||
|
metric_values = {}
|
||||||
|
for metric_name, metric_regex in metric_regex_map.items():
|
||||||
|
match = re.search(metric_regex, output)
|
||||||
|
if match:
|
||||||
|
metric_values[metric_name] = float(match.group(metric_name))
|
||||||
|
continue
|
||||||
|
metric_values[metric_name] = 0.0
|
||||||
|
load_result = LoadResults(**metric_values)
|
||||||
|
return load_result
|
||||||
|
|
||||||
|
@allure.step("Try to stop K6 with SIGTERM")
|
||||||
|
def _stop_k6(self) -> None:
|
||||||
|
for __attempt in range(self._k6_stop_attempts):
|
||||||
|
if not self._k6_process.running():
|
||||||
|
break
|
||||||
|
|
||||||
|
self._k6_process.stop()
|
||||||
|
sleep(self._k6_stop_timeout)
|
||||||
|
else:
|
||||||
|
raise AssertionError("Can not stop K6 process within timeout")
|
||||||
|
|
||||||
|
def _kill_k6(self) -> None:
|
||||||
|
self._k6_process.kill()
|
||||||
|
|
||||||
|
@allure.step("Log K6 output")
|
||||||
|
def __log_k6_output(self) -> None:
|
||||||
|
allure.attach(self._k6_process.stdout(full=True), "K6 output", allure.attachment_type.TEXT)
|
187
pytest_tests/helpers/remote_process.py
Normal file
187
pytest_tests/helpers/remote_process.py
Normal file
|
@ -0,0 +1,187 @@
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import uuid
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_testlib.shell.interfaces import CommandOptions
|
||||||
|
from tenacity import retry, stop_after_attempt, wait_fixed
|
||||||
|
|
||||||
|
|
||||||
|
class RemoteProcess:
|
||||||
|
def __init__(self, cmd: str, process_dir: str, shell: Shell):
|
||||||
|
self.process_dir = process_dir
|
||||||
|
self.cmd = cmd
|
||||||
|
self.stdout_last_line_number = 0
|
||||||
|
self.stderr_last_line_number = 0
|
||||||
|
self.pid: Optional[str] = None
|
||||||
|
self.proc_rc: Optional[int] = None
|
||||||
|
self.saved_stdout: Optional[str] = None
|
||||||
|
self.saved_stderr: Optional[str] = None
|
||||||
|
self.shell = shell
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
@allure.step("Create remote process")
|
||||||
|
def create(cls, command: str, shell: Shell) -> RemoteProcess:
|
||||||
|
"""
|
||||||
|
Create a process on a remote host.
|
||||||
|
|
||||||
|
Created dir for process with following files:
|
||||||
|
command.sh: script to execute
|
||||||
|
pid: contains process id
|
||||||
|
rc: contains script return code
|
||||||
|
stderr: contains script errors
|
||||||
|
stdout: contains script output
|
||||||
|
|
||||||
|
Args:
|
||||||
|
shell: Shell instance
|
||||||
|
command: command to be run on a remote host
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
RemoteProcess instance for further examination
|
||||||
|
"""
|
||||||
|
remote_process = cls(cmd=command, process_dir=f"/tmp/proc_{uuid.uuid4()}", shell=shell)
|
||||||
|
remote_process._create_process_dir()
|
||||||
|
remote_process._generate_command_script(command)
|
||||||
|
remote_process._start_process()
|
||||||
|
remote_process.pid = remote_process._get_pid()
|
||||||
|
return remote_process
|
||||||
|
|
||||||
|
@allure.step("Get process stdout")
|
||||||
|
def stdout(self, full: bool = False) -> str:
|
||||||
|
"""
|
||||||
|
Method to get process stdout, either fresh info or full.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
full: returns full stdout that we have to this moment
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Fresh stdout. By means of stdout_last_line_number only new stdout lines are returned.
|
||||||
|
If process is finished (proc_rc is not None) saved stdout is returned
|
||||||
|
"""
|
||||||
|
if self.saved_stdout is not None:
|
||||||
|
cur_stdout = self.saved_stdout
|
||||||
|
else:
|
||||||
|
terminal = self.shell.exec(f"cat {self.process_dir}/stdout")
|
||||||
|
if self.proc_rc is not None:
|
||||||
|
self.saved_stdout = terminal.stdout
|
||||||
|
cur_stdout = terminal.stdout
|
||||||
|
|
||||||
|
if full:
|
||||||
|
return cur_stdout
|
||||||
|
whole_stdout = cur_stdout.split("\n")
|
||||||
|
if len(whole_stdout) > self.stdout_last_line_number:
|
||||||
|
resulted_stdout = "\n".join(whole_stdout[self.stdout_last_line_number :])
|
||||||
|
self.stdout_last_line_number = len(whole_stdout)
|
||||||
|
return resulted_stdout
|
||||||
|
return ""
|
||||||
|
|
||||||
|
@allure.step("Get process stderr")
|
||||||
|
def stderr(self, full: bool = False) -> str:
|
||||||
|
"""
|
||||||
|
Method to get process stderr, either fresh info or full.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
full: returns full stderr that we have to this moment
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Fresh stderr. By means of stderr_last_line_number only new stderr lines are returned.
|
||||||
|
If process is finished (proc_rc is not None) saved stderr is returned
|
||||||
|
"""
|
||||||
|
if self.saved_stderr is not None:
|
||||||
|
cur_stderr = self.saved_stderr
|
||||||
|
else:
|
||||||
|
terminal = self.shell.exec(f"cat {self.process_dir}/stderr")
|
||||||
|
if self.proc_rc is not None:
|
||||||
|
self.saved_stderr = terminal.stdout
|
||||||
|
cur_stderr = terminal.stdout
|
||||||
|
if full:
|
||||||
|
return cur_stderr
|
||||||
|
whole_stderr = cur_stderr.split("\n")
|
||||||
|
if len(whole_stderr) > self.stderr_last_line_number:
|
||||||
|
resulted_stderr = "\n".join(whole_stderr[self.stderr_last_line_number :])
|
||||||
|
self.stderr_last_line_number = len(whole_stderr)
|
||||||
|
return resulted_stderr
|
||||||
|
return ""
|
||||||
|
|
||||||
|
@allure.step("Get process rc")
|
||||||
|
def rc(self) -> Optional[int]:
|
||||||
|
if self.proc_rc is not None:
|
||||||
|
return self.proc_rc
|
||||||
|
|
||||||
|
terminal = self.shell.exec(f"cat {self.process_dir}/rc", CommandOptions(check=False))
|
||||||
|
if "No such file or directory" in terminal.stderr:
|
||||||
|
return None
|
||||||
|
elif terminal.stderr or terminal.return_code != 0:
|
||||||
|
raise AssertionError(f"cat process rc was not successfull: {terminal.stderr}")
|
||||||
|
|
||||||
|
self.proc_rc = int(terminal.stdout)
|
||||||
|
return self.proc_rc
|
||||||
|
|
||||||
|
@allure.step("Check if process is running")
|
||||||
|
def running(self) -> bool:
|
||||||
|
return self.rc() is None
|
||||||
|
|
||||||
|
@allure.step("Send signal to process")
|
||||||
|
def send_signal(self, signal: int) -> None:
|
||||||
|
kill_res = self.shell.exec(f"kill -{signal} {self.pid}", CommandOptions(check=False))
|
||||||
|
if "No such process" in kill_res.stderr:
|
||||||
|
return
|
||||||
|
if kill_res.return_code:
|
||||||
|
raise AssertionError(
|
||||||
|
f"Signal {signal} not sent. Return code of kill: {kill_res.return_code}"
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Stop process")
|
||||||
|
def stop(self) -> None:
|
||||||
|
self.send_signal(15)
|
||||||
|
|
||||||
|
@allure.step("Kill process")
|
||||||
|
def kill(self) -> None:
|
||||||
|
self.send_signal(9)
|
||||||
|
|
||||||
|
@allure.step("Clear process directory")
|
||||||
|
def clear(self) -> None:
|
||||||
|
if self.process_dir == "/":
|
||||||
|
raise AssertionError(f"Invalid path to delete: {self.process_dir}")
|
||||||
|
self.shell.exec(f"rm -rf {self.process_dir}")
|
||||||
|
|
||||||
|
@allure.step("Start remote process")
|
||||||
|
def _start_process(self) -> None:
|
||||||
|
self.shell.exec(
|
||||||
|
f"nohup {self.process_dir}/command.sh </dev/null "
|
||||||
|
f">{self.process_dir}/stdout "
|
||||||
|
f"2>{self.process_dir}/stderr &"
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Create process directory")
|
||||||
|
def _create_process_dir(self) -> None:
|
||||||
|
self.shell.exec(f"mkdir {self.process_dir}; chmod 777 {self.process_dir}")
|
||||||
|
terminal = self.shell.exec(f"realpath {self.process_dir}")
|
||||||
|
self.process_dir = terminal.stdout.strip()
|
||||||
|
|
||||||
|
@allure.step("Get pid")
|
||||||
|
@retry(wait=wait_fixed(10), stop=stop_after_attempt(5), reraise=True)
|
||||||
|
def _get_pid(self) -> str:
|
||||||
|
terminal = self.shell.exec(f"cat {self.process_dir}/pid")
|
||||||
|
assert terminal.stdout, f"invalid pid: {terminal.stdout}"
|
||||||
|
return terminal.stdout.strip()
|
||||||
|
|
||||||
|
@allure.step("Generate command script")
|
||||||
|
def _generate_command_script(self, command: str) -> None:
|
||||||
|
command = command.replace('"', '\\"').replace("\\", "\\\\")
|
||||||
|
script = (
|
||||||
|
f"#!/bin/bash\n"
|
||||||
|
f"cd {self.process_dir}\n"
|
||||||
|
f"{command} &\n"
|
||||||
|
f"pid=\$!\n"
|
||||||
|
f"cd {self.process_dir}\n"
|
||||||
|
f"echo \$pid > {self.process_dir}/pid\n"
|
||||||
|
f"wait \$pid\n"
|
||||||
|
f"echo $? > {self.process_dir}/rc"
|
||||||
|
)
|
||||||
|
|
||||||
|
self.shell.exec(f'echo "{script}" > {self.process_dir}/command.sh')
|
||||||
|
self.shell.exec(f"cat {self.process_dir}/command.sh")
|
||||||
|
self.shell.exec(f"chmod +x {self.process_dir}/command.sh")
|
129
pytest_tests/helpers/s3_helper.py
Normal file
129
pytest_tests/helpers/s3_helper.py
Normal file
|
@ -0,0 +1,129 @@
|
||||||
|
import datetime
|
||||||
|
import os
|
||||||
|
from datetime import datetime, timedelta
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import s3_gate_bucket
|
||||||
|
import s3_gate_object
|
||||||
|
from dateutil.parser import parse
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Expected all objects are presented in the bucket")
|
||||||
|
def check_objects_in_bucket(
|
||||||
|
s3_client, bucket, expected_objects: list, unexpected_objects: Optional[list] = None
|
||||||
|
) -> None:
|
||||||
|
unexpected_objects = unexpected_objects or []
|
||||||
|
bucket_objects = s3_gate_object.list_objects_s3(s3_client, bucket)
|
||||||
|
assert len(bucket_objects) == len(
|
||||||
|
expected_objects
|
||||||
|
), f"Expected {len(expected_objects)} objects in the bucket"
|
||||||
|
for bucket_object in expected_objects:
|
||||||
|
assert (
|
||||||
|
bucket_object in bucket_objects
|
||||||
|
), f"Expected object {bucket_object} in objects list {bucket_objects}"
|
||||||
|
|
||||||
|
for bucket_object in unexpected_objects:
|
||||||
|
assert (
|
||||||
|
bucket_object not in bucket_objects
|
||||||
|
), f"Expected object {bucket_object} not in objects list {bucket_objects}"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Try to get object and got error")
|
||||||
|
def try_to_get_objects_and_expect_error(s3_client, bucket: str, object_keys: list) -> None:
|
||||||
|
for obj in object_keys:
|
||||||
|
try:
|
||||||
|
s3_gate_object.get_object_s3(s3_client, bucket, obj)
|
||||||
|
raise AssertionError(f"Object {obj} found in bucket {bucket}")
|
||||||
|
except Exception as err:
|
||||||
|
assert "The specified key does not exist" in str(
|
||||||
|
err
|
||||||
|
), f"Expected error in exception {err}"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Set versioning enable for bucket")
|
||||||
|
def set_bucket_versioning(s3_client, bucket: str, status: s3_gate_bucket.VersioningStatus):
|
||||||
|
s3_gate_bucket.get_bucket_versioning_status(s3_client, bucket)
|
||||||
|
s3_gate_bucket.set_bucket_versioning(s3_client, bucket, status=status)
|
||||||
|
bucket_status = s3_gate_bucket.get_bucket_versioning_status(s3_client, bucket)
|
||||||
|
assert bucket_status == status.value, f"Expected {bucket_status} status. Got {status.value}"
|
||||||
|
|
||||||
|
|
||||||
|
def object_key_from_file_path(full_path: str) -> str:
|
||||||
|
return os.path.basename(full_path)
|
||||||
|
|
||||||
|
|
||||||
|
def assert_tags(
|
||||||
|
actual_tags: list, expected_tags: Optional[list] = None, unexpected_tags: Optional[list] = None
|
||||||
|
) -> None:
|
||||||
|
expected_tags = (
|
||||||
|
[{"Key": key, "Value": value} for key, value in expected_tags] if expected_tags else []
|
||||||
|
)
|
||||||
|
unexpected_tags = (
|
||||||
|
[{"Key": key, "Value": value} for key, value in unexpected_tags] if unexpected_tags else []
|
||||||
|
)
|
||||||
|
if expected_tags == []:
|
||||||
|
assert not actual_tags, f"Expected there is no tags, got {actual_tags}"
|
||||||
|
assert len(expected_tags) == len(actual_tags)
|
||||||
|
for tag in expected_tags:
|
||||||
|
assert tag in actual_tags, f"Tag {tag} must be in {actual_tags}"
|
||||||
|
for tag in unexpected_tags:
|
||||||
|
assert tag not in actual_tags, f"Tag {tag} should not be in {actual_tags}"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Expected all tags are presented in object")
|
||||||
|
def check_tags_by_object(
|
||||||
|
s3_client,
|
||||||
|
bucket: str,
|
||||||
|
key_name: str,
|
||||||
|
expected_tags: list,
|
||||||
|
unexpected_tags: Optional[list] = None,
|
||||||
|
) -> None:
|
||||||
|
actual_tags = s3_gate_object.get_object_tagging(s3_client, bucket, key_name)
|
||||||
|
assert_tags(
|
||||||
|
expected_tags=expected_tags, unexpected_tags=unexpected_tags, actual_tags=actual_tags
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Expected all tags are presented in bucket")
|
||||||
|
def check_tags_by_bucket(
|
||||||
|
s3_client, bucket: str, expected_tags: list, unexpected_tags: Optional[list] = None
|
||||||
|
) -> None:
|
||||||
|
actual_tags = s3_gate_bucket.get_bucket_tagging(s3_client, bucket)
|
||||||
|
assert_tags(
|
||||||
|
expected_tags=expected_tags, unexpected_tags=unexpected_tags, actual_tags=actual_tags
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def assert_object_lock_mode(
|
||||||
|
s3_client,
|
||||||
|
bucket: str,
|
||||||
|
file_name: str,
|
||||||
|
object_lock_mode: str,
|
||||||
|
retain_untile_date: datetime,
|
||||||
|
legal_hold_status: str = "OFF",
|
||||||
|
retain_period: Optional[int] = None,
|
||||||
|
):
|
||||||
|
object_dict = s3_gate_object.get_object_s3(s3_client, bucket, file_name, full_output=True)
|
||||||
|
assert (
|
||||||
|
object_dict.get("ObjectLockMode") == object_lock_mode
|
||||||
|
), f"Expected Object Lock Mode is {object_lock_mode}"
|
||||||
|
assert (
|
||||||
|
object_dict.get("ObjectLockLegalHoldStatus") == legal_hold_status
|
||||||
|
), f"Expected Object Lock Legal Hold Status is {legal_hold_status}"
|
||||||
|
object_retain_date = object_dict.get("ObjectLockRetainUntilDate")
|
||||||
|
retain_date = (
|
||||||
|
parse(object_retain_date) if isinstance(object_retain_date, str) else object_retain_date
|
||||||
|
)
|
||||||
|
if retain_untile_date:
|
||||||
|
assert retain_date.strftime("%Y-%m-%dT%H:%M:%S") == retain_untile_date.strftime(
|
||||||
|
"%Y-%m-%dT%H:%M:%S"
|
||||||
|
), f'Expected Object Lock Retain Until Date is {str(retain_untile_date.strftime("%Y-%m-%dT%H:%M:%S"))}'
|
||||||
|
elif retain_period:
|
||||||
|
last_modify_date = object_dict.get("LastModified")
|
||||||
|
last_modify = (
|
||||||
|
parse(last_modify_date) if isinstance(last_modify_date, str) else last_modify_date
|
||||||
|
)
|
||||||
|
assert (
|
||||||
|
retain_date - last_modify + timedelta(seconds=1)
|
||||||
|
).days == retain_period, f"Expected retention period is {retain_period} days"
|
25
pytest_tests/helpers/storage_object_info.py
Normal file
25
pytest_tests/helpers/storage_object_info.py
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class ObjectRef:
|
||||||
|
cid: str
|
||||||
|
oid: str
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class LockObjectInfo(ObjectRef):
|
||||||
|
lifetime: Optional[int] = None
|
||||||
|
expire_at: Optional[int] = None
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class StorageObjectInfo(ObjectRef):
|
||||||
|
size: Optional[int] = None
|
||||||
|
wallet_file_path: Optional[str] = None
|
||||||
|
file_path: Optional[str] = None
|
||||||
|
file_hash: Optional[str] = None
|
||||||
|
attributes: Optional[list[dict[str, str]]] = None
|
||||||
|
tombstone: Optional[str] = None
|
||||||
|
locks: Optional[list[LockObjectInfo]] = None
|
80
pytest_tests/helpers/test_control.py
Normal file
80
pytest_tests/helpers/test_control.py
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
import logging
|
||||||
|
from functools import wraps
|
||||||
|
from time import sleep, time
|
||||||
|
|
||||||
|
from _pytest.outcomes import Failed
|
||||||
|
from pytest import fail
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
class expect_not_raises:
|
||||||
|
"""
|
||||||
|
Decorator/Context manager check that some action, method or test does not raises exceptions
|
||||||
|
|
||||||
|
Useful to set proper state of failed test cases in allure
|
||||||
|
|
||||||
|
Example:
|
||||||
|
def do_stuff():
|
||||||
|
raise Exception("Fail")
|
||||||
|
|
||||||
|
def test_yellow(): <- this test is marked yellow (Test Defect) in allure
|
||||||
|
do_stuff()
|
||||||
|
|
||||||
|
def test_red(): <- this test is marked red (Failed) in allure
|
||||||
|
with expect_not_raises():
|
||||||
|
do_stuff()
|
||||||
|
|
||||||
|
@expect_not_raises()
|
||||||
|
def test_also_red(): <- this test is also marked red (Failed) in allure
|
||||||
|
do_stuff()
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __enter__(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def __exit__(self, exception_type, exception_value, exception_traceback):
|
||||||
|
if exception_value:
|
||||||
|
fail(str(exception_value))
|
||||||
|
|
||||||
|
def __call__(self, func):
|
||||||
|
@wraps(func)
|
||||||
|
def impl(*a, **kw):
|
||||||
|
with expect_not_raises():
|
||||||
|
func(*a, **kw)
|
||||||
|
|
||||||
|
return impl
|
||||||
|
|
||||||
|
|
||||||
|
def wait_for_success(max_wait_time: int = 60, interval: int = 1):
|
||||||
|
"""
|
||||||
|
Decorator to wait for some conditions/functions to pass successfully.
|
||||||
|
This is useful if you don't know exact time when something should pass successfully and do not
|
||||||
|
want to use sleep(X) with too big X.
|
||||||
|
|
||||||
|
Be careful though, wrapped function should only check the state of something, not change it.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def wrapper(func):
|
||||||
|
@wraps(func)
|
||||||
|
def impl(*a, **kw):
|
||||||
|
start = int(round(time()))
|
||||||
|
last_exception = None
|
||||||
|
while start + max_wait_time >= int(round(time())):
|
||||||
|
try:
|
||||||
|
return func(*a, **kw)
|
||||||
|
except Exception as ex:
|
||||||
|
logger.debug(ex)
|
||||||
|
last_exception = ex
|
||||||
|
sleep(interval)
|
||||||
|
except Failed as ex:
|
||||||
|
logger.debug(ex)
|
||||||
|
last_exception = ex
|
||||||
|
sleep(interval)
|
||||||
|
|
||||||
|
# timeout exceeded with no success, raise last_exception
|
||||||
|
raise last_exception
|
||||||
|
|
||||||
|
return impl
|
||||||
|
|
||||||
|
return wrapper
|
|
@ -1,8 +1,33 @@
|
||||||
import time
|
import time
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
from frostfs_testlib.resources.common import STORAGE_GC_TIME
|
from common import STORAGE_GC_TIME
|
||||||
from frostfs_testlib.utils import datetime_utils
|
|
||||||
|
|
||||||
|
def parse_time(value: str) -> int:
|
||||||
|
"""Converts time interval in text form into time interval as number of seconds.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
value: time interval as text.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Number of seconds in the parsed time interval.
|
||||||
|
"""
|
||||||
|
value = value.lower()
|
||||||
|
|
||||||
|
for suffix in ["s", "sec"]:
|
||||||
|
if value.endswith(suffix):
|
||||||
|
return int(value[: -len(suffix)])
|
||||||
|
|
||||||
|
for suffix in ["m", "min"]:
|
||||||
|
if value.endswith(suffix):
|
||||||
|
return int(value[: -len(suffix)]) * 60
|
||||||
|
|
||||||
|
for suffix in ["h", "hr", "hour"]:
|
||||||
|
if value.endswith(suffix):
|
||||||
|
return int(value[: -len(suffix)]) * 60 * 60
|
||||||
|
|
||||||
|
raise ValueError(f"Unknown units in time value '{value}'")
|
||||||
|
|
||||||
|
|
||||||
def placement_policy_from_container(container_info: str) -> str:
|
def placement_policy_from_container(container_info: str) -> str:
|
||||||
|
@ -22,7 +47,7 @@ def placement_policy_from_container(container_info: str) -> str:
|
||||||
FILTER Country EQ Sweden AS LOC_SW
|
FILTER Country EQ Sweden AS LOC_SW
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
container_info: output from frostfs-cli container get command
|
container_info: output from neofs-cli container get command
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
placement policy as a string
|
placement policy as a string
|
||||||
|
@ -32,6 +57,6 @@ def placement_policy_from_container(container_info: str) -> str:
|
||||||
|
|
||||||
|
|
||||||
def wait_for_gc_pass_on_storage_nodes() -> None:
|
def wait_for_gc_pass_on_storage_nodes() -> None:
|
||||||
wait_time = datetime_utils.parse_time(STORAGE_GC_TIME)
|
wait_time = parse_time(STORAGE_GC_TIME)
|
||||||
with allure.step(f"Wait {wait_time}s until GC completes on storage nodes"):
|
with allure.step(f"Wait {wait_time}s until GC completes on storage nodes"):
|
||||||
time.sleep(wait_time)
|
time.sleep(wait_time)
|
||||||
|
|
70
pytest_tests/helpers/wallet.py
Normal file
70
pytest_tests/helpers/wallet.py
Normal file
|
@ -0,0 +1,70 @@
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from dataclasses import dataclass
|
||||||
|
|
||||||
|
from cluster import Cluster, NodeBase
|
||||||
|
from common import FREE_STORAGE, WALLET_CONFIG, WALLET_PASS
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_testlib.utils.wallet import get_last_address_from_wallet, init_wallet
|
||||||
|
from python_keywords.payment_neogo import deposit_gas, transfer_gas
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class WalletFile:
|
||||||
|
path: str
|
||||||
|
password: str = WALLET_PASS
|
||||||
|
config_path: str = WALLET_CONFIG
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_node(node: NodeBase):
|
||||||
|
return WalletFile(
|
||||||
|
node.get_wallet_path(), node.get_wallet_password(), node.get_wallet_config_path()
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_address(self) -> str:
|
||||||
|
"""
|
||||||
|
Extracts the last address from wallet.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The address of the wallet.
|
||||||
|
"""
|
||||||
|
return get_last_address_from_wallet(self.path, self.password)
|
||||||
|
|
||||||
|
|
||||||
|
class WalletFactory:
|
||||||
|
def __init__(self, wallets_dir: str, shell: Shell, cluster: Cluster) -> None:
|
||||||
|
self.shell = shell
|
||||||
|
self.wallets_dir = wallets_dir
|
||||||
|
self.cluster = cluster
|
||||||
|
|
||||||
|
def create_wallet(self, password: str = WALLET_PASS) -> WalletFile:
|
||||||
|
"""
|
||||||
|
Creates new default wallet
|
||||||
|
Args:
|
||||||
|
password: wallet password
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
WalletFile object of new wallet
|
||||||
|
"""
|
||||||
|
wallet_path = os.path.join(self.wallets_dir, f"{str(uuid.uuid4())}.json")
|
||||||
|
init_wallet(wallet_path, password)
|
||||||
|
|
||||||
|
if not FREE_STORAGE:
|
||||||
|
main_chain = self.cluster.main_chain_nodes[0]
|
||||||
|
deposit = 30
|
||||||
|
transfer_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit + 1,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_to_path=wallet_path,
|
||||||
|
wallet_to_password=password,
|
||||||
|
)
|
||||||
|
deposit_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_from_path=wallet_path,
|
||||||
|
wallet_from_password=password,
|
||||||
|
)
|
||||||
|
|
||||||
|
return WalletFile(wallet_path, password)
|
|
@ -13,7 +13,7 @@ markers =
|
||||||
# functional markers
|
# functional markers
|
||||||
container: tests for container creation
|
container: tests for container creation
|
||||||
grpc_api: standard gRPC API tests
|
grpc_api: standard gRPC API tests
|
||||||
grpc_control: tests related to using frostfs-cli control commands
|
grpc_control: tests related to using neofs-cli control commands
|
||||||
grpc_object_lock: gRPC lock tests
|
grpc_object_lock: gRPC lock tests
|
||||||
http_gate: HTTP gate contract
|
http_gate: HTTP gate contract
|
||||||
s3_gate: All S3 gate tests
|
s3_gate: All S3 gate tests
|
||||||
|
@ -25,7 +25,7 @@ markers =
|
||||||
s3_gate_tagging: Tagging S3 gate tests
|
s3_gate_tagging: Tagging S3 gate tests
|
||||||
s3_gate_versioning: Versioning S3 gate tests
|
s3_gate_versioning: Versioning S3 gate tests
|
||||||
long: long tests (with long execution time)
|
long: long tests (with long execution time)
|
||||||
node_mgmt: frostfs control commands
|
node_mgmt: neofs control commands
|
||||||
session_token: tests for operations with session token
|
session_token: tests for operations with session token
|
||||||
static_session: tests for operations with static session token
|
static_session: tests for operations with static session token
|
||||||
bearer: tests for bearer tokens
|
bearer: tests for bearer tokens
|
||||||
|
@ -40,6 +40,6 @@ markers =
|
||||||
failover_network: tests for network failure
|
failover_network: tests for network failure
|
||||||
failover_reboot: tests for system recovery after reboot of a node
|
failover_reboot: tests for system recovery after reboot of a node
|
||||||
add_nodes: add nodes to cluster
|
add_nodes: add nodes to cluster
|
||||||
check_binaries: check frostfs installed binaries versions
|
check_binaries: check neofs installed binaries versions
|
||||||
payments: tests for payment associated operations
|
payments: tests for payment associated operations
|
||||||
load: performance tests
|
load: performance tests
|
||||||
|
|
|
@ -1,7 +0,0 @@
|
||||||
import os
|
|
||||||
|
|
||||||
TEST_CYCLES_COUNT = int(os.getenv("TEST_CYCLES_COUNT", "1"))
|
|
||||||
|
|
||||||
BIN_VERSIONS_FILE = os.getenv("BIN_VERSIONS_FILE")
|
|
||||||
DEVENV_PATH = os.getenv("DEVENV_PATH", os.path.join("..", "frostfs-dev-env"))
|
|
||||||
HOSTING_CONFIG_FILE = os.getenv("HOSTING_CONFIG_FILE", ".devenv.hosting.yaml")
|
|
27
pytest_tests/resources/load_params.py
Normal file
27
pytest_tests/resources/load_params.py
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
import os
|
||||||
|
|
||||||
|
# Load node parameters
|
||||||
|
LOAD_NODES = os.getenv("LOAD_NODES", "").split(",")
|
||||||
|
LOAD_NODE_SSH_USER = os.getenv("LOAD_NODE_SSH_USER", "root")
|
||||||
|
LOAD_NODE_SSH_PRIVATE_KEY_PATH = os.getenv("LOAD_NODE_SSH_PRIVATE_KEY_PATH")
|
||||||
|
BACKGROUND_WRITERS_COUNT = os.getenv("BACKGROUND_WRITERS_COUNT", 10)
|
||||||
|
BACKGROUND_READERS_COUNT = os.getenv("BACKGROUND_READERS_COUNT", 10)
|
||||||
|
BACKGROUND_OBJ_SIZE = os.getenv("BACKGROUND_OBJ_SIZE", 1024)
|
||||||
|
BACKGROUND_LOAD_MAX_TIME = os.getenv("BACKGROUND_LOAD_MAX_TIME", 600)
|
||||||
|
|
||||||
|
# Load run parameters
|
||||||
|
|
||||||
|
OBJ_SIZE = os.getenv("OBJ_SIZE", "1000").split(",")
|
||||||
|
CONTAINERS_COUNT = os.getenv("CONTAINERS_COUNT", "1").split(",")
|
||||||
|
OUT_FILE = os.getenv("OUT_FILE", "1mb_200.json").split(",")
|
||||||
|
OBJ_COUNT = os.getenv("OBJ_COUNT", "4").split(",")
|
||||||
|
WRITERS = os.getenv("WRITERS", "200").split(",")
|
||||||
|
READERS = os.getenv("READER", "0").split(",")
|
||||||
|
DELETERS = os.getenv("DELETERS", "0").split(",")
|
||||||
|
LOAD_TIME = os.getenv("LOAD_TIME", "200").split(",")
|
||||||
|
LOAD_TYPE = os.getenv("LOAD_TYPE", "grpc").split(",")
|
||||||
|
LOAD_NODES_COUNT = os.getenv("LOAD_NODES_COUNT", "1").split(",")
|
||||||
|
STORAGE_NODE_COUNT = os.getenv("STORAGE_NODE_COUNT", "4").split(",")
|
||||||
|
CONTAINER_PLACEMENT_POLICY = os.getenv(
|
||||||
|
"CONTAINER_PLACEMENT_POLICY", "REP 1 IN X CBF 1 SELECT 1 FROM * AS X"
|
||||||
|
)
|
0
pytest_tests/steps/__init__.py
Normal file
0
pytest_tests/steps/__init__.py
Normal file
34
pytest_tests/steps/cluster_test_base.py
Normal file
34
pytest_tests/steps/cluster_test_base.py
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
import allure
|
||||||
|
import epoch
|
||||||
|
import pytest
|
||||||
|
from cluster import Cluster
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
|
||||||
|
# To skip adding every mandatory singleton dependency to EACH test function
|
||||||
|
class ClusterTestBase:
|
||||||
|
shell: Shell
|
||||||
|
cluster: Cluster
|
||||||
|
|
||||||
|
@pytest.fixture(scope="session", autouse=True)
|
||||||
|
def fill_mandatory_dependencies(self, cluster: Cluster, client_shell: Shell):
|
||||||
|
ClusterTestBase.shell = client_shell
|
||||||
|
ClusterTestBase.cluster = cluster
|
||||||
|
yield
|
||||||
|
|
||||||
|
@allure.title("Tick {epochs_to_tick} epochs")
|
||||||
|
def tick_epochs(self, epochs_to_tick: int):
|
||||||
|
for _ in range(epochs_to_tick):
|
||||||
|
self.tick_epoch()
|
||||||
|
|
||||||
|
def tick_epoch(self):
|
||||||
|
epoch.tick_epoch(self.shell, self.cluster)
|
||||||
|
|
||||||
|
def wait_for_epochs_align(self):
|
||||||
|
epoch.wait_for_epochs_align(self.shell, self.cluster)
|
||||||
|
|
||||||
|
def get_epoch(self):
|
||||||
|
return epoch.get_epoch(self.shell, self.cluster)
|
||||||
|
|
||||||
|
def ensure_fresh_epoch(self):
|
||||||
|
return epoch.ensure_fresh_epoch(self.shell, self.cluster)
|
162
pytest_tests/steps/load.py
Normal file
162
pytest_tests/steps/load.py
Normal file
|
@ -0,0 +1,162 @@
|
||||||
|
import concurrent.futures
|
||||||
|
import re
|
||||||
|
from dataclasses import asdict
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from common import STORAGE_NODE_SERVICE_NAME_REGEX
|
||||||
|
from k6 import K6, LoadParams, LoadResults
|
||||||
|
from neofs_testlib.cli.neofs_authmate import NeofsAuthmate
|
||||||
|
from neofs_testlib.cli.neogo import NeoGo
|
||||||
|
from neofs_testlib.hosting import Hosting
|
||||||
|
from neofs_testlib.shell import CommandOptions, SSHShell
|
||||||
|
from neofs_testlib.shell.interfaces import InteractiveInput
|
||||||
|
|
||||||
|
NEOFS_AUTHMATE_PATH = "neofs-s3-authmate"
|
||||||
|
STOPPED_HOSTS = []
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Get services endpoints")
|
||||||
|
def get_services_endpoints(
|
||||||
|
hosting: Hosting, service_name_regex: str, endpoint_attribute: str
|
||||||
|
) -> list[str]:
|
||||||
|
service_configs = hosting.find_service_configs(service_name_regex)
|
||||||
|
return [service_config.attributes[endpoint_attribute] for service_config in service_configs]
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Stop nodes")
|
||||||
|
def stop_unused_nodes(storage_nodes: list, used_nodes_count: int):
|
||||||
|
for node in storage_nodes[used_nodes_count:]:
|
||||||
|
host = node.host
|
||||||
|
STOPPED_HOSTS.append(host)
|
||||||
|
host.stop_host("hard")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Start nodes")
|
||||||
|
def start_stopped_nodes():
|
||||||
|
for host in STOPPED_HOSTS:
|
||||||
|
host.start_host()
|
||||||
|
STOPPED_HOSTS.remove(host)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Init s3 client")
|
||||||
|
def init_s3_client(
|
||||||
|
load_nodes: list, login: str, pkey: str, container_placement_policy: str, hosting: Hosting
|
||||||
|
):
|
||||||
|
service_configs = hosting.find_service_configs(STORAGE_NODE_SERVICE_NAME_REGEX)
|
||||||
|
host = hosting.get_host_by_service(service_configs[0].name)
|
||||||
|
wallet_path = service_configs[0].attributes["wallet_path"]
|
||||||
|
neogo_cli_config = host.get_cli_config("neo-go")
|
||||||
|
neogo_wallet = NeoGo(shell=host.get_shell(), neo_go_exec_path=neogo_cli_config.exec_path).wallet
|
||||||
|
dump_keys_output = neogo_wallet.dump_keys(wallet=wallet_path, wallet_config=None).stdout
|
||||||
|
public_key = str(re.search(r":\n(?P<public_key>.*)", dump_keys_output).group("public_key"))
|
||||||
|
node_endpoint = service_configs[0].attributes["rpc_endpoint"]
|
||||||
|
# prompt_pattern doesn't work at the moment
|
||||||
|
for load_node in load_nodes:
|
||||||
|
ssh_client = SSHShell(host=load_node, login=login, private_key_path=pkey)
|
||||||
|
path = ssh_client.exec(r"sudo find . -name 'k6' -exec dirname {} \; -quit").stdout.strip(
|
||||||
|
"\n"
|
||||||
|
)
|
||||||
|
neofs_authmate_exec = NeofsAuthmate(ssh_client, NEOFS_AUTHMATE_PATH)
|
||||||
|
issue_secret_output = neofs_authmate_exec.secret.issue(
|
||||||
|
wallet=f"{path}/scenarios/files/wallet.json",
|
||||||
|
peer=node_endpoint,
|
||||||
|
bearer_rules=f"{path}/scenarios/files/rules.json",
|
||||||
|
gate_public_key=public_key,
|
||||||
|
container_placement_policy=container_placement_policy,
|
||||||
|
container_policy=f"{path}/scenarios/files/policy.json",
|
||||||
|
wallet_password="",
|
||||||
|
).stdout
|
||||||
|
aws_access_key_id = str(
|
||||||
|
re.search(r"access_key_id.*:\s.(?P<aws_access_key_id>\w*)", issue_secret_output).group(
|
||||||
|
"aws_access_key_id"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
aws_secret_access_key = str(
|
||||||
|
re.search(
|
||||||
|
r"secret_access_key.*:\s.(?P<aws_secret_access_key>\w*)", issue_secret_output
|
||||||
|
).group("aws_secret_access_key")
|
||||||
|
)
|
||||||
|
# prompt_pattern doesn't work at the moment
|
||||||
|
configure_input = [
|
||||||
|
InteractiveInput(prompt_pattern=r"AWS Access Key ID.*", input=aws_access_key_id),
|
||||||
|
InteractiveInput(
|
||||||
|
prompt_pattern=r"AWS Secret Access Key.*", input=aws_secret_access_key
|
||||||
|
),
|
||||||
|
InteractiveInput(prompt_pattern=r".*", input=""),
|
||||||
|
InteractiveInput(prompt_pattern=r".*", input=""),
|
||||||
|
]
|
||||||
|
ssh_client.exec("aws configure", CommandOptions(interactive_inputs=configure_input))
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Clear cache and data from storage nodes")
|
||||||
|
def clear_cache_and_data(hosting: Hosting):
|
||||||
|
service_configs = hosting.find_service_configs(STORAGE_NODE_SERVICE_NAME_REGEX)
|
||||||
|
for service_config in service_configs:
|
||||||
|
host = hosting.get_host_by_service(service_config.name)
|
||||||
|
host.stop_service(service_config.name)
|
||||||
|
host.delete_storage_node_data(service_config.name)
|
||||||
|
host.start_service(service_config.name)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Prepare objects")
|
||||||
|
def prepare_objects(k6_instance: K6):
|
||||||
|
k6_instance.prepare()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Prepare K6 instances and objects")
|
||||||
|
def prepare_k6_instances(
|
||||||
|
load_nodes: list, login: str, pkey: str, load_params: LoadParams, prepare: bool = True
|
||||||
|
) -> list[K6]:
|
||||||
|
k6_load_objects = []
|
||||||
|
for load_node in load_nodes:
|
||||||
|
ssh_client = SSHShell(host=load_node, login=login, private_key_path=pkey)
|
||||||
|
k6_load_object = K6(load_params, ssh_client)
|
||||||
|
k6_load_objects.append(k6_load_object)
|
||||||
|
for k6_load_object in k6_load_objects:
|
||||||
|
if prepare:
|
||||||
|
with allure.step("Prepare objects"):
|
||||||
|
prepare_objects(k6_load_object)
|
||||||
|
return k6_load_objects
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Run K6")
|
||||||
|
def run_k6_load(k6_instance: K6) -> LoadResults:
|
||||||
|
with allure.step("Executing load"):
|
||||||
|
k6_instance.start()
|
||||||
|
k6_instance.wait_until_finished(k6_instance.load_params.load_time * 2)
|
||||||
|
with allure.step("Printing results"):
|
||||||
|
k6_instance.get_k6_results()
|
||||||
|
return k6_instance.parsing_results()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("MultiNode K6 Run")
|
||||||
|
def multi_node_k6_run(k6_instances: list) -> dict:
|
||||||
|
results = []
|
||||||
|
avg_results = {}
|
||||||
|
with concurrent.futures.ThreadPoolExecutor() as executor:
|
||||||
|
futures = []
|
||||||
|
for k6_instance in k6_instances:
|
||||||
|
futures.append(executor.submit(run_k6_load, k6_instance))
|
||||||
|
for future in concurrent.futures.as_completed(futures):
|
||||||
|
results.append(asdict(future.result()))
|
||||||
|
for k6_result in results:
|
||||||
|
for key in k6_result:
|
||||||
|
try:
|
||||||
|
avg_results[key] += k6_result[key] / len(results)
|
||||||
|
except KeyError:
|
||||||
|
avg_results[key] = k6_result[key] / len(results)
|
||||||
|
return avg_results
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Compare results")
|
||||||
|
def compare_load_results(result: dict, result_new: dict):
|
||||||
|
for key in result:
|
||||||
|
if result[key] != 0 and result_new[key] != 0:
|
||||||
|
if (abs(result[key] - result_new[key]) / min(result[key], result_new[key])) < 0.25:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
raise AssertionError(f"Difference in {key} values more than 25%")
|
||||||
|
elif result[key] == 0 and result_new[key] == 0:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
raise AssertionError(f"Unexpected zero value in {key}")
|
206
pytest_tests/steps/s3_gate_base.py
Normal file
206
pytest_tests/steps/s3_gate_base.py
Normal file
|
@ -0,0 +1,206 @@
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import uuid
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import boto3
|
||||||
|
import pytest
|
||||||
|
import s3_gate_bucket
|
||||||
|
import s3_gate_object
|
||||||
|
import urllib3
|
||||||
|
from aws_cli_client import AwsCliClient
|
||||||
|
from botocore.config import Config
|
||||||
|
from botocore.exceptions import ClientError
|
||||||
|
from cli_helpers import _cmd_run, _configure_aws_cli, _run_with_passwd
|
||||||
|
from cluster import Cluster
|
||||||
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from common import NEOFS_AUTHMATE_EXEC
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from pytest import FixtureRequest
|
||||||
|
from python_keywords.container import list_containers
|
||||||
|
|
||||||
|
# Disable warnings on self-signed certificate which the
|
||||||
|
# boto library produces on requests to S3-gate in dev-env
|
||||||
|
urllib3.disable_warnings()
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
CREDENTIALS_CREATE_TIMEOUT = "1m"
|
||||||
|
|
||||||
|
# Number of attempts that S3 clients will attempt per each request (1 means single attempt
|
||||||
|
# without any retries)
|
||||||
|
MAX_REQUEST_ATTEMPTS = 1
|
||||||
|
RETRY_MODE = "standard"
|
||||||
|
|
||||||
|
|
||||||
|
class TestS3GateBase(ClusterTestBase):
|
||||||
|
s3_client: Any = None
|
||||||
|
|
||||||
|
@pytest.fixture(scope="class", autouse=True)
|
||||||
|
@allure.title("[Class/Autouse]: Create S3 client")
|
||||||
|
def s3_client(
|
||||||
|
self, default_wallet, client_shell: Shell, request: FixtureRequest, cluster: Cluster
|
||||||
|
) -> Any:
|
||||||
|
wallet = default_wallet
|
||||||
|
s3_bearer_rules_file = f"{os.getcwd()}/robot/resources/files/s3_bearer_rules.json"
|
||||||
|
policy = None if isinstance(request.param, str) else request.param[1]
|
||||||
|
(cid, bucket, access_key_id, secret_access_key, owner_private_key,) = init_s3_credentials(
|
||||||
|
wallet, cluster, s3_bearer_rules_file=s3_bearer_rules_file, policy=policy
|
||||||
|
)
|
||||||
|
containers_list = list_containers(
|
||||||
|
wallet, shell=client_shell, endpoint=self.cluster.default_rpc_endpoint
|
||||||
|
)
|
||||||
|
assert cid in containers_list, f"Expected cid {cid} in {containers_list}"
|
||||||
|
|
||||||
|
if "aws cli" in request.param:
|
||||||
|
client = configure_cli_client(
|
||||||
|
access_key_id, secret_access_key, cluster.default_s3_gate_endpoint
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
client = configure_boto3_client(
|
||||||
|
access_key_id, secret_access_key, cluster.default_s3_gate_endpoint
|
||||||
|
)
|
||||||
|
TestS3GateBase.s3_client = client
|
||||||
|
TestS3GateBase.wallet = wallet
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
@allure.title("Create/delete bucket")
|
||||||
|
def bucket(self):
|
||||||
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
yield bucket
|
||||||
|
self.delete_all_object_in_bucket(bucket)
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
@allure.title("Create two buckets")
|
||||||
|
def two_buckets(self):
|
||||||
|
bucket_1 = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
bucket_2 = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
yield bucket_1, bucket_2
|
||||||
|
for bucket in [bucket_1, bucket_2]:
|
||||||
|
self.delete_all_object_in_bucket(bucket)
|
||||||
|
|
||||||
|
def delete_all_object_in_bucket(self, bucket):
|
||||||
|
versioning_status = s3_gate_bucket.get_bucket_versioning_status(self.s3_client, bucket)
|
||||||
|
if versioning_status == s3_gate_bucket.VersioningStatus.ENABLED.value:
|
||||||
|
# From versioned bucket we should delete all versions and delete markers of all objects
|
||||||
|
objects_versions = s3_gate_object.list_objects_versions_s3(self.s3_client, bucket)
|
||||||
|
if objects_versions:
|
||||||
|
s3_gate_object.delete_object_versions_s3_without_dm(
|
||||||
|
self.s3_client, bucket, objects_versions
|
||||||
|
)
|
||||||
|
objects_delete_markers = s3_gate_object.list_objects_delete_markers_s3(
|
||||||
|
self.s3_client, bucket
|
||||||
|
)
|
||||||
|
if objects_delete_markers:
|
||||||
|
s3_gate_object.delete_object_versions_s3_without_dm(
|
||||||
|
self.s3_client, bucket, objects_delete_markers
|
||||||
|
)
|
||||||
|
|
||||||
|
else:
|
||||||
|
# From non-versioned bucket it's sufficient to delete objects by key
|
||||||
|
objects = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
|
if objects:
|
||||||
|
s3_gate_object.delete_objects_s3(self.s3_client, bucket, objects)
|
||||||
|
objects_delete_markers = s3_gate_object.list_objects_delete_markers_s3(
|
||||||
|
self.s3_client, bucket
|
||||||
|
)
|
||||||
|
if objects_delete_markers:
|
||||||
|
s3_gate_object.delete_object_versions_s3_without_dm(
|
||||||
|
self.s3_client, bucket, objects_delete_markers
|
||||||
|
)
|
||||||
|
|
||||||
|
# Delete the bucket itself
|
||||||
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Init S3 Credentials")
|
||||||
|
def init_s3_credentials(
|
||||||
|
wallet_path: str,
|
||||||
|
cluster: Cluster,
|
||||||
|
s3_bearer_rules_file: Optional[str] = None,
|
||||||
|
policy: Optional[dict] = None,
|
||||||
|
):
|
||||||
|
bucket = str(uuid.uuid4())
|
||||||
|
s3_bearer_rules = s3_bearer_rules_file or "robot/resources/files/s3_bearer_rules.json"
|
||||||
|
|
||||||
|
s3gate_node = cluster.s3gates[0]
|
||||||
|
gate_public_key = s3gate_node.get_wallet_public_key()
|
||||||
|
cmd = (
|
||||||
|
f"{NEOFS_AUTHMATE_EXEC} --debug --with-log --timeout {CREDENTIALS_CREATE_TIMEOUT} "
|
||||||
|
f"issue-secret --wallet {wallet_path} --gate-public-key={gate_public_key} "
|
||||||
|
f"--peer {cluster.default_rpc_endpoint} --container-friendly-name {bucket} "
|
||||||
|
f"--bearer-rules {s3_bearer_rules}"
|
||||||
|
)
|
||||||
|
if policy:
|
||||||
|
cmd += f" --container-policy {policy}'"
|
||||||
|
logger.info(f"Executing command: {cmd}")
|
||||||
|
|
||||||
|
try:
|
||||||
|
output = _run_with_passwd(cmd)
|
||||||
|
logger.info(f"Command completed with output: {output}")
|
||||||
|
|
||||||
|
# output contains some debug info and then several JSON structures, so we find each
|
||||||
|
# JSON structure by curly brackets (naive approach, but works while JSON is not nested)
|
||||||
|
# and then we take JSON containing secret_access_key
|
||||||
|
json_blocks = re.findall(r"\{.*?\}", output, re.DOTALL)
|
||||||
|
for json_block in json_blocks:
|
||||||
|
try:
|
||||||
|
parsed_json_block = json.loads(json_block)
|
||||||
|
if "secret_access_key" in parsed_json_block:
|
||||||
|
return (
|
||||||
|
parsed_json_block["container_id"],
|
||||||
|
bucket,
|
||||||
|
parsed_json_block["access_key_id"],
|
||||||
|
parsed_json_block["secret_access_key"],
|
||||||
|
parsed_json_block["owner_private_key"],
|
||||||
|
)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
raise AssertionError(f"Could not parse info from output\n{output}")
|
||||||
|
raise AssertionError(f"Could not find AWS credentials in output:\n{output}")
|
||||||
|
|
||||||
|
except Exception as exc:
|
||||||
|
raise RuntimeError(f"Failed to init s3 credentials because of error\n{exc}") from exc
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Configure S3 client (boto3)")
|
||||||
|
def configure_boto3_client(access_key_id: str, secret_access_key: str, s3gate_endpoint: str):
|
||||||
|
try:
|
||||||
|
session = boto3.Session()
|
||||||
|
config = Config(
|
||||||
|
retries={
|
||||||
|
"max_attempts": MAX_REQUEST_ATTEMPTS,
|
||||||
|
"mode": RETRY_MODE,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
s3_client = session.client(
|
||||||
|
service_name="s3",
|
||||||
|
aws_access_key_id=access_key_id,
|
||||||
|
aws_secret_access_key=secret_access_key,
|
||||||
|
config=config,
|
||||||
|
endpoint_url=s3gate_endpoint,
|
||||||
|
verify=False,
|
||||||
|
)
|
||||||
|
return s3_client
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Configure S3 client (aws cli)")
|
||||||
|
def configure_cli_client(access_key_id: str, secret_access_key: str, s3gate_endpoint: str):
|
||||||
|
try:
|
||||||
|
client = AwsCliClient(s3gate_endpoint)
|
||||||
|
_configure_aws_cli("aws configure", access_key_id, secret_access_key)
|
||||||
|
_cmd_run(f"aws configure set max_attempts {MAX_REQUEST_ATTEMPTS}")
|
||||||
|
_cmd_run(f"aws configure set retry_mode {RETRY_MODE}")
|
||||||
|
return client
|
||||||
|
except Exception as err:
|
||||||
|
if "command was not found or was not executable" in str(err):
|
||||||
|
pytest.skip("AWS CLI was not found")
|
||||||
|
else:
|
||||||
|
raise RuntimeError("Error while configuring AwsCliClient") from err
|
315
pytest_tests/steps/s3_gate_bucket.py
Normal file
315
pytest_tests/steps/s3_gate_bucket.py
Normal file
|
@ -0,0 +1,315 @@
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import uuid
|
||||||
|
from enum import Enum
|
||||||
|
from time import sleep
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from botocore.exceptions import ClientError
|
||||||
|
from cli_helpers import log_command_execution
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
# Artificial delay that we add after object deletion and container creation
|
||||||
|
# Delay is added because sometimes immediately after deletion object still appears
|
||||||
|
# to be existing (probably because tombstone object takes some time to replicate)
|
||||||
|
# TODO: remove after https://github.com/nspcc-dev/neofs-s3-gw/issues/610 is fixed
|
||||||
|
S3_SYNC_WAIT_TIME = 5
|
||||||
|
|
||||||
|
|
||||||
|
class VersioningStatus(Enum):
|
||||||
|
ENABLED = "Enabled"
|
||||||
|
SUSPENDED = "Suspended"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Create bucket S3")
|
||||||
|
def create_bucket_s3(
|
||||||
|
s3_client,
|
||||||
|
object_lock_enabled_for_bucket: Optional[bool] = None,
|
||||||
|
acl: Optional[str] = None,
|
||||||
|
grant_write: Optional[str] = None,
|
||||||
|
grant_read: Optional[str] = None,
|
||||||
|
grant_full_control: Optional[str] = None,
|
||||||
|
bucket_configuration: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
bucket_name = str(uuid.uuid4())
|
||||||
|
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket_name}
|
||||||
|
if object_lock_enabled_for_bucket is not None:
|
||||||
|
params.update({"ObjectLockEnabledForBucket": object_lock_enabled_for_bucket})
|
||||||
|
if acl is not None:
|
||||||
|
params.update({"ACL": acl})
|
||||||
|
elif grant_write or grant_read or grant_full_control:
|
||||||
|
if grant_write:
|
||||||
|
params.update({"GrantWrite": grant_write})
|
||||||
|
elif grant_read:
|
||||||
|
params.update({"GrantRead": grant_read})
|
||||||
|
elif grant_full_control:
|
||||||
|
params.update({"GrantFullControl": grant_full_control})
|
||||||
|
if bucket_configuration:
|
||||||
|
params.update(
|
||||||
|
{"CreateBucketConfiguration": {"LocationConstraint": bucket_configuration}}
|
||||||
|
)
|
||||||
|
|
||||||
|
s3_bucket = s3_client.create_bucket(**params)
|
||||||
|
log_command_execution(f"Created S3 bucket {bucket_name}", s3_bucket)
|
||||||
|
sleep(S3_SYNC_WAIT_TIME)
|
||||||
|
return bucket_name
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List buckets S3")
|
||||||
|
def list_buckets_s3(s3_client):
|
||||||
|
found_buckets = []
|
||||||
|
try:
|
||||||
|
response = s3_client.list_buckets()
|
||||||
|
log_command_execution("S3 List buckets result", response)
|
||||||
|
|
||||||
|
for bucket in response["Buckets"]:
|
||||||
|
found_buckets.append(bucket["Name"])
|
||||||
|
|
||||||
|
return found_buckets
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete bucket S3")
|
||||||
|
def delete_bucket_s3(s3_client, bucket: str):
|
||||||
|
try:
|
||||||
|
response = s3_client.delete_bucket(Bucket=bucket)
|
||||||
|
log_command_execution("S3 Delete bucket result", response)
|
||||||
|
sleep(S3_SYNC_WAIT_TIME)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
log_command_execution("S3 Delete bucket error", str(err))
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Head bucket S3")
|
||||||
|
def head_bucket(s3_client, bucket: str):
|
||||||
|
try:
|
||||||
|
response = s3_client.head_bucket(Bucket=bucket)
|
||||||
|
log_command_execution("S3 Head bucket result", response)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
log_command_execution("S3 Head bucket error", str(err))
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Set bucket versioning status")
|
||||||
|
def set_bucket_versioning(s3_client, bucket_name: str, status: VersioningStatus) -> None:
|
||||||
|
try:
|
||||||
|
response = s3_client.put_bucket_versioning(
|
||||||
|
Bucket=bucket_name, VersioningConfiguration={"Status": status.value}
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Set bucket versioning to", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during set bucket versioning: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get bucket versioning status")
|
||||||
|
def get_bucket_versioning_status(s3_client, bucket_name: str) -> str:
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_versioning(Bucket=bucket_name)
|
||||||
|
status = response.get("Status")
|
||||||
|
log_command_execution("S3 Got bucket versioning status", response)
|
||||||
|
return status
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during get bucket versioning status: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put bucket tagging")
|
||||||
|
def put_bucket_tagging(s3_client, bucket_name: str, tags: list):
|
||||||
|
try:
|
||||||
|
tags = [{"Key": tag_key, "Value": tag_value} for tag_key, tag_value in tags]
|
||||||
|
tagging = {"TagSet": tags}
|
||||||
|
response = s3_client.put_bucket_tagging(Bucket=bucket_name, Tagging=tagging)
|
||||||
|
log_command_execution("S3 Put bucket tagging", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during put bucket tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get bucket acl")
|
||||||
|
def get_bucket_acl(s3_client, bucket_name: str) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_acl(Bucket=bucket_name)
|
||||||
|
log_command_execution("S3 Get bucket acl", response)
|
||||||
|
return response.get("Grants")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during get bucket tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get bucket tagging")
|
||||||
|
def get_bucket_tagging(s3_client, bucket_name: str) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_tagging(Bucket=bucket_name)
|
||||||
|
log_command_execution("S3 Get bucket tagging", response)
|
||||||
|
return response.get("TagSet")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during get bucket tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete bucket tagging")
|
||||||
|
def delete_bucket_tagging(s3_client, bucket_name: str) -> None:
|
||||||
|
try:
|
||||||
|
response = s3_client.delete_bucket_tagging(Bucket=bucket_name)
|
||||||
|
log_command_execution("S3 Delete bucket tagging", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during delete bucket tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put bucket ACL")
|
||||||
|
def put_bucket_acl_s3(
|
||||||
|
s3_client,
|
||||||
|
bucket: str,
|
||||||
|
acl: Optional[str] = None,
|
||||||
|
grant_write: Optional[str] = None,
|
||||||
|
grant_read: Optional[str] = None,
|
||||||
|
) -> list:
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
if acl:
|
||||||
|
params.update({"ACL": acl})
|
||||||
|
elif grant_write or grant_read:
|
||||||
|
if grant_write:
|
||||||
|
params.update({"GrantWrite": grant_write})
|
||||||
|
elif grant_read:
|
||||||
|
params.update({"GrantRead": grant_read})
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = s3_client.put_bucket_acl(**params)
|
||||||
|
log_command_execution("S3 ACL bucket result", response)
|
||||||
|
return response.get("Grants")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object lock configuration")
|
||||||
|
def put_object_lock_configuration(s3_client, bucket: str, configuration: dict):
|
||||||
|
params = {"Bucket": bucket, "ObjectLockConfiguration": configuration}
|
||||||
|
try:
|
||||||
|
response = s3_client.put_object_lock_configuration(**params)
|
||||||
|
log_command_execution("S3 put_object_lock_configuration result", response)
|
||||||
|
return response
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object lock configuration")
|
||||||
|
def get_object_lock_configuration(s3_client, bucket: str):
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
try:
|
||||||
|
response = s3_client.get_object_lock_configuration(**params)
|
||||||
|
log_command_execution("S3 get_object_lock_configuration result", response)
|
||||||
|
return response.get("ObjectLockConfiguration")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def get_bucket_policy(s3_client, bucket: str):
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_policy(**params)
|
||||||
|
log_command_execution("S3 get_object_lock_configuration result", response)
|
||||||
|
return response.get("ObjectLockConfiguration")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def put_bucket_policy(s3_client, bucket: str, policy: dict):
|
||||||
|
params = {"Bucket": bucket, "Policy": json.dumps(policy)}
|
||||||
|
try:
|
||||||
|
response = s3_client.put_bucket_policy(**params)
|
||||||
|
log_command_execution("S3 put_bucket_policy result", response)
|
||||||
|
return response
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def get_bucket_cors(s3_client, bucket: str):
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_cors(**params)
|
||||||
|
log_command_execution("S3 get_bucket_cors result", response)
|
||||||
|
return response.get("CORSRules")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def get_bucket_location(s3_client, bucket: str):
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
try:
|
||||||
|
response = s3_client.get_bucket_location(**params)
|
||||||
|
log_command_execution("S3 get_bucket_location result", response)
|
||||||
|
return response.get("LocationConstraint")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def put_bucket_cors(s3_client, bucket: str, cors_configuration: dict):
|
||||||
|
params = {"Bucket": bucket, "CORSConfiguration": cors_configuration}
|
||||||
|
try:
|
||||||
|
response = s3_client.put_bucket_cors(**params)
|
||||||
|
log_command_execution("S3 put_bucket_cors result", response)
|
||||||
|
return response
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
def delete_bucket_cors(s3_client, bucket: str):
|
||||||
|
params = {"Bucket": bucket}
|
||||||
|
try:
|
||||||
|
response = s3_client.delete_bucket_cors(**params)
|
||||||
|
log_command_execution("S3 delete_bucket_cors result", response)
|
||||||
|
return response.get("ObjectLockConfiguration")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
594
pytest_tests/steps/s3_gate_object.py
Normal file
594
pytest_tests/steps/s3_gate_object.py
Normal file
|
@ -0,0 +1,594 @@
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from time import sleep
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import pytest
|
||||||
|
import urllib3
|
||||||
|
from aws_cli_client import AwsCliClient
|
||||||
|
from botocore.exceptions import ClientError
|
||||||
|
from cli_helpers import log_command_execution
|
||||||
|
from s3_gate_bucket import S3_SYNC_WAIT_TIME
|
||||||
|
|
||||||
|
##########################################################
|
||||||
|
# Disabling warnings on self-signed certificate which the
|
||||||
|
# boto library produces on requests to S3-gate in dev-env.
|
||||||
|
urllib3.disable_warnings()
|
||||||
|
##########################################################
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
ACL_COPY = [
|
||||||
|
"private",
|
||||||
|
"public-read",
|
||||||
|
"public-read-write",
|
||||||
|
"authenticated-read",
|
||||||
|
"aws-exec-read",
|
||||||
|
"bucket-owner-read",
|
||||||
|
"bucket-owner-full-control",
|
||||||
|
]
|
||||||
|
|
||||||
|
ASSETS_DIR = os.getenv("ASSETS_DIR", "TemporaryDir/")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List objects S3 v2")
|
||||||
|
def list_objects_s3_v2(s3_client, bucket: str, full_output: bool = False) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_objects_v2(Bucket=bucket)
|
||||||
|
content = response.get("Contents", [])
|
||||||
|
log_command_execution("S3 v2 List objects result", response)
|
||||||
|
obj_list = []
|
||||||
|
for obj in content:
|
||||||
|
obj_list.append(obj["Key"])
|
||||||
|
logger.info(f"Found s3 objects: {obj_list}")
|
||||||
|
return response if full_output else obj_list
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List objects S3")
|
||||||
|
def list_objects_s3(s3_client, bucket: str, full_output: bool = False) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_objects(Bucket=bucket)
|
||||||
|
content = response.get("Contents", [])
|
||||||
|
log_command_execution("S3 List objects result", response)
|
||||||
|
obj_list = []
|
||||||
|
for obj in content:
|
||||||
|
obj_list.append(obj["Key"])
|
||||||
|
logger.info(f"Found s3 objects: {obj_list}")
|
||||||
|
return response if full_output else obj_list
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List objects versions S3")
|
||||||
|
def list_objects_versions_s3(s3_client, bucket: str, full_output: bool = False) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_object_versions(Bucket=bucket)
|
||||||
|
versions = response.get("Versions", [])
|
||||||
|
log_command_execution("S3 List objects versions result", response)
|
||||||
|
return response if full_output else versions
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List objects delete markers S3")
|
||||||
|
def list_objects_delete_markers_s3(s3_client, bucket: str, full_output: bool = False) -> list:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_object_versions(Bucket=bucket)
|
||||||
|
delete_markers = response.get("DeleteMarkers", [])
|
||||||
|
log_command_execution("S3 List objects delete markers result", response)
|
||||||
|
return response if full_output else delete_markers
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object S3")
|
||||||
|
def put_object_s3(s3_client, bucket: str, filepath: str, **kwargs):
|
||||||
|
filename = os.path.basename(filepath)
|
||||||
|
|
||||||
|
if isinstance(s3_client, AwsCliClient):
|
||||||
|
file_content = filepath
|
||||||
|
else:
|
||||||
|
with open(filepath, "rb") as put_file:
|
||||||
|
file_content = put_file.read()
|
||||||
|
|
||||||
|
try:
|
||||||
|
params = {"Body": file_content, "Bucket": bucket, "Key": filename}
|
||||||
|
if kwargs:
|
||||||
|
params = {**params, **kwargs}
|
||||||
|
response = s3_client.put_object(**params)
|
||||||
|
log_command_execution("S3 Put object result", response)
|
||||||
|
return response.get("VersionId")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Head object S3")
|
||||||
|
def head_object_s3(s3_client, bucket: str, object_key: str, version_id: Optional[str] = None):
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket, "Key": object_key}
|
||||||
|
if version_id:
|
||||||
|
params["VersionId"] = version_id
|
||||||
|
response = s3_client.head_object(**params)
|
||||||
|
log_command_execution("S3 Head object result", response)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete object S3")
|
||||||
|
def delete_object_s3(
|
||||||
|
s3_client, bucket: str, object_key: str, version_id: Optional[str] = None
|
||||||
|
) -> dict:
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket, "Key": object_key}
|
||||||
|
if version_id:
|
||||||
|
params["VersionId"] = version_id
|
||||||
|
response = s3_client.delete_object(**params)
|
||||||
|
log_command_execution("S3 Delete object result", response)
|
||||||
|
sleep(S3_SYNC_WAIT_TIME)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete objects S3")
|
||||||
|
def delete_objects_s3(s3_client, bucket: str, object_keys: list):
|
||||||
|
try:
|
||||||
|
response = s3_client.delete_objects(Bucket=bucket, Delete=_make_objs_dict(object_keys))
|
||||||
|
log_command_execution("S3 Delete objects result", response)
|
||||||
|
sleep(S3_SYNC_WAIT_TIME)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete object versions S3")
|
||||||
|
def delete_object_versions_s3(s3_client, bucket: str, object_versions: list):
|
||||||
|
try:
|
||||||
|
# Build deletion list in S3 format
|
||||||
|
delete_list = {
|
||||||
|
"Objects": [
|
||||||
|
{
|
||||||
|
"Key": object_version["Key"],
|
||||||
|
"VersionId": object_version["VersionId"],
|
||||||
|
}
|
||||||
|
for object_version in object_versions
|
||||||
|
]
|
||||||
|
}
|
||||||
|
response = s3_client.delete_objects(Bucket=bucket, Delete=delete_list)
|
||||||
|
log_command_execution("S3 Delete objects result", response)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete object versions S3 without delete markers")
|
||||||
|
def delete_object_versions_s3_without_dm(s3_client, bucket: str, object_versions: list):
|
||||||
|
try:
|
||||||
|
# Delete objects without creating delete markers
|
||||||
|
for object_version in object_versions:
|
||||||
|
params = {
|
||||||
|
"Bucket": bucket,
|
||||||
|
"Key": object_version["Key"],
|
||||||
|
"VersionId": object_version["VersionId"],
|
||||||
|
}
|
||||||
|
response = s3_client.delete_object(**params)
|
||||||
|
log_command_execution("S3 Delete object result", response)
|
||||||
|
return response
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object ACL")
|
||||||
|
def put_object_acl_s3(
|
||||||
|
s3_client,
|
||||||
|
bucket: str,
|
||||||
|
object_key: str,
|
||||||
|
acl: Optional[str] = None,
|
||||||
|
grant_write: Optional[str] = None,
|
||||||
|
grant_read: Optional[str] = None,
|
||||||
|
) -> list:
|
||||||
|
if not isinstance(s3_client, AwsCliClient):
|
||||||
|
pytest.skip("Method put_object_acl is not supported by boto3 client")
|
||||||
|
params = {"Bucket": bucket, "Key": object_key}
|
||||||
|
if acl:
|
||||||
|
params.update({"ACL": acl})
|
||||||
|
elif grant_write or grant_read:
|
||||||
|
if grant_write:
|
||||||
|
params.update({"GrantWrite": grant_write})
|
||||||
|
elif grant_read:
|
||||||
|
params.update({"GrantRead": grant_read})
|
||||||
|
try:
|
||||||
|
response = s3_client.put_object_acl(**params)
|
||||||
|
log_command_execution("S3 ACL objects result", response)
|
||||||
|
return response.get("Grants")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object ACL")
|
||||||
|
def get_object_acl_s3(
|
||||||
|
s3_client, bucket: str, object_key: str, version_id: Optional[str] = None
|
||||||
|
) -> list:
|
||||||
|
params = {"Bucket": bucket, "Key": object_key}
|
||||||
|
try:
|
||||||
|
if version_id:
|
||||||
|
params.update({"VersionId": version_id})
|
||||||
|
response = s3_client.get_object_acl(**params)
|
||||||
|
log_command_execution("S3 ACL objects result", response)
|
||||||
|
return response.get("Grants")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Copy object S3")
|
||||||
|
def copy_object_s3(
|
||||||
|
s3_client, bucket: str, object_key: str, bucket_dst: Optional[str] = None, **kwargs
|
||||||
|
) -> str:
|
||||||
|
filename = os.path.join(os.getcwd(), str(uuid.uuid4()))
|
||||||
|
try:
|
||||||
|
params = {
|
||||||
|
"Bucket": bucket_dst or bucket,
|
||||||
|
"CopySource": f"{bucket}/{object_key}",
|
||||||
|
"Key": filename,
|
||||||
|
}
|
||||||
|
if "ACL" in kwargs and kwargs["ACL"] in ACL_COPY:
|
||||||
|
params.update({"ACL": kwargs["ACL"]})
|
||||||
|
if "metadata_directive" in kwargs.keys():
|
||||||
|
params.update({"MetadataDirective": kwargs["metadata_directive"]})
|
||||||
|
if "metadata_directive" in kwargs.keys() and "metadata" in kwargs.keys():
|
||||||
|
params.update({"Metadata": kwargs["metadata"]})
|
||||||
|
if "tagging_directive" in kwargs.keys():
|
||||||
|
params.update({"TaggingDirective": kwargs["tagging_directive"]})
|
||||||
|
if "tagging_directive" in kwargs.keys() and "tagging" in kwargs.keys():
|
||||||
|
params.update({"Tagging": kwargs["tagging"]})
|
||||||
|
response = s3_client.copy_object(**params)
|
||||||
|
log_command_execution("S3 Copy objects result", response)
|
||||||
|
return filename
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object S3")
|
||||||
|
def get_object_s3(
|
||||||
|
s3_client,
|
||||||
|
bucket: str,
|
||||||
|
object_key: str,
|
||||||
|
version_id: Optional[str] = None,
|
||||||
|
range: Optional[list] = None,
|
||||||
|
full_output: bool = False,
|
||||||
|
):
|
||||||
|
filename = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket, "Key": object_key}
|
||||||
|
if version_id:
|
||||||
|
params["VersionId"] = version_id
|
||||||
|
|
||||||
|
if isinstance(s3_client, AwsCliClient):
|
||||||
|
params["file_path"] = filename
|
||||||
|
|
||||||
|
if range:
|
||||||
|
params["Range"] = f"bytes={range[0]}-{range[1]}"
|
||||||
|
|
||||||
|
response = s3_client.get_object(**params)
|
||||||
|
log_command_execution("S3 Get objects result", response)
|
||||||
|
|
||||||
|
if not isinstance(s3_client, AwsCliClient):
|
||||||
|
with open(f"{filename}", "wb") as get_file:
|
||||||
|
chunk = response["Body"].read(1024)
|
||||||
|
while chunk:
|
||||||
|
get_file.write(chunk)
|
||||||
|
chunk = response["Body"].read(1024)
|
||||||
|
return response if full_output else filename
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Create multipart upload S3")
|
||||||
|
def create_multipart_upload_s3(s3_client, bucket_name: str, object_key: str) -> str:
|
||||||
|
try:
|
||||||
|
response = s3_client.create_multipart_upload(Bucket=bucket_name, Key=object_key)
|
||||||
|
log_command_execution("S3 Created multipart upload", response)
|
||||||
|
assert response.get("UploadId"), f"Expected UploadId in response:\n{response}"
|
||||||
|
|
||||||
|
return response.get("UploadId")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List multipart uploads S3")
|
||||||
|
def list_multipart_uploads_s3(s3_client, bucket_name: str) -> Optional[list[dict]]:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_multipart_uploads(Bucket=bucket_name)
|
||||||
|
log_command_execution("S3 List multipart upload", response)
|
||||||
|
|
||||||
|
return response.get("Uploads")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Abort multipart upload S3")
|
||||||
|
def abort_multipart_uploads_s3(s3_client, bucket_name: str, object_key: str, upload_id: str):
|
||||||
|
try:
|
||||||
|
response = s3_client.abort_multipart_upload(
|
||||||
|
Bucket=bucket_name, Key=object_key, UploadId=upload_id
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Abort multipart upload", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Upload part S3")
|
||||||
|
def upload_part_s3(
|
||||||
|
s3_client, bucket_name: str, object_key: str, upload_id: str, part_num: int, filepath: str
|
||||||
|
) -> str:
|
||||||
|
if isinstance(s3_client, AwsCliClient):
|
||||||
|
file_content = filepath
|
||||||
|
else:
|
||||||
|
with open(filepath, "rb") as put_file:
|
||||||
|
file_content = put_file.read()
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = s3_client.upload_part(
|
||||||
|
UploadId=upload_id,
|
||||||
|
Bucket=bucket_name,
|
||||||
|
Key=object_key,
|
||||||
|
PartNumber=part_num,
|
||||||
|
Body=file_content,
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Upload part", response)
|
||||||
|
assert response.get("ETag"), f"Expected ETag in response:\n{response}"
|
||||||
|
|
||||||
|
return response.get("ETag")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Upload copy part S3")
|
||||||
|
def upload_part_copy_s3(
|
||||||
|
s3_client, bucket_name: str, object_key: str, upload_id: str, part_num: int, copy_source: str
|
||||||
|
) -> str:
|
||||||
|
|
||||||
|
try:
|
||||||
|
response = s3_client.upload_part_copy(
|
||||||
|
UploadId=upload_id,
|
||||||
|
Bucket=bucket_name,
|
||||||
|
Key=object_key,
|
||||||
|
PartNumber=part_num,
|
||||||
|
CopySource=copy_source,
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Upload copy part", response)
|
||||||
|
assert response.get("CopyPartResult").get("ETag"), f"Expected ETag in response:\n{response}"
|
||||||
|
|
||||||
|
return response.get("CopyPartResult").get("ETag")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List parts S3")
|
||||||
|
def list_parts_s3(s3_client, bucket_name: str, object_key: str, upload_id: str) -> list[dict]:
|
||||||
|
try:
|
||||||
|
response = s3_client.list_parts(UploadId=upload_id, Bucket=bucket_name, Key=object_key)
|
||||||
|
log_command_execution("S3 List part", response)
|
||||||
|
assert response.get("Parts"), f"Expected Parts in response:\n{response}"
|
||||||
|
|
||||||
|
return response.get("Parts")
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Complete multipart upload S3")
|
||||||
|
def complete_multipart_upload_s3(
|
||||||
|
s3_client, bucket_name: str, object_key: str, upload_id: str, parts: list
|
||||||
|
):
|
||||||
|
try:
|
||||||
|
parts = [{"ETag": etag, "PartNumber": part_num} for part_num, etag in parts]
|
||||||
|
response = s3_client.complete_multipart_upload(
|
||||||
|
Bucket=bucket_name, Key=object_key, UploadId=upload_id, MultipartUpload={"Parts": parts}
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Complete multipart upload", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(
|
||||||
|
f'Error Message: {err.response["Error"]["Message"]}\n'
|
||||||
|
f'Http status code: {err.response["ResponseMetadata"]["HTTPStatusCode"]}'
|
||||||
|
) from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object retention")
|
||||||
|
def put_object_retention(
|
||||||
|
s3_client,
|
||||||
|
bucket_name: str,
|
||||||
|
object_key: str,
|
||||||
|
retention: dict,
|
||||||
|
version_id: Optional[str] = None,
|
||||||
|
bypass_governance_retention: Optional[bool] = None,
|
||||||
|
):
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket_name, "Key": object_key, "Retention": retention}
|
||||||
|
if version_id:
|
||||||
|
params.update({"VersionId": version_id})
|
||||||
|
if not bypass_governance_retention is None:
|
||||||
|
params.update({"BypassGovernanceRetention": bypass_governance_retention})
|
||||||
|
s3_client.put_object_retention(**params)
|
||||||
|
log_command_execution("S3 Put object retention ", str(retention))
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during put object tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object legal hold")
|
||||||
|
def put_object_legal_hold(
|
||||||
|
s3_client, bucket_name: str, object_key: str, legal_hold: str, version_id: Optional[str] = None
|
||||||
|
):
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket_name, "Key": object_key, "LegalHold": {"Status": legal_hold}}
|
||||||
|
if version_id:
|
||||||
|
params.update({"VersionId": version_id})
|
||||||
|
s3_client.put_object_legal_hold(**params)
|
||||||
|
log_command_execution("S3 Put object legal hold ", str(legal_hold))
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during put object tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object tagging")
|
||||||
|
def put_object_tagging(s3_client, bucket_name: str, object_key: str, tags: list):
|
||||||
|
try:
|
||||||
|
tags = [{"Key": tag_key, "Value": tag_value} for tag_key, tag_value in tags]
|
||||||
|
tagging = {"TagSet": tags}
|
||||||
|
s3_client.put_object_tagging(Bucket=bucket_name, Key=object_key, Tagging=tagging)
|
||||||
|
log_command_execution("S3 Put object tagging", str(tags))
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during put object tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object tagging")
|
||||||
|
def get_object_tagging(
|
||||||
|
s3_client, bucket_name: str, object_key: str, version_id: Optional[str] = None
|
||||||
|
) -> list:
|
||||||
|
try:
|
||||||
|
params = {"Bucket": bucket_name, "Key": object_key}
|
||||||
|
if version_id:
|
||||||
|
params.update({"VersionId": version_id})
|
||||||
|
response = s3_client.get_object_tagging(**params)
|
||||||
|
log_command_execution("S3 Get object tagging", response)
|
||||||
|
return response.get("TagSet")
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during get object tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete object tagging")
|
||||||
|
def delete_object_tagging(s3_client, bucket_name: str, object_key: str):
|
||||||
|
try:
|
||||||
|
response = s3_client.delete_object_tagging(Bucket=bucket_name, Key=object_key)
|
||||||
|
log_command_execution("S3 Delete object tagging", response)
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during delete object tagging: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object attributes")
|
||||||
|
def get_object_attributes(
|
||||||
|
s3_client,
|
||||||
|
bucket_name: str,
|
||||||
|
object_key: str,
|
||||||
|
*attributes: str,
|
||||||
|
version_id: Optional[str] = None,
|
||||||
|
max_parts: Optional[int] = None,
|
||||||
|
part_number: Optional[int] = None,
|
||||||
|
get_full_resp: bool = True,
|
||||||
|
) -> dict:
|
||||||
|
try:
|
||||||
|
if not isinstance(s3_client, AwsCliClient):
|
||||||
|
logger.warning("Method get_object_attributes is not supported by boto3 client")
|
||||||
|
return {}
|
||||||
|
response = s3_client.get_object_attributes(
|
||||||
|
bucket_name,
|
||||||
|
object_key,
|
||||||
|
*attributes,
|
||||||
|
version_id=version_id,
|
||||||
|
max_parts=max_parts,
|
||||||
|
part_number=part_number,
|
||||||
|
)
|
||||||
|
log_command_execution("S3 Get object attributes", response)
|
||||||
|
for attr in attributes:
|
||||||
|
assert attr in response, f"Expected attribute {attr} in {response}"
|
||||||
|
|
||||||
|
if get_full_resp:
|
||||||
|
return response
|
||||||
|
else:
|
||||||
|
return response.get(attributes[0])
|
||||||
|
|
||||||
|
except ClientError as err:
|
||||||
|
raise Exception(f"Got error during get object attributes: {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
def _make_objs_dict(key_names):
|
||||||
|
objs_list = []
|
||||||
|
for key in key_names:
|
||||||
|
obj_dict = {"Key": key}
|
||||||
|
objs_list.append(obj_dict)
|
||||||
|
objs_dict = {"Objects": objs_list}
|
||||||
|
return objs_dict
|
281
pytest_tests/steps/session_token.py
Normal file
281
pytest_tests/steps/session_token.py
Normal file
|
@ -0,0 +1,281 @@
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from enum import Enum
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import json_transformers
|
||||||
|
from common import ASSETS_DIR, NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from data_formatters import get_wallet_public_key
|
||||||
|
from json_transformers import encode_for_json
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from storage_object_info import StorageObjectInfo
|
||||||
|
from wallet import WalletFile
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
UNRELATED_KEY = "unrelated key in the session"
|
||||||
|
UNRELATED_OBJECT = "unrelated object in the session"
|
||||||
|
UNRELATED_CONTAINER = "unrelated container in the session"
|
||||||
|
WRONG_VERB = "wrong verb of the session"
|
||||||
|
INVALID_SIGNATURE = "invalid signature of the session data"
|
||||||
|
|
||||||
|
|
||||||
|
class ObjectVerb(Enum):
|
||||||
|
PUT = "PUT"
|
||||||
|
DELETE = "DELETE"
|
||||||
|
GET = "GET"
|
||||||
|
RANGEHASH = "RANGEHASH"
|
||||||
|
RANGE = "RANGE"
|
||||||
|
HEAD = "HEAD"
|
||||||
|
SEARCH = "SEARCH"
|
||||||
|
|
||||||
|
|
||||||
|
class ContainerVerb(Enum):
|
||||||
|
CREATE = "PUT"
|
||||||
|
DELETE = "DELETE"
|
||||||
|
SETEACL = "SETEACL"
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class Lifetime:
|
||||||
|
exp: int = 100000000
|
||||||
|
nbf: int = 0
|
||||||
|
iat: int = 0
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Generate Session Token")
|
||||||
|
def generate_session_token(
|
||||||
|
owner_wallet: WalletFile,
|
||||||
|
session_wallet: WalletFile,
|
||||||
|
session: dict[str, dict[str, Any]],
|
||||||
|
tokens_dir: str,
|
||||||
|
lifetime: Optional[Lifetime] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
This function generates session token and writes it to the file.
|
||||||
|
Args:
|
||||||
|
owner_wallet: wallet of container owner
|
||||||
|
session_wallet: wallet to which we grant the access via session token
|
||||||
|
session: Contains allowed operation with parameters
|
||||||
|
tokens_dir: Dir for token
|
||||||
|
lifetime: lifetime options for session
|
||||||
|
Returns:
|
||||||
|
The path to the generated session token file
|
||||||
|
"""
|
||||||
|
|
||||||
|
file_path = os.path.join(tokens_dir, str(uuid.uuid4()))
|
||||||
|
|
||||||
|
pub_key_64 = get_wallet_public_key(session_wallet.path, session_wallet.password, "base64")
|
||||||
|
|
||||||
|
lifetime = lifetime or Lifetime()
|
||||||
|
|
||||||
|
session_token = {
|
||||||
|
"body": {
|
||||||
|
"id": f"{base64.b64encode(uuid.uuid4().bytes).decode('utf-8')}",
|
||||||
|
"ownerID": {
|
||||||
|
"value": f"{json_transformers.encode_for_json(owner_wallet.get_address())}"
|
||||||
|
},
|
||||||
|
"lifetime": {
|
||||||
|
"exp": f"{lifetime.exp}",
|
||||||
|
"nbf": f"{lifetime.nbf}",
|
||||||
|
"iat": f"{lifetime.iat}",
|
||||||
|
},
|
||||||
|
"sessionKey": pub_key_64,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
session_token["body"].update(session)
|
||||||
|
|
||||||
|
logger.info(f"Got this Session Token: {session_token}")
|
||||||
|
with open(file_path, "w", encoding="utf-8") as session_token_file:
|
||||||
|
json.dump(session_token, session_token_file, ensure_ascii=False, indent=4)
|
||||||
|
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Generate Session Token For Container")
|
||||||
|
def generate_container_session_token(
|
||||||
|
owner_wallet: WalletFile,
|
||||||
|
session_wallet: WalletFile,
|
||||||
|
verb: ContainerVerb,
|
||||||
|
tokens_dir: str,
|
||||||
|
lifetime: Optional[Lifetime] = None,
|
||||||
|
cid: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
This function generates session token for ContainerSessionContext
|
||||||
|
and writes it to the file. It is able to prepare session token file
|
||||||
|
for a specific container (<cid>) or for every container (adds
|
||||||
|
"wildcard" field).
|
||||||
|
Args:
|
||||||
|
owner_wallet: wallet of container owner.
|
||||||
|
session_wallet: wallet to which we grant the access via session token.
|
||||||
|
verb: verb to grant access to.
|
||||||
|
lifetime: lifetime options for session.
|
||||||
|
cid: container ID of the container
|
||||||
|
Returns:
|
||||||
|
The path to the generated session token file
|
||||||
|
"""
|
||||||
|
session = {
|
||||||
|
"container": {
|
||||||
|
"verb": verb.value,
|
||||||
|
"wildcard": cid is None,
|
||||||
|
**({"containerID": {"value": f"{encode_for_json(cid)}"}} if cid is not None else {}),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
return generate_session_token(
|
||||||
|
owner_wallet=owner_wallet,
|
||||||
|
session_wallet=session_wallet,
|
||||||
|
session=session,
|
||||||
|
tokens_dir=tokens_dir,
|
||||||
|
lifetime=lifetime,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Generate Session Token For Object")
|
||||||
|
def generate_object_session_token(
|
||||||
|
owner_wallet: WalletFile,
|
||||||
|
session_wallet: WalletFile,
|
||||||
|
oids: list[str],
|
||||||
|
cid: str,
|
||||||
|
verb: ObjectVerb,
|
||||||
|
tokens_dir: str,
|
||||||
|
lifetime: Optional[Lifetime] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
This function generates session token for ObjectSessionContext
|
||||||
|
and writes it to the file.
|
||||||
|
Args:
|
||||||
|
owner_wallet: wallet of container owner
|
||||||
|
session_wallet: wallet to which we grant the access via session token
|
||||||
|
cid: container ID of the container
|
||||||
|
oids: list of objectIDs to put into session
|
||||||
|
verb: verb to grant access to; Valid verbs are: ObjectVerb.
|
||||||
|
lifetime: lifetime options for session
|
||||||
|
Returns:
|
||||||
|
The path to the generated session token file
|
||||||
|
"""
|
||||||
|
session = {
|
||||||
|
"object": {
|
||||||
|
"verb": verb.value,
|
||||||
|
"target": {
|
||||||
|
"container": {"value": encode_for_json(cid)},
|
||||||
|
"objects": [{"value": encode_for_json(oid)} for oid in oids],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
return generate_session_token(
|
||||||
|
owner_wallet=owner_wallet,
|
||||||
|
session_wallet=session_wallet,
|
||||||
|
session=session,
|
||||||
|
tokens_dir=tokens_dir,
|
||||||
|
lifetime=lifetime,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get signed token for container session")
|
||||||
|
def get_container_signed_token(
|
||||||
|
owner_wallet: WalletFile,
|
||||||
|
user_wallet: WalletFile,
|
||||||
|
verb: ContainerVerb,
|
||||||
|
shell: Shell,
|
||||||
|
tokens_dir: str,
|
||||||
|
lifetime: Optional[Lifetime] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Returns signed token file path for static container session
|
||||||
|
"""
|
||||||
|
session_token_file = generate_container_session_token(
|
||||||
|
owner_wallet=owner_wallet,
|
||||||
|
session_wallet=user_wallet,
|
||||||
|
verb=verb,
|
||||||
|
tokens_dir=tokens_dir,
|
||||||
|
lifetime=lifetime,
|
||||||
|
)
|
||||||
|
return sign_session_token(shell, session_token_file, owner_wallet)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get signed token for object session")
|
||||||
|
def get_object_signed_token(
|
||||||
|
owner_wallet: WalletFile,
|
||||||
|
user_wallet: WalletFile,
|
||||||
|
cid: str,
|
||||||
|
storage_objects: list[StorageObjectInfo],
|
||||||
|
verb: ObjectVerb,
|
||||||
|
shell: Shell,
|
||||||
|
tokens_dir: str,
|
||||||
|
lifetime: Optional[Lifetime] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Returns signed token file path for static object session
|
||||||
|
"""
|
||||||
|
storage_object_ids = [storage_object.oid for storage_object in storage_objects]
|
||||||
|
session_token_file = generate_object_session_token(
|
||||||
|
owner_wallet=owner_wallet,
|
||||||
|
session_wallet=user_wallet,
|
||||||
|
oids=storage_object_ids,
|
||||||
|
cid=cid,
|
||||||
|
verb=verb,
|
||||||
|
tokens_dir=tokens_dir,
|
||||||
|
lifetime=lifetime,
|
||||||
|
)
|
||||||
|
return sign_session_token(shell, session_token_file, owner_wallet)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Create Session Token")
|
||||||
|
def create_session_token(
|
||||||
|
shell: Shell,
|
||||||
|
owner: str,
|
||||||
|
wallet_path: str,
|
||||||
|
wallet_password: str,
|
||||||
|
rpc_endpoint: str,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Create session token for an object.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
owner: User that writes the token.
|
||||||
|
wallet_path: The path to wallet to which we grant the access via session token.
|
||||||
|
wallet_password: Wallet password.
|
||||||
|
rpc_endpoint: Remote node address (as 'multiaddr' or '<host>:<port>').
|
||||||
|
Returns:
|
||||||
|
The path to the generated session token file.
|
||||||
|
"""
|
||||||
|
session_token = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC)
|
||||||
|
neofscli.session.create(
|
||||||
|
rpc_endpoint=rpc_endpoint,
|
||||||
|
address=owner,
|
||||||
|
wallet=wallet_path,
|
||||||
|
wallet_password=wallet_password,
|
||||||
|
out=session_token,
|
||||||
|
)
|
||||||
|
return session_token
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Sign Session Token")
|
||||||
|
def sign_session_token(shell: Shell, session_token_file: str, wlt: WalletFile) -> str:
|
||||||
|
"""
|
||||||
|
This function signs the session token by the given wallet.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
session_token_file: The path to the session token file.
|
||||||
|
wlt: The path to the signing wallet.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
The path to the signed token.
|
||||||
|
"""
|
||||||
|
signed_token_file = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=WALLET_CONFIG)
|
||||||
|
neofscli.util.sign_session_token(
|
||||||
|
wallet=wlt.path, from_file=session_token_file, to_file=signed_token_file
|
||||||
|
)
|
||||||
|
return signed_token_file
|
61
pytest_tests/steps/storage_object.py
Normal file
61
pytest_tests/steps/storage_object.py
Normal file
|
@ -0,0 +1,61 @@
|
||||||
|
import logging
|
||||||
|
from time import sleep
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import pytest
|
||||||
|
from cluster import Cluster
|
||||||
|
from epoch import tick_epoch
|
||||||
|
from grpc_responses import OBJECT_ALREADY_REMOVED
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from python_keywords.neofs_verbs import delete_object, get_object
|
||||||
|
from storage_object_info import StorageObjectInfo
|
||||||
|
from tombstone import verify_head_tombstone
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
CLEANUP_TIMEOUT = 10
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete Objects")
|
||||||
|
def delete_objects(
|
||||||
|
storage_objects: list[StorageObjectInfo], shell: Shell, cluster: Cluster
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Deletes given storage objects.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
storage_objects: list of objects to delete
|
||||||
|
shell: executor for cli command
|
||||||
|
"""
|
||||||
|
|
||||||
|
with allure.step("Delete objects"):
|
||||||
|
for storage_object in storage_objects:
|
||||||
|
storage_object.tombstone = delete_object(
|
||||||
|
storage_object.wallet_file_path,
|
||||||
|
storage_object.cid,
|
||||||
|
storage_object.oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
verify_head_tombstone(
|
||||||
|
wallet_path=storage_object.wallet_file_path,
|
||||||
|
cid=storage_object.cid,
|
||||||
|
oid_ts=storage_object.tombstone,
|
||||||
|
oid=storage_object.oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
|
||||||
|
tick_epoch(shell, cluster)
|
||||||
|
sleep(CLEANUP_TIMEOUT)
|
||||||
|
|
||||||
|
with allure.step("Get objects and check errors"):
|
||||||
|
for storage_object in storage_objects:
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ALREADY_REMOVED):
|
||||||
|
get_object(
|
||||||
|
storage_object.wallet_file_path,
|
||||||
|
storage_object.cid,
|
||||||
|
storage_object.oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=cluster.default_rpc_endpoint,
|
||||||
|
)
|
|
@ -5,16 +5,15 @@ from typing import Optional
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.common import DEFAULT_WALLET_CONFIG, DEFAULT_WALLET_PASS
|
from cluster import Cluster
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from common import WALLET_CONFIG, WALLET_PASS
|
||||||
from frostfs_testlib.shell import Shell
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
from neofs_testlib.utils.wallet import init_wallet
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
from python_keywords.acl import EACLRole
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLRole
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.storage.dataclasses.frostfs_services import InnerRing, StorageNode
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
from frostfs_testlib.utils import wallet_utils
|
from wellknown_acl import PUBLIC_ACL
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
|
||||||
OBJECT_COUNT = 5
|
OBJECT_COUNT = 5
|
||||||
|
|
||||||
|
@ -37,15 +36,15 @@ class Wallets:
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def wallets(default_wallet: str, temp_directory: str, cluster: Cluster) -> Wallets:
|
def wallets(default_wallet, temp_directory, cluster: Cluster) -> Wallets:
|
||||||
other_wallets_paths = [
|
other_wallets_paths = [
|
||||||
os.path.join(temp_directory, f"{str(uuid.uuid4())}.json") for _ in range(2)
|
os.path.join(temp_directory, f"{str(uuid.uuid4())}.json") for _ in range(2)
|
||||||
]
|
]
|
||||||
for other_wallet_path in other_wallets_paths:
|
for other_wallet_path in other_wallets_paths:
|
||||||
wallet_utils.init_wallet(other_wallet_path, DEFAULT_WALLET_PASS)
|
init_wallet(other_wallet_path, WALLET_PASS)
|
||||||
|
|
||||||
ir_node: InnerRing = cluster.ir_nodes[0]
|
ir_node = cluster.ir_nodes[0]
|
||||||
storage_node: StorageNode = cluster.storage_nodes[0]
|
storage_node = cluster.storage_nodes[0]
|
||||||
|
|
||||||
ir_wallet_path = ir_node.get_wallet_path()
|
ir_wallet_path = ir_node.get_wallet_path()
|
||||||
ir_wallet_config = ir_node.get_wallet_config_path()
|
ir_wallet_config = ir_node.get_wallet_config_path()
|
||||||
|
@ -55,9 +54,9 @@ def wallets(default_wallet: str, temp_directory: str, cluster: Cluster) -> Walle
|
||||||
|
|
||||||
yield Wallets(
|
yield Wallets(
|
||||||
wallets={
|
wallets={
|
||||||
EACLRole.USER: [Wallet(wallet_path=default_wallet, config_path=DEFAULT_WALLET_CONFIG)],
|
EACLRole.USER: [Wallet(wallet_path=default_wallet, config_path=WALLET_CONFIG)],
|
||||||
EACLRole.OTHERS: [
|
EACLRole.OTHERS: [
|
||||||
Wallet(wallet_path=other_wallet_path, config_path=DEFAULT_WALLET_CONFIG)
|
Wallet(wallet_path=other_wallet_path, config_path=WALLET_CONFIG)
|
||||||
for other_wallet_path in other_wallets_paths
|
for other_wallet_path in other_wallets_paths
|
||||||
],
|
],
|
||||||
EACLRole.SYSTEM: [
|
EACLRole.SYSTEM: [
|
||||||
|
@ -69,14 +68,14 @@ def wallets(default_wallet: str, temp_directory: str, cluster: Cluster) -> Walle
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def file_path(simple_object_size: int) -> str:
|
def file_path(simple_object_size):
|
||||||
yield generate_file(simple_object_size)
|
yield generate_file(simple_object_size)
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def eacl_container_with_objects(
|
def eacl_container_with_objects(
|
||||||
wallets: Wallets, client_shell: Shell, cluster: Cluster, file_path: str
|
wallets: Wallets, client_shell: Shell, cluster: Cluster, file_path: str
|
||||||
) -> tuple[str, list[str], str]:
|
):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
with allure.step("Create eACL public container"):
|
with allure.step("Create eACL public container"):
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
|
|
472
pytest_tests/testsuites/acl/storage_group/test_storagegroup.py
Normal file
472
pytest_tests/testsuites/acl/storage_group/test_storagegroup.py
Normal file
|
@ -0,0 +1,472 @@
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import pytest
|
||||||
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from common import ASSETS_DIR, FREE_STORAGE, WALLET_PASS
|
||||||
|
from file_helper import generate_file
|
||||||
|
from grpc_responses import OBJECT_ACCESS_DENIED, OBJECT_NOT_FOUND
|
||||||
|
from neofs_testlib.utils.wallet import init_wallet
|
||||||
|
from python_keywords.acl import (
|
||||||
|
EACLAccess,
|
||||||
|
EACLOperation,
|
||||||
|
EACLRole,
|
||||||
|
EACLRule,
|
||||||
|
create_eacl,
|
||||||
|
form_bearertoken_file,
|
||||||
|
set_eacl,
|
||||||
|
)
|
||||||
|
from python_keywords.container import create_container
|
||||||
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
|
from python_keywords.payment_neogo import deposit_gas, transfer_gas
|
||||||
|
from python_keywords.storage_group import (
|
||||||
|
delete_storagegroup,
|
||||||
|
get_storagegroup,
|
||||||
|
list_storagegroup,
|
||||||
|
put_storagegroup,
|
||||||
|
verify_get_storage_group,
|
||||||
|
verify_list_storage_group,
|
||||||
|
)
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
deposit = 30
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"object_size",
|
||||||
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
|
ids=["simple object", "complex object"],
|
||||||
|
)
|
||||||
|
@pytest.mark.sanity
|
||||||
|
@pytest.mark.acl
|
||||||
|
@pytest.mark.storage_group
|
||||||
|
class TestStorageGroup(ClusterTestBase):
|
||||||
|
@pytest.fixture(autouse=True)
|
||||||
|
def prepare_two_wallets(self, default_wallet):
|
||||||
|
self.main_wallet = default_wallet
|
||||||
|
self.other_wallet = os.path.join(os.getcwd(), ASSETS_DIR, f"{str(uuid.uuid4())}.json")
|
||||||
|
init_wallet(self.other_wallet, WALLET_PASS)
|
||||||
|
if not FREE_STORAGE:
|
||||||
|
main_chain = self.cluster.main_chain_nodes[0]
|
||||||
|
deposit = 30
|
||||||
|
transfer_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit + 1,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_to_path=self.other_wallet,
|
||||||
|
wallet_to_password=WALLET_PASS,
|
||||||
|
)
|
||||||
|
deposit_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_from_path=self.other_wallet,
|
||||||
|
wallet_from_password=WALLET_PASS,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test Storage Group in Private Container")
|
||||||
|
def test_storagegroup_basic_private_container(self, object_size, max_object_size):
|
||||||
|
cid = create_container(
|
||||||
|
self.main_wallet, shell=self.shell, endpoint=self.cluster.default_rpc_endpoint
|
||||||
|
)
|
||||||
|
file_path = generate_file(object_size)
|
||||||
|
oid = put_object_to_random_node(self.main_wallet, file_path, cid, self.shell, self.cluster)
|
||||||
|
objects = [oid]
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
objects=objects,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
self.expect_failure_for_storagegroup_operations(
|
||||||
|
wallet=self.other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
gid=storage_group,
|
||||||
|
)
|
||||||
|
self.storagegroup_operations_by_system_ro_container(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test Storage Group in Public Container")
|
||||||
|
def test_storagegroup_basic_public_container(self, object_size, max_object_size):
|
||||||
|
cid = create_container(
|
||||||
|
self.main_wallet,
|
||||||
|
basic_acl="public-read-write",
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
file_path = generate_file(object_size)
|
||||||
|
oid = put_object_to_random_node(
|
||||||
|
self.main_wallet, file_path, cid, shell=self.shell, cluster=self.cluster
|
||||||
|
)
|
||||||
|
objects = [oid]
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
self.storagegroup_operations_by_system_ro_container(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test Storage Group in Read-Only Container")
|
||||||
|
def test_storagegroup_basic_ro_container(self, object_size, max_object_size):
|
||||||
|
cid = create_container(
|
||||||
|
self.main_wallet,
|
||||||
|
basic_acl="public-read",
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
file_path = generate_file(object_size)
|
||||||
|
oid = put_object_to_random_node(
|
||||||
|
self.main_wallet, file_path, cid, shell=self.shell, cluster=self.cluster
|
||||||
|
)
|
||||||
|
objects = [oid]
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
self.storagegroup_operations_by_other_ro_container(
|
||||||
|
owner_wallet=self.main_wallet,
|
||||||
|
other_wallet=self.other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
self.storagegroup_operations_by_system_ro_container(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test Storage Group with Bearer Allow")
|
||||||
|
def test_storagegroup_bearer_allow(self, object_size, max_object_size):
|
||||||
|
cid = create_container(
|
||||||
|
self.main_wallet,
|
||||||
|
basic_acl="eacl-public-read-write",
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
file_path = generate_file(object_size)
|
||||||
|
oid = put_object_to_random_node(
|
||||||
|
self.main_wallet, file_path, cid, shell=self.shell, cluster=self.cluster
|
||||||
|
)
|
||||||
|
objects = [oid]
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
self.shell, self.cluster.default_rpc_endpoint, self.main_wallet, cid, objects
|
||||||
|
)
|
||||||
|
eacl_deny = [
|
||||||
|
EACLRule(access=EACLAccess.DENY, role=role, operation=op)
|
||||||
|
for op in EACLOperation
|
||||||
|
for role in EACLRole
|
||||||
|
]
|
||||||
|
set_eacl(
|
||||||
|
self.main_wallet,
|
||||||
|
cid,
|
||||||
|
create_eacl(cid, eacl_deny, shell=self.shell),
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
self.expect_failure_for_storagegroup_operations(
|
||||||
|
self.main_wallet, cid, objects, storage_group
|
||||||
|
)
|
||||||
|
bearer_file = form_bearertoken_file(
|
||||||
|
self.main_wallet,
|
||||||
|
cid,
|
||||||
|
[
|
||||||
|
EACLRule(operation=op, access=EACLAccess.ALLOW, role=EACLRole.USER)
|
||||||
|
for op in EACLOperation
|
||||||
|
],
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
self.expect_success_for_storagegroup_operations(
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
obj_list=objects,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
bearer=bearer_file,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test to check Storage Group lifetime")
|
||||||
|
def test_storagegroup_lifetime(self, object_size):
|
||||||
|
cid = create_container(
|
||||||
|
self.main_wallet, shell=self.shell, endpoint=self.cluster.default_rpc_endpoint
|
||||||
|
)
|
||||||
|
file_path = generate_file(object_size)
|
||||||
|
oid = put_object_to_random_node(
|
||||||
|
self.main_wallet, file_path, cid, shell=self.shell, cluster=self.cluster
|
||||||
|
)
|
||||||
|
objects = [oid]
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
self.shell,
|
||||||
|
self.cluster.default_rpc_endpoint,
|
||||||
|
self.main_wallet,
|
||||||
|
cid,
|
||||||
|
objects,
|
||||||
|
lifetime=1,
|
||||||
|
)
|
||||||
|
with allure.step("Tick two epochs"):
|
||||||
|
for _ in range(2):
|
||||||
|
self.tick_epoch()
|
||||||
|
self.wait_for_epochs_align()
|
||||||
|
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
||||||
|
get_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=self.main_wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Run Storage Group Operations And Expect Success")
|
||||||
|
def expect_success_for_storagegroup_operations(
|
||||||
|
self,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
obj_list: list,
|
||||||
|
object_size: int,
|
||||||
|
max_object_size: int,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
This func verifies if the Object's owner is allowed to
|
||||||
|
Put, List, Get and Delete the Storage Group which contains
|
||||||
|
the Object.
|
||||||
|
"""
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
self.shell, self.cluster.default_rpc_endpoint, wallet, cid, obj_list, bearer
|
||||||
|
)
|
||||||
|
verify_list_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
bearer=bearer,
|
||||||
|
)
|
||||||
|
verify_get_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
cluster=self.cluster,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
obj_list=obj_list,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
bearer=bearer,
|
||||||
|
)
|
||||||
|
delete_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
bearer=bearer,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Run Storage Group Operations And Expect Failure")
|
||||||
|
def expect_failure_for_storagegroup_operations(
|
||||||
|
self, wallet: str, cid: str, obj_list: list, gid: str
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
This func verifies if the Object's owner isn't allowed to
|
||||||
|
Put, List, Get and Delete the Storage Group which contains
|
||||||
|
the Object.
|
||||||
|
"""
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
put_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
objects=obj_list,
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
list_storagegroup(
|
||||||
|
shell=self.shell, endpoint=self.cluster.default_rpc_endpoint, wallet=wallet, cid=cid
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
get_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=gid,
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
delete_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=gid,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Run Storage Group Operations On Other's Behalf In RO Container")
|
||||||
|
def storagegroup_operations_by_other_ro_container(
|
||||||
|
self,
|
||||||
|
owner_wallet: str,
|
||||||
|
other_wallet: str,
|
||||||
|
cid: str,
|
||||||
|
obj_list: list,
|
||||||
|
object_size: int,
|
||||||
|
max_object_size: int,
|
||||||
|
):
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
self.shell, self.cluster.default_rpc_endpoint, owner_wallet, cid, obj_list
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
put_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
objects=obj_list,
|
||||||
|
)
|
||||||
|
verify_list_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
)
|
||||||
|
verify_get_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
cluster=self.cluster,
|
||||||
|
wallet=other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
obj_list=obj_list,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
delete_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=other_wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.step("Run Storage Group Operations On Systems's Behalf In RO Container")
|
||||||
|
def storagegroup_operations_by_system_ro_container(
|
||||||
|
self,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
obj_list: list,
|
||||||
|
object_size: int,
|
||||||
|
max_object_size: int,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
In this func we create a Storage Group on Inner Ring's key behalf
|
||||||
|
and include an Object created on behalf of some user. We expect
|
||||||
|
that System key is granted to make all operations except PUT and DELETE.
|
||||||
|
"""
|
||||||
|
ir_node = self.cluster.ir_nodes[0]
|
||||||
|
ir_wallet_path = ir_node.get_wallet_path()
|
||||||
|
ir_wallet_password = ir_node.get_wallet_password()
|
||||||
|
ir_wallet_config = ir_node.get_wallet_config_path()
|
||||||
|
|
||||||
|
if not FREE_STORAGE:
|
||||||
|
main_chain = self.cluster.main_chain_nodes[0]
|
||||||
|
deposit = 30
|
||||||
|
transfer_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit + 1,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_to_path=ir_wallet_path,
|
||||||
|
wallet_to_password=ir_wallet_password,
|
||||||
|
)
|
||||||
|
deposit_gas(
|
||||||
|
shell=self.shell,
|
||||||
|
amount=deposit,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_from_path=ir_wallet_path,
|
||||||
|
wallet_from_password=ir_wallet_password,
|
||||||
|
)
|
||||||
|
storage_group = put_storagegroup(
|
||||||
|
self.shell, self.cluster.default_rpc_endpoint, wallet, cid, obj_list
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
put_storagegroup(
|
||||||
|
self.shell,
|
||||||
|
self.cluster.default_rpc_endpoint,
|
||||||
|
ir_wallet_path,
|
||||||
|
cid,
|
||||||
|
obj_list,
|
||||||
|
wallet_config=ir_wallet_config,
|
||||||
|
)
|
||||||
|
verify_list_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=ir_wallet_path,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
wallet_config=ir_wallet_config,
|
||||||
|
)
|
||||||
|
verify_get_storage_group(
|
||||||
|
shell=self.shell,
|
||||||
|
cluster=self.cluster,
|
||||||
|
wallet=ir_wallet_path,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
obj_list=obj_list,
|
||||||
|
object_size=object_size,
|
||||||
|
max_object_size=max_object_size,
|
||||||
|
wallet_config=ir_wallet_config,
|
||||||
|
)
|
||||||
|
with pytest.raises(Exception, match=OBJECT_ACCESS_DENIED):
|
||||||
|
delete_storagegroup(
|
||||||
|
shell=self.shell,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=ir_wallet_path,
|
||||||
|
cid=cid,
|
||||||
|
gid=storage_group,
|
||||||
|
wallet_config=ir_wallet_config,
|
||||||
|
)
|
|
@ -1,18 +1,15 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PRIVATE_ACL_F, PUBLIC_ACL_F, READONLY_ACL_F
|
from cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.shell import Shell
|
from python_keywords.acl import EACLRole
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
from python_keywords.container_access import (
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLRole
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
|
|
||||||
from pytest_tests.helpers.container_access import (
|
|
||||||
check_full_access_to_container,
|
check_full_access_to_container,
|
||||||
check_no_access_to_container,
|
check_no_access_to_container,
|
||||||
check_read_only_container,
|
check_read_only_container,
|
||||||
)
|
)
|
||||||
from pytest_tests.testsuites.acl.conftest import Wallets
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
|
from wellknown_acl import PRIVATE_ACL_F, PUBLIC_ACL_F, READONLY_ACL_F
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -37,7 +34,7 @@ class TestACLBasic(ClusterTestBase):
|
||||||
# delete_container(user_wallet.wallet_path, cid_public)
|
# delete_container(user_wallet.wallet_path, cid_public)
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def private_container(self, wallets: Wallets):
|
def private_container(self, wallets):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
with allure.step("Create private container"):
|
with allure.step("Create private container"):
|
||||||
cid_private = create_container(
|
cid_private = create_container(
|
||||||
|
@ -53,7 +50,7 @@ class TestACLBasic(ClusterTestBase):
|
||||||
# delete_container(user_wallet.wallet_path, cid_private)
|
# delete_container(user_wallet.wallet_path, cid_private)
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def read_only_container(self, wallets: Wallets):
|
def read_only_container(self, wallets):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
with allure.step("Create public readonly container"):
|
with allure.step("Create public readonly container"):
|
||||||
cid_read_only = create_container(
|
cid_read_only = create_container(
|
||||||
|
@ -69,7 +66,7 @@ class TestACLBasic(ClusterTestBase):
|
||||||
# delete_container(user_wallet.wallet_path, cid_read_only)
|
# delete_container(user_wallet.wallet_path, cid_read_only)
|
||||||
|
|
||||||
@allure.title("Test basic ACL on public container")
|
@allure.title("Test basic ACL on public container")
|
||||||
def test_basic_acl_public(self, wallets: Wallets, public_container: str, file_path: str):
|
def test_basic_acl_public(self, wallets, public_container, file_path):
|
||||||
"""
|
"""
|
||||||
Test basic ACL set during public container creation.
|
Test basic ACL set during public container creation.
|
||||||
"""
|
"""
|
||||||
|
@ -115,7 +112,7 @@ class TestACLBasic(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test basic ACL on private container")
|
@allure.title("Test basic ACL on private container")
|
||||||
def test_basic_acl_private(self, wallets: Wallets, private_container: str, file_path: str):
|
def test_basic_acl_private(self, wallets, private_container, file_path):
|
||||||
"""
|
"""
|
||||||
Test basic ACL set during private container creation.
|
Test basic ACL set during private container creation.
|
||||||
"""
|
"""
|
||||||
|
@ -149,9 +146,7 @@ class TestACLBasic(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test basic ACL on readonly container")
|
@allure.title("Test basic ACL on readonly container")
|
||||||
def test_basic_acl_readonly(
|
def test_basic_acl_readonly(self, wallets, client_shell, read_only_container, file_path):
|
||||||
self, wallets: Wallets, client_shell: Shell, read_only_container: str, file_path: str
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test basic ACL Operations for Read-Only Container.
|
Test basic ACL Operations for Read-Only Container.
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -1,20 +1,21 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.steps.acl import (
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from python_keywords.acl import (
|
||||||
|
EACLAccess,
|
||||||
|
EACLOperation,
|
||||||
|
EACLRole,
|
||||||
|
EACLRule,
|
||||||
create_eacl,
|
create_eacl,
|
||||||
form_bearertoken_file,
|
form_bearertoken_file,
|
||||||
set_eacl,
|
set_eacl,
|
||||||
wait_for_cache_expired,
|
wait_for_cache_expired,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLAccess, EACLOperation, EACLRole, EACLRule
|
from python_keywords.container_access import (
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
|
|
||||||
from pytest_tests.helpers.container_access import (
|
|
||||||
check_custom_access_to_container,
|
check_custom_access_to_container,
|
||||||
check_full_access_to_container,
|
check_full_access_to_container,
|
||||||
check_no_access_to_container,
|
check_no_access_to_container,
|
||||||
)
|
)
|
||||||
from pytest_tests.testsuites.acl.conftest import Wallets
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -22,15 +23,8 @@ from pytest_tests.testsuites.acl.conftest import Wallets
|
||||||
@pytest.mark.acl_bearer
|
@pytest.mark.acl_bearer
|
||||||
class TestACLBearer(ClusterTestBase):
|
class TestACLBearer(ClusterTestBase):
|
||||||
@pytest.mark.parametrize("role", [EACLRole.USER, EACLRole.OTHERS])
|
@pytest.mark.parametrize("role", [EACLRole.USER, EACLRole.OTHERS])
|
||||||
def test_bearer_token_operations(
|
def test_bearer_token_operations(self, wallets, eacl_container_with_objects, role):
|
||||||
self,
|
allure.dynamic.title(f"Testcase to validate NeoFS operations with {role.value} BearerToken")
|
||||||
wallets: Wallets,
|
|
||||||
eacl_container_with_objects: tuple[str, list[str], str],
|
|
||||||
role: EACLRole,
|
|
||||||
):
|
|
||||||
allure.dynamic.title(
|
|
||||||
f"Testcase to validate FrostFS operations with {role.value} BearerToken"
|
|
||||||
)
|
|
||||||
cid, objects_oids, file_path = eacl_container_with_objects
|
cid, objects_oids, file_path = eacl_container_with_objects
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
deny_wallet = wallets.get_wallet(role)
|
deny_wallet = wallets.get_wallet(role)
|
||||||
|
|
|
@ -1,19 +1,25 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.steps.acl import create_eacl, set_eacl, wait_for_cache_expired
|
from failover_utils import wait_object_replication
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
from python_keywords.acl import (
|
||||||
from frostfs_testlib.steps.node_management import drop_object
|
EACLAccess,
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLAccess, EACLOperation, EACLRole, EACLRule
|
EACLOperation,
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
EACLRole,
|
||||||
from frostfs_testlib.utils.failover_utils import wait_object_replication
|
EACLRule,
|
||||||
|
create_eacl,
|
||||||
from pytest_tests.helpers.container_access import (
|
set_eacl,
|
||||||
|
wait_for_cache_expired,
|
||||||
|
)
|
||||||
|
from python_keywords.container import create_container
|
||||||
|
from python_keywords.container_access import (
|
||||||
check_full_access_to_container,
|
check_full_access_to_container,
|
||||||
check_no_access_to_container,
|
check_no_access_to_container,
|
||||||
)
|
)
|
||||||
from pytest_tests.helpers.object_access import (
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
|
from python_keywords.node_management import drop_object
|
||||||
|
from python_keywords.object_access import (
|
||||||
can_delete_object,
|
can_delete_object,
|
||||||
can_get_head_object,
|
can_get_head_object,
|
||||||
can_get_object,
|
can_get_object,
|
||||||
|
@ -22,7 +28,7 @@ from pytest_tests.helpers.object_access import (
|
||||||
can_put_object,
|
can_put_object,
|
||||||
can_search_object,
|
can_search_object,
|
||||||
)
|
)
|
||||||
from pytest_tests.testsuites.acl.conftest import Wallets
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -30,7 +36,7 @@ from pytest_tests.testsuites.acl.conftest import Wallets
|
||||||
@pytest.mark.acl_extended
|
@pytest.mark.acl_extended
|
||||||
class TestEACLContainer(ClusterTestBase):
|
class TestEACLContainer(ClusterTestBase):
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def eacl_full_placement_container_with_object(self, wallets: Wallets, file_path: str) -> str:
|
def eacl_full_placement_container_with_object(self, wallets, file_path) -> str:
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
storage_nodes = self.cluster.storage_nodes
|
storage_nodes = self.cluster.storage_nodes
|
||||||
node_count = len(storage_nodes)
|
node_count = len(storage_nodes)
|
||||||
|
@ -60,10 +66,7 @@ class TestEACLContainer(ClusterTestBase):
|
||||||
|
|
||||||
@pytest.mark.parametrize("deny_role", [EACLRole.USER, EACLRole.OTHERS])
|
@pytest.mark.parametrize("deny_role", [EACLRole.USER, EACLRole.OTHERS])
|
||||||
def test_extended_acl_deny_all_operations(
|
def test_extended_acl_deny_all_operations(
|
||||||
self,
|
self, wallets, eacl_container_with_objects, deny_role
|
||||||
wallets: Wallets,
|
|
||||||
eacl_container_with_objects: tuple[str, list[str], str],
|
|
||||||
deny_role: EACLRole,
|
|
||||||
):
|
):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
||||||
|
@ -71,7 +74,7 @@ class TestEACLContainer(ClusterTestBase):
|
||||||
not_deny_role_wallet = user_wallet if deny_role == EACLRole.OTHERS else other_wallet
|
not_deny_role_wallet = user_wallet if deny_role == EACLRole.OTHERS else other_wallet
|
||||||
deny_role_str = "all others" if deny_role == EACLRole.OTHERS else "user"
|
deny_role_str = "all others" if deny_role == EACLRole.OTHERS else "user"
|
||||||
not_deny_role_str = "user" if deny_role == EACLRole.OTHERS else "all others"
|
not_deny_role_str = "user" if deny_role == EACLRole.OTHERS else "all others"
|
||||||
allure.dynamic.title(f"Testcase to deny FrostFS operations for {deny_role_str}.")
|
allure.dynamic.title(f"Testcase to deny NeoFS operations for {deny_role_str}.")
|
||||||
cid, object_oids, file_path = eacl_container_with_objects
|
cid, object_oids, file_path = eacl_container_with_objects
|
||||||
|
|
||||||
with allure.step(f"Deny all operations for {deny_role_str} via eACL"):
|
with allure.step(f"Deny all operations for {deny_role_str} via eACL"):
|
||||||
|
@ -145,9 +148,9 @@ class TestEACLContainer(ClusterTestBase):
|
||||||
cluster=self.cluster,
|
cluster=self.cluster,
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Testcase to allow FrostFS operations for only one other pubkey.")
|
@allure.title("Testcase to allow NeoFS operations for only one other pubkey.")
|
||||||
def test_extended_acl_deny_all_operations_exclude_pubkey(
|
def test_extended_acl_deny_all_operations_exclude_pubkey(
|
||||||
self, wallets: Wallets, eacl_container_with_objects: tuple[str, list[str], str]
|
self, wallets, eacl_container_with_objects
|
||||||
):
|
):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
other_wallet, other_wallet_allow = wallets.get_wallets_list(EACLRole.OTHERS)[0:2]
|
other_wallet, other_wallet_allow = wallets.get_wallets_list(EACLRole.OTHERS)[0:2]
|
||||||
|
@ -206,11 +209,11 @@ class TestEACLContainer(ClusterTestBase):
|
||||||
cluster=self.cluster,
|
cluster=self.cluster,
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Testcase to validate FrostFS replication with eACL deny rules.")
|
@allure.title("Testcase to validate NeoFS replication with eACL deny rules.")
|
||||||
def test_extended_acl_deny_replication(
|
def test_extended_acl_deny_replication(
|
||||||
self,
|
self,
|
||||||
wallets: Wallets,
|
wallets,
|
||||||
eacl_full_placement_container_with_object: tuple[str, list[str], str],
|
eacl_full_placement_container_with_object,
|
||||||
):
|
):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
cid, oid, file_path = eacl_full_placement_container_with_object
|
cid, oid, file_path = eacl_full_placement_container_with_object
|
||||||
|
@ -248,10 +251,8 @@ class TestEACLContainer(ClusterTestBase):
|
||||||
storage_nodes,
|
storage_nodes,
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Testcase to validate FrostFS system operations with extended ACL")
|
@allure.title("Testcase to validate NeoFS system operations with extended ACL")
|
||||||
def test_extended_actions_system(
|
def test_extended_actions_system(self, wallets, eacl_container_with_objects):
|
||||||
self, wallets: Wallets, eacl_container_with_objects: tuple[str, list[str], str]
|
|
||||||
):
|
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
ir_wallet, storage_wallet = wallets.get_wallets_list(role=EACLRole.SYSTEM)[:2]
|
ir_wallet, storage_wallet = wallets.get_wallets_list(role=EACLRole.SYSTEM)[:2]
|
||||||
|
|
||||||
|
|
|
@ -1,15 +1,7 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.steps.acl import (
|
from python_keywords.acl import (
|
||||||
create_eacl,
|
|
||||||
form_bearertoken_file,
|
|
||||||
set_eacl,
|
|
||||||
wait_for_cache_expired,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.cli.container import create_container, delete_container
|
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
|
||||||
from frostfs_testlib.storage.dataclasses.acl import (
|
|
||||||
EACLAccess,
|
EACLAccess,
|
||||||
EACLFilter,
|
EACLFilter,
|
||||||
EACLFilters,
|
EACLFilters,
|
||||||
|
@ -18,15 +10,19 @@ from frostfs_testlib.storage.dataclasses.acl import (
|
||||||
EACLOperation,
|
EACLOperation,
|
||||||
EACLRole,
|
EACLRole,
|
||||||
EACLRule,
|
EACLRule,
|
||||||
|
create_eacl,
|
||||||
|
form_bearertoken_file,
|
||||||
|
set_eacl,
|
||||||
|
wait_for_cache_expired,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.container import create_container, delete_container
|
||||||
|
from python_keywords.container_access import (
|
||||||
from pytest_tests.helpers.container_access import (
|
|
||||||
check_full_access_to_container,
|
check_full_access_to_container,
|
||||||
check_no_access_to_container,
|
check_no_access_to_container,
|
||||||
)
|
)
|
||||||
from pytest_tests.helpers.object_access import can_get_head_object, can_get_object, can_put_object
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
from pytest_tests.testsuites.acl.conftest import Wallets
|
from python_keywords.object_access import can_get_head_object, can_get_object, can_put_object
|
||||||
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -72,7 +68,7 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
]
|
]
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
@pytest.fixture(scope="function")
|
||||||
def eacl_container_with_objects(self, wallets: Wallets, file_path: str):
|
def eacl_container_with_objects(self, wallets, file_path):
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
with allure.step("Create eACL public container"):
|
with allure.step("Create eACL public container"):
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
|
@ -131,13 +127,8 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
||||||
)
|
)
|
||||||
def test_extended_acl_filters_request(
|
def test_extended_acl_filters_request(self, wallets, eacl_container_with_objects, match_type):
|
||||||
self,
|
allure.dynamic.title(f"Validate NeoFS operations with request filter: {match_type.name}")
|
||||||
wallets: Wallets,
|
|
||||||
eacl_container_with_objects: tuple[str, list[str], str],
|
|
||||||
match_type: EACLMatchType,
|
|
||||||
):
|
|
||||||
allure.dynamic.title(f"Validate FrostFS operations with request filter: {match_type.name}")
|
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
||||||
(
|
(
|
||||||
|
@ -249,13 +240,10 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
||||||
)
|
)
|
||||||
def test_extended_acl_deny_filters_object(
|
def test_extended_acl_deny_filters_object(
|
||||||
self,
|
self, wallets, eacl_container_with_objects, match_type
|
||||||
wallets: Wallets,
|
|
||||||
eacl_container_with_objects: tuple[str, list[str], str],
|
|
||||||
match_type: EACLMatchType,
|
|
||||||
):
|
):
|
||||||
allure.dynamic.title(
|
allure.dynamic.title(
|
||||||
f"Validate FrostFS operations with deny user headers filter: {match_type.name}"
|
f"Validate NeoFS operations with deny user headers filter: {match_type.name}"
|
||||||
)
|
)
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
other_wallet = wallets.get_wallet(EACLRole.OTHERS)
|
||||||
|
@ -434,13 +422,10 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
"match_type", [EACLMatchType.STRING_EQUAL, EACLMatchType.STRING_NOT_EQUAL]
|
||||||
)
|
)
|
||||||
def test_extended_acl_allow_filters_object(
|
def test_extended_acl_allow_filters_object(
|
||||||
self,
|
self, wallets, eacl_container_with_objects, match_type
|
||||||
wallets: Wallets,
|
|
||||||
eacl_container_with_objects: tuple[str, list[str], str],
|
|
||||||
match_type: EACLMatchType,
|
|
||||||
):
|
):
|
||||||
allure.dynamic.title(
|
allure.dynamic.title(
|
||||||
"Testcase to validate FrostFS operation with allow eACL user headers filters:"
|
"Testcase to validate NeoFS operation with allow eACL user headers filters:"
|
||||||
f"{match_type.name}"
|
f"{match_type.name}"
|
||||||
)
|
)
|
||||||
user_wallet = wallets.get_wallet()
|
user_wallet = wallets.get_wallet()
|
||||||
|
@ -490,7 +475,7 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
allow_attribute = self.OTHER_ATTRIBUTE
|
allow_attribute = self.OTHER_ATTRIBUTE
|
||||||
deny_attribute = self.ATTRIBUTE
|
deny_attribute = self.ATTRIBUTE
|
||||||
|
|
||||||
with allure.step("Check other cannot get and put objects without attributes"):
|
with allure.step(f"Check other cannot get and put objects without attributes"):
|
||||||
oid = objects_without_header.pop()
|
oid = objects_without_header.pop()
|
||||||
with pytest.raises(AssertionError):
|
with pytest.raises(AssertionError):
|
||||||
assert can_get_head_object(
|
assert can_get_head_object(
|
||||||
|
@ -557,7 +542,7 @@ class TestEACLFilters(ClusterTestBase):
|
||||||
bearer=bearer_other,
|
bearer=bearer_other,
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step("Check other can get objects with attributes matching the filter"):
|
with allure.step(f"Check other can get objects with attributes matching the filter"):
|
||||||
oid = allow_objects.pop()
|
oid = allow_objects.pop()
|
||||||
assert can_get_head_object(
|
assert can_get_head_object(
|
||||||
other_wallet.wallet_path,
|
other_wallet.wallet_path,
|
||||||
|
|
|
@ -1,76 +1,62 @@
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
|
import re
|
||||||
import shutil
|
import shutil
|
||||||
|
import uuid
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from typing import Optional
|
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
import yaml
|
import yaml
|
||||||
from frostfs_testlib.hosting import Hosting
|
from binary_version_helper import get_local_binaries_versions, get_remote_binaries_versions
|
||||||
from frostfs_testlib.reporter import AllureHandler, get_reporter
|
from cluster import Cluster
|
||||||
from frostfs_testlib.resources.common import (
|
from common import (
|
||||||
ASSETS_DIR,
|
ASSETS_DIR,
|
||||||
COMPLEX_OBJECT_CHUNKS_COUNT,
|
COMPLEX_OBJECT_CHUNKS_COUNT,
|
||||||
COMPLEX_OBJECT_TAIL_SIZE,
|
COMPLEX_OBJECT_TAIL_SIZE,
|
||||||
DEFAULT_WALLET_PASS,
|
FREE_STORAGE,
|
||||||
|
HOSTING_CONFIG_FILE,
|
||||||
SIMPLE_OBJECT_SIZE,
|
SIMPLE_OBJECT_SIZE,
|
||||||
|
STORAGE_NODE_SERVICE_NAME_REGEX,
|
||||||
|
WALLET_PASS,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.s3.interfaces import S3ClientWrapper, VersioningStatus
|
from env_properties import save_env_properties
|
||||||
from frostfs_testlib.shell import LocalShell, Shell
|
from k6 import LoadParams
|
||||||
from frostfs_testlib.steps.cli.container import list_containers
|
from load import get_services_endpoints, prepare_k6_instances
|
||||||
from frostfs_testlib.steps.cli.object import get_netmap_netinfo
|
from load_params import (
|
||||||
from frostfs_testlib.steps.node_management import storage_node_healthcheck
|
BACKGROUND_LOAD_MAX_TIME,
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
BACKGROUND_OBJ_SIZE,
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
BACKGROUND_READERS_COUNT,
|
||||||
from frostfs_testlib.storage.controllers.cluster_state_controller import ClusterStateController
|
BACKGROUND_WRITERS_COUNT,
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletFactory, WalletInfo
|
LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
LOAD_NODE_SSH_USER,
|
||||||
from frostfs_testlib.utils import env_utils, version_utils
|
LOAD_NODES,
|
||||||
|
)
|
||||||
|
from neofs_testlib.hosting import Hosting
|
||||||
|
from neofs_testlib.reporter import AllureHandler, get_reporter
|
||||||
|
from neofs_testlib.shell import LocalShell, Shell
|
||||||
|
from neofs_testlib.utils.wallet import init_wallet
|
||||||
|
from payment_neogo import deposit_gas, transfer_gas
|
||||||
|
from python_keywords.neofs_verbs import get_netmap_netinfo
|
||||||
|
from python_keywords.node_management import storage_node_healthcheck
|
||||||
|
|
||||||
from pytest_tests.resources.common import HOSTING_CONFIG_FILE, TEST_CYCLES_COUNT
|
from helpers.wallet import WalletFactory
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
# Add logs check test even if it's not fit to mark selectors
|
def pytest_collection_modifyitems(items):
|
||||||
def pytest_configure(config: pytest.Config):
|
# Make network tests last based on @pytest.mark.node_mgmt
|
||||||
markers = config.option.markexpr
|
|
||||||
if markers != "":
|
|
||||||
config.option.markexpr = f"logs_after_session or ({markers})"
|
|
||||||
|
|
||||||
|
|
||||||
# pytest hook. Do not rename
|
|
||||||
def pytest_collection_modifyitems(items: list[pytest.Item]):
|
|
||||||
# Make network tests last based on @pytest.mark.node_mgmt and logs_test to be latest
|
|
||||||
def priority(item: pytest.Item) -> int:
|
def priority(item: pytest.Item) -> int:
|
||||||
is_node_mgmt_test = 1 if item.get_closest_marker("node_mgmt") else 0
|
is_node_mgmt_test = item.get_closest_marker("node_mgmt")
|
||||||
is_logs_check_test = 100 if item.get_closest_marker("logs_after_session") else 0
|
return 0 if not is_node_mgmt_test else 1
|
||||||
return is_node_mgmt_test + is_logs_check_test
|
|
||||||
|
|
||||||
items.sort(key=lambda item: priority(item))
|
items.sort(key=lambda item: priority(item))
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
|
||||||
if (
|
|
||||||
TEST_CYCLES_COUNT <= 1
|
|
||||||
or metafunc.definition.get_closest_marker("logs_after_session")
|
|
||||||
or metafunc.definition.get_closest_marker("no_cycles")
|
|
||||||
):
|
|
||||||
return
|
|
||||||
|
|
||||||
metafunc.fixturenames.append("cycle")
|
|
||||||
metafunc.parametrize(
|
|
||||||
"cycle",
|
|
||||||
range(1, TEST_CYCLES_COUNT + 1),
|
|
||||||
ids=[f"cycle {cycle}" for cycle in range(1, TEST_CYCLES_COUNT + 1)],
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
def configure_testlib():
|
def configure_testlib():
|
||||||
get_reporter().register_handler(AllureHandler())
|
get_reporter().register_handler(AllureHandler())
|
||||||
logging.getLogger("paramiko").setLevel(logging.INFO)
|
|
||||||
yield
|
yield
|
||||||
|
|
||||||
|
|
||||||
|
@ -129,105 +115,25 @@ def wallet_factory(temp_directory: str, client_shell: Shell, cluster: Cluster) -
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
def cluster(temp_directory: str, hosting: Hosting, client_shell: Shell) -> Cluster:
|
def cluster(temp_directory: str, hosting: Hosting) -> Cluster:
|
||||||
cluster = Cluster(hosting)
|
cluster = Cluster(hosting)
|
||||||
if cluster.is_local_devenv():
|
if cluster.is_local_devevn():
|
||||||
cluster.create_wallet_configs(hosting)
|
cluster.create_wallet_configs(hosting)
|
||||||
|
|
||||||
ClusterTestBase.shell = client_shell
|
|
||||||
ClusterTestBase.cluster = cluster
|
|
||||||
|
|
||||||
yield cluster
|
yield cluster
|
||||||
|
|
||||||
|
|
||||||
@allure.step("[Class]: Provide S3 policy")
|
|
||||||
@pytest.fixture(scope="class")
|
|
||||||
def s3_policy(request: pytest.FixtureRequest):
|
|
||||||
policy = None
|
|
||||||
if "param" in request.__dict__:
|
|
||||||
policy = request.param
|
|
||||||
|
|
||||||
return policy
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="session")
|
|
||||||
def cluster_state_controller(client_shell: Shell, cluster: Cluster) -> ClusterStateController:
|
|
||||||
controller = ClusterStateController(client_shell, cluster)
|
|
||||||
yield controller
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("[Class]: Create S3 client")
|
|
||||||
@pytest.fixture(scope="class")
|
|
||||||
def s3_client(
|
|
||||||
default_wallet: str,
|
|
||||||
client_shell: Shell,
|
|
||||||
s3_policy: Optional[str],
|
|
||||||
cluster: Cluster,
|
|
||||||
request: pytest.FixtureRequest,
|
|
||||||
) -> S3ClientWrapper:
|
|
||||||
wallet = WalletInfo(path=default_wallet, password=DEFAULT_WALLET_PASS)
|
|
||||||
|
|
||||||
(cid, access_key_id, secret_access_key) = s3_helper.init_s3_credentials(
|
|
||||||
wallet,
|
|
||||||
client_shell,
|
|
||||||
cluster,
|
|
||||||
s3gates=[cluster_node.s3_gate for cluster_node in cluster.cluster_nodes],
|
|
||||||
policy=s3_policy,
|
|
||||||
)
|
|
||||||
containers_list = list_containers(
|
|
||||||
wallet.path, shell=client_shell, endpoint=cluster.default_rpc_endpoint
|
|
||||||
)
|
|
||||||
assert cid in containers_list, f"Expected cid {cid} in {containers_list}"
|
|
||||||
|
|
||||||
s3_client_cls = request.param
|
|
||||||
client = s3_client_cls(access_key_id, secret_access_key, cluster.default_s3_gate_endpoint)
|
|
||||||
yield client
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Create/delete bucket")
|
|
||||||
@pytest.fixture
|
|
||||||
def bucket(s3_client: S3ClientWrapper, request: pytest.FixtureRequest):
|
|
||||||
bucket_name = s3_client.create_bucket()
|
|
||||||
|
|
||||||
versioning_status: Optional[VersioningStatus] = None
|
|
||||||
if "param" in request.__dict__:
|
|
||||||
versioning_status = request.param
|
|
||||||
|
|
||||||
if versioning_status:
|
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket_name, versioning_status)
|
|
||||||
|
|
||||||
yield bucket_name
|
|
||||||
s3_helper.delete_bucket_with_objects(s3_client, bucket_name)
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Create two buckets")
|
|
||||||
@pytest.fixture
|
|
||||||
def two_buckets(s3_client: S3ClientWrapper):
|
|
||||||
bucket_1 = s3_client.create_bucket()
|
|
||||||
bucket_2 = s3_client.create_bucket()
|
|
||||||
yield bucket_1, bucket_2
|
|
||||||
for bucket_name in [bucket_1, bucket_2]:
|
|
||||||
s3_helper.delete_bucket_with_objects(s3_client, bucket_name)
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Check binary versions")
|
|
||||||
@pytest.fixture(scope="session", autouse=True)
|
@pytest.fixture(scope="session", autouse=True)
|
||||||
def check_binary_versions(hosting: Hosting, client_shell: Shell, request: pytest.FixtureRequest):
|
@allure.title("Check binary versions")
|
||||||
local_versions = version_utils.get_local_binaries_versions(client_shell)
|
def check_binary_versions(request, hosting: Hosting, client_shell: Shell):
|
||||||
remote_versions = version_utils.get_remote_binaries_versions(hosting)
|
local_versions = get_local_binaries_versions(client_shell)
|
||||||
|
remote_versions = get_remote_binaries_versions(hosting)
|
||||||
|
|
||||||
all_versions = {**local_versions, **remote_versions}
|
all_versions = {**local_versions, **remote_versions}
|
||||||
|
save_env_properties(request.config, all_versions)
|
||||||
environment_dir = request.config.getoption("--alluredir")
|
|
||||||
if not environment_dir:
|
|
||||||
return None
|
|
||||||
|
|
||||||
file_path = f"{environment_dir}/environment.properties"
|
|
||||||
env_utils.save_env_properties(file_path, all_versions)
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Prepare tmp directory")
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
|
@allure.title("Prepare tmp directory")
|
||||||
def temp_directory():
|
def temp_directory():
|
||||||
with allure.step("Prepare tmp directory"):
|
with allure.step("Prepare tmp directory"):
|
||||||
full_path = os.path.join(os.getcwd(), ASSETS_DIR)
|
full_path = os.path.join(os.getcwd(), ASSETS_DIR)
|
||||||
|
@ -240,16 +146,23 @@ def temp_directory():
|
||||||
shutil.rmtree(full_path)
|
shutil.rmtree(full_path)
|
||||||
|
|
||||||
|
|
||||||
@allure.step("[Autouse/Session] Test session start time")
|
|
||||||
@pytest.fixture(scope="session", autouse=True)
|
@pytest.fixture(scope="session", autouse=True)
|
||||||
def session_start_time():
|
@allure.title("Collect logs")
|
||||||
|
def collect_logs(temp_directory, hosting: Hosting):
|
||||||
start_time = datetime.utcnow()
|
start_time = datetime.utcnow()
|
||||||
return start_time
|
yield
|
||||||
|
end_time = datetime.utcnow()
|
||||||
|
|
||||||
|
# Dump logs to temp directory (because they might be too large to keep in RAM)
|
||||||
|
logs_dir = os.path.join(temp_directory, "logs")
|
||||||
|
dump_logs(hosting, logs_dir, start_time, end_time)
|
||||||
|
attach_logs(logs_dir)
|
||||||
|
check_logs(logs_dir)
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Run health check for all storage nodes")
|
|
||||||
@pytest.fixture(scope="session", autouse=True)
|
@pytest.fixture(scope="session", autouse=True)
|
||||||
def run_health_check(session_start_time, cluster: Cluster):
|
@allure.title("Run health check for all storage nodes")
|
||||||
|
def run_health_check(collect_logs, cluster: Cluster):
|
||||||
failed_nodes = []
|
failed_nodes = []
|
||||||
for node in cluster.storage_nodes:
|
for node in cluster.storage_nodes:
|
||||||
health_check = storage_node_healthcheck(node)
|
health_check = storage_node_healthcheck(node)
|
||||||
|
@ -260,9 +173,135 @@ def run_health_check(session_start_time, cluster: Cluster):
|
||||||
raise AssertionError(f"Nodes {failed_nodes} are not healthy")
|
raise AssertionError(f"Nodes {failed_nodes} are not healthy")
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Prepare wallet and deposit")
|
|
||||||
@pytest.fixture(scope="session")
|
@pytest.fixture(scope="session")
|
||||||
def default_wallet(wallet_factory: WalletFactory) -> str:
|
def background_grpc_load(client_shell: Shell, hosting: Hosting):
|
||||||
wallet = wallet_factory.create_wallet(password=DEFAULT_WALLET_PASS)
|
registry_file = os.path.join("/tmp/", f"{str(uuid.uuid4())}.bolt")
|
||||||
allure.attach.file(wallet.path, os.path.basename(wallet.path), allure.attachment_type.JSON)
|
prepare_file = os.path.join("/tmp/", f"{str(uuid.uuid4())}.json")
|
||||||
return wallet.path
|
allure.dynamic.title(
|
||||||
|
f"Start background load with parameters: "
|
||||||
|
f"writers = {BACKGROUND_WRITERS_COUNT}, "
|
||||||
|
f"obj_size = {BACKGROUND_OBJ_SIZE}, "
|
||||||
|
f"load_time = {BACKGROUND_LOAD_MAX_TIME}"
|
||||||
|
f"prepare_json = {prepare_file}"
|
||||||
|
)
|
||||||
|
with allure.step("Get endpoints"):
|
||||||
|
endpoints_list = get_services_endpoints(
|
||||||
|
hosting=hosting,
|
||||||
|
service_name_regex=STORAGE_NODE_SERVICE_NAME_REGEX,
|
||||||
|
endpoint_attribute="rpc_endpoint",
|
||||||
|
)
|
||||||
|
endpoints = ",".join(endpoints_list)
|
||||||
|
load_params = LoadParams(
|
||||||
|
endpoint=endpoints,
|
||||||
|
obj_size=BACKGROUND_OBJ_SIZE,
|
||||||
|
registry_file=registry_file,
|
||||||
|
containers_count=1,
|
||||||
|
obj_count=0,
|
||||||
|
out_file=prepare_file,
|
||||||
|
readers=0,
|
||||||
|
writers=BACKGROUND_WRITERS_COUNT,
|
||||||
|
deleters=0,
|
||||||
|
load_time=BACKGROUND_LOAD_MAX_TIME,
|
||||||
|
load_type="grpc",
|
||||||
|
)
|
||||||
|
k6_load_instances = prepare_k6_instances(
|
||||||
|
load_nodes=LOAD_NODES,
|
||||||
|
login=LOAD_NODE_SSH_USER,
|
||||||
|
pkey=LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
|
load_params=load_params,
|
||||||
|
)
|
||||||
|
with allure.step("Run background load"):
|
||||||
|
for k6_load_instance in k6_load_instances:
|
||||||
|
k6_load_instance.start()
|
||||||
|
yield
|
||||||
|
with allure.step("Stop background load"):
|
||||||
|
for k6_load_instance in k6_load_instances:
|
||||||
|
k6_load_instance.stop()
|
||||||
|
with allure.step("Verify background load data"):
|
||||||
|
verify_params = LoadParams(
|
||||||
|
endpoint=endpoints,
|
||||||
|
clients=BACKGROUND_READERS_COUNT,
|
||||||
|
registry_file=registry_file,
|
||||||
|
load_time=BACKGROUND_LOAD_MAX_TIME,
|
||||||
|
load_type="verify",
|
||||||
|
)
|
||||||
|
k6_verify_instances = prepare_k6_instances(
|
||||||
|
load_nodes=LOAD_NODES,
|
||||||
|
login=LOAD_NODE_SSH_USER,
|
||||||
|
pkey=LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
|
load_params=verify_params,
|
||||||
|
prepare=False,
|
||||||
|
)
|
||||||
|
with allure.step("Run verify background load data"):
|
||||||
|
for k6_verify_instance in k6_verify_instances:
|
||||||
|
k6_verify_instance.start()
|
||||||
|
k6_verify_instance.wait_until_finished(BACKGROUND_LOAD_MAX_TIME)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(scope="session")
|
||||||
|
@allure.title("Prepare wallet and deposit")
|
||||||
|
def default_wallet(client_shell: Shell, temp_directory: str, cluster: Cluster):
|
||||||
|
wallet_path = os.path.join(os.getcwd(), ASSETS_DIR, f"{str(uuid.uuid4())}.json")
|
||||||
|
init_wallet(wallet_path, WALLET_PASS)
|
||||||
|
allure.attach.file(wallet_path, os.path.basename(wallet_path), allure.attachment_type.JSON)
|
||||||
|
|
||||||
|
if not FREE_STORAGE:
|
||||||
|
main_chain = cluster.main_chain_nodes[0]
|
||||||
|
deposit = 30
|
||||||
|
transfer_gas(
|
||||||
|
shell=client_shell,
|
||||||
|
amount=deposit + 1,
|
||||||
|
main_chain=main_chain,
|
||||||
|
wallet_to_path=wallet_path,
|
||||||
|
wallet_to_password=WALLET_PASS,
|
||||||
|
)
|
||||||
|
deposit_gas(
|
||||||
|
shell=client_shell,
|
||||||
|
main_chain=main_chain,
|
||||||
|
amount=deposit,
|
||||||
|
wallet_from_path=wallet_path,
|
||||||
|
wallet_from_password=WALLET_PASS,
|
||||||
|
)
|
||||||
|
|
||||||
|
return wallet_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Check logs for OOM and PANIC entries in {logs_dir}")
|
||||||
|
def check_logs(logs_dir: str):
|
||||||
|
problem_pattern = r"\Wpanic\W|\Woom\W|\Wtoo many open files\W"
|
||||||
|
|
||||||
|
log_file_paths = []
|
||||||
|
for directory_path, _, file_names in os.walk(logs_dir):
|
||||||
|
log_file_paths += [
|
||||||
|
os.path.join(directory_path, file_name)
|
||||||
|
for file_name in file_names
|
||||||
|
if re.match(r"\.(txt|log)", os.path.splitext(file_name)[-1], flags=re.IGNORECASE)
|
||||||
|
]
|
||||||
|
|
||||||
|
logs_with_problem = []
|
||||||
|
for file_path in log_file_paths:
|
||||||
|
with allure.step(f"Check log file {file_path}"):
|
||||||
|
with open(file_path, "r") as log_file:
|
||||||
|
if re.search(problem_pattern, log_file.read(), flags=re.IGNORECASE):
|
||||||
|
logs_with_problem.append(file_path)
|
||||||
|
if logs_with_problem:
|
||||||
|
raise pytest.fail(f"System logs {', '.join(logs_with_problem)} contain critical errors")
|
||||||
|
|
||||||
|
|
||||||
|
def dump_logs(hosting: Hosting, logs_dir: str, since: datetime, until: datetime) -> None:
|
||||||
|
# Dump logs to temp directory (because they might be too large to keep in RAM)
|
||||||
|
os.makedirs(logs_dir)
|
||||||
|
|
||||||
|
for host in hosting.hosts:
|
||||||
|
with allure.step(f"Dump logs from host {host.config.address}"):
|
||||||
|
try:
|
||||||
|
host.dump_logs(logs_dir, since=since, until=until)
|
||||||
|
except Exception as ex:
|
||||||
|
logger.warning(f"Exception during logs collection: {ex}")
|
||||||
|
|
||||||
|
|
||||||
|
def attach_logs(logs_dir: str) -> None:
|
||||||
|
# Zip all files and attach to Allure because it is more convenient to download a single
|
||||||
|
# zip with all logs rather than mess with individual logs files per service or node
|
||||||
|
logs_zip_file_path = shutil.make_archive(logs_dir, "zip", logs_dir)
|
||||||
|
allure.attach.file(logs_zip_file_path, name="logs.zip", extension="zip")
|
||||||
|
|
|
@ -2,8 +2,8 @@ import json
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PRIVATE_ACL_F
|
from epoch import tick_epoch
|
||||||
from frostfs_testlib.steps.cli.container import (
|
from python_keywords.container import (
|
||||||
create_container,
|
create_container,
|
||||||
delete_container,
|
delete_container,
|
||||||
get_container,
|
get_container,
|
||||||
|
@ -11,17 +11,19 @@ from frostfs_testlib.steps.cli.container import (
|
||||||
wait_for_container_creation,
|
wait_for_container_creation,
|
||||||
wait_for_container_deletion,
|
wait_for_container_deletion,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from utility import placement_policy_from_container
|
||||||
|
from wellknown_acl import PRIVATE_ACL_F
|
||||||
|
|
||||||
from pytest_tests.helpers.utility import placement_policy_from_container
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.container
|
@pytest.mark.container
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@pytest.mark.container
|
||||||
class TestContainer(ClusterTestBase):
|
class TestContainer(ClusterTestBase):
|
||||||
@pytest.mark.parametrize("name", ["", "test-container"], ids=["No name", "Set particular name"])
|
@pytest.mark.parametrize("name", ["", "test-container"], ids=["No name", "Set particular name"])
|
||||||
@pytest.mark.smoke
|
@pytest.mark.smoke
|
||||||
def test_container_creation(self, default_wallet: str, name: str):
|
def test_container_creation(self, default_wallet, name):
|
||||||
scenario_title = f"with name {name}" if name else "without name"
|
scenario_title = f"with name {name}" if name else "without name"
|
||||||
allure.dynamic.title(f"User can create container {scenario_title}")
|
allure.dynamic.title(f"User can create container {scenario_title}")
|
||||||
|
|
||||||
|
@ -85,7 +87,7 @@ class TestContainer(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Parallel container creation and deletion")
|
@allure.title("Parallel container creation and deletion")
|
||||||
def test_container_creation_deletion_parallel(self, default_wallet: str):
|
def test_container_creation_deletion_parallel(self, default_wallet):
|
||||||
containers_count = 3
|
containers_count = 3
|
||||||
wallet = default_wallet
|
wallet = default_wallet
|
||||||
placement_rule = "REP 2 IN X CBF 1 SELECT 2 FROM * AS X"
|
placement_rule = "REP 2 IN X CBF 1 SELECT 2 FROM * AS X"
|
||||||
|
@ -104,7 +106,7 @@ class TestContainer(ClusterTestBase):
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step("Wait for containers occur in container list"):
|
with allure.step(f"Wait for containers occur in container list"):
|
||||||
for cid in cids:
|
for cid in cids:
|
||||||
wait_for_container_creation(
|
wait_for_container_creation(
|
||||||
wallet,
|
wallet,
|
||||||
|
|
0
pytest_tests/testsuites/failovers/__init__.py
Normal file
0
pytest_tests/testsuites/failovers/__init__.py
Normal file
|
@ -1,29 +0,0 @@
|
||||||
import random
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
import allure
|
|
||||||
import pytest
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster, ClusterNode
|
|
||||||
from frostfs_testlib.storage.controllers import ShardsWatcher
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
|
||||||
@allure.title("Select random node for testing")
|
|
||||||
def node_under_test(cluster: Cluster) -> ClusterNode:
|
|
||||||
selected_node = random.choice(cluster.cluster_nodes)
|
|
||||||
allure.attach(f"{selected_node}", "Selected node", allure.attachment_type.TEXT)
|
|
||||||
return selected_node
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
|
||||||
@allure.title("Provide Shards watcher")
|
|
||||||
def shards_watcher(node_under_test: ClusterNode) -> ShardsWatcher:
|
|
||||||
watcher = ShardsWatcher(node_under_test)
|
|
||||||
return watcher
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture()
|
|
||||||
@allure.title("Test start time")
|
|
||||||
def test_start_time() -> datetime:
|
|
||||||
start_time = datetime.utcnow()
|
|
||||||
return start_time
|
|
|
@ -1,21 +1,18 @@
|
||||||
import logging
|
import logging
|
||||||
import random
|
from random import choices
|
||||||
from time import sleep
|
from time import sleep
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from cluster import StorageNode
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from failover_utils import wait_all_storage_nodes_returned, wait_object_replication
|
||||||
from frostfs_testlib.steps.cli.object import get_object, put_object_to_random_node
|
from file_helper import generate_file, get_file_hash
|
||||||
from frostfs_testlib.storage.cluster import StorageNode
|
from iptables_helper import IpTablesHelper
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.utils.failover_utils import (
|
from python_keywords.neofs_verbs import get_object, put_object_to_random_node
|
||||||
wait_all_storage_nodes_returned,
|
from wellknown_acl import PUBLIC_ACL
|
||||||
wait_object_replication,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_hash
|
|
||||||
|
|
||||||
from pytest_tests.helpers.iptables_helper import IpTablesHelper
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
STORAGE_NODE_COMMUNICATION_PORT = "8080"
|
STORAGE_NODE_COMMUNICATION_PORT = "8080"
|
||||||
|
@ -38,11 +35,11 @@ class TestFailoverNetwork(ClusterTestBase):
|
||||||
IpTablesHelper.restore_input_traffic_to_port(node.host.get_shell(), PORTS_TO_BLOCK)
|
IpTablesHelper.restore_input_traffic_to_port(node.host.get_shell(), PORTS_TO_BLOCK)
|
||||||
blocked_nodes.remove(node)
|
blocked_nodes.remove(node)
|
||||||
if not_empty:
|
if not_empty:
|
||||||
wait_all_storage_nodes_returned(self.shell, self.cluster)
|
wait_all_storage_nodes_returned(self.cluster)
|
||||||
|
|
||||||
@allure.title("Block Storage node traffic")
|
@allure.title("Block Storage node traffic")
|
||||||
def test_block_storage_node_traffic(
|
def test_block_storage_node_traffic(
|
||||||
self, default_wallet: str, require_multiple_hosts, simple_object_size: int
|
self, default_wallet, require_multiple_hosts, simple_object_size
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
Block storage nodes traffic using iptables and wait for replication for objects.
|
Block storage nodes traffic using iptables and wait for replication for objects.
|
||||||
|
@ -72,7 +69,7 @@ class TestFailoverNetwork(ClusterTestBase):
|
||||||
nodes_to_block = nodes
|
nodes_to_block = nodes
|
||||||
if nodes_to_block_count > len(nodes):
|
if nodes_to_block_count > len(nodes):
|
||||||
# TODO: the intent of this logic is not clear, need to revisit
|
# TODO: the intent of this logic is not clear, need to revisit
|
||||||
nodes_to_block = random.choices(nodes, k=2)
|
nodes_to_block = choices(nodes, k=2)
|
||||||
|
|
||||||
excluded_nodes = []
|
excluded_nodes = []
|
||||||
for node in nodes_to_block:
|
for node in nodes_to_block:
|
||||||
|
@ -92,7 +89,7 @@ class TestFailoverNetwork(ClusterTestBase):
|
||||||
)
|
)
|
||||||
assert node not in new_nodes
|
assert node not in new_nodes
|
||||||
|
|
||||||
with allure.step("Check object data is not corrupted"):
|
with allure.step(f"Check object data is not corrupted"):
|
||||||
got_file_path = get_object(
|
got_file_path = get_object(
|
||||||
wallet, cid, oid, endpoint=new_nodes[0].get_rpc_endpoint(), shell=self.shell
|
wallet, cid, oid, endpoint=new_nodes[0].get_rpc_endpoint(), shell=self.shell
|
||||||
)
|
)
|
||||||
|
@ -104,7 +101,7 @@ class TestFailoverNetwork(ClusterTestBase):
|
||||||
blocked_nodes.remove(node)
|
blocked_nodes.remove(node)
|
||||||
sleep(wakeup_node_timeout)
|
sleep(wakeup_node_timeout)
|
||||||
|
|
||||||
with allure.step("Check object data is not corrupted"):
|
with allure.step(f"Check object data is not corrupted"):
|
||||||
new_nodes = wait_object_replication(
|
new_nodes = wait_object_replication(
|
||||||
cid, oid, 2, shell=self.shell, nodes=self.cluster.storage_nodes
|
cid, oid, 2, shell=self.shell, nodes=self.cluster.storage_nodes
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,232 +0,0 @@
|
||||||
import logging
|
|
||||||
import os.path
|
|
||||||
import random
|
|
||||||
import time
|
|
||||||
|
|
||||||
import allure
|
|
||||||
import pytest
|
|
||||||
from frostfs_testlib.resources.common import MORPH_BLOCK_TIME
|
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
|
||||||
from frostfs_testlib.steps.cli.container import (
|
|
||||||
StorageContainer,
|
|
||||||
StorageContainerInfo,
|
|
||||||
create_container,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.cli.object import get_object
|
|
||||||
from frostfs_testlib.steps.node_management import check_node_in_map, check_node_not_in_map
|
|
||||||
from frostfs_testlib.storage.cluster import ClusterNode, StorageNode
|
|
||||||
from frostfs_testlib.storage.controllers import ClusterStateController
|
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import StorageObjectInfo
|
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import wait_for_success
|
|
||||||
from frostfs_testlib.utils import datetime_utils
|
|
||||||
from frostfs_testlib.utils.failover_utils import wait_for_host_offline, wait_object_replication
|
|
||||||
from frostfs_testlib.utils.file_utils import get_file_hash
|
|
||||||
from pytest import FixtureRequest
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.failover
|
|
||||||
@pytest.mark.failover_server
|
|
||||||
class TestFailoverServer(ClusterTestBase):
|
|
||||||
@wait_for_success(max_wait_time=120, interval=1)
|
|
||||||
def wait_node_not_in_map(self, *args, **kwargs):
|
|
||||||
check_node_not_in_map(*args, **kwargs)
|
|
||||||
|
|
||||||
@wait_for_success(max_wait_time=120, interval=1)
|
|
||||||
def wait_node_in_map(self, *args, **kwargs):
|
|
||||||
check_node_in_map(*args, **kwargs)
|
|
||||||
|
|
||||||
@allure.step("Create {count_containers} containers and {count_files} objects")
|
|
||||||
@pytest.fixture
|
|
||||||
def containers(
|
|
||||||
self,
|
|
||||||
request: FixtureRequest,
|
|
||||||
default_wallet: str,
|
|
||||||
) -> list[StorageContainer]:
|
|
||||||
|
|
||||||
placement_rule = "REP 2 CBF 2 SELECT 2 FROM * AS X"
|
|
||||||
|
|
||||||
containers = []
|
|
||||||
|
|
||||||
for _ in range(request.param):
|
|
||||||
cont_id = create_container(
|
|
||||||
default_wallet,
|
|
||||||
shell=self.shell,
|
|
||||||
endpoint=self.cluster.default_rpc_endpoint,
|
|
||||||
rule=placement_rule,
|
|
||||||
basic_acl=PUBLIC_ACL,
|
|
||||||
)
|
|
||||||
wallet = WalletInfo(path=default_wallet)
|
|
||||||
storage_cont_info = StorageContainerInfo(id=cont_id, wallet_file=wallet)
|
|
||||||
containers.append(
|
|
||||||
StorageContainer(
|
|
||||||
storage_container_info=storage_cont_info, shell=self.shell, cluster=self.cluster
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
return containers
|
|
||||||
|
|
||||||
@allure.step("Create object and delete after test")
|
|
||||||
@pytest.fixture(scope="class")
|
|
||||||
def storage_objects(
|
|
||||||
self,
|
|
||||||
request: FixtureRequest,
|
|
||||||
containers: list[StorageContainer],
|
|
||||||
simple_object_size: int,
|
|
||||||
complex_object_size: int,
|
|
||||||
) -> StorageObjectInfo:
|
|
||||||
count_object = request.param
|
|
||||||
object_size = [simple_object_size, complex_object_size]
|
|
||||||
object_list = []
|
|
||||||
for cont in containers:
|
|
||||||
for _ in range(count_object):
|
|
||||||
object_list.append(cont.generate_object(size=random.choice(object_size)))
|
|
||||||
|
|
||||||
for storage_object in object_list:
|
|
||||||
os.remove(storage_object.file_path)
|
|
||||||
|
|
||||||
yield object_list
|
|
||||||
|
|
||||||
@allure.step("Select random node to stop and start it after test")
|
|
||||||
@pytest.fixture
|
|
||||||
def node_to_stop(
|
|
||||||
self, node_under_test: ClusterNode, cluster_state_controller: ClusterStateController
|
|
||||||
) -> ClusterNode:
|
|
||||||
yield node_under_test
|
|
||||||
with allure.step(f"start {node_under_test.storage_node}"):
|
|
||||||
cluster_state_controller.start_stopped_hosts()
|
|
||||||
|
|
||||||
@allure.step("Upload object with nodes and compare")
|
|
||||||
def get_corrupted_objects_list(
|
|
||||||
self, nodes: list[StorageNode], storage_objects: list[StorageObjectInfo]
|
|
||||||
) -> list[StorageObjectInfo]:
|
|
||||||
corrupted_objects = []
|
|
||||||
for node in nodes:
|
|
||||||
for storage_object in storage_objects:
|
|
||||||
got_file_path = get_object(
|
|
||||||
storage_object.wallet_file_path,
|
|
||||||
storage_object.cid,
|
|
||||||
storage_object.oid,
|
|
||||||
endpoint=node.get_rpc_endpoint(),
|
|
||||||
shell=self.shell,
|
|
||||||
timeout="60s",
|
|
||||||
)
|
|
||||||
if storage_object.file_hash != get_file_hash(got_file_path):
|
|
||||||
corrupted_objects.append(storage_object)
|
|
||||||
os.remove(got_file_path)
|
|
||||||
|
|
||||||
return corrupted_objects
|
|
||||||
|
|
||||||
def check_objects_replication(
|
|
||||||
self, storage_objects: list[StorageObjectInfo], storage_nodes: list[StorageNode]
|
|
||||||
) -> None:
|
|
||||||
for storage_object in storage_objects:
|
|
||||||
wait_object_replication(
|
|
||||||
storage_object.cid,
|
|
||||||
storage_object.oid,
|
|
||||||
2,
|
|
||||||
shell=self.shell,
|
|
||||||
nodes=storage_nodes,
|
|
||||||
sleep_interval=45,
|
|
||||||
attempts=60,
|
|
||||||
)
|
|
||||||
|
|
||||||
@allure.title("Full shutdown node")
|
|
||||||
@pytest.mark.parametrize("containers, storage_objects", [(5, 10)], indirect=True)
|
|
||||||
def test_complete_node_shutdown(
|
|
||||||
self,
|
|
||||||
containers: list[StorageContainer],
|
|
||||||
storage_objects: list[StorageObjectInfo],
|
|
||||||
default_wallet: str,
|
|
||||||
node_to_stop: ClusterNode,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
|
|
||||||
with allure.step(f"Remove {node_to_stop} from the list of nodes"):
|
|
||||||
alive_nodes = list(set(self.cluster.cluster_nodes) - {node_to_stop})
|
|
||||||
|
|
||||||
storage_nodes = [cluster.storage_node for cluster in alive_nodes]
|
|
||||||
|
|
||||||
with allure.step("Tick epoch"):
|
|
||||||
self.tick_epochs(1, storage_nodes[0])
|
|
||||||
|
|
||||||
with allure.step("Wait 2 block time"):
|
|
||||||
time.sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME) * 2)
|
|
||||||
|
|
||||||
cluster_state_controller.stop_node_host(node=node_to_stop, mode="hard")
|
|
||||||
|
|
||||||
with allure.step(f"Check if the node {node_to_stop.storage_node} has stopped"):
|
|
||||||
wait_for_host_offline(self.shell, node_to_stop.storage_node)
|
|
||||||
|
|
||||||
with allure.step("Verify that there are no corrupted objects"):
|
|
||||||
corrupted_objects_list = self.get_corrupted_objects_list(storage_nodes, storage_objects)
|
|
||||||
|
|
||||||
assert not corrupted_objects_list
|
|
||||||
|
|
||||||
with allure.step(f"check {node_to_stop.storage_node} in map"):
|
|
||||||
self.wait_node_in_map(
|
|
||||||
node_to_stop.storage_node, self.shell, alive_node=storage_nodes[0]
|
|
||||||
)
|
|
||||||
|
|
||||||
count_tick_epoch = int(alive_nodes[0].ir_node.get_netmap_cleaner_threshold()) + 2
|
|
||||||
|
|
||||||
with allure.step(f"Tick {count_tick_epoch} epoch, in {storage_nodes[0]} node"):
|
|
||||||
for tick in range(count_tick_epoch):
|
|
||||||
self.tick_epoch(storage_nodes[0])
|
|
||||||
time.sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME) * 2)
|
|
||||||
|
|
||||||
with allure.step(f"Check if the node {node_to_stop.storage_node} has stopped"):
|
|
||||||
wait_for_host_offline(self.shell, node_to_stop.storage_node)
|
|
||||||
|
|
||||||
with allure.step(f"Check {node_to_stop} in not map"):
|
|
||||||
self.wait_node_not_in_map(
|
|
||||||
node_to_stop.storage_node, self.shell, alive_node=storage_nodes[0]
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step(
|
|
||||||
f"Verify that there are no corrupted objects after {count_tick_epoch} epoch"
|
|
||||||
):
|
|
||||||
corrupted_objects_list = self.get_corrupted_objects_list(storage_nodes, storage_objects)
|
|
||||||
assert not corrupted_objects_list
|
|
||||||
|
|
||||||
@allure.title("Temporarily disable a node")
|
|
||||||
@pytest.mark.parametrize("containers, storage_objects", [(5, 10)], indirect=True)
|
|
||||||
def test_temporarily_disable_a_node(
|
|
||||||
self,
|
|
||||||
containers: list[StorageContainer],
|
|
||||||
storage_objects: list[StorageObjectInfo],
|
|
||||||
default_wallet: str,
|
|
||||||
node_to_stop: ClusterNode,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
with allure.step(f"Remove {node_to_stop} from the list of nodes"):
|
|
||||||
storage_nodes = list(set(self.cluster.storage_nodes) - {node_to_stop.storage_node})
|
|
||||||
|
|
||||||
with allure.step("Tick epoch"):
|
|
||||||
self.tick_epochs(1, storage_nodes[0])
|
|
||||||
|
|
||||||
with allure.step("Wait 2 block time"):
|
|
||||||
time.sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME) * 2)
|
|
||||||
|
|
||||||
cluster_state_controller.stop_node_host(node=node_to_stop, mode="hard")
|
|
||||||
|
|
||||||
with allure.step(f"Check if the node {node_to_stop} has stopped"):
|
|
||||||
wait_for_host_offline(self.shell, node_to_stop.storage_node)
|
|
||||||
|
|
||||||
with allure.step("Verify that there are no corrupted objects"):
|
|
||||||
corrupted_objects_list = self.get_corrupted_objects_list(storage_nodes, storage_objects)
|
|
||||||
assert not corrupted_objects_list
|
|
||||||
|
|
||||||
with allure.step(f"Check {node_to_stop} in map"):
|
|
||||||
self.wait_node_in_map(
|
|
||||||
node_to_stop.storage_node, self.shell, alive_node=storage_nodes[0]
|
|
||||||
)
|
|
||||||
|
|
||||||
cluster_state_controller.start_node_host(node_to_stop)
|
|
||||||
|
|
||||||
with allure.step("Verify that there are no corrupted objects"):
|
|
||||||
corrupted_objects_list = self.get_corrupted_objects_list(storage_nodes, storage_objects)
|
|
||||||
assert not corrupted_objects_list
|
|
|
@ -1,75 +1,27 @@
|
||||||
import logging
|
import logging
|
||||||
from datetime import datetime
|
|
||||||
from time import sleep
|
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.analytics import test_case
|
from cluster import Cluster, StorageNode
|
||||||
from frostfs_testlib.hosting import Host
|
from failover_utils import wait_all_storage_nodes_returned, wait_object_replication
|
||||||
from frostfs_testlib.resources.common import MORPH_BLOCK_TIME
|
from file_helper import generate_file, get_file_hash
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from neofs_testlib.hosting import Host
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
from neofs_testlib.shell import CommandOptions
|
||||||
from frostfs_testlib.shell import CommandOptions, Shell
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.steps.cli.container import (
|
from python_keywords.neofs_verbs import get_object, put_object_to_random_node
|
||||||
StorageContainer,
|
from wellknown_acl import PUBLIC_ACL
|
||||||
StorageContainerInfo,
|
|
||||||
create_container,
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.cli.object import get_object, put_object_to_random_node
|
|
||||||
from frostfs_testlib.steps.node_management import (
|
|
||||||
check_node_in_map,
|
|
||||||
check_node_not_in_map,
|
|
||||||
exclude_node_from_network_map,
|
|
||||||
include_node_to_network_map,
|
|
||||||
remove_nodes_from_map_morph,
|
|
||||||
wait_for_node_to_be_ready,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster, ClusterNode, StorageNode
|
|
||||||
from frostfs_testlib.storage.controllers import ClusterStateController, ShardsWatcher
|
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import StorageObjectInfo
|
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import expect_not_raises
|
|
||||||
from frostfs_testlib.utils import datetime_utils
|
|
||||||
from frostfs_testlib.utils.failover_utils import (
|
|
||||||
wait_all_storage_nodes_returned,
|
|
||||||
wait_object_replication,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.utils.file_keeper import FileKeeper
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_hash
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
stopped_nodes: list[StorageNode] = []
|
stopped_nodes: list[StorageNode] = []
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="function")
|
|
||||||
@allure.title("Provide File Keeper")
|
|
||||||
def file_keeper():
|
|
||||||
keeper = FileKeeper()
|
|
||||||
yield keeper
|
|
||||||
keeper.restore_files()
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Return all stopped hosts")
|
|
||||||
@pytest.fixture(scope="function", autouse=True)
|
@pytest.fixture(scope="function", autouse=True)
|
||||||
def after_run_return_all_stopped_hosts(client_shell: Shell, cluster: Cluster) -> str:
|
@allure.step("Return all stopped hosts")
|
||||||
yield "After this test stopped services will be started automatically via fixture"
|
def after_run_return_all_stopped_hosts(cluster: Cluster):
|
||||||
return_stopped_hosts(client_shell, cluster)
|
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Return all stopped storage services after test")
|
|
||||||
@pytest.fixture(scope="function")
|
|
||||||
def after_run_return_all_stopped_services(cluster_state_controller: ClusterStateController):
|
|
||||||
yield
|
yield
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
return_stopped_hosts(cluster)
|
||||||
|
|
||||||
|
|
||||||
@allure.step("Return all stopped S3 GateWay services after test")
|
|
||||||
@pytest.fixture(scope="function")
|
|
||||||
def after_run_return_all_stopped_s3(cluster_state_controller: ClusterStateController):
|
|
||||||
yield
|
|
||||||
cluster_state_controller.start_stopped_s3_gate()
|
|
||||||
|
|
||||||
|
|
||||||
def panic_reboot_host(host: Host) -> None:
|
def panic_reboot_host(host: Host) -> None:
|
||||||
|
@ -80,13 +32,13 @@ def panic_reboot_host(host: Host) -> None:
|
||||||
shell.exec('sudo sh -c "echo b > /proc/sysrq-trigger"', options)
|
shell.exec('sudo sh -c "echo b > /proc/sysrq-trigger"', options)
|
||||||
|
|
||||||
|
|
||||||
def return_stopped_hosts(shell: Shell, cluster: Cluster) -> None:
|
def return_stopped_hosts(cluster: Cluster) -> None:
|
||||||
for node in list(stopped_nodes):
|
for node in list(stopped_nodes):
|
||||||
with allure.step(f"Start host {node}"):
|
with allure.step(f"Start host {node}"):
|
||||||
node.host.start_host()
|
node.host.start_host()
|
||||||
stopped_nodes.remove(node)
|
stopped_nodes.remove(node)
|
||||||
|
|
||||||
wait_all_storage_nodes_returned(shell, cluster)
|
wait_all_storage_nodes_returned(cluster)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.failover
|
@pytest.mark.failover
|
||||||
|
@ -136,7 +88,7 @@ class TestFailoverStorage(ClusterTestBase):
|
||||||
assert get_file_hash(source_file_path) == get_file_hash(got_file_path)
|
assert get_file_hash(source_file_path) == get_file_hash(got_file_path)
|
||||||
|
|
||||||
with allure.step("Return all hosts"):
|
with allure.step("Return all hosts"):
|
||||||
return_stopped_hosts(self.shell, self.cluster)
|
return_stopped_hosts(self.cluster)
|
||||||
|
|
||||||
with allure.step("Check object data is not corrupted"):
|
with allure.step("Check object data is not corrupted"):
|
||||||
new_nodes = wait_object_replication(
|
new_nodes = wait_object_replication(
|
||||||
|
@ -218,628 +170,3 @@ class TestFailoverStorage(ClusterTestBase):
|
||||||
wallet, cid, oid, shell=self.shell, endpoint=new_nodes[0].get_rpc_endpoint()
|
wallet, cid, oid, shell=self.shell, endpoint=new_nodes[0].get_rpc_endpoint()
|
||||||
)
|
)
|
||||||
assert get_file_hash(source_file_path) == get_file_hash(got_file_path)
|
assert get_file_hash(source_file_path) == get_file_hash(got_file_path)
|
||||||
|
|
||||||
@allure.title("Do not ignore unhealthy tree endpoints")
|
|
||||||
def test_unhealthy_tree(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
after_run_return_all_stopped_s3,
|
|
||||||
after_run_return_all_stopped_services,
|
|
||||||
):
|
|
||||||
default_node = self.cluster.cluster_nodes[0]
|
|
||||||
default_s3gate = self.cluster.s3_gates[0]
|
|
||||||
|
|
||||||
with allure.step("Turn S3 GW off on default node"):
|
|
||||||
default_s3gate.stop_service()
|
|
||||||
|
|
||||||
with allure.step("Turn off storage on default node"):
|
|
||||||
cluster_state_controller.stop_storage_service(default_node)
|
|
||||||
|
|
||||||
with allure.step("Turn on S3 GW on default node"):
|
|
||||||
default_s3gate.start_service()
|
|
||||||
|
|
||||||
with allure.step("Turn on storage on default node"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Create bucket with REP 1 SELECT 1 policy"):
|
|
||||||
bucket = s3_client.create_bucket(
|
|
||||||
location_constraint="load-1-1",
|
|
||||||
)
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
with allure.step("Put object into bucket"):
|
|
||||||
put_object = s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
|
|
||||||
with allure.step("Turn off all storage nodes except default"):
|
|
||||||
for node in self.cluster.cluster_nodes[1:]:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node)
|
|
||||||
|
|
||||||
with allure.step("Check that object is available"):
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
|
||||||
if "s3_client" in metafunc.fixturenames:
|
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.failover
|
|
||||||
@pytest.mark.failover_empty_map
|
|
||||||
class TestEmptyMap(ClusterTestBase):
|
|
||||||
"""
|
|
||||||
A set of tests for makes map empty and verify that we can read objects after that
|
|
||||||
"""
|
|
||||||
|
|
||||||
@allure.step("Teardown after EmptyMap offline test")
|
|
||||||
@pytest.fixture()
|
|
||||||
def empty_map_offline_teardown(self):
|
|
||||||
yield
|
|
||||||
with allure.step("Return all storage nodes to network map"):
|
|
||||||
for node in list(stopped_nodes):
|
|
||||||
include_node_to_network_map(node, node, shell=self.shell, cluster=self.cluster)
|
|
||||||
stopped_nodes.remove(node)
|
|
||||||
|
|
||||||
@test_case.title("Test makes network map empty (offline all storage nodes)")
|
|
||||||
@test_case.priority(test_case.TestCasePriority.HIGH)
|
|
||||||
@test_case.suite_name("failovers")
|
|
||||||
@test_case.suite_section("test_failover_storage")
|
|
||||||
@pytest.mark.failover_empty_map_offlne
|
|
||||||
@allure.title("Test makes network map empty (offline all storage nodes)")
|
|
||||||
def test_offline_all_storage_nodes(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
bucket: str,
|
|
||||||
simple_object_size: int,
|
|
||||||
empty_map_offline_teardown,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
The test makes network map empty (set offline status on all storage nodes) then returns all nodes to map and checks that object can read through s3.
|
|
||||||
|
|
||||||
Steps:
|
|
||||||
1. Check that bucket is empty
|
|
||||||
2: PUT object into bucket
|
|
||||||
3: Check that object exists in bucket
|
|
||||||
4: Exclude all storage nodes from network map (set status OFFLINE)
|
|
||||||
5: Return all storage nodes to network map
|
|
||||||
6: Check that we can read object from #2
|
|
||||||
Args:
|
|
||||||
bucket: bucket which contains tested object
|
|
||||||
simple_object_size: size of object
|
|
||||||
"""
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
bucket_objects = [file_name]
|
|
||||||
|
|
||||||
objects_list = s3_client.list_objects(bucket)
|
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
|
||||||
s3_client.put_object(bucket, file_path)
|
|
||||||
|
|
||||||
with allure.step("Check that object exists in bucket"):
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, bucket_objects)
|
|
||||||
|
|
||||||
storage_nodes = self.cluster.storage_nodes
|
|
||||||
with allure.step("Exclude all storage nodes from network map"):
|
|
||||||
for node in storage_nodes:
|
|
||||||
exclude_node_from_network_map(node, node, shell=self.shell, cluster=self.cluster)
|
|
||||||
stopped_nodes.append(node)
|
|
||||||
|
|
||||||
with allure.step("Return all storage nodes to network map"):
|
|
||||||
for node in storage_nodes:
|
|
||||||
include_node_to_network_map(node, node, shell=self.shell, cluster=self.cluster)
|
|
||||||
stopped_nodes.remove(node)
|
|
||||||
|
|
||||||
with allure.step("Check that we can read object"):
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, bucket_objects)
|
|
||||||
|
|
||||||
@allure.step("Teardown after EmptyMap stop service test")
|
|
||||||
@pytest.fixture()
|
|
||||||
def empty_map_stop_service_teardown(self):
|
|
||||||
yield
|
|
||||||
with allure.step("Return all storage nodes to network map"):
|
|
||||||
for node in list(list(stopped_nodes)):
|
|
||||||
with allure.step(f"Start node {node}"):
|
|
||||||
node.start_service()
|
|
||||||
with allure.step(f"Waiting status ready for node {node}"):
|
|
||||||
wait_for_node_to_be_ready(node)
|
|
||||||
|
|
||||||
sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME))
|
|
||||||
self.tick_epochs(1)
|
|
||||||
check_node_in_map(node, shell=self.shell, alive_node=node)
|
|
||||||
stopped_nodes.remove(node)
|
|
||||||
|
|
||||||
@test_case.title("Test makes network map empty (stop storage service on all nodes)")
|
|
||||||
@test_case.priority(test_case.TestCasePriority.HIGH)
|
|
||||||
@test_case.suite_name("failovers")
|
|
||||||
@test_case.suite_section("test_failover_storage")
|
|
||||||
@pytest.mark.failover_empty_map_stop_service
|
|
||||||
@allure.title("Test makes network map empty (stop storage service on all nodes)")
|
|
||||||
def test_stop_all_storage_nodes(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
bucket: str,
|
|
||||||
simple_object_size: int,
|
|
||||||
empty_map_stop_service_teardown,
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
The test makes network map empty (stop storage service on all nodes
|
|
||||||
then use 'frostfs-adm morph delete-nodes' to delete nodes from map)
|
|
||||||
then start all services and checks that object can read through s3.
|
|
||||||
|
|
||||||
Steps:
|
|
||||||
1. Check that bucket is empty
|
|
||||||
2: PUT object into bucket
|
|
||||||
3: Check that object exists in bucket
|
|
||||||
4: Exclude all storage nodes from network map (stop storage service
|
|
||||||
and manual exclude from map)
|
|
||||||
5: Return all storage nodes to network map
|
|
||||||
6: Check that we can read object from #2
|
|
||||||
Args:
|
|
||||||
bucket: bucket which contains tested object
|
|
||||||
simple_object_size: size of object
|
|
||||||
"""
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
bucket_objects = [file_name]
|
|
||||||
|
|
||||||
objects_list = s3_client.list_objects(bucket)
|
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
|
||||||
s3_client.put_object(bucket, file_path)
|
|
||||||
|
|
||||||
with allure.step("Check that object exists in bucket"):
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, bucket_objects)
|
|
||||||
|
|
||||||
with allure.step("Stop all storage nodes"):
|
|
||||||
for node in self.cluster.storage_nodes:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
node.stop_service()
|
|
||||||
stopped_nodes.append(node)
|
|
||||||
|
|
||||||
with allure.step("Remove all nodes from network map"):
|
|
||||||
remove_nodes_from_map_morph(
|
|
||||||
shell=self.shell, cluster=self.cluster, remove_nodes=stopped_nodes
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Return all storage nodes to network map"):
|
|
||||||
self.return_nodes_after_stop_with_check_empty_map(stopped_nodes)
|
|
||||||
|
|
||||||
with allure.step("Check that object exists in bucket"):
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, bucket_objects)
|
|
||||||
|
|
||||||
@allure.step("Return all nodes to cluster with check empty map first")
|
|
||||||
def return_nodes_after_stop_with_check_empty_map(self, return_nodes=None) -> None:
|
|
||||||
first_node = True
|
|
||||||
for node in list(return_nodes):
|
|
||||||
with allure.step(f"Start node {node}"):
|
|
||||||
node.start_service()
|
|
||||||
with allure.step(f"Waiting status ready for node {node}"):
|
|
||||||
wait_for_node_to_be_ready(node)
|
|
||||||
|
|
||||||
with allure.step("Make sure that network map is empty"):
|
|
||||||
if first_node:
|
|
||||||
for check_node in list(return_nodes):
|
|
||||||
check_node_not_in_map(check_node, shell=self.shell, alive_node=node)
|
|
||||||
first_node = False
|
|
||||||
|
|
||||||
sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME))
|
|
||||||
self.tick_epochs(1)
|
|
||||||
check_node_in_map(node, shell=self.shell, alive_node=node)
|
|
||||||
stopped_nodes.remove(node)
|
|
||||||
|
|
||||||
@allure.title("Test S3 Object loss from fstree/blobovnicza, versioning is enabled")
|
|
||||||
def test_s3_fstree_blobovnicza_loss_versioning_on(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
bucket = s3_client.create_bucket()
|
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
|
|
||||||
object_versions = []
|
|
||||||
with allure.step("Put object into one bucket"):
|
|
||||||
put_object = s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
object_versions.append(put_object)
|
|
||||||
|
|
||||||
with allure.step("Stop all storage nodes"):
|
|
||||||
for node in self.cluster.cluster_nodes:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node)
|
|
||||||
|
|
||||||
with allure.step("Delete blobovnicza and fstree from all nodes"):
|
|
||||||
for node in self.cluster.storage_nodes:
|
|
||||||
node.delete_blobovnicza()
|
|
||||||
node.delete_fstree()
|
|
||||||
|
|
||||||
with allure.step("Start all storage nodes"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
# need to get Delete Marker first
|
|
||||||
with allure.step("Delete the object from the bucket"):
|
|
||||||
delete_object = s3_client.delete_object(bucket, file_name)
|
|
||||||
object_versions.append(delete_object["VersionId"])
|
|
||||||
|
|
||||||
# and now delete all versions of object (including Delete Markers)
|
|
||||||
with allure.step("Delete all versions of the object from the bucket"):
|
|
||||||
for version in object_versions:
|
|
||||||
delete_object = s3_client.delete_object(bucket, file_name, version_id=version)
|
|
||||||
|
|
||||||
with allure.step("Delete bucket"):
|
|
||||||
s3_client.delete_bucket(bucket)
|
|
||||||
|
|
||||||
@allure.title("Test S3 Object loss from fstree/blobovnicza, versioning is disabled")
|
|
||||||
def test_s3_fstree_blobovnicza_loss_versioning_off(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
bucket = s3_client.create_bucket()
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
|
|
||||||
with allure.step("Put object into one bucket"):
|
|
||||||
s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
|
|
||||||
with allure.step("Stop all storage nodes"):
|
|
||||||
for node in self.cluster.cluster_nodes:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node)
|
|
||||||
|
|
||||||
with allure.step("Delete blobovnicza and fstree from all nodes"):
|
|
||||||
for node in self.cluster.storage_nodes:
|
|
||||||
node.delete_blobovnicza()
|
|
||||||
node.delete_fstree()
|
|
||||||
|
|
||||||
with allure.step("Start all storage nodes"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Delete the object from the bucket"):
|
|
||||||
s3_client.delete_object(bucket, file_name)
|
|
||||||
|
|
||||||
with allure.step("Delete bucket"):
|
|
||||||
s3_client.delete_bucket(bucket)
|
|
||||||
|
|
||||||
@pytest.mark.skip(reason="Need to increase cache lifetime")
|
|
||||||
@pytest.mark.parametrize(
|
|
||||||
# versioning should NOT be VersioningStatus.SUSPENDED, it needs to be undefined
|
|
||||||
"versioning_status",
|
|
||||||
[VersioningStatus.ENABLED, None],
|
|
||||||
)
|
|
||||||
@allure.title(
|
|
||||||
"After Pilorama.db loss on all nodes list objects should return nothing in second listing"
|
|
||||||
)
|
|
||||||
def test_s3_pilorama_loss(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
versioning_status: VersioningStatus,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
bucket = s3_client.create_bucket()
|
|
||||||
if versioning_status:
|
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, versioning_status)
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
|
|
||||||
with allure.step("Put object into one bucket"):
|
|
||||||
s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
|
|
||||||
with allure.step("Stop all storage nodes"):
|
|
||||||
for node in self.cluster.cluster_nodes:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node)
|
|
||||||
|
|
||||||
with allure.step("Delete pilorama.db from all nodes"):
|
|
||||||
for node in self.cluster.storage_nodes:
|
|
||||||
node.delete_pilorama()
|
|
||||||
|
|
||||||
with allure.step("Start all storage nodes"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Check list objects first time"):
|
|
||||||
objects_list = s3_client.list_objects(bucket)
|
|
||||||
assert objects_list, f"Expected not empty bucket"
|
|
||||||
|
|
||||||
with allure.step("Check list objects second time"):
|
|
||||||
objects_list = s3_client.list_objects(bucket)
|
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
|
||||||
|
|
||||||
with allure.step("Delete bucket"):
|
|
||||||
s3_client.delete_bucket(bucket)
|
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.failover
|
|
||||||
@pytest.mark.failover_data_loss
|
|
||||||
class TestStorageDataLoss(ClusterTestBase):
|
|
||||||
@allure.step("Get list of all piloramas on node")
|
|
||||||
def get_piloramas_list(self, cluster_state_controller, node) -> list:
|
|
||||||
data_directory_path = cluster_state_controller.get_data_directory()
|
|
||||||
|
|
||||||
cmd = f"sudo ls -1 {data_directory_path}/meta*/pilorama*"
|
|
||||||
shell = cluster_state_controller.host.get_shell()
|
|
||||||
stdout = shell.exec(cmd).stdout
|
|
||||||
|
|
||||||
piloramas = stdout.split("\n")
|
|
||||||
return piloramas
|
|
||||||
|
|
||||||
@allure.title(
|
|
||||||
"After metabase loss on all nodes operations on objects and buckets should be still available via S3"
|
|
||||||
)
|
|
||||||
@pytest.mark.metabase_loss
|
|
||||||
def test_metabase_loss(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
complex_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
after_run_return_all_stopped_services: str,
|
|
||||||
file_keeper: FileKeeper,
|
|
||||||
):
|
|
||||||
allure.dynamic.description(after_run_return_all_stopped_services)
|
|
||||||
|
|
||||||
with allure.step("Create bucket"):
|
|
||||||
bucket = s3_client.create_bucket()
|
|
||||||
|
|
||||||
with allure.step("Put objects into bucket"):
|
|
||||||
simple_object_path = generate_file(simple_object_size)
|
|
||||||
simple_object_key = s3_helper.object_key_from_file_path(simple_object_path)
|
|
||||||
|
|
||||||
complex_object_path = generate_file(complex_object_size)
|
|
||||||
complex_object_key = s3_helper.object_key_from_file_path(complex_object_path)
|
|
||||||
|
|
||||||
s3_client.put_object(bucket, simple_object_path)
|
|
||||||
s3_client.put_object(bucket, complex_object_path)
|
|
||||||
|
|
||||||
with allure.step("Check objects are in bucket"):
|
|
||||||
s3_helper.check_objects_in_bucket(
|
|
||||||
s3_client, bucket, expected_objects=[simple_object_key, complex_object_key]
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Stop storage services on all nodes"):
|
|
||||||
cluster_state_controller.stop_all_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Delete metabase from all nodes"):
|
|
||||||
for node in cluster_state_controller.cluster.storage_nodes:
|
|
||||||
node.delete_metabase()
|
|
||||||
|
|
||||||
with allure.step("Enable resync_metabase option for storage services"):
|
|
||||||
for storage_node in cluster_state_controller.cluster.storage_nodes:
|
|
||||||
with allure.step(f"Enable resync_metabase option for {storage_node}"):
|
|
||||||
config_file_path, config = storage_node.get_config()
|
|
||||||
if not config["storage"]["shard"]["default"]["resync_metabase"]:
|
|
||||||
file_keeper.add(storage_node, config_file_path)
|
|
||||||
config["storage"]["shard"]["default"]["resync_metabase"] = True
|
|
||||||
storage_node.save_config(config)
|
|
||||||
|
|
||||||
with allure.step("Start storage services on all nodes"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Delete objects from bucket"):
|
|
||||||
with allure.step("Delete simple object from bucket"):
|
|
||||||
with expect_not_raises():
|
|
||||||
s3_client.delete_object(bucket, simple_object_key)
|
|
||||||
|
|
||||||
with allure.step("Delete complex object from bucket"):
|
|
||||||
with expect_not_raises():
|
|
||||||
s3_client.delete_object(bucket, complex_object_key)
|
|
||||||
|
|
||||||
with allure.step("Delete bucket"):
|
|
||||||
with expect_not_raises():
|
|
||||||
s3_client.delete_bucket(bucket)
|
|
||||||
|
|
||||||
@allure.title(
|
|
||||||
"Write cache loss on one node should not affect shards and should not produce errors in log"
|
|
||||||
)
|
|
||||||
@pytest.mark.write_cache_loss
|
|
||||||
def test_write_cache_loss_on_one_node(
|
|
||||||
self,
|
|
||||||
node_under_test: ClusterNode,
|
|
||||||
simple_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
shards_watcher: ShardsWatcher,
|
|
||||||
default_wallet: str,
|
|
||||||
test_start_time: datetime,
|
|
||||||
after_run_return_all_stopped_services: str,
|
|
||||||
):
|
|
||||||
exception_messages = []
|
|
||||||
allure.dynamic.description(after_run_return_all_stopped_services)
|
|
||||||
|
|
||||||
with allure.step(f"Create container on node {node_under_test}"):
|
|
||||||
locode = node_under_test.storage_node.get_un_locode()
|
|
||||||
placement_rule = f"""REP 1 IN X
|
|
||||||
CBF 1
|
|
||||||
SELECT 1 FROM C AS X
|
|
||||||
FILTER 'UN-LOCODE' EQ '{locode}' AS C"""
|
|
||||||
cid = create_container(
|
|
||||||
default_wallet,
|
|
||||||
self.shell,
|
|
||||||
node_under_test.storage_node.get_rpc_endpoint(),
|
|
||||||
rule=placement_rule,
|
|
||||||
)
|
|
||||||
container = StorageContainer(
|
|
||||||
StorageContainerInfo(cid, WalletInfo(default_wallet)),
|
|
||||||
self.shell,
|
|
||||||
cluster_state_controller.cluster,
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step(f"Put couple objects to container on node {node_under_test}"):
|
|
||||||
storage_objects: list[StorageObjectInfo] = []
|
|
||||||
for _ in range(5):
|
|
||||||
storage_object = container.generate_object(
|
|
||||||
simple_object_size, endpoint=node_under_test.storage_node.get_rpc_endpoint()
|
|
||||||
)
|
|
||||||
storage_objects.append(storage_object)
|
|
||||||
|
|
||||||
with allure.step("Take shards snapshot"):
|
|
||||||
shards_watcher.take_shards_snapshot()
|
|
||||||
|
|
||||||
with allure.step(f"Stop storage service on node {node_under_test}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node_under_test)
|
|
||||||
|
|
||||||
with allure.step(f"Delete write cache from node {node_under_test}"):
|
|
||||||
node_under_test.storage_node.delete_write_cache()
|
|
||||||
|
|
||||||
with allure.step(f"Start storage service on node {node_under_test}"):
|
|
||||||
cluster_state_controller.start_storage_service(node_under_test)
|
|
||||||
|
|
||||||
with allure.step("Objects should be available"):
|
|
||||||
for storage_object in storage_objects:
|
|
||||||
get_object(
|
|
||||||
storage_object.wallet_file_path,
|
|
||||||
container.get_id(),
|
|
||||||
storage_object.oid,
|
|
||||||
self.shell,
|
|
||||||
node_under_test.storage_node.get_rpc_endpoint(),
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("No shards should have new errors"):
|
|
||||||
shards_watcher.take_shards_snapshot()
|
|
||||||
shards_with_errors = shards_watcher.get_shards_with_new_errors()
|
|
||||||
if shards_with_errors:
|
|
||||||
exception_messages.append(f"Shards have new errors: {shards_with_errors}")
|
|
||||||
|
|
||||||
with allure.step("No shards should have degraded status"):
|
|
||||||
snapshot = shards_watcher.get_shards_snapshot()
|
|
||||||
for shard in snapshot:
|
|
||||||
status = snapshot[shard]["mode"]
|
|
||||||
if status != "read-write":
|
|
||||||
exception_messages.append(f"Shard {shard} changed status to {status}")
|
|
||||||
|
|
||||||
with allure.step("No related errors should be in log"):
|
|
||||||
if node_under_test.host.is_message_in_logs(
|
|
||||||
message_regex=r"\Wno such file or directory\W", since=test_start_time
|
|
||||||
):
|
|
||||||
exception_messages.append(f"Node {node_under_test} have shard errors in logs")
|
|
||||||
|
|
||||||
with allure.step("Pass test if no errors found"):
|
|
||||||
assert not exception_messages, "\n".join(exception_messages)
|
|
||||||
|
|
||||||
@allure.title(
|
|
||||||
"Test S3 Loss of one node should trigger use of tree and storage service in another node"
|
|
||||||
)
|
|
||||||
def test_s3_one_endpoint_loss(
|
|
||||||
self,
|
|
||||||
bucket,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
after_run_return_all_stopped_services,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
# TODO: need to check that s3 gate is connected to localhost (such metric will be supported in 1.3)
|
|
||||||
|
|
||||||
with allure.step(
|
|
||||||
"Stop one node and wait for rebalance connection of s3 gate to storage service"
|
|
||||||
):
|
|
||||||
current_node = self.cluster.cluster_nodes[0]
|
|
||||||
cluster_state_controller.stop_storage_service(current_node)
|
|
||||||
# waiting for rebalance connection of s3 gate to storage service
|
|
||||||
sleep(60)
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
with allure.step("Put object into one bucket"):
|
|
||||||
put_object = s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
|
|
||||||
@allure.title("After Pilorama.db loss on one node object are retrievable")
|
|
||||||
def test_s3_one_pilorama_loss(
|
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
simple_object_size: int,
|
|
||||||
cluster_state_controller: ClusterStateController,
|
|
||||||
):
|
|
||||||
bucket = s3_client.create_bucket(
|
|
||||||
location_constraint="load-1-4",
|
|
||||||
grant_read="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
|
||||||
)
|
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
|
||||||
|
|
||||||
with allure.step("Check bucket versioning"):
|
|
||||||
bucket_versioning = s3_client.get_bucket_versioning_status(bucket)
|
|
||||||
assert bucket_versioning == "Enabled", "Bucket should have enabled versioning"
|
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
|
||||||
|
|
||||||
object_versions = []
|
|
||||||
with allure.step("Put object into one bucket"):
|
|
||||||
put_object = s3_client.put_object(bucket, file_path)
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, expected_objects=[file_name])
|
|
||||||
object_versions.append(put_object)
|
|
||||||
|
|
||||||
node_to_check = self.cluster.storage_nodes[0]
|
|
||||||
piloramas_list_before_removing = {}
|
|
||||||
with allure.step("Get list of all pilorama.db"):
|
|
||||||
piloramas_list_before_removing = self.get_piloramas_list(
|
|
||||||
node_to_check, cluster_state_controller
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Stop all storage nodes"):
|
|
||||||
for node in self.cluster.cluster_nodes:
|
|
||||||
with allure.step(f"Stop storage service on node: {node}"):
|
|
||||||
cluster_state_controller.stop_storage_service(node)
|
|
||||||
|
|
||||||
with allure.step("Delete pilorama.db from one node"):
|
|
||||||
node_to_check.delete_pilorama()
|
|
||||||
|
|
||||||
with allure.step("Start all storage nodes"):
|
|
||||||
cluster_state_controller.start_stopped_storage_services()
|
|
||||||
|
|
||||||
with allure.step("Tick epoch to trigger sync and then wait for 1 minute"):
|
|
||||||
self.tick_epochs(1)
|
|
||||||
sleep(120)
|
|
||||||
|
|
||||||
piloramas_list_afrer_removing = {}
|
|
||||||
with allure.step("Get list of all pilorama.db after sync"):
|
|
||||||
piloramas_list_afrer_removing = self.get_piloramas_list(
|
|
||||||
node_to_check, cluster_state_controller
|
|
||||||
)
|
|
||||||
assert (
|
|
||||||
piloramas_list_afrer_removing == piloramas_list_before_removing
|
|
||||||
), "List of pilorama.db is different"
|
|
||||||
|
|
||||||
with allure.step("Check bucket versioning"):
|
|
||||||
bucket_versioning = s3_client.get_bucket_versioning_status(bucket)
|
|
||||||
assert bucket_versioning == "Enabled", "Bucket should have enabled versioning"
|
|
||||||
|
|
||||||
with allure.step("Check list objects"):
|
|
||||||
objects_list = s3_client.list_objects(bucket)
|
|
||||||
assert objects_list, f"Expected not empty bucket"
|
|
||||||
|
|
||||||
with allure.step("Delete the object from the bucket"):
|
|
||||||
delete_object = s3_client.delete_object(bucket, file_name)
|
|
||||||
assert "DeleteMarker" in delete_object.keys(), "Delete markers not found"
|
|
||||||
|
|
||||||
with allure.step("Check list objects"):
|
|
||||||
objects_list = s3_client.list_objects_versions(bucket)
|
|
||||||
assert objects_list, f"Expected not empty bucket"
|
|
||||||
object_versions.append(delete_object["VersionId"])
|
|
||||||
|
|
||||||
# and now delete all versions of object (including Delete Markers)
|
|
||||||
with allure.step("Delete all versions of the object from the bucket"):
|
|
||||||
for version in object_versions:
|
|
||||||
delete_object = s3_client.delete_object(bucket, file_name, version_id=version)
|
|
||||||
|
|
||||||
with allure.step("Check list objects"):
|
|
||||||
objects_list = s3_client.list_objects_versions(bucket)
|
|
||||||
assert not objects_list, f"Expected empty bucket"
|
|
||||||
|
|
||||||
with allure.step("Delete bucket"):
|
|
||||||
s3_client.delete_bucket(bucket)
|
|
||||||
|
|
123
pytest_tests/testsuites/load/test_load.py
Normal file
123
pytest_tests/testsuites/load/test_load.py
Normal file
|
@ -0,0 +1,123 @@
|
||||||
|
import allure
|
||||||
|
import pytest
|
||||||
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from common import (
|
||||||
|
HTTP_GATE_SERVICE_NAME_REGEX,
|
||||||
|
S3_GATE_SERVICE_NAME_REGEX,
|
||||||
|
STORAGE_NODE_SERVICE_NAME_REGEX,
|
||||||
|
)
|
||||||
|
from k6 import LoadParams
|
||||||
|
from load import (
|
||||||
|
clear_cache_and_data,
|
||||||
|
get_services_endpoints,
|
||||||
|
init_s3_client,
|
||||||
|
multi_node_k6_run,
|
||||||
|
prepare_k6_instances,
|
||||||
|
start_stopped_nodes,
|
||||||
|
stop_unused_nodes,
|
||||||
|
)
|
||||||
|
from load_params import (
|
||||||
|
CONTAINER_PLACEMENT_POLICY,
|
||||||
|
CONTAINERS_COUNT,
|
||||||
|
DELETERS,
|
||||||
|
LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
|
LOAD_NODE_SSH_USER,
|
||||||
|
LOAD_NODES,
|
||||||
|
LOAD_NODES_COUNT,
|
||||||
|
LOAD_TIME,
|
||||||
|
LOAD_TYPE,
|
||||||
|
OBJ_COUNT,
|
||||||
|
OBJ_SIZE,
|
||||||
|
OUT_FILE,
|
||||||
|
READERS,
|
||||||
|
STORAGE_NODE_COUNT,
|
||||||
|
WRITERS,
|
||||||
|
)
|
||||||
|
from neofs_testlib.hosting import Hosting
|
||||||
|
|
||||||
|
ENDPOINTS_ATTRIBUTES = {
|
||||||
|
"http": {"regex": HTTP_GATE_SERVICE_NAME_REGEX, "endpoint_attribute": "endpoint"},
|
||||||
|
"grpc": {"regex": STORAGE_NODE_SERVICE_NAME_REGEX, "endpoint_attribute": "rpc_endpoint"},
|
||||||
|
"s3": {"regex": S3_GATE_SERVICE_NAME_REGEX, "endpoint_attribute": "endpoint"},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.load
|
||||||
|
class TestLoad(ClusterTestBase):
|
||||||
|
@pytest.fixture(autouse=True)
|
||||||
|
def clear_cache_and_data(self, hosting: Hosting):
|
||||||
|
clear_cache_and_data(hosting=hosting)
|
||||||
|
yield
|
||||||
|
start_stopped_nodes()
|
||||||
|
|
||||||
|
@pytest.fixture(scope="session", autouse=True)
|
||||||
|
def init_s3_client(self, hosting: Hosting):
|
||||||
|
if "s3" in list(map(lambda x: x.lower(), LOAD_TYPE)):
|
||||||
|
init_s3_client(
|
||||||
|
load_nodes=LOAD_NODES,
|
||||||
|
login=LOAD_NODE_SSH_USER,
|
||||||
|
pkey=LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
|
hosting=hosting,
|
||||||
|
container_placement_policy=CONTAINER_PLACEMENT_POLICY,
|
||||||
|
)
|
||||||
|
|
||||||
|
@pytest.mark.parametrize("obj_size, out_file", list(zip(OBJ_SIZE, OUT_FILE)))
|
||||||
|
@pytest.mark.parametrize("writers, readers, deleters", list(zip(WRITERS, READERS, DELETERS)))
|
||||||
|
@pytest.mark.parametrize("load_time", LOAD_TIME)
|
||||||
|
@pytest.mark.parametrize("node_count", STORAGE_NODE_COUNT)
|
||||||
|
@pytest.mark.parametrize("containers_count", CONTAINERS_COUNT)
|
||||||
|
@pytest.mark.parametrize("load_type", LOAD_TYPE)
|
||||||
|
@pytest.mark.parametrize("obj_count", OBJ_COUNT)
|
||||||
|
@pytest.mark.parametrize("load_nodes_count", LOAD_NODES_COUNT)
|
||||||
|
@pytest.mark.benchmark
|
||||||
|
@pytest.mark.grpc
|
||||||
|
def test_custom_load(
|
||||||
|
self,
|
||||||
|
obj_size,
|
||||||
|
out_file,
|
||||||
|
writers,
|
||||||
|
readers,
|
||||||
|
deleters,
|
||||||
|
load_time,
|
||||||
|
node_count,
|
||||||
|
obj_count,
|
||||||
|
load_type,
|
||||||
|
load_nodes_count,
|
||||||
|
containers_count,
|
||||||
|
hosting: Hosting,
|
||||||
|
):
|
||||||
|
allure.dynamic.title(
|
||||||
|
f"Load test - node_count = {node_count}, "
|
||||||
|
f"writers = {writers} readers = {readers}, "
|
||||||
|
f"deleters = {deleters}, obj_size = {obj_size}, "
|
||||||
|
f"load_time = {load_time}"
|
||||||
|
)
|
||||||
|
stop_unused_nodes(self.cluster.storage_nodes, node_count)
|
||||||
|
with allure.step("Get endpoints"):
|
||||||
|
endpoints_list = get_services_endpoints(
|
||||||
|
hosting=hosting,
|
||||||
|
service_name_regex=ENDPOINTS_ATTRIBUTES[LOAD_TYPE]["regex"],
|
||||||
|
endpoint_attribute=ENDPOINTS_ATTRIBUTES[LOAD_TYPE]["endpoint_attribute"],
|
||||||
|
)
|
||||||
|
endpoints = ",".join(endpoints_list[:node_count])
|
||||||
|
load_params = LoadParams(
|
||||||
|
endpoint=endpoints,
|
||||||
|
obj_size=obj_size,
|
||||||
|
containers_count=containers_count,
|
||||||
|
out_file=out_file,
|
||||||
|
obj_count=obj_count,
|
||||||
|
writers=writers,
|
||||||
|
readers=readers,
|
||||||
|
deleters=deleters,
|
||||||
|
load_time=load_time,
|
||||||
|
load_type=load_type,
|
||||||
|
)
|
||||||
|
load_nodes_list = LOAD_NODES[:load_nodes_count]
|
||||||
|
k6_load_instances = prepare_k6_instances(
|
||||||
|
load_nodes=load_nodes_list,
|
||||||
|
login=LOAD_NODE_SSH_USER,
|
||||||
|
pkey=LOAD_NODE_SSH_PRIVATE_KEY_PATH,
|
||||||
|
load_params=load_params,
|
||||||
|
)
|
||||||
|
with allure.step("Run load"):
|
||||||
|
multi_node_k6_run(k6_load_instances)
|
|
@ -5,11 +5,15 @@ from typing import Optional, Tuple
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.common import FROSTFS_CONTRACT_CACHE_TIMEOUT, MORPH_BLOCK_TIME
|
from cluster import StorageNode
|
||||||
from frostfs_testlib.resources.error_patterns import OBJECT_NOT_FOUND
|
from cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from common import MORPH_BLOCK_TIME, NEOFS_CONTRACT_CACHE_TIMEOUT
|
||||||
from frostfs_testlib.steps.cli.container import create_container, get_container
|
from epoch import tick_epoch
|
||||||
from frostfs_testlib.steps.cli.object import (
|
from file_helper import generate_file
|
||||||
|
from grpc_responses import OBJECT_NOT_FOUND, error_matches_status
|
||||||
|
from python_keywords.container import create_container, get_container
|
||||||
|
from python_keywords.failover_utils import wait_object_replication
|
||||||
|
from python_keywords.neofs_verbs import (
|
||||||
delete_object,
|
delete_object,
|
||||||
get_object,
|
get_object,
|
||||||
get_object_from_random_node,
|
get_object_from_random_node,
|
||||||
|
@ -17,31 +21,23 @@ from frostfs_testlib.steps.cli.object import (
|
||||||
put_object,
|
put_object,
|
||||||
put_object_to_random_node,
|
put_object_to_random_node,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.epoch import tick_epoch
|
from python_keywords.node_management import (
|
||||||
from frostfs_testlib.steps.node_management import (
|
|
||||||
check_node_in_map,
|
check_node_in_map,
|
||||||
delete_node_data,
|
delete_node_data,
|
||||||
drop_object,
|
drop_object,
|
||||||
exclude_node_from_network_map,
|
exclude_node_from_network_map,
|
||||||
get_locode_from_random_node,
|
get_locode_from_random_node,
|
||||||
|
get_netmap_snapshot,
|
||||||
include_node_to_network_map,
|
include_node_to_network_map,
|
||||||
node_shard_list,
|
node_shard_list,
|
||||||
node_shard_set_mode,
|
node_shard_set_mode,
|
||||||
|
start_storage_nodes,
|
||||||
storage_node_healthcheck,
|
storage_node_healthcheck,
|
||||||
storage_node_set_status,
|
storage_node_set_status,
|
||||||
wait_for_node_to_be_ready,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.storage_policy import get_nodes_with_object, get_simple_object_copies
|
|
||||||
from frostfs_testlib.storage.cluster import StorageNode
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils import datetime_utils, string_utils
|
|
||||||
from frostfs_testlib.utils.failover_utils import wait_object_replication
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
|
||||||
from pytest_tests.helpers.utility import (
|
|
||||||
placement_policy_from_container,
|
|
||||||
wait_for_gc_pass_on_storage_nodes,
|
|
||||||
)
|
)
|
||||||
|
from storage_policy import get_nodes_with_object, get_simple_object_copies
|
||||||
|
from utility import parse_time, placement_policy_from_container, wait_for_gc_pass_on_storage_nodes
|
||||||
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
check_nodes: list[StorageNode] = []
|
check_nodes: list[StorageNode] = []
|
||||||
|
@ -111,17 +107,17 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
with allure.step(f"Start node {node}"):
|
with allure.step(f"Start node {node}"):
|
||||||
node.start_service()
|
node.start_service()
|
||||||
with allure.step(f"Waiting status ready for node {node}"):
|
with allure.step(f"Waiting status ready for node {node}"):
|
||||||
wait_for_node_to_be_ready(node)
|
self.wait_for_node_to_be_ready(node)
|
||||||
|
|
||||||
# We need to wait for node to establish notifications from morph-chain
|
# We need to wait for node to establish notifications from morph-chain
|
||||||
# Otherwise it will hang up when we will try to set status
|
# Otherwise it will hang up when we will try to set status
|
||||||
sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME))
|
sleep(parse_time(MORPH_BLOCK_TIME))
|
||||||
|
|
||||||
with allure.step(f"Move node {node} to online state"):
|
with allure.step(f"Move node {node} to online state"):
|
||||||
storage_node_set_status(node, status="online", retries=2)
|
storage_node_set_status(node, status="online", retries=2)
|
||||||
|
|
||||||
check_nodes.remove(node)
|
check_nodes.remove(node)
|
||||||
sleep(datetime_utils.parse_time(MORPH_BLOCK_TIME))
|
sleep(parse_time(MORPH_BLOCK_TIME))
|
||||||
self.tick_epoch_with_retries(3)
|
self.tick_epoch_with_retries(3)
|
||||||
check_node_in_map(node, shell=self.shell, alive_node=alive_node)
|
check_node_in_map(node, shell=self.shell, alive_node=alive_node)
|
||||||
|
|
||||||
|
@ -134,7 +130,7 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
simple_object_size,
|
simple_object_size,
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
This test remove one node from frostfs_testlib.storage.cluster then add it back. Test uses base control operations with storage nodes (healthcheck, netmap-snapshot, set-status).
|
This test remove one node from cluster then add it back. Test uses base control operations with storage nodes (healthcheck, netmap-snapshot, set-status).
|
||||||
"""
|
"""
|
||||||
wallet = default_wallet
|
wallet = default_wallet
|
||||||
placement_rule_3 = "REP 3 IN X CBF 1 SELECT 3 FROM * AS X"
|
placement_rule_3 = "REP 3 IN X CBF 1 SELECT 3 FROM * AS X"
|
||||||
|
@ -328,10 +324,10 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
self.validate_object_copies(wallet, placement_rule, file_path, expected_copies)
|
self.validate_object_copies(wallet, placement_rule, file_path, expected_copies)
|
||||||
|
|
||||||
@pytest.mark.node_mgmt
|
@pytest.mark.node_mgmt
|
||||||
@allure.title("FrostFS object could be dropped using control command")
|
@allure.title("NeoFS object could be dropped using control command")
|
||||||
def test_drop_object(self, default_wallet, complex_object_size: int, simple_object_size: int):
|
def test_drop_object(self, default_wallet, complex_object_size, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test checks object could be dropped using `frostfs-cli control drop-objects` command.
|
Test checks object could be dropped using `neofs-cli control drop-objects` command.
|
||||||
"""
|
"""
|
||||||
wallet = default_wallet
|
wallet = default_wallet
|
||||||
endpoint = self.cluster.default_rpc_endpoint
|
endpoint = self.cluster.default_rpc_endpoint
|
||||||
|
@ -413,38 +409,6 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
oid = put_object_to_random_node(wallet, file_path, cid, self.shell, self.cluster)
|
oid = put_object_to_random_node(wallet, file_path, cid, self.shell, self.cluster)
|
||||||
delete_object(wallet, cid, oid, self.shell, self.cluster.default_rpc_endpoint)
|
delete_object(wallet, cid, oid, self.shell, self.cluster.default_rpc_endpoint)
|
||||||
|
|
||||||
@pytest.mark.node_mgmt
|
|
||||||
@allure.title("Put object with stopped node")
|
|
||||||
def test_stop_node(self, default_wallet, return_nodes_after_test_run, simple_object_size: int):
|
|
||||||
wallet = default_wallet
|
|
||||||
placement_rule = "REP 3 SELECT 4 FROM * AS X"
|
|
||||||
source_file_path = generate_file(simple_object_size)
|
|
||||||
storage_nodes = self.cluster.storage_nodes
|
|
||||||
random_node = random.choice(storage_nodes[1:])
|
|
||||||
alive_node = random.choice(
|
|
||||||
[storage_node for storage_node in storage_nodes if storage_node.id != random_node.id]
|
|
||||||
)
|
|
||||||
|
|
||||||
cid = create_container(
|
|
||||||
wallet,
|
|
||||||
rule=placement_rule,
|
|
||||||
basic_acl=PUBLIC_ACL,
|
|
||||||
shell=self.shell,
|
|
||||||
endpoint=random_node.get_rpc_endpoint(),
|
|
||||||
)
|
|
||||||
with allure.step("Stop the random node"):
|
|
||||||
check_nodes.append(random_node)
|
|
||||||
random_node.stop_service()
|
|
||||||
with allure.step("Try to put an object and expect success"):
|
|
||||||
put_object(
|
|
||||||
wallet,
|
|
||||||
source_file_path,
|
|
||||||
cid,
|
|
||||||
shell=self.shell,
|
|
||||||
endpoint=alive_node.get_rpc_endpoint(),
|
|
||||||
)
|
|
||||||
self.return_nodes(alive_node)
|
|
||||||
|
|
||||||
@allure.step("Validate object has {expected_copies} copies")
|
@allure.step("Validate object has {expected_copies} copies")
|
||||||
def validate_object_copies(
|
def validate_object_copies(
|
||||||
self, wallet: str, placement_rule: str, file_path: str, expected_copies: int
|
self, wallet: str, placement_rule: str, file_path: str, expected_copies: int
|
||||||
|
@ -485,6 +449,21 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
f"Node {node} hasn't gone to the READY and ONLINE state after {timeout * attempts} second"
|
f"Node {node} hasn't gone to the READY and ONLINE state after {timeout * attempts} second"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@allure.step("Wait for node {node} is ready")
|
||||||
|
def wait_for_node_to_be_ready(self, node: StorageNode) -> None:
|
||||||
|
timeout, attempts = 30, 6
|
||||||
|
for _ in range(attempts):
|
||||||
|
try:
|
||||||
|
health_check = storage_node_healthcheck(node)
|
||||||
|
if health_check.health_status == "READY":
|
||||||
|
return
|
||||||
|
except Exception as err:
|
||||||
|
logger.warning(f"Node {node} is not ready:\n{err}")
|
||||||
|
sleep(timeout)
|
||||||
|
raise AssertionError(
|
||||||
|
f"Node {node} hasn't gone to the READY state after {timeout * attempts} seconds"
|
||||||
|
)
|
||||||
|
|
||||||
@allure.step("Wait for {expected_copies} object copies in the wallet")
|
@allure.step("Wait for {expected_copies} object copies in the wallet")
|
||||||
def wait_for_expected_object_copies(
|
def wait_for_expected_object_copies(
|
||||||
self, wallet: str, cid: str, oid: str, expected_copies: int = 2
|
self, wallet: str, cid: str, oid: str, expected_copies: int = 2
|
||||||
|
@ -495,7 +474,7 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
if copies == expected_copies:
|
if copies == expected_copies:
|
||||||
break
|
break
|
||||||
tick_epoch(self.shell, self.cluster)
|
tick_epoch(self.shell, self.cluster)
|
||||||
sleep(datetime_utils.parse_time(FROSTFS_CONTRACT_CACHE_TIMEOUT))
|
sleep(parse_time(NEOFS_CONTRACT_CACHE_TIMEOUT))
|
||||||
else:
|
else:
|
||||||
raise AssertionError(f"There are no {expected_copies} copies during time")
|
raise AssertionError(f"There are no {expected_copies} copies during time")
|
||||||
|
|
||||||
|
@ -506,7 +485,7 @@ class TestNodeManagement(ClusterTestBase):
|
||||||
checker(wallet, cid, oid, shell=self.shell, endpoint=endpoint)
|
checker(wallet, cid, oid, shell=self.shell, endpoint=endpoint)
|
||||||
wait_for_gc_pass_on_storage_nodes()
|
wait_for_gc_pass_on_storage_nodes()
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
if string_utils.is_str_match_pattern(err, OBJECT_NOT_FOUND):
|
if error_matches_status(err, OBJECT_NOT_FOUND):
|
||||||
return
|
return
|
||||||
raise AssertionError(f'Expected "{OBJECT_NOT_FOUND}" error, got\n{err}')
|
raise AssertionError(f'Expected "{OBJECT_NOT_FOUND}" error, got\n{err}')
|
||||||
|
|
||||||
|
|
|
@ -4,16 +4,20 @@ import sys
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.error_patterns import (
|
from cluster import Cluster
|
||||||
|
from complex_object_actions import get_complex_object_split_ranges
|
||||||
|
from file_helper import generate_file, get_file_content, get_file_hash
|
||||||
|
from grpc_responses import (
|
||||||
INVALID_LENGTH_SPECIFIER,
|
INVALID_LENGTH_SPECIFIER,
|
||||||
INVALID_OFFSET_SPECIFIER,
|
INVALID_OFFSET_SPECIFIER,
|
||||||
INVALID_RANGE_OVERFLOW,
|
INVALID_RANGE_OVERFLOW,
|
||||||
INVALID_RANGE_ZERO_LENGTH,
|
INVALID_RANGE_ZERO_LENGTH,
|
||||||
OUT_OF_RANGE,
|
OUT_OF_RANGE,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.shell import Shell
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from pytest import FixtureRequest
|
||||||
from frostfs_testlib.steps.cli.object import (
|
from python_keywords.container import create_container
|
||||||
|
from python_keywords.neofs_verbs import (
|
||||||
get_object_from_random_node,
|
get_object_from_random_node,
|
||||||
get_range,
|
get_range,
|
||||||
get_range_hash,
|
get_range_hash,
|
||||||
|
@ -21,14 +25,11 @@ from frostfs_testlib.steps.cli.object import (
|
||||||
put_object_to_random_node,
|
put_object_to_random_node,
|
||||||
search_object,
|
search_object,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.complex_object_actions import get_complex_object_split_ranges
|
from python_keywords.storage_policy import get_complex_object_copies, get_simple_object_copies
|
||||||
from frostfs_testlib.steps.storage_object import delete_objects
|
|
||||||
from frostfs_testlib.steps.storage_policy import get_complex_object_copies, get_simple_object_copies
|
from helpers.storage_object_info import StorageObjectInfo
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import StorageObjectInfo
|
from steps.storage_object import delete_objects
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_content, get_file_hash
|
|
||||||
from pytest import FixtureRequest
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
|
@ -1,24 +1,19 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import EACL_PUBLIC_READ_WRITE
|
from cluster import Cluster
|
||||||
from frostfs_testlib.shell import Shell
|
from container import REP_2_FOR_3_NODES_PLACEMENT_RULE, SINGLE_PLACEMENT_RULE, create_container
|
||||||
from frostfs_testlib.steps.acl import form_bearertoken_file
|
from epoch import get_epoch
|
||||||
from frostfs_testlib.steps.cli.container import (
|
from neofs_testlib.shell import Shell
|
||||||
REP_2_FOR_3_NODES_PLACEMENT_RULE,
|
from neofs_verbs import delete_object, get_object
|
||||||
SINGLE_PLACEMENT_RULE,
|
|
||||||
StorageContainer,
|
|
||||||
StorageContainerInfo,
|
|
||||||
create_container,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.cli.object import delete_object, get_object
|
|
||||||
from frostfs_testlib.steps.epoch import get_epoch
|
|
||||||
from frostfs_testlib.steps.storage_object import StorageObjectInfo
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLAccess, EACLOperation, EACLRole, EACLRule
|
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import expect_not_raises
|
|
||||||
from pytest import FixtureRequest
|
from pytest import FixtureRequest
|
||||||
|
from python_keywords.acl import EACLAccess, EACLOperation, EACLRole, EACLRule, form_bearertoken_file
|
||||||
|
from wellknown_acl import EACL_PUBLIC_READ_WRITE
|
||||||
|
|
||||||
|
from helpers.container import StorageContainer, StorageContainerInfo
|
||||||
|
from helpers.test_control import expect_not_raises
|
||||||
|
from helpers.wallet import WalletFile
|
||||||
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
from steps.storage_object import StorageObjectInfo
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
|
@ -51,9 +46,9 @@ def user_container(
|
||||||
endpoint=cluster.default_rpc_endpoint,
|
endpoint=cluster.default_rpc_endpoint,
|
||||||
)
|
)
|
||||||
# Deliberately using s3gate wallet here to test bearer token
|
# Deliberately using s3gate wallet here to test bearer token
|
||||||
s3gate = cluster.s3_gates[0]
|
s3gate = cluster.s3gates[0]
|
||||||
return StorageContainer(
|
return StorageContainer(
|
||||||
StorageContainerInfo(container_id, WalletInfo.from_node(s3gate)),
|
StorageContainerInfo(container_id, WalletFile.from_node(s3gate)),
|
||||||
client_shell,
|
client_shell,
|
||||||
cluster,
|
cluster,
|
||||||
)
|
)
|
||||||
|
@ -106,7 +101,7 @@ class TestObjectApiWithBearerToken(ClusterTestBase):
|
||||||
f"Object can be deleted from any node using s3gate wallet with bearer token for {request.node.callspec.id}"
|
f"Object can be deleted from any node using s3gate wallet with bearer token for {request.node.callspec.id}"
|
||||||
)
|
)
|
||||||
|
|
||||||
s3_gate_wallet = self.cluster.s3_gates[0]
|
s3_gate_wallet = self.cluster.s3gates[0]
|
||||||
with allure.step("Try to delete each object from first storage node"):
|
with allure.step("Try to delete each object from first storage node"):
|
||||||
for storage_object in storage_objects:
|
for storage_object in storage_objects:
|
||||||
with expect_not_raises():
|
with expect_not_raises():
|
||||||
|
@ -142,7 +137,7 @@ class TestObjectApiWithBearerToken(ClusterTestBase):
|
||||||
f"Object can be fetched from any node using s3gate wallet with bearer token for {request.node.callspec.id}"
|
f"Object can be fetched from any node using s3gate wallet with bearer token for {request.node.callspec.id}"
|
||||||
)
|
)
|
||||||
|
|
||||||
s3_gate_wallet = self.cluster.s3_gates[0]
|
s3_gate_wallet = self.cluster.s3gates[0]
|
||||||
with allure.step("Put one object to container"):
|
with allure.step("Put one object to container"):
|
||||||
epoch = self.get_epoch()
|
epoch = self.get_epoch()
|
||||||
storage_object = user_container.generate_object(
|
storage_object = user_container.generate_object(
|
||||||
|
|
|
@ -2,15 +2,15 @@ import logging
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.error_patterns import OBJECT_NOT_FOUND
|
from epoch import get_epoch, tick_epoch
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file, get_file_hash
|
||||||
from frostfs_testlib.steps.cli.object import get_object_from_random_node, put_object_to_random_node, head_object
|
from grpc_responses import OBJECT_NOT_FOUND
|
||||||
from frostfs_testlib.steps.epoch import get_epoch
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_hash
|
|
||||||
from pytest import FixtureRequest
|
from pytest import FixtureRequest
|
||||||
|
from python_keywords.container import create_container
|
||||||
|
from python_keywords.neofs_verbs import get_object_from_random_node, put_object_to_random_node
|
||||||
|
from utility import wait_for_gc_pass_on_storage_nodes
|
||||||
|
|
||||||
from pytest_tests.helpers.utility import wait_for_gc_pass_on_storage_nodes
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -54,19 +54,6 @@ class TestObjectApiLifetime(ClusterTestBase):
|
||||||
# Wait for GC, because object with expiration is counted as alive until GC removes it
|
# Wait for GC, because object with expiration is counted as alive until GC removes it
|
||||||
wait_for_gc_pass_on_storage_nodes()
|
wait_for_gc_pass_on_storage_nodes()
|
||||||
|
|
||||||
with allure.step("Check object deleted because it expires on epoch"):
|
with allure.step("Check object deleted because it expires-on epoch"):
|
||||||
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
|
||||||
head_object(wallet, cid, oid, self.shell, self.cluster.default_rpc_endpoint)
|
|
||||||
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
|
||||||
get_object_from_random_node(wallet, cid, oid, self.shell, self.cluster)
|
|
||||||
|
|
||||||
with allure.step("Tick additional epoch"):
|
|
||||||
self.tick_epoch()
|
|
||||||
|
|
||||||
wait_for_gc_pass_on_storage_nodes()
|
|
||||||
|
|
||||||
with allure.step("Check object deleted because it expires on previous epoch"):
|
|
||||||
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
|
||||||
head_object(wallet, cid, oid, self.shell, self.cluster.default_rpc_endpoint)
|
|
||||||
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
||||||
get_object_from_random_node(wallet, cid, oid, self.shell, self.cluster)
|
get_object_from_random_node(wallet, cid, oid, self.shell, self.cluster)
|
||||||
|
|
|
@ -3,8 +3,12 @@ import re
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.common import STORAGE_GC_TIME
|
from cluster import Cluster
|
||||||
from frostfs_testlib.resources.error_patterns import (
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from common import STORAGE_GC_TIME
|
||||||
|
from complex_object_actions import get_link_object, get_storage_object_chunks
|
||||||
|
from epoch import ensure_fresh_epoch, get_epoch, tick_epoch
|
||||||
|
from grpc_responses import (
|
||||||
LIFETIME_REQUIRED,
|
LIFETIME_REQUIRED,
|
||||||
LOCK_NON_REGULAR_OBJECT,
|
LOCK_NON_REGULAR_OBJECT,
|
||||||
LOCK_OBJECT_EXPIRATION,
|
LOCK_OBJECT_EXPIRATION,
|
||||||
|
@ -13,30 +17,19 @@ from frostfs_testlib.resources.error_patterns import (
|
||||||
OBJECT_IS_LOCKED,
|
OBJECT_IS_LOCKED,
|
||||||
OBJECT_NOT_FOUND,
|
OBJECT_NOT_FOUND,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.shell import Shell
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.steps.cli.container import (
|
from node_management import drop_object
|
||||||
StorageContainer,
|
|
||||||
StorageContainerInfo,
|
|
||||||
create_container,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.cli.object import delete_object, head_object, lock_object
|
|
||||||
from frostfs_testlib.steps.complex_object_actions import get_link_object, get_storage_object_chunks
|
|
||||||
from frostfs_testlib.steps.epoch import ensure_fresh_epoch, get_epoch, tick_epoch
|
|
||||||
from frostfs_testlib.steps.node_management import drop_object
|
|
||||||
from frostfs_testlib.steps.storage_object import delete_objects
|
|
||||||
from frostfs_testlib.steps.storage_policy import get_nodes_with_object
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import (
|
|
||||||
LockObjectInfo,
|
|
||||||
StorageObjectInfo,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletFactory, WalletInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import expect_not_raises, wait_for_success
|
|
||||||
from frostfs_testlib.utils import datetime_utils
|
|
||||||
from pytest import FixtureRequest
|
from pytest import FixtureRequest
|
||||||
|
from python_keywords.container import create_container
|
||||||
|
from python_keywords.neofs_verbs import delete_object, head_object, lock_object
|
||||||
|
from storage_policy import get_nodes_with_object
|
||||||
|
from test_control import expect_not_raises, wait_for_success
|
||||||
|
from utility import parse_time, wait_for_gc_pass_on_storage_nodes
|
||||||
|
|
||||||
from pytest_tests.helpers.utility import wait_for_gc_pass_on_storage_nodes
|
from helpers.container import StorageContainer, StorageContainerInfo
|
||||||
|
from helpers.storage_object_info import LockObjectInfo, StorageObjectInfo
|
||||||
|
from helpers.wallet import WalletFactory, WalletFile
|
||||||
|
from steps.storage_object import delete_objects
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -56,7 +49,7 @@ def user_wallet(wallet_factory: WalletFactory):
|
||||||
@pytest.fixture(
|
@pytest.fixture(
|
||||||
scope="module",
|
scope="module",
|
||||||
)
|
)
|
||||||
def user_container(user_wallet: WalletInfo, client_shell: Shell, cluster: Cluster):
|
def user_container(user_wallet: WalletFile, client_shell: Shell, cluster: Cluster):
|
||||||
container_id = create_container(
|
container_id = create_container(
|
||||||
user_wallet.path, shell=client_shell, endpoint=cluster.default_rpc_endpoint
|
user_wallet.path, shell=client_shell, endpoint=cluster.default_rpc_endpoint
|
||||||
)
|
)
|
||||||
|
@ -328,7 +321,7 @@ class TestObjectLockWithGrpc(ClusterTestBase):
|
||||||
self.cluster.default_rpc_endpoint,
|
self.cluster.default_rpc_endpoint,
|
||||||
)
|
)
|
||||||
|
|
||||||
@wait_for_success(datetime_utils.parse_time(STORAGE_GC_TIME))
|
@wait_for_success(parse_time(STORAGE_GC_TIME))
|
||||||
def check_object_not_found():
|
def check_object_not_found():
|
||||||
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
with pytest.raises(Exception, match=OBJECT_NOT_FOUND):
|
||||||
head_object(
|
head_object(
|
||||||
|
|
101
pytest_tests/testsuites/payment/test_balance.py
Normal file
101
pytest_tests/testsuites/payment/test_balance.py
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import pytest
|
||||||
|
import yaml
|
||||||
|
from cluster_test_base import ClusterTestBase
|
||||||
|
from common import FREE_STORAGE, NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import CommandResult, Shell
|
||||||
|
from wallet import WalletFactory, WalletFile
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
DEPOSIT_AMOUNT = 30
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.sanity
|
||||||
|
@pytest.mark.payments
|
||||||
|
@pytest.mark.skipif(FREE_STORAGE, reason="Test only works on public network with paid storage")
|
||||||
|
class TestBalanceAccounting(ClusterTestBase):
|
||||||
|
@pytest.fixture(scope="class")
|
||||||
|
def main_wallet(self, wallet_factory: WalletFactory) -> WalletFile:
|
||||||
|
return wallet_factory.create_wallet()
|
||||||
|
|
||||||
|
@pytest.fixture(scope="class")
|
||||||
|
def other_wallet(self, wallet_factory: WalletFactory) -> WalletFile:
|
||||||
|
return wallet_factory.create_wallet()
|
||||||
|
|
||||||
|
@pytest.fixture(scope="class")
|
||||||
|
def cli(self, client_shell: Shell) -> NeofsCli:
|
||||||
|
return NeofsCli(client_shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
|
||||||
|
@allure.step("Check deposit amount")
|
||||||
|
def check_amount(self, result: CommandResult) -> None:
|
||||||
|
amount_str = result.stdout.rstrip()
|
||||||
|
|
||||||
|
try:
|
||||||
|
amount = int(amount_str)
|
||||||
|
except Exception as ex:
|
||||||
|
pytest.fail(
|
||||||
|
f"Amount parse error, should be parsable as int({DEPOSIT_AMOUNT}), but given {amount_str}: {ex}"
|
||||||
|
)
|
||||||
|
|
||||||
|
assert amount == DEPOSIT_AMOUNT
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
@allure.step("Write config with API endpoint")
|
||||||
|
def write_api_config(config_dir: str, endpoint: str, wallet: str) -> str:
|
||||||
|
with open(WALLET_CONFIG, "r") as file:
|
||||||
|
wallet_config = yaml.full_load(file)
|
||||||
|
api_config = {
|
||||||
|
**wallet_config,
|
||||||
|
"rpc-endpoint": endpoint,
|
||||||
|
"wallet": wallet,
|
||||||
|
}
|
||||||
|
api_config_file = os.path.join(config_dir, "neofs-cli-api-config.yaml")
|
||||||
|
with open(api_config_file, "w") as file:
|
||||||
|
yaml.dump(api_config, file)
|
||||||
|
return api_config_file
|
||||||
|
|
||||||
|
@allure.title("Test balance request with wallet and address")
|
||||||
|
def test_balance_wallet_address(self, main_wallet: WalletFile, cli: NeofsCli):
|
||||||
|
result = cli.accounting.balance(
|
||||||
|
wallet=main_wallet.path,
|
||||||
|
rpc_endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
address=main_wallet.get_address(),
|
||||||
|
)
|
||||||
|
|
||||||
|
self.check_amount(result)
|
||||||
|
|
||||||
|
@allure.title("Test balance request with wallet only")
|
||||||
|
def test_balance_wallet(self, main_wallet: WalletFile, cli: NeofsCli):
|
||||||
|
result = cli.accounting.balance(
|
||||||
|
wallet=main_wallet.path, rpc_endpoint=self.cluster.default_rpc_endpoint
|
||||||
|
)
|
||||||
|
self.check_amount(result)
|
||||||
|
|
||||||
|
@allure.title("Test balance request with wallet and wrong address")
|
||||||
|
def test_balance_wrong_address(
|
||||||
|
self, main_wallet: WalletFile, other_wallet: WalletFile, cli: NeofsCli
|
||||||
|
):
|
||||||
|
with pytest.raises(Exception, match="address option must be specified and valid"):
|
||||||
|
cli.accounting.balance(
|
||||||
|
wallet=main_wallet.path,
|
||||||
|
rpc_endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
address=other_wallet.get_address(),
|
||||||
|
)
|
||||||
|
|
||||||
|
@allure.title("Test balance request with config file")
|
||||||
|
def test_balance_api(self, temp_directory: str, main_wallet: WalletFile, client_shell: Shell):
|
||||||
|
config_file = self.write_api_config(
|
||||||
|
config_dir=temp_directory,
|
||||||
|
endpoint=self.cluster.default_rpc_endpoint,
|
||||||
|
wallet=main_wallet.path,
|
||||||
|
)
|
||||||
|
logger.info(f"Config with API endpoint: {config_file}")
|
||||||
|
|
||||||
|
cli = NeofsCli(client_shell, NEOFS_CLI_EXEC, config_file=config_file)
|
||||||
|
result = cli.accounting.balance()
|
||||||
|
|
||||||
|
self.check_amount(result)
|
|
@ -2,8 +2,14 @@ import logging
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from container import create_container
|
||||||
from frostfs_testlib.steps.acl import (
|
from file_helper import generate_file
|
||||||
|
from http_gate import get_object_and_verify_hashes, upload_via_http_gate_curl
|
||||||
|
from python_keywords.acl import (
|
||||||
|
EACLAccess,
|
||||||
|
EACLOperation,
|
||||||
|
EACLRole,
|
||||||
|
EACLRule,
|
||||||
bearer_token_base64_from_file,
|
bearer_token_base64_from_file,
|
||||||
create_eacl,
|
create_eacl,
|
||||||
form_bearertoken_file,
|
form_bearertoken_file,
|
||||||
|
@ -11,16 +17,13 @@ from frostfs_testlib.steps.acl import (
|
||||||
sign_bearer,
|
sign_bearer,
|
||||||
wait_for_cache_expired,
|
wait_for_cache_expired,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from wellknown_acl import PUBLIC_ACL
|
||||||
from frostfs_testlib.steps.http.http_gate import upload_via_http_gate_curl, verify_object_hash
|
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLAccess, EACLOperation, EACLRole, EACLRule
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.skip("Temporarly skip for v0.37")
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.http_gate
|
@pytest.mark.http_gate
|
||||||
class Test_http_bearer(ClusterTestBase):
|
class Test_http_bearer(ClusterTestBase):
|
||||||
|
@ -92,7 +95,6 @@ class Test_http_bearer(ClusterTestBase):
|
||||||
error_pattern="access to object operation denied",
|
error_pattern="access to object operation denied",
|
||||||
)
|
)
|
||||||
|
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"object_size",
|
"object_size",
|
||||||
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
|
@ -118,7 +120,7 @@ class Test_http_bearer(ClusterTestBase):
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
headers=headers,
|
headers=headers,
|
||||||
)
|
)
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
|
|
@ -1,13 +1,14 @@
|
||||||
|
import logging
|
||||||
import os
|
import os
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from epoch import get_epoch, tick_epoch
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file, get_file_hash
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.steps.epoch import get_epoch
|
from python_keywords.http_gate import (
|
||||||
from frostfs_testlib.steps.http.http_gate import (
|
|
||||||
attr_into_header,
|
attr_into_header,
|
||||||
|
get_object_and_verify_hashes,
|
||||||
get_object_by_attr_and_verify_hashes,
|
get_object_by_attr_and_verify_hashes,
|
||||||
get_via_http_curl,
|
get_via_http_curl,
|
||||||
get_via_http_gate,
|
get_via_http_gate,
|
||||||
|
@ -15,22 +16,22 @@ from frostfs_testlib.steps.http.http_gate import (
|
||||||
try_to_get_object_and_expect_error,
|
try_to_get_object_and_expect_error,
|
||||||
upload_via_http_gate,
|
upload_via_http_gate,
|
||||||
upload_via_http_gate_curl,
|
upload_via_http_gate_curl,
|
||||||
verify_object_hash,
|
|
||||||
)
|
)
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_hash
|
from utility import wait_for_gc_pass_on_storage_nodes
|
||||||
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
from pytest_tests.helpers.utility import wait_for_gc_pass_on_storage_nodes
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
OBJECT_NOT_FOUND_ERROR = "not found"
|
OBJECT_NOT_FOUND_ERROR = "not found"
|
||||||
|
|
||||||
|
|
||||||
@allure.link(
|
@allure.link(
|
||||||
"https://github.com/TrueCloudLab/frostfs-http-gw#frostfs-http-gateway",
|
"https://github.com/nspcc-dev/neofs-http-gw#neofs-http-gateway", name="neofs-http-gateway"
|
||||||
name="frostfs-http-gateway",
|
|
||||||
)
|
)
|
||||||
@allure.link("https://github.com/TrueCloudLab/frostfs-http-gw#uploading", name="uploading")
|
@allure.link("https://github.com/nspcc-dev/neofs-http-gw#uploading", name="uploading")
|
||||||
@allure.link("https://github.com/TrueCloudLab/frostfs-http-gw#downloading", name="downloading")
|
@allure.link("https://github.com/nspcc-dev/neofs-http-gw#downloading", name="downloading")
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.http_gate
|
@pytest.mark.http_gate
|
||||||
class TestHttpGate(ClusterTestBase):
|
class TestHttpGate(ClusterTestBase):
|
||||||
|
@ -43,15 +44,15 @@ class TestHttpGate(ClusterTestBase):
|
||||||
TestHttpGate.wallet = default_wallet
|
TestHttpGate.wallet = default_wallet
|
||||||
|
|
||||||
@allure.title("Test Put over gRPC, Get over HTTP")
|
@allure.title("Test Put over gRPC, Get over HTTP")
|
||||||
def test_put_grpc_get_http(self, complex_object_size: int, simple_object_size: int):
|
def test_put_grpc_get_http(self, complex_object_size, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test that object can be put using gRPC interface and get using HTTP.
|
Test that object can be put using gRPC interface and get using HTTP.
|
||||||
|
|
||||||
Steps:
|
Steps:
|
||||||
1. Create simple and large objects.
|
1. Create simple and large objects.
|
||||||
2. Put objects using gRPC (frostfs-cli).
|
2. Put objects using gRPC (neofs-cli).
|
||||||
3. Download objects using HTTP gate (https://github.com/TrueCloudLab/frostfs-http-gw#downloading).
|
3. Download objects using HTTP gate (https://github.com/nspcc-dev/neofs-http-gw#downloading).
|
||||||
4. Get objects using gRPC (frostfs-cli).
|
4. Get objects using gRPC (neofs-cli).
|
||||||
5. Compare hashes for got objects.
|
5. Compare hashes for got objects.
|
||||||
6. Compare hashes for got and original objects.
|
6. Compare hashes for got and original objects.
|
||||||
|
|
||||||
|
@ -86,7 +87,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
@ -96,18 +97,18 @@ class TestHttpGate(ClusterTestBase):
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.link("https://github.com/TrueCloudLab/frostfs-http-gw#uploading", name="uploading")
|
@allure.link("https://github.com/nspcc-dev/neofs-http-gw#uploading", name="uploading")
|
||||||
@allure.link("https://github.com/TrueCloudLab/frostfs-http-gw#downloading", name="downloading")
|
@allure.link("https://github.com/nspcc-dev/neofs-http-gw#downloading", name="downloading")
|
||||||
@allure.title("Test Put over HTTP, Get over HTTP")
|
@allure.title("Test Put over HTTP, Get over HTTP")
|
||||||
@pytest.mark.smoke
|
@pytest.mark.smoke
|
||||||
def test_put_http_get_http(self, complex_object_size: int, simple_object_size: int):
|
def test_put_http_get_http(self, complex_object_size, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test that object can be put and get using HTTP interface.
|
Test that object can be put and get using HTTP interface.
|
||||||
|
|
||||||
Steps:
|
Steps:
|
||||||
1. Create simple and large objects.
|
1. Create simple and large objects.
|
||||||
2. Upload objects using HTTP (https://github.com/TrueCloudLab/frostfs-http-gw#uploading).
|
2. Upload objects using HTTP (https://github.com/nspcc-dev/neofs-http-gw#uploading).
|
||||||
3. Download objects using HTTP gate (https://github.com/TrueCloudLab/frostfs-http-gw#downloading).
|
3. Download objects using HTTP gate (https://github.com/nspcc-dev/neofs-http-gw#downloading).
|
||||||
4. Compare hashes for got and original objects.
|
4. Compare hashes for got and original objects.
|
||||||
|
|
||||||
Expected result:
|
Expected result:
|
||||||
|
@ -133,7 +134,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
@ -144,8 +145,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.link(
|
@allure.link(
|
||||||
"https://github.com/TrueCloudLab/frostfs-http-gw#by-attributes",
|
"https://github.com/nspcc-dev/neofs-http-gw#by-attributes", name="download by attributes"
|
||||||
name="download by attributes",
|
|
||||||
)
|
)
|
||||||
@allure.title("Test Put over HTTP, Get over HTTP with headers")
|
@allure.title("Test Put over HTTP, Get over HTTP with headers")
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
|
@ -157,14 +157,14 @@ class TestHttpGate(ClusterTestBase):
|
||||||
],
|
],
|
||||||
ids=["simple", "hyphen", "percent"],
|
ids=["simple", "hyphen", "percent"],
|
||||||
)
|
)
|
||||||
def test_put_http_get_http_with_headers(self, attributes: dict, simple_object_size: int):
|
def test_put_http_get_http_with_headers(self, attributes: dict, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test that object can be downloaded using different attributes in HTTP header.
|
Test that object can be downloaded using different attributes in HTTP header.
|
||||||
|
|
||||||
Steps:
|
Steps:
|
||||||
1. Create simple and large objects.
|
1. Create simple and large objects.
|
||||||
2. Upload objects using HTTP with particular attributes in the header.
|
2. Upload objects using HTTP with particular attributes in the header.
|
||||||
3. Download objects by attributes using HTTP gate (https://github.com/TrueCloudLab/frostfs-http-gw#by-attributes).
|
3. Download objects by attributes using HTTP gate (https://github.com/nspcc-dev/neofs-http-gw#by-attributes).
|
||||||
4. Compare hashes for got and original objects.
|
4. Compare hashes for got and original objects.
|
||||||
|
|
||||||
Expected result:
|
Expected result:
|
||||||
|
@ -197,11 +197,9 @@ class TestHttpGate(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test Expiration-Epoch in HTTP header")
|
@allure.title("Test Expiration-Epoch in HTTP header")
|
||||||
@pytest.mark.parametrize("epoch_gap", [0, 1])
|
def test_expiration_epoch_in_http(self, simple_object_size):
|
||||||
def test_expiration_epoch_in_http(self, simple_object_size: int, epoch_gap):
|
|
||||||
endpoint = self.cluster.default_rpc_endpoint
|
endpoint = self.cluster.default_rpc_endpoint
|
||||||
http_endpoint = self.cluster.default_http_gate_endpoint
|
http_endpoint = self.cluster.default_http_gate_endpoint
|
||||||
min_valid_epoch = get_epoch(self.shell, self.cluster) + epoch_gap
|
|
||||||
|
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
self.wallet,
|
self.wallet,
|
||||||
|
@ -211,43 +209,47 @@ class TestHttpGate(ClusterTestBase):
|
||||||
basic_acl=PUBLIC_ACL,
|
basic_acl=PUBLIC_ACL,
|
||||||
)
|
)
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
oids_to_be_expired = []
|
oids = []
|
||||||
oids_to_be_valid = []
|
|
||||||
|
|
||||||
for gap_until in (0, 1, 2, 100):
|
curr_epoch = get_epoch(self.shell, self.cluster)
|
||||||
valid_until = min_valid_epoch + gap_until
|
epochs = (curr_epoch, curr_epoch + 1, curr_epoch + 2, curr_epoch + 100)
|
||||||
headers = {"X-Attribute-System-Expiration-Epoch": str(valid_until)}
|
|
||||||
|
for epoch in epochs:
|
||||||
|
headers = {"X-Attribute-Neofs-Expiration-Epoch": str(epoch)}
|
||||||
|
|
||||||
with allure.step("Put objects using HTTP with attribute Expiration-Epoch"):
|
with allure.step("Put objects using HTTP with attribute Expiration-Epoch"):
|
||||||
oid = upload_via_http_gate(
|
oids.append(
|
||||||
cid=cid, path=file_path, headers=headers, endpoint=http_endpoint
|
upload_via_http_gate(
|
||||||
|
cid=cid, path=file_path, headers=headers, endpoint=http_endpoint
|
||||||
|
)
|
||||||
)
|
)
|
||||||
if get_epoch(self.shell, self.cluster) + 1 <= valid_until:
|
|
||||||
oids_to_be_valid.append(oid)
|
|
||||||
else:
|
|
||||||
oids_to_be_expired.append(oid)
|
|
||||||
with allure.step("This object can be got"):
|
|
||||||
get_via_http_gate(cid=cid, oid=oid, endpoint=http_endpoint)
|
|
||||||
|
|
||||||
self.tick_epoch()
|
assert len(oids) == len(epochs), "Expected all objects have been put successfully"
|
||||||
|
|
||||||
# Wait for GC, because object with expiration is counted as alive until GC removes it
|
with allure.step("All objects can be get"):
|
||||||
wait_for_gc_pass_on_storage_nodes()
|
for oid in oids:
|
||||||
|
get_via_http_gate(cid=cid, oid=oid, endpoint=http_endpoint)
|
||||||
|
|
||||||
for oid in oids_to_be_expired:
|
for expired_objects, not_expired_objects in [(oids[:1], oids[1:]), (oids[:2], oids[2:])]:
|
||||||
with allure.step(f"{oid} shall be expired and cannot be got"):
|
self.tick_epoch()
|
||||||
|
|
||||||
|
# Wait for GC, because object with expiration is counted as alive until GC removes it
|
||||||
|
wait_for_gc_pass_on_storage_nodes()
|
||||||
|
|
||||||
|
for oid in expired_objects:
|
||||||
try_to_get_object_and_expect_error(
|
try_to_get_object_and_expect_error(
|
||||||
cid=cid,
|
cid=cid,
|
||||||
oid=oid,
|
oid=oid,
|
||||||
error_pattern=OBJECT_NOT_FOUND_ERROR,
|
error_pattern=OBJECT_NOT_FOUND_ERROR,
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
)
|
)
|
||||||
for oid in oids_to_be_valid:
|
|
||||||
with allure.step(f"{oid} shall be valid and can be got"):
|
with allure.step("Other objects can be get"):
|
||||||
get_via_http_gate(cid=cid, oid=oid, endpoint=http_endpoint)
|
for oid in not_expired_objects:
|
||||||
|
get_via_http_gate(cid=cid, oid=oid, endpoint=http_endpoint)
|
||||||
|
|
||||||
@allure.title("Test Zip in HTTP header")
|
@allure.title("Test Zip in HTTP header")
|
||||||
def test_zip_in_http(self, complex_object_size: int, simple_object_size: int):
|
def test_zip_in_http(self, complex_object_size, simple_object_size):
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
self.wallet,
|
self.wallet,
|
||||||
shell=self.shell,
|
shell=self.shell,
|
||||||
|
@ -285,9 +287,8 @@ class TestHttpGate(ClusterTestBase):
|
||||||
assert get_file_hash(f"{dir_path}/file2") == get_file_hash(file_path_large)
|
assert get_file_hash(f"{dir_path}/file2") == get_file_hash(file_path_large)
|
||||||
|
|
||||||
@pytest.mark.long
|
@pytest.mark.long
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
@allure.title("Test Put over HTTP/Curl, Get over HTTP/Curl for large object")
|
@allure.title("Test Put over HTTP/Curl, Get over HTTP/Curl for large object")
|
||||||
def test_put_http_get_http_large_file(self, complex_object_size: int):
|
def test_put_http_get_http_large_file(self, complex_object_size):
|
||||||
"""
|
"""
|
||||||
This test checks upload and download using curl with 'large' object.
|
This test checks upload and download using curl with 'large' object.
|
||||||
Large is object with size up to 20Mb.
|
Large is object with size up to 20Mb.
|
||||||
|
@ -313,7 +314,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
)
|
)
|
||||||
|
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid_gate,
|
oid=oid_gate,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
@ -322,7 +323,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
nodes=self.cluster.storage_nodes,
|
nodes=self.cluster.storage_nodes,
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
)
|
)
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid_curl,
|
oid=oid_curl,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
@ -333,9 +334,8 @@ class TestHttpGate(ClusterTestBase):
|
||||||
object_getter=get_via_http_curl,
|
object_getter=get_via_http_curl,
|
||||||
)
|
)
|
||||||
|
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
@allure.title("Test Put/Get over HTTP using Curl utility")
|
@allure.title("Test Put/Get over HTTP using Curl utility")
|
||||||
def test_put_http_get_http_curl(self, complex_object_size: int, simple_object_size: int):
|
def test_put_http_get_http_curl(self, complex_object_size, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test checks upload and download over HTTP using curl utility.
|
Test checks upload and download over HTTP using curl utility.
|
||||||
"""
|
"""
|
||||||
|
@ -361,7 +361,7 @@ class TestHttpGate(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
for oid, file_path in ((oid_simple, file_path_simple), (oid_large, file_path_large)):
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
|
|
@ -3,25 +3,27 @@ import os
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from container import (
|
||||||
from frostfs_testlib.steps.cli.container import (
|
|
||||||
create_container,
|
create_container,
|
||||||
delete_container,
|
delete_container,
|
||||||
list_containers,
|
list_containers,
|
||||||
wait_for_container_deletion,
|
wait_for_container_deletion,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.cli.object import delete_object
|
from epoch import tick_epoch
|
||||||
from frostfs_testlib.steps.http.http_gate import (
|
from file_helper import generate_file
|
||||||
|
from http_gate import (
|
||||||
attr_into_str_header_curl,
|
attr_into_str_header_curl,
|
||||||
get_object_by_attr_and_verify_hashes,
|
get_object_by_attr_and_verify_hashes,
|
||||||
try_to_get_object_and_expect_error,
|
try_to_get_object_and_expect_error,
|
||||||
try_to_get_object_via_passed_request_and_expect_error,
|
try_to_get_object_via_passed_request_and_expect_error,
|
||||||
upload_via_http_gate_curl,
|
upload_via_http_gate_curl,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import StorageObjectInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
from pytest import FixtureRequest
|
from pytest import FixtureRequest
|
||||||
|
from python_keywords.neofs_verbs import delete_object
|
||||||
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
|
from helpers.storage_object_info import StorageObjectInfo
|
||||||
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
OBJECT_ALREADY_REMOVED_ERROR = "object already removed"
|
OBJECT_ALREADY_REMOVED_ERROR = "object already removed"
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
@ -47,12 +49,9 @@ class Test_http_headers(ClusterTestBase):
|
||||||
@pytest.fixture(
|
@pytest.fixture(
|
||||||
params=[
|
params=[
|
||||||
pytest.lazy_fixture("simple_object_size"),
|
pytest.lazy_fixture("simple_object_size"),
|
||||||
# TODO: Temp disable for v0.37
|
pytest.lazy_fixture("complex_object_size"),
|
||||||
# pytest.lazy_fixture("complex_object_size"),
|
|
||||||
],
|
],
|
||||||
# TODO: Temp disable for v0.37
|
ids=["simple object", "complex object"],
|
||||||
# ids=["simple object", "complex object"],
|
|
||||||
ids=["simple object"],
|
|
||||||
scope="class",
|
scope="class",
|
||||||
)
|
)
|
||||||
def storage_objects_with_attributes(self, request: FixtureRequest) -> list[StorageObjectInfo]:
|
def storage_objects_with_attributes(self, request: FixtureRequest) -> list[StorageObjectInfo]:
|
||||||
|
|
|
@ -1,17 +1,19 @@
|
||||||
import logging
|
import logging
|
||||||
|
import os
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from container import create_container
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.cli.object import put_object_to_random_node
|
from http_gate import (
|
||||||
from frostfs_testlib.steps.http.http_gate import (
|
get_object_and_verify_hashes,
|
||||||
get_object_by_attr_and_verify_hashes,
|
get_object_by_attr_and_verify_hashes,
|
||||||
try_to_get_object_via_passed_request_and_expect_error,
|
try_to_get_object_via_passed_request_and_expect_error,
|
||||||
verify_object_hash,
|
|
||||||
)
|
)
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.neofs_verbs import put_object_to_random_node
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -38,8 +40,8 @@ class Test_http_object(ClusterTestBase):
|
||||||
|
|
||||||
Steps:
|
Steps:
|
||||||
1. Create object;
|
1. Create object;
|
||||||
2. Put objects using gRPC (frostfs-cli) with attributes [--attributes chapter1=peace,chapter2=war];
|
2. Put objects using gRPC (neofs-cli) with attributes [--attributes chapter1=peace,chapter2=war];
|
||||||
3. Download object using HTTP gate (https://github.com/TrueCloudLab/frostfs-http-gw#downloading);
|
3. Download object using HTTP gate (https://github.com/nspcc-dev/neofs-http-gw#downloading);
|
||||||
4. Compare hashes between original and downloaded object;
|
4. Compare hashes between original and downloaded object;
|
||||||
5. [Negative] Try to the get object with specified attributes and `get` request: [get/$CID/chapter1/peace];
|
5. [Negative] Try to the get object with specified attributes and `get` request: [get/$CID/chapter1/peace];
|
||||||
6. Download the object with specified attributes and `get_by_attribute` request: [get_by_attribute/$CID/chapter1/peace];
|
6. Download the object with specified attributes and `get_by_attribute` request: [get_by_attribute/$CID/chapter1/peace];
|
||||||
|
@ -50,7 +52,6 @@ class Test_http_object(ClusterTestBase):
|
||||||
Expected result:
|
Expected result:
|
||||||
Hashes must be the same.
|
Hashes must be the same.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
with allure.step("Create public container"):
|
with allure.step("Create public container"):
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
self.wallet,
|
self.wallet,
|
||||||
|
@ -83,7 +84,7 @@ class Test_http_object(ClusterTestBase):
|
||||||
attributes=f"{key_value1},{key_value2}",
|
attributes=f"{key_value1},{key_value2}",
|
||||||
)
|
)
|
||||||
with allure.step("Get object and verify hashes [ get/$CID/$OID ]"):
|
with allure.step("Get object and verify hashes [ get/$CID/$OID ]"):
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
|
|
@ -2,11 +2,12 @@ import logging
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from container import create_container
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.http.http_gate import upload_via_http_gate_curl, verify_object_hash
|
from http_gate import get_object_and_verify_hashes, upload_via_http_gate_curl
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from wellknown_acl import PUBLIC_ACL
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -27,7 +28,6 @@ class Test_http_streaming(ClusterTestBase):
|
||||||
[pytest.lazy_fixture("complex_object_size")],
|
[pytest.lazy_fixture("complex_object_size")],
|
||||||
ids=["complex object"],
|
ids=["complex object"],
|
||||||
)
|
)
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
def test_object_can_be_put_get_by_streaming(self, object_size: int):
|
def test_object_can_be_put_get_by_streaming(self, object_size: int):
|
||||||
"""
|
"""
|
||||||
Test that object can be put using gRPC interface and get using HTTP.
|
Test that object can be put using gRPC interface and get using HTTP.
|
||||||
|
@ -35,7 +35,7 @@ class Test_http_streaming(ClusterTestBase):
|
||||||
Steps:
|
Steps:
|
||||||
1. Create big object;
|
1. Create big object;
|
||||||
2. Put object using curl with pipe (streaming);
|
2. Put object using curl with pipe (streaming);
|
||||||
3. Download object using HTTP gate (https://github.com/TrueCloudLab/frostfs-http-gw#downloading);
|
3. Download object using HTTP gate (https://github.com/nspcc-dev/neofs-http-gw#downloading);
|
||||||
4. Compare hashes between original and downloaded object;
|
4. Compare hashes between original and downloaded object;
|
||||||
|
|
||||||
Expected result:
|
Expected result:
|
||||||
|
@ -59,7 +59,7 @@ class Test_http_streaming(ClusterTestBase):
|
||||||
oid = upload_via_http_gate_curl(
|
oid = upload_via_http_gate_curl(
|
||||||
cid=cid, filepath=file_path, endpoint=self.cluster.default_http_gate_endpoint
|
cid=cid, filepath=file_path, endpoint=self.cluster.default_http_gate_endpoint
|
||||||
)
|
)
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
|
|
@ -5,33 +5,30 @@ from typing import Optional
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.error_patterns import OBJECT_NOT_FOUND
|
from container import create_container
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from epoch import get_epoch, wait_for_epochs_align
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.cli.object import (
|
from grpc_responses import OBJECT_NOT_FOUND
|
||||||
get_netmap_netinfo,
|
from http_gate import (
|
||||||
get_object_from_random_node,
|
|
||||||
head_object,
|
|
||||||
)
|
|
||||||
from frostfs_testlib.steps.epoch import get_epoch, wait_for_epochs_align
|
|
||||||
from frostfs_testlib.steps.http.http_gate import (
|
|
||||||
attr_into_str_header_curl,
|
attr_into_str_header_curl,
|
||||||
|
get_object_and_verify_hashes,
|
||||||
try_to_get_object_and_expect_error,
|
try_to_get_object_and_expect_error,
|
||||||
upload_via_http_gate_curl,
|
upload_via_http_gate_curl,
|
||||||
verify_object_hash,
|
|
||||||
)
|
)
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.neofs_verbs import get_netmap_netinfo, get_object_from_random_node, head_object
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
from wellknown_acl import PUBLIC_ACL
|
||||||
|
|
||||||
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
EXPIRATION_TIMESTAMP_HEADER = "__SYSTEM__EXPIRATION_TIMESTAMP"
|
EXPIRATION_TIMESTAMP_HEADER = "__NEOFS__EXPIRATION_TIMESTAMP"
|
||||||
EXPIRATION_EPOCH_HEADER = "__SYSTEM__EXPIRATION_EPOCH"
|
EXPIRATION_EPOCH_HEADER = "__NEOFS__EXPIRATION_EPOCH"
|
||||||
EXPIRATION_DURATION_HEADER = "__SYSTEM__EXPIRATION_DURATION"
|
EXPIRATION_DURATION_HEADER = "__NEOFS__EXPIRATION_DURATION"
|
||||||
EXPIRATION_EXPIRATION_RFC = "__SYSTEM__EXPIRATION_RFC3339"
|
EXPIRATION_EXPIRATION_RFC = "__NEOFS__EXPIRATION_RFC3339"
|
||||||
SYSTEM_EXPIRATION_EPOCH = "System-Expiration-Epoch"
|
NEOFS_EXPIRATION_EPOCH = "Neofs-Expiration-Epoch"
|
||||||
SYSTEM_EXPIRATION_DURATION = "System-Expiration-Duration"
|
NEOFS_EXPIRATION_DURATION = "Neofs-Expiration-Duration"
|
||||||
SYSTEM_EXPIRATION_TIMESTAMP = "System-Expiration-Timestamp"
|
NEOFS_EXPIRATION_TIMESTAMP = "Neofs-Expiration-Timestamp"
|
||||||
SYSTEM_EXPIRATION_RFC3339 = "System-Expiration-RFC3339"
|
NEOFS_EXIPRATION_RFC3339 = "Neofs-Expiration-RFC3339"
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -97,7 +94,7 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
f"Validate that only {EXPIRATION_EPOCH_HEADER} exists in header and other headers are abesent"
|
f"Validate that only {EXPIRATION_EPOCH_HEADER} exists in header and other headers are abesent"
|
||||||
)
|
)
|
||||||
def validation_for_http_header_attr(self, head_info: dict, expected_epoch: int) -> None:
|
def validation_for_http_header_attr(self, head_info: dict, expected_epoch: int) -> None:
|
||||||
# check that __SYSTEM__EXPIRATION_EPOCH attribute has corresponding epoch
|
# check that __NEOFS__EXPIRATION_EPOCH attribute has corresponding epoch
|
||||||
assert self.check_key_value_presented_header(
|
assert self.check_key_value_presented_header(
|
||||||
head_info, {EXPIRATION_EPOCH_HEADER: str(expected_epoch)}
|
head_info, {EXPIRATION_EPOCH_HEADER: str(expected_epoch)}
|
||||||
), f'Expected to find {EXPIRATION_EPOCH_HEADER}: {expected_epoch} in: {head_info["header"]["attributes"]}'
|
), f'Expected to find {EXPIRATION_EPOCH_HEADER}: {expected_epoch} in: {head_info["header"]["attributes"]}'
|
||||||
|
@ -122,7 +119,7 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
headers=attr_into_str_header_curl(attributes),
|
headers=attr_into_str_header_curl(attributes),
|
||||||
)
|
)
|
||||||
verify_object_hash(
|
get_object_and_verify_hashes(
|
||||||
oid=oid,
|
oid=oid,
|
||||||
file_name=file_path,
|
file_name=file_path,
|
||||||
wallet=self.wallet,
|
wallet=self.wallet,
|
||||||
|
@ -143,7 +140,7 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
@allure.title("[negative] attempt to put object with expired epoch")
|
@allure.title("[negative] attempt to put object with expired epoch")
|
||||||
def test_unable_put_expired_epoch(self, user_container: str, simple_object_size: int):
|
def test_unable_put_expired_epoch(self, user_container: str, simple_object_size: int):
|
||||||
headers = attr_into_str_header_curl(
|
headers = attr_into_str_header_curl(
|
||||||
{"System-Expiration-Epoch": str(get_epoch(self.shell, self.cluster) - 1)}
|
{"Neofs-Expiration-Epoch": str(get_epoch(self.shell, self.cluster) - 1)}
|
||||||
)
|
)
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
with allure.step(
|
with allure.step(
|
||||||
|
@ -154,15 +151,15 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
filepath=file_path,
|
filepath=file_path,
|
||||||
endpoint=self.cluster.default_http_gate_endpoint,
|
endpoint=self.cluster.default_http_gate_endpoint,
|
||||||
headers=headers,
|
headers=headers,
|
||||||
error_pattern="must be greater than current epoch",
|
error_pattern="object has expired",
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("[negative] attempt to put object with negative System-Expiration-Duration")
|
@allure.title("[negative] attempt to put object with negative Neofs-Expiration-Duration")
|
||||||
def test_unable_put_negative_duration(self, user_container: str, simple_object_size: int):
|
def test_unable_put_negative_duration(self, user_container: str, simple_object_size: int):
|
||||||
headers = attr_into_str_header_curl({"System-Expiration-Duration": "-1h"})
|
headers = attr_into_str_header_curl({"Neofs-Expiration-Duration": "-1h"})
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
with allure.step(
|
with allure.step(
|
||||||
"Put object using HTTP with attribute System-Expiration-Duration where duration is negative"
|
"Put object using HTTP with attribute Neofs-Expiration-Duration where duration is negative"
|
||||||
):
|
):
|
||||||
upload_via_http_gate_curl(
|
upload_via_http_gate_curl(
|
||||||
cid=user_container,
|
cid=user_container,
|
||||||
|
@ -173,13 +170,13 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title(
|
@allure.title(
|
||||||
"[negative] attempt to put object with System-Expiration-Timestamp value in the past"
|
"[negative] attempt to put object with Neofs-Expiration-Timestamp value in the past"
|
||||||
)
|
)
|
||||||
def test_unable_put_expired_timestamp(self, user_container: str, simple_object_size: int):
|
def test_unable_put_expired_timestamp(self, user_container: str, simple_object_size: int):
|
||||||
headers = attr_into_str_header_curl({"System-Expiration-Timestamp": "1635075727"})
|
headers = attr_into_str_header_curl({"Neofs-Expiration-Timestamp": "1635075727"})
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
with allure.step(
|
with allure.step(
|
||||||
"Put object using HTTP with attribute System-Expiration-Timestamp where duration is in the past"
|
"Put object using HTTP with attribute Neofs-Expiration-Timestamp where duration is in the past"
|
||||||
):
|
):
|
||||||
upload_via_http_gate_curl(
|
upload_via_http_gate_curl(
|
||||||
cid=user_container,
|
cid=user_container,
|
||||||
|
@ -190,10 +187,10 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title(
|
@allure.title(
|
||||||
"[negative] Put object using HTTP with attribute System-Expiration-RFC3339 where duration is in the past"
|
"[negative] Put object using HTTP with attribute Neofs-Expiration-RFC3339 where duration is in the past"
|
||||||
)
|
)
|
||||||
def test_unable_put_expired_rfc(self, user_container: str, simple_object_size: int):
|
def test_unable_put_expired_rfc(self, user_container: str, simple_object_size: int):
|
||||||
headers = attr_into_str_header_curl({"System-Expiration-RFC3339": "2021-11-22T09:55:49Z"})
|
headers = attr_into_str_header_curl({"Neofs-Expiration-RFC3339": "2021-11-22T09:55:49Z"})
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
upload_via_http_gate_curl(
|
upload_via_http_gate_curl(
|
||||||
cid=user_container,
|
cid=user_container,
|
||||||
|
@ -209,7 +206,6 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
ids=["simple object", "complex object"],
|
ids=["simple object", "complex object"],
|
||||||
)
|
)
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
def test_http_attr_priority_epoch_duration(
|
def test_http_attr_priority_epoch_duration(
|
||||||
self, user_container: str, object_size: int, epoch_duration: int
|
self, user_container: str, object_size: int, epoch_duration: int
|
||||||
):
|
):
|
||||||
|
@ -219,7 +215,7 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
logger.info(
|
logger.info(
|
||||||
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
||||||
)
|
)
|
||||||
attributes = {SYSTEM_EXPIRATION_EPOCH: expected_epoch, SYSTEM_EXPIRATION_DURATION: "1m"}
|
attributes = {NEOFS_EXPIRATION_EPOCH: expected_epoch, NEOFS_EXPIRATION_DURATION: "1m"}
|
||||||
file_path = generate_file(object_size)
|
file_path = generate_file(object_size)
|
||||||
with allure.step(
|
with allure.step(
|
||||||
f"Put objects using HTTP with attributes and head command should display {EXPIRATION_EPOCH_HEADER}: {expected_epoch} attr"
|
f"Put objects using HTTP with attributes and head command should display {EXPIRATION_EPOCH_HEADER}: {expected_epoch} attr"
|
||||||
|
@ -257,7 +253,6 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
ids=["simple object", "complex object"],
|
ids=["simple object", "complex object"],
|
||||||
)
|
)
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
def test_http_attr_priority_dur_timestamp(
|
def test_http_attr_priority_dur_timestamp(
|
||||||
self, user_container: str, object_size: int, epoch_duration: int
|
self, user_container: str, object_size: int, epoch_duration: int
|
||||||
):
|
):
|
||||||
|
@ -268,10 +263,10 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
||||||
)
|
)
|
||||||
attributes = {
|
attributes = {
|
||||||
SYSTEM_EXPIRATION_DURATION: self.epoch_count_into_mins(
|
NEOFS_EXPIRATION_DURATION: self.epoch_count_into_mins(
|
||||||
epoch_duration=epoch_duration, epoch=2
|
epoch_duration=epoch_duration, epoch=2
|
||||||
),
|
),
|
||||||
SYSTEM_EXPIRATION_TIMESTAMP: self.epoch_count_into_timestamp(
|
NEOFS_EXPIRATION_TIMESTAMP: self.epoch_count_into_timestamp(
|
||||||
epoch_duration=epoch_duration, epoch=1
|
epoch_duration=epoch_duration, epoch=1
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
@ -312,7 +307,6 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
ids=["simple object", "complex object"],
|
ids=["simple object", "complex object"],
|
||||||
)
|
)
|
||||||
@pytest.mark.skip("Temp disable for v0.37")
|
|
||||||
def test_http_attr_priority_timestamp_rfc(
|
def test_http_attr_priority_timestamp_rfc(
|
||||||
self, user_container: str, object_size: int, epoch_duration: int
|
self, user_container: str, object_size: int, epoch_duration: int
|
||||||
):
|
):
|
||||||
|
@ -323,10 +317,10 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
||||||
)
|
)
|
||||||
attributes = {
|
attributes = {
|
||||||
SYSTEM_EXPIRATION_TIMESTAMP: self.epoch_count_into_timestamp(
|
NEOFS_EXPIRATION_TIMESTAMP: self.epoch_count_into_timestamp(
|
||||||
epoch_duration=epoch_duration, epoch=2
|
epoch_duration=epoch_duration, epoch=2
|
||||||
),
|
),
|
||||||
SYSTEM_EXPIRATION_RFC3339: self.epoch_count_into_timestamp(
|
NEOFS_EXIPRATION_RFC3339: self.epoch_count_into_timestamp(
|
||||||
epoch_duration=epoch_duration, epoch=1, rfc3339=True
|
epoch_duration=epoch_duration, epoch=1, rfc3339=True
|
||||||
),
|
),
|
||||||
}
|
}
|
||||||
|
@ -362,11 +356,8 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
@allure.title("Test that object is automatically delete when expiration passed")
|
@allure.title("Test that object is automatically delete when expiration passed")
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"object_size",
|
"object_size",
|
||||||
# TODO: Temp disabled for v0.37
|
[pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
||||||
# [pytest.lazy_fixture("simple_object_size"), pytest.lazy_fixture("complex_object_size")],
|
ids=["simple object", "complex object"],
|
||||||
[pytest.lazy_fixture("simple_object_size")],
|
|
||||||
# ids=["simple object", "complex object"],
|
|
||||||
ids=["simple object"],
|
|
||||||
)
|
)
|
||||||
def test_http_rfc_object_unavailable_after_expir(
|
def test_http_rfc_object_unavailable_after_expir(
|
||||||
self, user_container: str, object_size: int, epoch_duration: int
|
self, user_container: str, object_size: int, epoch_duration: int
|
||||||
|
@ -378,7 +369,7 @@ class Test_http_system_header(ClusterTestBase):
|
||||||
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
f"epoch duration={epoch_duration}, current_epoch= {get_epoch(self.shell, self.cluster)} expected_epoch {expected_epoch}"
|
||||||
)
|
)
|
||||||
attributes = {
|
attributes = {
|
||||||
SYSTEM_EXPIRATION_RFC3339: self.epoch_count_into_timestamp(
|
NEOFS_EXIPRATION_RFC3339: self.epoch_count_into_timestamp(
|
||||||
epoch_duration=epoch_duration, epoch=2, rfc3339=True
|
epoch_duration=epoch_duration, epoch=2, rfc3339=True
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,67 +1,94 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from s3_helper import object_key_from_file_path
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
|
def pytest_generate_tests(metafunc):
|
||||||
|
if "s3_client" in metafunc.fixturenames:
|
||||||
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.acl
|
@pytest.mark.acl
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
class TestS3GateACL:
|
class TestS3GateACL(TestS3GateBase):
|
||||||
@allure.title("Test S3: Object ACL")
|
@allure.title("Test S3: Object ACL")
|
||||||
@pytest.mark.parametrize("s3_client", [AwsCliClient], indirect=True)
|
def test_s3_object_ACL(self, bucket, simple_object_size):
|
||||||
def test_s3_object_ACL(self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
|
|
||||||
with allure.step("Put object into bucket, Check ACL is empty"):
|
with allure.step("Put object into bucket, Check ACL is empty"):
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
obj_acl = s3_client.get_object_acl(bucket, file_name)
|
obj_acl = s3_gate_object.get_object_acl_s3(self.s3_client, bucket, file_name)
|
||||||
assert obj_acl == [], f"Expected ACL is empty, got {obj_acl}"
|
assert obj_acl == [], f"Expected ACL is empty, got {obj_acl}"
|
||||||
|
|
||||||
with allure.step("Put object ACL = public-read"):
|
with allure.step("Put object ACL = public-read"):
|
||||||
s3_client.put_object_acl(bucket, file_name, "public-read")
|
s3_gate_object.put_object_acl_s3(self.s3_client, bucket, file_name, "public-read")
|
||||||
obj_acl = s3_client.get_object_acl(bucket, file_name)
|
obj_acl = s3_gate_object.get_object_acl_s3(self.s3_client, bucket, file_name)
|
||||||
s3_helper.assert_s3_acl(acl_grants=obj_acl, permitted_users="AllUsers")
|
obj_permission = [permission.get("Permission") for permission in obj_acl]
|
||||||
|
assert obj_permission == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Put object ACL = private"):
|
with allure.step("Put object ACL = private"):
|
||||||
s3_client.put_object_acl(bucket, file_name, "private")
|
s3_gate_object.put_object_acl_s3(self.s3_client, bucket, file_name, "private")
|
||||||
obj_acl = s3_client.get_object_acl(bucket, file_name)
|
obj_acl = s3_gate_object.get_object_acl_s3(self.s3_client, bucket, file_name)
|
||||||
s3_helper.assert_s3_acl(acl_grants=obj_acl, permitted_users="CanonicalUser")
|
obj_permission = [permission.get("Permission") for permission in obj_acl]
|
||||||
|
assert obj_permission == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for Canonical User is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step(
|
with allure.step(
|
||||||
"Put object with grant-read uri=http://acs.amazonaws.com/groups/global/AllUsers"
|
"Put object with grant-read uri=http://acs.amazonaws.com/groups/global/AllUsers"
|
||||||
):
|
):
|
||||||
s3_client.put_object_acl(
|
s3_gate_object.put_object_acl_s3(
|
||||||
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
file_name,
|
file_name,
|
||||||
grant_read="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
grant_read="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
||||||
)
|
)
|
||||||
obj_acl = s3_client.get_object_acl(bucket, file_name)
|
obj_acl = s3_gate_object.get_object_acl_s3(self.s3_client, bucket, file_name)
|
||||||
s3_helper.assert_s3_acl(acl_grants=obj_acl, permitted_users="AllUsers")
|
obj_permission = [permission.get("Permission") for permission in obj_acl]
|
||||||
|
assert obj_permission == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
@allure.title("Test S3: Bucket ACL")
|
@allure.title("Test S3: Bucket ACL")
|
||||||
@pytest.mark.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
def test_s3_bucket_ACL(self):
|
||||||
def test_s3_bucket_ACL(self, s3_client: S3ClientWrapper):
|
|
||||||
with allure.step("Create bucket with ACL = public-read-write"):
|
with allure.step("Create bucket with ACL = public-read-write"):
|
||||||
bucket = s3_client.create_bucket(
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True, acl="public-read-write")
|
||||||
object_lock_enabled_for_bucket=True, acl="public-read-write"
|
bucket_acl = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket)
|
||||||
)
|
bucket_permission = [permission.get("Permission") for permission in bucket_acl]
|
||||||
bucket_acl = s3_client.get_bucket_acl(bucket)
|
assert bucket_permission == [
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl, permitted_users="AllUsers")
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Change bucket ACL to private"):
|
with allure.step("Change bucket ACL to private"):
|
||||||
s3_client.put_bucket_acl(bucket, acl="private")
|
s3_gate_bucket.put_bucket_acl_s3(self.s3_client, bucket, acl="private")
|
||||||
bucket_acl = s3_client.get_bucket_acl(bucket)
|
bucket_acl = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl, permitted_users="CanonicalUser")
|
bucket_permission = [permission.get("Permission") for permission in bucket_acl]
|
||||||
|
assert bucket_permission == [
|
||||||
|
"FULL_CONTROL"
|
||||||
|
], "Permission for CanonicalUser is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step(
|
with allure.step(
|
||||||
"Change bucket acl to --grant-write uri=http://acs.amazonaws.com/groups/global/AllUsers"
|
"Change bucket acl to --grant-write uri=http://acs.amazonaws.com/groups/global/AllUsers"
|
||||||
):
|
):
|
||||||
s3_client.put_bucket_acl(
|
s3_gate_bucket.put_bucket_acl_s3(
|
||||||
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
grant_write="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
grant_write="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
||||||
)
|
)
|
||||||
bucket_acl = s3_client.get_bucket_acl(bucket)
|
bucket_acl = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl, permitted_users="AllUsers")
|
bucket_permission = [permission.get("Permission") for permission in bucket_acl]
|
||||||
|
assert bucket_permission == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
|
@ -2,132 +2,164 @@ from datetime import datetime, timedelta
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from s3_helper import assert_object_lock_mode, check_objects_in_bucket, object_key_from_file_path
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_bucket
|
@pytest.mark.s3_gate_bucket
|
||||||
class TestS3GateBucket:
|
class TestS3GateBucket(TestS3GateBase):
|
||||||
@allure.title("Test S3: Create Bucket with different ACL")
|
@allure.title("Test S3: Create Bucket with different ACL")
|
||||||
def test_s3_create_bucket_with_ACL(self, s3_client: S3ClientWrapper):
|
def test_s3_create_bucket_with_ACL(self):
|
||||||
|
|
||||||
with allure.step("Create bucket with ACL private"):
|
with allure.step("Create bucket with ACL private"):
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True, acl="private")
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True, acl="private")
|
||||||
bucket_acl = s3_client.get_bucket_acl(bucket)
|
bucket_acl = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl, permitted_users="CanonicalUser")
|
bucket_permission = [permission.get("Permission") for permission in bucket_acl]
|
||||||
|
assert bucket_permission == [
|
||||||
|
"FULL_CONTROL"
|
||||||
|
], "Permission for CanonicalUser is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Create bucket with ACL = public-read"):
|
with allure.step("Create bucket with ACL = public-read"):
|
||||||
bucket_1 = s3_client.create_bucket(
|
bucket_1 = s3_gate_bucket.create_bucket_s3(self.s3_client, True, acl="public-read")
|
||||||
object_lock_enabled_for_bucket=True, acl="public-read"
|
bucket_acl_1 = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket_1)
|
||||||
)
|
bucket_permission_1 = [permission.get("Permission") for permission in bucket_acl_1]
|
||||||
bucket_acl_1 = s3_client.get_bucket_acl(bucket_1)
|
assert bucket_permission_1 == [
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl_1, permitted_users="AllUsers")
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Create bucket with ACL public-read-write"):
|
with allure.step("Create bucket with ACL public-read-write"):
|
||||||
bucket_2 = s3_client.create_bucket(
|
bucket_2 = s3_gate_bucket.create_bucket_s3(
|
||||||
object_lock_enabled_for_bucket=True, acl="public-read-write"
|
self.s3_client, True, acl="public-read-write"
|
||||||
)
|
)
|
||||||
bucket_acl_2 = s3_client.get_bucket_acl(bucket_2)
|
bucket_acl_2 = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket_2)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl_2, permitted_users="AllUsers")
|
bucket_permission_2 = [permission.get("Permission") for permission in bucket_acl_2]
|
||||||
|
assert bucket_permission_2 == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for CanonicalUser is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Create bucket with ACL = authenticated-read"):
|
with allure.step("Create bucket with ACL = authenticated-read"):
|
||||||
bucket_3 = s3_client.create_bucket(
|
bucket_3 = s3_gate_bucket.create_bucket_s3(
|
||||||
object_lock_enabled_for_bucket=True, acl="authenticated-read"
|
self.s3_client, True, acl="authenticated-read"
|
||||||
)
|
)
|
||||||
bucket_acl_3 = s3_client.get_bucket_acl(bucket_3)
|
bucket_acl_3 = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket_3)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl_3, permitted_users="AllUsers")
|
bucket_permission_3 = [permission.get("Permission") for permission in bucket_acl_3]
|
||||||
|
assert bucket_permission_3 == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
@allure.title("Test S3: Create Bucket with different ACL by grand")
|
@allure.title("Test S3: Create Bucket with different ACL by grand")
|
||||||
def test_s3_create_bucket_with_grands(self, s3_client: S3ClientWrapper):
|
def test_s3_create_bucket_with_grands(self):
|
||||||
|
|
||||||
with allure.step("Create bucket with --grant-read"):
|
with allure.step("Create bucket with --grant-read"):
|
||||||
bucket = s3_client.create_bucket(
|
bucket = s3_gate_bucket.create_bucket_s3(
|
||||||
object_lock_enabled_for_bucket=True,
|
self.s3_client,
|
||||||
|
True,
|
||||||
grant_read="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
grant_read="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
||||||
)
|
)
|
||||||
bucket_acl = s3_client.get_bucket_acl(bucket)
|
bucket_acl = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl, permitted_users="AllUsers")
|
bucket_permission = [permission.get("Permission") for permission in bucket_acl]
|
||||||
|
assert bucket_permission == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for CanonicalUser is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Create bucket with --grant-wtite"):
|
with allure.step("Create bucket with --grant-wtite"):
|
||||||
bucket_1 = s3_client.create_bucket(
|
bucket_1 = s3_gate_bucket.create_bucket_s3(
|
||||||
object_lock_enabled_for_bucket=True,
|
self.s3_client,
|
||||||
|
True,
|
||||||
grant_write="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
grant_write="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
||||||
)
|
)
|
||||||
bucket_acl_1 = s3_client.get_bucket_acl(bucket_1)
|
bucket_acl_1 = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket_1)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl_1, permitted_users="AllUsers")
|
bucket_permission_1 = [permission.get("Permission") for permission in bucket_acl_1]
|
||||||
|
assert bucket_permission_1 == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for all groups is FULL_CONTROL"
|
||||||
|
|
||||||
with allure.step("Create bucket with --grant-full-control"):
|
with allure.step("Create bucket with --grant-full-control"):
|
||||||
bucket_2 = s3_client.create_bucket(
|
bucket_2 = s3_gate_bucket.create_bucket_s3(
|
||||||
object_lock_enabled_for_bucket=True,
|
self.s3_client,
|
||||||
|
True,
|
||||||
grant_full_control="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
grant_full_control="uri=http://acs.amazonaws.com/groups/global/AllUsers",
|
||||||
)
|
)
|
||||||
bucket_acl_2 = s3_client.get_bucket_acl(bucket_2)
|
bucket_acl_2 = s3_gate_bucket.get_bucket_acl(self.s3_client, bucket_2)
|
||||||
s3_helper.assert_s3_acl(acl_grants=bucket_acl_2, permitted_users="AllUsers")
|
bucket_permission_2 = [permission.get("Permission") for permission in bucket_acl_2]
|
||||||
|
assert bucket_permission_2 == [
|
||||||
|
"FULL_CONTROL",
|
||||||
|
"FULL_CONTROL",
|
||||||
|
], "Permission for CanonicalUser is FULL_CONTROL"
|
||||||
|
|
||||||
@allure.title("Test S3: create bucket with object lock")
|
@allure.title("Test S3: create bucket with object lock")
|
||||||
def test_s3_bucket_object_lock(self, s3_client: S3ClientWrapper, simple_object_size: int):
|
def test_s3_bucket_object_lock(self, simple_object_size):
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
|
|
||||||
with allure.step("Create bucket with --no-object-lock-enabled-for-bucket"):
|
with allure.step("Create bucket with --no-object-lock-enabled-for-bucket"):
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=False)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, False)
|
||||||
date_obj = datetime.utcnow() + timedelta(days=1)
|
date_obj = datetime.utcnow() + timedelta(days=1)
|
||||||
with pytest.raises(
|
with pytest.raises(
|
||||||
Exception, match=r".*Object Lock configuration does not exist for this bucket.*"
|
Exception, match=r".*Object Lock configuration does not exist for this bucket.*"
|
||||||
):
|
):
|
||||||
# An error occurred (ObjectLockConfigurationNotFoundError) when calling the PutObject operation (reached max retries: 0):
|
# An error occurred (ObjectLockConfigurationNotFoundError) when calling the PutObject operation (reached max retries: 0):
|
||||||
# Object Lock configuration does not exist for this bucket
|
# Object Lock configuration does not exist for this bucket
|
||||||
s3_client.put_object(
|
s3_gate_object.put_object_s3(
|
||||||
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
file_path,
|
file_path,
|
||||||
object_lock_mode="COMPLIANCE",
|
ObjectLockMode="COMPLIANCE",
|
||||||
object_lock_retain_until_date=date_obj.strftime("%Y-%m-%dT%H:%M:%S"),
|
ObjectLockRetainUntilDate=date_obj.strftime("%Y-%m-%dT%H:%M:%S"),
|
||||||
)
|
)
|
||||||
with allure.step("Create bucket with --object-lock-enabled-for-bucket"):
|
with allure.step("Create bucket with --object-lock-enabled-for-bucket"):
|
||||||
bucket_1 = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket_1 = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
date_obj_1 = datetime.utcnow() + timedelta(days=1)
|
date_obj_1 = datetime.utcnow() + timedelta(days=1)
|
||||||
s3_client.put_object(
|
s3_gate_object.put_object_s3(
|
||||||
|
self.s3_client,
|
||||||
bucket_1,
|
bucket_1,
|
||||||
file_path,
|
file_path,
|
||||||
object_lock_mode="COMPLIANCE",
|
ObjectLockMode="COMPLIANCE",
|
||||||
object_lock_retain_until_date=date_obj_1.strftime("%Y-%m-%dT%H:%M:%S"),
|
ObjectLockRetainUntilDate=date_obj_1.strftime("%Y-%m-%dT%H:%M:%S"),
|
||||||
object_lock_legal_hold_status="ON",
|
ObjectLockLegalHoldStatus="ON",
|
||||||
)
|
)
|
||||||
s3_helper.assert_object_lock_mode(
|
assert_object_lock_mode(
|
||||||
s3_client, bucket_1, file_name, "COMPLIANCE", date_obj_1, "ON"
|
self.s3_client, bucket_1, file_name, "COMPLIANCE", date_obj_1, "ON"
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test S3: delete bucket")
|
@allure.title("Test S3: delete bucket")
|
||||||
def test_s3_delete_bucket(self, s3_client: S3ClientWrapper, simple_object_size: int):
|
def test_s3_delete_bucket(self, simple_object_size):
|
||||||
file_path_1 = generate_file(simple_object_size)
|
file_path_1 = generate_file(simple_object_size)
|
||||||
file_name_1 = s3_helper.object_key_from_file_path(file_path_1)
|
file_name_1 = object_key_from_file_path(file_path_1)
|
||||||
file_path_2 = generate_file(simple_object_size)
|
file_path_2 = generate_file(simple_object_size)
|
||||||
file_name_2 = s3_helper.object_key_from_file_path(file_path_2)
|
file_name_2 = object_key_from_file_path(file_path_2)
|
||||||
bucket = s3_client.create_bucket()
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
|
||||||
with allure.step("Put two objects into bucket"):
|
with allure.step("Put two objects into bucket"):
|
||||||
s3_client.put_object(bucket, file_path_1)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path_1)
|
||||||
s3_client.put_object(bucket, file_path_2)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path_2)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [file_name_1, file_name_2])
|
check_objects_in_bucket(self.s3_client, bucket, [file_name_1, file_name_2])
|
||||||
|
|
||||||
with allure.step("Try to delete not empty bucket and get error"):
|
with allure.step("Try to delete not empty bucket and get error"):
|
||||||
with pytest.raises(Exception, match=r".*The bucket you tried to delete is not empty.*"):
|
with pytest.raises(Exception, match=r".*The bucket you tried to delete is not empty.*"):
|
||||||
s3_client.delete_bucket(bucket)
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket)
|
||||||
|
|
||||||
with allure.step("Delete object in bucket"):
|
with allure.step("Delete object in bucket"):
|
||||||
s3_client.delete_object(bucket, file_name_1)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name_1)
|
||||||
s3_client.delete_object(bucket, file_name_2)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name_2)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [])
|
check_objects_in_bucket(self.s3_client, bucket, [])
|
||||||
|
|
||||||
with allure.step("Delete empty bucket"):
|
with allure.step(f"Delete empty bucket"):
|
||||||
s3_client.delete_bucket(bucket)
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket)
|
||||||
with pytest.raises(Exception, match=r".*Not Found.*"):
|
with pytest.raises(Exception, match=r".*Not Found.*"):
|
||||||
s3_client.head_bucket(bucket)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket)
|
||||||
|
|
|
@ -4,149 +4,141 @@ from random import choice, choices
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.common import ASSETS_DIR
|
from aws_cli_client import AwsCliClient
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
from common import ASSETS_DIR
|
||||||
from frostfs_testlib.shell import Shell
|
from epoch import tick_epoch
|
||||||
from frostfs_testlib.steps.epoch import tick_epoch
|
from file_helper import (
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
|
||||||
from frostfs_testlib.utils.file_utils import (
|
|
||||||
generate_file,
|
generate_file,
|
||||||
generate_file_with_content,
|
generate_file_with_content,
|
||||||
get_file_content,
|
get_file_content,
|
||||||
get_file_hash,
|
get_file_hash,
|
||||||
split_file,
|
split_file,
|
||||||
)
|
)
|
||||||
|
from s3_helper import (
|
||||||
|
check_objects_in_bucket,
|
||||||
|
check_tags_by_bucket,
|
||||||
|
check_tags_by_object,
|
||||||
|
set_bucket_versioning,
|
||||||
|
try_to_get_objects_and_expect_error,
|
||||||
|
)
|
||||||
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@allure.link(
|
@allure.link("https://github.com/nspcc-dev/neofs-s3-gw#neofs-s3-gateway", name="neofs-s3-gateway")
|
||||||
"https://github.com/TrueCloudLab/frostfs-s3-gw#frostfs-s3-gw", name="frostfs-s3-gateway"
|
|
||||||
)
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_base
|
@pytest.mark.s3_gate_base
|
||||||
class TestS3Gate:
|
class TestS3Gate(TestS3GateBase):
|
||||||
@allure.title("Test S3 Bucket API")
|
@allure.title("Test S3 Bucket API")
|
||||||
def test_s3_buckets(
|
def test_s3_buckets(self, simple_object_size):
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
client_shell: Shell,
|
|
||||||
cluster: Cluster,
|
|
||||||
simple_object_size: int,
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test base S3 Bucket API (Create/List/Head/Delete).
|
Test base S3 Bucket API (Create/List/Head/Delete).
|
||||||
"""
|
"""
|
||||||
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = self.object_key_from_file_path(file_path)
|
||||||
|
|
||||||
with allure.step("Create buckets"):
|
with allure.step("Create buckets"):
|
||||||
bucket_1 = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket_1 = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket_1, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket_1, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
bucket_2 = s3_client.create_bucket()
|
bucket_2 = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
|
||||||
with allure.step("Check buckets are presented in the system"):
|
with allure.step("Check buckets are presented in the system"):
|
||||||
buckets = s3_client.list_buckets()
|
buckets = s3_gate_bucket.list_buckets_s3(self.s3_client)
|
||||||
assert bucket_1 in buckets, f"Expected bucket {bucket_1} is in the list"
|
assert bucket_1 in buckets, f"Expected bucket {bucket_1} is in the list"
|
||||||
assert bucket_2 in buckets, f"Expected bucket {bucket_2} is in the list"
|
assert bucket_2 in buckets, f"Expected bucket {bucket_2} is in the list"
|
||||||
|
|
||||||
with allure.step("Bucket must be empty"):
|
with allure.step("Bucket must be empty"):
|
||||||
for bucket in (bucket_1, bucket_2):
|
for bucket in (bucket_1, bucket_2):
|
||||||
objects_list = s3_client.list_objects(bucket)
|
objects_list = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
||||||
|
|
||||||
with allure.step("Check buckets are visible with S3 head command"):
|
with allure.step("Check buckets are visible with S3 head command"):
|
||||||
s3_client.head_bucket(bucket_1)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket_1)
|
||||||
s3_client.head_bucket(bucket_2)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket_2)
|
||||||
|
|
||||||
with allure.step("Check we can put/list object with S3 commands"):
|
with allure.step("Check we can put/list object with S3 commands"):
|
||||||
version_id = s3_client.put_object(bucket_1, file_path)
|
version_id = s3_gate_object.put_object_s3(self.s3_client, bucket_1, file_path)
|
||||||
s3_client.head_object(bucket_1, file_name)
|
s3_gate_object.head_object_s3(self.s3_client, bucket_1, file_name)
|
||||||
|
|
||||||
bucket_objects = s3_client.list_objects(bucket_1)
|
bucket_objects = s3_gate_object.list_objects_s3(self.s3_client, bucket_1)
|
||||||
assert (
|
assert (
|
||||||
file_name in bucket_objects
|
file_name in bucket_objects
|
||||||
), f"Expected file {file_name} in objects list {bucket_objects}"
|
), f"Expected file {file_name} in objects list {bucket_objects}"
|
||||||
|
|
||||||
with allure.step("Try to delete not empty bucket and get error"):
|
with allure.step("Try to delete not empty bucket and get error"):
|
||||||
with pytest.raises(Exception, match=r".*The bucket you tried to delete is not empty.*"):
|
with pytest.raises(Exception, match=r".*The bucket you tried to delete is not empty.*"):
|
||||||
s3_client.delete_bucket(bucket_1)
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket_1)
|
||||||
|
|
||||||
s3_client.head_bucket(bucket_1)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket_1)
|
||||||
|
|
||||||
with allure.step(f"Delete empty bucket {bucket_2}"):
|
with allure.step(f"Delete empty bucket {bucket_2}"):
|
||||||
s3_client.delete_bucket(bucket_2)
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket_2)
|
||||||
tick_epoch(client_shell, cluster)
|
tick_epoch(self.shell, self.cluster)
|
||||||
|
|
||||||
with allure.step(f"Check bucket {bucket_2} deleted"):
|
with allure.step(f"Check bucket {bucket_2} deleted"):
|
||||||
with pytest.raises(Exception, match=r".*Not Found.*"):
|
with pytest.raises(Exception, match=r".*Not Found.*"):
|
||||||
s3_client.head_bucket(bucket_2)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket_2)
|
||||||
|
|
||||||
buckets = s3_client.list_buckets()
|
buckets = s3_gate_bucket.list_buckets_s3(self.s3_client)
|
||||||
assert bucket_1 in buckets, f"Expected bucket {bucket_1} is in the list"
|
assert bucket_1 in buckets, f"Expected bucket {bucket_1} is in the list"
|
||||||
assert bucket_2 not in buckets, f"Expected bucket {bucket_2} is not in the list"
|
assert bucket_2 not in buckets, f"Expected bucket {bucket_2} is not in the list"
|
||||||
|
|
||||||
with allure.step(f"Delete object from {bucket_1}"):
|
with allure.step(f"Delete object from {bucket_1}"):
|
||||||
s3_client.delete_object(bucket_1, file_name, version_id)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket_1, file_name, version_id)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_1, expected_objects=[])
|
check_objects_in_bucket(self.s3_client, bucket_1, expected_objects=[])
|
||||||
|
|
||||||
with allure.step(f"Delete bucket {bucket_1}"):
|
with allure.step(f"Delete bucket {bucket_1}"):
|
||||||
s3_client.delete_bucket(bucket_1)
|
s3_gate_bucket.delete_bucket_s3(self.s3_client, bucket_1)
|
||||||
tick_epoch(client_shell, cluster)
|
self.tick_epoch()
|
||||||
|
|
||||||
with allure.step(f"Check bucket {bucket_1} deleted"):
|
with allure.step(f"Check bucket {bucket_1} deleted"):
|
||||||
with pytest.raises(Exception, match=r".*Not Found.*"):
|
with pytest.raises(Exception, match=r".*Not Found.*"):
|
||||||
s3_client.head_bucket(bucket_1)
|
s3_gate_bucket.head_bucket(self.s3_client, bucket_1)
|
||||||
|
|
||||||
@allure.title("Test S3 Object API")
|
@allure.title("Test S3 Object API")
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"file_type", ["simple", "large"], ids=["Simple object", "Large object"]
|
"file_type", ["simple", "large"], ids=["Simple object", "Large object"]
|
||||||
)
|
)
|
||||||
def test_s3_api_object(
|
def test_s3_api_object(self, file_type, two_buckets, simple_object_size, complex_object_size):
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
file_type: str,
|
|
||||||
two_buckets: tuple[str, str],
|
|
||||||
simple_object_size: int,
|
|
||||||
complex_object_size: int,
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test base S3 Object API (Put/Head/List) for simple and large objects.
|
Test base S3 Object API (Put/Head/List) for simple and large objects.
|
||||||
"""
|
"""
|
||||||
file_path = generate_file(
|
file_path = generate_file(
|
||||||
simple_object_size if file_type == "simple" else complex_object_size
|
simple_object_size if file_type == "simple" else complex_object_size
|
||||||
)
|
)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = self.object_key_from_file_path(file_path)
|
||||||
|
|
||||||
bucket_1, bucket_2 = two_buckets
|
bucket_1, bucket_2 = two_buckets
|
||||||
|
|
||||||
for bucket in (bucket_1, bucket_2):
|
for bucket in (bucket_1, bucket_2):
|
||||||
with allure.step("Bucket must be empty"):
|
with allure.step("Bucket must be empty"):
|
||||||
objects_list = s3_client.list_objects(bucket)
|
objects_list = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
||||||
|
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
s3_client.head_object(bucket, file_name)
|
s3_gate_object.head_object_s3(self.s3_client, bucket, file_name)
|
||||||
|
|
||||||
bucket_objects = s3_client.list_objects(bucket)
|
bucket_objects = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert (
|
assert (
|
||||||
file_name in bucket_objects
|
file_name in bucket_objects
|
||||||
), f"Expected file {file_name} in objects list {bucket_objects}"
|
), f"Expected file {file_name} in objects list {bucket_objects}"
|
||||||
|
|
||||||
with allure.step("Check object's attributes"):
|
with allure.step("Check object's attributes"):
|
||||||
for attrs in (["ETag"], ["ObjectSize", "StorageClass"]):
|
for attrs in (["ETag"], ["ObjectSize", "StorageClass"]):
|
||||||
s3_client.get_object_attributes(bucket, file_name, attrs)
|
s3_gate_object.get_object_attributes(self.s3_client, bucket, file_name, *attrs)
|
||||||
|
|
||||||
@allure.title("Test S3 Sync directory")
|
@allure.title("Test S3 Sync directory")
|
||||||
def test_s3_sync_dir(self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int):
|
def test_s3_sync_dir(self, bucket, simple_object_size):
|
||||||
"""
|
"""
|
||||||
Test checks sync directory with AWS CLI utility.
|
Test checks sync directory with AWS CLI utility.
|
||||||
"""
|
"""
|
||||||
|
@ -154,31 +146,29 @@ class TestS3Gate:
|
||||||
file_path_2 = os.path.join(os.getcwd(), ASSETS_DIR, "test_sync", "test_file_2")
|
file_path_2 = os.path.join(os.getcwd(), ASSETS_DIR, "test_sync", "test_file_2")
|
||||||
key_to_path = {"test_file_1": file_path_1, "test_file_2": file_path_2}
|
key_to_path = {"test_file_1": file_path_1, "test_file_2": file_path_2}
|
||||||
|
|
||||||
if not isinstance(s3_client, AwsCliClient):
|
if not isinstance(self.s3_client, AwsCliClient):
|
||||||
pytest.skip("This test is not supported with boto3 client")
|
pytest.skip("This test is not supported with boto3 client")
|
||||||
|
|
||||||
generate_file_with_content(simple_object_size, file_path=file_path_1)
|
generate_file_with_content(simple_object_size, file_path=file_path_1)
|
||||||
generate_file_with_content(simple_object_size, file_path=file_path_2)
|
generate_file_with_content(simple_object_size, file_path=file_path_2)
|
||||||
|
|
||||||
s3_client.sync(bucket=bucket, dir_path=os.path.dirname(file_path_1))
|
self.s3_client.sync(bucket_name=bucket, dir_path=os.path.dirname(file_path_1))
|
||||||
|
|
||||||
with allure.step("Check objects are synced"):
|
with allure.step("Check objects are synced"):
|
||||||
objects = s3_client.list_objects(bucket)
|
objects = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
|
|
||||||
with allure.step("Check these are the same objects"):
|
with allure.step("Check these are the same objects"):
|
||||||
assert set(key_to_path.keys()) == set(
|
assert set(key_to_path.keys()) == set(
|
||||||
objects
|
objects
|
||||||
), f"Expected all objects saved. Got {objects}"
|
), f"Expected all objects saved. Got {objects}"
|
||||||
for obj_key in objects:
|
for obj_key in objects:
|
||||||
got_object = s3_client.get_object(bucket, obj_key)
|
got_object = s3_gate_object.get_object_s3(self.s3_client, bucket, obj_key)
|
||||||
assert get_file_hash(got_object) == get_file_hash(
|
assert get_file_hash(got_object) == get_file_hash(
|
||||||
key_to_path.get(obj_key)
|
key_to_path.get(obj_key)
|
||||||
), "Expected hashes are the same"
|
), "Expected hashes are the same"
|
||||||
|
|
||||||
@allure.title("Test S3 Object versioning")
|
@allure.title("Test S3 Object versioning")
|
||||||
def test_s3_api_versioning(
|
def test_s3_api_versioning(self, bucket, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test checks basic versioning functionality for S3 bucket.
|
Test checks basic versioning functionality for S3 bucket.
|
||||||
"""
|
"""
|
||||||
|
@ -186,17 +176,17 @@ class TestS3Gate:
|
||||||
version_2_content = "Version 2"
|
version_2_content = "Version 2"
|
||||||
file_name_simple = generate_file_with_content(simple_object_size, content=version_1_content)
|
file_name_simple = generate_file_with_content(simple_object_size, content=version_1_content)
|
||||||
obj_key = os.path.basename(file_name_simple)
|
obj_key = os.path.basename(file_name_simple)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
|
|
||||||
with allure.step("Put several versions of object into bucket"):
|
with allure.step("Put several versions of object into bucket"):
|
||||||
version_id_1 = s3_client.put_object(bucket, file_name_simple)
|
version_id_1 = s3_gate_object.put_object_s3(self.s3_client, bucket, file_name_simple)
|
||||||
generate_file_with_content(
|
generate_file_with_content(
|
||||||
simple_object_size, file_path=file_name_simple, content=version_2_content
|
simple_object_size, file_path=file_name_simple, content=version_2_content
|
||||||
)
|
)
|
||||||
version_id_2 = s3_client.put_object(bucket, file_name_simple)
|
version_id_2 = s3_gate_object.put_object_s3(self.s3_client, bucket, file_name_simple)
|
||||||
|
|
||||||
with allure.step("Check bucket shows all versions"):
|
with allure.step("Check bucket shows all versions"):
|
||||||
versions = s3_client.list_objects_versions(bucket)
|
versions = s3_gate_object.list_objects_versions_s3(self.s3_client, bucket)
|
||||||
obj_versions = {
|
obj_versions = {
|
||||||
version.get("VersionId") for version in versions if version.get("Key") == obj_key
|
version.get("VersionId") for version in versions if version.get("Key") == obj_key
|
||||||
}
|
}
|
||||||
|
@ -207,7 +197,9 @@ class TestS3Gate:
|
||||||
|
|
||||||
with allure.step("Show information about particular version"):
|
with allure.step("Show information about particular version"):
|
||||||
for version_id in (version_id_1, version_id_2):
|
for version_id in (version_id_1, version_id_2):
|
||||||
response = s3_client.head_object(bucket, obj_key, version_id=version_id)
|
response = s3_gate_object.head_object_s3(
|
||||||
|
self.s3_client, bucket, obj_key, version_id=version_id
|
||||||
|
)
|
||||||
assert "LastModified" in response, "Expected LastModified field"
|
assert "LastModified" in response, "Expected LastModified field"
|
||||||
assert "ETag" in response, "Expected ETag field"
|
assert "ETag" in response, "Expected ETag field"
|
||||||
assert (
|
assert (
|
||||||
|
@ -217,8 +209,8 @@ class TestS3Gate:
|
||||||
|
|
||||||
with allure.step("Check object's attributes"):
|
with allure.step("Check object's attributes"):
|
||||||
for version_id in (version_id_1, version_id_2):
|
for version_id in (version_id_1, version_id_2):
|
||||||
got_attrs = s3_client.get_object_attributes(
|
got_attrs = s3_gate_object.get_object_attributes(
|
||||||
bucket, obj_key, ["ETag"], version_id=version_id
|
self.s3_client, bucket, obj_key, "ETag", version_id=version_id
|
||||||
)
|
)
|
||||||
if got_attrs:
|
if got_attrs:
|
||||||
assert (
|
assert (
|
||||||
|
@ -226,27 +218,31 @@ class TestS3Gate:
|
||||||
), f"Expected VersionId is {version_id}"
|
), f"Expected VersionId is {version_id}"
|
||||||
|
|
||||||
with allure.step("Delete object and check it was deleted"):
|
with allure.step("Delete object and check it was deleted"):
|
||||||
response = s3_client.delete_object(bucket, obj_key)
|
response = s3_gate_object.delete_object_s3(self.s3_client, bucket, obj_key)
|
||||||
version_id_delete = response.get("VersionId")
|
version_id_delete = response.get("VersionId")
|
||||||
|
|
||||||
with pytest.raises(Exception, match=r".*Not Found.*"):
|
with pytest.raises(Exception, match=r".*Not Found.*"):
|
||||||
s3_client.head_object(bucket, obj_key)
|
s3_gate_object.head_object_s3(self.s3_client, bucket, obj_key)
|
||||||
|
|
||||||
with allure.step("Get content for all versions and check it is correct"):
|
with allure.step("Get content for all versions and check it is correct"):
|
||||||
for version, content in (
|
for version, content in (
|
||||||
(version_id_2, version_2_content),
|
(version_id_2, version_2_content),
|
||||||
(version_id_1, version_1_content),
|
(version_id_1, version_1_content),
|
||||||
):
|
):
|
||||||
file_name = s3_client.get_object(bucket, obj_key, version_id=version)
|
file_name = s3_gate_object.get_object_s3(
|
||||||
|
self.s3_client, bucket, obj_key, version_id=version
|
||||||
|
)
|
||||||
got_content = get_file_content(file_name)
|
got_content = get_file_content(file_name)
|
||||||
assert (
|
assert (
|
||||||
got_content == content
|
got_content == content
|
||||||
), f"Expected object content is\n{content}\nGot\n{got_content}"
|
), f"Expected object content is\n{content}\nGot\n{got_content}"
|
||||||
|
|
||||||
with allure.step("Restore previous object version"):
|
with allure.step("Restore previous object version"):
|
||||||
s3_client.delete_object(bucket, obj_key, version_id=version_id_delete)
|
s3_gate_object.delete_object_s3(
|
||||||
|
self.s3_client, bucket, obj_key, version_id=version_id_delete
|
||||||
|
)
|
||||||
|
|
||||||
file_name = s3_client.get_object(bucket, obj_key)
|
file_name = s3_gate_object.get_object_s3(self.s3_client, bucket, obj_key)
|
||||||
got_content = get_file_content(file_name)
|
got_content = get_file_content(file_name)
|
||||||
assert (
|
assert (
|
||||||
got_content == version_2_content
|
got_content == version_2_content
|
||||||
|
@ -254,9 +250,7 @@ class TestS3Gate:
|
||||||
|
|
||||||
@pytest.mark.s3_gate_multipart
|
@pytest.mark.s3_gate_multipart
|
||||||
@allure.title("Test S3 Object Multipart API")
|
@allure.title("Test S3 Object Multipart API")
|
||||||
def test_s3_api_multipart(
|
def test_s3_api_multipart(self, bucket, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test checks S3 Multipart API (Create multipart upload/Abort multipart upload/List multipart upload/
|
Test checks S3 Multipart API (Create multipart upload/Abort multipart upload/List multipart upload/
|
||||||
Upload part/List parts/Complete multipart upload).
|
Upload part/List parts/Complete multipart upload).
|
||||||
|
@ -265,16 +259,18 @@ class TestS3Gate:
|
||||||
file_name_large = generate_file(
|
file_name_large = generate_file(
|
||||||
simple_object_size * 1024 * 6 * parts_count
|
simple_object_size * 1024 * 6 * parts_count
|
||||||
) # 5Mb - min part
|
) # 5Mb - min part
|
||||||
object_key = s3_helper.object_key_from_file_path(file_name_large)
|
object_key = self.object_key_from_file_path(file_name_large)
|
||||||
part_files = split_file(file_name_large, parts_count)
|
part_files = split_file(file_name_large, parts_count)
|
||||||
parts = []
|
parts = []
|
||||||
|
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Create and abort multipart upload"):
|
with allure.step("Create and abort multipart upload"):
|
||||||
upload_id = s3_client.create_multipart_upload(bucket, object_key)
|
upload_id = s3_gate_object.create_multipart_upload_s3(
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
self.s3_client, bucket, object_key
|
||||||
|
)
|
||||||
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert uploads, f"Expected there one upload in bucket {bucket}"
|
assert uploads, f"Expected there one upload in bucket {bucket}"
|
||||||
assert (
|
assert (
|
||||||
uploads[0].get("Key") == object_key
|
uploads[0].get("Key") == object_key
|
||||||
|
@ -283,50 +279,54 @@ class TestS3Gate:
|
||||||
uploads[0].get("UploadId") == upload_id
|
uploads[0].get("UploadId") == upload_id
|
||||||
), f"Expected correct UploadId {upload_id} in upload {uploads}"
|
), f"Expected correct UploadId {upload_id} in upload {uploads}"
|
||||||
|
|
||||||
s3_client.abort_multipart_upload(bucket, object_key, upload_id)
|
s3_gate_object.abort_multipart_uploads_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Create new multipart upload and upload several parts"):
|
with allure.step("Create new multipart upload and upload several parts"):
|
||||||
upload_id = s3_client.create_multipart_upload(bucket, object_key)
|
upload_id = s3_gate_object.create_multipart_upload_s3(
|
||||||
|
self.s3_client, bucket, object_key
|
||||||
|
)
|
||||||
for part_id, file_path in enumerate(part_files, start=1):
|
for part_id, file_path in enumerate(part_files, start=1):
|
||||||
etag = s3_client.upload_part(bucket, object_key, upload_id, part_id, file_path)
|
etag = s3_gate_object.upload_part_s3(
|
||||||
|
self.s3_client, bucket, object_key, upload_id, part_id, file_path
|
||||||
|
)
|
||||||
parts.append((part_id, etag))
|
parts.append((part_id, etag))
|
||||||
|
|
||||||
with allure.step("Check all parts are visible in bucket"):
|
with allure.step("Check all parts are visible in bucket"):
|
||||||
got_parts = s3_client.list_parts(bucket, object_key, upload_id)
|
got_parts = s3_gate_object.list_parts_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
assert len(got_parts) == len(
|
assert len(got_parts) == len(
|
||||||
part_files
|
part_files
|
||||||
), f"Expected {parts_count} parts, got\n{got_parts}"
|
), f"Expected {parts_count} parts, got\n{got_parts}"
|
||||||
|
|
||||||
s3_client.complete_multipart_upload(bucket, object_key, upload_id, parts)
|
s3_gate_object.complete_multipart_upload_s3(
|
||||||
|
self.s3_client, bucket, object_key, upload_id, parts
|
||||||
|
)
|
||||||
|
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Check we can get whole object from bucket"):
|
with allure.step("Check we can get whole object from bucket"):
|
||||||
got_object = s3_client.get_object(bucket, object_key)
|
got_object = s3_gate_object.get_object_s3(self.s3_client, bucket, object_key)
|
||||||
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
||||||
|
|
||||||
self.check_object_attributes(s3_client, bucket, object_key, parts_count)
|
self.check_object_attributes(bucket, object_key, parts_count)
|
||||||
|
|
||||||
@allure.title("Test S3 Bucket tagging API")
|
@allure.title("Test S3 Bucket tagging API")
|
||||||
def test_s3_api_bucket_tagging(self, s3_client: S3ClientWrapper, bucket: str):
|
def test_s3_api_bucket_tagging(self, bucket):
|
||||||
"""
|
"""
|
||||||
Test checks S3 Bucket tagging API (Put tag/Get tag).
|
Test checks S3 Bucket tagging API (Put tag/Get tag).
|
||||||
"""
|
"""
|
||||||
key_value_pair = [("some-key", "some-value"), ("some-key-2", "some-value-2")]
|
key_value_pair = [("some-key", "some-value"), ("some-key-2", "some-value-2")]
|
||||||
|
|
||||||
s3_client.put_bucket_tagging(bucket, key_value_pair)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, key_value_pair)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, key_value_pair)
|
check_tags_by_bucket(self.s3_client, bucket, key_value_pair)
|
||||||
|
|
||||||
s3_client.delete_bucket_tagging(bucket)
|
s3_gate_bucket.delete_bucket_tagging(self.s3_client, bucket)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, [])
|
check_tags_by_bucket(self.s3_client, bucket, [])
|
||||||
|
|
||||||
@allure.title("Test S3 Object tagging API")
|
@allure.title("Test S3 Object tagging API")
|
||||||
def test_s3_api_object_tagging(
|
def test_s3_api_object_tagging(self, bucket, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test checks S3 Object tagging API (Put tag/Get tag/Update tag).
|
Test checks S3 Object tagging API (Put tag/Get tag/Update tag).
|
||||||
"""
|
"""
|
||||||
|
@ -337,32 +337,26 @@ class TestS3Gate:
|
||||||
]
|
]
|
||||||
key_value_pair_obj_new = [("some-key-obj-new", "some-value-obj-new")]
|
key_value_pair_obj_new = [("some-key-obj-new", "some-value-obj-new")]
|
||||||
file_name_simple = generate_file(simple_object_size)
|
file_name_simple = generate_file(simple_object_size)
|
||||||
obj_key = s3_helper.object_key_from_file_path(file_name_simple)
|
obj_key = self.object_key_from_file_path(file_name_simple)
|
||||||
|
|
||||||
s3_client.put_bucket_tagging(bucket, key_value_pair_bucket)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, key_value_pair_bucket)
|
||||||
|
|
||||||
s3_client.put_object(bucket, file_name_simple)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_name_simple)
|
||||||
|
|
||||||
for tags in (key_value_pair_obj, key_value_pair_obj_new):
|
for tags in (key_value_pair_obj, key_value_pair_obj_new):
|
||||||
s3_client.put_object_tagging(bucket, obj_key, tags)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, obj_key, tags)
|
||||||
s3_helper.check_tags_by_object(
|
check_tags_by_object(
|
||||||
s3_client,
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
obj_key,
|
obj_key,
|
||||||
tags,
|
tags,
|
||||||
)
|
)
|
||||||
|
|
||||||
s3_client.delete_object_tagging(bucket, obj_key)
|
s3_gate_object.delete_object_tagging(self.s3_client, bucket, obj_key)
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, obj_key, [])
|
check_tags_by_object(self.s3_client, bucket, obj_key, [])
|
||||||
|
|
||||||
@allure.title("Test S3: Delete object & delete objects S3 API")
|
@allure.title("Test S3: Delete object & delete objects S3 API")
|
||||||
def test_s3_api_delete(
|
def test_s3_api_delete(self, two_buckets, simple_object_size, complex_object_size):
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
two_buckets: tuple[str, str],
|
|
||||||
simple_object_size: int,
|
|
||||||
complex_object_size: int,
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Check delete_object and delete_objects S3 API operation. From first bucket some objects deleted one by one.
|
Check delete_object and delete_objects S3 API operation. From first bucket some objects deleted one by one.
|
||||||
From second bucket some objects deleted all at once.
|
From second bucket some objects deleted all at once.
|
||||||
|
@ -381,15 +375,15 @@ class TestS3Gate:
|
||||||
|
|
||||||
for bucket in (bucket_1, bucket_2):
|
for bucket in (bucket_1, bucket_2):
|
||||||
with allure.step(f"Bucket {bucket} must be empty as it just created"):
|
with allure.step(f"Bucket {bucket} must be empty as it just created"):
|
||||||
objects_list = s3_client.list_objects_v2(bucket)
|
objects_list = s3_gate_object.list_objects_s3_v2(self.s3_client, bucket)
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
||||||
|
|
||||||
for file_path in file_paths:
|
for file_path in file_paths:
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
put_objects.append(s3_helper.object_key_from_file_path(file_path))
|
put_objects.append(self.object_key_from_file_path(file_path))
|
||||||
|
|
||||||
with allure.step(f"Check all objects put in bucket {bucket} successfully"):
|
with allure.step(f"Check all objects put in bucket {bucket} successfully"):
|
||||||
bucket_objects = s3_client.list_objects_v2(bucket)
|
bucket_objects = s3_gate_object.list_objects_s3_v2(self.s3_client, bucket)
|
||||||
assert set(put_objects) == set(
|
assert set(put_objects) == set(
|
||||||
bucket_objects
|
bucket_objects
|
||||||
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
||||||
|
@ -397,38 +391,28 @@ class TestS3Gate:
|
||||||
with allure.step("Delete some objects from bucket_1 one by one"):
|
with allure.step("Delete some objects from bucket_1 one by one"):
|
||||||
objects_to_delete_b1 = choices(put_objects, k=max_delete_objects)
|
objects_to_delete_b1 = choices(put_objects, k=max_delete_objects)
|
||||||
for obj in objects_to_delete_b1:
|
for obj in objects_to_delete_b1:
|
||||||
s3_client.delete_object(bucket_1, obj)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket_1, obj)
|
||||||
|
|
||||||
with allure.step("Check deleted objects are not visible in bucket bucket_1"):
|
with allure.step("Check deleted objects are not visible in bucket bucket_1"):
|
||||||
bucket_objects = s3_client.list_objects_v2(bucket_1)
|
bucket_objects = s3_gate_object.list_objects_s3_v2(self.s3_client, bucket_1)
|
||||||
assert set(put_objects).difference(set(objects_to_delete_b1)) == set(
|
assert set(put_objects).difference(set(objects_to_delete_b1)) == set(
|
||||||
bucket_objects
|
bucket_objects
|
||||||
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
||||||
for object_key in objects_to_delete_b1:
|
try_to_get_objects_and_expect_error(self.s3_client, bucket_1, objects_to_delete_b1)
|
||||||
with pytest.raises(Exception, match="The specified key does not exist"):
|
|
||||||
s3_client.get_object(bucket_1, object_key)
|
|
||||||
|
|
||||||
with allure.step("Delete some objects from bucket_2 at once"):
|
with allure.step("Delete some objects from bucket_2 at once"):
|
||||||
objects_to_delete_b2 = choices(put_objects, k=max_delete_objects)
|
objects_to_delete_b2 = choices(put_objects, k=max_delete_objects)
|
||||||
s3_client.delete_objects(bucket_2, objects_to_delete_b2)
|
s3_gate_object.delete_objects_s3(self.s3_client, bucket_2, objects_to_delete_b2)
|
||||||
|
|
||||||
with allure.step("Check deleted objects are not visible in bucket bucket_2"):
|
with allure.step("Check deleted objects are not visible in bucket bucket_2"):
|
||||||
objects_list = s3_client.list_objects_v2(bucket_2)
|
objects_list = s3_gate_object.list_objects_s3_v2(self.s3_client, bucket_2)
|
||||||
assert set(put_objects).difference(set(objects_to_delete_b2)) == set(
|
assert set(put_objects).difference(set(objects_to_delete_b2)) == set(
|
||||||
objects_list
|
objects_list
|
||||||
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
), f"Expected all objects {put_objects} in objects list {bucket_objects}"
|
||||||
for object_key in objects_to_delete_b2:
|
try_to_get_objects_and_expect_error(self.s3_client, bucket_2, objects_to_delete_b2)
|
||||||
with pytest.raises(Exception, match="The specified key does not exist"):
|
|
||||||
s3_client.get_object(bucket_2, object_key)
|
|
||||||
|
|
||||||
@allure.title("Test S3: Copy object to the same bucket")
|
@allure.title("Test S3: Copy object to the same bucket")
|
||||||
def test_s3_copy_same_bucket(
|
def test_s3_copy_same_bucket(self, bucket, complex_object_size, simple_object_size):
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
bucket: str,
|
|
||||||
complex_object_size: int,
|
|
||||||
simple_object_size: int,
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test object can be copied to the same bucket.
|
Test object can be copied to the same bucket.
|
||||||
#TODO: delete after test_s3_copy_object will be merge
|
#TODO: delete after test_s3_copy_object will be merge
|
||||||
|
@ -436,49 +420,43 @@ class TestS3Gate:
|
||||||
file_path_simple, file_path_large = generate_file(simple_object_size), generate_file(
|
file_path_simple, file_path_large = generate_file(simple_object_size), generate_file(
|
||||||
complex_object_size
|
complex_object_size
|
||||||
)
|
)
|
||||||
file_name_simple = s3_helper.object_key_from_file_path(file_path_simple)
|
file_name_simple = self.object_key_from_file_path(file_path_simple)
|
||||||
file_name_large = s3_helper.object_key_from_file_path(file_path_large)
|
file_name_large = self.object_key_from_file_path(file_path_large)
|
||||||
bucket_objects = [file_name_simple, file_name_large]
|
bucket_objects = [file_name_simple, file_name_large]
|
||||||
|
|
||||||
with allure.step("Bucket must be empty"):
|
with allure.step("Bucket must be empty"):
|
||||||
objects_list = s3_client.list_objects(bucket)
|
objects_list = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
||||||
|
|
||||||
with allure.step("Put objects into bucket"):
|
with allure.step("Put objects into bucket"):
|
||||||
for file_path in (file_path_simple, file_path_large):
|
for file_path in (file_path_simple, file_path_large):
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
|
|
||||||
with allure.step("Copy one object into the same bucket"):
|
with allure.step("Copy one object into the same bucket"):
|
||||||
copy_obj_path = s3_client.copy_object(bucket, file_name_simple)
|
copy_obj_path = s3_gate_object.copy_object_s3(self.s3_client, bucket, file_name_simple)
|
||||||
bucket_objects.append(copy_obj_path)
|
bucket_objects.append(copy_obj_path)
|
||||||
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, bucket_objects)
|
check_objects_in_bucket(self.s3_client, bucket, bucket_objects)
|
||||||
|
|
||||||
with allure.step("Check copied object has the same content"):
|
with allure.step("Check copied object has the same content"):
|
||||||
got_copied_file = s3_client.get_object(bucket, copy_obj_path)
|
got_copied_file = s3_gate_object.get_object_s3(self.s3_client, bucket, copy_obj_path)
|
||||||
assert get_file_hash(file_path_simple) == get_file_hash(
|
assert get_file_hash(file_path_simple) == get_file_hash(
|
||||||
got_copied_file
|
got_copied_file
|
||||||
), "Hashes must be the same"
|
), "Hashes must be the same"
|
||||||
|
|
||||||
with allure.step("Delete one object from bucket"):
|
with allure.step("Delete one object from bucket"):
|
||||||
s3_client.delete_object(bucket, file_name_simple)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name_simple)
|
||||||
bucket_objects.remove(file_name_simple)
|
bucket_objects.remove(file_name_simple)
|
||||||
|
|
||||||
s3_helper.check_objects_in_bucket(
|
check_objects_in_bucket(
|
||||||
s3_client,
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
expected_objects=bucket_objects,
|
expected_objects=bucket_objects,
|
||||||
unexpected_objects=[file_name_simple],
|
unexpected_objects=[file_name_simple],
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test S3: Copy object to another bucket")
|
@allure.title("Test S3: Copy object to another bucket")
|
||||||
def test_s3_copy_to_another_bucket(
|
def test_s3_copy_to_another_bucket(self, two_buckets, complex_object_size, simple_object_size):
|
||||||
self,
|
|
||||||
s3_client: S3ClientWrapper,
|
|
||||||
two_buckets: tuple[str, str],
|
|
||||||
complex_object_size: int,
|
|
||||||
simple_object_size: int,
|
|
||||||
):
|
|
||||||
"""
|
"""
|
||||||
Test object can be copied to another bucket.
|
Test object can be copied to another bucket.
|
||||||
#TODO: delete after test_s3_copy_object will be merge
|
#TODO: delete after test_s3_copy_object will be merge
|
||||||
|
@ -486,53 +464,55 @@ class TestS3Gate:
|
||||||
file_path_simple, file_path_large = generate_file(simple_object_size), generate_file(
|
file_path_simple, file_path_large = generate_file(simple_object_size), generate_file(
|
||||||
complex_object_size
|
complex_object_size
|
||||||
)
|
)
|
||||||
file_name_simple = s3_helper.object_key_from_file_path(file_path_simple)
|
file_name_simple = self.object_key_from_file_path(file_path_simple)
|
||||||
file_name_large = s3_helper.object_key_from_file_path(file_path_large)
|
file_name_large = self.object_key_from_file_path(file_path_large)
|
||||||
bucket_1_objects = [file_name_simple, file_name_large]
|
bucket_1_objects = [file_name_simple, file_name_large]
|
||||||
|
|
||||||
bucket_1, bucket_2 = two_buckets
|
bucket_1, bucket_2 = two_buckets
|
||||||
|
|
||||||
with allure.step("Buckets must be empty"):
|
with allure.step("Buckets must be empty"):
|
||||||
for bucket in (bucket_1, bucket_2):
|
for bucket in (bucket_1, bucket_2):
|
||||||
objects_list = s3_client.list_objects(bucket)
|
objects_list = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
assert not objects_list, f"Expected empty bucket, got {objects_list}"
|
||||||
|
|
||||||
with allure.step("Put objects into one bucket"):
|
with allure.step("Put objects into one bucket"):
|
||||||
for file_path in (file_path_simple, file_path_large):
|
for file_path in (file_path_simple, file_path_large):
|
||||||
s3_client.put_object(bucket_1, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket_1, file_path)
|
||||||
|
|
||||||
with allure.step("Copy object from first bucket into second"):
|
with allure.step("Copy object from first bucket into second"):
|
||||||
copy_obj_path_b2 = s3_client.copy_object(bucket_1, file_name_large, bucket=bucket_2)
|
copy_obj_path_b2 = s3_gate_object.copy_object_s3(
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_1, expected_objects=bucket_1_objects)
|
self.s3_client, bucket_1, file_name_large, bucket_dst=bucket_2
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_2, expected_objects=[copy_obj_path_b2])
|
)
|
||||||
|
check_objects_in_bucket(self.s3_client, bucket_1, expected_objects=bucket_1_objects)
|
||||||
|
check_objects_in_bucket(self.s3_client, bucket_2, expected_objects=[copy_obj_path_b2])
|
||||||
|
|
||||||
with allure.step("Check copied object has the same content"):
|
with allure.step("Check copied object has the same content"):
|
||||||
got_copied_file_b2 = s3_client.get_object(bucket_2, copy_obj_path_b2)
|
got_copied_file_b2 = s3_gate_object.get_object_s3(
|
||||||
|
self.s3_client, bucket_2, copy_obj_path_b2
|
||||||
|
)
|
||||||
assert get_file_hash(file_path_large) == get_file_hash(
|
assert get_file_hash(file_path_large) == get_file_hash(
|
||||||
got_copied_file_b2
|
got_copied_file_b2
|
||||||
), "Hashes must be the same"
|
), "Hashes must be the same"
|
||||||
|
|
||||||
with allure.step("Delete one object from first bucket"):
|
with allure.step("Delete one object from first bucket"):
|
||||||
s3_client.delete_object(bucket_1, file_name_simple)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket_1, file_name_simple)
|
||||||
bucket_1_objects.remove(file_name_simple)
|
bucket_1_objects.remove(file_name_simple)
|
||||||
|
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_1, expected_objects=bucket_1_objects)
|
check_objects_in_bucket(self.s3_client, bucket_1, expected_objects=bucket_1_objects)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_2, expected_objects=[copy_obj_path_b2])
|
check_objects_in_bucket(self.s3_client, bucket_2, expected_objects=[copy_obj_path_b2])
|
||||||
|
|
||||||
with allure.step("Delete one object from second bucket and check it is empty"):
|
with allure.step("Delete one object from second bucket and check it is empty"):
|
||||||
s3_client.delete_object(bucket_2, copy_obj_path_b2)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket_2, copy_obj_path_b2)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_2, expected_objects=[])
|
check_objects_in_bucket(self.s3_client, bucket_2, expected_objects=[])
|
||||||
|
|
||||||
def check_object_attributes(
|
def check_object_attributes(self, bucket: str, object_key: str, parts_count: int):
|
||||||
self, s3_client: S3ClientWrapper, bucket: str, object_key: str, parts_count: int
|
if not isinstance(self.s3_client, AwsCliClient):
|
||||||
):
|
|
||||||
if not isinstance(s3_client, AwsCliClient):
|
|
||||||
logger.warning("Attributes check is not supported for boto3 implementation")
|
logger.warning("Attributes check is not supported for boto3 implementation")
|
||||||
return
|
return
|
||||||
|
|
||||||
with allure.step("Check object's attributes"):
|
with allure.step("Check object's attributes"):
|
||||||
obj_parts = s3_client.get_object_attributes(
|
obj_parts = s3_gate_object.get_object_attributes(
|
||||||
bucket, object_key, ["ObjectParts"], full_output=False
|
self.s3_client, bucket, object_key, "ObjectParts", get_full_resp=False
|
||||||
)
|
)
|
||||||
assert (
|
assert (
|
||||||
obj_parts.get("TotalPartsCount") == parts_count
|
obj_parts.get("TotalPartsCount") == parts_count
|
||||||
|
@ -543,12 +523,13 @@ class TestS3Gate:
|
||||||
|
|
||||||
with allure.step("Check object's attribute max-parts"):
|
with allure.step("Check object's attribute max-parts"):
|
||||||
max_parts = 2
|
max_parts = 2
|
||||||
obj_parts = s3_client.get_object_attributes(
|
obj_parts = s3_gate_object.get_object_attributes(
|
||||||
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
object_key,
|
object_key,
|
||||||
["ObjectParts"],
|
"ObjectParts",
|
||||||
max_parts=max_parts,
|
max_parts=max_parts,
|
||||||
full_output=False,
|
get_full_resp=False,
|
||||||
)
|
)
|
||||||
assert (
|
assert (
|
||||||
obj_parts.get("TotalPartsCount") == parts_count
|
obj_parts.get("TotalPartsCount") == parts_count
|
||||||
|
@ -560,12 +541,13 @@ class TestS3Gate:
|
||||||
|
|
||||||
with allure.step("Check object's attribute part-number-marker"):
|
with allure.step("Check object's attribute part-number-marker"):
|
||||||
part_number_marker = 3
|
part_number_marker = 3
|
||||||
obj_parts = s3_client.get_object_attributes(
|
obj_parts = s3_gate_object.get_object_attributes(
|
||||||
|
self.s3_client,
|
||||||
bucket,
|
bucket,
|
||||||
object_key,
|
object_key,
|
||||||
["ObjectParts"],
|
"ObjectParts",
|
||||||
part_number=part_number_marker,
|
part_number=part_number_marker,
|
||||||
full_output=False,
|
get_full_resp=False,
|
||||||
)
|
)
|
||||||
assert (
|
assert (
|
||||||
obj_parts.get("TotalPartsCount") == parts_count
|
obj_parts.get("TotalPartsCount") == parts_count
|
||||||
|
@ -574,3 +556,7 @@ class TestS3Gate:
|
||||||
obj_parts.get("PartNumberMarker") == part_number_marker
|
obj_parts.get("PartNumberMarker") == part_number_marker
|
||||||
), f"Expected PartNumberMarker is {part_number_marker}"
|
), f"Expected PartNumberMarker is {part_number_marker}"
|
||||||
assert len(obj_parts.get("Parts")) == 1, f"Expected Parts count is {parts_count}"
|
assert len(obj_parts.get("Parts")) == 1, f"Expected Parts count is {parts_count}"
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def object_key_from_file_path(full_path: str) -> str:
|
||||||
|
return os.path.basename(full_path)
|
||||||
|
|
|
@ -3,36 +3,36 @@ from datetime import datetime, timedelta
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper
|
from file_helper import generate_file, generate_file_with_content
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from s3_helper import assert_object_lock_mode, check_objects_in_bucket, object_key_from_file_path
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, generate_file_with_content
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_locking
|
@pytest.mark.s3_gate_locking
|
||||||
@pytest.mark.parametrize("version_id", [None, "second"])
|
@pytest.mark.parametrize("version_id", [None, "second"])
|
||||||
class TestS3GateLocking:
|
class TestS3GateLocking(TestS3GateBase):
|
||||||
@allure.title("Test S3: Checking the operation of retention period & legal lock on the object")
|
@allure.title("Test S3: Checking the operation of retention period & legal lock on the object")
|
||||||
def test_s3_object_locking(
|
def test_s3_object_locking(self, version_id, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, version_id: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
retention_period = 2
|
retention_period = 2
|
||||||
|
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
|
|
||||||
with allure.step("Put several versions of object into bucket"):
|
with allure.step("Put several versions of object into bucket"):
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
file_name_1 = generate_file_with_content(simple_object_size, file_path=file_path)
|
file_name_1 = generate_file_with_content(simple_object_size, file_path=file_path)
|
||||||
version_id_2 = s3_client.put_object(bucket, file_name_1)
|
version_id_2 = s3_gate_object.put_object_s3(self.s3_client, bucket, file_name_1)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [file_name])
|
check_objects_in_bucket(self.s3_client, bucket, [file_name])
|
||||||
if version_id:
|
if version_id:
|
||||||
version_id = version_id_2
|
version_id = version_id_2
|
||||||
|
|
||||||
|
@ -42,53 +42,51 @@ class TestS3GateLocking:
|
||||||
"Mode": "COMPLIANCE",
|
"Mode": "COMPLIANCE",
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
s3_helper.assert_object_lock_mode(
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
s3_client, bucket, file_name, "COMPLIANCE", date_obj, "OFF"
|
)
|
||||||
|
assert_object_lock_mode(
|
||||||
|
self.s3_client, bucket, file_name, "COMPLIANCE", date_obj, "OFF"
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step(f"Put legal hold to object {file_name}"):
|
with allure.step(f"Put legal hold to object {file_name}"):
|
||||||
s3_client.put_object_legal_hold(bucket, file_name, "ON", version_id)
|
s3_gate_object.put_object_legal_hold(
|
||||||
s3_helper.assert_object_lock_mode(
|
self.s3_client, bucket, file_name, "ON", version_id
|
||||||
s3_client, bucket, file_name, "COMPLIANCE", date_obj, "ON"
|
|
||||||
)
|
)
|
||||||
|
assert_object_lock_mode(self.s3_client, bucket, file_name, "COMPLIANCE", date_obj, "ON")
|
||||||
|
|
||||||
with allure.step("Fail with deleting object with legal hold and retention period"):
|
with allure.step(f"Fail with deleting object with legal hold and retention period"):
|
||||||
if version_id:
|
if version_id:
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
# An error occurred (AccessDenied) when calling the DeleteObject operation (reached max retries: 0): Access Denied.
|
# An error occurred (AccessDenied) when calling the DeleteObject operation (reached max retries: 0): Access Denied.
|
||||||
s3_client.delete_object(bucket, file_name, version_id)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name, version_id)
|
||||||
|
|
||||||
with allure.step("Check retention period is no longer set on the uploaded object"):
|
with allure.step(f"Check retention period is no longer set on the uploaded object"):
|
||||||
time.sleep((retention_period + 1) * 60)
|
time.sleep((retention_period + 1) * 60)
|
||||||
s3_helper.assert_object_lock_mode(
|
assert_object_lock_mode(self.s3_client, bucket, file_name, "COMPLIANCE", date_obj, "ON")
|
||||||
s3_client, bucket, file_name, "COMPLIANCE", date_obj, "ON"
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Fail with deleting object with legal hold and retention period"):
|
with allure.step(f"Fail with deleting object with legal hold and retention period"):
|
||||||
if version_id:
|
if version_id:
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
# An error occurred (AccessDenied) when calling the DeleteObject operation (reached max retries: 0): Access Denied.
|
# An error occurred (AccessDenied) when calling the DeleteObject operation (reached max retries: 0): Access Denied.
|
||||||
s3_client.delete_object(bucket, file_name, version_id)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name, version_id)
|
||||||
else:
|
else:
|
||||||
s3_client.delete_object(bucket, file_name, version_id)
|
s3_gate_object.delete_object_s3(self.s3_client, bucket, file_name, version_id)
|
||||||
|
|
||||||
@allure.title("Test S3: Checking the impossibility to change the retention mode COMPLIANCE")
|
@allure.title("Test S3: Checking the impossibility to change the retention mode COMPLIANCE")
|
||||||
def test_s3_mode_compliance(
|
def test_s3_mode_compliance(self, version_id, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, version_id: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
retention_period = 2
|
retention_period = 2
|
||||||
retention_period_1 = 1
|
retention_period_1 = 1
|
||||||
|
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
with allure.step("Put object into bucket"):
|
||||||
obj_version = s3_client.put_object(bucket, file_path)
|
obj_version = s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
if version_id:
|
if version_id:
|
||||||
version_id = obj_version
|
version_id = obj_version
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [file_name])
|
check_objects_in_bucket(self.s3_client, bucket, [file_name])
|
||||||
|
|
||||||
with allure.step(f"Put retention period {retention_period}min to object {file_name}"):
|
with allure.step(f"Put retention period {retention_period}min to object {file_name}"):
|
||||||
date_obj = datetime.utcnow() + timedelta(minutes=retention_period)
|
date_obj = datetime.utcnow() + timedelta(minutes=retention_period)
|
||||||
|
@ -96,9 +94,11 @@ class TestS3GateLocking:
|
||||||
"Mode": "COMPLIANCE",
|
"Mode": "COMPLIANCE",
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
s3_helper.assert_object_lock_mode(
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
s3_client, bucket, file_name, "COMPLIANCE", date_obj, "OFF"
|
)
|
||||||
|
assert_object_lock_mode(
|
||||||
|
self.s3_client, bucket, file_name, "COMPLIANCE", date_obj, "OFF"
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step(
|
with allure.step(
|
||||||
|
@ -110,25 +110,25 @@ class TestS3GateLocking:
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
|
)
|
||||||
|
|
||||||
@allure.title("Test S3: Checking the ability to change retention mode GOVERNANCE")
|
@allure.title("Test S3: Checking the ability to change retention mode GOVERNANCE")
|
||||||
def test_s3_mode_governance(
|
def test_s3_mode_governance(self, version_id, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, version_id: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
retention_period = 3
|
retention_period = 3
|
||||||
retention_period_1 = 2
|
retention_period_1 = 2
|
||||||
retention_period_2 = 5
|
retention_period_2 = 5
|
||||||
|
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
with allure.step("Put object into bucket"):
|
||||||
obj_version = s3_client.put_object(bucket, file_path)
|
obj_version = s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
if version_id:
|
if version_id:
|
||||||
version_id = obj_version
|
version_id = obj_version
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [file_name])
|
check_objects_in_bucket(self.s3_client, bucket, [file_name])
|
||||||
|
|
||||||
with allure.step(f"Put retention period {retention_period}min to object {file_name}"):
|
with allure.step(f"Put retention period {retention_period}min to object {file_name}"):
|
||||||
date_obj = datetime.utcnow() + timedelta(minutes=retention_period)
|
date_obj = datetime.utcnow() + timedelta(minutes=retention_period)
|
||||||
|
@ -136,9 +136,11 @@ class TestS3GateLocking:
|
||||||
"Mode": "GOVERNANCE",
|
"Mode": "GOVERNANCE",
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
s3_helper.assert_object_lock_mode(
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
s3_client, bucket, file_name, "GOVERNANCE", date_obj, "OFF"
|
)
|
||||||
|
assert_object_lock_mode(
|
||||||
|
self.s3_client, bucket, file_name, "GOVERNANCE", date_obj, "OFF"
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step(
|
with allure.step(
|
||||||
|
@ -150,7 +152,9 @@ class TestS3GateLocking:
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
|
)
|
||||||
|
|
||||||
with allure.step(
|
with allure.step(
|
||||||
f"Try to change retention period {retention_period_1}min to object {file_name}"
|
f"Try to change retention period {retention_period_1}min to object {file_name}"
|
||||||
|
@ -161,7 +165,9 @@ class TestS3GateLocking:
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id)
|
s3_gate_object.put_object_retention(
|
||||||
|
self.s3_client, bucket, file_name, retention, version_id
|
||||||
|
)
|
||||||
|
|
||||||
with allure.step(f"Put new retention period {retention_period_2}min to object {file_name}"):
|
with allure.step(f"Put new retention period {retention_period_2}min to object {file_name}"):
|
||||||
date_obj = datetime.utcnow() + timedelta(minutes=retention_period_2)
|
date_obj = datetime.utcnow() + timedelta(minutes=retention_period_2)
|
||||||
|
@ -169,55 +175,55 @@ class TestS3GateLocking:
|
||||||
"Mode": "GOVERNANCE",
|
"Mode": "GOVERNANCE",
|
||||||
"RetainUntilDate": date_obj,
|
"RetainUntilDate": date_obj,
|
||||||
}
|
}
|
||||||
s3_client.put_object_retention(bucket, file_name, retention, version_id, True)
|
s3_gate_object.put_object_retention(
|
||||||
s3_helper.assert_object_lock_mode(
|
self.s3_client, bucket, file_name, retention, version_id, True
|
||||||
s3_client, bucket, file_name, "GOVERNANCE", date_obj, "OFF"
|
)
|
||||||
|
assert_object_lock_mode(
|
||||||
|
self.s3_client, bucket, file_name, "GOVERNANCE", date_obj, "OFF"
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Test S3: Checking if an Object Cannot Be Locked")
|
@allure.title("Test S3: Checking if an Object Cannot Be Locked")
|
||||||
def test_s3_legal_hold(
|
def test_s3_legal_hold(self, version_id, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, version_id: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
|
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=False)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, False)
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
with allure.step("Put object into bucket"):
|
||||||
obj_version = s3_client.put_object(bucket, file_path)
|
obj_version = s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
if version_id:
|
if version_id:
|
||||||
version_id = obj_version
|
version_id = obj_version
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, [file_name])
|
check_objects_in_bucket(self.s3_client, bucket, [file_name])
|
||||||
|
|
||||||
with allure.step(f"Put legal hold to object {file_name}"):
|
with allure.step(f"Put legal hold to object {file_name}"):
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
s3_client.put_object_legal_hold(bucket, file_name, "ON", version_id)
|
s3_gate_object.put_object_legal_hold(
|
||||||
|
self.s3_client, bucket, file_name, "ON", version_id
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
class TestS3GateLockingBucket:
|
class TestS3GateLockingBucket(TestS3GateBase):
|
||||||
@allure.title("Test S3: Bucket Lock")
|
@allure.title("Test S3: Bucket Lock")
|
||||||
def test_s3_bucket_lock(self, s3_client: S3ClientWrapper, simple_object_size: int):
|
def test_s3_bucket_lock(self, simple_object_size):
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
configuration = {"Rule": {"DefaultRetention": {"Mode": "COMPLIANCE", "Days": 1}}}
|
configuration = {"Rule": {"DefaultRetention": {"Mode": "COMPLIANCE", "Days": 1}}}
|
||||||
|
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
|
|
||||||
with allure.step("PutObjectLockConfiguration with ObjectLockEnabled=False"):
|
with allure.step("PutObjectLockConfiguration with ObjectLockEnabled=False"):
|
||||||
s3_client.put_object_lock_configuration(bucket, configuration)
|
s3_gate_bucket.put_object_lock_configuration(self.s3_client, bucket, configuration)
|
||||||
|
|
||||||
with allure.step("PutObjectLockConfiguration with ObjectLockEnabled=True"):
|
with allure.step("PutObjectLockConfiguration with ObjectLockEnabled=True"):
|
||||||
configuration["ObjectLockEnabled"] = "Enabled"
|
configuration["ObjectLockEnabled"] = "Enabled"
|
||||||
s3_client.put_object_lock_configuration(bucket, configuration)
|
s3_gate_bucket.put_object_lock_configuration(self.s3_client, bucket, configuration)
|
||||||
|
|
||||||
with allure.step("GetObjectLockConfiguration"):
|
with allure.step("GetObjectLockConfiguration"):
|
||||||
config = s3_client.get_object_lock_configuration(bucket)
|
config = s3_gate_bucket.get_object_lock_configuration(self.s3_client, bucket)
|
||||||
configuration["Rule"]["DefaultRetention"]["Years"] = 0
|
configuration["Rule"]["DefaultRetention"]["Years"] = 0
|
||||||
assert config == configuration, f"Configurations must be equal {configuration}"
|
assert config == configuration, f"Configurations must be equal {configuration}"
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
with allure.step("Put object into bucket"):
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
s3_helper.assert_object_lock_mode(
|
assert_object_lock_mode(self.s3_client, bucket, file_name, "COMPLIANCE", None, "OFF", 1)
|
||||||
s3_client, bucket, file_name, "COMPLIANCE", None, "OFF", 1
|
|
||||||
)
|
|
||||||
|
|
|
@ -1,152 +1,132 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
from file_helper import generate_file, get_file_hash, split_file
|
||||||
from frostfs_testlib.steps.cli.container import list_objects, search_container_by_name
|
from s3_helper import check_objects_in_bucket, object_key_from_file_path, set_bucket_versioning
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, get_file_hash, split_file
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
PART_SIZE = 5 * 1024 * 1024
|
PART_SIZE = 5 * 1024 * 1024
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_multipart
|
@pytest.mark.s3_gate_multipart
|
||||||
class TestS3GateMultipart(ClusterTestBase):
|
class TestS3GateMultipart(TestS3GateBase):
|
||||||
NO_SUCH_UPLOAD = (
|
|
||||||
"The upload ID may be invalid, or the upload may have been aborted or completed."
|
|
||||||
)
|
|
||||||
|
|
||||||
@allure.title("Test S3 Object Multipart API")
|
@allure.title("Test S3 Object Multipart API")
|
||||||
@pytest.mark.parametrize("bucket", [VersioningStatus.ENABLED], indirect=True)
|
def test_s3_object_multipart(self):
|
||||||
def test_s3_object_multipart(self, s3_client: S3ClientWrapper, bucket: str):
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
parts_count = 5
|
parts_count = 5
|
||||||
file_name_large = generate_file(PART_SIZE * parts_count) # 5Mb - min part
|
file_name_large = generate_file(PART_SIZE * parts_count) # 5Mb - min part
|
||||||
object_key = s3_helper.object_key_from_file_path(file_name_large)
|
object_key = object_key_from_file_path(file_name_large)
|
||||||
part_files = split_file(file_name_large, parts_count)
|
part_files = split_file(file_name_large, parts_count)
|
||||||
parts = []
|
parts = []
|
||||||
|
|
||||||
with allure.step("Upload first part"):
|
with allure.step("Upload first part"):
|
||||||
upload_id = s3_client.create_multipart_upload(bucket, object_key)
|
upload_id = s3_gate_object.create_multipart_upload_s3(
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
self.s3_client, bucket, object_key
|
||||||
etag = s3_client.upload_part(bucket, object_key, upload_id, 1, part_files[0])
|
)
|
||||||
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
|
etag = s3_gate_object.upload_part_s3(
|
||||||
|
self.s3_client, bucket, object_key, upload_id, 1, part_files[0]
|
||||||
|
)
|
||||||
parts.append((1, etag))
|
parts.append((1, etag))
|
||||||
got_parts = s3_client.list_parts(bucket, object_key, upload_id)
|
got_parts = s3_gate_object.list_parts_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
assert len(got_parts) == 1, f"Expected {1} parts, got\n{got_parts}"
|
assert len(got_parts) == 1, f"Expected {1} parts, got\n{got_parts}"
|
||||||
|
|
||||||
with allure.step("Upload last parts"):
|
with allure.step("Upload last parts"):
|
||||||
for part_id, file_path in enumerate(part_files[1:], start=2):
|
for part_id, file_path in enumerate(part_files[1:], start=2):
|
||||||
etag = s3_client.upload_part(bucket, object_key, upload_id, part_id, file_path)
|
etag = s3_gate_object.upload_part_s3(
|
||||||
|
self.s3_client, bucket, object_key, upload_id, part_id, file_path
|
||||||
|
)
|
||||||
parts.append((part_id, etag))
|
parts.append((part_id, etag))
|
||||||
got_parts = s3_client.list_parts(bucket, object_key, upload_id)
|
got_parts = s3_gate_object.list_parts_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
s3_client.complete_multipart_upload(bucket, object_key, upload_id, parts)
|
s3_gate_object.complete_multipart_upload_s3(
|
||||||
|
self.s3_client, bucket, object_key, upload_id, parts
|
||||||
|
)
|
||||||
assert len(got_parts) == len(
|
assert len(got_parts) == len(
|
||||||
part_files
|
part_files
|
||||||
), f"Expected {parts_count} parts, got\n{got_parts}"
|
), f"Expected {parts_count} parts, got\n{got_parts}"
|
||||||
|
|
||||||
with allure.step("Check upload list is empty"):
|
with allure.step("Check upload list is empty"):
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Check we can get whole object from bucket"):
|
with allure.step("Check we can get whole object from bucket"):
|
||||||
got_object = s3_client.get_object(bucket, object_key)
|
got_object = s3_gate_object.get_object_s3(self.s3_client, bucket, object_key)
|
||||||
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
||||||
|
|
||||||
@allure.title("Test S3 Multipart abort")
|
@allure.title("Test S3 Multipart abord")
|
||||||
@pytest.mark.parametrize("bucket", [VersioningStatus.ENABLED], indirect=True)
|
def test_s3_abort_multipart(self):
|
||||||
def test_s3_abort_multipart(
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
self,
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
s3_client: S3ClientWrapper,
|
parts_count = 5
|
||||||
default_wallet: str,
|
file_name_large = generate_file(PART_SIZE * parts_count) # 5Mb - min part
|
||||||
bucket: str,
|
object_key = object_key_from_file_path(file_name_large)
|
||||||
simple_object_size: int,
|
part_files = split_file(file_name_large, parts_count)
|
||||||
complex_object_size: int,
|
parts = []
|
||||||
):
|
|
||||||
complex_file = generate_file(complex_object_size)
|
|
||||||
simple_file = generate_file(simple_object_size)
|
|
||||||
to_upload = [complex_file, complex_file, simple_file]
|
|
||||||
files_count = len(to_upload)
|
|
||||||
upload_key = "multipart_abort"
|
|
||||||
|
|
||||||
with allure.step(f"Get related container_id for bucket '{bucket}'"):
|
with allure.step("Upload first part"):
|
||||||
container_id = search_container_by_name(
|
upload_id = s3_gate_object.create_multipart_upload_s3(
|
||||||
default_wallet, bucket, self.shell, self.cluster.default_rpc_endpoint
|
self.s3_client, bucket, object_key
|
||||||
)
|
)
|
||||||
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
with allure.step("Create multipart upload"):
|
etag = s3_gate_object.upload_part_s3(
|
||||||
upload_id = s3_client.create_multipart_upload(bucket, upload_key)
|
self.s3_client, bucket, object_key, upload_id, 1, part_files[0]
|
||||||
|
|
||||||
with allure.step(f"Upload {files_count} files to multipart upload"):
|
|
||||||
for i, file in enumerate(to_upload, 1):
|
|
||||||
s3_client.upload_part(bucket, upload_key, upload_id, i, file)
|
|
||||||
|
|
||||||
with allure.step(f"Check that we have {files_count} files in bucket"):
|
|
||||||
parts = s3_client.list_parts(bucket, upload_key, upload_id)
|
|
||||||
assert len(parts) == files_count, f"Expected {files_count} parts, got\n{parts}"
|
|
||||||
|
|
||||||
with allure.step(f"Check that we have {files_count} files in container '{container_id}'"):
|
|
||||||
objects = list_objects(
|
|
||||||
default_wallet, self.shell, container_id, self.cluster.default_rpc_endpoint
|
|
||||||
)
|
)
|
||||||
assert (
|
parts.append((1, etag))
|
||||||
len(objects) == files_count
|
got_parts = s3_gate_object.list_parts_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
), f"Expected {files_count} objects in container, got\n{objects}"
|
assert len(got_parts) == 1, f"Expected {1} parts, got\n{got_parts}"
|
||||||
|
|
||||||
with allure.step("Abort multipart upload"):
|
with allure.step("Abort multipart upload"):
|
||||||
s3_client.abort_multipart_upload(bucket, upload_key, upload_id)
|
s3_gate_object.abort_multipart_uploads_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert not uploads, f"Expected no uploads in bucket {bucket}"
|
assert not uploads, f"Expected there is no uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Check that we have no files in bucket since upload was aborted"):
|
|
||||||
with pytest.raises(Exception, match=self.NO_SUCH_UPLOAD):
|
|
||||||
s3_client.list_parts(bucket, upload_key, upload_id)
|
|
||||||
|
|
||||||
with allure.step("Check that we have no files in container since upload was aborted"):
|
|
||||||
objects = list_objects(
|
|
||||||
default_wallet, self.shell, container_id, self.cluster.default_rpc_endpoint
|
|
||||||
)
|
|
||||||
assert len(objects) == 0, f"Expected no objects in container, got\n{objects}"
|
|
||||||
|
|
||||||
@allure.title("Test S3 Upload Part Copy")
|
@allure.title("Test S3 Upload Part Copy")
|
||||||
@pytest.mark.parametrize("bucket", [VersioningStatus.ENABLED], indirect=True)
|
def test_s3_multipart_copy(self):
|
||||||
def test_s3_multipart_copy(self, s3_client: S3ClientWrapper, bucket: str):
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
parts_count = 3
|
parts_count = 3
|
||||||
file_name_large = generate_file(PART_SIZE * parts_count) # 5Mb - min part
|
file_name_large = generate_file(PART_SIZE * parts_count) # 5Mb - min part
|
||||||
object_key = s3_helper.object_key_from_file_path(file_name_large)
|
object_key = object_key_from_file_path(file_name_large)
|
||||||
part_files = split_file(file_name_large, parts_count)
|
part_files = split_file(file_name_large, parts_count)
|
||||||
parts = []
|
parts = []
|
||||||
objs = []
|
objs = []
|
||||||
|
|
||||||
with allure.step(f"Put {parts_count} objects in bucket"):
|
with allure.step(f"Put {parts_count} objec in bucket"):
|
||||||
for part in part_files:
|
for part in part_files:
|
||||||
s3_client.put_object(bucket, part)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, part)
|
||||||
objs.append(s3_helper.object_key_from_file_path(part))
|
objs.append(object_key_from_file_path(part))
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket, objs)
|
check_objects_in_bucket(self.s3_client, bucket, objs)
|
||||||
|
|
||||||
with allure.step("Create multipart upload object"):
|
with allure.step("Create multipart upload object"):
|
||||||
upload_id = s3_client.create_multipart_upload(bucket, object_key)
|
upload_id = s3_gate_object.create_multipart_upload_s3(
|
||||||
uploads = s3_client.list_multipart_uploads(bucket)
|
self.s3_client, bucket, object_key
|
||||||
|
)
|
||||||
|
uploads = s3_gate_object.list_multipart_uploads_s3(self.s3_client, bucket)
|
||||||
assert uploads, f"Expected there are uploads in bucket {bucket}"
|
assert uploads, f"Expected there are uploads in bucket {bucket}"
|
||||||
|
|
||||||
with allure.step("Upload parts to multipart upload"):
|
with allure.step("Start multipart upload"):
|
||||||
for part_id, obj_key in enumerate(objs, start=1):
|
for part_id, obj_key in enumerate(objs, start=1):
|
||||||
etag = s3_client.upload_part_copy(
|
etag = s3_gate_object.upload_part_copy_s3(
|
||||||
bucket, object_key, upload_id, part_id, f"{bucket}/{obj_key}"
|
self.s3_client, bucket, object_key, upload_id, part_id, f"{bucket}/{obj_key}"
|
||||||
)
|
)
|
||||||
parts.append((part_id, etag))
|
parts.append((part_id, etag))
|
||||||
got_parts = s3_client.list_parts(bucket, object_key, upload_id)
|
got_parts = s3_gate_object.list_parts_s3(self.s3_client, bucket, object_key, upload_id)
|
||||||
|
s3_gate_object.complete_multipart_upload_s3(
|
||||||
with allure.step("Complete multipart upload"):
|
self.s3_client, bucket, object_key, upload_id, parts
|
||||||
s3_client.complete_multipart_upload(bucket, object_key, upload_id, parts)
|
)
|
||||||
assert len(got_parts) == len(
|
assert len(got_parts) == len(
|
||||||
part_files
|
part_files
|
||||||
), f"Expected {parts_count} parts, got\n{got_parts}"
|
), f"Expected {parts_count} parts, got\n{got_parts}"
|
||||||
|
|
||||||
with allure.step("Check we can get whole object from bucket"):
|
with allure.step("Check we can get whole object from bucket"):
|
||||||
got_object = s3_client.get_object(bucket, object_key)
|
got_object = s3_gate_object.get_object_s3(self.s3_client, bucket, object_key)
|
||||||
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
assert get_file_hash(got_object) == get_file_hash(file_name_large)
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -2,73 +2,71 @@ import os
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.cli.container import search_container_by_name
|
from python_keywords.container import search_container_by_name
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from python_keywords.storage_policy import get_simple_object_copies
|
||||||
from frostfs_testlib.steps.storage_policy import get_simple_object_copies
|
from s3_helper import check_objects_in_bucket, object_key_from_file_path, set_bucket_versioning
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import expect_not_raises
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
policy = f"{os.getcwd()}/pytest_tests/resources/files/policy.json"
|
policy = f"{os.getcwd()}/robot/resources/files/policy.json"
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize(
|
metafunc.parametrize(
|
||||||
"s3_client, s3_policy",
|
"s3_client",
|
||||||
[(AwsCliClient, policy), (Boto3ClientWrapper, policy)],
|
[("aws cli", policy), ("boto3", policy)],
|
||||||
indirect=True,
|
indirect=True,
|
||||||
ids=["aws cli", "boto3"],
|
ids=["aws cli", "boto3"],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
class TestS3GatePolicy(ClusterTestBase):
|
class TestS3GatePolicy(TestS3GateBase):
|
||||||
@allure.title("Test S3: Verify bucket creation with retention policy applied")
|
@allure.title("Test S3: Verify bucket creation with retention policy applied")
|
||||||
def test_s3_bucket_location(
|
def test_s3_bucket_location(self, simple_object_size):
|
||||||
self, default_wallet: str, s3_client: S3ClientWrapper, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path_1 = generate_file(simple_object_size)
|
file_path_1 = generate_file(simple_object_size)
|
||||||
file_name_1 = s3_helper.object_key_from_file_path(file_path_1)
|
file_name_1 = object_key_from_file_path(file_path_1)
|
||||||
file_path_2 = generate_file(simple_object_size)
|
file_path_2 = generate_file(simple_object_size)
|
||||||
file_name_2 = s3_helper.object_key_from_file_path(file_path_2)
|
file_name_2 = object_key_from_file_path(file_path_2)
|
||||||
|
|
||||||
with allure.step("Create two buckets with different bucket configuration"):
|
with allure.step("Create two buckets with different bucket configuration"):
|
||||||
bucket_1 = s3_client.create_bucket(location_constraint="complex")
|
bucket_1 = s3_gate_bucket.create_bucket_s3(
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket_1, VersioningStatus.ENABLED)
|
self.s3_client, bucket_configuration="complex"
|
||||||
bucket_2 = s3_client.create_bucket(location_constraint="rep-3")
|
)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket_2, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket_1, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
list_buckets = s3_client.list_buckets()
|
bucket_2 = s3_gate_bucket.create_bucket_s3(self.s3_client, bucket_configuration="rep-3")
|
||||||
|
set_bucket_versioning(self.s3_client, bucket_2, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
|
list_buckets = s3_gate_bucket.list_buckets_s3(self.s3_client)
|
||||||
assert (
|
assert (
|
||||||
bucket_1 in list_buckets and bucket_2 in list_buckets
|
bucket_1 in list_buckets and bucket_2 in list_buckets
|
||||||
), f"Expected two buckets {bucket_1, bucket_2}, got {list_buckets}"
|
), f"Expected two buckets {bucket_1, bucket_2}, got {list_buckets}"
|
||||||
|
|
||||||
with allure.step("Check head buckets"):
|
# with allure.step("Check head buckets"):
|
||||||
with expect_not_raises():
|
head_1 = s3_gate_bucket.head_bucket(self.s3_client, bucket_1)
|
||||||
s3_client.head_bucket(bucket_1)
|
head_2 = s3_gate_bucket.head_bucket(self.s3_client, bucket_2)
|
||||||
s3_client.head_bucket(bucket_2)
|
assert head_1 == {} or head_1.get("HEAD") == None, "Expected head is empty"
|
||||||
|
assert head_2 == {} or head_2.get("HEAD") == None, "Expected head is empty"
|
||||||
|
|
||||||
with allure.step("Put objects into buckets"):
|
with allure.step("Put objects into buckets"):
|
||||||
version_id_1 = s3_client.put_object(bucket_1, file_path_1)
|
version_id_1 = s3_gate_object.put_object_s3(self.s3_client, bucket_1, file_path_1)
|
||||||
version_id_2 = s3_client.put_object(bucket_2, file_path_2)
|
version_id_2 = s3_gate_object.put_object_s3(self.s3_client, bucket_2, file_path_2)
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_1, [file_name_1])
|
check_objects_in_bucket(self.s3_client, bucket_1, [file_name_1])
|
||||||
s3_helper.check_objects_in_bucket(s3_client, bucket_2, [file_name_2])
|
check_objects_in_bucket(self.s3_client, bucket_2, [file_name_2])
|
||||||
|
|
||||||
with allure.step("Check bucket location"):
|
with allure.step("Check bucket location"):
|
||||||
bucket_loc_1 = s3_client.get_bucket_location(bucket_1)
|
bucket_loc_1 = s3_gate_bucket.get_bucket_location(self.s3_client, bucket_1)
|
||||||
bucket_loc_2 = s3_client.get_bucket_location(bucket_2)
|
bucket_loc_2 = s3_gate_bucket.get_bucket_location(self.s3_client, bucket_2)
|
||||||
assert bucket_loc_1 == "complex"
|
assert bucket_loc_1 == "complex"
|
||||||
assert bucket_loc_2 == "rep-3"
|
assert bucket_loc_2 == "rep-3"
|
||||||
|
|
||||||
with allure.step("Check object policy"):
|
with allure.step("Check object policy"):
|
||||||
cid_1 = search_container_by_name(
|
cid_1 = search_container_by_name(
|
||||||
default_wallet,
|
self.wallet, bucket_1, shell=self.shell, endpoint=self.cluster.default_rpc_endpoint
|
||||||
bucket_1,
|
|
||||||
shell=self.shell,
|
|
||||||
endpoint=self.cluster.default_rpc_endpoint,
|
|
||||||
)
|
)
|
||||||
copies_1 = get_simple_object_copies(
|
copies_1 = get_simple_object_copies(
|
||||||
wallet=default_wallet,
|
wallet=self.wallet,
|
||||||
cid=cid_1,
|
cid=cid_1,
|
||||||
oid=version_id_1,
|
oid=version_id_1,
|
||||||
shell=self.shell,
|
shell=self.shell,
|
||||||
|
@ -76,13 +74,10 @@ class TestS3GatePolicy(ClusterTestBase):
|
||||||
)
|
)
|
||||||
assert copies_1 == 1
|
assert copies_1 == 1
|
||||||
cid_2 = search_container_by_name(
|
cid_2 = search_container_by_name(
|
||||||
default_wallet,
|
self.wallet, bucket_2, shell=self.shell, endpoint=self.cluster.default_rpc_endpoint
|
||||||
bucket_2,
|
|
||||||
shell=self.shell,
|
|
||||||
endpoint=self.cluster.default_rpc_endpoint,
|
|
||||||
)
|
)
|
||||||
copies_2 = get_simple_object_copies(
|
copies_2 = get_simple_object_copies(
|
||||||
wallet=default_wallet,
|
wallet=self.wallet,
|
||||||
cid=cid_2,
|
cid=cid_2,
|
||||||
oid=version_id_2,
|
oid=version_id_2,
|
||||||
shell=self.shell,
|
shell=self.shell,
|
||||||
|
@ -90,23 +85,17 @@ class TestS3GatePolicy(ClusterTestBase):
|
||||||
)
|
)
|
||||||
assert copies_2 == 3
|
assert copies_2 == 3
|
||||||
|
|
||||||
@allure.title("Test S3: bucket with unexisting location constraint")
|
|
||||||
def test_s3_bucket_wrong_location(self, s3_client: S3ClientWrapper):
|
|
||||||
with allure.step("Create bucket with unenxisting location constraint policy"):
|
|
||||||
with pytest.raises(Exception):
|
|
||||||
s3_client.create_bucket(location_constraint="UNEXISTING LOCATION CONSTRAINT")
|
|
||||||
|
|
||||||
@allure.title("Test S3: bucket policy ")
|
@allure.title("Test S3: bucket policy ")
|
||||||
def test_s3_bucket_policy(self, s3_client: S3ClientWrapper):
|
def test_s3_bucket_policy(self):
|
||||||
with allure.step("Create bucket with default policy"):
|
with allure.step("Create bucket with default policy"):
|
||||||
bucket = s3_client.create_bucket()
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
|
|
||||||
with allure.step("GetBucketPolicy"):
|
with allure.step("GetBucketPolicy"):
|
||||||
s3_client.get_bucket_policy(bucket)
|
s3_gate_bucket.get_bucket_policy(self.s3_client, bucket)
|
||||||
|
|
||||||
with allure.step("Put new policy"):
|
with allure.step("Put new policy"):
|
||||||
custom_policy = f"file://{os.getcwd()}/pytest_tests/resources/files/bucket_policy.json"
|
custom_policy = f"file://{os.getcwd()}/robot/resources/files/bucket_policy.json"
|
||||||
custom_policy = {
|
custom_policy = {
|
||||||
"Version": "2008-10-17",
|
"Version": "2008-10-17",
|
||||||
"Id": "aaaa-bbbb-cccc-dddd",
|
"Id": "aaaa-bbbb-cccc-dddd",
|
||||||
|
@ -121,19 +110,19 @@ class TestS3GatePolicy(ClusterTestBase):
|
||||||
],
|
],
|
||||||
}
|
}
|
||||||
|
|
||||||
s3_client.put_bucket_policy(bucket, custom_policy)
|
s3_gate_bucket.put_bucket_policy(self.s3_client, bucket, custom_policy)
|
||||||
with allure.step("GetBucketPolicy"):
|
with allure.step("GetBucketPolicy"):
|
||||||
policy_1 = s3_client.get_bucket_policy(bucket)
|
policy_1 = s3_gate_bucket.get_bucket_policy(self.s3_client, bucket)
|
||||||
print(policy_1)
|
print(policy_1)
|
||||||
|
|
||||||
@allure.title("Test S3: bucket CORS")
|
@allure.title("Test S3: bucket policy ")
|
||||||
def test_s3_cors(self, s3_client: S3ClientWrapper):
|
def test_s3_cors(self):
|
||||||
with allure.step("Create bucket without cors"):
|
with allure.step("Create bucket without cors"):
|
||||||
bucket = s3_client.create_bucket()
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
|
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
bucket_cors = s3_client.get_bucket_cors(bucket)
|
bucket_cors = s3_gate_bucket.get_bucket_cors(self.s3_client, bucket)
|
||||||
|
|
||||||
with allure.step("Put bucket cors"):
|
with allure.step("Put bucket cors"):
|
||||||
cors = {
|
cors = {
|
||||||
|
@ -153,14 +142,14 @@ class TestS3GatePolicy(ClusterTestBase):
|
||||||
},
|
},
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
s3_client.put_bucket_cors(bucket, cors)
|
s3_gate_bucket.put_bucket_cors(self.s3_client, bucket, cors)
|
||||||
bucket_cors = s3_client.get_bucket_cors(bucket)
|
bucket_cors = s3_gate_bucket.get_bucket_cors(self.s3_client, bucket)
|
||||||
assert bucket_cors == cors.get(
|
assert bucket_cors == cors.get(
|
||||||
"CORSRules"
|
"CORSRules"
|
||||||
), f"Expected CORSRules must be {cors.get('CORSRules')}"
|
), f"Expected corsrules must be {cors.get('CORSRules')}"
|
||||||
|
|
||||||
with allure.step("delete bucket cors"):
|
with allure.step("delete bucket cors"):
|
||||||
s3_client.delete_bucket_cors(bucket)
|
s3_gate_bucket.delete_bucket_cors(self.s3_client, bucket)
|
||||||
|
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
bucket_cors = s3_client.get_bucket_cors(bucket)
|
bucket_cors = s3_gate_bucket.get_bucket_cors(self.s3_client, bucket)
|
||||||
|
|
|
@ -1,23 +1,27 @@
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
from random import choice
|
from random import choice
|
||||||
from string import ascii_letters
|
from string import ascii_letters
|
||||||
from typing import Tuple
|
from typing import Tuple
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from s3_helper import check_tags_by_bucket, check_tags_by_object, object_key_from_file_path
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_tagging
|
@pytest.mark.s3_gate_tagging
|
||||||
class TestS3GateTagging:
|
class TestS3GateTagging(TestS3GateBase):
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def create_tags(count: int) -> Tuple[list, list]:
|
def create_tags(count: int) -> Tuple[list, list]:
|
||||||
tags = []
|
tags = []
|
||||||
|
@ -28,84 +32,82 @@ class TestS3GateTagging:
|
||||||
return tags
|
return tags
|
||||||
|
|
||||||
@allure.title("Test S3: Object tagging")
|
@allure.title("Test S3: Object tagging")
|
||||||
def test_s3_object_tagging(
|
def test_s3_object_tagging(self, bucket, simple_object_size):
|
||||||
self, s3_client: S3ClientWrapper, bucket: str, simple_object_size: int
|
|
||||||
):
|
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = object_key_from_file_path(file_path)
|
||||||
|
|
||||||
with allure.step("Put with 3 tags object into bucket"):
|
with allure.step("Put with 3 tags object into bucket"):
|
||||||
tag_1 = "Tag1=Value1"
|
tag_1 = "Tag1=Value1"
|
||||||
s3_client.put_object(bucket, file_path, tagging=tag_1)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path, Tagging=tag_1)
|
||||||
got_tags = s3_client.get_object_tagging(bucket, file_name)
|
got_tags = s3_gate_object.get_object_tagging(self.s3_client, bucket, file_name)
|
||||||
assert got_tags, f"Expected tags, got {got_tags}"
|
assert got_tags, f"Expected tags, got {got_tags}"
|
||||||
assert got_tags == [{"Key": "Tag1", "Value": "Value1"}], "Tags must be the same"
|
assert got_tags == [{"Key": "Tag1", "Value": "Value1"}], "Tags must be the same"
|
||||||
|
|
||||||
with allure.step("Put 10 new tags for object"):
|
with allure.step("Put 10 new tags for object"):
|
||||||
tags_2 = self.create_tags(10)
|
tags_2 = self.create_tags(10)
|
||||||
s3_client.put_object_tagging(bucket, file_name, tags=tags_2)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, file_name, tags=tags_2)
|
||||||
s3_helper.check_tags_by_object(
|
check_tags_by_object(self.s3_client, bucket, file_name, tags_2, [("Tag1", "Value1")])
|
||||||
s3_client, bucket, file_name, tags_2, [("Tag1", "Value1")]
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Put 10 extra new tags for object"):
|
with allure.step("Put 10 extra new tags for object"):
|
||||||
tags_3 = self.create_tags(10)
|
tags_3 = self.create_tags(10)
|
||||||
s3_client.put_object_tagging(bucket, file_name, tags=tags_3)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, file_name, tags=tags_3)
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, file_name, tags_3, tags_2)
|
check_tags_by_object(self.s3_client, bucket, file_name, tags_3, tags_2)
|
||||||
|
|
||||||
with allure.step("Copy one object with tag"):
|
with allure.step("Copy one object with tag"):
|
||||||
copy_obj_path_1 = s3_client.copy_object(bucket, file_name, tagging_directive="COPY")
|
copy_obj_path_1 = s3_gate_object.copy_object_s3(
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, copy_obj_path_1, tags_3, tags_2)
|
self.s3_client, bucket, file_name, tagging_directive="COPY"
|
||||||
|
)
|
||||||
|
check_tags_by_object(self.s3_client, bucket, copy_obj_path_1, tags_3, tags_2)
|
||||||
|
|
||||||
with allure.step("Put 11 new tags to object and expect an error"):
|
with allure.step("Put 11 new tags to object and expect an error"):
|
||||||
tags_4 = self.create_tags(11)
|
tags_4 = self.create_tags(11)
|
||||||
with pytest.raises(Exception, match=r".*Object tags cannot be greater than 10*"):
|
with pytest.raises(Exception, match=r".*Object tags cannot be greater than 10*"):
|
||||||
# An error occurred (BadRequest) when calling the PutObjectTagging operation: Object tags cannot be greater than 10
|
# An error occurred (BadRequest) when calling the PutObjectTagging operation: Object tags cannot be greater than 10
|
||||||
s3_client.put_object_tagging(bucket, file_name, tags=tags_4)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, file_name, tags=tags_4)
|
||||||
|
|
||||||
with allure.step("Put empty tag"):
|
with allure.step("Put empty tag"):
|
||||||
tags_5 = []
|
tags_5 = []
|
||||||
s3_client.put_object_tagging(bucket, file_name, tags=tags_5)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, file_name, tags=tags_5)
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, file_name, [])
|
check_tags_by_object(self.s3_client, bucket, file_name, [])
|
||||||
|
|
||||||
with allure.step("Put 10 object tags"):
|
with allure.step("Put 10 object tags"):
|
||||||
tags_6 = self.create_tags(10)
|
tags_6 = self.create_tags(10)
|
||||||
s3_client.put_object_tagging(bucket, file_name, tags=tags_6)
|
s3_gate_object.put_object_tagging(self.s3_client, bucket, file_name, tags=tags_6)
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, file_name, tags_6)
|
check_tags_by_object(self.s3_client, bucket, file_name, tags_6)
|
||||||
|
|
||||||
with allure.step("Delete tags by delete-object-tagging"):
|
with allure.step("Delete tags by delete-object-tagging"):
|
||||||
s3_client.delete_object_tagging(bucket, file_name)
|
s3_gate_object.delete_object_tagging(self.s3_client, bucket, file_name)
|
||||||
s3_helper.check_tags_by_object(s3_client, bucket, file_name, [])
|
check_tags_by_object(self.s3_client, bucket, file_name, [])
|
||||||
|
|
||||||
@allure.title("Test S3: bucket tagging")
|
@allure.title("Test S3: bucket tagging")
|
||||||
def test_s3_bucket_tagging(self, s3_client: S3ClientWrapper, bucket: str):
|
def test_s3_bucket_tagging(self, bucket):
|
||||||
|
|
||||||
with allure.step("Put 10 bucket tags"):
|
with allure.step("Put 10 bucket tags"):
|
||||||
tags_1 = self.create_tags(10)
|
tags_1 = self.create_tags(10)
|
||||||
s3_client.put_bucket_tagging(bucket, tags_1)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, tags_1)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, tags_1)
|
check_tags_by_bucket(self.s3_client, bucket, tags_1)
|
||||||
|
|
||||||
with allure.step("Put new 10 bucket tags"):
|
with allure.step("Put new 10 bucket tags"):
|
||||||
tags_2 = self.create_tags(10)
|
tags_2 = self.create_tags(10)
|
||||||
s3_client.put_bucket_tagging(bucket, tags_2)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, tags_2)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, tags_2, tags_1)
|
check_tags_by_bucket(self.s3_client, bucket, tags_2, tags_1)
|
||||||
|
|
||||||
with allure.step("Put 11 new tags to bucket and expect an error"):
|
with allure.step("Put 11 new tags to bucket and expect an error"):
|
||||||
tags_3 = self.create_tags(11)
|
tags_3 = self.create_tags(11)
|
||||||
with pytest.raises(Exception, match=r".*Object tags cannot be greater than 10.*"):
|
with pytest.raises(Exception, match=r".*Object tags cannot be greater than 10.*"):
|
||||||
# An error occurred (BadRequest) when calling the PutBucketTagging operation (reached max retries: 0): Object tags cannot be greater than 10
|
# An error occurred (BadRequest) when calling the PutBucketTagging operation (reached max retries: 0): Object tags cannot be greater than 10
|
||||||
s3_client.put_bucket_tagging(bucket, tags_3)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, tags_3)
|
||||||
|
|
||||||
with allure.step("Put empty tag"):
|
with allure.step("Put empty tag"):
|
||||||
tags_4 = []
|
tags_4 = []
|
||||||
s3_client.put_bucket_tagging(bucket, tags_4)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, tags_4)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, tags_4)
|
check_tags_by_bucket(self.s3_client, bucket, tags_4)
|
||||||
|
|
||||||
with allure.step("Put new 10 bucket tags"):
|
with allure.step("Put new 10 bucket tags"):
|
||||||
tags_5 = self.create_tags(10)
|
tags_5 = self.create_tags(10)
|
||||||
s3_client.put_bucket_tagging(bucket, tags_5)
|
s3_gate_bucket.put_bucket_tagging(self.s3_client, bucket, tags_5)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, tags_5, tags_2)
|
check_tags_by_bucket(self.s3_client, bucket, tags_5, tags_2)
|
||||||
|
|
||||||
with allure.step("Delete tags by delete-bucket-tagging"):
|
with allure.step("Delete tags by delete-bucket-tagging"):
|
||||||
s3_client.delete_bucket_tagging(bucket)
|
s3_gate_bucket.delete_bucket_tagging(self.s3_client, bucket)
|
||||||
s3_helper.check_tags_by_bucket(s3_client, bucket, [])
|
check_tags_by_bucket(self.s3_client, bucket, [])
|
||||||
|
|
|
@ -2,41 +2,48 @@ import os
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.s3 import AwsCliClient, Boto3ClientWrapper, S3ClientWrapper, VersioningStatus
|
from file_helper import generate_file, generate_file_with_content
|
||||||
from frostfs_testlib.steps.s3 import s3_helper
|
from s3_helper import set_bucket_versioning
|
||||||
from frostfs_testlib.utils.file_utils import generate_file, generate_file_with_content
|
|
||||||
|
from steps import s3_gate_bucket, s3_gate_object
|
||||||
|
from steps.s3_gate_base import TestS3GateBase
|
||||||
|
|
||||||
|
|
||||||
def pytest_generate_tests(metafunc: pytest.Metafunc):
|
def pytest_generate_tests(metafunc):
|
||||||
if "s3_client" in metafunc.fixturenames:
|
if "s3_client" in metafunc.fixturenames:
|
||||||
metafunc.parametrize("s3_client", [AwsCliClient, Boto3ClientWrapper], indirect=True)
|
metafunc.parametrize("s3_client", ["aws cli", "boto3"], indirect=True)
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.s3_gate
|
@pytest.mark.s3_gate
|
||||||
@pytest.mark.s3_gate_versioning
|
@pytest.mark.s3_gate_versioning
|
||||||
class TestS3GateVersioning:
|
class TestS3GateVersioning(TestS3GateBase):
|
||||||
|
@staticmethod
|
||||||
|
def object_key_from_file_path(full_path: str) -> str:
|
||||||
|
return os.path.basename(full_path)
|
||||||
|
|
||||||
@allure.title("Test S3: try to disable versioning")
|
@allure.title("Test S3: try to disable versioning")
|
||||||
def test_s3_version_off(self, s3_client: S3ClientWrapper):
|
def test_s3_version_off(self):
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=True)
|
|
||||||
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, True)
|
||||||
with pytest.raises(Exception):
|
with pytest.raises(Exception):
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.SUSPENDED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.SUSPENDED)
|
||||||
|
|
||||||
@allure.title("Test S3: Enable and disable versioning")
|
@allure.title("Test S3: Enable and disable versioning")
|
||||||
def test_s3_version(self, s3_client: S3ClientWrapper, simple_object_size: int):
|
def test_s3_version(self, simple_object_size):
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
file_name = s3_helper.object_key_from_file_path(file_path)
|
file_name = self.object_key_from_file_path(file_path)
|
||||||
bucket_objects = [file_name]
|
bucket_objects = [file_name]
|
||||||
bucket = s3_client.create_bucket(object_lock_enabled_for_bucket=False)
|
bucket = s3_gate_bucket.create_bucket_s3(self.s3_client, False)
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.SUSPENDED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.SUSPENDED)
|
||||||
|
|
||||||
with allure.step("Put object into bucket"):
|
with allure.step("Put object into bucket"):
|
||||||
s3_client.put_object(bucket, file_path)
|
s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
objects_list = s3_client.list_objects(bucket)
|
objects_list = s3_gate_object.list_objects_s3(self.s3_client, bucket)
|
||||||
assert (
|
assert (
|
||||||
objects_list == bucket_objects
|
objects_list == bucket_objects
|
||||||
), f"Expected list with single objects in bucket, got {objects_list}"
|
), f"Expected list with single objects in bucket, got {objects_list}"
|
||||||
object_version = s3_client.list_objects_versions(bucket)
|
object_version = s3_gate_object.list_objects_versions_s3(self.s3_client, bucket)
|
||||||
actual_version = [
|
actual_version = [
|
||||||
version.get("VersionId")
|
version.get("VersionId")
|
||||||
for version in object_version
|
for version in object_version
|
||||||
|
@ -45,20 +52,20 @@ class TestS3GateVersioning:
|
||||||
assert actual_version == [
|
assert actual_version == [
|
||||||
"null"
|
"null"
|
||||||
], f"Expected version is null in list-object-versions, got {object_version}"
|
], f"Expected version is null in list-object-versions, got {object_version}"
|
||||||
object_0 = s3_client.head_object(bucket, file_name)
|
object_0 = s3_gate_object.head_object_s3(self.s3_client, bucket, file_name)
|
||||||
assert (
|
assert (
|
||||||
object_0.get("VersionId") == "null"
|
object_0.get("VersionId") == "null"
|
||||||
), f"Expected version is null in head-object, got {object_0.get('VersionId')}"
|
), f"Expected version is null in head-object, got {object_0.get('VersionId')}"
|
||||||
|
|
||||||
s3_helper.set_bucket_versioning(s3_client, bucket, VersioningStatus.ENABLED)
|
set_bucket_versioning(self.s3_client, bucket, s3_gate_bucket.VersioningStatus.ENABLED)
|
||||||
|
|
||||||
with allure.step("Put several versions of object into bucket"):
|
with allure.step("Put several versions of object into bucket"):
|
||||||
version_id_1 = s3_client.put_object(bucket, file_path)
|
version_id_1 = s3_gate_object.put_object_s3(self.s3_client, bucket, file_path)
|
||||||
file_name_1 = generate_file_with_content(simple_object_size, file_path=file_path)
|
file_name_1 = generate_file_with_content(simple_object_size, file_path=file_path)
|
||||||
version_id_2 = s3_client.put_object(bucket, file_name_1)
|
version_id_2 = s3_gate_object.put_object_s3(self.s3_client, bucket, file_name_1)
|
||||||
|
|
||||||
with allure.step("Check bucket shows all versions"):
|
with allure.step("Check bucket shows all versions"):
|
||||||
versions = s3_client.list_objects_versions(bucket)
|
versions = s3_gate_object.list_objects_versions_s3(self.s3_client, bucket)
|
||||||
obj_versions = [
|
obj_versions = [
|
||||||
version.get("VersionId") for version in versions if version.get("Key") == file_name
|
version.get("VersionId") for version in versions if version.get("Key") == file_name
|
||||||
]
|
]
|
||||||
|
@ -67,19 +74,25 @@ class TestS3GateVersioning:
|
||||||
), f"Expected object has versions: {version_id_1, version_id_2, 'null'}"
|
), f"Expected object has versions: {version_id_1, version_id_2, 'null'}"
|
||||||
|
|
||||||
with allure.step("Get object"):
|
with allure.step("Get object"):
|
||||||
object_1 = s3_client.get_object(bucket, file_name, full_output=True)
|
object_1 = s3_gate_object.get_object_s3(
|
||||||
|
self.s3_client, bucket, file_name, full_output=True
|
||||||
|
)
|
||||||
assert (
|
assert (
|
||||||
object_1.get("VersionId") == version_id_2
|
object_1.get("VersionId") == version_id_2
|
||||||
), f"Get object with version {version_id_2}"
|
), f"Get object with version {version_id_2}"
|
||||||
|
|
||||||
with allure.step("Get first version of object"):
|
with allure.step("Get first version of object"):
|
||||||
object_2 = s3_client.get_object(bucket, file_name, version_id_1, full_output=True)
|
object_2 = s3_gate_object.get_object_s3(
|
||||||
|
self.s3_client, bucket, file_name, version_id_1, full_output=True
|
||||||
|
)
|
||||||
assert (
|
assert (
|
||||||
object_2.get("VersionId") == version_id_1
|
object_2.get("VersionId") == version_id_1
|
||||||
), f"Get object with version {version_id_1}"
|
), f"Get object with version {version_id_1}"
|
||||||
|
|
||||||
with allure.step("Get second version of object"):
|
with allure.step("Get second version of object"):
|
||||||
object_3 = s3_client.get_object(bucket, file_name, version_id_2, full_output=True)
|
object_3 = s3_gate_object.get_object_s3(
|
||||||
|
self.s3_client, bucket, file_name, version_id_2, full_output=True
|
||||||
|
)
|
||||||
assert (
|
assert (
|
||||||
object_3.get("VersionId") == version_id_2
|
object_3.get("VersionId") == version_id_2
|
||||||
), f"Get object with version {version_id_2}"
|
), f"Get object with version {version_id_2}"
|
||||||
|
|
|
@ -1,18 +1,14 @@
|
||||||
import logging
|
import logging
|
||||||
import os
|
|
||||||
from http import HTTPStatus
|
from http import HTTPStatus
|
||||||
from re import match
|
from re import match
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
import requests
|
import requests
|
||||||
from frostfs_testlib.hosting import Hosting
|
from binary_version_helper import get_remote_binaries_versions
|
||||||
from frostfs_testlib.resources.common import ASSETS_DIR
|
from common import BIN_VERSIONS_FILE
|
||||||
from frostfs_testlib.utils.env_utils import read_env_properties, save_env_properties
|
from env_properties import read_env_properties, save_env_properties
|
||||||
from frostfs_testlib.utils.version_utils import get_remote_binaries_versions
|
from neofs_testlib.hosting import Hosting
|
||||||
from pytest import FixtureRequest
|
|
||||||
|
|
||||||
from pytest_tests.resources.common import BIN_VERSIONS_FILE
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -21,7 +17,7 @@ logger = logging.getLogger("NeoLogger")
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
@pytest.mark.check_binaries
|
@pytest.mark.check_binaries
|
||||||
@pytest.mark.skip("Skipped due to https://j.yadro.com/browse/OBJECT-628")
|
@pytest.mark.skip("Skipped due to https://j.yadro.com/browse/OBJECT-628")
|
||||||
def test_binaries_versions(request: FixtureRequest, hosting: Hosting):
|
def test_binaries_versions(request, hosting: Hosting):
|
||||||
"""
|
"""
|
||||||
Compare binaries versions from external source (url) and deployed on servers.
|
Compare binaries versions from external source (url) and deployed on servers.
|
||||||
"""
|
"""
|
||||||
|
@ -32,9 +28,7 @@ def test_binaries_versions(request: FixtureRequest, hosting: Hosting):
|
||||||
with allure.step("Get binaries versions from servers"):
|
with allure.step("Get binaries versions from servers"):
|
||||||
got_versions = get_remote_binaries_versions(hosting)
|
got_versions = get_remote_binaries_versions(hosting)
|
||||||
|
|
||||||
environment_dir = request.config.getoption("--alluredir") or ASSETS_DIR
|
env_properties = read_env_properties(request.config)
|
||||||
env_file = os.path.join(environment_dir, "environment.properties")
|
|
||||||
env_properties = read_env_properties(env_file)
|
|
||||||
|
|
||||||
# compare versions from servers and file
|
# compare versions from servers and file
|
||||||
failed_versions = {}
|
failed_versions = {}
|
||||||
|
@ -50,7 +44,7 @@ def test_binaries_versions(request: FixtureRequest, hosting: Hosting):
|
||||||
additional_env_properties[binary] = actual_version
|
additional_env_properties[binary] = actual_version
|
||||||
|
|
||||||
if env_properties and additional_env_properties:
|
if env_properties and additional_env_properties:
|
||||||
save_env_properties(env_file, additional_env_properties)
|
save_env_properties(request.config, additional_env_properties)
|
||||||
|
|
||||||
# create clear beautiful error with aggregation info
|
# create clear beautiful error with aggregation info
|
||||||
if failed_versions:
|
if failed_versions:
|
||||||
|
|
|
@ -1,9 +1,9 @@
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletFactory, WalletInfo
|
from wallet import WalletFactory, WalletFile
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def owner_wallet(wallet_factory: WalletFactory) -> WalletInfo:
|
def owner_wallet(wallet_factory: WalletFactory) -> WalletFile:
|
||||||
"""
|
"""
|
||||||
Returns wallet which owns containers and objects
|
Returns wallet which owns containers and objects
|
||||||
"""
|
"""
|
||||||
|
@ -11,7 +11,7 @@ def owner_wallet(wallet_factory: WalletFactory) -> WalletInfo:
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def user_wallet(wallet_factory: WalletFactory) -> WalletInfo:
|
def user_wallet(wallet_factory: WalletFactory) -> WalletFile:
|
||||||
"""
|
"""
|
||||||
Returns wallet which will use objects from owner via static session
|
Returns wallet which will use objects from owner via static session
|
||||||
"""
|
"""
|
||||||
|
@ -19,7 +19,7 @@ def user_wallet(wallet_factory: WalletFactory) -> WalletInfo:
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def stranger_wallet(wallet_factory: WalletFactory) -> WalletInfo:
|
def stranger_wallet(wallet_factory: WalletFactory) -> WalletFile:
|
||||||
"""
|
"""
|
||||||
Returns stranger wallet which should fail to obtain data
|
Returns stranger wallet which should fail to obtain data
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -2,14 +2,15 @@ import random
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.common import DEFAULT_WALLET_PASS
|
from cluster_test_base import ClusterTestBase
|
||||||
from frostfs_testlib.resources.error_patterns import SESSION_NOT_FOUND
|
from common import WALLET_PASS
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.steps.cli.object import delete_object, put_object, put_object_to_random_node
|
from grpc_responses import SESSION_NOT_FOUND
|
||||||
from frostfs_testlib.steps.session_token import create_session_token
|
from neofs_testlib.utils.wallet import get_last_address_from_wallet
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.utils import wallet_utils
|
from python_keywords.neofs_verbs import delete_object, put_object, put_object_to_random_node
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
from steps.session_token import create_session_token
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.sanity
|
@pytest.mark.sanity
|
||||||
|
@ -37,19 +38,21 @@ class TestDynamicObjectSession(ClusterTestBase):
|
||||||
|
|
||||||
with allure.step("Init wallet"):
|
with allure.step("Init wallet"):
|
||||||
wallet = default_wallet
|
wallet = default_wallet
|
||||||
address = wallet_utils.get_last_address_from_wallet(wallet, "")
|
address = get_last_address_from_wallet(wallet, "")
|
||||||
|
|
||||||
with allure.step("Nodes Settlements"):
|
with allure.step("Nodes Settlements"):
|
||||||
session_token_node, container_node, non_container_node = random.sample(
|
(
|
||||||
self.cluster.storage_nodes, 3
|
session_token_node,
|
||||||
)
|
container_node,
|
||||||
|
non_container_node,
|
||||||
|
) = random.sample(self.cluster.storage_nodes, 3)
|
||||||
|
|
||||||
with allure.step("Create Session Token"):
|
with allure.step("Create Session Token"):
|
||||||
session_token = create_session_token(
|
session_token = create_session_token(
|
||||||
shell=self.shell,
|
shell=self.shell,
|
||||||
owner=address,
|
owner=address,
|
||||||
wallet_path=wallet,
|
wallet_path=wallet,
|
||||||
wallet_password=DEFAULT_WALLET_PASS,
|
wallet_password=WALLET_PASS,
|
||||||
rpc_endpoint=session_token_node.get_rpc_endpoint(),
|
rpc_endpoint=session_token_node.get_rpc_endpoint(),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -2,15 +2,15 @@ import logging
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.error_patterns import (
|
from cluster import Cluster
|
||||||
EXPIRED_SESSION_TOKEN,
|
from cluster_test_base import ClusterTestBase
|
||||||
MALFORMED_REQUEST,
|
from epoch import ensure_fresh_epoch
|
||||||
OBJECT_ACCESS_DENIED,
|
from file_helper import generate_file
|
||||||
OBJECT_NOT_FOUND,
|
from grpc_responses import MALFORMED_REQUEST, OBJECT_ACCESS_DENIED, OBJECT_NOT_FOUND
|
||||||
)
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.shell import Shell
|
from pytest import FixtureRequest
|
||||||
from frostfs_testlib.steps.cli.container import create_container
|
from python_keywords.container import create_container
|
||||||
from frostfs_testlib.steps.cli.object import (
|
from python_keywords.neofs_verbs import (
|
||||||
delete_object,
|
delete_object,
|
||||||
get_object,
|
get_object,
|
||||||
get_object_from_random_node,
|
get_object_from_random_node,
|
||||||
|
@ -20,8 +20,11 @@ from frostfs_testlib.steps.cli.object import (
|
||||||
put_object_to_random_node,
|
put_object_to_random_node,
|
||||||
search_object,
|
search_object,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.epoch import ensure_fresh_epoch
|
from test_control import expect_not_raises
|
||||||
from frostfs_testlib.steps.session_token import (
|
from wallet import WalletFile
|
||||||
|
|
||||||
|
from helpers.storage_object_info import StorageObjectInfo
|
||||||
|
from steps.session_token import (
|
||||||
INVALID_SIGNATURE,
|
INVALID_SIGNATURE,
|
||||||
UNRELATED_CONTAINER,
|
UNRELATED_CONTAINER,
|
||||||
UNRELATED_KEY,
|
UNRELATED_KEY,
|
||||||
|
@ -33,14 +36,7 @@ from frostfs_testlib.steps.session_token import (
|
||||||
get_object_signed_token,
|
get_object_signed_token,
|
||||||
sign_session_token,
|
sign_session_token,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.storage_object import delete_objects
|
from steps.storage_object import delete_objects
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
|
||||||
from frostfs_testlib.storage.dataclasses.storage_object_info import StorageObjectInfo
|
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.testing.test_control import expect_not_raises
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
from pytest import FixtureRequest
|
|
||||||
|
|
||||||
logger = logging.getLogger("NeoLogger")
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
@ -49,7 +45,7 @@ RANGE_OFFSET_FOR_COMPLEX_OBJECT = 200
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def storage_containers(
|
def storage_containers(
|
||||||
owner_wallet: WalletInfo, client_shell: Shell, cluster: Cluster
|
owner_wallet: WalletFile, client_shell: Shell, cluster: Cluster
|
||||||
) -> list[str]:
|
) -> list[str]:
|
||||||
cid = create_container(
|
cid = create_container(
|
||||||
owner_wallet.path, shell=client_shell, endpoint=cluster.default_rpc_endpoint
|
owner_wallet.path, shell=client_shell, endpoint=cluster.default_rpc_endpoint
|
||||||
|
@ -67,7 +63,7 @@ def storage_containers(
|
||||||
scope="module",
|
scope="module",
|
||||||
)
|
)
|
||||||
def storage_objects(
|
def storage_objects(
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
client_shell: Shell,
|
client_shell: Shell,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
cluster: Cluster,
|
cluster: Cluster,
|
||||||
|
@ -123,8 +119,8 @@ def get_ranges(
|
||||||
|
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def static_sessions(
|
def static_sessions(
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
client_shell: Shell,
|
client_shell: Shell,
|
||||||
|
@ -160,7 +156,7 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
def test_static_session_read(
|
def test_static_session_read(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
method_under_test,
|
method_under_test,
|
||||||
|
@ -186,13 +182,14 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session with range operations")
|
@allure.title("Validate static session with range operations")
|
||||||
|
@pytest.mark.static_session
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
"method_under_test,verb",
|
"method_under_test,verb",
|
||||||
[(get_range, ObjectVerb.RANGE), (get_range_hash, ObjectVerb.RANGEHASH)],
|
[(get_range, ObjectVerb.RANGE), (get_range_hash, ObjectVerb.RANGEHASH)],
|
||||||
)
|
)
|
||||||
def test_static_session_range(
|
def test_static_session_range(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
method_under_test,
|
method_under_test,
|
||||||
|
@ -225,9 +222,12 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session with search operation")
|
@allure.title("Validate static session with search operation")
|
||||||
|
@pytest.mark.static_session
|
||||||
|
@pytest.mark.xfail
|
||||||
|
# (see https://github.com/nspcc-dev/neofs-node/issues/2030)
|
||||||
def test_static_session_search(
|
def test_static_session_search(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -247,12 +247,13 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
session=static_sessions[ObjectVerb.SEARCH],
|
session=static_sessions[ObjectVerb.SEARCH],
|
||||||
root=True,
|
root=True,
|
||||||
)
|
)
|
||||||
assert sorted(expected_object_ids) == sorted(actual_object_ids)
|
assert expected_object_ids == actual_object_ids
|
||||||
|
|
||||||
@allure.title("Validate static session with object id not in session")
|
@allure.title("Validate static session with object id not in session")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_unrelated_object(
|
def test_static_session_unrelated_object(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -274,9 +275,10 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session with user id not in session")
|
@allure.title("Validate static session with user id not in session")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_head_unrelated_user(
|
def test_static_session_head_unrelated_user(
|
||||||
self,
|
self,
|
||||||
stranger_wallet: WalletInfo,
|
stranger_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -300,9 +302,10 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session with wrong verb in session")
|
@allure.title("Validate static session with wrong verb in session")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_head_wrong_verb(
|
def test_static_session_head_wrong_verb(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -326,9 +329,10 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session with container id not in session")
|
@allure.title("Validate static session with container id not in session")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_unrelated_container(
|
def test_static_session_unrelated_container(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
|
@ -353,11 +357,12 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session which signed by another wallet")
|
@allure.title("Validate static session which signed by another wallet")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_signed_by_other(
|
def test_static_session_signed_by_other(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
stranger_wallet: WalletInfo,
|
stranger_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -391,10 +396,11 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session which signed for another container")
|
@allure.title("Validate static session which signed for another container")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_signed_for_other_container(
|
def test_static_session_signed_for_other_container(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -429,10 +435,11 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session which wasn't signed")
|
@allure.title("Validate static session which wasn't signed")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_without_sign(
|
def test_static_session_without_sign(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -465,10 +472,11 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session which expires at next epoch")
|
@allure.title("Validate static session which expires at next epoch")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_expiration_at_next(
|
def test_static_session_expiration_at_next(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -486,60 +494,44 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
object_id = storage_objects[0].oid
|
object_id = storage_objects[0].oid
|
||||||
expiration = Lifetime(epoch + 1, epoch, epoch)
|
expiration = Lifetime(epoch + 1, epoch, epoch)
|
||||||
|
|
||||||
with allure.step("Create session token"):
|
token_expire_at_next_epoch = get_object_signed_token(
|
||||||
token_expire_at_next_epoch = get_object_signed_token(
|
owner_wallet,
|
||||||
owner_wallet,
|
user_wallet,
|
||||||
user_wallet,
|
container,
|
||||||
|
storage_objects,
|
||||||
|
ObjectVerb.HEAD,
|
||||||
|
self.shell,
|
||||||
|
temp_directory,
|
||||||
|
expiration,
|
||||||
|
)
|
||||||
|
|
||||||
|
head_object(
|
||||||
|
user_wallet.path,
|
||||||
|
container,
|
||||||
|
object_id,
|
||||||
|
self.shell,
|
||||||
|
self.cluster.default_rpc_endpoint,
|
||||||
|
session=token_expire_at_next_epoch,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.tick_epoch()
|
||||||
|
|
||||||
|
with pytest.raises(Exception, match=MALFORMED_REQUEST):
|
||||||
|
head_object(
|
||||||
|
user_wallet.path,
|
||||||
container,
|
container,
|
||||||
storage_objects,
|
object_id,
|
||||||
ObjectVerb.HEAD,
|
|
||||||
self.shell,
|
self.shell,
|
||||||
temp_directory,
|
self.cluster.default_rpc_endpoint,
|
||||||
expiration,
|
session=token_expire_at_next_epoch,
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step("Object should be available with session token after token creation"):
|
|
||||||
with expect_not_raises():
|
|
||||||
head_object(
|
|
||||||
user_wallet.path,
|
|
||||||
container,
|
|
||||||
object_id,
|
|
||||||
self.shell,
|
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_expire_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step(
|
|
||||||
"Object should be available at last epoch before session token expiration"
|
|
||||||
):
|
|
||||||
self.tick_epoch()
|
|
||||||
with expect_not_raises():
|
|
||||||
head_object(
|
|
||||||
user_wallet.path,
|
|
||||||
container,
|
|
||||||
object_id,
|
|
||||||
self.shell,
|
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_expire_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Object should NOT be available after session token expiration epoch"):
|
|
||||||
self.tick_epoch()
|
|
||||||
with pytest.raises(Exception, match=EXPIRED_SESSION_TOKEN):
|
|
||||||
head_object(
|
|
||||||
user_wallet.path,
|
|
||||||
container,
|
|
||||||
object_id,
|
|
||||||
self.shell,
|
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_expire_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
@allure.title("Validate static session which is valid starting from next epoch")
|
@allure.title("Validate static session which is valid starting from next epoch")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_start_at_next(
|
def test_static_session_start_at_next(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -557,74 +549,54 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
object_id = storage_objects[0].oid
|
object_id = storage_objects[0].oid
|
||||||
expiration = Lifetime(epoch + 2, epoch + 1, epoch)
|
expiration = Lifetime(epoch + 2, epoch + 1, epoch)
|
||||||
|
|
||||||
with allure.step("Create session token"):
|
token_start_at_next_epoch = get_object_signed_token(
|
||||||
token_start_at_next_epoch = get_object_signed_token(
|
owner_wallet,
|
||||||
owner_wallet,
|
user_wallet,
|
||||||
user_wallet,
|
container,
|
||||||
|
storage_objects,
|
||||||
|
ObjectVerb.HEAD,
|
||||||
|
self.shell,
|
||||||
|
temp_directory,
|
||||||
|
expiration,
|
||||||
|
)
|
||||||
|
|
||||||
|
with pytest.raises(Exception, match=MALFORMED_REQUEST):
|
||||||
|
head_object(
|
||||||
|
user_wallet.path,
|
||||||
container,
|
container,
|
||||||
storage_objects,
|
object_id,
|
||||||
ObjectVerb.HEAD,
|
|
||||||
self.shell,
|
self.shell,
|
||||||
temp_directory,
|
self.cluster.default_rpc_endpoint,
|
||||||
expiration,
|
session=token_start_at_next_epoch,
|
||||||
)
|
)
|
||||||
|
|
||||||
with allure.step("Object should NOT be available with session token after token creation"):
|
self.tick_epoch()
|
||||||
with pytest.raises(Exception, match=MALFORMED_REQUEST):
|
head_object(
|
||||||
head_object(
|
user_wallet.path,
|
||||||
user_wallet.path,
|
container,
|
||||||
container,
|
object_id,
|
||||||
object_id,
|
self.shell,
|
||||||
self.shell,
|
self.cluster.default_rpc_endpoint,
|
||||||
self.cluster.default_rpc_endpoint,
|
session=token_start_at_next_epoch,
|
||||||
session=token_start_at_next_epoch,
|
)
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step(
|
self.tick_epoch()
|
||||||
"Object should be available with session token starting from token nbf epoch"
|
with pytest.raises(Exception, match=MALFORMED_REQUEST):
|
||||||
):
|
head_object(
|
||||||
self.tick_epoch()
|
user_wallet.path,
|
||||||
with expect_not_raises():
|
container,
|
||||||
head_object(
|
object_id,
|
||||||
user_wallet.path,
|
self.shell,
|
||||||
container,
|
self.cluster.default_rpc_endpoint,
|
||||||
object_id,
|
session=token_start_at_next_epoch,
|
||||||
self.shell,
|
)
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_start_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step(
|
|
||||||
"Object should be available at last epoch before session token expiration"
|
|
||||||
):
|
|
||||||
self.tick_epoch()
|
|
||||||
with expect_not_raises():
|
|
||||||
head_object(
|
|
||||||
user_wallet.path,
|
|
||||||
container,
|
|
||||||
object_id,
|
|
||||||
self.shell,
|
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_start_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
with allure.step("Object should NOT be available after session token expiration epoch"):
|
|
||||||
self.tick_epoch()
|
|
||||||
with pytest.raises(Exception, match=EXPIRED_SESSION_TOKEN):
|
|
||||||
head_object(
|
|
||||||
user_wallet.path,
|
|
||||||
container,
|
|
||||||
object_id,
|
|
||||||
self.shell,
|
|
||||||
self.cluster.default_rpc_endpoint,
|
|
||||||
session=token_start_at_next_epoch,
|
|
||||||
)
|
|
||||||
|
|
||||||
@allure.title("Validate static session which is already expired")
|
@allure.title("Validate static session which is already expired")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_already_expired(
|
def test_static_session_already_expired(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
@ -653,7 +625,7 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
expiration,
|
expiration,
|
||||||
)
|
)
|
||||||
|
|
||||||
with pytest.raises(Exception, match=EXPIRED_SESSION_TOKEN):
|
with pytest.raises(Exception, match=MALFORMED_REQUEST):
|
||||||
head_object(
|
head_object(
|
||||||
user_wallet.path,
|
user_wallet.path,
|
||||||
container,
|
container,
|
||||||
|
@ -666,7 +638,7 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
@allure.title("Delete verb should be restricted for static session")
|
@allure.title("Delete verb should be restricted for static session")
|
||||||
def test_static_session_delete_verb(
|
def test_static_session_delete_verb(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -691,7 +663,7 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
@allure.title("Put verb should be restricted for static session")
|
@allure.title("Put verb should be restricted for static session")
|
||||||
def test_static_session_put_verb(
|
def test_static_session_put_verb(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
static_sessions: dict[ObjectVerb, str],
|
static_sessions: dict[ObjectVerb, str],
|
||||||
request: FixtureRequest,
|
request: FixtureRequest,
|
||||||
|
@ -714,10 +686,11 @@ class TestObjectStaticSession(ClusterTestBase):
|
||||||
)
|
)
|
||||||
|
|
||||||
@allure.title("Validate static session which is issued in future epoch")
|
@allure.title("Validate static session which is issued in future epoch")
|
||||||
|
@pytest.mark.static_session
|
||||||
def test_static_session_invalid_issued_epoch(
|
def test_static_session_invalid_issued_epoch(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
storage_containers: list[str],
|
storage_containers: list[str],
|
||||||
storage_objects: list[StorageObjectInfo],
|
storage_objects: list[StorageObjectInfo],
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
|
|
|
@ -1,21 +1,28 @@
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
from frostfs_testlib.resources.wellknown_acl import PUBLIC_ACL
|
from file_helper import generate_file
|
||||||
from frostfs_testlib.shell import Shell
|
from neofs_testlib.shell import Shell
|
||||||
from frostfs_testlib.steps.acl import create_eacl, set_eacl, wait_for_cache_expired
|
from python_keywords.acl import (
|
||||||
from frostfs_testlib.steps.cli.container import (
|
EACLAccess,
|
||||||
|
EACLOperation,
|
||||||
|
EACLRole,
|
||||||
|
EACLRule,
|
||||||
|
create_eacl,
|
||||||
|
set_eacl,
|
||||||
|
wait_for_cache_expired,
|
||||||
|
)
|
||||||
|
from python_keywords.container import (
|
||||||
create_container,
|
create_container,
|
||||||
delete_container,
|
delete_container,
|
||||||
get_container,
|
get_container,
|
||||||
list_containers,
|
list_containers,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.steps.session_token import ContainerVerb, get_container_signed_token
|
from python_keywords.object_access import can_put_object
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLAccess, EACLOperation, EACLRole, EACLRule
|
from wallet import WalletFile
|
||||||
from frostfs_testlib.storage.dataclasses.wallet import WalletInfo
|
from wellknown_acl import PUBLIC_ACL
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
from frostfs_testlib.utils.file_utils import generate_file
|
|
||||||
|
|
||||||
from pytest_tests.helpers.object_access import can_put_object
|
from steps.cluster_test_base import ClusterTestBase
|
||||||
|
from steps.session_token import ContainerVerb, get_container_signed_token
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.static_session_container
|
@pytest.mark.static_session_container
|
||||||
|
@ -23,8 +30,8 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
@pytest.fixture(scope="module")
|
@pytest.fixture(scope="module")
|
||||||
def static_sessions(
|
def static_sessions(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
client_shell: Shell,
|
client_shell: Shell,
|
||||||
temp_directory: str,
|
temp_directory: str,
|
||||||
) -> dict[ContainerVerb, str]:
|
) -> dict[ContainerVerb, str]:
|
||||||
|
@ -40,8 +47,8 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
|
|
||||||
def test_static_session_token_container_create(
|
def test_static_session_token_container_create(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
static_sessions: dict[ContainerVerb, str],
|
static_sessions: dict[ContainerVerb, str],
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
|
@ -70,7 +77,7 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
|
|
||||||
def test_static_session_token_container_create_with_other_verb(
|
def test_static_session_token_container_create_with_other_verb(
|
||||||
self,
|
self,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
static_sessions: dict[ContainerVerb, str],
|
static_sessions: dict[ContainerVerb, str],
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
|
@ -89,7 +96,7 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
|
|
||||||
def test_static_session_token_container_create_with_other_wallet(
|
def test_static_session_token_container_create_with_other_wallet(
|
||||||
self,
|
self,
|
||||||
stranger_wallet: WalletInfo,
|
stranger_wallet: WalletFile,
|
||||||
static_sessions: dict[ContainerVerb, str],
|
static_sessions: dict[ContainerVerb, str],
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
|
@ -107,8 +114,8 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
|
|
||||||
def test_static_session_token_container_delete(
|
def test_static_session_token_container_delete(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
static_sessions: dict[ContainerVerb, str],
|
static_sessions: dict[ContainerVerb, str],
|
||||||
):
|
):
|
||||||
"""
|
"""
|
||||||
|
@ -137,9 +144,9 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
|
|
||||||
def test_static_session_token_container_set_eacl(
|
def test_static_session_token_container_set_eacl(
|
||||||
self,
|
self,
|
||||||
owner_wallet: WalletInfo,
|
owner_wallet: WalletFile,
|
||||||
user_wallet: WalletInfo,
|
user_wallet: WalletFile,
|
||||||
stranger_wallet: WalletInfo,
|
stranger_wallet: WalletFile,
|
||||||
static_sessions: dict[ContainerVerb, str],
|
static_sessions: dict[ContainerVerb, str],
|
||||||
simple_object_size,
|
simple_object_size,
|
||||||
):
|
):
|
||||||
|
@ -156,7 +163,7 @@ class TestSessionTokenContainer(ClusterTestBase):
|
||||||
file_path = generate_file(simple_object_size)
|
file_path = generate_file(simple_object_size)
|
||||||
assert can_put_object(stranger_wallet.path, cid, file_path, self.shell, self.cluster)
|
assert can_put_object(stranger_wallet.path, cid, file_path, self.shell, self.cluster)
|
||||||
|
|
||||||
with allure.step("Deny all operations for other via eACL"):
|
with allure.step(f"Deny all operations for other via eACL"):
|
||||||
eacl_deny = [
|
eacl_deny = [
|
||||||
EACLRule(access=EACLAccess.DENY, role=EACLRole.OTHERS, operation=op)
|
EACLRule(access=EACLAccess.DENY, role=EACLRole.OTHERS, operation=op)
|
||||||
for op in EACLOperation
|
for op in EACLOperation
|
||||||
|
|
|
@ -7,13 +7,12 @@ from io import StringIO
|
||||||
import allure
|
import allure
|
||||||
import pytest
|
import pytest
|
||||||
import yaml
|
import yaml
|
||||||
|
from cluster import Cluster, StorageNode
|
||||||
|
from common import WALLET_CONFIG
|
||||||
from configobj import ConfigObj
|
from configobj import ConfigObj
|
||||||
from frostfs_testlib.cli import FrostfsCli
|
from neofs_testlib.cli import NeofsCli
|
||||||
from frostfs_testlib.resources.cli import CLI_DEFAULT_TIMEOUT
|
|
||||||
from frostfs_testlib.resources.common import DEFAULT_WALLET_CONFIG
|
|
||||||
from frostfs_testlib.storage.cluster import Cluster, StorageNode
|
|
||||||
|
|
||||||
SHARD_PREFIX = "FROSTFS_STORAGE_SHARD_"
|
SHARD_PREFIX = "NEOFS_STORAGE_SHARD_"
|
||||||
BLOBSTOR_PREFIX = "_BLOBSTOR_"
|
BLOBSTOR_PREFIX = "_BLOBSTOR_"
|
||||||
|
|
||||||
|
|
||||||
|
@ -135,15 +134,14 @@ class TestControlShard:
|
||||||
wallet_password = node.get_wallet_password()
|
wallet_password = node.get_wallet_password()
|
||||||
control_endpoint = node.get_control_endpoint()
|
control_endpoint = node.get_control_endpoint()
|
||||||
|
|
||||||
cli_config = node.host.get_cli_config("frostfs-cli")
|
cli_config = node.host.get_cli_config("neofs-cli")
|
||||||
|
|
||||||
cli = FrostfsCli(node.host.get_shell(), cli_config.exec_path, DEFAULT_WALLET_CONFIG)
|
cli = NeofsCli(node.host.get_shell(), cli_config.exec_path, WALLET_CONFIG)
|
||||||
result = cli.shards.list(
|
result = cli.shards.list(
|
||||||
endpoint=control_endpoint,
|
endpoint=control_endpoint,
|
||||||
wallet=wallet_path,
|
wallet=wallet_path,
|
||||||
wallet_password=wallet_password,
|
wallet_password=wallet_password,
|
||||||
json_mode=True,
|
json_mode=True,
|
||||||
timeout=CLI_DEFAULT_TIMEOUT,
|
|
||||||
)
|
)
|
||||||
return [Shard.from_object(shard) for shard in json.loads(result.stdout.split(">", 1)[1])]
|
return [Shard.from_object(shard) for shard in json.loads(result.stdout.split(">", 1)[1])]
|
||||||
|
|
||||||
|
|
|
@ -1,46 +0,0 @@
|
||||||
import os
|
|
||||||
import shutil
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
import allure
|
|
||||||
import pytest
|
|
||||||
from frostfs_testlib.testing.cluster_test_base import ClusterTestBase
|
|
||||||
|
|
||||||
|
|
||||||
class TestLogs(ClusterTestBase):
|
|
||||||
@pytest.mark.logs_after_session
|
|
||||||
def test_logs_after_session(self, temp_directory: str, session_start_time: datetime):
|
|
||||||
"""
|
|
||||||
This test automatically added to any test run to check logs from cluster for critical errors.
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
end_time = datetime.utcnow()
|
|
||||||
logs_dir = os.path.join(temp_directory, "logs")
|
|
||||||
os.makedirs(logs_dir)
|
|
||||||
issues_regex = r"\Wpanic\W|\Woom\W|\Wtoo many\W|\Winsufficient funds\W"
|
|
||||||
|
|
||||||
hosts_with_problems = []
|
|
||||||
for host in self.cluster.hosts:
|
|
||||||
with allure.step(f"Check logs on {host.config.address}"):
|
|
||||||
if host.is_message_in_logs(issues_regex, session_start_time, end_time):
|
|
||||||
hosts_with_problems.append(host.config.address)
|
|
||||||
host.dump_logs(
|
|
||||||
logs_dir,
|
|
||||||
since=session_start_time,
|
|
||||||
until=end_time,
|
|
||||||
filter_regex=issues_regex,
|
|
||||||
)
|
|
||||||
|
|
||||||
if hosts_with_problems:
|
|
||||||
self._attach_logs(logs_dir)
|
|
||||||
|
|
||||||
assert (
|
|
||||||
not hosts_with_problems
|
|
||||||
), f"The following hosts contains contain critical errors in system logs: {', '.join(hosts_with_problems)}"
|
|
||||||
|
|
||||||
def _attach_logs(self, logs_dir: str) -> None:
|
|
||||||
# Zip all files and attach to Allure because it is more convenient to download a single
|
|
||||||
# zip with all logs rather than mess with individual logs files per service or node
|
|
||||||
logs_zip_file_path = shutil.make_archive(logs_dir, "zip", logs_dir)
|
|
||||||
allure.attach.file(logs_zip_file_path, name="logs.zip", extension="zip")
|
|
1
pytest_tests/wallet_pass.yml
Normal file
1
pytest_tests/wallet_pass.yml
Normal file
|
@ -0,0 +1 @@
|
||||||
|
password: ""
|
|
@ -1,16 +1,71 @@
|
||||||
allure-pytest==2.13.2
|
aiodns==3.0.0
|
||||||
allure-python-commons==2.13.2
|
aiohttp==3.7.4.post0
|
||||||
|
aioresponses==0.7.2
|
||||||
|
allure-pytest==2.9.45
|
||||||
|
allure-python-commons==2.9.45
|
||||||
|
async-timeout==3.0.1
|
||||||
|
asynctest==0.13.0
|
||||||
|
attrs==21.4.0
|
||||||
base58==2.1.0
|
base58==2.1.0
|
||||||
|
bitarray==2.3.4
|
||||||
|
black==22.8.0
|
||||||
boto3==1.16.33
|
boto3==1.16.33
|
||||||
botocore==1.19.33
|
botocore==1.19.33
|
||||||
|
certifi==2022.5.18
|
||||||
|
cffi==1.15.0
|
||||||
|
chardet==4.0.0
|
||||||
|
charset-normalizer==2.0.12
|
||||||
|
coverage==6.3.3
|
||||||
configobj==5.0.6
|
configobj==5.0.6
|
||||||
frostfs-testlib>=2.0.1
|
docker==4.4.0
|
||||||
neo-mamba==1.0.0
|
docutils==0.17.1
|
||||||
|
Events==0.4
|
||||||
|
flake8==4.0.1
|
||||||
|
idna==3.3
|
||||||
|
iniconfig==1.1.1
|
||||||
|
isort==5.10.1
|
||||||
|
jmespath==0.10.0
|
||||||
|
jsonschema==4.5.1
|
||||||
|
lz4==3.1.3
|
||||||
|
mccabe==0.6.1
|
||||||
|
mmh3==3.0.0
|
||||||
|
multidict==6.0.2
|
||||||
|
mypy==0.950
|
||||||
|
mypy-extensions==0.4.3
|
||||||
|
neo-mamba==0.10.0
|
||||||
|
neo3crypto==0.2.1
|
||||||
|
neo3vm==0.9.0
|
||||||
|
neo3vm-stubs==0.9.0
|
||||||
|
neofs-testlib==0.8.1
|
||||||
|
netaddr==0.8.0
|
||||||
|
orjson==3.6.8
|
||||||
|
packaging==21.3
|
||||||
|
paramiko==2.10.3
|
||||||
pexpect==4.8.0
|
pexpect==4.8.0
|
||||||
pyyaml==6.0
|
pluggy==1.0.0
|
||||||
|
pre-commit==2.20.0
|
||||||
|
ptyprocess==0.7.0
|
||||||
|
py==1.11.0
|
||||||
|
pybiginteger==1.2.6
|
||||||
|
pybiginteger-stubs==1.2.6
|
||||||
|
pycares==4.1.2
|
||||||
|
pycodestyle==2.8.0
|
||||||
|
pycparser==2.21
|
||||||
|
pycryptodome==3.11.0
|
||||||
|
pyflakes==2.4.0
|
||||||
|
pyparsing==3.0.9
|
||||||
|
pyrsistent==0.18.1
|
||||||
pytest==7.1.2
|
pytest==7.1.2
|
||||||
pytest-lazy-fixture==0.6.3
|
pytest-lazy-fixture==0.6.3
|
||||||
python-dateutil==2.8.2
|
python-dateutil==2.8.2
|
||||||
|
pyyaml==6.0
|
||||||
requests==2.28.0
|
requests==2.28.0
|
||||||
|
robotframework==4.1.2
|
||||||
|
s3transfer==0.3.7
|
||||||
|
six==1.16.0
|
||||||
tenacity==8.0.1
|
tenacity==8.0.1
|
||||||
|
tomli==2.0.1
|
||||||
|
typing-extensions==4.2.0
|
||||||
urllib3==1.26.9
|
urllib3==1.26.9
|
||||||
|
websocket-client==1.3.2
|
||||||
|
yarl==1.7.2
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
pre-commit==2.20.0
|
|
||||||
isort==5.12.0
|
|
0
robot/resources/lib/python_keywords/__init__.py
Normal file
0
robot/resources/lib/python_keywords/__init__.py
Normal file
278
robot/resources/lib/python_keywords/acl.py
Normal file
278
robot/resources/lib/python_keywords/acl.py
Normal file
|
@ -0,0 +1,278 @@
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import uuid
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from enum import Enum
|
||||||
|
from time import sleep
|
||||||
|
from typing import Any, Dict, List, Optional, Union
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import base58
|
||||||
|
from common import ASSETS_DIR, NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from data_formatters import get_wallet_public_key
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
EACL_LIFETIME = 100500
|
||||||
|
NEOFS_CONTRACT_CACHE_TIMEOUT = 30
|
||||||
|
|
||||||
|
|
||||||
|
class EACLOperation(Enum):
|
||||||
|
PUT = "put"
|
||||||
|
GET = "get"
|
||||||
|
HEAD = "head"
|
||||||
|
GET_RANGE = "getrange"
|
||||||
|
GET_RANGE_HASH = "getrangehash"
|
||||||
|
SEARCH = "search"
|
||||||
|
DELETE = "delete"
|
||||||
|
|
||||||
|
|
||||||
|
class EACLAccess(Enum):
|
||||||
|
ALLOW = "allow"
|
||||||
|
DENY = "deny"
|
||||||
|
|
||||||
|
|
||||||
|
class EACLRole(Enum):
|
||||||
|
OTHERS = "others"
|
||||||
|
USER = "user"
|
||||||
|
SYSTEM = "system"
|
||||||
|
|
||||||
|
|
||||||
|
class EACLHeaderType(Enum):
|
||||||
|
REQUEST = "req" # Filter request headers
|
||||||
|
OBJECT = "obj" # Filter object headers
|
||||||
|
SERVICE = "SERVICE" # Filter service headers. These are not processed by NeoFS nodes and exist for service use only
|
||||||
|
|
||||||
|
|
||||||
|
class EACLMatchType(Enum):
|
||||||
|
STRING_EQUAL = "=" # Return true if strings are equal
|
||||||
|
STRING_NOT_EQUAL = "!=" # Return true if strings are different
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class EACLFilter:
|
||||||
|
header_type: EACLHeaderType = EACLHeaderType.REQUEST
|
||||||
|
match_type: EACLMatchType = EACLMatchType.STRING_EQUAL
|
||||||
|
key: Optional[str] = None
|
||||||
|
value: Optional[str] = None
|
||||||
|
|
||||||
|
def to_dict(self) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
"headerType": self.header_type,
|
||||||
|
"matchType": self.match_type,
|
||||||
|
"key": self.key,
|
||||||
|
"value": self.value,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class EACLFilters:
|
||||||
|
filters: Optional[List[EACLFilter]] = None
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return (
|
||||||
|
",".join(
|
||||||
|
[
|
||||||
|
f"{filter.header_type.value}:"
|
||||||
|
f"{filter.key}{filter.match_type.value}{filter.value}"
|
||||||
|
for filter in self.filters
|
||||||
|
]
|
||||||
|
)
|
||||||
|
if self.filters
|
||||||
|
else []
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class EACLPubKey:
|
||||||
|
keys: Optional[List[str]] = None
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class EACLRule:
|
||||||
|
operation: Optional[EACLOperation] = None
|
||||||
|
access: Optional[EACLAccess] = None
|
||||||
|
role: Optional[Union[EACLRole, str]] = None
|
||||||
|
filters: Optional[EACLFilters] = None
|
||||||
|
|
||||||
|
def to_dict(self) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
"Operation": self.operation,
|
||||||
|
"Access": self.access,
|
||||||
|
"Role": self.role,
|
||||||
|
"Filters": self.filters or [],
|
||||||
|
}
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
role = (
|
||||||
|
self.role.value
|
||||||
|
if isinstance(self.role, EACLRole)
|
||||||
|
else f'pubkey:{get_wallet_public_key(self.role, "")}'
|
||||||
|
)
|
||||||
|
return f'{self.access.value} {self.operation.value} {self.filters or ""} {role}'
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Get extended ACL")
|
||||||
|
def get_eacl(wallet_path: str, cid: str, shell: Shell, endpoint: str) -> Optional[str]:
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
try:
|
||||||
|
result = cli.container.get_eacl(wallet=wallet_path, rpc_endpoint=endpoint, cid=cid)
|
||||||
|
except RuntimeError as exc:
|
||||||
|
logger.info("Extended ACL table is not set for this container")
|
||||||
|
logger.info(f"Got exception while getting eacl: {exc}")
|
||||||
|
return None
|
||||||
|
if "extended ACL table is not set for this container" in result.stdout:
|
||||||
|
return None
|
||||||
|
return result.stdout
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Set extended ACL")
|
||||||
|
def set_eacl(
|
||||||
|
wallet_path: str,
|
||||||
|
cid: str,
|
||||||
|
eacl_table_path: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
session_token: Optional[str] = None,
|
||||||
|
) -> None:
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
cli.container.set_eacl(
|
||||||
|
wallet=wallet_path,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
cid=cid,
|
||||||
|
table=eacl_table_path,
|
||||||
|
await_mode=True,
|
||||||
|
session=session_token,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _encode_cid_for_eacl(cid: str) -> str:
|
||||||
|
cid_base58 = base58.b58decode(cid)
|
||||||
|
return base64.b64encode(cid_base58).decode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
def create_eacl(cid: str, rules_list: List[EACLRule], shell: Shell) -> str:
|
||||||
|
table_file_path = os.path.join(os.getcwd(), ASSETS_DIR, f"eacl_table_{str(uuid.uuid4())}.json")
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
cli.acl.extended_create(cid=cid, out=table_file_path, rule=rules_list)
|
||||||
|
|
||||||
|
with open(table_file_path, "r") as file:
|
||||||
|
table_data = file.read()
|
||||||
|
logger.info(f"Generated eACL:\n{table_data}")
|
||||||
|
|
||||||
|
return table_file_path
|
||||||
|
|
||||||
|
|
||||||
|
def form_bearertoken_file(
|
||||||
|
wif: str,
|
||||||
|
cid: str,
|
||||||
|
eacl_rule_list: List[Union[EACLRule, EACLPubKey]],
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
sign: Optional[bool] = True,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
This function fetches eACL for given <cid> on behalf of <wif>,
|
||||||
|
then extends it with filters taken from <eacl_rules>, signs
|
||||||
|
with bearer token and writes to file
|
||||||
|
"""
|
||||||
|
enc_cid = _encode_cid_for_eacl(cid) if cid else None
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
|
||||||
|
eacl = get_eacl(wif, cid, shell, endpoint)
|
||||||
|
json_eacl = dict()
|
||||||
|
if eacl:
|
||||||
|
eacl = eacl.replace("eACL: ", "").split("Signature")[0]
|
||||||
|
json_eacl = json.loads(eacl)
|
||||||
|
logger.info(json_eacl)
|
||||||
|
eacl_result = {
|
||||||
|
"body": {
|
||||||
|
"eaclTable": {"containerID": {"value": enc_cid} if cid else enc_cid, "records": []},
|
||||||
|
"lifetime": {"exp": EACL_LIFETIME, "nbf": "1", "iat": "0"},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert eacl_rules, "Got empty eacl_records list"
|
||||||
|
for rule in eacl_rule_list:
|
||||||
|
op_data = {
|
||||||
|
"operation": rule.operation.value.upper(),
|
||||||
|
"action": rule.access.value.upper(),
|
||||||
|
"filters": rule.filters or [],
|
||||||
|
"targets": [],
|
||||||
|
}
|
||||||
|
|
||||||
|
if isinstance(rule.role, EACLRole):
|
||||||
|
op_data["targets"] = [{"role": rule.role.value.upper()}]
|
||||||
|
elif isinstance(rule.role, EACLPubKey):
|
||||||
|
op_data["targets"] = [{"keys": rule.role.keys}]
|
||||||
|
|
||||||
|
eacl_result["body"]["eaclTable"]["records"].append(op_data)
|
||||||
|
|
||||||
|
# Add records from current eACL
|
||||||
|
if "records" in json_eacl.keys():
|
||||||
|
for record in json_eacl["records"]:
|
||||||
|
eacl_result["body"]["eaclTable"]["records"].append(record)
|
||||||
|
|
||||||
|
with open(file_path, "w", encoding="utf-8") as eacl_file:
|
||||||
|
json.dump(eacl_result, eacl_file, ensure_ascii=False, indent=4)
|
||||||
|
|
||||||
|
logger.info(f"Got these extended ACL records: {eacl_result}")
|
||||||
|
if sign:
|
||||||
|
sign_bearer(
|
||||||
|
shell=shell,
|
||||||
|
wallet_path=wif,
|
||||||
|
eacl_rules_file_from=file_path,
|
||||||
|
eacl_rules_file_to=file_path,
|
||||||
|
json=True,
|
||||||
|
)
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
def eacl_rules(access: str, verbs: list, user: str) -> list[str]:
|
||||||
|
"""
|
||||||
|
This function creates a list of eACL rules.
|
||||||
|
Args:
|
||||||
|
access (str): identifies if the following operation(s)
|
||||||
|
is allowed or denied
|
||||||
|
verbs (list): a list of operations to set rules for
|
||||||
|
user (str): a group of users (user/others) or a wallet of
|
||||||
|
a certain user for whom rules are set
|
||||||
|
Returns:
|
||||||
|
(list): a list of eACL rules
|
||||||
|
"""
|
||||||
|
if user not in ("others", "user"):
|
||||||
|
pubkey = get_wallet_public_key(user, wallet_password="")
|
||||||
|
user = f"pubkey:{pubkey}"
|
||||||
|
|
||||||
|
rules = []
|
||||||
|
for verb in verbs:
|
||||||
|
rule = f"{access} {verb} {user}"
|
||||||
|
rules.append(rule)
|
||||||
|
return rules
|
||||||
|
|
||||||
|
|
||||||
|
def sign_bearer(
|
||||||
|
shell: Shell, wallet_path: str, eacl_rules_file_from: str, eacl_rules_file_to: str, json: bool
|
||||||
|
) -> None:
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=WALLET_CONFIG)
|
||||||
|
neofscli.util.sign_bearer_token(
|
||||||
|
wallet=wallet_path, from_file=eacl_rules_file_from, to_file=eacl_rules_file_to, json=json
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Wait for eACL cache expired")
|
||||||
|
def wait_for_cache_expired():
|
||||||
|
sleep(NEOFS_CONTRACT_CACHE_TIMEOUT)
|
||||||
|
return
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Return bearer token in base64 to caller")
|
||||||
|
def bearer_token_base64_from_file(
|
||||||
|
bearer_path: str,
|
||||||
|
) -> str:
|
||||||
|
with open(bearer_path, "rb") as file:
|
||||||
|
signed = file.read()
|
||||||
|
return base64.b64encode(signed).decode("utf-8")
|
131
robot/resources/lib/python_keywords/cli_helpers.py
Normal file
131
robot/resources/lib/python_keywords/cli_helpers.py
Normal file
|
@ -0,0 +1,131 @@
|
||||||
|
#!/usr/bin/python3.9
|
||||||
|
|
||||||
|
"""
|
||||||
|
Helper functions to use with `neofs-cli`, `neo-go` and other CLIs.
|
||||||
|
"""
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import subprocess
|
||||||
|
import sys
|
||||||
|
from contextlib import suppress
|
||||||
|
from datetime import datetime
|
||||||
|
from textwrap import shorten
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import pexpect
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
COLOR_GREEN = "\033[92m"
|
||||||
|
COLOR_OFF = "\033[0m"
|
||||||
|
|
||||||
|
|
||||||
|
def _cmd_run(cmd: str, timeout: int = 30) -> str:
|
||||||
|
"""
|
||||||
|
Runs given shell command <cmd>, in case of success returns its stdout,
|
||||||
|
in case of failure returns error message.
|
||||||
|
"""
|
||||||
|
compl_proc = None
|
||||||
|
start_time = datetime.now()
|
||||||
|
try:
|
||||||
|
logger.info(f"{COLOR_GREEN}Executing command: {cmd}{COLOR_OFF}")
|
||||||
|
start_time = datetime.utcnow()
|
||||||
|
compl_proc = subprocess.run(
|
||||||
|
cmd,
|
||||||
|
check=True,
|
||||||
|
universal_newlines=True,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
timeout=timeout,
|
||||||
|
shell=True,
|
||||||
|
)
|
||||||
|
output = compl_proc.stdout
|
||||||
|
return_code = compl_proc.returncode
|
||||||
|
end_time = datetime.utcnow()
|
||||||
|
logger.info(f"{COLOR_GREEN}Output: {output}{COLOR_OFF}")
|
||||||
|
_attach_allure_log(cmd, output, return_code, start_time, end_time)
|
||||||
|
|
||||||
|
return output
|
||||||
|
except subprocess.CalledProcessError as exc:
|
||||||
|
logger.info(
|
||||||
|
f"Command: {cmd}\n" f"Error:\nreturn code: {exc.returncode} " f"\nOutput: {exc.output}"
|
||||||
|
)
|
||||||
|
end_time = datetime.now()
|
||||||
|
return_code, cmd_output = subprocess.getstatusoutput(cmd)
|
||||||
|
_attach_allure_log(cmd, cmd_output, return_code, start_time, end_time)
|
||||||
|
|
||||||
|
raise RuntimeError(
|
||||||
|
f"Command: {cmd}\n" f"Error:\nreturn code: {exc.returncode}\n" f"Output: {exc.output}"
|
||||||
|
) from exc
|
||||||
|
except OSError as exc:
|
||||||
|
raise RuntimeError(f"Command: {cmd}\n" f"Output: {exc.strerror}") from exc
|
||||||
|
except Exception as exc:
|
||||||
|
return_code, cmd_output = subprocess.getstatusoutput(cmd)
|
||||||
|
end_time = datetime.now()
|
||||||
|
_attach_allure_log(cmd, cmd_output, return_code, start_time, end_time)
|
||||||
|
logger.info(
|
||||||
|
f"Command: {cmd}\n"
|
||||||
|
f"Error:\nreturn code: {return_code}\n"
|
||||||
|
f"Output: {exc.output.decode('utf-8') if type(exc.output) is bytes else exc.output}"
|
||||||
|
)
|
||||||
|
raise
|
||||||
|
|
||||||
|
|
||||||
|
def _run_with_passwd(cmd: str) -> str:
|
||||||
|
child = pexpect.spawn(cmd)
|
||||||
|
child.delaybeforesend = 1
|
||||||
|
child.expect(".*")
|
||||||
|
child.sendline("\r")
|
||||||
|
if sys.platform == "darwin":
|
||||||
|
child.expect(pexpect.EOF)
|
||||||
|
cmd = child.before
|
||||||
|
else:
|
||||||
|
child.wait()
|
||||||
|
cmd = child.read()
|
||||||
|
return cmd.decode()
|
||||||
|
|
||||||
|
|
||||||
|
def _configure_aws_cli(cmd: str, key_id: str, access_key: str, out_format: str = "json") -> str:
|
||||||
|
child = pexpect.spawn(cmd)
|
||||||
|
child.delaybeforesend = 1
|
||||||
|
|
||||||
|
child.expect("AWS Access Key ID.*")
|
||||||
|
child.sendline(key_id)
|
||||||
|
|
||||||
|
child.expect("AWS Secret Access Key.*")
|
||||||
|
child.sendline(access_key)
|
||||||
|
|
||||||
|
child.expect("Default region name.*")
|
||||||
|
child.sendline("")
|
||||||
|
|
||||||
|
child.expect("Default output format.*")
|
||||||
|
child.sendline(out_format)
|
||||||
|
|
||||||
|
child.wait()
|
||||||
|
cmd = child.read()
|
||||||
|
# child.expect(pexpect.EOF)
|
||||||
|
# cmd = child.before
|
||||||
|
return cmd.decode()
|
||||||
|
|
||||||
|
|
||||||
|
def _attach_allure_log(
|
||||||
|
cmd: str, output: str, return_code: int, start_time: datetime, end_time: datetime
|
||||||
|
) -> None:
|
||||||
|
command_attachment = (
|
||||||
|
f"COMMAND: '{cmd}'\n"
|
||||||
|
f"OUTPUT:\n {output}\n"
|
||||||
|
f"RC: {return_code}\n"
|
||||||
|
f"Start / End / Elapsed\t {start_time.time()} / {end_time.time()} / {end_time - start_time}"
|
||||||
|
)
|
||||||
|
with allure.step(f'COMMAND: {shorten(cmd, width=60, placeholder="...")}'):
|
||||||
|
allure.attach(command_attachment, "Command execution", allure.attachment_type.TEXT)
|
||||||
|
|
||||||
|
|
||||||
|
def log_command_execution(cmd: str, output: Union[str, dict]) -> None:
|
||||||
|
logger.info(f"{cmd}: {output}")
|
||||||
|
with suppress(Exception):
|
||||||
|
json_output = json.dumps(output, indent=4, sort_keys=True)
|
||||||
|
output = json_output
|
||||||
|
command_attachment = f"COMMAND: '{cmd}'\n" f"OUTPUT:\n {output}\n"
|
||||||
|
with allure.step(f'COMMAND: {shorten(cmd, width=60, placeholder="...")}'):
|
||||||
|
allure.attach(command_attachment, "Command execution", allure.attachment_type.TEXT)
|
183
robot/resources/lib/python_keywords/complex_object_actions.py
Normal file
183
robot/resources/lib/python_keywords/complex_object_actions.py
Normal file
|
@ -0,0 +1,183 @@
|
||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
This module contains functions which are used for Large Object assembling:
|
||||||
|
getting Last Object and split and getting Link Object. It is not enough to
|
||||||
|
simply perform a "raw" HEAD request, as noted in the issue:
|
||||||
|
https://github.com/nspcc-dev/neofs-node/issues/1304. Therefore, the reliable
|
||||||
|
retrieval of the aforementioned objects must be done this way: send direct
|
||||||
|
"raw" HEAD request to the every Storage Node and return the desired OID on
|
||||||
|
first non-null response.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from typing import Optional, Tuple
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import neofs_verbs
|
||||||
|
from cluster import Cluster, StorageNode
|
||||||
|
from common import WALLET_CONFIG
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_verbs import head_object
|
||||||
|
from storage_object import StorageObjectInfo
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
def get_storage_object_chunks(
|
||||||
|
storage_object: StorageObjectInfo, shell: Shell, cluster: Cluster
|
||||||
|
) -> list[str]:
|
||||||
|
"""
|
||||||
|
Get complex object split objects ids (no linker object)
|
||||||
|
|
||||||
|
Args:
|
||||||
|
storage_object: storage_object to get it's chunks
|
||||||
|
shell: client shell to do cmd requests
|
||||||
|
cluster: cluster object under test
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
list of object ids of complex object chunks
|
||||||
|
"""
|
||||||
|
|
||||||
|
with allure.step(f"Get complex object chunks (f{storage_object.oid})"):
|
||||||
|
split_object_id = get_link_object(
|
||||||
|
storage_object.wallet_file_path,
|
||||||
|
storage_object.cid,
|
||||||
|
storage_object.oid,
|
||||||
|
shell,
|
||||||
|
cluster.storage_nodes,
|
||||||
|
is_direct=False,
|
||||||
|
)
|
||||||
|
head = head_object(
|
||||||
|
storage_object.wallet_file_path,
|
||||||
|
storage_object.cid,
|
||||||
|
split_object_id,
|
||||||
|
shell,
|
||||||
|
cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
|
||||||
|
chunks_object_ids = []
|
||||||
|
if "split" in head["header"] and "children" in head["header"]["split"]:
|
||||||
|
chunks_object_ids = head["header"]["split"]["children"]
|
||||||
|
|
||||||
|
return chunks_object_ids
|
||||||
|
|
||||||
|
|
||||||
|
def get_complex_object_split_ranges(
|
||||||
|
storage_object: StorageObjectInfo, shell: Shell, cluster: Cluster
|
||||||
|
) -> list[Tuple[int, int]]:
|
||||||
|
|
||||||
|
"""
|
||||||
|
Get list of split ranges tuples (offset, length) of a complex object
|
||||||
|
For example if object size if 100 and max object size in system is 30
|
||||||
|
the returned list should be
|
||||||
|
[(0, 30), (30, 30), (60, 30), (90, 10)]
|
||||||
|
|
||||||
|
Args:
|
||||||
|
storage_object: storage_object to get it's chunks
|
||||||
|
shell: client shell to do cmd requests
|
||||||
|
cluster: cluster object under test
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
list of object ids of complex object chunks
|
||||||
|
"""
|
||||||
|
|
||||||
|
ranges: list = []
|
||||||
|
offset = 0
|
||||||
|
chunks_ids = get_storage_object_chunks(storage_object, shell, cluster)
|
||||||
|
for chunk_id in chunks_ids:
|
||||||
|
head = head_object(
|
||||||
|
storage_object.wallet_file_path,
|
||||||
|
storage_object.cid,
|
||||||
|
chunk_id,
|
||||||
|
shell,
|
||||||
|
cluster.default_rpc_endpoint,
|
||||||
|
)
|
||||||
|
|
||||||
|
length = int(head["header"]["payloadLength"])
|
||||||
|
ranges.append((offset, length))
|
||||||
|
|
||||||
|
offset = offset + length
|
||||||
|
|
||||||
|
return ranges
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Link Object")
|
||||||
|
def get_link_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
nodes: list[StorageNode],
|
||||||
|
bearer: str = "",
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
is_direct: bool = True,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Args:
|
||||||
|
wallet (str): path to the wallet on whose behalf the Storage Nodes
|
||||||
|
are requested
|
||||||
|
cid (str): Container ID which stores the Large Object
|
||||||
|
oid (str): Large Object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
nodes: list of nodes to do search on
|
||||||
|
bearer (optional, str): path to Bearer token file
|
||||||
|
wallet_config (optional, str): path to the neofs-cli config file
|
||||||
|
is_direct: send request directly to the node or not; this flag
|
||||||
|
turns into `--ttl 1` key
|
||||||
|
Returns:
|
||||||
|
(str): Link Object ID
|
||||||
|
When no Link Object ID is found after all Storage Nodes polling,
|
||||||
|
the function throws an error.
|
||||||
|
"""
|
||||||
|
for node in nodes:
|
||||||
|
endpoint = node.get_rpc_endpoint()
|
||||||
|
try:
|
||||||
|
resp = neofs_verbs.head_object(
|
||||||
|
wallet,
|
||||||
|
cid,
|
||||||
|
oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=endpoint,
|
||||||
|
is_raw=True,
|
||||||
|
is_direct=is_direct,
|
||||||
|
bearer=bearer,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
if resp["link"]:
|
||||||
|
return resp["link"]
|
||||||
|
except Exception:
|
||||||
|
logger.info(f"No Link Object found on {endpoint}; continue")
|
||||||
|
logger.error(f"No Link Object for {cid}/{oid} found among all Storage Nodes")
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Last Object")
|
||||||
|
def get_last_object(
|
||||||
|
wallet: str, cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> Optional[str]:
|
||||||
|
"""
|
||||||
|
Args:
|
||||||
|
wallet (str): path to the wallet on whose behalf the Storage Nodes
|
||||||
|
are requested
|
||||||
|
cid (str): Container ID which stores the Large Object
|
||||||
|
oid (str): Large Object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
nodes: list of nodes to do search on
|
||||||
|
Returns:
|
||||||
|
(str): Last Object ID
|
||||||
|
When no Last Object ID is found after all Storage Nodes polling,
|
||||||
|
the function throws an error.
|
||||||
|
"""
|
||||||
|
for node in nodes:
|
||||||
|
endpoint = node.get_rpc_endpoint()
|
||||||
|
try:
|
||||||
|
resp = neofs_verbs.head_object(
|
||||||
|
wallet, cid, oid, shell=shell, endpoint=endpoint, is_raw=True, is_direct=True
|
||||||
|
)
|
||||||
|
if resp["lastPart"]:
|
||||||
|
return resp["lastPart"]
|
||||||
|
except Exception:
|
||||||
|
logger.info(f"No Last Object found on {endpoint}; continue")
|
||||||
|
logger.error(f"No Last Object for {cid}/{oid} found among all Storage Nodes")
|
||||||
|
return None
|
239
robot/resources/lib/python_keywords/container.py
Normal file
239
robot/resources/lib/python_keywords/container.py
Normal file
|
@ -0,0 +1,239 @@
|
||||||
|
#!/usr/bin/python3.9
|
||||||
|
|
||||||
|
"""
|
||||||
|
This module contains keywords that utilize `neofs-cli container` commands.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
from time import sleep
|
||||||
|
from typing import Optional, Union
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import json_transformers
|
||||||
|
from common import NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
DEFAULT_PLACEMENT_RULE = "REP 2 IN X CBF 1 SELECT 4 FROM * AS X"
|
||||||
|
SINGLE_PLACEMENT_RULE = "REP 1 IN X CBF 1 SELECT 4 FROM * AS X"
|
||||||
|
REP_2_FOR_3_NODES_PLACEMENT_RULE = "REP 2 IN X CBF 1 SELECT 3 FROM * AS X"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Create Container")
|
||||||
|
def create_container(
|
||||||
|
wallet: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
rule: str = DEFAULT_PLACEMENT_RULE,
|
||||||
|
basic_acl: str = "",
|
||||||
|
attributes: Optional[dict] = None,
|
||||||
|
session_token: str = "",
|
||||||
|
session_wallet: str = "",
|
||||||
|
name: str = None,
|
||||||
|
options: dict = None,
|
||||||
|
await_mode: bool = True,
|
||||||
|
wait_for_creation: bool = True,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
A wrapper for `neofs-cli container create` call.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet (str): a wallet on whose behalf a container is created
|
||||||
|
rule (optional, str): placement rule for container
|
||||||
|
basic_acl (optional, str): an ACL for container, will be
|
||||||
|
appended to `--basic-acl` key
|
||||||
|
attributes (optional, dict): container attributes , will be
|
||||||
|
appended to `--attributes` key
|
||||||
|
session_token (optional, str): a path to session token file
|
||||||
|
session_wallet(optional, str): a path to the wallet which signed
|
||||||
|
the session token; this parameter makes sense
|
||||||
|
when paired with `session_token`
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
options (optional, dict): any other options to pass to the call
|
||||||
|
name (optional, str): container name attribute
|
||||||
|
await_mode (bool): block execution until container is persisted
|
||||||
|
wait_for_creation (): Wait for container shows in container list
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
(str): CID of the created container
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
result = cli.container.create(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=session_wallet if session_wallet else wallet,
|
||||||
|
policy=rule,
|
||||||
|
basic_acl=basic_acl,
|
||||||
|
attributes=attributes,
|
||||||
|
name=name,
|
||||||
|
session=session_token,
|
||||||
|
await_mode=await_mode,
|
||||||
|
**options or {},
|
||||||
|
)
|
||||||
|
|
||||||
|
cid = _parse_cid(result.stdout)
|
||||||
|
|
||||||
|
logger.info("Container created; waiting until it is persisted in the sidechain")
|
||||||
|
|
||||||
|
if wait_for_creation:
|
||||||
|
wait_for_container_creation(wallet, cid, shell, endpoint)
|
||||||
|
|
||||||
|
return cid
|
||||||
|
|
||||||
|
|
||||||
|
def wait_for_container_creation(
|
||||||
|
wallet: str, cid: str, shell: Shell, endpoint: str, attempts: int = 15, sleep_interval: int = 1
|
||||||
|
):
|
||||||
|
for _ in range(attempts):
|
||||||
|
containers = list_containers(wallet, shell, endpoint)
|
||||||
|
if cid in containers:
|
||||||
|
return
|
||||||
|
logger.info(f"There is no {cid} in {containers} yet; sleep {sleep_interval} and continue")
|
||||||
|
sleep(sleep_interval)
|
||||||
|
raise RuntimeError(
|
||||||
|
f"After {attempts * sleep_interval} seconds container {cid} hasn't been persisted; exiting"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def wait_for_container_deletion(
|
||||||
|
wallet: str, cid: str, shell: Shell, endpoint: str, attempts: int = 30, sleep_interval: int = 1
|
||||||
|
):
|
||||||
|
for _ in range(attempts):
|
||||||
|
try:
|
||||||
|
get_container(wallet, cid, shell=shell, endpoint=endpoint)
|
||||||
|
sleep(sleep_interval)
|
||||||
|
continue
|
||||||
|
except Exception as err:
|
||||||
|
if "container not found" not in str(err):
|
||||||
|
raise AssertionError(f'Expected "container not found" in error, got\n{err}')
|
||||||
|
return
|
||||||
|
raise AssertionError(f"Expected container deleted during {attempts * sleep_interval} sec.")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List Containers")
|
||||||
|
def list_containers(wallet: str, shell: Shell, endpoint: str) -> list[str]:
|
||||||
|
"""
|
||||||
|
A wrapper for `neofs-cli container list` call. It returns all the
|
||||||
|
available containers for the given wallet.
|
||||||
|
Args:
|
||||||
|
wallet (str): a wallet on whose behalf we list the containers
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
Returns:
|
||||||
|
(list): list of containers
|
||||||
|
"""
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
result = cli.container.list(rpc_endpoint=endpoint, wallet=wallet)
|
||||||
|
logger.info(f"Containers: \n{result}")
|
||||||
|
return result.stdout.split()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Container")
|
||||||
|
def get_container(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
json_mode: bool = True,
|
||||||
|
) -> Union[dict, str]:
|
||||||
|
"""
|
||||||
|
A wrapper for `neofs-cli container get` call. It extracts container's
|
||||||
|
attributes and rearranges them into a more compact view.
|
||||||
|
Args:
|
||||||
|
wallet (str): path to a wallet on whose behalf we get the container
|
||||||
|
cid (str): ID of the container to get
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
json_mode (bool): return container in JSON format
|
||||||
|
Returns:
|
||||||
|
(dict, str): dict of container attributes
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
result = cli.container.get(rpc_endpoint=endpoint, wallet=wallet, cid=cid, json_mode=json_mode)
|
||||||
|
|
||||||
|
if not json_mode:
|
||||||
|
return result.stdout
|
||||||
|
|
||||||
|
container_info = json.loads(result.stdout)
|
||||||
|
attributes = dict()
|
||||||
|
for attr in container_info["attributes"]:
|
||||||
|
attributes[attr["key"]] = attr["value"]
|
||||||
|
container_info["attributes"] = attributes
|
||||||
|
container_info["ownerID"] = json_transformers.json_reencode(container_info["ownerID"]["value"])
|
||||||
|
return container_info
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete Container")
|
||||||
|
# TODO: make the error message about a non-found container more user-friendly
|
||||||
|
# https://github.com/nspcc-dev/neofs-contract/issues/121
|
||||||
|
def delete_container(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
force: bool = False,
|
||||||
|
session_token: Optional[str] = None,
|
||||||
|
await_mode: bool = False,
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
A wrapper for `neofs-cli container delete` call.
|
||||||
|
Args:
|
||||||
|
wallet (str): path to a wallet on whose behalf we delete the container
|
||||||
|
cid (str): ID of the container to delete
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
force (bool): do not check whether container contains locks and remove immediately
|
||||||
|
session_token: a path to session token file
|
||||||
|
This function doesn't return anything.
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, WALLET_CONFIG)
|
||||||
|
cli.container.delete(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
force=force,
|
||||||
|
session=session_token,
|
||||||
|
await_mode=await_mode,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _parse_cid(output: str) -> str:
|
||||||
|
"""
|
||||||
|
Parses container ID from a given CLI output. The input string we expect:
|
||||||
|
container ID: 2tz86kVTDpJxWHrhw3h6PbKMwkLtBEwoqhHQCKTre1FN
|
||||||
|
awaiting...
|
||||||
|
container has been persisted on sidechain
|
||||||
|
We want to take 'container ID' value from the string.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
output (str): CLI output to parse
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
(str): extracted CID
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
# taking first line from command's output
|
||||||
|
first_line = output.split("\n")[0]
|
||||||
|
except Exception:
|
||||||
|
first_line = ""
|
||||||
|
logger.error(f"Got empty output: {output}")
|
||||||
|
splitted = first_line.split(": ")
|
||||||
|
if len(splitted) != 2:
|
||||||
|
raise ValueError(f"no CID was parsed from command output: \t{first_line}")
|
||||||
|
return splitted[1]
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Search container by name")
|
||||||
|
def search_container_by_name(wallet: str, name: str, shell: Shell, endpoint: str):
|
||||||
|
list_cids = list_containers(wallet, shell, endpoint)
|
||||||
|
for cid in list_cids:
|
||||||
|
cont_info = get_container(wallet, cid, shell, endpoint, True)
|
||||||
|
if cont_info.get("attributes").get("Name", None) == name:
|
||||||
|
return cid
|
||||||
|
return None
|
|
@ -1,10 +1,9 @@
|
||||||
from typing import List, Optional
|
from typing import List, Optional
|
||||||
|
|
||||||
from frostfs_testlib.shell import Shell
|
from acl import EACLOperation
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
from cluster import Cluster
|
||||||
from frostfs_testlib.storage.dataclasses.acl import EACLOperation
|
from neofs_testlib.shell import Shell
|
||||||
|
from python_keywords.object_access import (
|
||||||
from pytest_tests.helpers.object_access import (
|
|
||||||
can_delete_object,
|
can_delete_object,
|
||||||
can_get_head_object,
|
can_get_head_object,
|
||||||
can_get_object,
|
can_get_object,
|
50
robot/resources/lib/python_keywords/data_formatters.py
Normal file
50
robot/resources/lib/python_keywords/data_formatters.py
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
|
||||||
|
import base58
|
||||||
|
from neo3 import wallet
|
||||||
|
|
||||||
|
|
||||||
|
def dict_to_attrs(attrs: dict) -> str:
|
||||||
|
"""
|
||||||
|
This function takes a dictionary of object's attributes and converts them
|
||||||
|
into string. The string is passed to `--attributes` key of neofs-cli.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
attrs (dict): object attributes in {"a": "b", "c": "d"} format.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
(str): string in "a=b,c=d" format.
|
||||||
|
"""
|
||||||
|
return ",".join(f"{key}={value}" for key, value in attrs.items())
|
||||||
|
|
||||||
|
|
||||||
|
def __fix_wallet_schema(wallet: dict) -> None:
|
||||||
|
# Temporary function to fix wallets that do not conform to the schema
|
||||||
|
# TODO: get rid of it once issue is solved
|
||||||
|
if "name" not in wallet:
|
||||||
|
wallet["name"] = None
|
||||||
|
for account in wallet["accounts"]:
|
||||||
|
if "extra" not in account:
|
||||||
|
account["extra"] = None
|
||||||
|
|
||||||
|
|
||||||
|
def get_wallet_public_key(wallet_path: str, wallet_password: str, format: str = "hex") -> str:
|
||||||
|
# Get public key from wallet file
|
||||||
|
with open(wallet_path, "r") as file:
|
||||||
|
wallet_content = json.load(file)
|
||||||
|
__fix_wallet_schema(wallet_content)
|
||||||
|
|
||||||
|
wallet_from_json = wallet.Wallet.from_json(wallet_content, password=wallet_password)
|
||||||
|
public_key_hex = str(wallet_from_json.accounts[0].public_key)
|
||||||
|
|
||||||
|
# Convert public key to specified format
|
||||||
|
if format == "hex":
|
||||||
|
return public_key_hex
|
||||||
|
if format == "base58":
|
||||||
|
public_key_base58 = base58.b58encode(bytes.fromhex(public_key_hex))
|
||||||
|
return public_key_base58.decode("utf-8")
|
||||||
|
if format == "base64":
|
||||||
|
public_key_base64 = base64.b64encode(bytes.fromhex(public_key_hex))
|
||||||
|
return public_key_base64.decode("utf-8")
|
||||||
|
raise ValueError(f"Invalid public key format: {format}")
|
111
robot/resources/lib/python_keywords/epoch.py
Normal file
111
robot/resources/lib/python_keywords/epoch.py
Normal file
|
@ -0,0 +1,111 @@
|
||||||
|
import logging
|
||||||
|
from time import sleep
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import Cluster, StorageNode
|
||||||
|
from common import (
|
||||||
|
MAINNET_BLOCK_TIME,
|
||||||
|
NEOFS_ADM_CONFIG_PATH,
|
||||||
|
NEOFS_ADM_EXEC,
|
||||||
|
NEOFS_CLI_EXEC,
|
||||||
|
NEOGO_EXECUTABLE,
|
||||||
|
)
|
||||||
|
from neofs_testlib.cli import NeofsAdm, NeofsCli, NeoGo
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_testlib.utils.wallet import get_last_address_from_wallet
|
||||||
|
from payment_neogo import get_contract_hash
|
||||||
|
from test_control import wait_for_success
|
||||||
|
from utility import parse_time
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Ensure fresh epoch")
|
||||||
|
def ensure_fresh_epoch(
|
||||||
|
shell: Shell, cluster: Cluster, alive_node: Optional[StorageNode] = None
|
||||||
|
) -> int:
|
||||||
|
# ensure new fresh epoch to avoid epoch switch during test session
|
||||||
|
alive_node = alive_node if alive_node else cluster.storage_nodes[0]
|
||||||
|
current_epoch = get_epoch(shell, cluster, alive_node)
|
||||||
|
tick_epoch(shell, cluster, alive_node)
|
||||||
|
epoch = get_epoch(shell, cluster, alive_node)
|
||||||
|
assert epoch > current_epoch, "Epoch wasn't ticked"
|
||||||
|
return epoch
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Wait for epochs align in whole cluster")
|
||||||
|
@wait_for_success(60, 5)
|
||||||
|
def wait_for_epochs_align(shell: Shell, cluster: Cluster) -> bool:
|
||||||
|
epochs = []
|
||||||
|
for node in cluster.storage_nodes:
|
||||||
|
epochs.append(get_epoch(shell, cluster, node))
|
||||||
|
unique_epochs = list(set(epochs))
|
||||||
|
assert (
|
||||||
|
len(unique_epochs) == 1
|
||||||
|
), f"unaligned epochs found, {epochs}, count of unique epochs {len(unique_epochs)}"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Epoch")
|
||||||
|
def get_epoch(shell: Shell, cluster: Cluster, alive_node: Optional[StorageNode] = None):
|
||||||
|
alive_node = alive_node if alive_node else cluster.storage_nodes[0]
|
||||||
|
endpoint = alive_node.get_rpc_endpoint()
|
||||||
|
wallet_path = alive_node.get_wallet_path()
|
||||||
|
wallet_config = alive_node.get_wallet_config_path()
|
||||||
|
|
||||||
|
cli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=wallet_config)
|
||||||
|
|
||||||
|
epoch = cli.netmap.epoch(endpoint, wallet_path)
|
||||||
|
return int(epoch.stdout)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Tick Epoch")
|
||||||
|
def tick_epoch(shell: Shell, cluster: Cluster, alive_node: Optional[StorageNode] = None):
|
||||||
|
"""
|
||||||
|
Tick epoch using neofs-adm or NeoGo if neofs-adm is not available (DevEnv)
|
||||||
|
Args:
|
||||||
|
shell: local shell to make queries about current epoch. Remote shell will be used to tick new one
|
||||||
|
cluster: cluster instance under test
|
||||||
|
alive_node: node to send requests to (first node in cluster by default)
|
||||||
|
"""
|
||||||
|
|
||||||
|
alive_node = alive_node if alive_node else cluster.storage_nodes[0]
|
||||||
|
remote_shell = alive_node.host.get_shell()
|
||||||
|
|
||||||
|
if NEOFS_ADM_EXEC and NEOFS_ADM_CONFIG_PATH:
|
||||||
|
# If neofs-adm is available, then we tick epoch with it (to be consistent with UAT tests)
|
||||||
|
neofsadm = NeofsAdm(
|
||||||
|
shell=remote_shell,
|
||||||
|
neofs_adm_exec_path=NEOFS_ADM_EXEC,
|
||||||
|
config_file=NEOFS_ADM_CONFIG_PATH,
|
||||||
|
)
|
||||||
|
neofsadm.morph.force_new_epoch()
|
||||||
|
return
|
||||||
|
|
||||||
|
# Otherwise we tick epoch using transaction
|
||||||
|
cur_epoch = get_epoch(shell, cluster)
|
||||||
|
|
||||||
|
# Use first node by default
|
||||||
|
ir_node = cluster.ir_nodes[0]
|
||||||
|
# In case if no local_wallet_path is provided, we use wallet_path
|
||||||
|
ir_wallet_path = ir_node.get_wallet_path()
|
||||||
|
ir_wallet_pass = ir_node.get_wallet_password()
|
||||||
|
ir_address = get_last_address_from_wallet(ir_wallet_path, ir_wallet_pass)
|
||||||
|
|
||||||
|
morph_chain = cluster.morph_chain_nodes[0]
|
||||||
|
morph_endpoint = morph_chain.get_endpoint()
|
||||||
|
|
||||||
|
neogo = NeoGo(shell, neo_go_exec_path=NEOGO_EXECUTABLE)
|
||||||
|
neogo.contract.invokefunction(
|
||||||
|
wallet=ir_wallet_path,
|
||||||
|
wallet_password=ir_wallet_pass,
|
||||||
|
scripthash=get_contract_hash(morph_chain, "netmap.neofs", shell=shell),
|
||||||
|
method="newEpoch",
|
||||||
|
arguments=f"int:{cur_epoch + 1}",
|
||||||
|
multisig_hash=f"{ir_address}:Global",
|
||||||
|
address=ir_address,
|
||||||
|
rpc_endpoint=morph_endpoint,
|
||||||
|
force=True,
|
||||||
|
gas=1,
|
||||||
|
)
|
||||||
|
sleep(parse_time(MAINNET_BLOCK_TIME))
|
54
robot/resources/lib/python_keywords/failover_utils.py
Normal file
54
robot/resources/lib/python_keywords/failover_utils.py
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
import logging
|
||||||
|
from time import sleep
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import Cluster, StorageNode
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from python_keywords.node_management import storage_node_healthcheck
|
||||||
|
from storage_policy import get_nodes_with_object
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Wait for object replication")
|
||||||
|
def wait_object_replication(
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
expected_copies: int,
|
||||||
|
shell: Shell,
|
||||||
|
nodes: list[StorageNode],
|
||||||
|
) -> list[StorageNode]:
|
||||||
|
sleep_interval, attempts = 15, 20
|
||||||
|
nodes_with_object = []
|
||||||
|
for _ in range(attempts):
|
||||||
|
nodes_with_object = get_nodes_with_object(cid, oid, shell=shell, nodes=nodes)
|
||||||
|
if len(nodes_with_object) >= expected_copies:
|
||||||
|
return nodes_with_object
|
||||||
|
sleep(sleep_interval)
|
||||||
|
raise AssertionError(
|
||||||
|
f"Expected {expected_copies} copies of object, but found {len(nodes_with_object)}. "
|
||||||
|
f"Waiting time {sleep_interval * attempts}"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Wait for storage nodes returned to cluster")
|
||||||
|
def wait_all_storage_nodes_returned(cluster: Cluster) -> None:
|
||||||
|
sleep_interval, attempts = 15, 20
|
||||||
|
for __attempt in range(attempts):
|
||||||
|
if is_all_storage_nodes_returned(cluster):
|
||||||
|
return
|
||||||
|
sleep(sleep_interval)
|
||||||
|
raise AssertionError("Storage node(s) is broken")
|
||||||
|
|
||||||
|
|
||||||
|
def is_all_storage_nodes_returned(cluster: Cluster) -> bool:
|
||||||
|
with allure.step("Run health check for all storage nodes"):
|
||||||
|
for node in cluster.storage_nodes:
|
||||||
|
try:
|
||||||
|
health_check = storage_node_healthcheck(node)
|
||||||
|
except Exception as err:
|
||||||
|
logger.warning(f"Node healthcheck fails with error {err}")
|
||||||
|
return False
|
||||||
|
if health_check.health_status != "READY" or health_check.network_status != "ONLINE":
|
||||||
|
return False
|
||||||
|
return True
|
353
robot/resources/lib/python_keywords/http_gate.py
Normal file
353
robot/resources/lib/python_keywords/http_gate.py
Normal file
|
@ -0,0 +1,353 @@
|
||||||
|
import base64
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import random
|
||||||
|
import re
|
||||||
|
import shutil
|
||||||
|
import uuid
|
||||||
|
import zipfile
|
||||||
|
from typing import Optional
|
||||||
|
from urllib.parse import quote_plus
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import requests
|
||||||
|
from aws_cli_client import LONG_TIMEOUT
|
||||||
|
from cli_helpers import _cmd_run
|
||||||
|
from cluster import StorageNode
|
||||||
|
from common import SIMPLE_OBJECT_SIZE
|
||||||
|
from file_helper import get_file_hash
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from python_keywords.neofs_verbs import get_object
|
||||||
|
from python_keywords.storage_policy import get_nodes_without_object
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
ASSETS_DIR = os.getenv("ASSETS_DIR", "TemporaryDir/")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get via HTTP Gate")
|
||||||
|
def get_via_http_gate(cid: str, oid: str, endpoint: str, request_path: Optional[str] = None):
|
||||||
|
"""
|
||||||
|
This function gets given object from HTTP gate
|
||||||
|
cid: container id to get object from
|
||||||
|
oid: object ID
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
request_path: (optional) http request, if ommited - use default [{endpoint}/get/{cid}/{oid}]
|
||||||
|
"""
|
||||||
|
|
||||||
|
# if `request_path` parameter ommited, use default
|
||||||
|
if request_path is None:
|
||||||
|
request = f"{endpoint}/get/{cid}/{oid}"
|
||||||
|
else:
|
||||||
|
request = f"{endpoint}{request_path}"
|
||||||
|
|
||||||
|
resp = requests.get(request, stream=True)
|
||||||
|
|
||||||
|
if not resp.ok:
|
||||||
|
raise Exception(
|
||||||
|
f"""Failed to get object via HTTP gate:
|
||||||
|
request: {resp.request.path_url},
|
||||||
|
response: {resp.text},
|
||||||
|
status code: {resp.status_code} {resp.reason}"""
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.info(f"Request: {request}")
|
||||||
|
_attach_allure_step(request, resp.status_code)
|
||||||
|
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, f"{cid}_{oid}")
|
||||||
|
with open(file_path, "wb") as file:
|
||||||
|
shutil.copyfileobj(resp.raw, file)
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get via Zip HTTP Gate")
|
||||||
|
def get_via_zip_http_gate(cid: str, prefix: str, endpoint: str):
|
||||||
|
"""
|
||||||
|
This function gets given object from HTTP gate
|
||||||
|
cid: container id to get object from
|
||||||
|
prefix: common prefix
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
"""
|
||||||
|
request = f"{endpoint}/zip/{cid}/{prefix}"
|
||||||
|
resp = requests.get(request, stream=True)
|
||||||
|
|
||||||
|
if not resp.ok:
|
||||||
|
raise Exception(
|
||||||
|
f"""Failed to get object via HTTP gate:
|
||||||
|
request: {resp.request.path_url},
|
||||||
|
response: {resp.text},
|
||||||
|
status code: {resp.status_code} {resp.reason}"""
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.info(f"Request: {request}")
|
||||||
|
_attach_allure_step(request, resp.status_code)
|
||||||
|
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, f"{cid}_archive.zip")
|
||||||
|
with open(file_path, "wb") as file:
|
||||||
|
shutil.copyfileobj(resp.raw, file)
|
||||||
|
|
||||||
|
with zipfile.ZipFile(file_path, "r") as zip_ref:
|
||||||
|
zip_ref.extractall(ASSETS_DIR)
|
||||||
|
|
||||||
|
return os.path.join(os.getcwd(), ASSETS_DIR, prefix)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get via HTTP Gate by attribute")
|
||||||
|
def get_via_http_gate_by_attribute(
|
||||||
|
cid: str, attribute: dict, endpoint: str, request_path: Optional[str] = None
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
This function gets given object from HTTP gate
|
||||||
|
cid: CID to get object from
|
||||||
|
attribute: attribute {name: attribute} value pair
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
request_path: (optional) http request path, if ommited - use default [{endpoint}/get_by_attribute/{Key}/{Value}]
|
||||||
|
"""
|
||||||
|
attr_name = list(attribute.keys())[0]
|
||||||
|
attr_value = quote_plus(str(attribute.get(attr_name)))
|
||||||
|
# if `request_path` parameter ommited, use default
|
||||||
|
if request_path is None:
|
||||||
|
request = f"{endpoint}/get_by_attribute/{cid}/{quote_plus(str(attr_name))}/{attr_value}"
|
||||||
|
else:
|
||||||
|
request = f"{endpoint}{request_path}"
|
||||||
|
|
||||||
|
resp = requests.get(request, stream=True)
|
||||||
|
|
||||||
|
if not resp.ok:
|
||||||
|
raise Exception(
|
||||||
|
f"""Failed to get object via HTTP gate:
|
||||||
|
request: {resp.request.path_url},
|
||||||
|
response: {resp.text},
|
||||||
|
status code: {resp.status_code} {resp.reason}"""
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.info(f"Request: {request}")
|
||||||
|
_attach_allure_step(request, resp.status_code)
|
||||||
|
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, f"{cid}_{str(uuid.uuid4())}")
|
||||||
|
with open(file_path, "wb") as file:
|
||||||
|
shutil.copyfileobj(resp.raw, file)
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Upload via HTTP Gate")
|
||||||
|
def upload_via_http_gate(cid: str, path: str, endpoint: str, headers: dict = None) -> str:
|
||||||
|
"""
|
||||||
|
This function upload given object through HTTP gate
|
||||||
|
cid: CID to get object from
|
||||||
|
path: File path to upload
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
headers: Object header
|
||||||
|
"""
|
||||||
|
request = f"{endpoint}/upload/{cid}"
|
||||||
|
files = {"upload_file": open(path, "rb")}
|
||||||
|
body = {"filename": path}
|
||||||
|
resp = requests.post(request, files=files, data=body, headers=headers)
|
||||||
|
|
||||||
|
if not resp.ok:
|
||||||
|
raise Exception(
|
||||||
|
f"""Failed to get object via HTTP gate:
|
||||||
|
request: {resp.request.path_url},
|
||||||
|
response: {resp.text},
|
||||||
|
status code: {resp.status_code} {resp.reason}"""
|
||||||
|
)
|
||||||
|
|
||||||
|
logger.info(f"Request: {request}")
|
||||||
|
_attach_allure_step(request, resp.json(), req_type="POST")
|
||||||
|
|
||||||
|
assert resp.json().get("object_id"), f"OID found in response {resp}"
|
||||||
|
|
||||||
|
return resp.json().get("object_id")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Check is the passed object large")
|
||||||
|
def is_object_large(filepath: str) -> bool:
|
||||||
|
"""
|
||||||
|
This function check passed file size and return True if file_size > SIMPLE_OBJECT_SIZE
|
||||||
|
filepath: File path to check
|
||||||
|
"""
|
||||||
|
file_size = os.path.getsize(filepath)
|
||||||
|
logger.info(f"Size= {file_size}")
|
||||||
|
if file_size > int(SIMPLE_OBJECT_SIZE):
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Upload via HTTP Gate using Curl")
|
||||||
|
def upload_via_http_gate_curl(
|
||||||
|
cid: str,
|
||||||
|
filepath: str,
|
||||||
|
endpoint: str,
|
||||||
|
headers: list = None,
|
||||||
|
error_pattern: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
This function upload given object through HTTP gate using curl utility.
|
||||||
|
cid: CID to get object from
|
||||||
|
filepath: File path to upload
|
||||||
|
headers: Object header
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
error_pattern: [optional] expected error message from the command
|
||||||
|
"""
|
||||||
|
request = f"{endpoint}/upload/{cid}"
|
||||||
|
attributes = ""
|
||||||
|
if headers:
|
||||||
|
# parse attributes
|
||||||
|
attributes = " ".join(headers)
|
||||||
|
|
||||||
|
large_object = is_object_large(filepath)
|
||||||
|
if large_object:
|
||||||
|
# pre-clean
|
||||||
|
_cmd_run("rm pipe -f")
|
||||||
|
files = f"file=@pipe;filename={os.path.basename(filepath)}"
|
||||||
|
cmd = f"mkfifo pipe;cat {filepath} > pipe & curl --no-buffer -F '{files}' {attributes} {request}"
|
||||||
|
output = _cmd_run(cmd, LONG_TIMEOUT)
|
||||||
|
# clean up pipe
|
||||||
|
_cmd_run("rm pipe")
|
||||||
|
else:
|
||||||
|
files = f"file=@{filepath};filename={os.path.basename(filepath)}"
|
||||||
|
cmd = f"curl -F '{files}' {attributes} {request}"
|
||||||
|
output = _cmd_run(cmd)
|
||||||
|
|
||||||
|
if error_pattern:
|
||||||
|
match = error_pattern.casefold() in str(output).casefold()
|
||||||
|
assert match, f"Expected {output} to match {error_pattern}"
|
||||||
|
return ""
|
||||||
|
|
||||||
|
oid_re = re.search(r'"object_id": "(.*)"', output)
|
||||||
|
if not oid_re:
|
||||||
|
raise AssertionError(f'Could not find "object_id" in {output}')
|
||||||
|
return oid_re.group(1)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get via HTTP Gate using Curl")
|
||||||
|
def get_via_http_curl(cid: str, oid: str, endpoint: str) -> str:
|
||||||
|
"""
|
||||||
|
This function gets given object from HTTP gate using curl utility.
|
||||||
|
cid: CID to get object from
|
||||||
|
oid: object OID
|
||||||
|
endpoint: http gate endpoint
|
||||||
|
"""
|
||||||
|
request = f"{endpoint}/get/{cid}/{oid}"
|
||||||
|
file_path = os.path.join(os.getcwd(), ASSETS_DIR, f"{cid}_{oid}_{str(uuid.uuid4())}")
|
||||||
|
|
||||||
|
cmd = f"curl {request} > {file_path}"
|
||||||
|
_cmd_run(cmd)
|
||||||
|
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
def _attach_allure_step(request: str, status_code: int, req_type="GET"):
|
||||||
|
command_attachment = f"REQUEST: '{request}'\n" f"RESPONSE:\n {status_code}\n"
|
||||||
|
with allure.step(f"{req_type} Request"):
|
||||||
|
allure.attach(command_attachment, f"{req_type} Request", allure.attachment_type.TEXT)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Try to get object and expect error")
|
||||||
|
def try_to_get_object_and_expect_error(
|
||||||
|
cid: str, oid: str, error_pattern: str, endpoint: str
|
||||||
|
) -> None:
|
||||||
|
try:
|
||||||
|
get_via_http_gate(cid=cid, oid=oid, endpoint=endpoint)
|
||||||
|
raise AssertionError(f"Expected error on getting object with cid: {cid}")
|
||||||
|
except Exception as err:
|
||||||
|
match = error_pattern.casefold() in str(err).casefold()
|
||||||
|
assert match, f"Expected {err} to match {error_pattern}"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Verify object can be get using HTTP header attribute")
|
||||||
|
def get_object_by_attr_and_verify_hashes(
|
||||||
|
oid: str, file_name: str, cid: str, attrs: dict, endpoint: str
|
||||||
|
) -> None:
|
||||||
|
got_file_path_http = get_via_http_gate(cid=cid, oid=oid, endpoint=endpoint)
|
||||||
|
got_file_path_http_attr = get_via_http_gate_by_attribute(
|
||||||
|
cid=cid, attribute=attrs, endpoint=endpoint
|
||||||
|
)
|
||||||
|
assert_hashes_are_equal(file_name, got_file_path_http, got_file_path_http_attr)
|
||||||
|
|
||||||
|
|
||||||
|
def get_object_and_verify_hashes(
|
||||||
|
oid: str,
|
||||||
|
file_name: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
nodes: list[StorageNode],
|
||||||
|
endpoint: str,
|
||||||
|
object_getter=None,
|
||||||
|
) -> None:
|
||||||
|
|
||||||
|
nodes_list = get_nodes_without_object(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
shell=shell,
|
||||||
|
nodes=nodes,
|
||||||
|
)
|
||||||
|
# for some reason we can face with case when nodes_list is empty due to object resides in all nodes
|
||||||
|
if nodes_list:
|
||||||
|
random_node = random.choice(nodes_list)
|
||||||
|
else:
|
||||||
|
random_node = random.choice(nodes)
|
||||||
|
|
||||||
|
object_getter = object_getter or get_via_http_gate
|
||||||
|
|
||||||
|
got_file_path = get_object(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=random_node.get_rpc_endpoint(),
|
||||||
|
)
|
||||||
|
got_file_path_http = object_getter(cid=cid, oid=oid, endpoint=endpoint)
|
||||||
|
|
||||||
|
assert_hashes_are_equal(file_name, got_file_path, got_file_path_http)
|
||||||
|
|
||||||
|
|
||||||
|
def assert_hashes_are_equal(orig_file_name: str, got_file_1: str, got_file_2: str) -> None:
|
||||||
|
msg = "Expected hashes are equal for files {f1} and {f2}"
|
||||||
|
got_file_hash_http = get_file_hash(got_file_1)
|
||||||
|
assert get_file_hash(got_file_2) == got_file_hash_http, msg.format(f1=got_file_2, f2=got_file_1)
|
||||||
|
assert get_file_hash(orig_file_name) == got_file_hash_http, msg.format(
|
||||||
|
f1=orig_file_name, f2=got_file_1
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def attr_into_header(attrs: dict) -> dict:
|
||||||
|
return {f"X-Attribute-{_key}": _value for _key, _value in attrs.items()}
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step(
|
||||||
|
"Convert each attribute (Key=Value) to the following format: -H 'X-Attribute-Key: Value'"
|
||||||
|
)
|
||||||
|
def attr_into_str_header_curl(attrs: dict) -> list:
|
||||||
|
headers = []
|
||||||
|
for k, v in attrs.items():
|
||||||
|
headers.append(f"-H 'X-Attribute-{k}: {v}'")
|
||||||
|
logger.info(f"[List of Attrs for curl:] {headers}")
|
||||||
|
return headers
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step(
|
||||||
|
"Try to get object via http (pass http_request and optional attributes) and expect error"
|
||||||
|
)
|
||||||
|
def try_to_get_object_via_passed_request_and_expect_error(
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
error_pattern: str,
|
||||||
|
endpoint: str,
|
||||||
|
http_request_path: str,
|
||||||
|
attrs: dict = None,
|
||||||
|
) -> None:
|
||||||
|
try:
|
||||||
|
if attrs is None:
|
||||||
|
get_via_http_gate(cid=cid, oid=oid, endpoint=endpoint, request_path=http_request_path)
|
||||||
|
else:
|
||||||
|
get_via_http_gate_by_attribute(
|
||||||
|
cid=cid, attribute=attrs, endpoint=endpoint, request_path=http_request_path
|
||||||
|
)
|
||||||
|
raise AssertionError(f"Expected error on getting object with cid: {cid}")
|
||||||
|
except Exception as err:
|
||||||
|
match = error_pattern.casefold() in str(err).casefold()
|
||||||
|
assert match, f"Expected {err} to match {error_pattern}"
|
136
robot/resources/lib/python_keywords/json_transformers.py
Normal file
136
robot/resources/lib/python_keywords/json_transformers.py
Normal file
|
@ -0,0 +1,136 @@
|
||||||
|
"""
|
||||||
|
When doing requests to NeoFS, we get JSON output as an automatically decoded
|
||||||
|
structure from protobuf. Some fields are decoded with boilerplates and binary
|
||||||
|
values are Base64-encoded.
|
||||||
|
|
||||||
|
This module contains functions which rearrange the structure and reencode binary
|
||||||
|
data from Base64 to Base58.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import base64
|
||||||
|
|
||||||
|
import base58
|
||||||
|
|
||||||
|
|
||||||
|
def decode_simple_header(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function reencodes Simple Object header and its attributes.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
data = decode_common_fields(data)
|
||||||
|
|
||||||
|
# Normalize object attributes
|
||||||
|
data["header"]["attributes"] = {
|
||||||
|
attr["key"]: attr["value"] for attr in data["header"]["attributes"]
|
||||||
|
}
|
||||||
|
except Exception as exc:
|
||||||
|
raise ValueError(f"failed to decode JSON output: {exc}") from exc
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def decode_split_header(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function rearranges Complex Object header.
|
||||||
|
The header holds SplitID, a random unique
|
||||||
|
number, which is common among all splitted objects, and IDs of the Linking
|
||||||
|
Object and the last splitted Object.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
data["splitId"] = json_reencode(data["splitId"])
|
||||||
|
data["lastPart"] = json_reencode(data["lastPart"]["value"]) if data["lastPart"] else None
|
||||||
|
data["link"] = json_reencode(data["link"]["value"]) if data["link"] else None
|
||||||
|
except Exception as exc:
|
||||||
|
raise ValueError(f"failed to decode JSON output: {exc}") from exc
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def decode_linking_object(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function reencodes Linking Object header.
|
||||||
|
It contains IDs of child Objects and Split Chain data.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
data = decode_simple_header(data)
|
||||||
|
split = data["header"]["split"]
|
||||||
|
split["children"] = [json_reencode(item["value"]) for item in split["children"]]
|
||||||
|
split["splitID"] = json_reencode(split["splitID"])
|
||||||
|
split["previous"] = json_reencode(split["previous"]["value"]) if split["previous"] else None
|
||||||
|
split["parent"] = json_reencode(split["parent"]["value"]) if split["parent"] else None
|
||||||
|
except Exception as exc:
|
||||||
|
raise ValueError(f"failed to decode JSON output: {exc}") from exc
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def decode_storage_group(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function reencodes Storage Group header.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
data = decode_common_fields(data)
|
||||||
|
except Exception as exc:
|
||||||
|
raise ValueError(f"failed to decode JSON output: {exc}") from exc
|
||||||
|
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def decode_tombstone(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function reencodes Tombstone header.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
data = decode_simple_header(data)
|
||||||
|
data["header"]["sessionToken"] = decode_session_token(data["header"]["sessionToken"])
|
||||||
|
except Exception as exc:
|
||||||
|
raise ValueError(f"failed to decode JSON output: {exc}") from exc
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def decode_session_token(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
This function reencodes a fragment of header which contains
|
||||||
|
information about session token.
|
||||||
|
"""
|
||||||
|
target = data["body"]["object"]["target"]
|
||||||
|
target["container"] = json_reencode(target["container"]["value"])
|
||||||
|
target["objects"] = [json_reencode(obj["value"]) for obj in target["objects"]]
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
def json_reencode(data: str) -> str:
|
||||||
|
"""
|
||||||
|
According to JSON protocol, binary data (Object/Container/Storage Group IDs, etc)
|
||||||
|
is converted to string via Base58 encoder. But we usually operate with Base64-encoded format.
|
||||||
|
This function reencodes given Base58 string into the Base64 one.
|
||||||
|
"""
|
||||||
|
return base58.b58encode(base64.b64decode(data)).decode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
def encode_for_json(data: str) -> str:
|
||||||
|
"""
|
||||||
|
This function encodes binary data for sending them as protobuf
|
||||||
|
structures.
|
||||||
|
"""
|
||||||
|
return base64.b64encode(base58.b58decode(data)).decode("utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
def decode_common_fields(data: dict) -> dict:
|
||||||
|
"""
|
||||||
|
Despite of type (simple/complex Object, Storage Group, etc) every Object
|
||||||
|
header contains several common fields.
|
||||||
|
This function rearranges these fields.
|
||||||
|
"""
|
||||||
|
data["objectID"] = json_reencode(data["objectID"]["value"])
|
||||||
|
|
||||||
|
header = data["header"]
|
||||||
|
header["containerID"] = json_reencode(header["containerID"]["value"])
|
||||||
|
header["ownerID"] = json_reencode(header["ownerID"]["value"])
|
||||||
|
header["payloadHash"] = json_reencode(header["payloadHash"]["sum"])
|
||||||
|
header["version"] = f"{header['version']['major']}{header['version']['minor']}"
|
||||||
|
# Homomorphic hash is optional and its calculation might be disabled in trusted network
|
||||||
|
if header.get("homomorphicHash"):
|
||||||
|
header["homomorphicHash"] = json_reencode(header["homomorphicHash"]["sum"])
|
||||||
|
|
||||||
|
return data
|
634
robot/resources/lib/python_keywords/neofs_verbs.py
Normal file
634
robot/resources/lib/python_keywords/neofs_verbs.py
Normal file
|
@ -0,0 +1,634 @@
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import os
|
||||||
|
import re
|
||||||
|
import uuid
|
||||||
|
from typing import Any, Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import json_transformers
|
||||||
|
from cluster import Cluster
|
||||||
|
from common import ASSETS_DIR, NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object from random node")
|
||||||
|
def get_object_from_random_node(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
write_object: Optional[str] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
no_progress: bool = True,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
GET from NeoFS random storage node
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf GET is done
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
oid: Object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer (optional, str): path to Bearer Token file, appends to `--bearer` key
|
||||||
|
write_object (optional, str): path to downloaded file, appends to `--file` key
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
wallet_config(optional, str): path to the wallet config
|
||||||
|
no_progress(optional, bool): do not show progress bar
|
||||||
|
xhdr (optional, dict): Request X-Headers in form of Key=Value
|
||||||
|
session (optional, dict): path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
(str): path to downloaded file
|
||||||
|
"""
|
||||||
|
endpoint = cluster.get_random_storage_rpc_endpoint()
|
||||||
|
return get_object(
|
||||||
|
wallet,
|
||||||
|
cid,
|
||||||
|
oid,
|
||||||
|
shell,
|
||||||
|
endpoint,
|
||||||
|
bearer,
|
||||||
|
write_object,
|
||||||
|
xhdr,
|
||||||
|
wallet_config,
|
||||||
|
no_progress,
|
||||||
|
session,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get object from {endpoint}")
|
||||||
|
def get_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str = None,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
write_object: Optional[str] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
no_progress: bool = True,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
GET from NeoFS.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet (str): wallet on whose behalf GET is done
|
||||||
|
cid (str): ID of Container where we get the Object from
|
||||||
|
oid (str): Object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
write_object: path to downloaded file, appends to `--file` key
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
wallet_config(optional, str): path to the wallet config
|
||||||
|
no_progress(optional, bool): do not show progress bar
|
||||||
|
xhdr (optional, dict): Request X-Headers in form of Key=Value
|
||||||
|
session (optional, dict): path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
(str): path to downloaded file
|
||||||
|
"""
|
||||||
|
|
||||||
|
if not write_object:
|
||||||
|
write_object = str(uuid.uuid4())
|
||||||
|
file_path = os.path.join(ASSETS_DIR, write_object)
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
cli.object.get(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
file=file_path,
|
||||||
|
bearer=bearer,
|
||||||
|
no_progress=no_progress,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
return file_path
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Range Hash from {endpoint}")
|
||||||
|
def get_range_hash(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
range_cut: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
GETRANGEHASH of given Object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf GETRANGEHASH is done
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
oid: Object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
range_cut: Range to take hash from in the form offset1:length1,...,
|
||||||
|
value to pass to the `--range` parameter
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
xhdr: Request X-Headers in form of Key=Values
|
||||||
|
session: Filepath to a JSON- or binary-encoded token of the object RANGEHASH session.
|
||||||
|
Returns:
|
||||||
|
None
|
||||||
|
"""
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.hash(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
range=range_cut,
|
||||||
|
bearer=bearer,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
# cutting off output about range offset and length
|
||||||
|
return result.stdout.split(":")[1].strip()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object to random node")
|
||||||
|
def put_object_to_random_node(
|
||||||
|
wallet: str,
|
||||||
|
path: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
attributes: Optional[dict] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
expire_at: Optional[int] = None,
|
||||||
|
no_progress: bool = True,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
PUT of given file to a random storage node.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf PUT is done
|
||||||
|
path: path to file to be PUT
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
shell: executor for cli command
|
||||||
|
cluster: cluster under test
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
attributes: User attributes in form of Key1=Value1,Key2=Value2
|
||||||
|
cluster: cluster under test
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
no_progress: do not show progress bar
|
||||||
|
expire_at: Last epoch in the life of the object
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
session: path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
ID of uploaded Object
|
||||||
|
"""
|
||||||
|
|
||||||
|
endpoint = cluster.get_random_storage_rpc_endpoint()
|
||||||
|
return put_object(
|
||||||
|
wallet,
|
||||||
|
path,
|
||||||
|
cid,
|
||||||
|
shell,
|
||||||
|
endpoint,
|
||||||
|
bearer,
|
||||||
|
attributes,
|
||||||
|
xhdr,
|
||||||
|
wallet_config,
|
||||||
|
expire_at,
|
||||||
|
no_progress,
|
||||||
|
session,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put object at {endpoint} in container {cid}")
|
||||||
|
def put_object(
|
||||||
|
wallet: str,
|
||||||
|
path: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
attributes: Optional[dict] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
expire_at: Optional[int] = None,
|
||||||
|
no_progress: bool = True,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
PUT of given file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf PUT is done
|
||||||
|
path: path to file to be PUT
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
attributes: User attributes in form of Key1=Value1,Key2=Value2
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
no_progress: do not show progress bar
|
||||||
|
expire_at: Last epoch in the life of the object
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
session: path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
(str): ID of uploaded Object
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.put(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
file=path,
|
||||||
|
cid=cid,
|
||||||
|
attributes=attributes,
|
||||||
|
bearer=bearer,
|
||||||
|
expire_at=expire_at,
|
||||||
|
no_progress=no_progress,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
# splitting CLI output to lines and taking the penultimate line
|
||||||
|
id_str = result.stdout.strip().split("\n")[-2]
|
||||||
|
oid = id_str.split(":")[1]
|
||||||
|
return oid.strip()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete object {cid}/{oid} from {endpoint}")
|
||||||
|
def delete_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str = None,
|
||||||
|
bearer: str = "",
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
DELETE an Object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf DELETE is done
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
oid: ID of Object we are going to delete
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
session: path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
(str): Tombstone ID
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.delete(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
bearer=bearer,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
id_str = result.stdout.split("\n")[1]
|
||||||
|
tombstone = id_str.split(":")[1]
|
||||||
|
return tombstone.strip()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Range")
|
||||||
|
def get_range(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
range_cut: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
bearer: str = "",
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
GETRANGE an Object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf GETRANGE is done
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
oid: ID of Object we are going to request
|
||||||
|
range_cut: range to take data from in the form offset:length
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
session: path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
(str, bytes) - path to the file with range content and content of this file as bytes
|
||||||
|
"""
|
||||||
|
range_file_path = os.path.join(ASSETS_DIR, str(uuid.uuid4()))
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
cli.object.range(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
range=range_cut,
|
||||||
|
file=range_file_path,
|
||||||
|
bearer=bearer,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
with open(range_file_path, "rb") as file:
|
||||||
|
content = file.read()
|
||||||
|
return range_file_path, content
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Lock Object")
|
||||||
|
def lock_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
lifetime: Optional[int] = None,
|
||||||
|
expire_at: Optional[int] = None,
|
||||||
|
address: Optional[str] = None,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
ttl: Optional[int] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Lock object in container.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
address: Address of wallet account.
|
||||||
|
bearer: File with signed JSON or binary encoded bearer token.
|
||||||
|
cid: Container ID.
|
||||||
|
oid: Object ID.
|
||||||
|
lifetime: Lock lifetime.
|
||||||
|
expire_at: Lock expiration epoch.
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
session: Path to a JSON-encoded container session token.
|
||||||
|
ttl: TTL value in request meta header (default 2).
|
||||||
|
wallet: WIF (NEP-2) string or path to the wallet or binary key.
|
||||||
|
xhdr: Dict with request X-Headers.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Lock object ID
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.lock(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
lifetime=lifetime,
|
||||||
|
expire_at=expire_at,
|
||||||
|
address=address,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
bearer=bearer,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
ttl=ttl,
|
||||||
|
)
|
||||||
|
|
||||||
|
# splitting CLI output to lines and taking the penultimate line
|
||||||
|
id_str = result.stdout.strip().split("\n")[0]
|
||||||
|
oid = id_str.split(":")[1]
|
||||||
|
return oid.strip()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Search object")
|
||||||
|
def search_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
bearer: str = "",
|
||||||
|
filters: Optional[dict] = None,
|
||||||
|
expected_objects_list: Optional[list] = None,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
phy: bool = False,
|
||||||
|
root: bool = False,
|
||||||
|
) -> list:
|
||||||
|
"""
|
||||||
|
SEARCH an Object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet: wallet on whose behalf SEARCH is done
|
||||||
|
cid: ID of Container where we get the Object from
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer: path to Bearer Token file, appends to `--bearer` key
|
||||||
|
endpoint: NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
filters: key=value pairs to filter Objects
|
||||||
|
expected_objects_list: a list of ObjectIDs to compare found Objects with
|
||||||
|
wallet_config: path to the wallet config
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
session: path to a JSON-encoded container session token
|
||||||
|
phy: Search physically stored objects.
|
||||||
|
root: Search for user objects.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
list of found ObjectIDs
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.search(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
bearer=bearer,
|
||||||
|
xhdr=xhdr,
|
||||||
|
filters=[f"{filter_key} EQ {filter_val}" for filter_key, filter_val in filters.items()]
|
||||||
|
if filters
|
||||||
|
else None,
|
||||||
|
session=session,
|
||||||
|
phy=phy,
|
||||||
|
root=root,
|
||||||
|
)
|
||||||
|
|
||||||
|
found_objects = re.findall(r"(\w{43,44})", result.stdout)
|
||||||
|
|
||||||
|
if expected_objects_list:
|
||||||
|
if sorted(found_objects) == sorted(expected_objects_list):
|
||||||
|
logger.info(
|
||||||
|
f"Found objects list '{found_objects}' "
|
||||||
|
f"is equal for expected list '{expected_objects_list}'"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
logger.warning(
|
||||||
|
f"Found object list {found_objects} "
|
||||||
|
f"is not equal to expected list '{expected_objects_list}'"
|
||||||
|
)
|
||||||
|
|
||||||
|
return found_objects
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get netmap netinfo")
|
||||||
|
def get_netmap_netinfo(
|
||||||
|
wallet: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
address: Optional[str] = None,
|
||||||
|
ttl: Optional[int] = None,
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
) -> dict[str, Any]:
|
||||||
|
"""
|
||||||
|
Get netmap netinfo output from node
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet (str): wallet on whose behalf request is done
|
||||||
|
shell: executor for cli command
|
||||||
|
endpoint (optional, str): NeoFS endpoint to send request to, appends to `--rpc-endpoint` key
|
||||||
|
address: Address of wallet account
|
||||||
|
ttl: TTL value in request meta header (default 2)
|
||||||
|
wallet: Path to the wallet or binary key
|
||||||
|
xhdr: Request X-Headers in form of Key=Value
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
(dict): dict of parsed command output
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
output = cli.netmap.netinfo(
|
||||||
|
wallet=wallet,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
address=address,
|
||||||
|
ttl=ttl,
|
||||||
|
xhdr=xhdr,
|
||||||
|
)
|
||||||
|
|
||||||
|
settings = dict()
|
||||||
|
|
||||||
|
patterns = [
|
||||||
|
(re.compile("(.*): (\d+)"), int),
|
||||||
|
(re.compile("(.*): (false|true)"), bool),
|
||||||
|
(re.compile("(.*): (\d+\.\d+)"), float),
|
||||||
|
]
|
||||||
|
for pattern, func in patterns:
|
||||||
|
for setting, value in re.findall(pattern, output.stdout):
|
||||||
|
settings[setting.lower().strip().replace(" ", "_")] = func(value)
|
||||||
|
|
||||||
|
return settings
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Head object")
|
||||||
|
def head_object(
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
oid: str,
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
bearer: str = "",
|
||||||
|
xhdr: Optional[dict] = None,
|
||||||
|
json_output: bool = True,
|
||||||
|
is_raw: bool = False,
|
||||||
|
is_direct: bool = False,
|
||||||
|
wallet_config: Optional[str] = None,
|
||||||
|
session: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
HEAD an Object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
wallet (str): wallet on whose behalf HEAD is done
|
||||||
|
cid (str): ID of Container where we get the Object from
|
||||||
|
oid (str): ObjectID to HEAD
|
||||||
|
shell: executor for cli command
|
||||||
|
bearer (optional, str): path to Bearer Token file, appends to `--bearer` key
|
||||||
|
endpoint(optional, str): NeoFS endpoint to send request to
|
||||||
|
json_output(optional, bool): return response in JSON format or not; this flag
|
||||||
|
turns into `--json` key
|
||||||
|
is_raw(optional, bool): send "raw" request or not; this flag
|
||||||
|
turns into `--raw` key
|
||||||
|
is_direct(optional, bool): send request directly to the node or not; this flag
|
||||||
|
turns into `--ttl 1` key
|
||||||
|
wallet_config(optional, str): path to the wallet config
|
||||||
|
xhdr (optional, dict): Request X-Headers in form of Key=Value
|
||||||
|
session (optional, dict): path to a JSON-encoded container session token
|
||||||
|
Returns:
|
||||||
|
depending on the `json_output` parameter value, the function returns
|
||||||
|
(dict): HEAD response in JSON format
|
||||||
|
or
|
||||||
|
(str): HEAD response as a plain text
|
||||||
|
"""
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, wallet_config or WALLET_CONFIG)
|
||||||
|
result = cli.object.head(
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=oid,
|
||||||
|
bearer=bearer,
|
||||||
|
json_mode=json_output,
|
||||||
|
raw=is_raw,
|
||||||
|
ttl=1 if is_direct else None,
|
||||||
|
xhdr=xhdr,
|
||||||
|
session=session,
|
||||||
|
)
|
||||||
|
|
||||||
|
if not json_output:
|
||||||
|
return result
|
||||||
|
|
||||||
|
try:
|
||||||
|
decoded = json.loads(result.stdout)
|
||||||
|
except Exception as exc:
|
||||||
|
# If we failed to parse output as JSON, the cause might be
|
||||||
|
# the plain text string in the beginning of the output.
|
||||||
|
# Here we cut off first string and try to parse again.
|
||||||
|
logger.info(f"failed to parse output: {exc}")
|
||||||
|
logger.info("parsing output in another way")
|
||||||
|
fst_line_idx = result.stdout.find("\n")
|
||||||
|
decoded = json.loads(result.stdout[fst_line_idx:])
|
||||||
|
|
||||||
|
# If response is Complex Object header, it has `splitId` key
|
||||||
|
if "splitId" in decoded.keys():
|
||||||
|
logger.info("decoding split header")
|
||||||
|
return json_transformers.decode_split_header(decoded)
|
||||||
|
|
||||||
|
# If response is Last or Linking Object header,
|
||||||
|
# it has `header` dictionary and non-null `split` dictionary
|
||||||
|
if "split" in decoded["header"].keys():
|
||||||
|
if decoded["header"]["split"]:
|
||||||
|
logger.info("decoding linking object")
|
||||||
|
return json_transformers.decode_linking_object(decoded)
|
||||||
|
|
||||||
|
if decoded["header"]["objectType"] == "STORAGE_GROUP":
|
||||||
|
logger.info("decoding storage group")
|
||||||
|
return json_transformers.decode_storage_group(decoded)
|
||||||
|
|
||||||
|
if decoded["header"]["objectType"] == "TOMBSTONE":
|
||||||
|
logger.info("decoding tombstone")
|
||||||
|
return json_transformers.decode_tombstone(decoded)
|
||||||
|
|
||||||
|
logger.info("decoding simple header")
|
||||||
|
return json_transformers.decode_simple_header(decoded)
|
246
robot/resources/lib/python_keywords/node_management.py
Normal file
246
robot/resources/lib/python_keywords/node_management.py
Normal file
|
@ -0,0 +1,246 @@
|
||||||
|
import logging
|
||||||
|
import random
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import Cluster, StorageNode
|
||||||
|
from common import MORPH_BLOCK_TIME, NEOFS_CLI_EXEC
|
||||||
|
from epoch import tick_epoch
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from utility import parse_time
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class HealthStatus:
|
||||||
|
network_status: Optional[str] = None
|
||||||
|
health_status: Optional[str] = None
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_stdout(output: str) -> "HealthStatus":
|
||||||
|
network, health = None, None
|
||||||
|
for line in output.split("\n"):
|
||||||
|
if "Network status" in line:
|
||||||
|
network = line.split(":")[-1].strip()
|
||||||
|
if "Health status" in line:
|
||||||
|
health = line.split(":")[-1].strip()
|
||||||
|
return HealthStatus(network, health)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Stop random storage nodes")
|
||||||
|
def stop_random_storage_nodes(number: int, nodes: list[StorageNode]) -> list[StorageNode]:
|
||||||
|
"""
|
||||||
|
Shuts down the given number of randomly selected storage nodes.
|
||||||
|
Args:
|
||||||
|
number: the number of storage nodes to stop
|
||||||
|
nodes: the list of storage nodes to stop
|
||||||
|
Returns:
|
||||||
|
the list of nodes that were stopped
|
||||||
|
"""
|
||||||
|
nodes_to_stop = random.sample(nodes, number)
|
||||||
|
for node in nodes_to_stop:
|
||||||
|
node.stop_service()
|
||||||
|
return nodes_to_stop
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Start storage node")
|
||||||
|
def start_storage_nodes(nodes: list[StorageNode]) -> None:
|
||||||
|
"""
|
||||||
|
The function starts specified storage nodes.
|
||||||
|
Args:
|
||||||
|
nodes: the list of nodes to start
|
||||||
|
"""
|
||||||
|
for node in nodes:
|
||||||
|
node.start_service()
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Locode from random storage node")
|
||||||
|
def get_locode_from_random_node(cluster: Cluster) -> str:
|
||||||
|
node = random.choice(cluster.storage_nodes)
|
||||||
|
locode = node.get_un_locode()
|
||||||
|
logger.info(f"Chosen '{locode}' locode from node {node}")
|
||||||
|
return locode
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Healthcheck for storage node {node}")
|
||||||
|
def storage_node_healthcheck(node: StorageNode) -> HealthStatus:
|
||||||
|
"""
|
||||||
|
The function returns storage node's health status.
|
||||||
|
Args:
|
||||||
|
node: storage node for which health status should be retrieved.
|
||||||
|
Returns:
|
||||||
|
health status as HealthStatus object.
|
||||||
|
"""
|
||||||
|
command = "control healthcheck"
|
||||||
|
output = _run_control_command_with_retries(node, command)
|
||||||
|
return HealthStatus.from_stdout(output)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Set status for {node}")
|
||||||
|
def storage_node_set_status(node: StorageNode, status: str, retries: int = 0) -> None:
|
||||||
|
"""
|
||||||
|
The function sets particular status for given node.
|
||||||
|
Args:
|
||||||
|
node: node for which status should be set.
|
||||||
|
status: online or offline.
|
||||||
|
retries (optional, int): number of retry attempts if it didn't work from the first time
|
||||||
|
"""
|
||||||
|
command = f"control set-status --status {status}"
|
||||||
|
_run_control_command_with_retries(node, command, retries)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get netmap snapshot")
|
||||||
|
def get_netmap_snapshot(node: StorageNode, shell: Shell) -> str:
|
||||||
|
"""
|
||||||
|
The function returns string representation of netmap snapshot.
|
||||||
|
Args:
|
||||||
|
node: node from which netmap snapshot should be requested.
|
||||||
|
Returns:
|
||||||
|
string representation of netmap
|
||||||
|
"""
|
||||||
|
|
||||||
|
storage_wallet_config = node.get_wallet_config_path()
|
||||||
|
storage_wallet_path = node.get_wallet_path()
|
||||||
|
|
||||||
|
cli = NeofsCli(shell, NEOFS_CLI_EXEC, config_file=storage_wallet_config)
|
||||||
|
return cli.netmap.snapshot(
|
||||||
|
rpc_endpoint=node.get_rpc_endpoint(),
|
||||||
|
wallet=storage_wallet_path,
|
||||||
|
).stdout
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get shard list for {node}")
|
||||||
|
def node_shard_list(node: StorageNode) -> list[str]:
|
||||||
|
"""
|
||||||
|
The function returns list of shards for specified storage node.
|
||||||
|
Args:
|
||||||
|
node: node for which shards should be returned.
|
||||||
|
Returns:
|
||||||
|
list of shards.
|
||||||
|
"""
|
||||||
|
command = "control shards list"
|
||||||
|
output = _run_control_command_with_retries(node, command)
|
||||||
|
return re.findall(r"Shard (.*):", output)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Shard set for {node}")
|
||||||
|
def node_shard_set_mode(node: StorageNode, shard: str, mode: str) -> str:
|
||||||
|
"""
|
||||||
|
The function sets mode for specified shard.
|
||||||
|
Args:
|
||||||
|
node: node on which shard mode should be set.
|
||||||
|
"""
|
||||||
|
command = f"control shards set-mode --id {shard} --mode {mode}"
|
||||||
|
return _run_control_command_with_retries(node, command)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Drop object from {node}")
|
||||||
|
def drop_object(node: StorageNode, cid: str, oid: str) -> str:
|
||||||
|
"""
|
||||||
|
The function drops object from specified node.
|
||||||
|
Args:
|
||||||
|
node_id str: node from which object should be dropped.
|
||||||
|
"""
|
||||||
|
command = f"control drop-objects -o {cid}/{oid}"
|
||||||
|
return _run_control_command_with_retries(node, command)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete data from host for node {node}")
|
||||||
|
def delete_node_data(node: StorageNode) -> None:
|
||||||
|
node.stop_service()
|
||||||
|
node.host.delete_storage_node_data(node.name)
|
||||||
|
time.sleep(parse_time(MORPH_BLOCK_TIME))
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Exclude node {node_to_exclude} from network map")
|
||||||
|
def exclude_node_from_network_map(
|
||||||
|
node_to_exclude: StorageNode,
|
||||||
|
alive_node: StorageNode,
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
) -> None:
|
||||||
|
node_netmap_key = node_to_exclude.get_wallet_public_key()
|
||||||
|
|
||||||
|
storage_node_set_status(node_to_exclude, status="offline")
|
||||||
|
|
||||||
|
time.sleep(parse_time(MORPH_BLOCK_TIME))
|
||||||
|
tick_epoch(shell, cluster)
|
||||||
|
|
||||||
|
snapshot = get_netmap_snapshot(node=alive_node, shell=shell)
|
||||||
|
assert (
|
||||||
|
node_netmap_key not in snapshot
|
||||||
|
), f"Expected node with key {node_netmap_key} to be absent in network map"
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Include node {node_to_include} into network map")
|
||||||
|
def include_node_to_network_map(
|
||||||
|
node_to_include: StorageNode,
|
||||||
|
alive_node: StorageNode,
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
) -> None:
|
||||||
|
storage_node_set_status(node_to_include, status="online")
|
||||||
|
|
||||||
|
# Per suggestion of @fyrchik we need to wait for 2 blocks after we set status and after tick epoch.
|
||||||
|
# First sleep can be omitted after https://github.com/nspcc-dev/neofs-node/issues/1790 complete.
|
||||||
|
|
||||||
|
time.sleep(parse_time(MORPH_BLOCK_TIME) * 2)
|
||||||
|
tick_epoch(shell, cluster)
|
||||||
|
time.sleep(parse_time(MORPH_BLOCK_TIME) * 2)
|
||||||
|
|
||||||
|
check_node_in_map(node_to_include, shell, alive_node)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Check node {node} in network map")
|
||||||
|
def check_node_in_map(
|
||||||
|
node: StorageNode, shell: Shell, alive_node: Optional[StorageNode] = None
|
||||||
|
) -> None:
|
||||||
|
alive_node = alive_node or node
|
||||||
|
|
||||||
|
node_netmap_key = node.get_wallet_public_key()
|
||||||
|
logger.info(f"Node ({node.label}) netmap key: {node_netmap_key}")
|
||||||
|
|
||||||
|
snapshot = get_netmap_snapshot(alive_node, shell)
|
||||||
|
assert (
|
||||||
|
node_netmap_key in snapshot
|
||||||
|
), f"Expected node with key {node_netmap_key} to be in network map"
|
||||||
|
|
||||||
|
|
||||||
|
def _run_control_command_with_retries(node: StorageNode, command: str, retries: int = 0) -> str:
|
||||||
|
for attempt in range(1 + retries): # original attempt + specified retries
|
||||||
|
try:
|
||||||
|
return _run_control_command(node, command)
|
||||||
|
except AssertionError as err:
|
||||||
|
if attempt < retries:
|
||||||
|
logger.warning(f"Command {command} failed with error {err} and will be retried")
|
||||||
|
continue
|
||||||
|
raise AssertionError(f"Command {command} failed with error {err}") from err
|
||||||
|
|
||||||
|
|
||||||
|
def _run_control_command(node: StorageNode, command: str) -> None:
|
||||||
|
host = node.host
|
||||||
|
|
||||||
|
service_config = host.get_service_config(node.name)
|
||||||
|
wallet_path = service_config.attributes["wallet_path"]
|
||||||
|
wallet_password = service_config.attributes["wallet_password"]
|
||||||
|
control_endpoint = service_config.attributes["control_endpoint"]
|
||||||
|
|
||||||
|
shell = host.get_shell()
|
||||||
|
wallet_config_path = f"/tmp/{node.name}-config.yaml"
|
||||||
|
wallet_config = f'password: "{wallet_password}"'
|
||||||
|
shell.exec(f"echo '{wallet_config}' > {wallet_config_path}")
|
||||||
|
|
||||||
|
cli_config = host.get_cli_config("neofs-cli")
|
||||||
|
|
||||||
|
# TODO: implement cli.control
|
||||||
|
# cli = NeofsCli(shell, cli_config.exec_path, wallet_config_path)
|
||||||
|
result = shell.exec(
|
||||||
|
f"{cli_config.exec_path} {command} --endpoint {control_endpoint} "
|
||||||
|
f"--wallet {wallet_path} --config {wallet_config_path}"
|
||||||
|
)
|
||||||
|
return result.stdout
|
|
@ -1,10 +1,11 @@
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
|
||||||
import allure
|
import allure
|
||||||
from frostfs_testlib.resources.cli import CLI_DEFAULT_TIMEOUT
|
from cluster import Cluster
|
||||||
from frostfs_testlib.resources.error_patterns import OBJECT_ACCESS_DENIED
|
from file_helper import get_file_hash
|
||||||
from frostfs_testlib.shell import Shell
|
from grpc_responses import OBJECT_ACCESS_DENIED, error_matches_status
|
||||||
from frostfs_testlib.steps.cli.object import (
|
from neofs_testlib.shell import Shell
|
||||||
|
from python_keywords.neofs_verbs import (
|
||||||
delete_object,
|
delete_object,
|
||||||
get_object_from_random_node,
|
get_object_from_random_node,
|
||||||
get_range,
|
get_range,
|
||||||
|
@ -13,9 +14,6 @@ from frostfs_testlib.steps.cli.object import (
|
||||||
put_object_to_random_node,
|
put_object_to_random_node,
|
||||||
search_object,
|
search_object,
|
||||||
)
|
)
|
||||||
from frostfs_testlib.storage.cluster import Cluster
|
|
||||||
from frostfs_testlib.utils import string_utils
|
|
||||||
from frostfs_testlib.utils.file_utils import get_file_hash
|
|
||||||
|
|
||||||
OPERATION_ERROR_TYPE = RuntimeError
|
OPERATION_ERROR_TYPE = RuntimeError
|
||||||
|
|
||||||
|
@ -44,7 +42,7 @@ def can_get_object(
|
||||||
cluster=cluster,
|
cluster=cluster,
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -77,7 +75,7 @@ def can_put_object(
|
||||||
cluster=cluster,
|
cluster=cluster,
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -107,7 +105,7 @@ def can_delete_object(
|
||||||
endpoint=endpoint,
|
endpoint=endpoint,
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -123,7 +121,6 @@ def can_get_head_object(
|
||||||
bearer: Optional[str] = None,
|
bearer: Optional[str] = None,
|
||||||
wallet_config: Optional[str] = None,
|
wallet_config: Optional[str] = None,
|
||||||
xhdr: Optional[dict] = None,
|
xhdr: Optional[dict] = None,
|
||||||
timeout: Optional[str] = CLI_DEFAULT_TIMEOUT,
|
|
||||||
) -> bool:
|
) -> bool:
|
||||||
with allure.step("Try get head of object"):
|
with allure.step("Try get head of object"):
|
||||||
try:
|
try:
|
||||||
|
@ -136,10 +133,9 @@ def can_get_head_object(
|
||||||
xhdr=xhdr,
|
xhdr=xhdr,
|
||||||
shell=shell,
|
shell=shell,
|
||||||
endpoint=endpoint,
|
endpoint=endpoint,
|
||||||
timeout=timeout,
|
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -155,7 +151,6 @@ def can_get_range_of_object(
|
||||||
bearer: Optional[str] = None,
|
bearer: Optional[str] = None,
|
||||||
wallet_config: Optional[str] = None,
|
wallet_config: Optional[str] = None,
|
||||||
xhdr: Optional[dict] = None,
|
xhdr: Optional[dict] = None,
|
||||||
timeout: Optional[str] = CLI_DEFAULT_TIMEOUT,
|
|
||||||
) -> bool:
|
) -> bool:
|
||||||
with allure.step("Try get range of object"):
|
with allure.step("Try get range of object"):
|
||||||
try:
|
try:
|
||||||
|
@ -169,10 +164,9 @@ def can_get_range_of_object(
|
||||||
xhdr=xhdr,
|
xhdr=xhdr,
|
||||||
shell=shell,
|
shell=shell,
|
||||||
endpoint=endpoint,
|
endpoint=endpoint,
|
||||||
timeout=timeout,
|
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -188,7 +182,6 @@ def can_get_range_hash_of_object(
|
||||||
bearer: Optional[str] = None,
|
bearer: Optional[str] = None,
|
||||||
wallet_config: Optional[str] = None,
|
wallet_config: Optional[str] = None,
|
||||||
xhdr: Optional[dict] = None,
|
xhdr: Optional[dict] = None,
|
||||||
timeout: Optional[str] = CLI_DEFAULT_TIMEOUT,
|
|
||||||
) -> bool:
|
) -> bool:
|
||||||
with allure.step("Try get range hash of object"):
|
with allure.step("Try get range hash of object"):
|
||||||
try:
|
try:
|
||||||
|
@ -202,10 +195,9 @@ def can_get_range_hash_of_object(
|
||||||
xhdr=xhdr,
|
xhdr=xhdr,
|
||||||
shell=shell,
|
shell=shell,
|
||||||
endpoint=endpoint,
|
endpoint=endpoint,
|
||||||
timeout=timeout,
|
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
||||||
|
@ -221,7 +213,6 @@ def can_search_object(
|
||||||
bearer: Optional[str] = None,
|
bearer: Optional[str] = None,
|
||||||
wallet_config: Optional[str] = None,
|
wallet_config: Optional[str] = None,
|
||||||
xhdr: Optional[dict] = None,
|
xhdr: Optional[dict] = None,
|
||||||
timeout: Optional[str] = CLI_DEFAULT_TIMEOUT,
|
|
||||||
) -> bool:
|
) -> bool:
|
||||||
with allure.step("Try search object in container"):
|
with allure.step("Try search object in container"):
|
||||||
try:
|
try:
|
||||||
|
@ -233,10 +224,9 @@ def can_search_object(
|
||||||
xhdr=xhdr,
|
xhdr=xhdr,
|
||||||
shell=shell,
|
shell=shell,
|
||||||
endpoint=endpoint,
|
endpoint=endpoint,
|
||||||
timeout=timeout,
|
|
||||||
)
|
)
|
||||||
except OPERATION_ERROR_TYPE as err:
|
except OPERATION_ERROR_TYPE as err:
|
||||||
assert string_utils.is_str_match_pattern(
|
assert error_matches_status(
|
||||||
err, OBJECT_ACCESS_DENIED
|
err, OBJECT_ACCESS_DENIED
|
||||||
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
), f"Expected {err} to match {OBJECT_ACCESS_DENIED}"
|
||||||
return False
|
return False
|
213
robot/resources/lib/python_keywords/payment_neogo.py
Normal file
213
robot/resources/lib/python_keywords/payment_neogo.py
Normal file
|
@ -0,0 +1,213 @@
|
||||||
|
import base64
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
import re
|
||||||
|
import time
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import MainChain, MorphChain
|
||||||
|
from common import GAS_HASH, MAINNET_BLOCK_TIME, NEOFS_CONTRACT, NEOGO_EXECUTABLE
|
||||||
|
from neo3 import wallet as neo3_wallet
|
||||||
|
from neofs_testlib.cli import NeoGo
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_testlib.utils.converters import contract_hash_to_address
|
||||||
|
from neofs_testlib.utils.wallet import get_last_address_from_wallet
|
||||||
|
from utility import parse_time
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
EMPTY_PASSWORD = ""
|
||||||
|
TX_PERSIST_TIMEOUT = 15 # seconds
|
||||||
|
ASSET_POWER_MAINCHAIN = 10**8
|
||||||
|
ASSET_POWER_SIDECHAIN = 10**12
|
||||||
|
|
||||||
|
|
||||||
|
def get_nns_contract_hash(morph_chain: MorphChain) -> str:
|
||||||
|
return morph_chain.rpc_client.get_contract_state(1)["hash"]
|
||||||
|
|
||||||
|
|
||||||
|
def get_contract_hash(morph_chain: MorphChain, resolve_name: str, shell: Shell) -> str:
|
||||||
|
nns_contract_hash = get_nns_contract_hash(morph_chain)
|
||||||
|
neogo = NeoGo(shell=shell, neo_go_exec_path=NEOGO_EXECUTABLE)
|
||||||
|
out = neogo.contract.testinvokefunction(
|
||||||
|
scripthash=nns_contract_hash,
|
||||||
|
method="resolve",
|
||||||
|
arguments=f"string:{resolve_name} int:16",
|
||||||
|
rpc_endpoint=morph_chain.get_endpoint(),
|
||||||
|
)
|
||||||
|
stack_data = json.loads(out.stdout.replace("\n", ""))["stack"][0]["value"]
|
||||||
|
return bytes.decode(base64.b64decode(stack_data[0]["value"]))
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Withdraw Mainnet Gas")
|
||||||
|
def withdraw_mainnet_gas(shell: Shell, main_chain: MainChain, wlt: str, amount: int):
|
||||||
|
address = get_last_address_from_wallet(wlt, EMPTY_PASSWORD)
|
||||||
|
scripthash = neo3_wallet.Account.address_to_script_hash(address)
|
||||||
|
|
||||||
|
neogo = NeoGo(shell=shell, neo_go_exec_path=NEOGO_EXECUTABLE)
|
||||||
|
out = neogo.contract.invokefunction(
|
||||||
|
wallet=wlt,
|
||||||
|
address=address,
|
||||||
|
rpc_endpoint=main_chain.get_endpoint(),
|
||||||
|
scripthash=NEOFS_CONTRACT,
|
||||||
|
method="withdraw",
|
||||||
|
arguments=f"{scripthash} int:{amount}",
|
||||||
|
multisig_hash=f"{scripthash}:Global",
|
||||||
|
wallet_password="",
|
||||||
|
)
|
||||||
|
|
||||||
|
m = re.match(r"^Sent invocation transaction (\w{64})$", out.stdout)
|
||||||
|
if m is None:
|
||||||
|
raise Exception("Can not get Tx.")
|
||||||
|
tx = m.group(1)
|
||||||
|
if not transaction_accepted(tx):
|
||||||
|
raise AssertionError(f"TX {tx} hasn't been processed")
|
||||||
|
|
||||||
|
|
||||||
|
def transaction_accepted(main_chain: MainChain, tx_id: str):
|
||||||
|
"""
|
||||||
|
This function returns True in case of accepted TX.
|
||||||
|
Args:
|
||||||
|
tx_id(str): transaction ID
|
||||||
|
Returns:
|
||||||
|
(bool)
|
||||||
|
"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
for _ in range(0, TX_PERSIST_TIMEOUT):
|
||||||
|
time.sleep(1)
|
||||||
|
resp = main_chain.rpc_client.get_transaction_height(tx_id)
|
||||||
|
if resp is not None:
|
||||||
|
logger.info(f"TX is accepted in block: {resp}")
|
||||||
|
return True
|
||||||
|
except Exception as out:
|
||||||
|
logger.info(f"request failed with error: {out}")
|
||||||
|
raise out
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get NeoFS Balance")
|
||||||
|
def get_balance(shell: Shell, morph_chain: MorphChain, wallet_path: str, wallet_password: str = ""):
|
||||||
|
"""
|
||||||
|
This function returns NeoFS balance for given wallet.
|
||||||
|
"""
|
||||||
|
with open(wallet_path) as wallet_file:
|
||||||
|
wallet = neo3_wallet.Wallet.from_json(json.load(wallet_file), password=wallet_password)
|
||||||
|
acc = wallet.accounts[-1]
|
||||||
|
payload = [{"type": "Hash160", "value": str(acc.script_hash)}]
|
||||||
|
try:
|
||||||
|
resp = morph_chain.rpc_client.invoke_function(
|
||||||
|
get_contract_hash(morph_chain, "balance.neofs", shell=shell), "balanceOf", payload
|
||||||
|
)
|
||||||
|
logger.info(f"Got response \n{resp}")
|
||||||
|
value = int(resp["stack"][0]["value"])
|
||||||
|
return value / ASSET_POWER_SIDECHAIN
|
||||||
|
except Exception as out:
|
||||||
|
logger.error(f"failed to get wallet balance: {out}")
|
||||||
|
raise out
|
||||||
|
|
||||||
|
|
||||||
|
@allure.title("Transfer Gas")
|
||||||
|
def transfer_gas(
|
||||||
|
shell: Shell,
|
||||||
|
amount: int,
|
||||||
|
main_chain: MainChain,
|
||||||
|
wallet_from_path: Optional[str] = None,
|
||||||
|
wallet_from_password: Optional[str] = None,
|
||||||
|
address_from: Optional[str] = None,
|
||||||
|
address_to: Optional[str] = None,
|
||||||
|
wallet_to_path: Optional[str] = None,
|
||||||
|
wallet_to_password: Optional[str] = None,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
This function transfer GAS in main chain from mainnet wallet to
|
||||||
|
the provided wallet. If the wallet contains more than one address,
|
||||||
|
the assets will be transferred to the last one.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
wallet_from_password: Password of the wallet; it is required to decode the wallet
|
||||||
|
and extract its addresses.
|
||||||
|
wallet_from_path: Path to chain node wallet.
|
||||||
|
address_from: The address of the wallet to transfer assets from.
|
||||||
|
wallet_to_path: The path to the wallet to transfer assets to.
|
||||||
|
wallet_to_password: The password to the wallet to transfer assets to.
|
||||||
|
address_to: The address of the wallet to transfer assets to.
|
||||||
|
amount: Amount of gas to transfer.
|
||||||
|
"""
|
||||||
|
wallet_from_path = wallet_from_path or main_chain.get_wallet_path()
|
||||||
|
wallet_from_password = (
|
||||||
|
wallet_from_password
|
||||||
|
if wallet_from_password is not None
|
||||||
|
else main_chain.get_wallet_password()
|
||||||
|
)
|
||||||
|
address_from = address_from or get_last_address_from_wallet(
|
||||||
|
wallet_from_path, wallet_from_password
|
||||||
|
)
|
||||||
|
address_to = address_to or get_last_address_from_wallet(wallet_to_path, wallet_to_password)
|
||||||
|
|
||||||
|
neogo = NeoGo(shell, neo_go_exec_path=NEOGO_EXECUTABLE)
|
||||||
|
out = neogo.nep17.transfer(
|
||||||
|
rpc_endpoint=main_chain.get_endpoint(),
|
||||||
|
wallet=wallet_from_path,
|
||||||
|
wallet_password=wallet_from_password,
|
||||||
|
amount=amount,
|
||||||
|
from_address=address_from,
|
||||||
|
to_address=address_to,
|
||||||
|
token="GAS",
|
||||||
|
force=True,
|
||||||
|
)
|
||||||
|
txid = out.stdout.strip().split("\n")[-1]
|
||||||
|
if len(txid) != 64:
|
||||||
|
raise Exception("Got no TXID after run the command")
|
||||||
|
if not transaction_accepted(main_chain, txid):
|
||||||
|
raise AssertionError(f"TX {txid} hasn't been processed")
|
||||||
|
time.sleep(parse_time(MAINNET_BLOCK_TIME))
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("NeoFS Deposit")
|
||||||
|
def deposit_gas(
|
||||||
|
shell: Shell,
|
||||||
|
main_chain: MainChain,
|
||||||
|
amount: int,
|
||||||
|
wallet_from_path: str,
|
||||||
|
wallet_from_password: str,
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Transferring GAS from given wallet to NeoFS contract address.
|
||||||
|
"""
|
||||||
|
# get NeoFS contract address
|
||||||
|
deposit_addr = contract_hash_to_address(NEOFS_CONTRACT)
|
||||||
|
logger.info(f"NeoFS contract address: {deposit_addr}")
|
||||||
|
address_from = get_last_address_from_wallet(
|
||||||
|
wallet_path=wallet_from_path, wallet_password=wallet_from_password
|
||||||
|
)
|
||||||
|
transfer_gas(
|
||||||
|
shell=shell,
|
||||||
|
main_chain=main_chain,
|
||||||
|
amount=amount,
|
||||||
|
wallet_from_path=wallet_from_path,
|
||||||
|
wallet_from_password=wallet_from_password,
|
||||||
|
address_to=deposit_addr,
|
||||||
|
address_from=address_from,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Mainnet Balance")
|
||||||
|
def get_mainnet_balance(main_chain: MainChain, address: str):
|
||||||
|
resp = main_chain.rpc_client.get_nep17_balances(address=address)
|
||||||
|
logger.info(f"Got getnep17balances response: {resp}")
|
||||||
|
for balance in resp["balance"]:
|
||||||
|
if balance["assethash"] == GAS_HASH:
|
||||||
|
return float(balance["amount"]) / ASSET_POWER_MAINCHAIN
|
||||||
|
return float(0)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Sidechain Balance")
|
||||||
|
def get_sidechain_balance(morph_chain: MorphChain, address: str):
|
||||||
|
resp = morph_chain.rpc_client.get_nep17_balances(address=address)
|
||||||
|
logger.info(f"Got getnep17balances response: {resp}")
|
||||||
|
for balance in resp["balance"]:
|
||||||
|
if balance["assethash"] == GAS_HASH:
|
||||||
|
return float(balance["amount"]) / ASSET_POWER_SIDECHAIN
|
||||||
|
return float(0)
|
249
robot/resources/lib/python_keywords/storage_group.py
Normal file
249
robot/resources/lib/python_keywords/storage_group.py
Normal file
|
@ -0,0 +1,249 @@
|
||||||
|
"""
|
||||||
|
This module contains keywords for work with Storage Groups.
|
||||||
|
It contains wrappers for `neofs-cli storagegroup` verbs.
|
||||||
|
"""
|
||||||
|
import logging
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import allure
|
||||||
|
from cluster import Cluster
|
||||||
|
from common import NEOFS_CLI_EXEC, WALLET_CONFIG
|
||||||
|
from complex_object_actions import get_link_object
|
||||||
|
from neofs_testlib.cli import NeofsCli
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
from neofs_verbs import head_object
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Put Storagegroup")
|
||||||
|
def put_storagegroup(
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
objects: list,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
lifetime: int = 10,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Wrapper for `neofs-cli storagegroup put`. Before the SG is created,
|
||||||
|
neofs-cli performs HEAD on `objects`, so this verb must be allowed
|
||||||
|
for `wallet` in `cid`.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
wallet: Path to wallet on whose behalf the SG is created.
|
||||||
|
cid: ID of Container to put SG to.
|
||||||
|
lifetime: Storage group lifetime in epochs.
|
||||||
|
objects: List of Object IDs to include into the SG.
|
||||||
|
bearer: Path to Bearer token file.
|
||||||
|
wallet_config: Path to neofs-cli config file.
|
||||||
|
Returns:
|
||||||
|
Object ID of created Storage Group.
|
||||||
|
"""
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=wallet_config)
|
||||||
|
result = neofscli.storagegroup.put(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
lifetime=lifetime,
|
||||||
|
members=objects,
|
||||||
|
bearer=bearer,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
)
|
||||||
|
gid = result.stdout.split("\n")[1].split(": ")[1]
|
||||||
|
return gid
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("List Storagegroup")
|
||||||
|
def list_storagegroup(
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
bearer: Optional[str] = None,
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
) -> list:
|
||||||
|
"""
|
||||||
|
Wrapper for `neofs-cli storagegroup list`. This operation
|
||||||
|
requires SEARCH allowed for `wallet` in `cid`.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
wallet: Path to wallet on whose behalf the SGs are listed in the container
|
||||||
|
cid: ID of Container to list.
|
||||||
|
bearer: Path to Bearer token file.
|
||||||
|
wallet_config: Path to neofs-cli config file.
|
||||||
|
Returns:
|
||||||
|
Object IDs of found Storage Groups.
|
||||||
|
"""
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=wallet_config)
|
||||||
|
result = neofscli.storagegroup.list(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
bearer=bearer,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
)
|
||||||
|
# throwing off the first string of output
|
||||||
|
found_objects = result.stdout.split("\n")[1:]
|
||||||
|
return found_objects
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Storagegroup")
|
||||||
|
def get_storagegroup(
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
gid: str,
|
||||||
|
bearer: str = "",
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
) -> dict:
|
||||||
|
"""
|
||||||
|
Wrapper for `neofs-cli storagegroup get`.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
wallet: Path to wallet on whose behalf the SG is got.
|
||||||
|
cid: ID of Container where SG is stored.
|
||||||
|
gid: ID of the Storage Group.
|
||||||
|
bearer: Path to Bearer token file.
|
||||||
|
wallet_config: Path to neofs-cli config file.
|
||||||
|
Returns:
|
||||||
|
Detailed information on the Storage Group.
|
||||||
|
"""
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=wallet_config)
|
||||||
|
result = neofscli.storagegroup.get(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
bearer=bearer,
|
||||||
|
id=gid,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
)
|
||||||
|
|
||||||
|
# TODO: temporary solution for parsing output. Needs to be replaced with
|
||||||
|
# JSON parsing when https://github.com/nspcc-dev/neofs-node/issues/1355
|
||||||
|
# is done.
|
||||||
|
strings = result.stdout.strip().split("\n")
|
||||||
|
# first three strings go to `data`;
|
||||||
|
# skip the 'Members:' string;
|
||||||
|
# the rest of strings go to `members`
|
||||||
|
data, members = strings[:3], strings[3:]
|
||||||
|
sg_dict = {}
|
||||||
|
for i in data:
|
||||||
|
key, val = i.split(": ")
|
||||||
|
sg_dict[key] = val
|
||||||
|
sg_dict["Members"] = []
|
||||||
|
for member in members[1:]:
|
||||||
|
sg_dict["Members"].append(member.strip())
|
||||||
|
return sg_dict
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Delete Storagegroup")
|
||||||
|
def delete_storagegroup(
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
gid: str,
|
||||||
|
bearer: str = "",
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
) -> str:
|
||||||
|
"""
|
||||||
|
Wrapper for `neofs-cli storagegroup delete`.
|
||||||
|
Args:
|
||||||
|
shell: Shell instance.
|
||||||
|
wallet: Path to wallet on whose behalf the SG is deleted.
|
||||||
|
cid: ID of Container where SG is stored.
|
||||||
|
gid: ID of the Storage Group.
|
||||||
|
bearer: Path to Bearer token file.
|
||||||
|
wallet_config: Path to neofs-cli config file.
|
||||||
|
Returns:
|
||||||
|
Tombstone ID of the deleted Storage Group.
|
||||||
|
"""
|
||||||
|
neofscli = NeofsCli(shell=shell, neofs_cli_exec_path=NEOFS_CLI_EXEC, config_file=wallet_config)
|
||||||
|
result = neofscli.storagegroup.delete(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
bearer=bearer,
|
||||||
|
id=gid,
|
||||||
|
rpc_endpoint=endpoint,
|
||||||
|
)
|
||||||
|
tombstone_id = result.stdout.strip().split("\n")[1].split(": ")[1]
|
||||||
|
return tombstone_id
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Verify list operation over Storagegroup")
|
||||||
|
def verify_list_storage_group(
|
||||||
|
shell: Shell,
|
||||||
|
endpoint: str,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
gid: str,
|
||||||
|
bearer: str = None,
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
):
|
||||||
|
storage_groups = list_storagegroup(
|
||||||
|
shell=shell,
|
||||||
|
endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
bearer=bearer,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
assert gid in storage_groups
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Verify get operation over Storagegroup")
|
||||||
|
def verify_get_storage_group(
|
||||||
|
shell: Shell,
|
||||||
|
cluster: Cluster,
|
||||||
|
wallet: str,
|
||||||
|
cid: str,
|
||||||
|
gid: str,
|
||||||
|
obj_list: list,
|
||||||
|
object_size: int,
|
||||||
|
max_object_size: int,
|
||||||
|
bearer: str = None,
|
||||||
|
wallet_config: str = WALLET_CONFIG,
|
||||||
|
):
|
||||||
|
obj_parts = []
|
||||||
|
endpoint = cluster.default_rpc_endpoint
|
||||||
|
if object_size > max_object_size:
|
||||||
|
for obj in obj_list:
|
||||||
|
link_oid = get_link_object(
|
||||||
|
wallet,
|
||||||
|
cid,
|
||||||
|
obj,
|
||||||
|
shell=shell,
|
||||||
|
nodes=cluster.storage_nodes,
|
||||||
|
bearer=bearer,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
obj_head = head_object(
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
oid=link_oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=endpoint,
|
||||||
|
is_raw=True,
|
||||||
|
bearer=bearer,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
obj_parts = obj_head["header"]["split"]["children"]
|
||||||
|
|
||||||
|
obj_num = len(obj_list)
|
||||||
|
storagegroup_data = get_storagegroup(
|
||||||
|
shell=shell,
|
||||||
|
endpoint=endpoint,
|
||||||
|
wallet=wallet,
|
||||||
|
cid=cid,
|
||||||
|
gid=gid,
|
||||||
|
bearer=bearer,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
exp_size = object_size * obj_num
|
||||||
|
if object_size < max_object_size:
|
||||||
|
assert int(storagegroup_data["Group size"]) == exp_size
|
||||||
|
assert storagegroup_data["Members"] == obj_list
|
||||||
|
else:
|
||||||
|
assert int(storagegroup_data["Group size"]) == exp_size
|
||||||
|
assert storagegroup_data["Members"] == obj_parts
|
173
robot/resources/lib/python_keywords/storage_policy.py
Normal file
173
robot/resources/lib/python_keywords/storage_policy.py
Normal file
|
@ -0,0 +1,173 @@
|
||||||
|
#!/usr/bin/python3
|
||||||
|
|
||||||
|
"""
|
||||||
|
This module contains keywords which are used for asserting
|
||||||
|
that storage policies are respected.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
import allure
|
||||||
|
import complex_object_actions
|
||||||
|
import neofs_verbs
|
||||||
|
from cluster import StorageNode
|
||||||
|
from grpc_responses import OBJECT_NOT_FOUND, error_matches_status
|
||||||
|
from neofs_testlib.shell import Shell
|
||||||
|
|
||||||
|
logger = logging.getLogger("NeoLogger")
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Object Copies")
|
||||||
|
def get_object_copies(
|
||||||
|
complexity: str, wallet: str, cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> int:
|
||||||
|
"""
|
||||||
|
The function performs requests to all nodes of the container and
|
||||||
|
finds out if they store a copy of the object. The procedure is
|
||||||
|
different for simple and complex object, so the function requires
|
||||||
|
a sign of object complexity.
|
||||||
|
Args:
|
||||||
|
complexity (str): the tag of object size and complexity,
|
||||||
|
[Simple|Complex]
|
||||||
|
wallet (str): the path to the wallet on whose behalf the
|
||||||
|
copies are got
|
||||||
|
cid (str): ID of the container
|
||||||
|
oid (str): ID of the Object
|
||||||
|
shell: executor for cli command
|
||||||
|
Returns:
|
||||||
|
(int): the number of object copies in the container
|
||||||
|
"""
|
||||||
|
return (
|
||||||
|
get_simple_object_copies(wallet, cid, oid, shell, nodes)
|
||||||
|
if complexity == "Simple"
|
||||||
|
else get_complex_object_copies(wallet, cid, oid, shell, nodes)
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Simple Object Copies")
|
||||||
|
def get_simple_object_copies(
|
||||||
|
wallet: str, cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> int:
|
||||||
|
"""
|
||||||
|
To figure out the number of a simple object copies, only direct
|
||||||
|
HEAD requests should be made to the every node of the container.
|
||||||
|
We consider non-empty HEAD response as a stored object copy.
|
||||||
|
Args:
|
||||||
|
wallet (str): the path to the wallet on whose behalf the
|
||||||
|
copies are got
|
||||||
|
cid (str): ID of the container
|
||||||
|
oid (str): ID of the Object
|
||||||
|
shell: executor for cli command
|
||||||
|
nodes: nodes to search on
|
||||||
|
Returns:
|
||||||
|
(int): the number of object copies in the container
|
||||||
|
"""
|
||||||
|
copies = 0
|
||||||
|
for node in nodes:
|
||||||
|
try:
|
||||||
|
response = neofs_verbs.head_object(
|
||||||
|
wallet, cid, oid, shell=shell, endpoint=node.get_rpc_endpoint(), is_direct=True
|
||||||
|
)
|
||||||
|
if response:
|
||||||
|
logger.info(f"Found object {oid} on node {node}")
|
||||||
|
copies += 1
|
||||||
|
except Exception:
|
||||||
|
logger.info(f"No {oid} object copy found on {node}, continue")
|
||||||
|
continue
|
||||||
|
return copies
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Complex Object Copies")
|
||||||
|
def get_complex_object_copies(
|
||||||
|
wallet: str, cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> int:
|
||||||
|
"""
|
||||||
|
To figure out the number of a complex object copies, we firstly
|
||||||
|
need to retrieve its Last object. We consider that the number of
|
||||||
|
complex object copies is equal to the number of its last object
|
||||||
|
copies. When we have the Last object ID, the task is reduced
|
||||||
|
to getting simple object copies.
|
||||||
|
Args:
|
||||||
|
wallet (str): the path to the wallet on whose behalf the
|
||||||
|
copies are got
|
||||||
|
cid (str): ID of the container
|
||||||
|
oid (str): ID of the Object
|
||||||
|
shell: executor for cli command
|
||||||
|
Returns:
|
||||||
|
(int): the number of object copies in the container
|
||||||
|
"""
|
||||||
|
last_oid = complex_object_actions.get_last_object(wallet, cid, oid, shell, nodes)
|
||||||
|
assert last_oid, f"No Last Object for {cid}/{oid} found among all Storage Nodes"
|
||||||
|
return get_simple_object_copies(wallet, cid, last_oid, shell, nodes)
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Nodes With Object")
|
||||||
|
def get_nodes_with_object(
|
||||||
|
cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> list[StorageNode]:
|
||||||
|
"""
|
||||||
|
The function returns list of nodes which store
|
||||||
|
the given object.
|
||||||
|
Args:
|
||||||
|
cid (str): ID of the container which store the object
|
||||||
|
oid (str): object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
nodes: nodes to find on
|
||||||
|
Returns:
|
||||||
|
(list): nodes which store the object
|
||||||
|
"""
|
||||||
|
|
||||||
|
nodes_list = []
|
||||||
|
for node in nodes:
|
||||||
|
wallet = node.get_wallet_path()
|
||||||
|
wallet_config = node.get_wallet_config_path()
|
||||||
|
try:
|
||||||
|
res = neofs_verbs.head_object(
|
||||||
|
wallet,
|
||||||
|
cid,
|
||||||
|
oid,
|
||||||
|
shell=shell,
|
||||||
|
endpoint=node.get_rpc_endpoint(),
|
||||||
|
is_direct=True,
|
||||||
|
wallet_config=wallet_config,
|
||||||
|
)
|
||||||
|
if res is not None:
|
||||||
|
logger.info(f"Found object {oid} on node {node}")
|
||||||
|
nodes_list.append(node)
|
||||||
|
except Exception:
|
||||||
|
logger.info(f"No {oid} object copy found on {node}, continue")
|
||||||
|
continue
|
||||||
|
return nodes_list
|
||||||
|
|
||||||
|
|
||||||
|
@allure.step("Get Nodes Without Object")
|
||||||
|
def get_nodes_without_object(
|
||||||
|
wallet: str, cid: str, oid: str, shell: Shell, nodes: list[StorageNode]
|
||||||
|
) -> list[StorageNode]:
|
||||||
|
"""
|
||||||
|
The function returns list of nodes which do not store
|
||||||
|
the given object.
|
||||||
|
Args:
|
||||||
|
wallet (str): the path to the wallet on whose behalf
|
||||||
|
we request the nodes
|
||||||
|
cid (str): ID of the container which store the object
|
||||||
|
oid (str): object ID
|
||||||
|
shell: executor for cli command
|
||||||
|
Returns:
|
||||||
|
(list): nodes which do not store the object
|
||||||
|
"""
|
||||||
|
nodes_list = []
|
||||||
|
for node in nodes:
|
||||||
|
try:
|
||||||
|
res = neofs_verbs.head_object(
|
||||||
|
wallet, cid, oid, shell=shell, endpoint=node.get_rpc_endpoint(), is_direct=True
|
||||||
|
)
|
||||||
|
if res is None:
|
||||||
|
nodes_list.append(node)
|
||||||
|
except Exception as err:
|
||||||
|
if error_matches_status(err, OBJECT_NOT_FOUND):
|
||||||
|
nodes_list.append(node)
|
||||||
|
else:
|
||||||
|
raise Exception(f"Got error {err} on head object command") from err
|
||||||
|
return nodes_list
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Reference in a new issue