523 lines
17 KiB
Python
523 lines
17 KiB
Python
import json
|
|
from copy import deepcopy
|
|
|
|
import pytest
|
|
import sure # noqa # pylint: disable=unused-import
|
|
|
|
import moto.server as server
|
|
from moto import mock_eks
|
|
from moto.core import ACCOUNT_ID
|
|
from moto.eks.exceptions import ResourceInUseException, ResourceNotFoundException
|
|
from moto.eks.models import (
|
|
CLUSTER_EXISTS_MSG,
|
|
CLUSTER_IN_USE_MSG,
|
|
CLUSTER_NOT_FOUND_MSG,
|
|
NODEGROUP_EXISTS_MSG,
|
|
NODEGROUP_NOT_FOUND_MSG,
|
|
)
|
|
from moto.eks.responses import DEFAULT_MAX_RESULTS, DEFAULT_NEXT_TOKEN
|
|
from tests.test_eks.test_eks import all_arn_values_should_be_valid
|
|
from tests.test_eks.test_eks_constants import (
|
|
AddonAttributes,
|
|
ClusterAttributes,
|
|
DEFAULT_ENCODING,
|
|
DEFAULT_HTTP_HEADERS,
|
|
DEFAULT_REGION,
|
|
Endpoints,
|
|
FargateProfileAttributes,
|
|
HttpHeaders,
|
|
NodegroupAttributes,
|
|
NODEROLE_ARN_KEY,
|
|
NODEROLE_ARN_VALUE,
|
|
PARTITIONS,
|
|
RegExTemplates,
|
|
ResponseAttributes,
|
|
ROLE_ARN_KEY,
|
|
ROLE_ARN_VALUE,
|
|
SERVICE,
|
|
StatusCodes,
|
|
SUBNETS_KEY,
|
|
SUBNETS_VALUE,
|
|
)
|
|
|
|
"""
|
|
Test the different server responses
|
|
"""
|
|
|
|
NAME_LIST = ["foo", "bar", "baz", "qux"]
|
|
|
|
|
|
class TestCluster:
|
|
cluster_name = "example_cluster"
|
|
data = {ClusterAttributes.NAME: cluster_name, ROLE_ARN_KEY: ROLE_ARN_VALUE}
|
|
endpoint = Endpoints.CREATE_CLUSTER
|
|
expected_arn_values = [
|
|
PARTITIONS,
|
|
DEFAULT_REGION,
|
|
ACCOUNT_ID,
|
|
cluster_name,
|
|
]
|
|
|
|
|
|
class TestNodegroup:
|
|
cluster_name = TestCluster.cluster_name
|
|
nodegroup_name = "example_nodegroup"
|
|
data = {
|
|
ClusterAttributes.CLUSTER_NAME: cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: nodegroup_name,
|
|
NODEROLE_ARN_KEY: NODEROLE_ARN_VALUE,
|
|
SUBNETS_KEY: SUBNETS_VALUE,
|
|
}
|
|
endpoint = Endpoints.CREATE_NODEGROUP.format(clusterName=cluster_name)
|
|
expected_arn_values = [
|
|
PARTITIONS,
|
|
DEFAULT_REGION,
|
|
ACCOUNT_ID,
|
|
cluster_name,
|
|
nodegroup_name,
|
|
None,
|
|
]
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def test_client():
|
|
backend = server.create_backend_app(SERVICE)
|
|
yield backend.test_client()
|
|
|
|
|
|
@pytest.fixture(scope="function")
|
|
def create_cluster(test_client):
|
|
def create_and_verify_cluster(client, name):
|
|
"""Creates one valid cluster and verifies return status code 200."""
|
|
data = deepcopy(TestCluster.data)
|
|
data.update(name=name)
|
|
response = client.post(
|
|
TestCluster.endpoint, data=json.dumps(data), headers=DEFAULT_HTTP_HEADERS,
|
|
)
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
|
|
return json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.CLUSTER
|
|
]
|
|
|
|
def _execute(name=TestCluster.cluster_name):
|
|
return create_and_verify_cluster(test_client, name=name)
|
|
|
|
yield _execute
|
|
|
|
|
|
@pytest.fixture(scope="function", autouse=True)
|
|
def create_nodegroup(test_client):
|
|
def create_and_verify_nodegroup(client, name):
|
|
"""Creates one valid nodegroup and verifies return status code 200."""
|
|
data = deepcopy(TestNodegroup.data)
|
|
data.update(nodegroupName=name)
|
|
response = client.post(
|
|
TestNodegroup.endpoint, data=json.dumps(data), headers=DEFAULT_HTTP_HEADERS,
|
|
)
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
|
|
return json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.NODEGROUP
|
|
]
|
|
|
|
def _execute(name=TestNodegroup.nodegroup_name):
|
|
return create_and_verify_nodegroup(test_client, name=name)
|
|
|
|
yield _execute
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_create_single_cluster(create_cluster):
|
|
result_cluster = create_cluster()
|
|
|
|
result_cluster[ClusterAttributes.NAME].should.equal(TestCluster.cluster_name)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestCluster.expected_arn_values,
|
|
pattern=RegExTemplates.CLUSTER_ARN,
|
|
arn_under_test=result_cluster[ClusterAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_create_multiple_clusters_with_same_name(test_client, create_cluster):
|
|
create_cluster()
|
|
expected_exception = ResourceInUseException
|
|
expected_msg = CLUSTER_EXISTS_MSG.format(clusterName=TestCluster.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestCluster.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.post(
|
|
TestCluster.endpoint,
|
|
data=json.dumps(TestCluster.data),
|
|
headers=DEFAULT_HTTP_HEADERS,
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_create_nodegroup_without_cluster(test_client):
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = CLUSTER_NOT_FOUND_MSG.format(clusterName=TestCluster.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: None,
|
|
NodegroupAttributes.NODEGROUP_NAME: None,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
endpoint = Endpoints.CREATE_NODEGROUP.format(clusterName=TestCluster.cluster_name)
|
|
|
|
response = test_client.post(
|
|
endpoint, data=json.dumps(TestNodegroup.data), headers=DEFAULT_HTTP_HEADERS
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_create_nodegroup_on_existing_cluster(create_cluster, create_nodegroup):
|
|
create_cluster()
|
|
result_data = create_nodegroup()
|
|
|
|
result_data[NodegroupAttributes.NODEGROUP_NAME].should.equal(
|
|
TestNodegroup.nodegroup_name
|
|
)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestNodegroup.expected_arn_values,
|
|
pattern=RegExTemplates.NODEGROUP_ARN,
|
|
arn_under_test=result_data[NodegroupAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_create_multiple_nodegroups_with_same_name(
|
|
test_client, create_cluster, create_nodegroup
|
|
):
|
|
create_cluster()
|
|
create_nodegroup()
|
|
expected_exception = ResourceInUseException
|
|
expected_msg = NODEGROUP_EXISTS_MSG.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestNodegroup.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: TestNodegroup.nodegroup_name,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.post(
|
|
TestNodegroup.endpoint,
|
|
data=json.dumps(TestNodegroup.data),
|
|
headers=DEFAULT_HTTP_HEADERS,
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_list_clusters(test_client, create_cluster):
|
|
[create_cluster(name) for name in NAME_LIST]
|
|
|
|
response = test_client.get(
|
|
Endpoints.LIST_CLUSTERS.format(
|
|
maxResults=DEFAULT_MAX_RESULTS, nextToken=DEFAULT_NEXT_TOKEN
|
|
)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.CLUSTERS
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
len(result_data).should.equal(len(NAME_LIST))
|
|
sorted(result_data).should.equal(sorted(NAME_LIST))
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_list_nodegroups(test_client, create_cluster, create_nodegroup):
|
|
create_cluster()
|
|
[create_nodegroup(name) for name in NAME_LIST]
|
|
|
|
response = test_client.get(
|
|
Endpoints.LIST_NODEGROUPS.format(
|
|
clusterName=TestCluster.cluster_name,
|
|
maxResults=DEFAULT_MAX_RESULTS,
|
|
nextToken=DEFAULT_NEXT_TOKEN,
|
|
)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.NODEGROUPS
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
sorted(result_data).should.equal(sorted(NAME_LIST))
|
|
len(result_data).should.equal(len(NAME_LIST))
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_describe_existing_cluster(test_client, create_cluster):
|
|
create_cluster()
|
|
|
|
response = test_client.get(
|
|
Endpoints.DESCRIBE_CLUSTER.format(clusterName=TestCluster.cluster_name)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.CLUSTER
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
result_data[ClusterAttributes.NAME].should.equal(TestCluster.cluster_name)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestCluster.expected_arn_values,
|
|
pattern=RegExTemplates.CLUSTER_ARN,
|
|
arn_under_test=result_data[ClusterAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_describe_nonexisting_cluster(test_client):
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = CLUSTER_NOT_FOUND_MSG.format(clusterName=TestCluster.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: None,
|
|
NodegroupAttributes.NODEGROUP_NAME: None,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.get(
|
|
Endpoints.DESCRIBE_CLUSTER.format(clusterName=TestCluster.cluster_name)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_describe_existing_nodegroup(test_client, create_cluster, create_nodegroup):
|
|
create_cluster()
|
|
create_nodegroup()
|
|
|
|
response = test_client.get(
|
|
Endpoints.DESCRIBE_NODEGROUP.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.NODEGROUP
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
result_data[ClusterAttributes.CLUSTER_NAME].should.equal(TestNodegroup.cluster_name)
|
|
result_data[NodegroupAttributes.NODEGROUP_NAME].should.equal(
|
|
TestNodegroup.nodegroup_name
|
|
)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestNodegroup.expected_arn_values,
|
|
pattern=RegExTemplates.NODEGROUP_ARN,
|
|
arn_under_test=result_data[NodegroupAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_describe_nonexisting_nodegroup(test_client, create_cluster):
|
|
create_cluster()
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = NODEGROUP_NOT_FOUND_MSG.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestNodegroup.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: TestNodegroup.nodegroup_name,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.get(
|
|
Endpoints.DESCRIBE_NODEGROUP.format(
|
|
clusterName=TestCluster.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_describe_nodegroup_nonexisting_cluster(test_client):
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = CLUSTER_NOT_FOUND_MSG.format(clusterName=TestNodegroup.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestNodegroup.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: TestNodegroup.nodegroup_name,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.get(
|
|
Endpoints.DESCRIBE_NODEGROUP.format(
|
|
clusterName=TestCluster.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_cluster(test_client, create_cluster):
|
|
create_cluster()
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_CLUSTER.format(clusterName=TestCluster.cluster_name)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.CLUSTER
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
result_data[ClusterAttributes.NAME].should.equal(TestCluster.cluster_name)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestCluster.expected_arn_values,
|
|
pattern=RegExTemplates.CLUSTER_ARN,
|
|
arn_under_test=result_data[ClusterAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_nonexisting_cluster(test_client):
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = CLUSTER_NOT_FOUND_MSG.format(clusterName=TestCluster.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: None,
|
|
NodegroupAttributes.NODEGROUP_NAME: None,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_CLUSTER.format(clusterName=TestCluster.cluster_name)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_cluster_with_nodegroups(
|
|
test_client, create_cluster, create_nodegroup
|
|
):
|
|
create_cluster()
|
|
create_nodegroup()
|
|
expected_exception = ResourceInUseException
|
|
expected_msg = CLUSTER_IN_USE_MSG.format(clusterName=TestCluster.cluster_name)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestCluster.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: TestNodegroup.nodegroup_name,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_CLUSTER.format(clusterName=TestCluster.cluster_name)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_nodegroup(test_client, create_cluster, create_nodegroup):
|
|
create_cluster()
|
|
create_nodegroup()
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_NODEGROUP.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))[
|
|
ResponseAttributes.NODEGROUP
|
|
]
|
|
|
|
response.status_code.should.equal(StatusCodes.OK)
|
|
result_data[ClusterAttributes.CLUSTER_NAME].should.equal(TestNodegroup.cluster_name)
|
|
result_data[NodegroupAttributes.NODEGROUP_NAME].should.equal(
|
|
TestNodegroup.nodegroup_name
|
|
)
|
|
all_arn_values_should_be_valid(
|
|
expected_arn_values=TestNodegroup.expected_arn_values,
|
|
pattern=RegExTemplates.NODEGROUP_ARN,
|
|
arn_under_test=result_data[NodegroupAttributes.ARN],
|
|
)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_nonexisting_nodegroup(test_client, create_cluster):
|
|
create_cluster()
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = NODEGROUP_NOT_FOUND_MSG.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: TestNodegroup.cluster_name,
|
|
NodegroupAttributes.NODEGROUP_NAME: TestNodegroup.nodegroup_name,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_NODEGROUP.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
@mock_eks
|
|
def test_eks_delete_nodegroup_nonexisting_cluster(test_client):
|
|
expected_exception = ResourceNotFoundException
|
|
expected_msg = CLUSTER_NOT_FOUND_MSG.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
expected_data = {
|
|
ClusterAttributes.CLUSTER_NAME: None,
|
|
NodegroupAttributes.NODEGROUP_NAME: None,
|
|
FargateProfileAttributes.FARGATE_PROFILE_NAME: None,
|
|
AddonAttributes.ADDON_NAME: None,
|
|
ResponseAttributes.MESSAGE: expected_msg,
|
|
}
|
|
|
|
response = test_client.delete(
|
|
Endpoints.DELETE_NODEGROUP.format(
|
|
clusterName=TestNodegroup.cluster_name,
|
|
nodegroupName=TestNodegroup.nodegroup_name,
|
|
)
|
|
)
|
|
|
|
should_return_expected_exception(response, expected_exception, expected_data)
|
|
|
|
|
|
def should_return_expected_exception(response, expected_exception, expected_data):
|
|
result_data = json.loads(response.data.decode(DEFAULT_ENCODING))
|
|
|
|
response.status_code.should.equal(expected_exception.STATUS)
|
|
response.headers.get(HttpHeaders.ErrorType).should.equal(expected_exception.TYPE)
|
|
result_data.should.equal(expected_data)
|