2018-12-04 13:36:56 -08:00
|
|
|
# Copyright (C) 2018 Google 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.
|
|
|
|
|
|
|
|
|
|
"""Types of values."""
|
|
|
|
|
|
|
|
|
|
from __future__ import absolute_import
|
|
|
|
|
from __future__ import division
|
|
|
|
|
from __future__ import print_function
|
|
|
|
|
|
|
|
|
|
import inspect
|
|
|
|
|
|
2020-02-28 09:37:23 -08:00
|
|
|
from fire import inspectutils
|
2018-12-04 13:36:56 -08:00
|
|
|
import six
|
|
|
|
|
|
|
|
|
|
|
2019-07-22 10:42:16 -07:00
|
|
|
VALUE_TYPES = (bool, six.string_types, six.integer_types, float, complex,
|
|
|
|
|
type(Ellipsis), type(None), type(NotImplemented))
|
2018-12-04 13:36:56 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def IsGroup(component):
|
|
|
|
|
# TODO(dbieber): Check if there are any subcomponents.
|
|
|
|
|
return not IsCommand(component) and not IsValue(component)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def IsCommand(component):
|
|
|
|
|
return inspect.isroutine(component) or inspect.isclass(component)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def IsValue(component):
|
2020-02-28 09:13:42 -08:00
|
|
|
return isinstance(component, VALUE_TYPES) or HasCustomStr(component)
|
2019-07-22 10:41:09 -07:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def IsSimpleGroup(component):
|
|
|
|
|
"""If a group is simple enough, then we treat it as a value in PrintResult.
|
|
|
|
|
|
|
|
|
|
Only if a group contains all value types do we consider it simple enough to
|
|
|
|
|
print as a value.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
component: The group to check for value-group status.
|
|
|
|
|
Returns:
|
|
|
|
|
A boolean indicating if the group should be treated as a value for printing
|
|
|
|
|
purposes.
|
|
|
|
|
"""
|
|
|
|
|
assert isinstance(component, dict)
|
|
|
|
|
for unused_key, value in component.items():
|
|
|
|
|
if not IsValue(value) and not isinstance(value, (list, dict)):
|
|
|
|
|
return False
|
|
|
|
|
return True
|
2020-02-28 09:13:42 -08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def HasCustomStr(component):
|
|
|
|
|
"""Determines if a component has a custom __str__ method.
|
|
|
|
|
|
|
|
|
|
Uses inspect.classify_class_attrs to determine the origin of the object's
|
|
|
|
|
__str__ method, if one is present. If it defined by `object` itself, then
|
|
|
|
|
it is not considered custom. Otherwise it is. This means that the __str__
|
|
|
|
|
methods of primitives like ints and floats are considered custom.
|
|
|
|
|
|
|
|
|
|
Objects with custom __str__ methods are treated as values and can be
|
|
|
|
|
serialized in places where more complex objects would have their help screen
|
|
|
|
|
shown instead.
|
|
|
|
|
|
|
|
|
|
Args:
|
|
|
|
|
component: The object to check for a custom __str__ method.
|
|
|
|
|
Returns:
|
|
|
|
|
Whether `component` has a custom __str__ method.
|
|
|
|
|
"""
|
|
|
|
|
if hasattr(component, '__str__'):
|
2020-02-28 09:37:23 -08:00
|
|
|
class_attrs = inspectutils.GetClassAttrsDict(type(component)) or {}
|
2020-02-28 09:13:42 -08:00
|
|
|
str_attr = class_attrs.get('__str__')
|
|
|
|
|
if str_attr and str_attr.defining_class is not object:
|
|
|
|
|
return True
|
|
|
|
|
return False
|