diff options
-rw-r--r-- | glustolibs-gluster/glustolibs/gluster/lib_utils.py | 19 | ||||
-rw-r--r-- | tests/functional/disperse/test_ec_brick_consumable_size.py | 113 |
2 files changed, 132 insertions, 0 deletions
diff --git a/glustolibs-gluster/glustolibs/gluster/lib_utils.py b/glustolibs-gluster/glustolibs/gluster/lib_utils.py index 0b86291b0..e4459eec2 100644 --- a/glustolibs-gluster/glustolibs/gluster/lib_utils.py +++ b/glustolibs-gluster/glustolibs/gluster/lib_utils.py @@ -921,3 +921,22 @@ def add_services_to_firewall(nodes, firewall_service, permanent=False): _rc = False return _rc + + +def get_size_of_mountpoint(node, mount_point): + """ + get_size_of_mountpoint: + Returns the size in blocks for the mount point + + Args: + node - node on which path is mounted + mount_point - mount point path + + Returns: + Size of the mount point in blocks or none. + """ + + cmd = "df %s | grep -v '^Filesystem' | awk '{print $4}'" % (mount_point) + _, out, _ = g.run(node, cmd) + + return out diff --git a/tests/functional/disperse/test_ec_brick_consumable_size.py b/tests/functional/disperse/test_ec_brick_consumable_size.py new file mode 100644 index 000000000..e2cee80b1 --- /dev/null +++ b/tests/functional/disperse/test_ec_brick_consumable_size.py @@ -0,0 +1,113 @@ +# Copyright (C) 2018 Red Hat, Inc. <http://www.redhat.com> +# +# This program is free software; you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation; either version 2 of the License, or +# any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License along +# with this program; if not, write to the Free Software Foundation, Inc., +# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. + +""" +EcBrickConsumableSize: + + This test verifies that the size of the volume will be + 'number of data bricks * least of brick size'. + +""" +from glusto.core import Glusto as g +from glustolibs.gluster.brick_libs import get_all_bricks +from glustolibs.gluster.exceptions import ExecutionError +from glustolibs.gluster.volume_libs import (get_volume_info) +from glustolibs.gluster.lib_utils import get_size_of_mountpoint +from glustolibs.gluster.gluster_base_class import (GlusterBaseClass, runs_on) + + +@runs_on([['dispersed', 'distributed-dispersed'], ['glusterfs']]) +class EcBrickConsumableSize(GlusterBaseClass): + + # Method to setup the environment for test case + def setUp(self): + # Setup Volume and Mount Volume + g.log.info("Starting to Setup Volume and Mount Volume") + ret = self.setup_volume_and_mount_volume(mounts=self.mounts) + if not ret: + raise ExecutionError("Failed to Setup_Volume and Mount_Volume") + g.log.info("Successful in Setup Volume and Mount Volume") + + # Test Case + def test_disperse_vol_size(self): + # pylint: disable=too-many-locals + mnode = self.mnode + volname = self.volname + client = self.mounts[0].client_system + mountpoint = self.mounts[0].mountpoint + + # Obtain the volume size + vol_size = get_size_of_mountpoint(client, mountpoint) + self.assertIsNotNone(vol_size, ("Unable to get the volsize \ + of %s.", volname)) + + # Retrieve the minimum brick size + min_brick_size = -1 + bricks_list = get_all_bricks(mnode, volname) + for brick in bricks_list: + brick_node, brick_path = brick.split(":") + brick_size = get_size_of_mountpoint(brick_node, brick_path) + if ((brick_size is not None) and (min_brick_size == -1) or + (int(min_brick_size) > int(brick_size))): + min_brick_size = brick_size + + # Calculate the consumable size + vol_info = get_volume_info(mnode, volname) + self.assertIsNotNone(vol_info, ("Unable to get the volinfo \ + of %s.", volname)) + + disp_data_bricks = (int(vol_info[volname]['disperseCount']) - + int(vol_info[volname]['redundancyCount'])) + dist_count = (int(vol_info[volname]['brickCount']) / + int(vol_info[volname]['disperseCount'])) + consumable_size = ((int(min_brick_size) * int(disp_data_bricks)) * + int(dist_count)) + + # Verify the volume size is in allowable range + # Volume size should be above 98% of consumable size. + delta = (100 - ((float(vol_size)/float(consumable_size)) * 100)) + self.assertTrue(delta < 2, ("Volume size is not in allowable range")) + + g.log.info("Volume size is in allowable range") + + # Write to the available size + block_size = 1024 + write_size = ((int(vol_size) * (0.95) * int(block_size)) / + (int(dist_count))) + for i in range(1, int(dist_count)): + ret, _, _ = g.run(client, "fallocate -l {} {}/testfile{} \ + ".format(int(write_size), mountpoint, i)) + self.assertTrue(ret == 0, ("Writing file of available size failed \ + on volume %s", volname)) + g.log.info("Successfully verified volume size") + + # Try writing more than the available size + write_size = ((int(vol_size) * int(block_size)) * 1.2) + ret, _, _ = g.run(client, "fallocate -l {} {}/testfile1 \ + ".format(int(write_size), mountpoint)) + self.assertTrue(ret != 0, ("Writing file of more than available \ + size passed on volume %s", volname)) + + g.log.info("Successfully verified brick consumable size") + + # Method to cleanup test setup + def tearDown(self): + # Stopping the volume + g.log.info("Starting to Unmount Volume and Cleanup Volume") + ret = self.unmount_volume_and_cleanup_volume(mounts=self.mounts) + if not ret: + raise ExecutionError("Failed to Unmount Volume and Cleanup Volume") + g.log.info("Successful in Unmount Volume and Cleanup Volume") |