Service: Scheduler (#6197)
This commit is contained in:
parent
01e2d11ed3
commit
b4346e2eea
@ -6172,6 +6172,24 @@
|
|||||||
- [ ] update_workteam
|
- [ ] update_workteam
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
## scheduler
|
||||||
|
<details>
|
||||||
|
<summary>100% implemented</summary>
|
||||||
|
|
||||||
|
- [X] create_schedule
|
||||||
|
- [X] create_schedule_group
|
||||||
|
- [X] delete_schedule
|
||||||
|
- [X] delete_schedule_group
|
||||||
|
- [X] get_schedule
|
||||||
|
- [X] get_schedule_group
|
||||||
|
- [X] list_schedule_groups
|
||||||
|
- [X] list_schedules
|
||||||
|
- [X] list_tags_for_resource
|
||||||
|
- [X] tag_resource
|
||||||
|
- [X] untag_resource
|
||||||
|
- [X] update_schedule
|
||||||
|
</details>
|
||||||
|
|
||||||
## sdb
|
## sdb
|
||||||
<details>
|
<details>
|
||||||
<summary>50% implemented</summary>
|
<summary>50% implemented</summary>
|
||||||
@ -7081,7 +7099,6 @@
|
|||||||
- sagemaker-metrics
|
- sagemaker-metrics
|
||||||
- sagemaker-runtime
|
- sagemaker-runtime
|
||||||
- savingsplans
|
- savingsplans
|
||||||
- scheduler
|
|
||||||
- schemas
|
- schemas
|
||||||
- securityhub
|
- securityhub
|
||||||
- securitylake
|
- securitylake
|
||||||
|
58
docs/docs/services/scheduler.rst
Normal file
58
docs/docs/services/scheduler.rst
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
.. _implementedservice_scheduler:
|
||||||
|
|
||||||
|
.. |start-h3| raw:: html
|
||||||
|
|
||||||
|
<h3>
|
||||||
|
|
||||||
|
.. |end-h3| raw:: html
|
||||||
|
|
||||||
|
</h3>
|
||||||
|
|
||||||
|
=========
|
||||||
|
scheduler
|
||||||
|
=========
|
||||||
|
|
||||||
|
.. autoclass:: moto.scheduler.models.EventBridgeSchedulerBackend
|
||||||
|
|
||||||
|
|start-h3| Example usage |end-h3|
|
||||||
|
|
||||||
|
.. sourcecode:: python
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_scheduler_behaviour:
|
||||||
|
boto3.client("scheduler")
|
||||||
|
...
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|start-h3| Implemented features for this service |end-h3|
|
||||||
|
|
||||||
|
- [X] create_schedule
|
||||||
|
|
||||||
|
The ClientToken parameter is not yet implemented
|
||||||
|
|
||||||
|
|
||||||
|
- [X] create_schedule_group
|
||||||
|
|
||||||
|
The ClientToken parameter is not yet implemented
|
||||||
|
|
||||||
|
|
||||||
|
- [X] delete_schedule
|
||||||
|
- [X] delete_schedule_group
|
||||||
|
- [X] get_schedule
|
||||||
|
- [X] get_schedule_group
|
||||||
|
- [X] list_schedule_groups
|
||||||
|
|
||||||
|
The MaxResults-parameter and pagination options are not yet implemented
|
||||||
|
|
||||||
|
|
||||||
|
- [ ] list_schedules
|
||||||
|
- [X] list_tags_for_resource
|
||||||
|
- [X] tag_resource
|
||||||
|
- [X] untag_resource
|
||||||
|
- [X] update_schedule
|
||||||
|
|
||||||
|
The ClientToken is not yet implemented
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -143,6 +143,7 @@ mock_route53resolver = lazy_load(
|
|||||||
mock_s3 = lazy_load(".s3", "mock_s3")
|
mock_s3 = lazy_load(".s3", "mock_s3")
|
||||||
mock_s3control = lazy_load(".s3control", "mock_s3control")
|
mock_s3control = lazy_load(".s3control", "mock_s3control")
|
||||||
mock_sagemaker = lazy_load(".sagemaker", "mock_sagemaker")
|
mock_sagemaker = lazy_load(".sagemaker", "mock_sagemaker")
|
||||||
|
mock_scheduler = lazy_load(".scheduler", "mock_scheduler")
|
||||||
mock_sdb = lazy_load(".sdb", "mock_sdb")
|
mock_sdb = lazy_load(".sdb", "mock_sdb")
|
||||||
mock_secretsmanager = lazy_load(".secretsmanager", "mock_secretsmanager")
|
mock_secretsmanager = lazy_load(".secretsmanager", "mock_secretsmanager")
|
||||||
mock_servicequotas = lazy_load(
|
mock_servicequotas = lazy_load(
|
||||||
|
@ -148,6 +148,7 @@ backend_url_patterns = [
|
|||||||
re.compile("https?://([0-9]+)\\.s3-control\\.(.+)\\.amazonaws\\.com"),
|
re.compile("https?://([0-9]+)\\.s3-control\\.(.+)\\.amazonaws\\.com"),
|
||||||
),
|
),
|
||||||
("sagemaker", re.compile("https?://api\\.sagemaker\\.(.+)\\.amazonaws.com")),
|
("sagemaker", re.compile("https?://api\\.sagemaker\\.(.+)\\.amazonaws.com")),
|
||||||
|
("scheduler", re.compile("https?://scheduler\\.(.+)\\.amazonaws\\.com")),
|
||||||
("sdb", re.compile("https?://sdb\\.(.+)\\.amazonaws\\.com")),
|
("sdb", re.compile("https?://sdb\\.(.+)\\.amazonaws\\.com")),
|
||||||
("secretsmanager", re.compile("https?://secretsmanager\\.(.+)\\.amazonaws\\.com")),
|
("secretsmanager", re.compile("https?://secretsmanager\\.(.+)\\.amazonaws\\.com")),
|
||||||
(
|
(
|
||||||
|
5
moto/scheduler/__init__.py
Normal file
5
moto/scheduler/__init__.py
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
"""scheduler module initialization; sets value for base decorator."""
|
||||||
|
from .models import scheduler_backends
|
||||||
|
from ..core.models import base_decorator
|
||||||
|
|
||||||
|
mock_scheduler = base_decorator(scheduler_backends)
|
12
moto/scheduler/exceptions.py
Normal file
12
moto/scheduler/exceptions.py
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
"""Exceptions raised by the scheduler service."""
|
||||||
|
from moto.core.exceptions import JsonRESTError
|
||||||
|
|
||||||
|
|
||||||
|
class ScheduleNotFound(JsonRESTError):
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__("ResourceNotFoundException", "Schedule not found")
|
||||||
|
|
||||||
|
|
||||||
|
class ScheduleGroupNotFound(JsonRESTError):
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__("ResourceNotFoundException", "ScheduleGroup not found")
|
240
moto/scheduler/models.py
Normal file
240
moto/scheduler/models.py
Normal file
@ -0,0 +1,240 @@
|
|||||||
|
"""EventBridgeSchedulerBackend class with methods for supported APIs."""
|
||||||
|
from typing import Any, Dict, List, Iterable, Optional
|
||||||
|
|
||||||
|
from moto.core import BaseBackend, BackendDict, BaseModel
|
||||||
|
from moto.core.utils import unix_time
|
||||||
|
from moto.utilities.tagging_service import TaggingService
|
||||||
|
|
||||||
|
from .exceptions import ScheduleNotFound, ScheduleGroupNotFound
|
||||||
|
|
||||||
|
|
||||||
|
class Schedule(BaseModel):
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
region: str,
|
||||||
|
account_id: str,
|
||||||
|
group_name: str,
|
||||||
|
name: str,
|
||||||
|
description: Optional[str],
|
||||||
|
schedule_expression: str,
|
||||||
|
schedule_expression_timezone: Optional[str],
|
||||||
|
flexible_time_window: Dict[str, Any],
|
||||||
|
target: Dict[str, Any],
|
||||||
|
state: Optional[str],
|
||||||
|
kms_key_arn: Optional[str],
|
||||||
|
start_date: Optional[str],
|
||||||
|
end_date: Optional[str],
|
||||||
|
):
|
||||||
|
self.name = name
|
||||||
|
self.group_name = group_name
|
||||||
|
self.description = description
|
||||||
|
self.arn = (
|
||||||
|
f"arn:aws:scheduler:{region}:{account_id}:schedule/{group_name}/{name}"
|
||||||
|
)
|
||||||
|
self.schedule_expression = schedule_expression
|
||||||
|
self.schedule_expression_timezone = schedule_expression_timezone
|
||||||
|
self.flexible_time_window = flexible_time_window
|
||||||
|
self.target = Schedule.validate_target(target)
|
||||||
|
self.state = state or "ENABLED"
|
||||||
|
self.kms_key_arn = kms_key_arn
|
||||||
|
self.start_date = start_date
|
||||||
|
self.end_date = end_date
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def validate_target(target: Dict[str, Any]) -> Dict[str, Any]: # type: ignore[misc]
|
||||||
|
if "RetryPolicy" not in target:
|
||||||
|
target["RetryPolicy"] = {
|
||||||
|
"MaximumEventAgeInSeconds": 86400,
|
||||||
|
"MaximumRetryAttempts": 185,
|
||||||
|
}
|
||||||
|
return target
|
||||||
|
|
||||||
|
def to_dict(self, short: bool = False) -> Dict[str, Any]:
|
||||||
|
dct: Dict[str, Any] = {
|
||||||
|
"Arn": self.arn,
|
||||||
|
"Name": self.name,
|
||||||
|
"GroupName": self.group_name,
|
||||||
|
"Description": self.description,
|
||||||
|
"ScheduleExpression": self.schedule_expression,
|
||||||
|
"ScheduleExpressionTimezone": self.schedule_expression_timezone,
|
||||||
|
"FlexibleTimeWindow": self.flexible_time_window,
|
||||||
|
"Target": self.target,
|
||||||
|
"State": self.state,
|
||||||
|
"KmsKeyArn": self.kms_key_arn,
|
||||||
|
"StartDate": self.start_date,
|
||||||
|
"EndDate": self.end_date,
|
||||||
|
}
|
||||||
|
if short:
|
||||||
|
dct["Target"] = {"Arn": dct["Target"]["Arn"]}
|
||||||
|
return dct
|
||||||
|
|
||||||
|
|
||||||
|
class ScheduleGroup(BaseModel):
|
||||||
|
def __init__(self, region: str, account_id: str, name: str):
|
||||||
|
self.name = name
|
||||||
|
self.arn = f"arn:aws:scheduler:{region}:{account_id}:schedule-group/{name}"
|
||||||
|
self.schedules: Dict[str, Schedule] = dict()
|
||||||
|
self.created_on = None if self.name == "default" else unix_time()
|
||||||
|
self.last_modified = None if self.name == "default" else unix_time()
|
||||||
|
|
||||||
|
def add_schedule(self, schedule: Schedule) -> None:
|
||||||
|
self.schedules[schedule.name] = schedule
|
||||||
|
|
||||||
|
def get_schedule(self, name: str) -> Schedule:
|
||||||
|
if name not in self.schedules:
|
||||||
|
raise ScheduleNotFound
|
||||||
|
return self.schedules[name]
|
||||||
|
|
||||||
|
def delete_schedule(self, name: str) -> None:
|
||||||
|
self.schedules.pop(name)
|
||||||
|
|
||||||
|
def to_dict(self) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
"Arn": self.arn,
|
||||||
|
"CreationDate": self.created_on,
|
||||||
|
"LastModificationDate": self.last_modified,
|
||||||
|
"Name": self.name,
|
||||||
|
"State": "ACTIVE",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class EventBridgeSchedulerBackend(BaseBackend):
|
||||||
|
"""Implementation of EventBridgeScheduler APIs."""
|
||||||
|
|
||||||
|
def __init__(self, region_name: str, account_id: str):
|
||||||
|
super().__init__(region_name, account_id)
|
||||||
|
self.schedules: List[Schedule] = list()
|
||||||
|
self.schedule_groups = {
|
||||||
|
"default": ScheduleGroup(
|
||||||
|
region=region_name, account_id=account_id, name="default"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
self.tagger = TaggingService()
|
||||||
|
|
||||||
|
def create_schedule(
|
||||||
|
self,
|
||||||
|
description: str,
|
||||||
|
end_date: str,
|
||||||
|
flexible_time_window: Dict[str, Any],
|
||||||
|
group_name: str,
|
||||||
|
kms_key_arn: str,
|
||||||
|
name: str,
|
||||||
|
schedule_expression: str,
|
||||||
|
schedule_expression_timezone: str,
|
||||||
|
start_date: str,
|
||||||
|
state: str,
|
||||||
|
target: Dict[str, Any],
|
||||||
|
) -> Schedule:
|
||||||
|
"""
|
||||||
|
The ClientToken parameter is not yet implemented
|
||||||
|
"""
|
||||||
|
group = self.schedule_groups[group_name or "default"]
|
||||||
|
schedule = Schedule(
|
||||||
|
region=self.region_name,
|
||||||
|
account_id=self.account_id,
|
||||||
|
group_name=group.name,
|
||||||
|
name=name,
|
||||||
|
description=description,
|
||||||
|
schedule_expression=schedule_expression,
|
||||||
|
schedule_expression_timezone=schedule_expression_timezone,
|
||||||
|
flexible_time_window=flexible_time_window,
|
||||||
|
target=target,
|
||||||
|
state=state,
|
||||||
|
kms_key_arn=kms_key_arn,
|
||||||
|
start_date=start_date,
|
||||||
|
end_date=end_date,
|
||||||
|
)
|
||||||
|
group.add_schedule(schedule)
|
||||||
|
return schedule
|
||||||
|
|
||||||
|
def get_schedule(self, group_name: Optional[str], name: str) -> Schedule:
|
||||||
|
group = self.get_schedule_group(group_name)
|
||||||
|
return group.get_schedule(name)
|
||||||
|
|
||||||
|
def delete_schedule(self, group_name: Optional[str], name: str) -> None:
|
||||||
|
group = self.get_schedule_group(group_name)
|
||||||
|
group.delete_schedule(name)
|
||||||
|
|
||||||
|
def update_schedule(
|
||||||
|
self,
|
||||||
|
description: str,
|
||||||
|
end_date: str,
|
||||||
|
flexible_time_window: Dict[str, Any],
|
||||||
|
group_name: str,
|
||||||
|
kms_key_arn: str,
|
||||||
|
name: str,
|
||||||
|
schedule_expression: str,
|
||||||
|
schedule_expression_timezone: str,
|
||||||
|
start_date: str,
|
||||||
|
state: str,
|
||||||
|
target: Dict[str, Any],
|
||||||
|
) -> Schedule:
|
||||||
|
"""
|
||||||
|
The ClientToken is not yet implemented
|
||||||
|
"""
|
||||||
|
schedule = self.get_schedule(group_name=group_name, name=name)
|
||||||
|
schedule.schedule_expression = schedule_expression
|
||||||
|
schedule.schedule_expression_timezone = schedule_expression_timezone
|
||||||
|
schedule.flexible_time_window = flexible_time_window
|
||||||
|
schedule.target = Schedule.validate_target(target)
|
||||||
|
schedule.description = description
|
||||||
|
schedule.state = state
|
||||||
|
schedule.kms_key_arn = kms_key_arn
|
||||||
|
schedule.start_date = start_date
|
||||||
|
schedule.end_date = end_date
|
||||||
|
return schedule
|
||||||
|
|
||||||
|
def list_schedules(
|
||||||
|
self, group_names: Optional[str], state: Optional[str]
|
||||||
|
) -> Iterable[Schedule]:
|
||||||
|
"""
|
||||||
|
The following parameters are not yet implemented: MaxResults, NamePrefix, NextToken
|
||||||
|
"""
|
||||||
|
results = []
|
||||||
|
for group in self.schedule_groups.values():
|
||||||
|
if not group_names or group.name in group_names:
|
||||||
|
for schedule in group.schedules.values():
|
||||||
|
if not state or schedule.state == state:
|
||||||
|
results.append(schedule)
|
||||||
|
return results
|
||||||
|
|
||||||
|
def create_schedule_group(
|
||||||
|
self, name: str, tags: List[Dict[str, str]]
|
||||||
|
) -> ScheduleGroup:
|
||||||
|
"""
|
||||||
|
The ClientToken parameter is not yet implemented
|
||||||
|
"""
|
||||||
|
group = ScheduleGroup(
|
||||||
|
region=self.region_name, account_id=self.account_id, name=name
|
||||||
|
)
|
||||||
|
self.schedule_groups[name] = group
|
||||||
|
self.tagger.tag_resource(group.arn, tags)
|
||||||
|
return group
|
||||||
|
|
||||||
|
def get_schedule_group(self, group_name: Optional[str]) -> ScheduleGroup:
|
||||||
|
if (group_name or "default") not in self.schedule_groups:
|
||||||
|
raise ScheduleGroupNotFound
|
||||||
|
return self.schedule_groups[group_name or "default"]
|
||||||
|
|
||||||
|
def list_schedule_groups(self) -> Iterable[ScheduleGroup]:
|
||||||
|
"""
|
||||||
|
The MaxResults-parameter and pagination options are not yet implemented
|
||||||
|
"""
|
||||||
|
return self.schedule_groups.values()
|
||||||
|
|
||||||
|
def delete_schedule_group(self, name: Optional[str]) -> None:
|
||||||
|
self.schedule_groups.pop(name or "default")
|
||||||
|
|
||||||
|
def list_tags_for_resource(
|
||||||
|
self, resource_arn: str
|
||||||
|
) -> Dict[str, List[Dict[str, str]]]:
|
||||||
|
return self.tagger.list_tags_for_resource(resource_arn)
|
||||||
|
|
||||||
|
def tag_resource(self, resource_arn: str, tags: List[Dict[str, str]]) -> None:
|
||||||
|
self.tagger.tag_resource(resource_arn, tags)
|
||||||
|
|
||||||
|
def untag_resource(self, resource_arn: str, tag_keys: List[str]) -> None:
|
||||||
|
self.tagger.untag_resource_using_names(resource_arn, tag_keys)
|
||||||
|
|
||||||
|
|
||||||
|
scheduler_backends = BackendDict(EventBridgeSchedulerBackend, "scheduler")
|
142
moto/scheduler/responses.py
Normal file
142
moto/scheduler/responses.py
Normal file
@ -0,0 +1,142 @@
|
|||||||
|
"""Handles incoming scheduler requests, invokes methods, returns responses."""
|
||||||
|
import json
|
||||||
|
from typing import Any
|
||||||
|
from urllib.parse import unquote
|
||||||
|
|
||||||
|
from moto.core.common_types import TYPE_RESPONSE
|
||||||
|
from moto.core.responses import BaseResponse
|
||||||
|
from .models import scheduler_backends, EventBridgeSchedulerBackend
|
||||||
|
|
||||||
|
|
||||||
|
class EventBridgeSchedulerResponse(BaseResponse):
|
||||||
|
"""Handler for EventBridgeScheduler requests and responses."""
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
super().__init__(service_name="scheduler")
|
||||||
|
|
||||||
|
@property
|
||||||
|
def scheduler_backend(self) -> EventBridgeSchedulerBackend:
|
||||||
|
"""Return backend instance specific for this region."""
|
||||||
|
return scheduler_backends[self.current_account][self.region]
|
||||||
|
|
||||||
|
def create_schedule(self) -> str:
|
||||||
|
description = self._get_param("Description")
|
||||||
|
end_date = self._get_param("EndDate")
|
||||||
|
flexible_time_window = self._get_param("FlexibleTimeWindow")
|
||||||
|
group_name = self._get_param("GroupName")
|
||||||
|
kms_key_arn = self._get_param("KmsKeyArn")
|
||||||
|
name = self.uri.split("/")[-1]
|
||||||
|
schedule_expression = self._get_param("ScheduleExpression")
|
||||||
|
schedule_expression_timezone = self._get_param("ScheduleExpressionTimezone")
|
||||||
|
start_date = self._get_param("StartDate")
|
||||||
|
state = self._get_param("State")
|
||||||
|
target = self._get_param("Target")
|
||||||
|
schedule = self.scheduler_backend.create_schedule(
|
||||||
|
description=description,
|
||||||
|
end_date=end_date,
|
||||||
|
flexible_time_window=flexible_time_window,
|
||||||
|
group_name=group_name,
|
||||||
|
kms_key_arn=kms_key_arn,
|
||||||
|
name=name,
|
||||||
|
schedule_expression=schedule_expression,
|
||||||
|
schedule_expression_timezone=schedule_expression_timezone,
|
||||||
|
start_date=start_date,
|
||||||
|
state=state,
|
||||||
|
target=target,
|
||||||
|
)
|
||||||
|
return json.dumps(dict(ScheduleArn=schedule.arn))
|
||||||
|
|
||||||
|
def get_schedule(self) -> str:
|
||||||
|
group_name = self._get_param("groupName")
|
||||||
|
full_url = self.uri.split("?")[0]
|
||||||
|
name = full_url.split("/")[-1]
|
||||||
|
schedule = self.scheduler_backend.get_schedule(group_name, name)
|
||||||
|
return json.dumps(schedule.to_dict())
|
||||||
|
|
||||||
|
def delete_schedule(self) -> str:
|
||||||
|
group_name = self._get_param("groupName")
|
||||||
|
name = self.uri.split("?")[0].split("/")[-1]
|
||||||
|
self.scheduler_backend.delete_schedule(group_name, name)
|
||||||
|
return "{}"
|
||||||
|
|
||||||
|
def update_schedule(self) -> str:
|
||||||
|
group_name = self._get_param("groupName")
|
||||||
|
name = self.uri.split("?")[0].split("/")[-1]
|
||||||
|
description = self._get_param("Description")
|
||||||
|
end_date = self._get_param("EndDate")
|
||||||
|
flexible_time_window = self._get_param("FlexibleTimeWindow")
|
||||||
|
kms_key_arn = self._get_param("KmsKeyArn")
|
||||||
|
schedule_expression = self._get_param("ScheduleExpression")
|
||||||
|
schedule_expression_timezone = self._get_param("ScheduleExpressionTimezone")
|
||||||
|
start_date = self._get_param("StartDate")
|
||||||
|
state = self._get_param("State")
|
||||||
|
target = self._get_param("Target")
|
||||||
|
schedule = self.scheduler_backend.update_schedule(
|
||||||
|
description=description,
|
||||||
|
end_date=end_date,
|
||||||
|
flexible_time_window=flexible_time_window,
|
||||||
|
group_name=group_name,
|
||||||
|
kms_key_arn=kms_key_arn,
|
||||||
|
name=name,
|
||||||
|
schedule_expression=schedule_expression,
|
||||||
|
schedule_expression_timezone=schedule_expression_timezone,
|
||||||
|
start_date=start_date,
|
||||||
|
state=state,
|
||||||
|
target=target,
|
||||||
|
)
|
||||||
|
return json.dumps(dict(ScheduleArn=schedule.arn))
|
||||||
|
|
||||||
|
def list_schedules(self) -> str:
|
||||||
|
group_names = self.querystring.get("ScheduleGroup")
|
||||||
|
state = self._get_param("State")
|
||||||
|
schedules = self.scheduler_backend.list_schedules(group_names, state)
|
||||||
|
return json.dumps({"Schedules": [sch.to_dict(short=True) for sch in schedules]})
|
||||||
|
|
||||||
|
def create_schedule_group(self) -> str:
|
||||||
|
name = self._get_param("Name")
|
||||||
|
tags = self._get_param("Tags")
|
||||||
|
schedule_group = self.scheduler_backend.create_schedule_group(
|
||||||
|
name=name,
|
||||||
|
tags=tags,
|
||||||
|
)
|
||||||
|
return json.dumps(dict(ScheduleGroupArn=schedule_group.arn))
|
||||||
|
|
||||||
|
def get_schedule_group(self) -> str:
|
||||||
|
group_name = self.uri.split("?")[0].split("/")[-1]
|
||||||
|
group = self.scheduler_backend.get_schedule_group(group_name)
|
||||||
|
return json.dumps(group.to_dict())
|
||||||
|
|
||||||
|
def delete_schedule_group(self) -> str:
|
||||||
|
group_name = self.uri.split("?")[0].split("/")[-1]
|
||||||
|
self.scheduler_backend.delete_schedule_group(group_name)
|
||||||
|
return "{}"
|
||||||
|
|
||||||
|
def list_schedule_groups(self) -> str:
|
||||||
|
schedule_groups = self.scheduler_backend.list_schedule_groups()
|
||||||
|
return json.dumps(dict(ScheduleGroups=[sg.to_dict() for sg in schedule_groups]))
|
||||||
|
|
||||||
|
def list_tags_for_resource(self) -> TYPE_RESPONSE:
|
||||||
|
resource_arn = unquote(self.uri.split("/tags/")[-1])
|
||||||
|
tags = self.scheduler_backend.list_tags_for_resource(resource_arn)
|
||||||
|
return 200, {}, json.dumps(tags)
|
||||||
|
|
||||||
|
def tag_resource(self) -> TYPE_RESPONSE:
|
||||||
|
resource_arn = unquote(self.uri.split("/tags/")[-1])
|
||||||
|
tags = json.loads(self.body)["Tags"]
|
||||||
|
self.scheduler_backend.tag_resource(resource_arn, tags)
|
||||||
|
return 200, {}, "{}"
|
||||||
|
|
||||||
|
def untag_resource(self) -> TYPE_RESPONSE:
|
||||||
|
resource_arn = unquote(self.uri.split("?")[0].split("/tags/")[-1])
|
||||||
|
tag_keys = self.querystring.get("TagKeys")
|
||||||
|
self.scheduler_backend.untag_resource(resource_arn, tag_keys) # type: ignore
|
||||||
|
return 200, {}, "{}"
|
||||||
|
|
||||||
|
def tags(self, request: Any, full_url: str, headers: Any) -> TYPE_RESPONSE:
|
||||||
|
super().setup_class(request, full_url, headers)
|
||||||
|
if request.method == "POST":
|
||||||
|
return self.tag_resource()
|
||||||
|
elif request.method == "DELETE":
|
||||||
|
return self.untag_resource()
|
||||||
|
else:
|
||||||
|
return self.list_tags_for_resource()
|
19
moto/scheduler/urls.py
Normal file
19
moto/scheduler/urls.py
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
"""scheduler base URL and path."""
|
||||||
|
from .responses import EventBridgeSchedulerResponse
|
||||||
|
|
||||||
|
url_bases = [
|
||||||
|
r"https?://scheduler\.(.+)\.amazonaws\.com",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
response = EventBridgeSchedulerResponse()
|
||||||
|
|
||||||
|
|
||||||
|
url_paths = {
|
||||||
|
"{0}/schedules$": response.dispatch,
|
||||||
|
"{0}/schedules/(?P<name>[^/]+)$": response.dispatch,
|
||||||
|
"{0}/schedule-groups$": response.dispatch,
|
||||||
|
"{0}/schedule-groups/(?P<name>[^/]+)$": response.dispatch,
|
||||||
|
"{0}/tags/(?P<ResourceArn>.+)$": response.tags,
|
||||||
|
"{0}/tags/arn:aws:scheduler:(?P<region_name>[^/]+):(?P<account_id>[^/]+):schedule/(?P<group_name>[^/]+)/(?P<schedule_name>[^/]+)/?$": response.tags,
|
||||||
|
}
|
@ -235,7 +235,7 @@ disable = W,C,R,E
|
|||||||
enable = anomalous-backslash-in-string, arguments-renamed, dangerous-default-value, deprecated-module, function-redefined, import-self, redefined-builtin, redefined-outer-name, reimported, pointless-statement, super-with-arguments, unused-argument, unused-import, unused-variable, useless-else-on-loop, wildcard-import
|
enable = anomalous-backslash-in-string, arguments-renamed, dangerous-default-value, deprecated-module, function-redefined, import-self, redefined-builtin, redefined-outer-name, reimported, pointless-statement, super-with-arguments, unused-argument, unused-import, unused-variable, useless-else-on-loop, wildcard-import
|
||||||
|
|
||||||
[mypy]
|
[mypy]
|
||||||
files= moto/a*,moto/b*,moto/c*,moto/d*,moto/e*,moto/f*,moto/g*,moto/i*,moto/k*,moto/l*,moto/m*,moto/n*,moto/o*,moto/p*,moto/q*,moto/rdsdata
|
files= moto/a*,moto/b*,moto/c*,moto/d*,moto/e*,moto/f*,moto/g*,moto/i*,moto/k*,moto/l*,moto/m*,moto/n*,moto/o*,moto/p*,moto/q*,moto/rdsdata,moto/scheduler
|
||||||
show_column_numbers=True
|
show_column_numbers=True
|
||||||
show_error_codes = True
|
show_error_codes = True
|
||||||
disable_error_code=abstract
|
disable_error_code=abstract
|
||||||
|
@ -494,6 +494,9 @@ s3:
|
|||||||
- TestAccS3ObjectsDataSource_fetchOwner
|
- TestAccS3ObjectsDataSource_fetchOwner
|
||||||
sagemaker:
|
sagemaker:
|
||||||
- TestAccSageMakerPrebuiltECRImageDataSource
|
- TestAccSageMakerPrebuiltECRImageDataSource
|
||||||
|
scheduler:
|
||||||
|
- TestAccSchedulerSchedule_
|
||||||
|
- TestAccSchedulerScheduleGroup_
|
||||||
secretsmanager:
|
secretsmanager:
|
||||||
- TestAccSecretsManagerSecretDataSource_basic
|
- TestAccSecretsManagerSecretDataSource_basic
|
||||||
- TestAccSecretsManagerSecretPolicy_
|
- TestAccSecretsManagerSecretPolicy_
|
||||||
|
0
tests/test_scheduler/__init__.py
Normal file
0
tests/test_scheduler/__init__.py
Normal file
49
tests/test_scheduler/test_schedule_groups.py
Normal file
49
tests/test_scheduler/test_schedule_groups.py
Normal file
@ -0,0 +1,49 @@
|
|||||||
|
"""Unit tests for scheduler-supported APIs."""
|
||||||
|
import boto3
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from botocore.exceptions import ClientError
|
||||||
|
from moto import mock_scheduler
|
||||||
|
from moto.core import DEFAULT_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_scheduler
|
||||||
|
def test_create_get_delete_schedule_group():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
arn = client.create_schedule_group(Name="sg")["ScheduleGroupArn"]
|
||||||
|
|
||||||
|
assert arn == f"arn:aws:scheduler:eu-west-1:{DEFAULT_ACCOUNT_ID}:schedule-group/sg"
|
||||||
|
|
||||||
|
group = client.get_schedule_group(Name="sg")
|
||||||
|
assert group["Arn"] == arn
|
||||||
|
assert group["Name"] == "sg"
|
||||||
|
assert group["State"] == "ACTIVE"
|
||||||
|
|
||||||
|
client.delete_schedule_group(Name="sg")
|
||||||
|
|
||||||
|
with pytest.raises(ClientError) as exc:
|
||||||
|
client.get_schedule_group(Name="sg")
|
||||||
|
err = exc.value.response["Error"]
|
||||||
|
assert err["Code"] == "ResourceNotFoundException"
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_list_schedule_groups():
|
||||||
|
client = boto3.client("scheduler", region_name="ap-southeast-1")
|
||||||
|
|
||||||
|
# The default group is always active
|
||||||
|
groups = client.list_schedule_groups()["ScheduleGroups"]
|
||||||
|
assert len(groups) == 1
|
||||||
|
assert (
|
||||||
|
groups[0]["Arn"]
|
||||||
|
== f"arn:aws:scheduler:ap-southeast-1:{DEFAULT_ACCOUNT_ID}:schedule-group/default"
|
||||||
|
)
|
||||||
|
|
||||||
|
arn1 = client.create_schedule_group(Name="sg")["ScheduleGroupArn"]
|
||||||
|
|
||||||
|
groups = client.list_schedule_groups()["ScheduleGroups"]
|
||||||
|
assert len(groups) == 2
|
||||||
|
assert groups[1]["Arn"] == arn1
|
167
tests/test_scheduler/test_scheduler.py
Normal file
167
tests/test_scheduler/test_scheduler.py
Normal file
@ -0,0 +1,167 @@
|
|||||||
|
"""Unit tests for scheduler-supported APIs."""
|
||||||
|
import boto3
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from botocore.client import ClientError
|
||||||
|
from moto import mock_scheduler
|
||||||
|
from moto.core import DEFAULT_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_scheduler
|
||||||
|
def test_create_get_schedule():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
arn = client.create_schedule(
|
||||||
|
Name="my-schedule",
|
||||||
|
ScheduleExpression="some cron",
|
||||||
|
FlexibleTimeWindow={
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
},
|
||||||
|
Target={
|
||||||
|
"Arn": "not supported yet",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
},
|
||||||
|
)["ScheduleArn"]
|
||||||
|
|
||||||
|
assert (
|
||||||
|
arn
|
||||||
|
== f"arn:aws:scheduler:eu-west-1:{DEFAULT_ACCOUNT_ID}:schedule/default/my-schedule"
|
||||||
|
)
|
||||||
|
|
||||||
|
resp = client.get_schedule(Name="my-schedule")
|
||||||
|
assert resp["Arn"] == arn
|
||||||
|
assert resp["Name"] == "my-schedule"
|
||||||
|
assert resp["ScheduleExpression"] == "some cron"
|
||||||
|
assert resp["FlexibleTimeWindow"] == {
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
}
|
||||||
|
assert resp["Target"] == {
|
||||||
|
"Arn": "not supported yet",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
"RetryPolicy": {"MaximumEventAgeInSeconds": 86400, "MaximumRetryAttempts": 185},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_create_get_delete__in_different_group():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
|
||||||
|
client.create_schedule_group(Name="sg")
|
||||||
|
schedule_arn = client.create_schedule(
|
||||||
|
Name="my-schedule",
|
||||||
|
GroupName="sg",
|
||||||
|
ScheduleExpression="some cron",
|
||||||
|
FlexibleTimeWindow={
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
},
|
||||||
|
Target={
|
||||||
|
"Arn": "not supported yet",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
},
|
||||||
|
)["ScheduleArn"]
|
||||||
|
|
||||||
|
assert (
|
||||||
|
schedule_arn
|
||||||
|
== "arn:aws:scheduler:eu-west-1:123456789012:schedule/sg/my-schedule"
|
||||||
|
)
|
||||||
|
|
||||||
|
schedule = client.get_schedule(GroupName="sg", Name="my-schedule")
|
||||||
|
assert schedule["Arn"] == schedule_arn
|
||||||
|
|
||||||
|
client.delete_schedule(GroupName="sg", Name="my-schedule")
|
||||||
|
|
||||||
|
with pytest.raises(ClientError) as exc:
|
||||||
|
client.get_schedule(GroupName="sg", Name="my-schedule")
|
||||||
|
err = exc.value.response["Error"]
|
||||||
|
assert err["Code"] == "ResourceNotFoundException"
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_update_schedule():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
|
||||||
|
client.create_schedule(
|
||||||
|
Name="my-schedule",
|
||||||
|
ScheduleExpression="some cron",
|
||||||
|
FlexibleTimeWindow={
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
},
|
||||||
|
Target={
|
||||||
|
"Arn": "not supported yet",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
client.update_schedule(
|
||||||
|
Name="my-schedule",
|
||||||
|
Description="new desc",
|
||||||
|
ScheduleExpression="new cron",
|
||||||
|
FlexibleTimeWindow={
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
},
|
||||||
|
State="DISABLED",
|
||||||
|
Target={
|
||||||
|
"Arn": "different arn",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
schedule = client.get_schedule(Name="my-schedule")
|
||||||
|
assert schedule["Description"] == "new desc"
|
||||||
|
assert schedule["ScheduleExpression"] == "new cron"
|
||||||
|
assert schedule["State"] == "DISABLED"
|
||||||
|
assert schedule["Target"] == {
|
||||||
|
"Arn": "different arn",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
"RetryPolicy": {"MaximumEventAgeInSeconds": 86400, "MaximumRetryAttempts": 185},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_get_schedule_for_unknown_group():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
|
||||||
|
with pytest.raises(ClientError) as exc:
|
||||||
|
client.get_schedule(GroupName="unknown", Name="my-schedule")
|
||||||
|
err = exc.value.response["Error"]
|
||||||
|
assert err["Code"] == "ResourceNotFoundException"
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_list_schedules():
|
||||||
|
client = boto3.client("scheduler", region_name="eu-west-1")
|
||||||
|
|
||||||
|
schedules = client.list_schedules()["Schedules"]
|
||||||
|
assert schedules == []
|
||||||
|
|
||||||
|
client.create_schedule_group(Name="group2")
|
||||||
|
|
||||||
|
for group in ["default", "group2"]:
|
||||||
|
for schedule in ["sch1", "sch2"]:
|
||||||
|
for state in ["ENABLED", "DISABLED"]:
|
||||||
|
client.create_schedule(
|
||||||
|
Name=f"{schedule}_{state}",
|
||||||
|
GroupName=group,
|
||||||
|
State=state,
|
||||||
|
ScheduleExpression="some cron",
|
||||||
|
FlexibleTimeWindow={"MaximumWindowInMinutes": 4, "Mode": "OFF"},
|
||||||
|
Target={"Arn": "not supported yet", "RoleArn": "n/a"},
|
||||||
|
)
|
||||||
|
|
||||||
|
schedules = client.list_schedules()["Schedules"]
|
||||||
|
assert len(schedules) == 8
|
||||||
|
# The ListSchedules command should not return the entire Target-dictionary
|
||||||
|
assert schedules[0]["Target"] == {"Arn": "not supported yet"}
|
||||||
|
|
||||||
|
schedules = client.list_schedules(GroupName="group2")["Schedules"]
|
||||||
|
assert len(schedules) == 4
|
||||||
|
|
||||||
|
schedules = client.list_schedules(State="ENABLED")["Schedules"]
|
||||||
|
assert len(schedules) == 4
|
57
tests/test_scheduler/test_scheduler_tags.py
Normal file
57
tests/test_scheduler/test_scheduler_tags.py
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
import boto3
|
||||||
|
|
||||||
|
from moto import mock_scheduler
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_schedule_tags():
|
||||||
|
client = boto3.client("scheduler", "us-east-1")
|
||||||
|
arn = client.create_schedule(
|
||||||
|
Name="my-schedule",
|
||||||
|
ScheduleExpression="some cron",
|
||||||
|
FlexibleTimeWindow={
|
||||||
|
"MaximumWindowInMinutes": 4,
|
||||||
|
"Mode": "OFF",
|
||||||
|
},
|
||||||
|
Target={
|
||||||
|
"Arn": "not supported yet",
|
||||||
|
"RoleArn": "n/a",
|
||||||
|
},
|
||||||
|
)["ScheduleArn"]
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == []
|
||||||
|
|
||||||
|
client.tag_resource(
|
||||||
|
ResourceArn=arn,
|
||||||
|
Tags=[{"Key": "k1", "Value": "v1"}, {"Key": "k2", "Value": "v2"}],
|
||||||
|
)
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == [{"Key": "k1", "Value": "v1"}, {"Key": "k2", "Value": "v2"}]
|
||||||
|
|
||||||
|
client.untag_resource(ResourceArn=arn, TagKeys=["k1"])
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == [{"Key": "k2", "Value": "v2"}]
|
||||||
|
|
||||||
|
|
||||||
|
@mock_scheduler
|
||||||
|
def test_schedule_group_tags():
|
||||||
|
client = boto3.client("scheduler", "us-east-1")
|
||||||
|
arn = client.create_schedule_group(
|
||||||
|
Name="my-schedule", Tags=[{"Key": "k1", "Value": "v1"}]
|
||||||
|
)["ScheduleGroupArn"]
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == [{"Key": "k1", "Value": "v1"}]
|
||||||
|
|
||||||
|
client.tag_resource(ResourceArn=arn, Tags=[{"Key": "k2", "Value": "v2"}])
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == [{"Key": "k1", "Value": "v1"}, {"Key": "k2", "Value": "v2"}]
|
||||||
|
|
||||||
|
client.untag_resource(ResourceArn=arn, TagKeys=["k1"])
|
||||||
|
|
||||||
|
resp = client.list_tags_for_resource(ResourceArn=arn)
|
||||||
|
assert resp["Tags"] == [{"Key": "k2", "Value": "v2"}]
|
10
tests/test_scheduler/test_server.py
Normal file
10
tests/test_scheduler/test_server.py
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
from moto import server as server
|
||||||
|
|
||||||
|
|
||||||
|
def test_list_tags():
|
||||||
|
test_client = server.create_backend_app("scheduler").test_client()
|
||||||
|
res = test_client.get(
|
||||||
|
"/tags/arn%3Aaws%3Ascheduler%3Aus-east-1%3A123456789012%3Aschedule%2Fdefault%2Fmy-schedule"
|
||||||
|
)
|
||||||
|
|
||||||
|
assert res.data == b'{"Tags": []}'
|
Loading…
Reference in New Issue
Block a user