From 5cd288b42ca90053a0a5d7853fa680d9f0171aef Mon Sep 17 00:00:00 2001
From: Pepijn <83588709+pepijn-motosumo@users.noreply.github.com>
Date: Tue, 10 Oct 2023 09:39:55 +0200
Subject: [PATCH] Rudimentary support for IVS (#6894)
---
IMPLEMENTATION_COVERAGE.md | 37 +++-
docs/docs/services/ivs.rst | 60 ++++++
moto/__init__.py | 1 +
moto/backend_index.py | 1 +
moto/ivs/__init__.py | 5 +
moto/ivs/exceptions.py | 9 +
moto/ivs/models.py | 134 ++++++++++++
moto/ivs/responses.py | 93 +++++++++
moto/ivs/urls.py | 20 ++
moto/moto_api/_internal/moto_random.py | 6 +
tests/test_ivs/__init__.py | 0
tests/test_ivs/test_ivs.py | 192 ++++++++++++++++++
.../mock_random/test_mock_random.py | 10 +-
13 files changed, 565 insertions(+), 3 deletions(-)
create mode 100644 docs/docs/services/ivs.rst
create mode 100644 moto/ivs/__init__.py
create mode 100644 moto/ivs/exceptions.py
create mode 100644 moto/ivs/models.py
create mode 100644 moto/ivs/responses.py
create mode 100644 moto/ivs/urls.py
create mode 100644 tests/test_ivs/__init__.py
create mode 100644 tests/test_ivs/test_ivs.py
diff --git a/IMPLEMENTATION_COVERAGE.md b/IMPLEMENTATION_COVERAGE.md
index 086121c49..34e5703b0 100644
--- a/IMPLEMENTATION_COVERAGE.md
+++ b/IMPLEMENTATION_COVERAGE.md
@@ -4105,6 +4105,42 @@
- [X] update_thing_shadow
+## ivs
+
+20% implemented
+
+- [X] batch_get_channel
+- [ ] batch_get_stream_key
+- [ ] batch_start_viewer_session_revocation
+- [X] create_channel
+- [ ] create_recording_configuration
+- [ ] create_stream_key
+- [X] delete_channel
+- [ ] delete_playback_key_pair
+- [ ] delete_recording_configuration
+- [ ] delete_stream_key
+- [X] get_channel
+- [ ] get_playback_key_pair
+- [ ] get_recording_configuration
+- [ ] get_stream
+- [ ] get_stream_key
+- [ ] get_stream_session
+- [ ] import_playback_key_pair
+- [X] list_channels
+- [ ] list_playback_key_pairs
+- [ ] list_recording_configurations
+- [ ] list_stream_keys
+- [ ] list_stream_sessions
+- [ ] list_streams
+- [ ] list_tags_for_resource
+- [ ] put_metadata
+- [ ] start_viewer_session_revocation
+- [ ] stop_stream
+- [ ] tag_resource
+- [ ] untag_resource
+- [X] update_channel
+
+
## kinesis
93% implemented
@@ -7481,7 +7517,6 @@
- iotthingsgraph
- iottwinmaker
- iotwireless
-- ivs
- ivs-realtime
- ivschat
- kafka
diff --git a/docs/docs/services/ivs.rst b/docs/docs/services/ivs.rst
new file mode 100644
index 000000000..e3b0f66e3
--- /dev/null
+++ b/docs/docs/services/ivs.rst
@@ -0,0 +1,60 @@
+.. _implementedservice_ivs:
+
+.. |start-h3| raw:: html
+
+
+
+.. |end-h3| raw:: html
+
+
+
+===
+ivs
+===
+
+.. autoclass:: moto.ivs.models.IVSBackend
+
+|start-h3| Example usage |end-h3|
+
+.. sourcecode:: python
+
+ @mock_ivs
+ def test_ivs_behaviour:
+ boto3.client("ivs")
+ ...
+
+
+
+|start-h3| Implemented features for this service |end-h3|
+
+- [X] batch_get_channel
+- [ ] batch_get_stream_key
+- [ ] batch_start_viewer_session_revocation
+- [X] create_channel
+- [ ] create_recording_configuration
+- [ ] create_stream_key
+- [X] delete_channel
+- [ ] delete_playback_key_pair
+- [ ] delete_recording_configuration
+- [ ] delete_stream_key
+- [X] get_channel
+- [ ] get_playback_key_pair
+- [ ] get_recording_configuration
+- [ ] get_stream
+- [ ] get_stream_key
+- [ ] get_stream_session
+- [ ] import_playback_key_pair
+- [X] list_channels
+- [ ] list_playback_key_pairs
+- [ ] list_recording_configurations
+- [ ] list_stream_keys
+- [ ] list_stream_sessions
+- [ ] list_streams
+- [ ] list_tags_for_resource
+- [ ] put_metadata
+- [ ] start_viewer_session_revocation
+- [ ] stop_stream
+- [ ] tag_resource
+- [ ] untag_resource
+- [X] update_channel
+
diff --git a/moto/__init__.py b/moto/__init__.py
index d6401ddb8..2c5e14720 100644
--- a/moto/__init__.py
+++ b/moto/__init__.py
@@ -106,6 +106,7 @@ mock_iam = lazy_load(".iam", "mock_iam")
mock_identitystore = lazy_load(".identitystore", "mock_identitystore")
mock_iot = lazy_load(".iot", "mock_iot")
mock_iotdata = lazy_load(".iotdata", "mock_iotdata", boto3_name="iot-data")
+mock_ivs = lazy_load(".ivs", "mock_ivs", boto3_name="ivs")
mock_kinesis = lazy_load(".kinesis", "mock_kinesis")
mock_kinesisvideo = lazy_load(".kinesisvideo", "mock_kinesisvideo")
mock_kinesisvideoarchivedmedia = lazy_load(
diff --git a/moto/backend_index.py b/moto/backend_index.py
index faf763d1d..a55cbdd3f 100644
--- a/moto/backend_index.py
+++ b/moto/backend_index.py
@@ -88,6 +88,7 @@ backend_url_patterns = [
("iot", re.compile("https?://iot\\.(.+)\\.amazonaws\\.com")),
("iot-data", re.compile("https?://data\\.iot\\.(.+)\\.amazonaws.com")),
("iot-data", re.compile("https?://data-ats\\.iot\\.(.+)\\.amazonaws.com")),
+ ("ivs", re.compile("https?://ivs\\.(.+)\\.amazonaws\\.com")),
("kinesis", re.compile("https?://kinesis\\.(.+)\\.amazonaws\\.com")),
("kinesis", re.compile("https?://(.+)\\.control-kinesis\\.(.+)\\.amazonaws\\.com")),
("kinesis", re.compile("https?://(.+)\\.data-kinesis\\.(.+)\\.amazonaws\\.com")),
diff --git a/moto/ivs/__init__.py b/moto/ivs/__init__.py
new file mode 100644
index 000000000..e68e33550
--- /dev/null
+++ b/moto/ivs/__init__.py
@@ -0,0 +1,5 @@
+"""ivs module initialization; sets value for base decorator."""
+from .models import ivs_backends
+from ..core.models import base_decorator
+
+mock_ivs = base_decorator(ivs_backends)
diff --git a/moto/ivs/exceptions.py b/moto/ivs/exceptions.py
new file mode 100644
index 000000000..73771dd07
--- /dev/null
+++ b/moto/ivs/exceptions.py
@@ -0,0 +1,9 @@
+"""Exceptions raised by the ivs service."""
+from moto.core.exceptions import JsonRESTError
+
+
+class ResourceNotFoundException(JsonRESTError):
+ code = 404
+
+ def __init__(self, message: str):
+ super().__init__("ResourceNotFoundException", message)
diff --git a/moto/ivs/models.py b/moto/ivs/models.py
new file mode 100644
index 000000000..ea762d44c
--- /dev/null
+++ b/moto/ivs/models.py
@@ -0,0 +1,134 @@
+"""IVSBackend class with methods for supported APIs."""
+from typing import Optional, Any, List, Dict, Tuple
+from moto.core import BaseBackend, BackendDict
+from moto.ivs.exceptions import ResourceNotFoundException
+from moto.utilities.paginator import paginate
+from moto.moto_api._internal import mock_random
+
+
+class IVSBackend(BaseBackend):
+ """Implementation of IVS APIs."""
+
+ PAGINATION_MODEL = {
+ "list_channels": {
+ "input_token": "next_token",
+ "limit_key": "max_results",
+ "limit_default": 100,
+ "unique_attribute": "arn",
+ },
+ }
+
+ def __init__(self, region_name: str, account_id: str):
+ super().__init__(region_name, account_id)
+ self.channels: List[Dict[str, Any]] = []
+
+ def create_channel(
+ self,
+ authorized: bool,
+ insecure_ingest: bool,
+ latency_mode: str,
+ name: str,
+ preset: str,
+ recording_configuration_arn: str,
+ tags: Dict[str, str],
+ channel_type: str,
+ ) -> Tuple[Dict[str, Any], Dict[str, Any]]:
+ channel_id = mock_random.get_random_string(12)
+ channel_arn = (
+ f"arn:aws:ivs:{self.region_name}:{self.account_id}:channel/{channel_id}"
+ )
+ channel = {
+ "arn": channel_arn,
+ "authorized": authorized,
+ "ingestEndpoint": "ingest.example.com",
+ "insecureIngest": insecure_ingest,
+ "latencyMode": latency_mode,
+ "name": name,
+ "playbackUrl": f"https://playback.example.com/{self.region_name}.{self.account_id}.{channel_id}.m3u8",
+ "preset": preset,
+ "recordingConfigurationArn": recording_configuration_arn,
+ "tags": tags,
+ "type": channel_type,
+ }
+ self.channels.append(channel)
+ stream_key_id = mock_random.get_random_string(12)
+ stream_key_arn = f"arn:aws:ivs:{self.region_name}:{self.account_id}:stream-key/{stream_key_id}"
+ stream_key = {
+ "arn": stream_key_arn,
+ "channelArn": channel_arn,
+ "tags": tags,
+ "value": f"sk_{self.region_name}_{mock_random.token_urlsafe(32)}",
+ }
+ return channel, stream_key
+
+ @paginate(pagination_model=PAGINATION_MODEL) # type: ignore[misc]
+ def list_channels(
+ self,
+ filter_by_name: Optional[str],
+ filter_by_recording_configuration_arn: Optional[str],
+ ) -> List[Dict[str, Any]]:
+ if filter_by_name is not None:
+ channels = [
+ channel
+ for channel in self.channels
+ if channel["name"] == filter_by_name
+ ]
+ elif filter_by_recording_configuration_arn is not None:
+ channels = [
+ channel
+ for channel in self.channels
+ if channel["recordingConfigurationArn"]
+ == filter_by_recording_configuration_arn
+ ]
+ else:
+ channels = self.channels
+ return channels
+
+ def _find_channel(self, arn: str) -> Dict[str, Any]:
+ try:
+ return next(channel for channel in self.channels if channel["arn"] == arn)
+ except StopIteration:
+ raise ResourceNotFoundException(f"Resource: {arn} not found")
+
+ def get_channel(self, arn: str) -> Dict[str, Any]:
+ return self._find_channel(arn)
+
+ def batch_get_channel(
+ self, arns: List[str]
+ ) -> Tuple[List[Dict[str, Any]], List[Dict[str, str]]]:
+ return [channel for channel in self.channels if channel["arn"] in arns], []
+
+ def update_channel(
+ self,
+ arn: str,
+ authorized: Optional[bool],
+ insecure_ingest: Optional[bool],
+ latency_mode: Optional[str],
+ name: Optional[str],
+ preset: Optional[str],
+ recording_configuration_arn: Optional[str],
+ channel_type: Optional[str],
+ ) -> Dict[str, Any]:
+ channel = self._find_channel(arn)
+ if authorized is not None:
+ channel["authorized"] = authorized
+ if insecure_ingest is not None:
+ channel["insecureIngest"] = insecure_ingest
+ if latency_mode is not None:
+ channel["latencyMode"] = latency_mode
+ if name is not None:
+ channel["name"] = name
+ if preset is not None:
+ channel["preset"] = preset
+ if recording_configuration_arn is not None:
+ channel["recordingConfigurationArn"] = recording_configuration_arn
+ if channel_type is not None:
+ channel["type"] = channel_type
+ return channel
+
+ def delete_channel(self, arn: str) -> None:
+ self._find_channel(arn)
+ self.channels = [channel for channel in self.channels if channel["arn"] != arn]
+
+
+ivs_backends = BackendDict(IVSBackend, "ivs")
diff --git a/moto/ivs/responses.py b/moto/ivs/responses.py
new file mode 100644
index 000000000..a15a7f1c8
--- /dev/null
+++ b/moto/ivs/responses.py
@@ -0,0 +1,93 @@
+"""Handles incoming ivs requests, invokes methods, returns responses."""
+import json
+from moto.core.responses import BaseResponse
+from .models import IVSBackend, ivs_backends
+
+
+class IVSResponse(BaseResponse):
+ """Handler for IVS requests and responses."""
+
+ def __init__(self) -> None:
+ super().__init__(service_name="ivs")
+
+ @property
+ def ivs_backend(self) -> IVSBackend:
+ """Return backend instance specific for this region."""
+ return ivs_backends[self.current_account][self.region]
+
+ def create_channel(self) -> str:
+ authorized = self._get_param("authorized", False)
+ insecure_ingest = self._get_param("insecureIngest", False)
+ latency_mode = self._get_param("latencyMode", "LOW")
+ name = self._get_param("name")
+ preset = self._get_param("preset", "")
+ recording_configuration_arn = self._get_param("recordingConfigurationArn", "")
+ tags = self._get_param("tags", {})
+ channel_type = self._get_param("type", "STANDARD")
+ channel, stream_key = self.ivs_backend.create_channel(
+ authorized=authorized,
+ insecure_ingest=insecure_ingest,
+ latency_mode=latency_mode,
+ name=name,
+ preset=preset,
+ recording_configuration_arn=recording_configuration_arn,
+ tags=tags,
+ channel_type=channel_type,
+ )
+ return json.dumps(dict(channel=channel, streamKey=stream_key))
+
+ def list_channels(self) -> str:
+ filter_by_name = self._get_param("filterByName")
+ filter_by_recording_configuration_arn = self._get_param(
+ "filterByRecordingConfigurationArn"
+ )
+ max_results = self._get_param("maxResults")
+ next_token = self._get_param("nextToken")
+ channels, next_token = self.ivs_backend.list_channels(
+ filter_by_name=filter_by_name,
+ filter_by_recording_configuration_arn=filter_by_recording_configuration_arn,
+ max_results=max_results,
+ next_token=next_token,
+ )
+ return json.dumps(dict(channels=channels, nextToken=next_token))
+
+ def get_channel(self) -> str:
+ arn = self._get_param("arn")
+ channel = self.ivs_backend.get_channel(
+ arn=arn,
+ )
+ return json.dumps(dict(channel=channel))
+
+ def batch_get_channel(self) -> str:
+ arns = self._get_param("arns")
+ channels, errors = self.ivs_backend.batch_get_channel(
+ arns=arns,
+ )
+ return json.dumps(dict(channels=channels, errors=errors))
+
+ def update_channel(self) -> str:
+ arn = self._get_param("arn")
+ authorized = self._get_param("authorized")
+ insecure_ingest = self._get_param("insecureIngest")
+ latency_mode = self._get_param("latencyMode")
+ name = self._get_param("name")
+ preset = self._get_param("preset")
+ recording_configuration_arn = self._get_param("recordingConfigurationArn")
+ channel_type = self._get_param("type")
+ channel = self.ivs_backend.update_channel(
+ arn=arn,
+ authorized=authorized,
+ insecure_ingest=insecure_ingest,
+ latency_mode=latency_mode,
+ name=name,
+ preset=preset,
+ recording_configuration_arn=recording_configuration_arn,
+ channel_type=channel_type,
+ )
+ return json.dumps(dict(channel=channel))
+
+ def delete_channel(self) -> None:
+ arn = self._get_param("arn")
+ self.ivs_backend.delete_channel(
+ arn=arn,
+ )
diff --git a/moto/ivs/urls.py b/moto/ivs/urls.py
new file mode 100644
index 000000000..7c4a85267
--- /dev/null
+++ b/moto/ivs/urls.py
@@ -0,0 +1,20 @@
+"""ivs base URL and path."""
+from .responses import IVSResponse
+
+
+url_bases = [
+ r"https?://ivs\.(.+)\.amazonaws\.com",
+]
+
+
+response = IVSResponse()
+
+
+url_paths = {
+ "{0}/CreateChannel": response.dispatch,
+ "{0}/ListChannels": response.dispatch,
+ "{0}/GetChannel": response.dispatch,
+ "{0}/BatchGetChannel": response.dispatch,
+ "{0}/UpdateChannel": response.dispatch,
+ "{0}/DeleteChannel": response.dispatch,
+}
diff --git a/moto/moto_api/_internal/moto_random.py b/moto/moto_api/_internal/moto_random.py
index 93f7f9c8d..0bd405b67 100644
--- a/moto/moto_api/_internal/moto_random.py
+++ b/moto/moto_api/_internal/moto_random.py
@@ -1,6 +1,7 @@
from random import Random
import string
from uuid import UUID
+from base64 import urlsafe_b64encode
HEX_CHARS = list(range(10)) + ["a", "b", "c", "d", "e", "f"]
@@ -30,3 +31,8 @@ class MotoRandom(Random):
pool += string.digits
random_str = "".join([self.choice(pool) for i in range(length)])
return random_str.lower() if lower_case else random_str
+
+ # Based on https://github.com/python/cpython/blob/main/Lib/secrets.py
+ def token_urlsafe(self, nbytes: int) -> str:
+ randbytes = self.getrandbits(nbytes * 8).to_bytes(nbytes, "little")
+ return urlsafe_b64encode(randbytes).rstrip(b"=").decode("ascii")
diff --git a/tests/test_ivs/__init__.py b/tests/test_ivs/__init__.py
new file mode 100644
index 000000000..e69de29bb
diff --git a/tests/test_ivs/test_ivs.py b/tests/test_ivs/test_ivs.py
new file mode 100644
index 000000000..b5a2f1867
--- /dev/null
+++ b/tests/test_ivs/test_ivs.py
@@ -0,0 +1,192 @@
+"""Unit tests for ivs-supported APIs."""
+import boto3
+from botocore.exceptions import ClientError
+from pytest import raises
+from moto import mock_ivs
+from re import fullmatch
+
+
+@mock_ivs
+def test_create_channel_with_name():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ assert create_response["channel"]["name"] == "foo"
+
+
+@mock_ivs
+def test_create_channel_defaults():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ assert create_response["channel"]["authorized"] is False
+ assert create_response["channel"]["insecureIngest"] is False
+ assert create_response["channel"]["latencyMode"] == "LOW"
+ assert create_response["channel"]["preset"] == ""
+ assert create_response["channel"]["recordingConfigurationArn"] == ""
+ assert create_response["channel"]["tags"] == {}
+ assert create_response["channel"]["type"] == "STANDARD"
+ assert create_response["streamKey"]["tags"] == {}
+
+
+@mock_ivs
+def test_create_channel_generated_values():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ assert fullmatch(r"arn:aws:ivs:.*:channel/.*", create_response["channel"]["arn"])
+ assert create_response["channel"]["ingestEndpoint"]
+ assert create_response["channel"]["playbackUrl"]
+ assert fullmatch(
+ r"arn:aws:ivs:.*:stream-key/.*", create_response["streamKey"]["arn"]
+ )
+ assert (
+ create_response["streamKey"]["channelArn"] == create_response["channel"]["arn"]
+ )
+ assert fullmatch(r"sk_.*", create_response["streamKey"]["value"])
+
+
+@mock_ivs
+def test_create_channel_with_name_and_recording_configuration():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(
+ name="foo",
+ recordingConfigurationArn="blah",
+ )
+ assert create_response["channel"]["name"] == "foo"
+ assert create_response["channel"]["recordingConfigurationArn"] == "blah"
+
+
+@mock_ivs
+def test_list_channels_empty():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ list_response = client.list_channels()
+ assert list_response["channels"] == []
+
+
+@mock_ivs
+def test_list_channels_one():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ client.create_channel(name="foo")
+ list_response = client.list_channels()
+ assert len(list_response["channels"]) == 1
+ assert list_response["channels"][0]["name"] == "foo"
+
+
+@mock_ivs
+def test_list_channels_two():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ client.create_channel(name="foo")
+ client.create_channel(
+ name="bar",
+ recordingConfigurationArn="blah",
+ )
+ list_response = client.list_channels()
+ assert len(list_response["channels"]) == 2
+ assert list_response["channels"][0]["name"] == "foo"
+ assert list_response["channels"][1]["name"] == "bar"
+
+
+@mock_ivs
+def test_list_channels_by_name():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ client.create_channel(name="foo")
+ client.create_channel(
+ name="bar",
+ recordingConfigurationArn="blah",
+ )
+ list_response = client.list_channels(filterByName="foo")
+ assert len(list_response["channels"]) == 1
+ assert list_response["channels"][0]["name"] == "foo"
+
+
+@mock_ivs
+def test_list_channels_by_recording_configuration():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ client.create_channel(name="foo")
+ client.create_channel(
+ name="bar",
+ recordingConfigurationArn="blah",
+ )
+ list_response = client.list_channels(filterByRecordingConfigurationArn="blah")
+ assert len(list_response["channels"]) == 1
+ assert list_response["channels"][0]["name"] == "bar"
+
+
+@mock_ivs
+def test_list_channels_pagination():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ client.create_channel(name="foo")
+ client.create_channel(name="bar")
+ first_list_response = client.list_channels(maxResults=1)
+ assert len(first_list_response["channels"]) == 1
+ assert "nextToken" in first_list_response
+ second_list_response = client.list_channels(
+ maxResults=1, nextToken=first_list_response["nextToken"]
+ )
+ assert len(second_list_response["channels"]) == 1
+ assert "nextToken" not in second_list_response
+
+
+@mock_ivs
+def test_get_channel_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ get_response = client.get_channel(arn=create_response["channel"]["arn"])
+ assert get_response["channel"]["name"] == "foo"
+
+
+@mock_ivs
+def test_get_channel_not_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ with raises(ClientError) as exc:
+ client.get_channel(arn="nope")
+ assert exc.value.response["Error"]["Code"] == "ResourceNotFoundException"
+
+
+@mock_ivs
+def test_batch_get_channel():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ batch_get_response = client.batch_get_channel(
+ arns=[create_response["channel"]["arn"]]
+ )
+ assert len(batch_get_response["channels"]) == 1
+ assert batch_get_response["channels"][0]["name"] == "foo"
+
+
+@mock_ivs
+def test_update_channel_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(
+ name="foo",
+ recordingConfigurationArn="blah",
+ )
+ update_response = client.update_channel(
+ arn=create_response["channel"]["arn"],
+ name="bar",
+ )
+ assert update_response["channel"]["name"] == "bar"
+ assert update_response["channel"]["recordingConfigurationArn"] == "blah"
+
+
+@mock_ivs
+def test_update_channel_not_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ with raises(ClientError) as exc:
+ client.update_channel(arn="nope", name="bar")
+ assert exc.value.response["Error"]["Code"] == "ResourceNotFoundException"
+
+
+@mock_ivs
+def test_delete_channel_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ create_response = client.create_channel(name="foo")
+ client.delete_channel(arn=create_response["channel"]["arn"])
+ list_response = client.list_channels()
+ assert list_response["channels"] == []
+
+
+@mock_ivs
+def test_delete_channel_not_exists():
+ client = boto3.client("ivs", region_name="eu-west-1")
+ with raises(ClientError) as exc:
+ client.delete_channel(arn="nope")
+ assert exc.value.response["Error"]["Code"] == "ResourceNotFoundException"
diff --git a/tests/test_moto_api/mock_random/test_mock_random.py b/tests/test_moto_api/mock_random/test_mock_random.py
index dd415a644..8c5c4d120 100644
--- a/tests/test_moto_api/mock_random/test_mock_random.py
+++ b/tests/test_moto_api/mock_random/test_mock_random.py
@@ -31,7 +31,13 @@ def test_semi_random_hex_strings():
# Ensure they are different
assert fixed_hex != random_hex
- # Retrieving another 'fixed' UUID should not return a known UUID
- second_hex = mock_random.uuid4()
+ # Retrieving another 'fixed' HEX should not return a known HEX
+ second_hex = mock_random.get_random_hex()
assert second_hex != random_hex
assert second_hex != fixed_hex
+
+
+def test_semi_random_token_urlsafe():
+ mock_random.seed(42)
+ token = mock_random.token_urlsafe(32)
+ assert token == "nXmxo38xgBzRGmcG-0DWvVdSaEaQO7E-3lYkOenBuCM"