Fix: EMR ReleaseLabel validation does not respect semantic versioning (#3489)

Fixes #3474
This commit is contained in:
Brian Pandola 2020-11-22 10:54:59 -08:00 committed by GitHub
parent 93b393c679
commit 53a3e52c67
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 129 additions and 3 deletions

View File

@ -3,6 +3,7 @@
exclude_lines = exclude_lines =
if __name__ == .__main__.: if __name__ == .__main__.:
raise NotImplemented. raise NotImplemented.
return NotImplemented
def __repr__ def __repr__
[run] [run]

View File

@ -13,7 +13,7 @@ from moto.core.responses import xml_to_json_response
from moto.core.utils import tags_from_query_string from moto.core.utils import tags_from_query_string
from .exceptions import EmrError from .exceptions import EmrError
from .models import emr_backends from .models import emr_backends
from .utils import steps_from_query_string, Unflattener from .utils import steps_from_query_string, Unflattener, ReleaseLabel
def generate_boto3_response(operation): def generate_boto3_response(operation):
@ -323,7 +323,9 @@ class ElasticMapReduceResponse(BaseResponse):
custom_ami_id = self._get_param("CustomAmiId") custom_ami_id = self._get_param("CustomAmiId")
if custom_ami_id: if custom_ami_id:
kwargs["custom_ami_id"] = custom_ami_id kwargs["custom_ami_id"] = custom_ami_id
if release_label and release_label < "emr-5.7.0": if release_label and (
ReleaseLabel(release_label) < ReleaseLabel("emr-5.7.0")
):
message = "Custom AMI is not allowed" message = "Custom AMI is not allowed"
raise EmrError( raise EmrError(
error_type="ValidationException", error_type="ValidationException",

View File

@ -1,5 +1,6 @@
from __future__ import unicode_literals from __future__ import unicode_literals
import random import random
import re
import string import string
from moto.core.utils import camelcase_to_underscores from moto.core.utils import camelcase_to_underscores
@ -144,3 +145,76 @@ class CamelToUnderscoresWalker:
@staticmethod @staticmethod
def parse_scalar(x): def parse_scalar(x):
return x return x
class ReleaseLabel(object):
version_re = re.compile(r"^emr-(\d+)\.(\d+)\.(\d+)$")
def __init__(self, release_label):
major, minor, patch = self.parse(release_label)
self.major = major
self.minor = minor
self.patch = patch
@classmethod
def parse(cls, release_label):
if not release_label:
raise ValueError("Invalid empty ReleaseLabel: %r" % release_label)
match = cls.version_re.match(release_label)
if not match:
raise ValueError("Invalid ReleaseLabel: %r" % release_label)
major, minor, patch = match.groups()
major = int(major)
minor = int(minor)
patch = int(patch)
return major, minor, patch
def __str__(self):
version = "emr-%d.%d.%d" % (self.major, self.minor, self.patch)
return version
def __repr__(self):
return "%s(%r)" % (self.__class__.__name__, str(self))
def __iter__(self):
return iter((self.major, self.minor, self.patch))
def __eq__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return (
self.major == other.major
and self.minor == other.minor
and self.patch == other.patch
)
def __ne__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return tuple(self) != tuple(other)
def __lt__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return tuple(self) < tuple(other)
def __le__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return tuple(self) <= tuple(other)
def __gt__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return tuple(self) > tuple(other)
def __ge__(self, other):
if not isinstance(other, self.__class__):
return NotImplemented
return tuple(self) >= tuple(other)

View File

@ -636,7 +636,7 @@ def test_run_job_flow_with_custom_ami():
args = deepcopy(run_job_flow_args) args = deepcopy(run_job_flow_args)
args["CustomAmiId"] = "MyEmrCustomAmi" args["CustomAmiId"] = "MyEmrCustomAmi"
args["ReleaseLabel"] = "emr-5.7.0" args["ReleaseLabel"] = "emr-5.31.0"
cluster_id = client.run_job_flow(**args)["JobFlowId"] cluster_id = client.run_job_flow(**args)["JobFlowId"]
resp = client.describe_cluster(ClusterId=cluster_id) resp = client.describe_cluster(ClusterId=cluster_id)
resp["Cluster"]["CustomAmiId"].should.equal("MyEmrCustomAmi") resp["Cluster"]["CustomAmiId"].should.equal("MyEmrCustomAmi")

View File

@ -0,0 +1,49 @@
import pytest
from moto.emr.utils import ReleaseLabel
def test_invalid_release_labels_raise_exception():
invalid_releases = [
"",
"0",
"1.0",
"emr-2.0",
]
for invalid_release in invalid_releases:
with pytest.raises(ValueError):
ReleaseLabel(invalid_release)
def test_release_label_comparisons():
assert str(ReleaseLabel("emr-5.1.2")) == "emr-5.1.2"
assert ReleaseLabel("emr-5.0.0") != ReleaseLabel("emr-5.0.1")
assert ReleaseLabel("emr-5.0.0") == ReleaseLabel("emr-5.0.0")
assert ReleaseLabel("emr-5.31.0") > ReleaseLabel("emr-5.7.0")
assert ReleaseLabel("emr-6.0.0") > ReleaseLabel("emr-5.7.0")
assert ReleaseLabel("emr-5.7.0") < ReleaseLabel("emr-5.10.0")
assert ReleaseLabel("emr-5.10.0") < ReleaseLabel("emr-5.10.1")
assert ReleaseLabel("emr-5.60.0") >= ReleaseLabel("emr-5.7.0")
assert ReleaseLabel("emr-6.0.0") >= ReleaseLabel("emr-6.0.0")
assert ReleaseLabel("emr-5.7.0") <= ReleaseLabel("emr-5.17.0")
assert ReleaseLabel("emr-5.7.0") <= ReleaseLabel("emr-5.7.0")
releases_unsorted = [
ReleaseLabel("emr-5.60.2"),
ReleaseLabel("emr-4.0.1"),
ReleaseLabel("emr-4.0.0"),
ReleaseLabel("emr-5.7.3"),
]
releases_sorted = [str(label) for label in sorted(releases_unsorted)]
expected = [
"emr-4.0.0",
"emr-4.0.1",
"emr-5.7.3",
"emr-5.60.2",
]
assert releases_sorted == expected