import unittest
from stone.backend import (
remove_aliases_from_api,
CodeBackend
)
from stone.ir import (
Alias,
Api,
ApiNamespace,
ApiRoute,
List,
Nullable,
Boolean,
String,
Struct,
StructField,
Union,
UnionField,
resolve_aliases,
strip_alias
)
_MYPY = False
if _MYPY:
import typing
import argparse
class _Tester(CodeBackend):
def generate(self, api):
pass
class _TesterCmdline(CodeBackend):
cmdline_parser = argparse.ArgumentParser()
cmdline_parser.add_argument('-v', '--verbose', action='store_true')
def generate(self, api):
pass
class TestBackend(unittest.TestCase):
def test_api_namespace(self):
ns = ApiNamespace('files')
a1 = Struct('A1', None, ns)
a1.set_attributes(None, [StructField('f1', Boolean(), None, None)])
a2 = Struct('A2', None, ns)
a2.set_attributes(None, [StructField('f2', Boolean(), None, None)])
l1 = List(a1)
s = String()
route = ApiRoute('test/route', 1, None)
route.set_attributes(None, None, l1, a2, s, None)
ns.add_route(route)
route_io = ns.get_route_io_data_types()
self.assertIn(a1, route_io)
self.assertIn(a2, route_io)
self.assertNotIn(l1, route_io)
self.assertNotIn(s, route_io)
def test_code_backend_helpers(self):
t = _Tester(None, [])
self.assertEqual(t.filter_out_none_valued_keys({}), {})
self.assertEqual(t.filter_out_none_valued_keys({'a': None}), {})
self.assertEqual(t.filter_out_none_valued_keys({'a': None, 'b': 3}), {'b': 3})
def test_code_backend_basic_emitters(self):
t = _Tester(None, [])
t.emit('hello')
self.assertEqual(t.output_buffer_to_string(), 'hello\n')
t.clear_output_buffer()
self.assertRaises(AssertionError, lambda: t.emit('hello\n'))
t.emit('hello')
with t.indent():
t.emit('world')
with t.indent():
t.emit('!')
expected = """\
hello
world
!
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
with t.indent():
t.emit_wrapped_text('Colorless green ideas sleep furiously',
prefix='$', initial_prefix='>',
subsequent_prefix='|', width=13)
expected = """\
$>Colorless
$|green
$|ideas
$|sleep
$|furiously
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
def test_code_backend_list_gen(self):
t = _Tester(None, [])
t.generate_multiline_list(['a=1', 'b=2'])
expected = """\
(a=1,
b=2)
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1', 'b=2'], 'def __init__', after=':')
expected = """\
def __init__(a=1,
b=2):
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1', 'b=2'], before='function_to_call', compact=False)
expected = """\
function_to_call(
a=1,
b=2,
)
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1', 'b=2'], 'function_to_call',
compact=False, skip_last_sep=True)
expected = """\
function_to_call(
a=1,
b=2
)
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1', 'b=2'], 'def func', ':',
compact=False, skip_last_sep=True)
expected = """\
def func(
a=1,
b=2
):
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1'], 'function_to_call', compact=False)
expected = 'function_to_call(a=1)\n'
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['a=1'], 'function_to_call', compact=True)
expected = 'function_to_call(a=1)\n'
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list([], 'function_to_call', compact=False)
expected = 'function_to_call()\n'
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list([], 'function_to_call', compact=True)
expected = 'function_to_call()\n'
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
t.generate_multiline_list(['String'], 'List', delim=('<', '>'), compact=True)
expected = 'List<String>\n'
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
def test_code_backend_block_gen(self):
t = _Tester(None, [])
with t.block('int sq(int x)', ';'):
t.emit('return x*x;')
expected = """\
int sq(int x) {
return x*x;
};
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
with t.block('int sq(int x)', allman=True):
t.emit('return x*x;')
expected = """\
int sq(int x)
{
return x*x;
}
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
with t.block('int sq(int x)', delim=('<', '>'), dent=8):
t.emit('return x*x;')
expected = """\
int sq(int x) <
return x*x;
>
"""
self.assertEqual(t.output_buffer_to_string(), expected)
t.clear_output_buffer()
def test_backend_cmdline(self):
t = _TesterCmdline(None, ['-v'])
self.assertTrue(t.args.verbose)
def test_resolve_aliases(self):
first_alias = Alias(None, None, None)
first_alias.data_type = String()
resolved_type = resolve_aliases(first_alias.data_type)
self.assertIsInstance(resolved_type, String)
self.assertIsInstance(first_alias.data_type, String)
first_alias = Alias(None, None, None)
second_alias = Alias(None, None, None)
first_alias.data_type = second_alias
second_alias.data_type = String()
resolved_type = resolve_aliases(first_alias.data_type)
first_alias.data_type = resolved_type
self.assertIsInstance(resolved_type, String)
self.assertIsInstance(first_alias.data_type, String)
self.assertIsInstance(second_alias.data_type, String)
def test_strip_alias(self):
first_alias = Alias(None, None, None)
second_alias = Alias(None, None, None)
third_alias = Alias(None, None, None)
first_alias.data_type = second_alias
second_alias.data_type = third_alias
third_alias.data_type = String()
test_struct = Struct('TestStruct', None, None)
test_struct.set_attributes(None, [
StructField('field1', List(Nullable(first_alias)), None, None)
])
curr_type = first_alias
while hasattr(curr_type, 'data_type'):
curr_type.data_type = resolve_aliases(curr_type.data_type)
curr_type = curr_type.data_type
self.assertIsInstance(first_alias.data_type, String)
self.assertEqual(len(test_struct.fields), 1)
field = test_struct.fields[0]
strip_alias(field.data_type)
list_type = field.data_type
self.assertIsInstance(list_type, List)
nullable_type = list_type.data_type
self.assertIsInstance(nullable_type, Nullable)
string_type = nullable_type.data_type
self.assertIsInstance(string_type, String)
def test_preserve_aliases_from_api(self):
api = Api(version="0.1b1")
api.ensure_namespace('preserve_alias')
api.ensure_namespace('zzzz')
ns = api.namespaces['preserve_alias']
imported = api.namespaces['zzzz']
namespace_id = Alias('NamespaceId', ns, None)
namespace_id.data_type = String()
shared_folder_id = Alias('SharedFolderId', ns, None)
shared_folder_id.set_attributes(None, namespace_id)
path_root_id = Alias('PathRootId', ns, None)
path_root_id.set_attributes(None, shared_folder_id)
foo_alias = Alias('FooAlias', None, None)
foo_alias.set_attributes(None, String())
bar_alias = Alias('BarAlias', None, None)
bar_alias.set_attributes(None, foo_alias)
ns.add_alias(namespace_id)
ns.add_alias(shared_folder_id)
ns.add_alias(path_root_id)
imported.add_alias(foo_alias)
imported.add_alias(bar_alias)
test_struct = Struct('TestStruct', ns, None)
test_struct.set_attributes(None, [StructField('field1', path_root_id, None, None)])
test_union = Union('TestUnion', ns, None, None)
test_union.set_attributes(None, [UnionField('test', path_root_id, None, None)])
dependent_struct = Struct('DependentStruct', ns, None)
dependent_struct.set_attributes(None, [
StructField('field_alias', imported.alias_by_name['BarAlias'], None, None)
])
ns.add_data_type(test_struct)
ns.add_data_type(test_union)
ns.add_data_type(dependent_struct)
struct_alias = Alias('StructAlias', ns, None)
struct_alias.set_attributes(None, test_struct)
ns.add_alias(struct_alias)
self.assertEqual(len(api.namespaces), 2)
self.assertTrue('preserve_alias' in api.namespaces)
self.assertTrue('zzzz' in api.namespaces)
ns = api.namespaces['preserve_alias']
imported = api.namespaces['zzzz']
self.assertEqual(len(ns.aliases), 4)
self.assertEqual(len(imported.aliases), 2)
aliases = {
alias._name: alias for alias in ns.aliases
}
imported_aliases = {
alias.name: alias for alias in imported.aliases
}
data_types = {
data_type._name: data_type for data_type in ns.data_types
}
self.assertTrue('NamespaceId' in aliases)
self.assertTrue('SharedFolderId' in aliases)
self.assertTrue('PathRootId' in aliases)
self.assertTrue('StructAlias' in aliases)
self.assertTrue('FooAlias' in imported_aliases)
self.assertTrue('BarAlias' in imported_aliases)
self.assertIsInstance(aliases['NamespaceId'].data_type, String)
self.assertIsInstance(aliases['SharedFolderId'].data_type, Alias)
self.assertIsInstance(aliases['PathRootId'].data_type, Alias)
self.assertIsInstance(aliases['StructAlias'].data_type, Struct)
self.assertIsInstance(imported_aliases['FooAlias'].data_type, String)
self.assertIsInstance(imported_aliases['BarAlias'].data_type, Alias)
self.assertIsInstance(data_types['TestStruct'], Struct)
test_struct = data_types.get('TestStruct')
dependent_struct = data_types.get('DependentStruct')
self.assertTrue(len(test_struct.fields), 1)
self.assertTrue(len(dependent_struct.fields), 1)
field = test_struct.fields[0]
self.assertEqual(field.name, 'field1')
self.assertIsInstance(field.data_type, Alias)
field = dependent_struct.fields[0]
self.assertEqual(field.name, 'field_alias')
self.assertIsInstance(field.data_type, Alias)
test_union = data_types['TestUnion']
self.assertTrue(len(test_union.fields), 1)
field = test_union.fields[0]
self.assertEqual(field.name, 'test')
self.assertIsInstance(field.data_type, Alias)
def test_no_preserve_aliases_from_api(self):
api = Api(version="0.1b1")
api.ensure_namespace('preserve_alias')
api.ensure_namespace('zzzz')
ns = api.namespaces['preserve_alias']
imported = api.namespaces['zzzz']
namespace_id = Alias('NamespaceId', ns, None)
namespace_id.data_type = String()
shared_folder_id = Alias('SharedFolderId', ns, None)
shared_folder_id.set_attributes(None, namespace_id)
path_root_id = Alias('PathRootId', ns, None)
path_root_id.set_attributes(None, shared_folder_id)
nullable_alias = Alias('NullableAlias', ns, None)
nullable_alias.set_attributes(None, Nullable(path_root_id))
foo_alias = Alias('FooAlias', None, None)
foo_alias.set_attributes(None, String())
bar_alias = Alias('BarAlias', None, None)
bar_alias.set_attributes(None, foo_alias)
ns.add_alias(namespace_id)
ns.add_alias(shared_folder_id)
ns.add_alias(path_root_id)
ns.add_alias(nullable_alias)
imported.add_alias(foo_alias)
imported.add_alias(bar_alias)
test_struct = Struct('TestStruct', ns, None)
test_struct.set_attributes(None, [
StructField('field_alias', path_root_id, None, None),
StructField('field_nullable_alias', nullable_alias, None, None),
StructField('field_list_of_alias', List(path_root_id), None, None)
])
test_union = Union('TestUnion', ns, None, None)
test_union.set_attributes(None, [UnionField('test', path_root_id, None, None)])
dependent_struct = Struct('DependentStruct', ns, None)
dependent_struct.set_attributes(None, [
StructField('field_alias', imported.alias_by_name['BarAlias'], None, None)
])
ns.add_data_type(test_struct)
ns.add_data_type(test_union)
ns.add_data_type(dependent_struct)
struct_alias = Alias('StructAlias', ns, None)
struct_alias.set_attributes(None, test_struct)
ns.add_alias(struct_alias)
api = remove_aliases_from_api(api)
self.assertEqual(len(api.namespaces), 2)
self.assertTrue('preserve_alias' in api.namespaces)
self.assertTrue('zzzz' in api.namespaces)
ns = api.namespaces['preserve_alias']
imported = api.namespaces['zzzz']
self.assertEqual(len(ns.aliases), 0)
self.assertEqual(len(imported.aliases), 0)
data_types = {
data_type._name: data_type for data_type in ns.data_types
}
test_struct = data_types.get('TestStruct')
self.assertIsInstance(test_struct, Struct)
self.assertEqual(len(test_struct.fields), 3)
for field in test_struct.fields:
if field.name == 'field_list_of_alias':
self.assertIsInstance(field.data_type, List)
list_type = field.data_type.data_type
self.assertIsInstance(list_type, String)
elif field.name == 'field_nullable_alias':
field_type = field.data_type
self.assertIsInstance(field_type, Nullable)
self.assertIsInstance(field_type.data_type, String)
else:
self.assertIsInstance(field.data_type, String)
test_union = data_types['TestUnion']
self.assertTrue(len(test_union.fields), 1)
field = test_union.fields[0]
self.assertEqual(field.name, 'test')
self.assertIsInstance(field.data_type, String)
dependent_struct = data_types.get('DependentStruct')
self.assertIsInstance(dependent_struct, Struct)
self.assertEqual(len(dependent_struct.fields), 1)
field = dependent_struct.fields[0]
self.assertIsInstance(field.data_type, String)
def test_route_hashing(self):
route = ApiRoute('test/route', 1, None)
same_route = ApiRoute("test/route", 1, None)
diff_version = ApiRoute("test/route", 2, None)
diff_route = ApiRoute("test/route_2", 1, None)
self.assertEqual(hash(route), hash(same_route))
self.assertNotEqual(hash(route), hash(diff_version))
self.assertNotEqual(hash(route), hash(diff_route))
if __name__ == '__main__':
unittest.main()