Skip to content

Commit

Permalink
fixes #63098 add funcs to return block device, mount point, and files…
Browse files Browse the repository at this point in the history
…ystem type for a path
  • Loading branch information
nicholasmhughes authored and Megan Wilhite committed Nov 29, 2022
1 parent 841584c commit f4f4416
Show file tree
Hide file tree
Showing 6 changed files with 412 additions and 313 deletions.
1 change: 1 addition & 0 deletions changelog/63098.added
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Add functions that will return the underlying block device, mount point, and filesystem type for a given path
19 changes: 19 additions & 0 deletions salt/modules/disk.py
Original file line number Diff line number Diff line change
Expand Up @@ -1015,3 +1015,22 @@ def _iostat_aix(interval, count, disks):
iostats[disk][disk_mode] = _iostats_dict(fields, stats)

return iostats


def get_fstype_from_path(path):
"""
Return the filesystem type of the underlying device for a specified path.
.. versionadded:: 3006.0
path
The path for the function to evaluate.
CLI Example:
.. code-block:: bash
salt '*' disk.get_fstype_from_path /root
"""
dev = __salt__["mount.get_device_from_path"](path)
return fstype(dev)
43 changes: 43 additions & 0 deletions salt/modules/mount.py
Original file line number Diff line number Diff line change
Expand Up @@ -1976,3 +1976,46 @@ def rm_filesystems(name, device, config="/etc/filesystems"):
raise CommandExecutionError("rm_filesystems error exception {exc}")

return modified


def get_mount_from_path(path):
"""
Return the mount providing a specified path.
.. versionadded:: 3006.0
path
The path for the function to evaluate.
CLI Example:
.. code-block:: bash
salt '*' mount.get_mount_from_path /opt/some/nested/path
"""
path = os.path.realpath(os.path.abspath(path))
while path != os.path.sep:
if os.path.ismount(path):
return path
path = os.path.abspath(os.path.join(path, os.pardir))
return path


def get_device_from_path(path):
"""
Return the underlying device for a specified path.
.. versionadded:: 3006.0
path
The path for the function to evaluate.
CLI Example:
.. code-block:: bash
salt '*' mount.get_device_from_path /
"""
mount = get_mount_from_path(path)
mounts = active()
return mounts.get(mount, {}).get("device")
322 changes: 322 additions & 0 deletions tests/pytests/unit/modules/test_disk.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,322 @@
"""
:codeauthor: Jayesh Kariya <[email protected]>
"""
import sys

import pytest

import salt.modules.disk as disk
import salt.utils.path
import salt.utils.platform
from tests.support.mock import MagicMock, patch
from tests.support.unit import skipIf

STUB_DISK_USAGE = {
"/": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/dev": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/run": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/run/lock": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/run/shm": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/run/user": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
"/sys/fs/cgroup": {
"filesystem": None,
"1K-blocks": 10000,
"used": 10000,
"available": 10000,
"capacity": 10000,
},
}


STUB_DISK_INODEUSAGE = {
"/": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/dev": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/run": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/run/lock": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/run/shm": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/run/user": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
"/sys/fs/cgroup": {
"inodes": 10000,
"used": 10000,
"free": 10000,
"use": 10000,
"filesystem": None,
},
}

STUB_DISK_PERCENT = {
"/": 50,
"/dev": 10,
"/run": 10,
"/run/lock": 10,
"/run/shm": 10,
"/run/user": 10,
"/sys/fs/cgroup": 10,
}

STUB_DISK_BLKID = {"/dev/sda": {"TYPE": "ext4", "UUID": None}}


@pytest.fixture
def configure_loader_modules():
return {disk: {}}


def test_usage_dict():
with patch.dict(disk.__grains__, {"kernel": "Linux"}), patch(
"salt.modules.disk.usage", MagicMock(return_value=STUB_DISK_USAGE)
):
mock_cmd = MagicMock(return_value=1)
with patch.dict(disk.__salt__, {"cmd.run": mock_cmd}):
assert STUB_DISK_USAGE == disk.usage(args=None)


def test_usage_none():
with patch.dict(disk.__grains__, {"kernel": "Linux"}), patch(
"salt.modules.disk.usage", MagicMock(return_value="")
):
mock_cmd = MagicMock(return_value=1)
with patch.dict(disk.__salt__, {"cmd.run": mock_cmd}):
assert "" == disk.usage(args=None)


def test_inodeusage():
with patch.dict(disk.__grains__, {"kernel": "OpenBSD"}), patch(
"salt.modules.disk.inodeusage", MagicMock(return_value=STUB_DISK_INODEUSAGE)
):
mock = MagicMock()
with patch.dict(disk.__salt__, {"cmd.run": mock}):
assert STUB_DISK_INODEUSAGE == disk.inodeusage(args=None)


def test_percent():
with patch.dict(disk.__grains__, {"kernel": "Linux"}), patch(
"salt.modules.disk.percent", MagicMock(return_value=STUB_DISK_PERCENT)
):
mock = MagicMock()
with patch.dict(disk.__salt__, {"cmd.run": mock}):
assert STUB_DISK_PERCENT == disk.percent(args=None)


def test_percent_args():
with patch.dict(disk.__grains__, {"kernel": "Linux"}), patch(
"salt.modules.disk.percent", MagicMock(return_value="/")
):
mock = MagicMock()
with patch.dict(disk.__salt__, {"cmd.run": mock}):
assert "/" == disk.percent("/")


