Merge "Use on-clause and retry_policy get_spec for validation"

This commit is contained in:
Zuul 2018-07-20 11:47:53 +00:00 committed by Gerrit Code Review
commit 6c4e089f00
5 changed files with 131 additions and 87 deletions

View File

@ -19,25 +19,80 @@ from mistral.lang import types
from mistral.lang.v2 import base
from mistral.lang.v2 import publish
class OnClauseSpec(base.BaseSpec):
_simple_schema = {
NEXT_TASK = {
"oneOf": [
types.NONEMPTY_STRING,
types.UNIQUE_STRING_OR_EXPRESSION_CONDITION_LIST
{
"type": "string",
"pattern": "^\w+$",
"description": "Task name (e.g.: `task1`)"
},
{
"type": "string",
"pattern": "^\w+ \w+=(.*)$",
"description": "Task name with dict parameter "
"(e.g.: `fail msg=\"test\"`, "
"`fail msg=<% task() %>`)"
},
{
"type": "string",
"pattern": "^\w+\\(\w+=(.*)\\)$",
"description": "Task name with func parameter "
"(e.g.: `fail(msg=\"test\")`, "
"`fail(msg=<% task() %>)`)"
}
]
}
_advanced_schema = {
TASK_WITH_EXPRESSION = {
"type": "object",
"minProperties": 1,
"maxProperties": 1,
"patternProperties": {
x['pattern']: types.EXPRESSION for x in NEXT_TASK['oneOf']
},
"description": "All next task variants plus expression (e.g.: "
"`task1: <% $.vm_id != null %>`, "
"`fail(msg=\"test\"): <% $.vm_id != null %>`)."
}
LIST_OF_TASKS = {
"type": "array",
"items": {
"oneOf": [
NEXT_TASK,
TASK_WITH_EXPRESSION
]
},
"uniqueItems": True,
"minItems": 1
}
ADVANCED_PUBLISHING_DICT = {
"type": "object",
"minProperties": 1,
"properties": {
"publish": types.NONEMPTY_DICT,
"next": _simple_schema,
"publish": publish.PublishSpec.get_schema(),
"next": {
"oneOf": [
NEXT_TASK,
TASK_WITH_EXPRESSION,
LIST_OF_TASKS
]
}
},
"additionalProperties": False
}
_schema = {"oneOf": [_simple_schema, _advanced_schema]}
class OnClauseSpec(base.BaseSpec):
_schema = {
"oneOf": [
NEXT_TASK,
TASK_WITH_EXPRESSION,
LIST_OF_TASKS,
ADVANCED_PUBLISHING_DICT
]
}
def __init__(self, data):
super(OnClauseSpec, self).__init__(data)

View File

@ -23,7 +23,7 @@ class PoliciesSpec(base.BaseSpec):
_schema = {
"type": "object",
"properties": {
"retry": types.ANY,
"retry": retry_policy.RetrySpec.get_schema(),
"wait-before": types.EXPRESSION_OR_POSITIVE_INTEGER,
"wait-after": types.EXPRESSION_OR_POSITIVE_INTEGER,
"timeout": types.EXPRESSION_OR_POSITIVE_INTEGER,

View File

@ -19,6 +19,7 @@ from mistral.lang import types
from mistral.lang.v2 import base
from mistral.lang.v2 import on_clause
from mistral.lang.v2 import policies
from mistral.lang.v2 import retry_policy
# TODO(rakhmerov): This specification should be broken into two separate
@ -31,15 +32,15 @@ class TaskDefaultsSpec(base.BaseSpec):
_schema = {
"type": "object",
"properties": {
"retry": types.ANY,
"wait-before": types.ANY,
"wait-after": types.ANY,
"timeout": types.ANY,
"pause-before": types.ANY,
"concurrency": types.ANY,
"on-complete": types.ANY,
"on-success": types.ANY,
"on-error": types.ANY,
"retry": retry_policy.RetrySpec.get_schema(),
"wait-before": types.EXPRESSION_OR_POSITIVE_INTEGER,
"wait-after": types.EXPRESSION_OR_POSITIVE_INTEGER,
"timeout": types.EXPRESSION_OR_POSITIVE_INTEGER,
"pause-before": types.EXPRESSION_OR_BOOLEAN,
"concurrency": types.EXPRESSION_OR_POSITIVE_INTEGER,
"on-complete": on_clause.OnClauseSpec.get_schema(),
"on-success": on_clause.OnClauseSpec.get_schema(),
"on-error": on_clause.OnClauseSpec.get_schema(),
"safe-rerun": types.EXPRESSION_OR_BOOLEAN,
"requires": {
"oneOf": [types.NONEMPTY_STRING, types.UNIQUE_STRING_LIST]

View File

@ -25,6 +25,7 @@ from mistral.lang.v2 import base
from mistral.lang.v2 import on_clause
from mistral.lang.v2 import policies
from mistral.lang.v2 import publish
from mistral.lang.v2 import retry_policy
from mistral import utils
from mistral.workflow import states
@ -73,12 +74,12 @@ class TaskSpec(base.BaseSpec):
},
"publish": types.NONEMPTY_DICT,
"publish-on-error": types.NONEMPTY_DICT,
"retry": types.ANY,
"wait-before": types.ANY,
"wait-after": types.ANY,
"timeout": types.ANY,
"pause-before": types.ANY,
"concurrency": types.ANY,
"retry": retry_policy.RetrySpec.get_schema(),
"wait-before": types.EXPRESSION_OR_POSITIVE_INTEGER,
"wait-after": types.EXPRESSION_OR_POSITIVE_INTEGER,
"timeout": types.EXPRESSION_OR_POSITIVE_INTEGER,
"pause-before": types.EXPRESSION_OR_BOOLEAN,
"concurrency": types.EXPRESSION_OR_POSITIVE_INTEGER,
"target": types.NONEMPTY_STRING,
"keep-result": types.EXPRESSION_OR_BOOLEAN,
"safe-rerun": types.EXPRESSION_OR_BOOLEAN
@ -279,9 +280,9 @@ class DirectWorkflowTaskSpec(TaskSpec):
types.POSITIVE_INTEGER
]
},
"on-complete": types.ANY,
"on-success": types.ANY,
"on-error": types.ANY
"on-complete": on_clause.OnClauseSpec.get_schema(),
"on-success": on_clause.OnClauseSpec.get_schema(),
"on-error": on_clause.OnClauseSpec.get_schema()
}
}

View File

@ -276,57 +276,35 @@ class TaskSpecValidation(v2_base.WorkflowSpecValidationTestCase):
def test_direct_transition(self):
tests = [
({'on-success': ['email']}, False),
({'on-success': [{'email': '<% 1 %>'}]}, False),
({'on-success': [{'email': '<% 1 %>'}, 'echo']}, False),
({'on-success': [{'email': '<% $.v1 in $.v2 %>'}]}, False),
({'on-success': [{'email': '<% * %>'}]}, True),
({'on-success': [{'email': '{{ 1 }}'}]}, False),
({'on-success': [{'email': '{{ 1 }}'}, 'echo']}, False),
({'on-success': [{'email': '{{ _.v1 in _.v2 }}'}]}, False),
({'on-success': [{'email': '{{ * }}'}]}, True),
({'on-success': 'email'}, False),
({'on-success': None}, True),
({'on-success': ['']}, True),
({'on-success': []}, True),
({'on-success': ['email', 'email']}, True),
({'on-success': ['email', 12345]}, True),
({'on-error': ['email']}, False),
({'on-error': [{'email': '<% 1 %>'}]}, False),
({'on-error': [{'email': '<% 1 %>'}, 'echo']}, False),
({'on-error': [{'email': '<% $.v1 in $.v2 %>'}]}, False),
({'on-error': [{'email': '<% * %>'}]}, True),
({'on-error': [{'email': '{{ 1 }}'}]}, False),
({'on-error': [{'email': '{{ 1 }}'}, 'echo']}, False),
({'on-error': [{'email': '{{ _.v1 in _.v2 }}'}]}, False),
({'on-error': [{'email': '{{ * }}'}]}, True),
({'on-error': 'email'}, False),
({'on-error': None}, True),
({'on-error': ['']}, True),
({'on-error': []}, True),
({'on-error': ['email', 'email']}, True),
({'on-error': ['email', 12345]}, True),
({'on-complete': ['email']}, False),
({'on-complete': [{'email': '<% 1 %>'}]}, False),
({'on-complete': [{'email': '<% 1 %>'}, 'echo']}, False),
({'on-complete': [{'email': '<% $.v1 in $.v2 %>'}]}, False),
({'on-complete': [{'email': '<% * %>'}]}, True),
({'on-complete': [{'email': '{{ 1 }}'}]}, False),
({'on-complete': [{'email': '{{ 1 }}'}, 'echo']}, False),
({'on-complete': [{'email': '{{ _.v1 in _.v2 }}'}]}, False),
({'on-complete': [{'email': '{{ * }}'}]}, True),
({'on-complete': 'email'}, False),
({'on-complete': None}, True),
({'on-complete': ['']}, True),
({'on-complete': []}, True),
({'on-complete': ['email', 'email']}, True),
({'on-complete': ['email', 12345]}, True)
(['email'], False),
(['email%'], True),
([{'email': '<% 1 %>'}], False),
([{'email': '<% 1 %>'}, {'email': '<% 1 %>'}], True),
([{'email': '<% 1 %>', 'more_email': '<% 2 %>'}], True),
(['email'], False),
([{'email': '<% 1 %>'}, 'echo'], False),
([{'email': '<% $.v1 in $.v2 %>'}], False),
([{'email': '<% * %>'}], True),
([{'email': '{{ 1 }}'}], False),
([{'email': '{{ 1 }}'}, 'echo'], False),
([{'email': '{{ _.v1 in _.v2 }}'}], False),
([{'email': '{{ * }}'}], True),
('email', False),
('fail msg="<% task().result %>"', False),
('fail(msg=<% task() %>)', False),
(None, True),
([''], True),
([], True),
(['email', 'email'], True),
(['email', 12345], True)
]
for transition, expect_error in tests:
for on_clause_key in ['on-error', 'on-success', 'on-complete']:
for on_clause_value, expect_error in tests:
overlay = {'test': {'tasks': {}}}
utils.merge_dicts(overlay['test']['tasks'], {'get': transition})
utils.merge_dicts(overlay['test']['tasks'],
{'get': {on_clause_key: on_clause_value}})
self._parse_dsl_spec(
add_tasks=True,
@ -431,6 +409,9 @@ class TaskSpecValidation(v2_base.WorkflowSpecValidationTestCase):
({'on-success': {'next': 'email'}}, False),
({'on-success': {'next': ['email']}}, False),
({'on-success': {'next': [{'email': 'email'}]}}, True),
({'on-success': {'next': [{'email': 'email',
'more_email': 'more_email'}]}}, True),
({'on-success': {'next': {'email': 'email'}}}, True),
({'on-error': {'publish': {'var1': 1234}}}, True),
({'on-error': {'publish': {'branch': {'var1': 1234}}}}, False),
(
@ -526,6 +507,9 @@ class TaskSpecValidation(v2_base.WorkflowSpecValidationTestCase):
({'on-error': {'next': 'email'}}, False),
({'on-error': {'next': ['email']}}, False),
({'on-error': {'next': [{'email': 'email'}]}}, True),
({'on-error': {'next': [{'email': 'email',
'more_email': 'more_email'}]}}, True),
({'on-error': {'next': {'email': 'email'}}}, True),
({'on-complete': {'publish': {'var1': 1234}}}, True),
({'on-complete': {'publish': {'branch': {'var1': 1234}}}}, False),
(
@ -620,7 +604,10 @@ class TaskSpecValidation(v2_base.WorkflowSpecValidationTestCase):
({'on-complete': {'next': [{'email': '<% $.v1 %>'}]}}, False),
({'on-complete': {'next': 'email'}}, False),
({'on-complete': {'next': ['email']}}, False),
({'on-complete': {'next': [{'email': 'email'}]}}, True)
({'on-complete': {'next': [{'email': 'email'}]}}, True),
({'on-complete': {'next': [{'email': 'email',
'more_email': 'more_email'}]}}, True),
({'on-complete': {'next': {'email': 'email'}}}, True)
]
for transition, expect_error in tests: