Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

cbt: creating a common base class for fio subclasses #318

Open
wants to merge 6 commits into
base: master
Choose a base branch
from

Conversation

harriscr
Copy link
Contributor

This is the first part of an effort to rationalise the 5 current fio classes into a single class structure. It will mean the users of cbt no longer need to know the subtle differences between the differing fio classes.
The change adds typing for methods and variables (using mypy) and pep8 compliance (using ruff) to the fio* files changed.

This is my first pull request so I expect there is something I've missed in the process.

This has been tested by:

  1. running the unit tests in the test sub directory
  2. Validating that the client_endpoint_rbd_4K_rand_write.yaml configuration file used in teuthology results in the same generated fio cli with and without these changes
  3. Validating that the example yaml for fio (example-client_endpoints.yaml) generated the same cli to run fio with and without the changes

@perezjosibm
Copy link
Contributor

Adding @sseshasa as well since the refactoring changes seem important

@perezjosibm
Copy link
Contributor

@harriscr can you kindly post the snapshots showing the tests that you run? Also, it would be very useful the class diagrams that you got from pyreverse (new vs old).

I have some problems to understand what you mean by:

It will mean the users of cbt no longer need to know the subtle differences between the differing fio classes.

I'm guessing the main difference between each FIO subclass instance would be the underlying FIO engine (eg.libaio, librd, etc.) Or you mean "block" vs 'file" IO? Could you illustrate with an example please?

Thanks

Copy link
Contributor

@perezjosibm perezjosibm left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've added comments to the points I consider might need amending/clarification. It looks good to me so far, but will be useful to have further insight from Mark and Seshasayee.
Thanks

'{:0>8}'.format(config.get('iteration')),
'id-{}'.format(digest))
self.archive_dir = os.path.join(
archive_dir, "results", "{:0>8}".format(config.get("iteration")), "id-{}".format(digest)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wonder whether a f-string would be better for the future instead of the old .format() style?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree. I plan to look at benchmark.py in the future, so will look at changing this line to an f-string when I do.

self.log_lat = config.get('log_lat', True)
logger.info("Results dir: %s", self.archive_dir)
self.run_dir = os.path.join(
settings.cluster.get("tmp_dir"), "{:0>8}".format(config.get("iteration")), self.getclass()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like the convention of splitting lines on '{', '(', can you share the config file to instruct the formatter? If that can be part of the commit, much better 👍

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am working on a pyproject.toml file which will contain the formatting instructions, as well as linting and typing. I would like to deliver under a separate PR however.

@@ -130,27 +135,27 @@ def prefill(self):

def run(self):
if self.osd_ra and self.osd_ra_changed:
logger.info('Setting OSD Read Ahead to: %s', self.osd_ra)
self.cluster.set_osd_param('read_ahead_kb', self.osd_ra)
logger.info("Setting OSD Read Ahead to: %s", self.osd_ra)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seems to me that OSD read ahead is very Ceph specific (rather than a Benchmark attribute), so we might need to consider whether is convenient to abstract it out into the cluster class instead.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like that idea.
Currently osd_ra is an attribute of the benchmark in the yaml file, so it would be good to change it to a cluster attribute in the yaml at the same time as moving it to an attribute of the Ceph object

if self.valgrind is not None:
logger.debug('Adding valgrind to the command path.')
logger.debug("Adding valgrind to the command path.")
self.cmd_path_full = common.setup_valgrind(self.valgrind, self.getclass(), self.run_dir)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Wonder if setting up valgrid as a profile_tool instead would make it better? @markhpc what are your thoughts please?

yaml.dump(config_dict, fd, default_flow_style=False)

def exists(self):
def exists(self) -> bool:
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I guess the type annotation is going to be a continuous effort since not all the methods have been annotated

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Correct. I will type the methods as I need to edit/change them.
This one was changed so the signature matches in the child class and the parent class, as I was changing the child class.

return command

def _build_prefill_command(self, endpoint_number: int) -> str:
command = f"sudo {self.cmd_path} --rw=write --bs=4M --iodepth={self._configuration_with_defaults.get('prefill_iodepth')}"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here is assuming that prefill can only be done using block size of 4MB, which I don't object per se, but wondering whether that could be the default value but to allow the flexibility of different block sizes?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The idea of this particular PR is not to change the current behaviour. It is something to consider for the future however

Comment on lines -220 to +136
if 'recovery_test' in self.cluster.config:
if self.recov_test_type == 'blocking':
if "recovery_test" in self._cluster.config:
if self._recovery_test_type == "blocking":
recovery_callback = self.recovery_callback_blocking
elif self.recov_test_type == 'background':
elif self._recovery_test_type == "background":
recovery_callback = self.recovery_callback_background
self.cluster.create_recovery_test(self.run_dir, recovery_callback, self.recov_test_type)
self._cluster.create_recovery_test(self.run_dir, recovery_callback, self._recovery_test_type) # type: ignore [no-untyped-call]

if 'recovery_test' in self.cluster.config and self.recov_test_type == 'background':
if "recovery_test" in self._cluster.config and self._recovery_test_type == "background":
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Probably this block of code requires its own method

ps.append(p)
for p in ps:
p.wait()
log.info("Running fio %s test.", self._cli_options["rw"])
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the original 'self.mode' was more generic to refer to the IO mode (random read, random write, etc), whereas the suggested '._cli_options["rw"]' is too specific to FIO, so not sure if that conveys the intention.
If in the near future we need to compare two different benchmarks over the same workload, we need that the options generic enough to compare apples with apples, so my vote would be towards a generic IO mode rather than specific FIO "rw".
@markhpc , @sseshasa what are your thoughts please?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This matches the existing behaviour. self.mode is taken from the options in the yaml file, and is used for the -rw parameter to fio. The original code is:

self.mode = config.get('mode', 'write')

and the new code is:

self._cli_options["rw"] = self._configuration_with_defaults.get("mode", "write")

I'm happy to change the code to take the value from the configuration instead if you think that would be a better approach, but it doesn't change what is printed to the screen, or what the code does

common.sync_files('%s/*' % self.run_dir, self.out_dir)
self.analyze(self.out_dir)

def cleanup(self):
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

|I think its a good opportunity to add method documentation

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can't believe I missed this. Will add in.
I'm surprised that the linter didn't pick this up either so may take a look at my settings

host = settings.host_info(client)["host"]
for i in range(self.endpoints_per_client):
def analyze(self, output_directory: str) -> None:
log.info("Converting results to json format.")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Method documentation?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will add

Chris Harris added 2 commits October 18, 2024 17:24
Reverting the change to fio.py as it was made in error

This reverts commit 3c75e3d.

Signed-off-by: Chris Harris <[email protected]>
@harriscr harriscr requested a review from perezjosibm October 21, 2024 09:53
Chris Harris added 3 commits October 21, 2024 11:21
Reverting the change to fio.py as it was made in error

This reverts commit 3c75e3d.

Signed-off-by: Chris Harris <[email protected]>
Reverting the change to fio.py as it was made in error

This reverts commit 3c75e3d.

Signed-off-by: Chris Harris <[email protected]>
Reverting the change to fio.py as it was made in error

This reverts commit 3c75e3d.

Signed-off-by: Chris Harris <[email protected]>
@harriscr
Copy link
Contributor Author

harriscr commented Nov 7, 2024

@harriscr
Copy link
Contributor Author

harriscr commented Nov 7, 2024

Here are the pyreverse diagrams:
Classes before the change:
classes_before
Classes after the change:
classes_after
Packages before the change
packages_before
packages after the change:
packages_after

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants