Skip to content
This repository has been archived by the owner on Apr 14, 2023. It is now read-only.

Commit

Permalink
add implementation and unit tests for a trigger step
Browse files Browse the repository at this point in the history
  • Loading branch information
rwu committed Sep 25, 2017
1 parent e03427f commit 7dccf11
Show file tree
Hide file tree
Showing 5 changed files with 980 additions and 207 deletions.
228 changes: 21 additions & 207 deletions core/step.py
Original file line number Diff line number Diff line change
@@ -1,38 +1,32 @@
import json
from collections import namedtuple
import logging
from core import contextdecorator
import core.config.config
from core.decorators import ActionResult
from core.executionelement import ExecutionElement
from core.helpers import (get_app_action_api, InvalidElementConstructed, InvalidInput,
dereference_step_routing, format_exception_message)
from core.nextstep import NextStep
from core.widgetsignals import get_widget_signal
from apps import get_app_action
from core.validator import validate_app_action_parameters
import uuid
from core.stepaction import StepAction
from core.steptrigger import StepTrigger

logger = logging.getLogger(__name__)

_Widget = namedtuple('Widget', ['app', 'widget'])


class Step(ExecutionElement):
class Step(StepAction, StepTrigger):
def __init__(self,
name='',
action='',
app='',
device='',
inputs=None,
next_steps=None,
flags=None,
status='Success',
position=None,
widgets=None,
risk=0,
uid=None,
templated=False,
raw_json=None):
"""Initializes a new Step object. A Workflow has many steps that it executes.
"""Initializes a new Step object. A Workflow has many steps that it executes. A Step object can be either a Step
Action or Step Trigger object.
Args:
name (str, optional): The name of the Step object. Defaults to an empty string.
action (str, optional): The name of the action associated with a Step. Defaults to an empty string.
Expand All @@ -51,205 +45,25 @@ def __init__(self,
Created from uuid.uuid4().hex in Python
raw_json (dict, optional): JSON representation of this object. Used for Jinja templating
"""
ExecutionElement.__init__(self, name, uid)
if action == '' or app == '':
raise InvalidElementConstructed('Either both action and app or xml must be '
'specified in step constructor')
self.action = action
self.app = app
self.run, self.input_api = get_app_action_api(self.app, self.action)
get_app_action(self.app, self.run)
inputs = inputs if inputs is not None else {}
self.templated = templated
if not self.templated:
self.input = validate_app_action_parameters(self.input_api, inputs, self.app, self.action)
else:
self.input = inputs
self.device = device
self.risk = risk
self.conditionals = next_steps if next_steps is not None else []
self.position = position if position is not None else {}
self.widgets = [_Widget(widget_app, widget_name)
for (widget_app, widget_name) in widgets] if widgets is not None else []

self.output = None
self.next_up = None
self.raw_json = raw_json if raw_json is not None else {}
self.execution_uid = 'default'
self.results_sock = None
self.execution_uid = None

def _update_json(self, updated_json):
self.action = updated_json['action']
self.app = updated_json['app']
self.device = updated_json['device'] if 'device' in updated_json else ''
self.risk = updated_json['risk'] if 'risk' in updated_json else 0
inputs = {arg['name']: arg['value'] for arg in updated_json['inputs']} if 'inputs' in updated_json else {}
if inputs is not None:
if not self.templated:
self.input = validate_app_action_parameters(self.input_api, inputs, self.app, self.action)
else:
self.input = inputs
else:
self.input = validate_app_action_parameters(self.input_api, {}, self.app, self.action)
self.conditionals = [NextStep.from_json(cond_json) for cond_json in updated_json['next']]

@contextdecorator.context
def render_step(self, **kwargs):
"""Uses JINJA templating to render a Step object.
Args:
kwargs (dict[str]): Arguments to use in the JINJA templating.
"""
if self.templated:
from jinja2 import Environment
env = Environment().from_string(json.dumps(self.raw_json)).render(core.config.config.JINJA_GLOBALS, **kwargs)
self._update_json(updated_json=json.loads(env))

def set_input(self, new_input):
"""Updates the input for a Step object.
Args:
new_input (dict): The new inputs for the Step object.
"""
self.input = validate_app_action_parameters(self.input_api, new_input, self.app, self.action)

def __send_callback(self, callback_name, data={}):
data['sender'] = {}
data['sender']['name'] = self.name
data['sender']['app'] = self.app
data['sender']['action'] = self.action
data['sender']['inputs'] = self.input
data['callback_name'] = callback_name
data['sender']['id'] = self.name
data['sender']['execution_uid'] = self.execution_uid
data['sender']['uid'] = self.uid
if self.results_sock:
self.results_sock.send_json(data)

def execute(self, instance, accumulator):
"""Executes a Step by calling the associated app function.
Args:
instance (App): The instance of an App object to be used to execute the associated function.
accumulator (dict): Dict containing the results of the previous steps
Returns:
The result of the executed function.
"""
self.execution_uid = uuid.uuid4().hex
self.__send_callback('Step Started')
try:
args = dereference_step_routing(self.input, accumulator, 'In step {0}'.format(self.name))
args = validate_app_action_parameters(self.input_api, args, self.app, self.action)
action = get_app_action(self.app, self.run)
result = action(instance, **args)
self.__send_callback('Function Execution Success',
{'name': self.name, 'data': {'result': result.as_json()}})
except InvalidInput as e:
formatted_error = format_exception_message(e)
logger.error('Error calling step {0}. Error: {1}'.format(self.name, formatted_error))
self.__send_callback('Step Input Invalid')
self.output = ActionResult('error: {0}'.format(formatted_error), 'InvalidInput')
raise
except Exception as e:
formatted_error = format_exception_message(e)
logger.error('Error calling step {0}. Error: {1}'.format(self.name, formatted_error))
self.output = ActionResult('error: {0}'.format(formatted_error), 'UnhandledException')
raise
else:
self.output = result
for widget in self.widgets:
get_widget_signal(widget.app, widget.widget).send(self, data=json.dumps({"result": result.as_json()}))
logger.debug('Step {0}-{1} (uid {2}) executed successfully'.format(self.app, self.action, self.uid))
return result

def get_next_step(self, accumulator):
"""Gets the NextStep object to be executed after the current Step.
Args:
accumulator (dict): A record of teh previously-executed steps. Of form {step_name: result}
Returns:
The NextStep object to be executed.
"""

for next_step in self.conditionals:
next_step.results_sock = self.results_sock
next_step = next_step(self.output, accumulator)
if next_step is not None:
self.next_up = next_step
self.__send_callback('Conditionals Executed')
return next_step
super(Step, self).__init__(name, action, app, device, inputs, next_steps, flags, status, position, widgets,
risk, uid, templated, raw_json)

def __repr__(self):
output = {'uid': self.uid,
'name': self.name,
'action': self.action,
'app': self.app,
'device': self.device,
'risk': str(self.risk),
'input': self.input,
'next': [next_step for next_step in self.conditionals],
'nextUp': self.next_up,
'position': self.position,
'widget': str([{'app': widget.app, 'name': widget.widget} for widget in self.widgets])}
if self.output:
output["output"] = self.output.as_json()
return str(output)
for attr, value in self.__dict__.items():
if attr == 'risk':
output[attr] = str(value)
elif attr == 'next_step':
output[attr] = [next_step for next_step in self.conditionals]
elif attr == 'flags':
output[attr] = [flag.as_json() for flag in self.flags]
else:
output[attr] = value

def as_json(self):
"""Gets the JSON representation of a Step object.
Returns:
The JSON representation of a Step object.
"""
output = {"uid": self.uid,
"name": str(self.name),
"action": str(self.action),
"app": str(self.app),
"device": str(self.device),
"risk": self.risk,
"inputs": [{'name': arg_name, 'value': arg_value} for arg_name, arg_value in self.input.items()],
'widgets': [{'app': widget.app, 'name': widget.widget} for widget in self.widgets],
"position": self.position,
"next": [next_step.as_json() for next_step in self.conditionals if next_step.name is not None]}
if self.output:
output["output"] = self.output.as_json()
return output

@staticmethod
def from_json(json_in, position):
"""Forms a Step object from the provided JSON object.
Args:
json_in (dict): The JSON object to convert from.
position (dict): position of the step node of the form {'x': <x position>, 'y': <y position>}
Returns:
The Step object parsed from the JSON object.
"""
device = json_in['device'] if ('device' in json_in
and json_in['device'] is not None
and json_in['device'] != 'None') else ''
risk = json_in['risk'] if 'risk' in json_in else 0
widgets = []
uid = json_in['uid'] if 'uid' in json_in else uuid.uuid4().hex
if 'widgets' in json_in:
widgets = [(widget['app'], widget['name'])
for widget in json_in['widgets'] if ('app' in widget and 'name' in widget)]
conditionals = []
if 'next' in json_in:
conditionals = [NextStep.from_json(next_step) for next_step in json_in['next'] if next_step]
return Step(name=json_in['name'],
action=json_in['action'],
app=json_in['app'],
device=device,
risk=risk,
inputs={arg['name']: arg['value'] for arg in json_in['inputs']},
next_steps=conditionals,
position={key: value for key, value in position.items()},
widgets=widgets,
uid=uid,
templated=json_in['templated'] if 'templated' in json_in else False,
raw_json=json_in)
return str(output)
Loading

0 comments on commit 7dccf11

Please sign in to comment.