mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Teaching objects to load themselves, making the JSON/YAML parsing ambidexterous.
This commit is contained in:
parent
c75aeca435
commit
56b6cb5328
12 changed files with 180 additions and 59 deletions
|
@ -0,0 +1 @@
|
|||
|
85
test/v2/parsing/test_general.py
Normal file
85
test/v2/parsing/test_general.py
Normal file
|
@ -0,0 +1,85 @@
|
|||
# TODO: header
|
||||
|
||||
import unittest
|
||||
from ansible.parsing import load
|
||||
from ansible.errors import AnsibleParserError
|
||||
|
||||
import json
|
||||
|
||||
class MockFile(file):
|
||||
|
||||
def __init__(self, ds, method='json'):
|
||||
self.ds = ds
|
||||
self.method = method
|
||||
|
||||
def read(self):
|
||||
if method == 'json':
|
||||
return json.dumps(ds)
|
||||
elif method == 'yaml':
|
||||
return yaml.dumps(ds)
|
||||
elif method == 'fail':
|
||||
return """
|
||||
AAARGGGGH
|
||||
THIS WON'T PARSE !!!
|
||||
NOOOOOOOOOOOOOOOOOO
|
||||
"""
|
||||
else:
|
||||
raise Exception("untestable serializer")
|
||||
|
||||
def close(self):
|
||||
pass
|
||||
|
||||
class TestGeneralParsing(unittest.TestCase):
|
||||
|
||||
def __init__(self):
|
||||
pass
|
||||
|
||||
def setUp(self):
|
||||
pass
|
||||
|
||||
def tearDown(self):
|
||||
pass
|
||||
|
||||
def parse_json_from_string(self):
|
||||
input = """
|
||||
{
|
||||
"asdf" : "1234",
|
||||
"jkl" : 5678
|
||||
}
|
||||
"""
|
||||
output = load(input)
|
||||
assert output['asdf'] == '1234'
|
||||
assert output['jkl'] == 5678
|
||||
|
||||
def parse_json_from_file(self):
|
||||
output = load(MockFile(dict(a=1,b=2,c=3)),'json')
|
||||
assert ouput == dict(a=1,b=2,c=3)
|
||||
|
||||
def parse_yaml_from_dict(self):
|
||||
input = """
|
||||
asdf: '1234'
|
||||
jkl: 5678
|
||||
"""
|
||||
output = load(input)
|
||||
assert output['asdf'] == '1234'
|
||||
assert output['jkl'] == 5678
|
||||
|
||||
def parse_yaml_from_file(self):
|
||||
output = load(MockFile(dict(a=1,b=2,c=3),'yaml'))
|
||||
assert output == dict(a=1,b=2,c=3)
|
||||
|
||||
def parse_fail(self):
|
||||
input = """
|
||||
TEXT
|
||||
***
|
||||
NOT VALID
|
||||
"""
|
||||
self.failUnlessRaises(load(input), AnsibleParserError)
|
||||
|
||||
def parse_fail_from_file(self):
|
||||
self.failUnlessRaises(load(MockFile(None,'fail')), AnsibleParserError)
|
||||
|
||||
def parse_fail_invalid_type(self):
|
||||
self.failUnlessRaises(3000, AnsibleParsingError)
|
||||
self.failUnlessRaises(dict(a=1,b=2,c=3), AnsibleParserError)
|
||||
|
|
@ -5,6 +5,10 @@ import unittest
|
|||
|
||||
class TestModArgsDwim(unittest.TestCase):
|
||||
|
||||
# TODO: add tests that construct ModuleArgsParser with a task reference
|
||||
# TODO: verify the AnsibleError raised on failure knows the task
|
||||
# and the task knows the line numbers
|
||||
|
||||
def setUp(self):
|
||||
self.m = ModuleArgsParser()
|
||||
pass
|
||||
|
@ -78,4 +82,3 @@ class TestModArgsDwim(unittest.TestCase):
|
|||
assert mod == 'copy'
|
||||
assert args == dict(src='a', dest='b')
|
||||
assert to is 'localhost'
|
||||
|
||||
|
|
|
@ -1,2 +1 @@
|
|||
# TODO: header
|
||||
|
||||
|
|
|
@ -67,5 +67,3 @@ class TestTask(unittest.TestCase):
|
|||
|
||||
def test_delegate_to_parses(self):
|
||||
pass
|
||||
|
||||
|
||||
|
|
|
@ -16,4 +16,31 @@
|
|||
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
class AnsibleError(Exception):
|
||||
pass
|
||||
def __init__(self, message, object=None):
|
||||
self.message = message
|
||||
self.object = object
|
||||
|
||||
# TODO: nice __repr__ message that includes the line number if the object
|
||||
# it was constructed with had the line number
|
||||
|
||||
# TODO: tests for the line number functionality
|
||||
|
||||
class AnsibleParserError(AnsibleError):
|
||||
''' something was detected early that is wrong about a playbook or data file '''
|
||||
pass
|
||||
|
||||
class AnsibleInternalError(AnsibleError):
|
||||
''' internal safeguards tripped, something happened in the code that should never happen '''
|
||||
pass
|
||||
|
||||
class AnsibleRuntimeError(AnsibleError):
|
||||
''' ansible had a problem while running a playbook '''
|
||||
pass
|
||||
|
||||
class AnsibleModuleError(AnsibleRuntimeError):
|
||||
''' a module failed somehow '''
|
||||
pass
|
||||
|
||||
class AnsibleConnectionFailure(AnsibleRuntimeError):
|
||||
''' the transport / connection_plugin had a fatal error '''
|
||||
pass
|
||||
|
|
|
@ -1 +1,18 @@
|
|||
# TODO: header
|
||||
|
||||
from ansible.errors import AnsibleError, AnsibleInternalError
|
||||
|
||||
def load(self, data):
|
||||
|
||||
if instanceof(data, file):
|
||||
fd = open(f)
|
||||
data = fd.read()
|
||||
fd.close()
|
||||
|
||||
if instanceof(data, basestring):
|
||||
try:
|
||||
return json.loads(data)
|
||||
except:
|
||||
return safe_load(data)
|
||||
|
||||
raise AnsibleInternalError("expected file or string, got %s" % type(data))
|
||||
|
|
|
@ -55,15 +55,16 @@ class ModuleArgsParser(object):
|
|||
will tell you about the modules in a predictable way.
|
||||
"""
|
||||
|
||||
def __init__(self):
|
||||
def __init__(self, task=None):
|
||||
self._ds = None
|
||||
self._task = task
|
||||
|
||||
def _get_delegate_to(self):
|
||||
'''
|
||||
Returns the value of the delegate_to key from the task datastructure,
|
||||
or None if the value was not directly specified
|
||||
'''
|
||||
return self._ds.get('delegate_to')
|
||||
return self._ds.get('delegate_to', None)
|
||||
|
||||
def _get_old_style_action(self):
|
||||
'''
|
||||
|
@ -108,29 +109,24 @@ class ModuleArgsParser(object):
|
|||
if 'module' in other_args:
|
||||
del other_args['module']
|
||||
args.update(other_args)
|
||||
|
||||
elif isinstance(action_data, basestring):
|
||||
action_data = action_data.strip()
|
||||
if not action_data:
|
||||
# TODO: change to an AnsibleParsingError so that the
|
||||
# filename/line number can be reported in the error
|
||||
raise AnsibleError("when using 'action:' or 'local_action:', the module name must be specified")
|
||||
raise AnsibleError("when using 'action:' or 'local_action:', the module name must be specified", object=self._task)
|
||||
else:
|
||||
# split up the string based on spaces, where the first
|
||||
# item specified must be a valid module name
|
||||
parts = action_data.split(' ', 1)
|
||||
action = parts[0]
|
||||
if action not in module_finder:
|
||||
# TODO: change to an AnsibleParsingError so that the
|
||||
# filename/line number can be reported in the error
|
||||
raise AnsibleError("the module '%s' was not found in the list of loaded modules")
|
||||
raise AnsibleError("the module '%s' was not found in the list of loaded modules" % action, object=self._task)
|
||||
if len(parts) > 1:
|
||||
args = self._get_args_from_action(action, ' '.join(parts[1:]))
|
||||
else:
|
||||
args = {}
|
||||
else:
|
||||
# TODO: change to an AnsibleParsingError so that the
|
||||
# filename/line number can be reported in the error
|
||||
raise AnsibleError('module args must be specified as a dictionary or string')
|
||||
raise AnsibleError('module args must be specified as a dictionary or string', object=self._task)
|
||||
|
||||
return dict(action=action, args=args, delegate_to=delegate_to)
|
||||
|
||||
|
@ -286,9 +282,7 @@ class ModuleArgsParser(object):
|
|||
# where 'action' or 'local_action' is the key
|
||||
result = self._get_old_style_action()
|
||||
if result is None:
|
||||
# TODO: change to an AnsibleParsingError so that the
|
||||
# filename/line number can be reported in the error
|
||||
raise AnsibleError('no action specified for this task')
|
||||
raise AnsibleError('no action specified for this task', object=self._task)
|
||||
|
||||
# if the action is set to 'shell', we switch that to 'command' and
|
||||
# set the special parameter '_uses_shell' to true in the args dict
|
||||
|
@ -302,11 +296,8 @@ class ModuleArgsParser(object):
|
|||
specified_delegate_to = self._get_delegate_to()
|
||||
if specified_delegate_to is not None:
|
||||
if result['delegate_to'] is not None:
|
||||
# TODO: change to an AnsibleParsingError so that the
|
||||
# filename/line number can be reported in the error
|
||||
raise AnsibleError('delegate_to cannot be used with local_action')
|
||||
else:
|
||||
result['delegate_to'] = specified_delegate_to
|
||||
|
||||
return (result['action'], result['args'], result['delegate_to'])
|
||||
|
||||
|
|
|
@ -4,4 +4,3 @@ from ansible.parsing.yaml.loader import AnsibleLoader
|
|||
def safe_load(stream):
|
||||
''' implements yaml.safe_load(), except using our custom loader class '''
|
||||
return load(stream, AnsibleLoader)
|
||||
|
||||
|
|
|
@ -31,4 +31,3 @@ class Attribute(object):
|
|||
|
||||
class FieldAttribute(Attribute):
|
||||
pass
|
||||
|
||||
|
|
|
@ -16,6 +16,7 @@
|
|||
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
from ansible.playbook.attribute import Attribute, FieldAttribute
|
||||
from ansible.parsing import load as ds_load
|
||||
|
||||
class Base(object):
|
||||
|
||||
|
@ -39,6 +40,9 @@ class Base(object):
|
|||
|
||||
assert ds is not None
|
||||
|
||||
if isinstance(ds, basestring) or isinstance(ds, file):
|
||||
ds = ds_load(ds)
|
||||
|
||||
# we currently don't do anything with private attributes but may
|
||||
# later decide to filter them out of 'ds' here.
|
||||
|
||||
|
@ -107,4 +111,3 @@ class Base(object):
|
|||
return self._attributes[needle]
|
||||
|
||||
raise AttributeError("attribute not found: %s" % needle)
|
||||
|
||||
|
|
|
@ -342,4 +342,3 @@ LEGACY = """
|
|||
raise AnsibleError("with_(plugin), and first_available_file are mutually incompatible in a single task")
|
||||
|
||||
"""
|
||||
|
||||
|
|
Loading…
Reference in a new issue