def test_blkid():
with patch.dict(
disk.__salt__, {"cmd.run_stdout": MagicMock(return_value=1)}
), patch("salt.modules.disk.blkid", MagicMock(return_value=STUB_DISK_BLKID)):
assert STUB_DISK_BLKID == disk.blkid()


@skipIf(salt.utils.platform.is_windows(), "Skip on Windows")
@skipIf(salt.utils.platform.is_darwin(), "Skip on Darwin")
@skipIf(salt.utils.platform.is_freebsd(), "Skip on FreeBSD")
def test_blkid_token():
run_stdout_mock = MagicMock(return_value={"retcode": 1})
with patch.dict(disk.__salt__, {"cmd.run_all": run_stdout_mock}):
disk.blkid(token="TYPE=ext4")
run_stdout_mock.assert_called_with(
["blkid", "-t", "TYPE=ext4"], python_shell=False
)


def test_dump():
mock = MagicMock(return_value={"retcode": 0, "stdout": ""})
with patch.dict(disk.__salt__, {"cmd.run_all": mock}):
disk.dump("/dev/sda")
mock.assert_called_once_with(
"blockdev --getro --getsz --getss --getpbsz --getiomin "
"--getioopt --getalignoff --getmaxsect --getsize "
"--getsize64 --getra --getfra /dev/sda",
python_shell=False,
)


def test_wipe():
mock = MagicMock(return_value={"retcode": 0, "stdout": ""})
with patch.dict(disk.__salt__, {"cmd.run_all": mock}):
disk.wipe("/dev/sda")
mock.assert_called_once_with("wipefs -a /dev/sda", python_shell=False)


@pytest.mark.skipif(
sys.version_info < (3, 6), reason="Py3.5 dictionaries are not ordered"
)
def test_tune():
mock = MagicMock(
return_value=(
"712971264\n512\n512\n512\n0\n0\n88\n712971264\n365041287168\n512\n512"
)
)
with patch.dict(disk.__salt__, {"cmd.run": mock}):
mock_dump = MagicMock(return_value={"retcode": 0, "stdout": ""})
with patch("salt.modules.disk.dump", mock_dump):
kwargs = {"read-ahead": 512, "filesystem-read-ahead": 1024}
disk.tune("/dev/sda", **kwargs)

mock.assert_called_with(
"blockdev --setra 512 --setfra 1024 /dev/sda", python_shell=False
)


def test_format():
"""
unit tests for disk.format
"""
device = "/dev/sdX1"
mock = MagicMock(return_value=0)
with patch.dict(disk.__salt__, {"cmd.retcode": mock}), patch(
"salt.utils.path.which", MagicMock(return_value=True)
):
assert disk.format_(device) is True


def test_fat_format():
"""
unit tests for disk.format when using fat argument
"""
device = "/dev/sdX1"
expected = ["mkfs", "-t", "fat", "-F", 12, "/dev/sdX1"]
mock = MagicMock(return_value=0)
with patch.dict(disk.__salt__, {"cmd.retcode": mock}), patch(
"salt.utils.path.which", MagicMock(return_value=True)
):
assert disk.format_(device, fs_type="fat", fat=12) is True
args, kwargs = mock.call_args_list[0]
assert expected == args[0]


@skipIf(
not salt.utils.path.which("lsblk") and not salt.utils.path.which("df"),
"lsblk or df not found",
)
def test_fstype():
"""
unit tests for disk.fstype
"""
device = "/dev/sdX1"
fs_type = "ext4"
mock = MagicMock(return_value="FSTYPE\n{}".format(fs_type))
with patch.dict(disk.__grains__, {"kernel": "Linux"}), patch.dict(
disk.__salt__, {"cmd.run": mock}
), patch("salt.utils.path.which", MagicMock(return_value=True)):
assert disk.fstype(device) == fs_type


def test_resize2fs():
"""
unit tests for disk.resize2fs
"""
device = "/dev/sdX1"
mock = MagicMock()
with patch.dict(disk.__salt__, {"cmd.run_all": mock}), patch(
"salt.utils.path.which", MagicMock(return_value=True)
):
disk.resize2fs(device)
mock.assert_called_once_with("resize2fs {}".format(device), python_shell=False)


@skipIf(salt.utils.platform.is_windows(), "Skip on Windows")
@skipIf(not salt.utils.path.which("mkfs"), "mkfs not found")
def test_format_():
"""
unit tests for disk.format_
"""
device = "/dev/sdX1"
mock = MagicMock(return_value=0)
with patch.dict(disk.__salt__, {"cmd.retcode": mock}):
disk.format_(device=device)
mock.assert_any_call(["mkfs", "-t", "ext4", device], ignore_retcode=True)


@skipIf(salt.utils.platform.is_windows(), "Skip on Windows")
@skipIf(not salt.utils.path.which("mkfs"), "mkfs not found")
def test_format__fat():
"""
unit tests for disk.format_ with FAT parameter
"""
device = "/dev/sdX1"
mock = MagicMock(return_value=0)
with patch.dict(disk.__salt__, {"cmd.retcode": mock}):
disk.format_(device=device, fs_type="fat", fat=12)
mock.assert_any_call(
["mkfs", "-t", "fat", "-F", 12, device], ignore_retcode=True
)
Loading

0 comments on commit f4f4416

Please sign in to comment.