Fix ContextView JSON serialization

* With disabled YAQL data output conversion, YAQL may return
  instances of ContextView which can't be properly saved into
  DB. This happens because Mistral serialization code doesn't
  turn on JSON conversion of custom objects, and they are just
  ignored by the "json" lib when it encounters them.
* Fixed how Mistral serializes context for Javascript evaluation
  to address the same problem.
* Implemented __repr__ method of ContextView.
* Removed logging of "data_context" from YAQL evaluation because
  previously it was always empty (because the string represetation
  of ContextView was always "{}") and now it may be very big, like
  megabytes, and the log gets populated too fast. It makes sense to
  log YAQL data context only when an error happened. In this case
  it helps to investigate an issue.
* Added all required unit tests.
* Fixed the tests for disabled YAQL conversion. In fact, they
  didn't test it properly because data conversion wasn't disabled.

Closes-Bug: #1867899
Change-Id: I12b4d0c5f1f49990d8ae09b72f73c0da96254a86
This commit is contained in:
Renat Akhmerov 2020-03-13 16:55:25 +07:00
parent dbfc0bea22
commit 019cffb3ab
7 changed files with 228 additions and 31 deletions

View File

@ -29,12 +29,20 @@ class JsonEncoded(sa.TypeDecorator):
def process_bind_param(self, value, dialect): def process_bind_param(self, value, dialect):
if value is not None: if value is not None:
value = jsonutils.dumps(value) # We need to convert the root of the given object graph into
# a primitive by hand so that we also enable conversion of
# object of custom classes into primitives. Otherwise, they are
# ignored by the "json" lib.
value = jsonutils.dumps(
jsonutils.to_primitive(value, convert_instances=True)
)
return value return value
def process_result_value(self, value, dialect): def process_result_value(self, value, dialect):
if value is not None: if value is not None:
value = jsonutils.loads(value) value = jsonutils.loads(value)
return value return value

View File

@ -208,10 +208,8 @@ class InlineYAQLEvaluator(YAQLEvaluator):
@classmethod @classmethod
def evaluate(cls, expression, data_context): def evaluate(cls, expression, data_context):
LOG.debug( LOG.debug(
"Start to evaluate YAQL expression. " "Starting to evaluate YAQL expression. "
"[expression='%s', context=%s]", "[expression='%s']", expression
expression,
data_context
) )
result = expression result = expression
@ -220,19 +218,17 @@ class InlineYAQLEvaluator(YAQLEvaluator):
if found_expressions: if found_expressions:
for expr in found_expressions: for expr in found_expressions:
trim_expr = expr.strip("<%>") trim_expr = expr.strip("<%>")
evaluated = super(InlineYAQLEvaluator,
cls).evaluate(trim_expr, data_context) evaluated = super(InlineYAQLEvaluator, cls).evaluate(
trim_expr,
data_context
)
if len(expression) == len(expr): if len(expression) == len(expr):
result = evaluated result = evaluated
else: else:
result = result.replace(expr, str(evaluated)) result = result.replace(expr, str(evaluated))
LOG.debug(
"Finished evaluation. [expression='%s', result: %s]",
expression,
utils.cut(result, length=200)
)
return result return result
@classmethod @classmethod

View File

