Skip to content

Commit

Permalink
Verify python source files with flake8.
Browse files Browse the repository at this point in the history
Fixes #994

Signed-off-by: Gil Bregman <[email protected]>
  • Loading branch information
gbregman committed Dec 31, 2024
1 parent 0842c8f commit 0562359
Show file tree
Hide file tree
Showing 28 changed files with 3,801 additions and 2,270 deletions.
6 changes: 6 additions & 0 deletions .github/workflows/build-container.yml
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,12 @@ jobs:
with:
submodules: recursive

- name: Install Flake8
run: pip install flake8

- name: Verify Python source files
run: make verify

- name: Build container images - spdk
run: make build SVC="spdk" SPDK_TARGET_ARCH=x86-64-v2

Expand Down
4 changes: 4 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,10 @@ include mk/autohelp.mk
.DEFAULT_GOAL := all
all: setup $(ALL)

verify: ## Run Python source files through flake8
@echo Verifying Python source files
flake8 control/*.py tests/*.py

setup: ## Configure huge-pages (requires sudo/root password)

@echo Setup core dump pattern as /tmp/coredump/core.*
Expand Down
7 changes: 7 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,12 @@ sudo groupadd docker
sudo usermod -aG docker $USER
```

In order to use the "make verify" option to validate the Python source files you need to have flake8 installed on the build machine:

```bash
pip install flake8
```

### Steps

To launch a containerized environment with a Ceph cluster and a NVMe-oF gateway (this is not the [prescribed deployment for production purposes](https://docs.ceph.com/en/quincy/install/#recommended-methods), but for testing and development tasks alone):
Expand Down Expand Up @@ -473,6 +479,7 @@ Targets:
Miscellaneous:
alias Print bash alias command for the nvmeof-cli. Usage: "eval $(make alias)"
verify Run flake8 on the Python source files
```

Targets may accept options: `make run SVC=nvme OPTS=--entrypoint=bash`.
Expand Down
79 changes: 44 additions & 35 deletions control/cephutils.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
import json
from .utils import GatewayLogger


class CephUtils:
"""Miscellaneous functions which connect to Ceph
"""
Expand All @@ -27,11 +28,12 @@ def __init__(self, config):
self.last_sent = time.time()

def execute_ceph_monitor_command(self, cmd):
self.logger.debug(f"Execute monitor command: {cmd}")
with rados.Rados(conffile=self.ceph_conf, rados_id=self.rados_id) as cluster:
self.logger.debug(f"Execute monitor command: {cmd}")
with rados.Rados(conffile=self.ceph_conf, rados_id=self.rados_id) as cluster:
rply = cluster.mon_command(cmd, b'')
self.logger.debug(f"Monitor reply: {rply}")
return rply

def get_gw_id_owner_ana_group(self, pool, group, anagrp):
str = '{' + f'"prefix":"nvme-gw show", "pool":"{pool}", "group":"{group}"' + '}'
self.logger.debug(f"nvme-show string: {str}")
Expand All @@ -45,23 +47,23 @@ def get_gw_id_owner_ana_group(self, pool, group, anagrp):
comp_str = f"{anagrp}: ACTIVE"
for gateway in data["Created Gateways:"]:
if comp_str in gateway["ana states"]:
gw_id = gateway["gw-id"]
self.logger.debug(f"found gw owner of anagrp {anagrp}: gw {gw_id}")
break
gw_id = gateway["gw-id"]
self.logger.debug(f"found gw owner of anagrp {anagrp}: gw {gw_id}")
break
return gw_id

def is_rebalance_supported(self):
return self.rebalance_supported
return self.rebalance_supported

def get_rebalance_ana_group(self):
return self.rebalance_ana_group
return self.rebalance_ana_group

def get_number_created_gateways(self, pool, group):
now = time.time()
if (now - self.last_sent) < 10 and self.anagroup_list :
self.logger.info(f"Caching response of the monitor: {self.anagroup_list}")
return self.anagroup_list
else :
if (now - self.last_sent) < 10 and self.anagroup_list:
self.logger.info(f"Caching response of the monitor: {self.anagroup_list}")
return self.anagroup_list
else:
try:
self.anagroup_list = []
self.last_sent = now
Expand All @@ -76,12 +78,12 @@ def get_number_created_gateways(self, pool, group):
self.rebalance_supported = True
self.rebalance_ana_group = data.get("rebalance_ana_group", None)
self.logger.debug(f"Rebalance ana_group: {self.rebalance_ana_group}")
else :
else:
self.rebalance_supported = False
pos = conv_str.find("[")
if pos != -1:
new_str = conv_str[pos + len("[") :]
pos = new_str.find("]")
new_str = conv_str[pos + len("["):]
pos = new_str.find("]")
new_str = new_str[: pos].strip()
int_str_list = new_str.split(' ')
self.logger.debug(f"new_str : {new_str}")
Expand All @@ -92,19 +94,20 @@ def get_number_created_gateways(self, pool, group):
self.logger.warning("GWs not found")

except Exception:
self.logger.exception(f"Failure get number created gateways:")
self.logger.exception("Failure get number created gateways")
self.anagroup_list = []

return self.anagroup_list

def fetch_and_display_ceph_version(self):
try:
rply = self.execute_ceph_monitor_command('{"prefix":"mon versions"}')
ceph_ver = rply[1].decode().removeprefix("{").strip().split(":")[0].removeprefix('"').removesuffix('"')
ceph_ver = rply[1].decode().removeprefix("{").strip().split(":")[0]
ceph_ver = ceph_ver.removeprefix('"').removesuffix('"')
ceph_ver = ceph_ver.removeprefix("ceph version ")
self.logger.info(f"Connected to Ceph with version \"{ceph_ver}\"")
except Exception:
self.logger.exception(f"Failure fetching Ceph version:")
self.logger.exception("Failure fetching Ceph version")
pass

def fetch_ceph_fsid(self) -> str:
Expand All @@ -113,7 +116,7 @@ def fetch_ceph_fsid(self) -> str:
with rados.Rados(conffile=self.ceph_conf, rados_id=self.rados_id) as cluster:
fsid = cluster.get_fsid()
except Exception:
self.logger.exception(f"Failure fetching Ceph fsid:")
self.logger.exception("Failure fetching Ceph fsid")

return fsid

Expand All @@ -130,69 +133,75 @@ def pool_exists(self, pool) -> bool:

def service_daemon_register(self, cluster, metadata):
try:
if cluster: # rados client
if cluster: # rados client
daemon_name = metadata['id']
cluster.service_daemon_register("nvmeof", daemon_name, metadata)
self.logger.info(f"Registered {daemon_name} to service_map!")
except Exception:
self.logger.exception(f"Can't register daemon to service_map!")
self.logger.exception("Can't register daemon to service_map!")

def service_daemon_update(self, cluster, status_buffer):
try:
if cluster and status_buffer:
cluster.service_daemon_update(status_buffer)
except Exception:
self.logger.exception(f"Can't update daemon status to service_map!")
self.logger.exception("Can't update daemon status to service_map!")

def create_image(self, pool_name, image_name, size) -> bool:
# Check for pool existence in advance as we don't create it if it's not there
if not self.pool_exists(pool_name):
raise rbd.ImageNotFound(f"Pool {pool_name} doesn't exist", errno = errno.ENODEV)
raise rbd.ImageNotFound(f"Pool {pool_name} doesn't exist", errno=errno.ENODEV)

image_exists = False
try:
image_size = self.get_image_size(pool_name, image_name)
image_exists = True
except rbd.ImageNotFound:
self.logger.debug(f"Image {pool_name}/{image_name} doesn't exist, will create it using size {size}")
self.logger.debug(f"Image {pool_name}/{image_name} doesn't exist, will "
f"create it using size {size}")
pass

if image_exists:
if image_size != size:
raise rbd.ImageExists(f"Image {pool_name}/{image_name} already exists with a size of {image_size} bytes which differs from the requested size of {size} bytes",
errno = errno.EEXIST)
raise rbd.ImageExists(f"Image {pool_name}/{image_name} already exists with "
f"a size of {image_size} bytes which differs from the "
f"requested size of {size} bytes",
errno=errno.EEXIST)
return False # Image exists with an idetical size, there is nothing to do here

with rados.Rados(conffile=self.ceph_conf, rados_id=self.rados_id) as cluster:
with cluster.open_ioctx(pool_name) as ioctx:
rbd_inst = rbd.RBD()
try:
rbd_inst.create(ioctx, image_name, size)
except rbd.ImageExists as ex:
except rbd.ImageExists:
self.logger.exception(f"Image {pool_name}/{image_name} was created just now")
raise rbd.ImageExists(f"Image {pool_name}/{image_name} was just created by someone else, please retry",
errno = errno.EAGAIN)
except Exception as ex:
raise rbd.ImageExists(f"Image {pool_name}/{image_name} was just created by "
f"someone else, please retry",
errno=errno.EAGAIN)
except Exception:
self.logger.exception(f"Can't create image {pool_name}/{image_name}")
raise ex
raise

return True

def get_image_size(self, pool_name, image_name) -> int:
image_size = 0
if not self.pool_exists(pool_name):
raise rbd.ImageNotFound(f"Pool {pool_name} doesn't exist", errno = errno.ENODEV)
raise rbd.ImageNotFound(f"Pool {pool_name} doesn't exist", errno=errno.ENODEV)

with rados.Rados(conffile=self.ceph_conf, rados_id=self.rados_id) as cluster:
with cluster.open_ioctx(pool_name) as ioctx:
rbd_inst = rbd.RBD()
rbd.RBD()
try:
with rbd.Image(ioctx, image_name) as img:
image_size = img.size()
except rbd.ImageNotFound:
raise rbd.ImageNotFound(f"Image {pool_name}/{image_name} doesn't exist", errno = errno.ENODEV)
raise rbd.ImageNotFound(f"Image {pool_name}/{image_name} doesn't exist",
errno=errno.ENODEV)
except Exception as ex:
self.logger.exception(f"Error while trying to get the size of image {pool_name}/{image_name}")
self.logger.exception(f"Error while trying to get the size of image "
f"{pool_name}/{image_name}")
raise ex

return image_size
Expand All @@ -205,6 +214,6 @@ def get_rbd_exception_details(self, ex):
if msg.startswith("["):
pos = msg.find("]")
if pos >= 0:
msg = msg[pos + 1 :].strip()
msg = msg[pos + 1:].strip()
ex_details = (ex.errno, msg)
return ex_details
Loading

0 comments on commit 0562359

Please sign in to comment.