66f310b227
- Use 401 error to determine work of Murano API in shell via curl - Change tempest parameters in utils - Rename mistralclient.py to workflowclient.py due names conflict in path. Closes-Bug: #1614602 Change-Id: I91bdda91da929b183393c4210af86bce619a53ec
121 lines
4.0 KiB
Python
121 lines
4.0 KiB
Python
# Copyright (c) 2015 OpenStack Foundation.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
|
# implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
|
|
import json
|
|
|
|
import eventlet
|
|
try:
|
|
from mistralclient.api import client as mistralcli
|
|
except ImportError as mistral_import_error:
|
|
mistralcli = None
|
|
from oslo_config import cfg
|
|
from oslo_log import log as logging
|
|
|
|
from murano.common import auth_utils
|
|
from murano.common.i18n import _LW
|
|
from murano.dsl import dsl
|
|
from murano.dsl import session_local_storage
|
|
|
|
CONF = cfg.CONF
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
class MistralError(Exception):
|
|
pass
|
|
|
|
|
|
@dsl.name('io.murano.system.MistralClient')
|
|
class MistralClient(object):
|
|
def __init__(self, this):
|
|
self._owner = this.find_owner('io.murano.Environment')
|
|
|
|
@property
|
|
def _client(self):
|
|
region = None if self._owner is None else self._owner['region']
|
|
return self._create_client(region)
|
|
|
|
@staticmethod
|
|
@session_local_storage.execution_session_memoize
|
|
def _create_client(region):
|
|
if not mistralcli:
|
|
LOG.warning(_LW("Mistral client is not available"))
|
|
raise mistral_import_error
|
|
|
|
mistral_settings = CONF.mistral
|
|
|
|
endpoint_type = mistral_settings.endpoint_type
|
|
service_type = mistral_settings.service_type
|
|
session = auth_utils.get_client_session()
|
|
|
|
mistral_url = mistral_settings.url or session.get_endpoint(
|
|
service_type=service_type,
|
|
endpoint_type=endpoint_type,
|
|
region_name=region)
|
|
auth_ref = session.auth.get_access(session)
|
|
|
|
return mistralcli.client(
|
|
mistral_url=mistral_url,
|
|
project_id=auth_ref.project_id,
|
|
endpoint_type=endpoint_type,
|
|
service_type=service_type,
|
|
auth_token=auth_ref.auth_token,
|
|
user_id=auth_ref.user_id,
|
|
insecure=mistral_settings.insecure,
|
|
cacert=mistral_settings.ca_cert
|
|
)
|
|
|
|
def upload(self, definition):
|
|
self._client.workflows.create(definition)
|
|
|
|
def run(self, name, timeout=600, inputs=None, params=None):
|
|
execution = self._client.executions.create(
|
|
workflow_identifier=name, workflow_input=inputs, params=params)
|
|
# For the fire and forget functionality - when we do not want to wait
|
|
# for the result of the run.
|
|
if timeout == 0:
|
|
return execution.id
|
|
|
|
state = execution.state
|
|
try:
|
|
# While the workflow is running we continue to wait until timeout.
|
|
with eventlet.timeout.Timeout(timeout):
|
|
while state not in ('ERROR', 'SUCCESS'):
|
|
eventlet.sleep(2)
|
|
execution = self._client.executions.get(execution.id)
|
|
state = execution.state
|
|
except eventlet.timeout.Timeout:
|
|
error_message = (
|
|
'Mistral run timed out. Execution id: {0}.').format(
|
|
execution.id)
|
|
raise MistralError(error_message)
|
|
|
|
if state == 'ERROR':
|
|
error_message = ('Mistral execution completed with ERROR.'
|
|
' Execution id: {0}. Output: {1}').format(
|
|
execution.id, execution.output)
|
|
raise MistralError(error_message)
|
|
|
|
# Load the JSON we got from Mistral client to dictionary.
|
|
output = json.loads(execution.output)
|
|
|
|
# Clean the returned dictionary from unnecessary data.
|
|
# We want to keep only flow level outputs.
|
|
output.pop('openstack', None)
|
|
output.pop('__execution', None)
|
|
output.pop('task', None)
|
|
|
|
return output
|