@ -14,6 +14,7 @@
# limitations under the License. # limitations under the License.
from oslo_config import cfg from oslo_config import cfg
from oslo_serialization import jsonutils
from mistral.db.v2 import api as db_api from mistral.db.v2 import api as db_api
from mistral.db.v2.sqlalchemy import models from mistral.db.v2.sqlalchemy import models
@ -1419,3 +1420,59 @@ class DataFlowTest(test_base.BaseTest):
self.assertEqual(2, len(res)) self.assertEqual(2, len(res))
self.assertIn('v1', res) self.assertIn('v1', res)
self.assertIn('v2', res) self.assertIn('v2', res)
def test_context_view_repr(self):
ctx = data_flow.ContextView(
{'k1': 'v1'},
{'k2': 'v2'},
{3: 3}
)
str_repr = str(ctx)
self.assertIsNotNone(str_repr)
self.assertFalse(str_repr == "{}")
self.assertEqual("{'k1': 'v1', 'k2': 'v2', 3: 3}", str_repr)
ctx = data_flow.ContextView()
self.assertEqual('{}', str(ctx))
def test_context_view_as_root_json(self):
ctx = data_flow.ContextView(
{'k1': 'v1'},
{'k2': 'v2'},
)
json_str = jsonutils.dumps(
jsonutils.to_primitive(ctx, convert_instances=True)
)
self.assertIsNotNone(json_str)
self.assertNotEqual('{}', json_str)
# We can't use regular dict comparison because key order
# is not defined.
self.assertIn('"k1": "v1"', json_str)
self.assertIn('"k2": "v2"', json_str)
def test_context_view_as_nested_json(self):
ctx = data_flow.ContextView(
{'k1': 'v1'},
{'k2': 'v2'},
)
d = {'root': ctx}
json_str = jsonutils.dumps(
jsonutils.to_primitive(d, convert_instances=True)
)
self.assertIsNotNone(json_str)
self.assertNotEqual('{"root": {}}', json_str)
# We can't use regular dict comparison because key order
# is not defined.
self.assertIn('"k1": "v1"', json_str)
self.assertIn('"k1": "v1"', json_str)
self.assertIn('"root"', json_str)

View File

@ -16,6 +16,7 @@
from mistral.db.v2 import api as db_api from mistral.db.v2 import api as db_api
from mistral.engine import engine_server from mistral.engine import engine_server
from mistral import exceptions as exc from mistral import exceptions as exc
from mistral.expressions import yaql_expression
from mistral.services import workflows as wf_service from mistral.services import workflows as wf_service
from mistral.tests.unit.engine import base as engine_test_base from mistral.tests.unit.engine import base as engine_test_base
@ -51,6 +52,12 @@ class DisabledYAQLConversionTest(engine_test_base.EngineTestCase):
self.override_config('convert_input_data', False, 'yaql') self.override_config('convert_input_data', False, 'yaql')
self.override_config('convert_output_data', False, 'yaql') self.override_config('convert_output_data', False, 'yaql')
# At this point YAQL engine has already been initialized with the
# default value of config options. So we need to set the corresponding
# constant to None so it gets initialized again with the new values
# upon the first use.
yaql_expression.YAQL_ENGINE = None
wf_text = """--- wf_text = """---
version: '2.0' version: '2.0'
@ -101,6 +108,10 @@ class DisabledYAQLConversionTest(engine_test_base.EngineTestCase):
self.override_config('convert_input_data', True, 'yaql') self.override_config('convert_input_data', True, 'yaql')
self.override_config('convert_output_data', False, 'yaql') self.override_config('convert_output_data', False, 'yaql')
# Setting YAQL engine to None so it reinitialized again with the
# right values upon the next use.
yaql_expression.YAQL_ENGINE = None
eng_svc = engine_server.get_oslo_service(setup_profiler=False) eng_svc = engine_server.get_oslo_service(setup_profiler=False)
self.assertRaisesWithMessage( self.assertRaisesWithMessage(
@ -109,3 +120,50 @@ class DisabledYAQLConversionTest(engine_test_base.EngineTestCase):
"so 'yaql.convert_input_data' must also be set to False.", "so 'yaql.convert_input_data' must also be set to False.",
eng_svc.start eng_svc.start
) )
def test_root_context(self):
# Both input and output data conversion in YAQL need to be disabled
# so that we're sure that there won't be any surprises from YAQL
# like some YAQL internal types included in expression results.
self.override_config('convert_input_data', False, 'yaql')
self.override_config('convert_output_data', False, 'yaql')
# Setting YAQL engine to None so it reinitialized again with the
# right values upon the next use.
yaql_expression.YAQL_ENGINE = None
wf_text = """---
version: '2.0'
wf:
input:
- param: default_val
tasks:
task1:
action: std.echo output=<% $ %>
publish:
result: <% task().result %>
"""
wf_service.create_workflows(wf_text)
# Start workflow.
wf_ex = self.engine.start_workflow('wf')
self.await_workflow_success(wf_ex.id)
with db_api.transaction():
# Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id)
t_ex = self._assert_single_item(
wf_ex.task_executions,
name='task1'
)
action_ex = t_ex.action_executions[0]
self.assertTrue(len(action_ex.input) > 0)
self.assertIn('output', action_ex.input)
self.assertIn('param', action_ex.input['output'])

