Prepare SWF objects representations directly via json.dumps()
... instead of jinja2 templates that are absolutely not suited for this purpose, and hard to test.
This commit is contained in:
parent
6e6b325225
commit
9483355584
@ -4,6 +4,8 @@ from collections import defaultdict
|
|||||||
import boto.swf
|
import boto.swf
|
||||||
|
|
||||||
from moto.core import BaseBackend
|
from moto.core import BaseBackend
|
||||||
|
from moto.core.utils import camelcase_to_underscores
|
||||||
|
|
||||||
from .exceptions import (
|
from .exceptions import (
|
||||||
SWFUnknownResourceFault,
|
SWFUnknownResourceFault,
|
||||||
SWFDomainAlreadyExistsFault,
|
SWFDomainAlreadyExistsFault,
|
||||||
@ -26,6 +28,15 @@ class Domain(object):
|
|||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "Domain(name: %(name)s, status: %(status)s)" % self.__dict__
|
return "Domain(name: %(name)s, status: %(status)s)" % self.__dict__
|
||||||
|
|
||||||
|
def to_dict(self):
|
||||||
|
hsh = {
|
||||||
|
"name": self.name,
|
||||||
|
"status": self.status,
|
||||||
|
}
|
||||||
|
if self.description:
|
||||||
|
hsh["description"] = self.description
|
||||||
|
return hsh
|
||||||
|
|
||||||
def get_type(self, kind, name, version, ignore_empty=False):
|
def get_type(self, kind, name, version, ignore_empty=False):
|
||||||
try:
|
try:
|
||||||
_types = getattr(self, "{}_types".format(kind))
|
_types = getattr(self, "{}_types".format(kind))
|
||||||
@ -62,12 +73,57 @@ class ActivityType(object):
|
|||||||
self.name = name
|
self.name = name
|
||||||
self.version = version
|
self.version = version
|
||||||
self.status = "REGISTERED"
|
self.status = "REGISTERED"
|
||||||
|
if "description" in kwargs:
|
||||||
|
self.description = kwargs.pop("description")
|
||||||
for key, value in kwargs.iteritems():
|
for key, value in kwargs.iteritems():
|
||||||
self.__setattr__(key, value)
|
self.__setattr__(key, value)
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "ActivityType(name: %(name)s, version: %(version)s)" % self.__dict__
|
return "ActivityType(name: %(name)s, version: %(version)s)" % self.__dict__
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _configuration_keys(self):
|
||||||
|
return [
|
||||||
|
"defaultTaskHeartbeatTimeout",
|
||||||
|
"defaultTaskScheduleToCloseTimeout",
|
||||||
|
"defaultTaskScheduleToStartTimeout",
|
||||||
|
"defaultTaskStartToCloseTimeout",
|
||||||
|
]
|
||||||
|
|
||||||
|
def to_short_dict(self):
|
||||||
|
return {
|
||||||
|
"name": self.name,
|
||||||
|
"version": self.version,
|
||||||
|
}
|
||||||
|
|
||||||
|
def to_medium_dict(self):
|
||||||
|
hsh = {
|
||||||
|
"activityType": self.to_short_dict(),
|
||||||
|
"creationDate": 1420066800,
|
||||||
|
"status": self.status,
|
||||||
|
}
|
||||||
|
if self.status == "DEPRECATED":
|
||||||
|
hsh["deprecationDate"] = 1422745200
|
||||||
|
if hasattr(self, "description"):
|
||||||
|
hsh["description"] = self.description
|
||||||
|
return hsh
|
||||||
|
|
||||||
|
def to_full_dict(self):
|
||||||
|
hsh = {
|
||||||
|
"typeInfo": self.to_medium_dict(),
|
||||||
|
"configuration": {}
|
||||||
|
}
|
||||||
|
if hasattr(self, "task_list"):
|
||||||
|
hsh["configuration"]["defaultTaskList"] = {"name": self.task_list}
|
||||||
|
for key in self._configuration_keys:
|
||||||
|
attr = camelcase_to_underscores(key)
|
||||||
|
if not hasattr(self, attr):
|
||||||
|
continue
|
||||||
|
if not getattr(self, attr):
|
||||||
|
continue
|
||||||
|
hsh["configuration"][key] = getattr(self, attr)
|
||||||
|
return hsh
|
||||||
|
|
||||||
|
|
||||||
class WorkflowType(object):
|
class WorkflowType(object):
|
||||||
def __init__(self, name, version, **kwargs):
|
def __init__(self, name, version, **kwargs):
|
||||||
@ -80,6 +136,48 @@ class WorkflowType(object):
|
|||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return "WorkflowType(name: %(name)s, version: %(version)s)" % self.__dict__
|
return "WorkflowType(name: %(name)s, version: %(version)s)" % self.__dict__
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _configuration_keys(self):
|
||||||
|
return [
|
||||||
|
"defaultChildPolicy",
|
||||||
|
"defaultExecutionStartToCloseTimeout",
|
||||||
|
"defaultTaskStartToCloseTimeout",
|
||||||
|
]
|
||||||
|
|
||||||
|
def to_short_dict(self):
|
||||||
|
return {
|
||||||
|
"name": self.name,
|
||||||
|
"version": self.version,
|
||||||
|
}
|
||||||
|
|
||||||
|
def to_medium_dict(self):
|
||||||
|
hsh = {
|
||||||
|
"workflowType": self.to_short_dict(),
|
||||||
|
"creationDate": 1420066800,
|
||||||
|
"status": self.status,
|
||||||
|
}
|
||||||
|
if self.status == "DEPRECATED":
|
||||||
|
hsh["deprecationDate"] = 1422745200
|
||||||
|
if hasattr(self, "description"):
|
||||||
|
hsh["description"] = self.description
|
||||||
|
return hsh
|
||||||
|
|
||||||
|
def to_full_dict(self):
|
||||||
|
hsh = {
|
||||||
|
"typeInfo": self.to_medium_dict(),
|
||||||
|
"configuration": {}
|
||||||
|
}
|
||||||
|
if hasattr(self, "task_list"):
|
||||||
|
hsh["configuration"]["defaultTaskList"] = {"name": self.task_list}
|
||||||
|
for key in self._configuration_keys:
|
||||||
|
attr = camelcase_to_underscores(key)
|
||||||
|
if not hasattr(self, attr):
|
||||||
|
continue
|
||||||
|
if getattr(self, attr) is None:
|
||||||
|
continue
|
||||||
|
hsh["configuration"][key] = getattr(self, attr)
|
||||||
|
return hsh
|
||||||
|
|
||||||
|
|
||||||
class SWFBackend(BaseBackend):
|
class SWFBackend(BaseBackend):
|
||||||
def __init__(self, region_name):
|
def __init__(self, region_name):
|
||||||
|
@ -51,23 +51,24 @@ class SWFResponse(BaseResponse):
|
|||||||
def _params(self):
|
def _params(self):
|
||||||
return json.loads(self.body)
|
return json.loads(self.body)
|
||||||
|
|
||||||
def _list_types(self, kind, template_str):
|
def _list_types(self, kind):
|
||||||
domain_name = self._params.get("domain")
|
domain_name = self._params.get("domain")
|
||||||
status = self._params.get("registrationStatus")
|
status = self._params.get("registrationStatus")
|
||||||
reverse_order = self._params.get("reverseOrder", None)
|
reverse_order = self._params.get("reverseOrder", None)
|
||||||
types = self.swf_backend.list_types(kind, domain_name, status, reverse_order=reverse_order)
|
types = self.swf_backend.list_types(kind, domain_name, status, reverse_order=reverse_order)
|
||||||
template = self.response_template(template_str)
|
return json.dumps({
|
||||||
return template.render(types=types)
|
"typeInfos": [_type.to_medium_dict() for _type in types]
|
||||||
|
})
|
||||||
|
|
||||||
def _describe_type(self, kind, template_str):
|
def _describe_type(self, kind):
|
||||||
domain = self._params.get("domain")
|
domain = self._params.get("domain")
|
||||||
_type = self._params.get("{}Type".format(kind))
|
_type_args = self._params.get("{}Type".format(kind))
|
||||||
|
|
||||||
name = _type["name"]
|
name = _type_args["name"]
|
||||||
version = _type["version"]
|
version = _type_args["version"]
|
||||||
_type = self.swf_backend.describe_type(kind, domain, name, version)
|
_type = self.swf_backend.describe_type(kind, domain, name, version)
|
||||||
template = self.response_template(template_str)
|
|
||||||
return template.render(_type=_type)
|
return json.dumps(_type.to_full_dict())
|
||||||
|
|
||||||
def _deprecate_type(self, kind):
|
def _deprecate_type(self, kind):
|
||||||
domain = self._params.get("domain")
|
domain = self._params.get("domain")
|
||||||
@ -83,8 +84,9 @@ class SWFResponse(BaseResponse):
|
|||||||
status = self._params.get("registrationStatus")
|
status = self._params.get("registrationStatus")
|
||||||
reverse_order = self._params.get("reverseOrder", None)
|
reverse_order = self._params.get("reverseOrder", None)
|
||||||
domains = self.swf_backend.list_domains(status, reverse_order=reverse_order)
|
domains = self.swf_backend.list_domains(status, reverse_order=reverse_order)
|
||||||
template = self.response_template(LIST_DOMAINS_TEMPLATE)
|
return json.dumps({
|
||||||
return template.render(domains=domains)
|
"domainInfos": [domain.to_dict() for domain in domains]
|
||||||
|
})
|
||||||
|
|
||||||
def register_domain(self):
|
def register_domain(self):
|
||||||
name = self._params.get("name")
|
name = self._params.get("name")
|
||||||
@ -92,24 +94,24 @@ class SWFResponse(BaseResponse):
|
|||||||
retention = self._params.get("workflowExecutionRetentionPeriodInDays")
|
retention = self._params.get("workflowExecutionRetentionPeriodInDays")
|
||||||
domain = self.swf_backend.register_domain(name, retention,
|
domain = self.swf_backend.register_domain(name, retention,
|
||||||
description=description)
|
description=description)
|
||||||
template = self.response_template("")
|
return ""
|
||||||
return template.render()
|
|
||||||
|
|
||||||
def deprecate_domain(self):
|
def deprecate_domain(self):
|
||||||
name = self._params.get("name")
|
name = self._params.get("name")
|
||||||
domain = self.swf_backend.deprecate_domain(name)
|
domain = self.swf_backend.deprecate_domain(name)
|
||||||
template = self.response_template("")
|
return ""
|
||||||
return template.render()
|
|
||||||
|
|
||||||
def describe_domain(self):
|
def describe_domain(self):
|
||||||
name = self._params.get("name")
|
name = self._params.get("name")
|
||||||
domain = self.swf_backend.describe_domain(name)
|
domain = self.swf_backend.describe_domain(name)
|
||||||
template = self.response_template(DESCRIBE_DOMAIN_TEMPLATE)
|
return json.dumps({
|
||||||
return template.render(domain=domain)
|
"configuration": { "workflowExecutionRetentionPeriodInDays": domain.retention },
|
||||||
|
"domainInfo": domain.to_dict()
|
||||||
|
})
|
||||||
|
|
||||||
# TODO: implement pagination
|
# TODO: implement pagination
|
||||||
def list_activity_types(self):
|
def list_activity_types(self):
|
||||||
return self._list_types("activity", LIST_ACTIVITY_TYPES_TEMPLATE)
|
return self._list_types("activity")
|
||||||
|
|
||||||
def register_activity_type(self):
|
def register_activity_type(self):
|
||||||
domain = self._params.get("domain")
|
domain = self._params.get("domain")
|
||||||
@ -141,11 +143,11 @@ class SWFResponse(BaseResponse):
|
|||||||
return self._deprecate_type("activity")
|
return self._deprecate_type("activity")
|
||||||
|
|
||||||
def describe_activity_type(self):
|
def describe_activity_type(self):
|
||||||
return self._describe_type("activity", DESCRIBE_ACTIVITY_TYPE_TEMPLATE)
|
return self._describe_type("activity")
|
||||||
|
|
||||||
# TODO: refactor with list_activity_types()
|
# TODO: refactor with list_activity_types()
|
||||||
def list_workflow_types(self):
|
def list_workflow_types(self):
|
||||||
return self._list_types("workflow", LIST_WORKFLOW_TYPES_TEMPLATE)
|
return self._list_types("workflow")
|
||||||
|
|
||||||
def register_workflow_type(self):
|
def register_workflow_type(self):
|
||||||
domain = self._params.get("domain")
|
domain = self._params.get("domain")
|
||||||
@ -176,88 +178,4 @@ class SWFResponse(BaseResponse):
|
|||||||
return self._deprecate_type("workflow")
|
return self._deprecate_type("workflow")
|
||||||
|
|
||||||
def describe_workflow_type(self):
|
def describe_workflow_type(self):
|
||||||
return self._describe_type("workflow", DESCRIBE_WORKFLOW_TYPE_TEMPLATE)
|
return self._describe_type("workflow")
|
||||||
|
|
||||||
|
|
||||||
LIST_DOMAINS_TEMPLATE = """{
|
|
||||||
"domainInfos": [
|
|
||||||
{%- for domain in domains %}
|
|
||||||
{
|
|
||||||
"description": "{{ domain.description }}",
|
|
||||||
"name": "{{ domain.name }}",
|
|
||||||
"status": "{{ domain.status }}"
|
|
||||||
}{% if not loop.last %},{% endif %}
|
|
||||||
{%- endfor %}
|
|
||||||
]
|
|
||||||
}"""
|
|
||||||
|
|
||||||
DESCRIBE_DOMAIN_TEMPLATE = """{
|
|
||||||
"configuration": {
|
|
||||||
"workflowExecutionRetentionPeriodInDays": "{{ domain.retention }}"
|
|
||||||
},
|
|
||||||
"domainInfo": {
|
|
||||||
"description": "{{ domain.description }}",
|
|
||||||
"name": "{{ domain.name }}",
|
|
||||||
"status": "{{ domain.status }}"
|
|
||||||
}
|
|
||||||
}"""
|
|
||||||
|
|
||||||
LIST_ACTIVITY_TYPES_TEMPLATE = """{
|
|
||||||
"typeInfos": [
|
|
||||||
{%- for _type in types %}
|
|
||||||
{
|
|
||||||
"activityType": {
|
|
||||||
"name": "{{ _type.name }}",
|
|
||||||
"version": "{{ _type.version }}"
|
|
||||||
},
|
|
||||||
"creationDate": 1420066800,
|
|
||||||
{% if _type.status == "DEPRECATED" %}"deprecationDate": 1422745200,{% endif %}
|
|
||||||
{% if _type.description %}"description": "{{ _type.description }}",{% endif %}
|
|
||||||
"status": "{{ _type.status }}"
|
|
||||||
}{% if not loop.last %},{% endif %}
|
|
||||||
{%- endfor %}
|
|
||||||
]
|
|
||||||
}"""
|
|
||||||
|
|
||||||
DESCRIBE_ACTIVITY_TYPE_TEMPLATE = """{
|
|
||||||
"configuration": {
|
|
||||||
{% if _type.default_task_heartbeat_timeout %}"defaultTaskHeartbeatTimeout": "{{ _type.default_task_heartbeat_timeout }}",{% endif %}
|
|
||||||
{% if _type.task_list %}"defaultTaskList": { "name": "{{ _type.task_list }}" },{% endif %}
|
|
||||||
{% if _type.default_task_schedule_to_close_timeout %}"defaultTaskScheduleToCloseTimeout": "{{ _type.default_task_schedule_to_close_timeout }}",{% endif %}
|
|
||||||
{% if _type.default_task_schedule_to_start_timeout %}"defaultTaskScheduleToStartTimeout": "{{ _type.default_task_schedule_to_start_timeout }}",{% endif %}
|
|
||||||
{% if _type.default_task_start_to_close_timeout %}"defaultTaskStartToCloseTimeout": "{{ _type.default_task_start_to_close_timeout }}",{% endif %}
|
|
||||||
"__moto_placeholder": "(avoid dealing with coma in json)"
|
|
||||||
},
|
|
||||||
"typeInfo": {
|
|
||||||
"activityType": {
|
|
||||||
"name": "{{ _type.name }}",
|
|
||||||
"version": "{{ _type.version }}"
|
|
||||||
},
|
|
||||||
"creationDate": 1420066800,
|
|
||||||
{% if _type.status == "DEPRECATED" %}"deprecationDate": 1422745200,{% endif %}
|
|
||||||
{% if _type.description %}"description": "{{ _type.description }}",{% endif %}
|
|
||||||
"status": "{{ _type.status }}"
|
|
||||||
}
|
|
||||||
}"""
|
|
||||||
|
|
||||||
LIST_WORKFLOW_TYPES_TEMPLATE = LIST_ACTIVITY_TYPES_TEMPLATE.replace("activityType", "workflowType")
|
|
||||||
|
|
||||||
DESCRIBE_WORKFLOW_TYPE_TEMPLATE = """{
|
|
||||||
"configuration": {
|
|
||||||
{% if _type.default_child_policy %}"defaultChildPolicy": "{{ _type.default_child_policy }}",{% endif %}
|
|
||||||
{% if _type.default_execution_start_to_close_timeout %}"defaultExecutionStartToCloseTimeout": "{{ _type.default_execution_start_to_close_timeout }}",{% endif %}
|
|
||||||
{% if _type.task_list %}"defaultTaskList": { "name": "{{ _type.task_list }}" },{% endif %}
|
|
||||||
{% if _type.default_task_start_to_close_timeout %}"defaultTaskStartToCloseTimeout": "{{ _type.default_task_start_to_close_timeout }}",{% endif %}
|
|
||||||
"__moto_placeholder": "(avoid dealing with coma in json)"
|
|
||||||
},
|
|
||||||
"typeInfo": {
|
|
||||||
"workflowType": {
|
|
||||||
"name": "{{ _type.name }}",
|
|
||||||
"version": "{{ _type.version }}"
|
|
||||||
},
|
|
||||||
"creationDate": 1420066800,
|
|
||||||
{% if _type.status == "DEPRECATED" %}"deprecationDate": 1422745200,{% endif %}
|
|
||||||
{% if _type.description %}"description": "{{ _type.description }}",{% endif %}
|
|
||||||
"status": "{{ _type.status }}"
|
|
||||||
}
|
|
||||||
}"""
|
|
||||||
|
@ -3,6 +3,7 @@ from nose.tools import assert_raises
|
|||||||
from sure import expect
|
from sure import expect
|
||||||
|
|
||||||
from moto import mock_swf
|
from moto import mock_swf
|
||||||
|
from moto.swf.models import ActivityType
|
||||||
from moto.swf.exceptions import (
|
from moto.swf.exceptions import (
|
||||||
SWFUnknownResourceFault,
|
SWFUnknownResourceFault,
|
||||||
SWFTypeAlreadyExistsFault,
|
SWFTypeAlreadyExistsFault,
|
||||||
@ -11,6 +12,37 @@ from moto.swf.exceptions import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
# Models
|
||||||
|
def test_short_dict_representation():
|
||||||
|
_type = ActivityType("test-activity", "v1.0")
|
||||||
|
_type.to_short_dict().should.equal({"name": "test-activity", "version": "v1.0"})
|
||||||
|
|
||||||
|
def test_medium_dict_representation():
|
||||||
|
_type = ActivityType("test-activity", "v1.0")
|
||||||
|
_type.to_medium_dict()["activityType"].should.equal(_type.to_short_dict())
|
||||||
|
_type.to_medium_dict()["status"].should.equal("REGISTERED")
|
||||||
|
_type.to_medium_dict().should.contain("creationDate")
|
||||||
|
_type.to_medium_dict().should_not.contain("deprecationDate")
|
||||||
|
_type.to_medium_dict().should_not.contain("description")
|
||||||
|
|
||||||
|
_type.description = "foo bar"
|
||||||
|
_type.to_medium_dict()["description"].should.equal("foo bar")
|
||||||
|
|
||||||
|
_type.status = "DEPRECATED"
|
||||||
|
_type.to_medium_dict().should.contain("deprecationDate")
|
||||||
|
|
||||||
|
def test_full_dict_representation():
|
||||||
|
_type = ActivityType("test-activity", "v1.0")
|
||||||
|
_type.to_full_dict()["typeInfo"].should.equal(_type.to_medium_dict())
|
||||||
|
_type.to_full_dict()["configuration"].should.equal({})
|
||||||
|
|
||||||
|
_type.task_list = "foo"
|
||||||
|
_type.to_full_dict()["configuration"]["defaultTaskList"].should.equal({"name":"foo"})
|
||||||
|
|
||||||
|
_type.default_task_heartbeat_timeout = "60"
|
||||||
|
_type.to_full_dict()["configuration"]["defaultTaskHeartbeatTimeout"].should.equal("60")
|
||||||
|
|
||||||
|
|
||||||
# RegisterActivityType endpoint
|
# RegisterActivityType endpoint
|
||||||
@mock_swf
|
@mock_swf
|
||||||
def test_register_activity_type():
|
def test_register_activity_type():
|
||||||
@ -138,7 +170,6 @@ def test_describe_activity_type():
|
|||||||
|
|
||||||
actype = conn.describe_activity_type("test-domain", "test-activity", "v1.0")
|
actype = conn.describe_activity_type("test-domain", "test-activity", "v1.0")
|
||||||
actype["configuration"]["defaultTaskList"]["name"].should.equal("foo")
|
actype["configuration"]["defaultTaskList"]["name"].should.equal("foo")
|
||||||
actype["configuration"].keys().should_not.contain("defaultTaskScheduleToClose")
|
|
||||||
infos = actype["typeInfo"]
|
infos = actype["typeInfo"]
|
||||||
infos["activityType"]["name"].should.equal("test-activity")
|
infos["activityType"]["name"].should.equal("test-activity")
|
||||||
infos["activityType"]["version"].should.equal("v1.0")
|
infos["activityType"]["version"].should.equal("v1.0")
|
||||||
|
@ -3,6 +3,7 @@ from nose.tools import assert_raises
|
|||||||
from sure import expect
|
from sure import expect
|
||||||
|
|
||||||
from moto import mock_swf
|
from moto import mock_swf
|
||||||
|
from moto.swf.models import Domain
|
||||||
from moto.swf.exceptions import (
|
from moto.swf.exceptions import (
|
||||||
SWFUnknownResourceFault,
|
SWFUnknownResourceFault,
|
||||||
SWFDomainAlreadyExistsFault,
|
SWFDomainAlreadyExistsFault,
|
||||||
@ -11,6 +12,15 @@ from moto.swf.exceptions import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
# Models
|
||||||
|
def test_dict_representation():
|
||||||
|
domain = Domain("foo", "52")
|
||||||
|
domain.to_dict().should.equal({"name":"foo", "status":"REGISTERED"})
|
||||||
|
|
||||||
|
domain.description = "foo bar"
|
||||||
|
domain.to_dict()["description"].should.equal("foo bar")
|
||||||
|
|
||||||
|
|
||||||
# RegisterDomain endpoint
|
# RegisterDomain endpoint
|
||||||
@mock_swf
|
@mock_swf
|
||||||
def test_register_domain():
|
def test_register_domain():
|
||||||
@ -53,7 +63,7 @@ def test_register_with_wrong_parameter_type():
|
|||||||
ex.body["__type"].should.equal("com.amazonaws.swf.base.model#SerializationException")
|
ex.body["__type"].should.equal("com.amazonaws.swf.base.model#SerializationException")
|
||||||
|
|
||||||
|
|
||||||
# ListDomain endpoint
|
# ListDomains endpoint
|
||||||
@mock_swf
|
@mock_swf
|
||||||
def test_list_domains_order():
|
def test_list_domains_order():
|
||||||
conn = boto.connect_swf("the_key", "the_secret")
|
conn = boto.connect_swf("the_key", "the_secret")
|
||||||
|
Loading…
Reference in New Issue
Block a user