Kinesis:update_stream_mode() (#5456)

This commit is contained in:
Bert Blommers 2022-09-08 21:51:55 +00:00 committed by GitHub
parent dbef197de8
commit f61ffc81ec
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 137 additions and 6 deletions

View File

@ -3638,7 +3638,7 @@
## kinesis ## kinesis
<details> <details>
<summary>89% implemented</summary> <summary>93% implemented</summary>
- [X] add_tags_to_stream - [X] add_tags_to_stream
- [X] create_stream - [X] create_stream
@ -3668,7 +3668,7 @@
- [X] stop_stream_encryption - [X] stop_stream_encryption
- [ ] subscribe_to_shard - [ ] subscribe_to_shard
- [X] update_shard_count - [X] update_shard_count
- [ ] update_stream_mode - [X] update_stream_mode
</details> </details>
## kinesis-video-archived-media ## kinesis-video-archived-media

View File

@ -57,5 +57,5 @@ kinesis
- [X] stop_stream_encryption - [X] stop_stream_encryption
- [ ] subscribe_to_shard - [ ] subscribe_to_shard
- [X] update_shard_count - [X] update_shard_count
- [ ] update_stream_mode - [X] update_stream_mode

View File

@ -82,3 +82,36 @@ class ValidationException(BadRequest):
"__type": "ValidationException", "__type": "ValidationException",
} }
) )
class RecordSizeExceedsLimit(BadRequest):
def __init__(self, position):
super().__init__()
self.description = json.dumps(
{
"message": f"1 validation error detected: Value at 'records.{position}.member.data' failed to satisfy constraint: Member must have length less than or equal to 1048576",
"__type": "ValidationException",
}
)
class TotalRecordsSizeExceedsLimit(BadRequest):
def __init__(self):
super().__init__()
self.description = json.dumps(
{
"message": "Records size exceeds 5 MB limit",
"__type": "InvalidArgumentException",
}
)
class TooManyRecords(BadRequest):
def __init__(self):
super().__init__()
self.description = json.dumps(
{
"message": "1 validation error detected: Value at 'records' failed to satisfy constraint: Member must have length less than or equal to 500",
"__type": "ValidationException",
}
)

View File

@ -21,6 +21,9 @@ from .exceptions import (
InvalidDecreaseRetention, InvalidDecreaseRetention,
InvalidIncreaseRetention, InvalidIncreaseRetention,
ValidationException, ValidationException,
RecordSizeExceedsLimit,
TotalRecordsSizeExceedsLimit,
TooManyRecords,
) )
from .utils import ( from .utils import (
compose_shard_iterator, compose_shard_iterator,
@ -411,6 +414,7 @@ class Stream(CloudFormationModel):
"EnhancedMonitoring": [{"ShardLevelMetrics": self.shard_level_metrics}], "EnhancedMonitoring": [{"ShardLevelMetrics": self.shard_level_metrics}],
"OpenShardCount": self.shard_count, "OpenShardCount": self.shard_count,
"EncryptionType": self.encryption_type, "EncryptionType": self.encryption_type,
"KeyId": self.key_id,
} }
} }
@ -542,7 +546,7 @@ class KinesisBackend(BaseBackend):
self.streams[stream_name] = stream self.streams[stream_name] = stream
return stream return stream
def describe_stream(self, stream_name): def describe_stream(self, stream_name) -> Stream:
if stream_name in self.streams: if stream_name in self.streams:
return self.streams[stream_name] return self.streams[stream_name]
else: else:
@ -622,6 +626,17 @@ class KinesisBackend(BaseBackend):
response = {"FailedRecordCount": 0, "Records": []} response = {"FailedRecordCount": 0, "Records": []}
if len(records) > 500:
raise TooManyRecords
data_sizes = [len(r.get("Data", "")) for r in records]
if sum(data_sizes) >= 5000000:
raise TotalRecordsSizeExceedsLimit
idx_over_limit = next(
(idx for idx, x in enumerate(data_sizes) if x >= 1048576), None
)
if idx_over_limit is not None:
raise RecordSizeExceedsLimit(position=idx_over_limit + 1)
for record in records: for record in records:
partition_key = record.get("PartitionKey") partition_key = record.get("PartitionKey")
explicit_hash_key = record.get("ExplicitHashKey") explicit_hash_key = record.get("ExplicitHashKey")
@ -821,5 +836,9 @@ class KinesisBackend(BaseBackend):
stream.encryption_type = "NONE" stream.encryption_type = "NONE"
stream.key_id = None stream.key_id = None
def update_stream_mode(self, stream_arn, stream_mode):
stream = self._find_stream_by_arn(stream_arn)
stream.stream_mode = stream_mode
kinesis_backends = BackendDict(KinesisBackend, "kinesis") kinesis_backends = BackendDict(KinesisBackend, "kinesis")

