diff --git a/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/ec2_dev_2_volid.py b/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/ec2_dev_2_volid.py index 5265e4478e..0c23ade006 100644 --- a/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/ec2_dev_2_volid.py +++ b/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/ec2_dev_2_volid.py @@ -41,17 +41,7 @@ def validate_device_name(device_name): return True -def main(): - syslog.syslog("Starting ec2_dev_2_volid.py script") - # Get dev - try: - dev = str(sys.argv[1]) - validate_device_name(dev) - syslog.syslog(f"Input block device is {dev}") - except IndexError: - syslog.syslog(syslog.LOG_ERR, "Provide block device i.e. xvdf") - - # Convert dev to mapping format +def adapt_device_name(dev): if "nvme" in dev: # For newer instances which expose EBS volumes as NVMe devices, translate the # device name so boto can discover it. @@ -70,26 +60,10 @@ def main(): else: dev = dev.replace("xvd", "sd") dev = "/dev/" + dev + return dev - # Get IMDSv2 token - token = get_imdsv2_token() - - # Get instance ID - instance_id = requests.get( - "http://169.254.169.254/latest/meta-data/instance-id", - headers=token, - timeout=METADATA_REQUEST_TIMEOUT, - ).text - # Get region - region = requests.get( - "http://169.254.169.254/latest/meta-data/placement/availability-zone", - headers=token, - timeout=METADATA_REQUEST_TIMEOUT, - ).text - region = region[:-1] - - # Parse configuration file to read proxy settings +def parse_proxy_config(): config = configparser.RawConfigParser() config.read("/etc/boto.cfg") proxy_config = Config() @@ -97,18 +71,10 @@ def main(): proxy = config.get("Boto", "proxy") proxy_port = config.get("Boto", "proxy_port") proxy_config = Config(proxies={"https": f"{proxy}:{proxy_port}"}) + return proxy_config - # Configure the AWS CA bundle. - # In US isolated regions the dedicated CA bundle will be used. - # In any other region, the default bundle will be used (None stands for the default settings). - # Note: We want to apply a more general solution that applies to every region, - # but for the time being this is enough to support US isolated regions without - # impacting the other ones. - ca_bundle = f"/etc/pki/{region}/certs/ca-bundle.pem" if region.startswith("us-iso") else None - - # Connect to AWS using boto - ec2 = boto3.client("ec2", region_name=region, config=proxy_config, verify=ca_bundle) +def get_device_volume_id(ec2, dev, instance_id): # Poll for blockdevicemapping devices = ec2.describe_instance_attribute(InstanceId=instance_id, Attribute="blockDeviceMapping").get( "BlockDeviceMappings" @@ -127,8 +93,48 @@ def main(): dev_map = dict((d.get("DeviceName"), d) for d in devices) loop_count += 1 - # Return volume ID - volume_id = dev_map.get(dev).get("Ebs").get("VolumeId") + return dev_map.get(dev).get("Ebs").get("VolumeId") + + +def get_metadata_value(token, metadata_path): + return requests.get( + metadata_path, + headers=token, + timeout=METADATA_REQUEST_TIMEOUT, + ).text + + +def main(): + syslog.syslog("Starting ec2_dev_2_volid.py script") + try: + dev = str(sys.argv[1]) + validate_device_name(dev) + syslog.syslog(f"Input block device is {dev}") + except IndexError: + syslog.syslog(syslog.LOG_ERR, "Provide block device i.e. xvdf") + + dev = adapt_device_name(dev) + + token = get_imdsv2_token() + + instance_id = get_metadata_value(token, "http://169.254.169.254/latest/meta-data/instance-id") + + region = get_metadata_value(token, "http://169.254.169.254/latest/meta-data/placement/availability-zone") + region = region[:-1] + + proxy_config = parse_proxy_config() + + # Configure the AWS CA bundle. + # In US isolated regions the dedicated CA bundle will be used. + # In any other region, the default bundle will be used (None stands for the default settings). + # Note: We want to apply a more general solution that applies to every region, + # but for the time being this is enough to support US isolated regions without + # impacting the other ones. + ca_bundle = f"/etc/pki/{region}/certs/ca-bundle.pem" if region.startswith("us-iso") else None + + ec2 = boto3.client("ec2", region_name=region, config=proxy_config, verify=ca_bundle) + + volume_id = get_device_volume_id(ec2, dev, instance_id) print(volume_id) diff --git a/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/manageVolume.py b/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/manageVolume.py index 29b57b2d21..ef8948b51b 100644 --- a/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/manageVolume.py +++ b/cookbooks/aws-parallelcluster-environment/files/default/ec2_udev_rules/manageVolume.py @@ -32,7 +32,7 @@ def validate_device_name(device_name): return True -def convert_dev(dev): +def adapt_device_name(dev): # Translate the device name as provided by the OS to the one used by EC2 # FIXME This approach could be broken in some OS variants, see # pylint: disable=fixme # https://docs.aws.amazon.com/AWSEC2/latest/UserGuide/nvme-ebs-volumes.html#identify-nvme-ebs-device @@ -91,7 +91,7 @@ def get_imdsv2_token(): def attach_volume(volume_id, instance_id, ec2): # Generate a list of system paths minus the root path - paths = [convert_dev(device) for device in get_all_devices()] + paths = [adapt_device_name(device) for device in get_all_devices()] # List of possible block devices block_devices = [ @@ -175,7 +175,6 @@ def detach_volume(volume_id, ec2): def parse_proxy_config(): - """Parse configuration file to read proxy settings.""" config = configparser.RawConfigParser() config.read("/etc/boto.cfg") proxy_config = Config() @@ -186,38 +185,33 @@ def parse_proxy_config(): return proxy_config -def handle_volume(volume_id, attach, detach): - # Get IMDSv2 token - token = get_imdsv2_token() - - # Get instance ID - instance_id = requests.get( - "http://169.254.169.254/latest/meta-data/instance-id", +def get_metadata_value(token, metadata_path): + return requests.get( + metadata_path, headers=token, timeout=METADATA_REQUEST_TIMEOUT, ).text - # Get region - region = requests.get( - "http://169.254.169.254/latest/meta-data/placement/availability-zone", - headers=token, - timeout=METADATA_REQUEST_TIMEOUT, - ).text + +def handle_volume(volume_id, attach, detach): + token = get_imdsv2_token() + + instance_id = get_metadata_value(token, "http://169.254.169.254/latest/meta-data/instance-id") + + region = get_metadata_value(token, "http://169.254.169.254/latest/meta-data/placement/availability-zone") region = region[:-1] - # Parse configuration file to read proxy settings proxy_config = parse_proxy_config() - # Connect to AWS using boto ec2 = boto3.client("ec2", region_name=region, config=proxy_config) - if attach and is_volume_avaialble(ec2, volume_id): + if attach and is_volume_available(ec2, volume_id): attach_volume(volume_id, instance_id, ec2) elif detach and is_volume_attached(ec2, volume_id): detach_volume(volume_id, ec2) -def is_volume_avaialble(ec2, volume_id): +def is_volume_available(ec2, volume_id): try: state = ec2.describe_volumes(VolumeIds=[volume_id]).get("Volumes")[0].get("State") if state == "available": diff --git a/test/unit/ec2_udev_rules/test_ec2_dev_2_volid.py b/test/unit/ec2_udev_rules/test_ec2_dev_2_volid.py index ae937ee8d9..5305a109a7 100644 --- a/test/unit/ec2_udev_rules/test_ec2_dev_2_volid.py +++ b/test/unit/ec2_udev_rules/test_ec2_dev_2_volid.py @@ -11,7 +11,7 @@ # See the License for the specific language governing permissions and limitations under the License. import pytest from assertpy import assert_that -from ec2_dev_2_volid import validate_device_name +from ec2_dev_2_volid import adapt_device_name, get_device_volume_id, get_imdsv2_token, validate_device_name @pytest.mark.parametrize( @@ -31,3 +31,115 @@ def test_validate_device_name(device_name, raises): assert_that(validate_device_name).raises(ValueError).when_called_with(device_name).contains("invalid pattern") else: assert_that(validate_device_name(device_name)).is_true() + + +@pytest.mark.parametrize( + ("status_code", "content", "expected_value"), + [(200, {"key": "value"}, {"X-aws-ec2-metadata-token": {"key": "value"}}), (400, {"key": "value"}, {})], +) +def test_get_imdsv2_token(mocker, status_code, content, expected_value): + mock = mocker.Mock() + mocker.patch("requests.put", mock) + mock.return_value.status_code = status_code + mock.return_value.content = content + assert_that(get_imdsv2_token()).is_equal_to(expected_value) + + +@pytest.mark.parametrize( + ("dev", "expected_name", "raises"), + [ + ("nvme0n1", "sdf", True), + ("nvme0n1p1", "sdf", True), + ("nvme0n1p128", "sdf", True), + ("xvd0n1", "/dev/sd0n1", False), + ("xvd0n1p1", "/dev/sd0n1p1", False), + ], +) +def test_adapt_device_name(mocker, dev, expected_name, raises, capsys): + mocker.patch("os.popen", mocker.mock_open(read_data=":sdf")) + if raises: + with pytest.raises(SystemExit) as e: + adapt_device_name(dev) + captured = capsys.readouterr() + assert_that(expected_name).is_equal_to(captured.out) + assert_that(e.value.code).is_equal_to(0) + else: + assert_that(adapt_device_name(dev)).matches(expected_name) + + +@pytest.fixture(name="ec2_mock") +def fixture_ec2_mock(mocker): + mock = mocker.MagicMock() + mocker.patch("boto3.client", mock) + return mock + + +@pytest.mark.parametrize( + ("dev", "block", "output_value"), + [ + ( + "/dev/sda1", + { + "InstanceId": "i-1234567890abcdef0", + "BlockDeviceMappings": [ + { + "DeviceName": "/dev/sda1", + "Ebs": { + "Status": "attached", + "DeleteOnTermination": True, + "VolumeId": "vol-049df61146c4d7901", + "AttachTime": "2013-05-17T22:42:34.000Z", + }, + }, + { + "DeviceName": "/dev/sdf", + "Ebs": { + "Status": "attached", + "DeleteOnTermination": False, + "VolumeId": "vol-049df61146c4d7901", + "AttachTime": "2013-09-10T23:07:00.000Z", + }, + }, + ], + }, + "vol-049df61146c4d7901", + ), + ( + "/dev/sda2", + { + "InstanceId": "i-1234567890abcdef0", + "BlockDeviceMappings": [ + { + "DeviceName": "/dev/sda1", + "Ebs": { + "Status": "attached", + "DeleteOnTermination": True, + "VolumeId": "vol-049df61146c4d7901", + "AttachTime": "2013-05-17T22:42:34.000Z", + }, + }, + { + "DeviceName": "/dev/sdf", + "Ebs": { + "Status": "attached", + "DeleteOnTermination": False, + "VolumeId": "vol-049df61146c4d7901", + "AttachTime": "2013-09-10T23:07:00.000Z", + }, + }, + ], + }, + SystemExit, + ), + ], +) +def test_get_device_volume_id(ec2_mock, dev, block, output_value): + ec2_mock.describe_instance_attribute.return_value = block + if output_value == SystemExit: + with pytest.raises(SystemExit) as e: + get_device_volume_id(ec2_mock, dev, 1) + assert_that(e.type).is_equal_to(SystemExit) + assert_that(e.value.code).is_equal_to(1) + else: + volume_id = get_device_volume_id(ec2_mock, dev, 1) + assert_that(volume_id).is_equal_to(output_value) diff --git a/test/unit/ec2_udev_rules/test_manage_volume.py b/test/unit/ec2_udev_rules/test_manage_volume.py index 9696260b06..12fd63a316 100644 --- a/test/unit/ec2_udev_rules/test_manage_volume.py +++ b/test/unit/ec2_udev_rules/test_manage_volume.py @@ -11,7 +11,19 @@ # See the License for the specific language governing permissions and limitations under the License. import pytest from assertpy import assert_that -from manageVolume import validate_device_name +from manageVolume import ( + adapt_device_name, + argparse, + attach_volume, + detach_volume, + get_all_devices, + get_imdsv2_token, + is_volume_attached, + is_volume_available, + main, + subprocess, + validate_device_name, +) @pytest.mark.parametrize( @@ -31,3 +43,168 @@ def test_validate_device_name(device_name, raises): assert_that(validate_device_name).raises(ValueError).when_called_with(device_name).contains("invalid pattern") else: assert_that(validate_device_name(device_name)).is_true() + + +@pytest.mark.parametrize( + ("dev", "expected_name"), + [ + ("/dev/nvme0n1", "/dev/sdf"), + ("/dev/nvme0n1p1", "/dev/sdf"), + ("/dev/nvme0n1p128", "/dev/sdf"), + ("/dev/hd0n1", "/dev/sd0n1"), + ("/dev/hd0n1p1", "/dev/sd0n1p1"), + ("/dev/xvd0n1", "/dev/sd0n1"), + ("/dev/xvd0n1p1", "/dev/sd0n1p1"), + ], +) +def test_adapt_device_name(mocker, dev, expected_name): + mocker.patch("os.popen", mocker.mock_open(read_data="sdf")) + assert_that(adapt_device_name(dev)).matches(expected_name) + + +@pytest.mark.parametrize( + "name", + ["xvda", "xvdb"], +) +def test_get_all_devices(mocker, name): + mocker.patch( + "subprocess.check_output", + side_effect=[name, subprocess.CalledProcessError(returncode=0, cmd=["/bin/lsblk", "-d", "-n"])], + ) + assert_that(get_all_devices).raises(subprocess.CalledProcessError) + assert_that(get_all_devices()).contains("/dev/" + name) + + +@pytest.mark.parametrize( + ("status_code", "content", "expected_value"), + [(200, {"key": "value"}, {"X-aws-ec2-metadata-token": {"key": "value"}}), (400, {"key": "value"}, {})], +) +def test_get_imdsv2_token(mocker, status_code, content, expected_value): + mock = mocker.Mock() + mocker.patch("requests.put", mock) + mock.return_value.status_code = status_code + mock.return_value.content = content + assert_that(get_imdsv2_token()).is_equal_to(expected_value) + + +@pytest.fixture(name="volume_response") +def fixture_volume_response(): + return { + "Device": "dev-01", + "InstanceId": "instance-01", + "State": "attached", + "VolumeId": "vol-01", + "DeleteOnTermination": True, + } + + +@pytest.fixture(name="ec2_mock") +def fixture_ec2_mock(mocker): + mock = mocker.MagicMock() + mocker.patch("boto3.client", mock) + return mock + + +@pytest.mark.parametrize( + ("state", "message"), + [("attached", ""), ("busy", "bad state"), ("detached", "bad state"), ("available", "failed to mount")], +) +def test_attach_volume(mocker, volume_response, state, message, ec2_mock, capsys): + mocker.patch("os.popen", mocker.mock_open(read_data="sdf")) + mocker.patch("subprocess.check_output", return_value="xvda") + + volume_response["State"] = state + ec2_mock.attach_volume.return_value = volume_response + ec2_mock.describe_volumes.return_value = {"Volumes": [{"Attachments": [{"State": state}]}]} + + if state != "attached": + with pytest.raises(SystemExit) as e: + attach_volume(volume_response["VolumeId"], volume_response["InstanceId"], ec2_mock) + assert_that(e.type).is_equal_to(SystemExit) + assert_that(e.value.code).is_equal_to(1) + + captured = capsys.readouterr() + assert_that(captured.out).contains(message) + + +@pytest.mark.parametrize( + ("state", "message"), + [("attached", "bad state"), ("busy", "bad state"), ("detached", "failed to detach"), ("available", "")], +) +def test_detach_volume(volume_response, state, message, ec2_mock, capsys): + volume_response["State"] = state + ec2_mock.detach_volume.return_value = volume_response + ec2_mock.describe_volumes.return_value = {"Volumes": [{"State": state}]} + + if state != "available": + with pytest.raises(SystemExit) as e: + detach_volume(volume_response["VolumeId"], ec2_mock) + assert_that(e.type).is_equal_to(SystemExit) + assert_that(e.value.code).is_equal_to(1) + + captured = capsys.readouterr() + assert_that(captured.out).contains(message) + + +@pytest.mark.parametrize( + ("volume_output", "expected_available"), + [ + ({"Volumes": [{"State": "available"}]}, True), + ({"Volumes": [{"State": "in-use"}]}, False), + ({"Volumes": [{"State": "busy"}]}, False), + (Exception, False), + ], +) +def test_is_volume_available(ec2_mock, volume_output, expected_available, capsys): + ec2_mock.describe_volumes.return_value = volume_output + is_avail = is_volume_available(ec2_mock, 0) + + if volume_output == Exception: + captured = capsys.readouterr() + assert_that(captured.out).contains("exception") + assert_that(is_avail).is_false() + else: + assert_that(is_avail).is_equal_to(expected_available) + + +@pytest.mark.parametrize( + ("volume_output", "expected_attached"), + [ + ({"Volumes": [{"State": "available"}]}, False), + ({"Volumes": [{"State": "in-use"}]}, True), + ({"Volumes": [{"State": "busy"}]}, False), + (Exception, False), + ], +) +def test_is_volume_attached(ec2_mock, volume_output, expected_attached, capsys): + ec2_mock.describe_volumes.return_value = volume_output + is_attached = is_volume_attached(ec2_mock, 0) + + if volume_output == Exception: + captured = capsys.readouterr() + assert_that(captured.out).contains("exception") + assert_that(is_attached).is_false() + else: + assert_that(is_attached).is_equal_to(expected_attached) + + +@pytest.mark.parametrize( + ("attach", "detach", "raises"), + [(True, False, False), (True, True, False), (False, True, False), (False, False, True)], +) +def test_main(mocker, attach, detach, raises): + mocker.patch( + "argparse.ArgumentParser.parse_args", return_value=argparse.Namespace(attach=attach, detach=detach, volume_id=1) + ) + mocker.patch("manageVolume.handle_volume", return_value=True) + + if raises: + with pytest.raises(SystemExit) as e: + main() + assert_that(e.type).is_equal_to(SystemExit) + assert_that(e.value.code).is_equal_to(1) + else: + try: + main() + except Exception: + pytest.fail("Main raised an exception, should not fail")