View File

@ -18,12 +18,12 @@ from oslo_utils import importutils
import testtools import testtools
from mistral.db.v2 import api as db_api from mistral.db.v2 import api as db_api
from mistral.expressions import yaql_expression
from mistral.services import workflows as wf_service from mistral.services import workflows as wf_service
from mistral.tests.unit.engine import base from mistral.tests.unit.engine import base
from mistral.utils import javascript from mistral.utils import javascript
from mistral.workflow import states from mistral.workflow import states
# Use the set_default method to set value otherwise in certain test cases # Use the set_default method to set value otherwise in certain test cases
# the change in value is not permanent. # the change in value is not permanent.
cfg.CONF.set_default('auth_enable', False, group='pecan') cfg.CONF.set_default('auth_enable', False, group='pecan')
@ -60,10 +60,8 @@ class JavaScriptEngineTest(base.EngineTestCase):
'This test requires that py_mini_racer library was ' 'This test requires that py_mini_racer library was '
'installed') 'installed')
def test_py_mini_racer_javascript_action(self): def test_py_mini_racer_javascript_action(self):
cfg.CONF.set_default( cfg.CONF.set_default('js_implementation', 'py_mini_racer')
'js_implementation',
'py_mini_racer'
)
length = 1000 length = 1000
wf_service.create_workflows(JAVASCRIPT_WORKFLOW) wf_service.create_workflows(JAVASCRIPT_WORKFLOW)
@ -85,6 +83,58 @@ class JavaScriptEngineTest(base.EngineTestCase):
self.assertEqual(length / 2, task_ex.published['res']) self.assertEqual(length / 2, task_ex.published['res'])
@testtools.skipIf(not importutils.try_import('py_mini_racer'),
'This test requires that py_mini_racer library was '
'installed')
def test_py_mini_racer_javascript_action_disabled_yaql_conversion(self):
cfg.CONF.set_default('js_implementation', 'py_mini_racer')
# Both input and output data conversion in YAQL need to be disabled
# so that we're sure that there won't be any surprises from YAQL
# like some YAQL internal types included in expression results.
self.override_config('convert_input_data', False, 'yaql')
self.override_config('convert_output_data', False, 'yaql')
# Setting YAQL engine to None so it reinitialized again with the
# right values upon the next use.
yaql_expression.YAQL_ENGINE = None
wf_text = """---
version: '2.0'
wf:
input:
- param: default_val
tasks:
task1:
action: std.js
input:
context: <% $ %>
script: >
return $.param
publish:
result: <% task().result %>
"""
wf_service.create_workflows(wf_text)
# Start workflow.
wf_ex = self.engine.start_workflow('wf')
self.await_workflow_success(wf_ex.id)
with db_api.transaction():
# Note: We need to reread execution to access related tasks.
wf_ex = db_api.get_workflow_execution(wf_ex.id)
t_ex = self._assert_single_item(
wf_ex.task_executions,
name='task1'
)
self.assertDictEqual({'result': 'default_val'}, t_ex.published)
@mock.patch.object(javascript, 'evaluate', fake_evaluate) @mock.patch.object(javascript, 'evaluate', fake_evaluate)
def test_fake_javascript_action_data_context(self): def test_fake_javascript_action_data_context(self):
length = 1000 length = 1000

View File