View File

@ -279,3 +279,9 @@ class KinesisResponse(BaseResponse):
stream_name = self.parameters.get("StreamName") stream_name = self.parameters.get("StreamName")
self.kinesis_backend.stop_stream_encryption(stream_name=stream_name) self.kinesis_backend.stop_stream_encryption(stream_name=stream_name)
return json.dumps(dict()) return json.dumps(dict())
def update_stream_mode(self):
stream_arn = self.parameters.get("StreamARN")
stream_mode = self.parameters.get("StreamModeDetails")
self.kinesis_backend.update_stream_mode(stream_arn, stream_mode)
return "{}"

View File

@ -135,8 +135,10 @@ iam:
iot: iot:
- TestAccIoTEndpointDataSource - TestAccIoTEndpointDataSource
kinesis: kinesis:
- TestAccKinesisStream_basic - TestAccKinesisStreamConsumerDataSource_
- TestAccKinesisStream_disappear - TestAccKinesisStreamConsumer_
- TestAccKinesisStreamDataSource_
- TestAccKinesisStream_
kms: kms:
- TestAccKMSAlias - TestAccKMSAlias
- TestAccKMSGrant_arn - TestAccKMSGrant_arn

View File

@ -36,6 +36,25 @@ def test_stream_creation_on_demand():
) )
@mock_kinesis
def test_update_stream_mode():
client = boto3.client("kinesis", region_name="eu-west-1")
resp = client.create_stream(
StreamName="my_stream", StreamModeDetails={"StreamMode": "ON_DEMAND"}
)
arn = client.describe_stream(StreamName="my_stream")["StreamDescription"][
"StreamARN"
]
client.update_stream_mode(
StreamARN=arn, StreamModeDetails={"StreamMode": "PROVISIONED"}
)
resp = client.describe_stream_summary(StreamName="my_stream")
stream = resp["StreamDescriptionSummary"]
stream.should.have.key("StreamModeDetails").equals({"StreamMode": "PROVISIONED"})
@mock_kinesis @mock_kinesis
def test_describe_non_existent_stream_boto3(): def test_describe_non_existent_stream_boto3():
client = boto3.client("kinesis", region_name="us-west-2") client = boto3.client("kinesis", region_name="us-west-2")

View File

@ -0,0 +1,52 @@
import boto3
import pytest
import sure # noqa # pylint: disable=unused-import
from botocore.exceptions import ClientError
from moto import mock_kinesis
@mock_kinesis
def test_record_data_exceeds_1mb():
client = boto3.client("kinesis", region_name="us-east-1")
client.create_stream(StreamName="my_stream", ShardCount=1)
with pytest.raises(ClientError) as exc:
client.put_records(
Records=[{"Data": b"a" * (2**20 + 1), "PartitionKey": "key"}],
StreamName="my_stream",
)
err = exc.value.response["Error"]
err["Code"].should.equal("ValidationException")
err["Message"].should.equal(
"1 validation error detected: Value at 'records.1.member.data' failed to satisfy constraint: Member must have length less than or equal to 1048576"
)
@mock_kinesis
def test_total_record_data_exceeds_5mb():
client = boto3.client("kinesis", region_name="us-east-1")
client.create_stream(StreamName="my_stream", ShardCount=1)
with pytest.raises(ClientError) as exc:
client.put_records(
Records=[{"Data": b"a" * 2**20, "PartitionKey": "key"}] * 5,
StreamName="my_stream",
)
err = exc.value.response["Error"]
err["Code"].should.equal("InvalidArgumentException")
err["Message"].should.equal("Records size exceeds 5 MB limit")
@mock_kinesis
def test_too_many_records():
client = boto3.client("kinesis", region_name="us-east-1")
client.create_stream(StreamName="my_stream", ShardCount=1)
with pytest.raises(ClientError) as exc:
client.put_records(
Records=[{"Data": b"a", "PartitionKey": "key"}] * 501,
StreamName="my_stream",
)
err = exc.value.response["Error"]
err["Code"].should.equal("ValidationException")
err["Message"].should.equal(
"1 validation error detected: Value at 'records' failed to satisfy constraint: Member must have length less than or equal to 500"
)