Skip to content

Commit

Permalink
Move inspect utils to mistral-lib
Browse files Browse the repository at this point in the history
Used also in mistral-extra so let move it to here

Change-Id: Ifa60374818ff3fe6da649d04c84c832b313e90d7
  • Loading branch information
eyalb1 committed Jan 21, 2020
1 parent a2fed6c commit 77e640f
Show file tree
Hide file tree
Showing 2 changed files with 187 additions and 0 deletions.
93 changes: 93 additions & 0 deletions mistral_lib/tests/test_inspect_utils.py
@@ -0,0 +1,93 @@
# Copyright 2014 - Mirantis, Inc.
#
# 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 time

from mistral_lib import actions
from mistral_lib.tests import base
from mistral_lib.utils import inspect_utils as i_u


class DummyHTTPAction(actions.Action):
def __init__(self,
url,
method="GET",
params=None,
body=None,
headers=None,
cookies=None,
auth=None,
timeout=None,
allow_redirects=None,
proxies=None,
verify=None):
super(DummyHTTPAction, self).__init__()

def run(self, context):
return context


class DummyRunTask(object):
def __init__(self, wf_ex, wf_spec, task_spec, ctx, triggered_by=None,
handles_error=False):
pass


class ClassWithProperties(object):

a = 1

@property
def prop(self):
pass


class InspectUtilsTest(base.TestCase):
def test_get_parameters_str(self):
action_class = DummyHTTPAction
parameters_str = i_u.get_arg_list_as_str(action_class.__init__)

http_action_params = (
'url, method="GET", params=null, body=null, '
'headers=null, cookies=null, auth=null, '
'timeout=null, allow_redirects=null, '
'proxies=null, verify=null'
)

self.assertEqual(http_action_params, parameters_str)

def test_get_parameters_str_all_mandatory(self):
clazz = DummyRunTask
parameters_str = i_u.get_arg_list_as_str(clazz.__init__)

self.assertEqual(
'wf_ex, wf_spec, task_spec, ctx, triggered_by=null,'
' handles_error=false',
parameters_str
)

def test_get_parameters_str_with_function_parameter(self):

def test_func(foo, bar=None, test_func=time.sleep):
pass

parameters_str = i_u.get_arg_list_as_str(test_func)

self.assertEqual("foo, bar=null", parameters_str)

def test_get_public_fields(self):

attrs = i_u.get_public_fields(ClassWithProperties)

self.assertEqual(attrs, {'a': 1})
94 changes: 94 additions & 0 deletions mistral_lib/utils/inspect_utils.py
@@ -0,0 +1,94 @@
# Copyright 2014 - Mirantis, Inc.
#
# 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 inspect
import json

import six


def get_public_fields(obj):
"""Returns only public fields from object or class."""

public_attributes = [attr for attr in dir(obj)
if not attr.startswith("_")]

public_fields = {}

for attribute_str in public_attributes:
attr = getattr(obj, attribute_str)
is_field = not (inspect.isbuiltin(attr) or
inspect.isfunction(attr) or
inspect.ismethod(attr) or
isinstance(attr, property))

if is_field:
public_fields[attribute_str] = attr

return public_fields


def get_docstring(obj):
return inspect.getdoc(obj)


def get_arg_list(func):
argspec = get_args_spec(func)

args = argspec.args

if 'self' in args:
args.remove('self')

return args


def get_arg_list_as_str(func):
args = getattr(func, "__arguments__", None)
if args:
return args

argspec = get_args_spec(func)
defs = list(argspec.defaults or [])

args = get_arg_list(func)

diff_args_defs = len(args) - len(defs)
arg_str_list = []

for index, default in enumerate(args):
if index >= diff_args_defs:
try:
arg_str_list.append(
"%s=%s" % (
args[index],
json.dumps(defs[index - diff_args_defs])
)
)
except TypeError:
pass
else:
arg_str_list.append("%s" % args[index])

keywords = argspec.keywords if six.PY2 else argspec.varkw
if keywords:
arg_str_list.append("**%s" % keywords)

return ", ".join(arg_str_list)


def get_args_spec(func):
if six.PY2:
return inspect.getargspec(func)
return inspect.getfullargspec(func)

0 comments on commit 77e640f

Please sign in to comment.