@ -1,4 +1,5 @@
# Copyright 2015 - Mirantis, Inc. # Copyright 2015 - Mirantis, Inc.
# Copyright 2020 - Nokia Software.
# #
# Licensed under the Apache License, Version 2.0 (the "License"); # Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License. # you may not use this file except in compliance with the License.
@ -13,11 +14,11 @@
# limitations under the License. # limitations under the License.
import abc import abc
import json
from mistral import config as cfg from mistral import config as cfg
from mistral import exceptions as exc from mistral import exceptions as exc
from oslo_serialization import jsonutils
from oslo_utils import importutils from oslo_utils import importutils
from stevedore import driver from stevedore import driver
from stevedore import extension from stevedore import extension
@ -46,44 +47,66 @@ class JSEvaluator(object):
class PyV8Evaluator(JSEvaluator): class PyV8Evaluator(JSEvaluator):
@classmethod @classmethod
def evaluate(cls, script, context): def evaluate(cls, script, ctx):
if not _PYV8: if not _PYV8:
raise exc.MistralException( raise exc.MistralException(
"PyV8 module is not available. Please install PyV8." "PyV8 module is not available. Please install PyV8."
) )
with _PYV8.JSContext() as ctx: with _PYV8.JSContext() as js_ctx:
# Prepare data context and way for interaction with it. # Prepare data context and way for interaction with it.
ctx.eval('$ = %s' % json.dumps(context)) # NOTE: it's important to enable conversion of custom types
# into JSON to account for classes like ContextView.
ctx_str = jsonutils.dumps(
jsonutils.to_primitive(ctx, convert_instances=True)
)
js_ctx.eval('$ = %s' % ctx_str)
result = js_ctx.eval(script)
result = ctx.eval(script)
return _PYV8.convert(result) return _PYV8.convert(result)
class V8EvalEvaluator(JSEvaluator): class V8EvalEvaluator(JSEvaluator):
@classmethod @classmethod
def evaluate(cls, script, context): def evaluate(cls, script, ctx):
if not _V8EVAL: if not _V8EVAL:
raise exc.MistralException( raise exc.MistralException(
"v8eval module is not available. Please install v8eval." "v8eval module is not available. Please install v8eval."
) )
v8 = _V8EVAL.V8() v8 = _V8EVAL.V8()
return v8.eval(('$ = %s; %s' % (json.dumps(context), script)).encode(
encoding='UTF-8')) # NOTE: it's important to enable conversion of custom types
# into JSON to account for classes like ContextView.
ctx_str = jsonutils.dumps(
jsonutils.to_primitive(ctx, convert_instances=True)
)
return v8.eval(
('$ = %s; %s' % (ctx_str, script)).encode(encoding='UTF-8')
)
class PyMiniRacerEvaluator(JSEvaluator): class PyMiniRacerEvaluator(JSEvaluator):
@classmethod @classmethod
def evaluate(cls, script, context): def evaluate(cls, script, ctx):
if not _PY_MINI_RACER: if not _PY_MINI_RACER:
raise exc.MistralException( raise exc.MistralException(
"PyMiniRacer module is not available. Please install " "PyMiniRacer module is not available. Please install "
"PyMiniRacer." "PyMiniRacer."
) )
ctx = _PY_MINI_RACER.MiniRacer() js_ctx = _PY_MINI_RACER.MiniRacer()
return ctx.eval(('$ = {}; {}'.format(json.dumps(context), script)))
# NOTE: it's important to enable conversion of custom types
# into JSON to account for classes like ContextView.
ctx_str = jsonutils.dumps(
jsonutils.to_primitive(ctx, convert_instances=True)
)
return js_ctx.eval(('$ = {}; {}'.format(ctx_str, script)))
_mgr = extension.ExtensionManager( _mgr = extension.ExtensionManager(
@ -107,5 +130,5 @@ def get_js_evaluator():
return _EVALUATOR return _EVALUATOR
def evaluate(script, context): def evaluate(script, ctx):
return get_js_evaluator().evaluate(script, context) return get_js_evaluator().evaluate(script, ctx)

View File

@ -124,6 +124,11 @@ class ContextView(dict):
def __delitem__(self, key): def __delitem__(self, key):
self._raise_immutable_error() self._raise_immutable_error()
def __repr__(self):
return ''.join(
['{', ', '.join([str(d)[1:-1] for d in self.dicts]), '}']
)
def evaluate_upstream_context(upstream_task_execs): def evaluate_upstream_context(upstream_task_execs):
published_vars = {} published_vars = {}