2021-12-27 20:15:37 +00:00
""" Unit tests for dax-supported APIs. """
import boto3
import pytest
import sure # noqa # pylint: disable=unused-import
from botocore . exceptions import ClientError
from moto import mock_dax
from moto . core import ACCOUNT_ID
# See our Development Tips on writing tests for hints on how to write good tests:
# http://docs.getmoto.org/en/latest/docs/contributing/development_tips/tests.html
@mock_dax
def test_create_cluster_minimal ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-2 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
created_cluster = client . create_cluster (
ClusterName = " daxcluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = iam_role_arn ,
) [ " Cluster " ]
described_cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [
" Clusters "
] [ 0 ]
for cluster in [ created_cluster , described_cluster ] :
cluster [ " ClusterName " ] . should . equal ( " daxcluster " )
cluster [ " ClusterArn " ] . should . equal (
f " arn:aws:dax:us-east-2: { ACCOUNT_ID } :cache/daxcluster "
)
cluster [ " TotalNodes " ] . should . equal ( 3 )
cluster [ " ActiveNodes " ] . should . equal ( 0 )
cluster [ " NodeType " ] . should . equal ( " dax.t3.small " )
cluster [ " Status " ] . should . equal ( " creating " )
cluster [ " ClusterDiscoveryEndpoint " ] . should . equal ( { " Port " : 8111 } )
cluster [ " PreferredMaintenanceWindow " ] . should . equal ( " thu:23:30-fri:00:30 " )
cluster [ " SubnetGroup " ] . should . equal ( " default " )
cluster [ " SecurityGroups " ] . should . have . length_of ( 1 )
cluster [ " IamRoleArn " ] . should . equal ( iam_role_arn )
cluster . should . have . key ( " ParameterGroup " )
cluster [ " ParameterGroup " ] . should . have . key ( " ParameterGroupName " ) . equals (
" default.dax1.0 "
)
cluster [ " SSEDescription " ] . should . equal ( { " Status " : " DISABLED " } )
cluster . should . have . key ( " ClusterEndpointEncryptionType " ) . equals ( " NONE " )
@mock_dax
def test_create_cluster_description ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-2 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
created_cluster = client . create_cluster (
ClusterName = " daxcluster " ,
Description = " my cluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = iam_role_arn ,
) [ " Cluster " ]
described_cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [
" Clusters "
] [ 0 ]
for cluster in [ created_cluster , described_cluster ] :
cluster [ " ClusterName " ] . should . equal ( " daxcluster " )
cluster [ " Description " ] . should . equal ( " my cluster " )
@mock_dax
def test_create_cluster_with_sse_enabled ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-2 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
created_cluster = client . create_cluster (
ClusterName = " daxcluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = iam_role_arn ,
SSESpecification = { " Enabled " : True } ,
2022-05-19 11:08:02 +00:00
ClusterEndpointEncryptionType = " TLS " ,
2021-12-27 20:15:37 +00:00
) [ " Cluster " ]
described_cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [
" Clusters "
] [ 0 ]
for cluster in [ created_cluster , described_cluster ] :
cluster [ " ClusterName " ] . should . equal ( " daxcluster " )
cluster [ " SSEDescription " ] . should . equal ( { " Status " : " ENABLED " } )
2022-05-19 11:08:02 +00:00
cluster [ " ClusterEndpointEncryptionType " ] . should . equal ( " TLS " )
2021-12-27 20:15:37 +00:00
@mock_dax
def test_create_cluster_invalid_arn ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = " 1invalid " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " n/a " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal ( " ARNs must start with ' arn: ' : n/a " )
@mock_dax
def test_create_cluster_invalid_arn_no_partition ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = " 1invalid " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " arn:sth " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal ( " Second colon partition not found: arn:sth " )
@mock_dax
def test_create_cluster_invalid_arn_no_vendor ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = " 1invalid " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " arn:sth:aws " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal ( " Third colon vendor not found: arn:sth:aws " )
@mock_dax
def test_create_cluster_invalid_arn_no_region ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = " 1invalid " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " arn:sth:aws:else " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal (
" Fourth colon (region/namespace delimiter) not found: arn:sth:aws:else "
)
@mock_dax
def test_create_cluster_invalid_arn_no_namespace ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = " 1invalid " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " arn:sth:aws:else:eu-west-1 " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal (
" Fifth colon (namespace/relative-id delimiter) not found: arn:sth:aws:else:eu-west-1 "
)
@mock_dax
@pytest.mark.parametrize (
" name " , [ " 1invalid " , " iИvalid " , " in_valid " , " invalid- " , " in--valid " ]
)
def test_create_cluster_invalid_name ( name ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . create_cluster (
ClusterName = name ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = " arn:aws:iam::486285699788:role/apigatewayrole " ,
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal (
" Cluster ID specified is not a valid identifier. Identifiers must begin with a letter; must contain only ASCII letters, digits, and hyphens; and must not end with a hyphen or contain two consecutive hyphens. "
)
@mock_dax
@pytest.mark.parametrize (
" name " , [ " 1invalid " , " iИvalid " , " in_valid " , " invalid- " , " in--valid " ]
)
def test_describe_clusters_invalid_name ( name ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . describe_clusters ( ClusterNames = [ name ] )
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " InvalidParameterValueException " )
err [ " Message " ] . should . equal (
" Cluster ID specified is not a valid identifier. Identifiers must begin with a letter; must contain only ASCII letters, digits, and hyphens; and must not end with a hyphen or contain two consecutive hyphens. "
)
@mock_dax
def test_delete_cluster_unknown ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . delete_cluster ( ClusterName = " unknown " )
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equals ( " ClusterNotFoundFault " )
err [ " Message " ] . should . equal ( " Cluster not found. " )
@mock_dax
def test_delete_cluster ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
client . create_cluster (
ClusterName = " daxcluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 2 ,
IamRoleArn = iam_role_arn ,
)
client . delete_cluster ( ClusterName = " daxcluster " )
for _ in range ( 0 , 3 ) :
# Cluster takes a while to delete...
cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [ " Clusters " ] [ 0 ]
cluster [ " Status " ] . should . equal ( " deleting " )
cluster [ " TotalNodes " ] . should . equal ( 2 )
cluster [ " ActiveNodes " ] . should . equal ( 0 )
cluster . shouldnt . have . key ( " Nodes " )
with pytest . raises ( ClientError ) as exc :
client . describe_clusters ( ClusterNames = [ " daxcluster " ] )
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " ClusterNotFoundFault " )
@mock_dax
def test_describe_cluster_unknown ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
with pytest . raises ( ClientError ) as exc :
client . describe_clusters ( ClusterNames = [ " unknown " ] )
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " ClusterNotFoundFault " )
err [ " Message " ] . should . equal ( " Cluster unknown not found. " )
@mock_dax
def test_describe_clusters_returns_all ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-1 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
for i in range ( 0 , 50 ) :
client . create_cluster (
ClusterName = f " daxcluster { i } " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 1 ,
IamRoleArn = iam_role_arn ,
)
clusters = client . describe_clusters ( ) [ " Clusters " ]
clusters . should . have . length_of ( 50 )
@mock_dax
def test_describe_clusters_paginates ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-1 " )
iam_role_arn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX "
for i in range ( 0 , 50 ) :
client . create_cluster (
ClusterName = f " daxcluster { i } " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 1 ,
IamRoleArn = iam_role_arn ,
)
resp = client . describe_clusters ( MaxResults = 10 )
resp [ " Clusters " ] . should . have . length_of ( 10 )
resp . should . have . key ( " NextToken " )
resp = client . describe_clusters ( MaxResults = 10 , NextToken = resp [ " NextToken " ] )
resp [ " Clusters " ] . should . have . length_of ( 10 )
resp . should . have . key ( " NextToken " )
resp = client . describe_clusters ( NextToken = resp [ " NextToken " ] )
resp [ " Clusters " ] . should . have . length_of ( 30 )
resp . shouldnt . have . key ( " NextToken " )
@mock_dax
def test_describe_clusters_returns_nodes_after_some_time ( ) :
client = boto3 . client ( " dax " , region_name = " us-east-2 " )
client . create_cluster (
ClusterName = " daxcluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX " ,
) [ " Cluster " ]
for _ in range ( 0 , 3 ) :
# Cluster takes a while to load...
cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [ " Clusters " ] [ 0 ]
cluster [ " Status " ] . should . equal ( " creating " )
cluster . shouldnt . have . key ( " Nodes " )
# Finished loading by now
cluster = client . describe_clusters ( ClusterNames = [ " daxcluster " ] ) [ " Clusters " ] [ 0 ]
cluster [ " ClusterName " ] . should . equal ( " daxcluster " )
cluster [ " ClusterArn " ] . should . equal (
f " arn:aws:dax:us-east-2: { ACCOUNT_ID } :cache/daxcluster "
)
cluster [ " TotalNodes " ] . should . equal ( 3 )
cluster [ " ActiveNodes " ] . should . equal ( 0 )
cluster [ " NodeType " ] . should . equal ( " dax.t3.small " )
cluster [ " Status " ] . should . equal ( " available " )
# Address Info is only available when the cluster is ready
cluster . should . have . key ( " ClusterDiscoveryEndpoint " )
endpoint = cluster [ " ClusterDiscoveryEndpoint " ]
endpoint . should . have . key ( " Address " )
address = endpoint [ " Address " ]
cluster_id = address . split ( " . " ) [ 1 ]
address . should . equal (
f " daxcluster. { cluster_id } .dax-clusters.us-east-2.amazonaws.com "
)
endpoint . should . have . key ( " Port " ) . equal ( 8111 )
endpoint . should . have . key ( " URL " ) . equal ( f " dax:// { address } " )
# Nodes are only shown when the cluster is ready
cluster . should . have . key ( " Nodes " ) . length_of ( 3 )
for idx , a in enumerate ( [ " a " , " b " , " c " ] ) :
node = cluster [ " Nodes " ] [ idx ]
node . should . have . key ( " NodeId " ) . equals ( f " daxcluster- { a } " )
node . should . have . key ( " Endpoint " )
node_address = (
f " daxcluster- { a } . { cluster_id } .nodes.dax-clusters.us-east-2.amazonaws.com "
)
node [ " Endpoint " ] . should . have . key ( " Address " ) . equals ( node_address )
node [ " Endpoint " ] . should . have . key ( " Port " ) . equals ( 8111 )
node . should . contain ( " AvailabilityZone " )
node . should . have . key ( " NodeStatus " ) . equals ( " available " )
node . should . have . key ( " ParameterGroupStatus " ) . equals ( " in-sync " )
cluster [ " PreferredMaintenanceWindow " ] . should . equal ( " thu:23:30-fri:00:30 " )
cluster [ " SubnetGroup " ] . should . equal ( " default " )
cluster [ " SecurityGroups " ] . should . have . length_of ( 1 )
cluster . should . have . key ( " ParameterGroup " )
cluster [ " ParameterGroup " ] . should . have . key ( " ParameterGroupName " ) . equals (
" default.dax1.0 "
)
cluster [ " SSEDescription " ] . should . equal ( { " Status " : " DISABLED " } )
cluster . should . have . key ( " ClusterEndpointEncryptionType " ) . equals ( " NONE " )
@mock_dax
def test_list_tags_unknown ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
with pytest . raises ( ClientError ) as exc :
client . list_tags ( ResourceName = " unknown " )
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " ClusterNotFoundFault " )
@mock_dax
def test_list_tags ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
cluster = client . create_cluster (
ClusterName = " daxcluster " ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 3 ,
IamRoleArn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX " ,
Tags = [
{ " Key " : " tag1 " , " Value " : " value1 " } ,
{ " Key " : " tag2 " , " Value " : " value2 " } ,
{ " Key " : " tag3 " , " Value " : " value3 " } ,
] ,
) [ " Cluster " ]
for name in [ " daxcluster " , cluster [ " ClusterArn " ] ] :
resp = client . list_tags ( ResourceName = name )
resp . shouldnt . have . key ( " NextToken " )
resp . should . have . key ( " Tags " ) . equals (
[
{ " Key " : " tag1 " , " Value " : " value1 " } ,
{ " Key " : " tag2 " , " Value " : " value2 " } ,
{ " Key " : " tag3 " , " Value " : " value3 " } ,
]
)
@mock_dax
def test_increase_replication_factor_unknown ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
with pytest . raises ( ClientError ) as exc :
client . increase_replication_factor (
ClusterName = " unknown " , NewReplicationFactor = 2
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " ClusterNotFoundFault " )
@mock_dax
def test_increase_replication_factor ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
name = " daxcluster "
cluster = client . create_cluster (
ClusterName = name ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 2 ,
IamRoleArn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX " ,
Tags = [
{ " Key " : " tag1 " , " Value " : " value1 " } ,
{ " Key " : " tag2 " , " Value " : " value2 " } ,
{ " Key " : " tag3 " , " Value " : " value3 " } ,
] ,
) [ " Cluster " ]
cluster [ " TotalNodes " ] . should . equal ( 2 )
new_cluster = client . increase_replication_factor (
ClusterName = name , NewReplicationFactor = 5
) [ " Cluster " ]
new_cluster [ " TotalNodes " ] . should . equal ( 5 )
new_cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
new_cluster [ " TotalNodes " ] . should . equal ( 5 )
# Progress cluster until it's available
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
node_ids = set ( [ n [ " NodeId " ] for n in cluster [ " Nodes " ] ] )
node_ids . should . equal (
{ f " { name } -a " , f " { name } -b " , f " { name } -c " , f " { name } -d " , f " { name } -e " }
)
@mock_dax
def test_decrease_replication_factor_unknown ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
with pytest . raises ( ClientError ) as exc :
client . decrease_replication_factor (
ClusterName = " unknown " , NewReplicationFactor = 2
)
err = exc . value . response [ " Error " ]
err [ " Code " ] . should . equal ( " ClusterNotFoundFault " )
@mock_dax
def test_decrease_replication_factor ( ) :
client = boto3 . client ( " dax " , region_name = " eu-west-1 " )
name = " daxcluster "
client . create_cluster (
ClusterName = name ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 5 ,
IamRoleArn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX " ,
Tags = [
{ " Key " : " tag1 " , " Value " : " value1 " } ,
{ " Key " : " tag2 " , " Value " : " value2 " } ,
{ " Key " : " tag3 " , " Value " : " value3 " } ,
] ,
)
new_cluster = client . decrease_replication_factor (
ClusterName = name , NewReplicationFactor = 3
) [ " Cluster " ]
new_cluster [ " TotalNodes " ] . should . equal ( 3 )
new_cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
new_cluster [ " TotalNodes " ] . should . equal ( 3 )
# Progress cluster until it's available
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
node_ids = set ( [ n [ " NodeId " ] for n in cluster [ " Nodes " ] ] )
node_ids . should . equal ( { f " { name } -a " , f " { name } -b " , f " { name } -c " } )
@mock_dax
def test_decrease_replication_factor_specific_nodeids ( ) :
client = boto3 . client ( " dax " , region_name = " ap-southeast-1 " )
name = " daxcluster "
client . create_cluster (
ClusterName = name ,
NodeType = " dax.t3.small " ,
ReplicationFactor = 5 ,
IamRoleArn = f " arn:aws:iam:: { ACCOUNT_ID } :role/aws-service-role/dax.amazonaws.com/AWSServiceRoleForDAX " ,
Tags = [
{ " Key " : " tag1 " , " Value " : " value1 " } ,
{ " Key " : " tag2 " , " Value " : " value2 " } ,
{ " Key " : " tag3 " , " Value " : " value3 " } ,
] ,
)
new_cluster = client . decrease_replication_factor (
ClusterName = name ,
NewReplicationFactor = 3 ,
NodeIdsToRemove = [ " daxcluster-b " , " daxcluster-c " ] ,
) [ " Cluster " ]
new_cluster [ " TotalNodes " ] . should . equal ( 3 )
new_cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
new_cluster [ " TotalNodes " ] . should . equal ( 3 )
# Progress cluster until it's available
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
cluster = client . describe_clusters ( ClusterNames = [ name ] ) [ " Clusters " ] [ 0 ]
node_ids = set ( [ n [ " NodeId " ] for n in cluster [ " Nodes " ] ] )
node_ids . should . equal ( { f " { name } -a " , f " { name } -d " , f " { name } -e " } )