|  | #! /usr/bin/env python3 | 
|  | # | 
|  | # Protocol Buffers - Google's data interchange format | 
|  | # Copyright 2015 Google Inc.  All rights reserved. | 
|  | # | 
|  | # Use of this source code is governed by a BSD-style | 
|  | # license that can be found in the LICENSE file or at | 
|  | # https://developers.google.com/open-source/licenses/bsd | 
|  |  | 
|  | """Tests for pddm.py.""" | 
|  |  | 
|  | import io | 
|  | import unittest | 
|  |  | 
|  | from objectivec.DevTools import pddm | 
|  |  | 
|  |  | 
|  | class TestParsingMacros(unittest.TestCase): | 
|  |  | 
|  | def testParseEmpty(self): | 
|  | f = io.StringIO('') | 
|  | result = pddm.MacroCollection(f) | 
|  | self.assertEqual(len(result._macros), 0) | 
|  |  | 
|  | def testParseOne(self): | 
|  | f = io.StringIO("""PDDM-DEFINE foo( ) | 
|  | body""") | 
|  | result = pddm.MacroCollection(f) | 
|  | self.assertEqual(len(result._macros), 1) | 
|  | macro = result._macros.get('foo') | 
|  | self.assertIsNotNone(macro) | 
|  | self.assertEqual(macro.name, 'foo') | 
|  | self.assertEqual(macro.args, tuple()) | 
|  | self.assertEqual(macro.body, 'body') | 
|  |  | 
|  | def testParseGeneral(self): | 
|  | # Tests multiple defines, spaces in all places, etc. | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE noArgs( ) | 
|  | body1 | 
|  | body2 | 
|  |  | 
|  | PDDM-DEFINE-END | 
|  |  | 
|  | PDDM-DEFINE oneArg(foo) | 
|  | body3 | 
|  | PDDM-DEFINE  twoArgs( bar_ , baz ) | 
|  | body4 | 
|  | body5""") | 
|  | result = pddm.MacroCollection(f) | 
|  | self.assertEqual(len(result._macros), 3) | 
|  | macro = result._macros.get('noArgs') | 
|  | self.assertIsNotNone(macro) | 
|  | self.assertEqual(macro.name, 'noArgs') | 
|  | self.assertEqual(macro.args, tuple()) | 
|  | self.assertEqual(macro.body, 'body1\nbody2\n') | 
|  | macro = result._macros.get('oneArg') | 
|  | self.assertIsNotNone(macro) | 
|  | self.assertEqual(macro.name, 'oneArg') | 
|  | self.assertEqual(macro.args, ('foo',)) | 
|  | self.assertEqual(macro.body, 'body3') | 
|  | macro = result._macros.get('twoArgs') | 
|  | self.assertIsNotNone(macro) | 
|  | self.assertEqual(macro.name, 'twoArgs') | 
|  | self.assertEqual(macro.args, ('bar_', 'baz')) | 
|  | self.assertEqual(macro.body, 'body4\nbody5') | 
|  | # Add into existing collection | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE another(a,b,c) | 
|  | body1 | 
|  | body2""") | 
|  | result.ParseInput(f) | 
|  | self.assertEqual(len(result._macros), 4) | 
|  | macro = result._macros.get('another') | 
|  | self.assertIsNotNone(macro) | 
|  | self.assertEqual(macro.name, 'another') | 
|  | self.assertEqual(macro.args, ('a', 'b', 'c')) | 
|  | self.assertEqual(macro.body, 'body1\nbody2') | 
|  |  | 
|  | def testParseDirectiveIssues(self): | 
|  | test_list = [ | 
|  | # Unknown directive | 
|  | ('PDDM-DEFINE foo()\nbody\nPDDM-DEFINED foo\nbaz', | 
|  | 'Hit a line with an unknown directive: '), | 
|  | # End without begin | 
|  | ('PDDM-DEFINE foo()\nbody\nPDDM-DEFINE-END\nPDDM-DEFINE-END\n', | 
|  | 'Got DEFINE-END directive without an active macro: '), | 
|  | # Line not in macro block | 
|  | ('PDDM-DEFINE foo()\nbody\nPDDM-DEFINE-END\nmumble\n', | 
|  | 'Hit a line that wasn\'t a directive and no open macro definition: '), | 
|  | # Redefine macro | 
|  | ('PDDM-DEFINE foo()\nbody\nPDDM-DEFINE foo(a)\nmumble\n', | 
|  | 'Attempt to redefine macro: '), | 
|  | ] | 
|  | for idx, (input_str, expected_prefix) in enumerate(test_list, 1): | 
|  | f = io.StringIO(input_str) | 
|  | try: | 
|  | result = pddm.MacroCollection(f) | 
|  | self.fail('Should throw exception, entry %d' % idx) | 
|  | except pddm.PDDMError as e: | 
|  | self.assertTrue(e.message.startswith(expected_prefix), | 
|  | 'Entry %d failed: %r' % (idx, e)) | 
|  |  | 
|  | def testParseBeginIssues(self): | 
|  | test_list = [ | 
|  | # 1. No name | 
|  | ('PDDM-DEFINE\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 2. No name (with spaces) | 
|  | ('PDDM-DEFINE  \nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 3. No open paren | 
|  | ('PDDM-DEFINE  foo\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 4. No close paren | 
|  | ('PDDM-DEFINE foo(\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 5. No close paren (with args) | 
|  | ('PDDM-DEFINE foo(a, b\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 6. No name before args | 
|  | ('PDDM-DEFINE  (a, b)\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 7. No name before args | 
|  | ('PDDM-DEFINE foo bar(a, b)\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | # 8. Empty arg name | 
|  | ('PDDM-DEFINE foo(a, ,b)\nmumble', | 
|  | 'Empty arg name in macro definition: '), | 
|  | ('PDDM-DEFINE foo(a,,b)\nmumble', | 
|  | 'Empty arg name in macro definition: '), | 
|  | # 10. Duplicate name | 
|  | ('PDDM-DEFINE foo(a,b,a,c)\nmumble', | 
|  | 'Arg name "a" used more than once in macro definition: '), | 
|  | # 11. Invalid arg name | 
|  | ('PDDM-DEFINE foo(a b,c)\nmumble', | 
|  | 'Invalid arg name "a b" in macro definition: '), | 
|  | ('PDDM-DEFINE foo(a.b,c)\nmumble', | 
|  | 'Invalid arg name "a.b" in macro definition: '), | 
|  | ('PDDM-DEFINE foo(a-b,c)\nmumble', | 
|  | 'Invalid arg name "a-b" in macro definition: '), | 
|  | ('PDDM-DEFINE foo(a,b,c.)\nmumble', | 
|  | 'Invalid arg name "c." in macro definition: '), | 
|  | # 15. Extra stuff after the name | 
|  | ('PDDM-DEFINE foo(a,c) foo\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | ('PDDM-DEFINE foo(a,c) foo)\nmumble', | 
|  | 'Failed to parse macro definition: '), | 
|  | ] | 
|  | for idx, (input_str, expected_prefix) in enumerate(test_list, 1): | 
|  | f = io.StringIO(input_str) | 
|  | try: | 
|  | result = pddm.MacroCollection(f) | 
|  | self.fail('Should throw exception, entry %d' % idx) | 
|  | except pddm.PDDMError as e: | 
|  | self.assertTrue(e.message.startswith(expected_prefix), | 
|  | 'Entry %d failed: %r' % (idx, e)) | 
|  |  | 
|  |  | 
|  | class TestExpandingMacros(unittest.TestCase): | 
|  |  | 
|  | def testExpandBasics(self): | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE noArgs( ) | 
|  | body1 | 
|  | body2 | 
|  |  | 
|  | PDDM-DEFINE-END | 
|  |  | 
|  | PDDM-DEFINE oneArg(a) | 
|  | body3 a | 
|  |  | 
|  | PDDM-DEFINE-END | 
|  |  | 
|  | PDDM-DEFINE twoArgs(b,c) | 
|  | body4 b c | 
|  | body5 | 
|  | PDDM-DEFINE-END | 
|  |  | 
|  | """) | 
|  | mc = pddm.MacroCollection(f) | 
|  | test_list = [ | 
|  | ('noArgs()', | 
|  | 'body1\nbody2\n'), | 
|  | ('oneArg(wee)', | 
|  | 'body3 wee\n'), | 
|  | ('twoArgs(having some, fun)', | 
|  | 'body4 having some fun\nbody5'), | 
|  | # One arg, pass empty. | 
|  | ('oneArg()', | 
|  | 'body3 \n'), | 
|  | # Two args, gets empty in each slot. | 
|  | ('twoArgs(, empty)', | 
|  | 'body4  empty\nbody5'), | 
|  | ('twoArgs(empty, )', | 
|  | 'body4 empty \nbody5'), | 
|  | ('twoArgs(, )', | 
|  | 'body4  \nbody5'), | 
|  | ] | 
|  | for idx, (input_str, expected) in enumerate(test_list, 1): | 
|  | result = mc.Expand(input_str) | 
|  | self.assertEqual(result, expected, | 
|  | 'Entry %d --\n       Result: %r\n     Expected: %r' % | 
|  | (idx, result, expected)) | 
|  |  | 
|  | def testExpandArgOptions(self): | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE bar(a) | 
|  | a-a$S-a$l-a$L-a$u-a$U | 
|  | PDDM-DEFINE-END | 
|  | """) | 
|  | mc = pddm.MacroCollection(f) | 
|  |  | 
|  | self.assertEqual(mc.Expand('bar(xYz)'), 'xYz-   -xYz-xyz-XYz-XYZ') | 
|  | self.assertEqual(mc.Expand('bar(MnoP)'), 'MnoP-    -mnoP-mnop-MnoP-MNOP') | 
|  | # Test empty | 
|  | self.assertEqual(mc.Expand('bar()'), '-----') | 
|  |  | 
|  | def testExpandSimpleMacroErrors(self): | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE foo(a, b) | 
|  | <a-z> | 
|  | PDDM-DEFINE baz(a) | 
|  | a - a$z | 
|  | """) | 
|  | mc = pddm.MacroCollection(f) | 
|  | test_list = [ | 
|  | # 1. Unknown macro | 
|  | ('bar()', | 
|  | 'No macro named "bar".'), | 
|  | ('bar(a)', | 
|  | 'No macro named "bar".'), | 
|  | # 3. Arg mismatch | 
|  | ('foo()', | 
|  | 'Expected 2 args, got: "foo()".'), | 
|  | ('foo(a b)', | 
|  | 'Expected 2 args, got: "foo(a b)".'), | 
|  | ('foo(a,b,c)', | 
|  | 'Expected 2 args, got: "foo(a,b,c)".'), | 
|  | # 6. Unknown option in expansion | 
|  | ('baz(mumble)', | 
|  | 'Unknown arg option "a$z" while expanding "baz(mumble)".'), | 
|  | ] | 
|  | for idx, (input_str, expected_err) in enumerate(test_list, 1): | 
|  | try: | 
|  | result = mc.Expand(input_str) | 
|  | self.fail('Should throw exception, entry %d' % idx) | 
|  | except pddm.PDDMError as e: | 
|  | self.assertEqual(e.message, expected_err, | 
|  | 'Entry %d failed: %r' % (idx, e)) | 
|  |  | 
|  | def testExpandReferences(self): | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE StartIt() | 
|  | foo(abc, def) | 
|  | foo(ghi, jkl) | 
|  | PDDM-DEFINE foo(a, b) | 
|  | bar(a, int) | 
|  | bar(b, NSString *) | 
|  | PDDM-DEFINE bar(n, t) | 
|  | - (t)n; | 
|  | - (void)set##n$u##:(t)value; | 
|  |  | 
|  | """) | 
|  | mc = pddm.MacroCollection(f) | 
|  | expected = """- (int)abc; | 
|  | - (void)setAbc:(int)value; | 
|  |  | 
|  | - (NSString *)def; | 
|  | - (void)setDef:(NSString *)value; | 
|  |  | 
|  | - (int)ghi; | 
|  | - (void)setGhi:(int)value; | 
|  |  | 
|  | - (NSString *)jkl; | 
|  | - (void)setJkl:(NSString *)value; | 
|  | """ | 
|  | self.assertEqual(mc.Expand('StartIt()'), expected) | 
|  |  | 
|  | def testCatchRecursion(self): | 
|  | f = io.StringIO(""" | 
|  | PDDM-DEFINE foo(a, b) | 
|  | bar(1, a) | 
|  | bar(2, b) | 
|  | PDDM-DEFINE bar(x, y) | 
|  | foo(x, y) | 
|  | """) | 
|  | mc = pddm.MacroCollection(f) | 
|  | try: | 
|  | result = mc.Expand('foo(A,B)') | 
|  | self.fail('Should throw exception! Test failed to catch recursion.') | 
|  | except pddm.PDDMError as e: | 
|  | self.assertEqual(e.message, | 
|  | 'Found macro recursion, invoking "foo(1, A)":\n...while expanding "bar(1, A)".\n...while expanding "foo(A,B)".') | 
|  |  | 
|  |  | 
|  | class TestParsingSource(unittest.TestCase): | 
|  |  | 
|  | def testBasicParse(self): | 
|  | test_list = [ | 
|  | # 1. no directives | 
|  | ('a\nb\nc', | 
|  | (3,) ), | 
|  | # 2. One define | 
|  | ('a\n//%PDDM-DEFINE foo()\n//%body\nc', | 
|  | (1, 2, 1) ), | 
|  | # 3. Two defines | 
|  | ('a\n//%PDDM-DEFINE foo()\n//%body\n//%PDDM-DEFINE bar()\n//%body2\nc', | 
|  | (1, 4, 1) ), | 
|  | # 4. Two defines with ends | 
|  | ('a\n//%PDDM-DEFINE foo()\n//%body\n//%PDDM-DEFINE-END\n' | 
|  | '//%PDDM-DEFINE bar()\n//%body2\n//%PDDM-DEFINE-END\nc', | 
|  | (1, 6, 1) ), | 
|  | # 5. One expand, one define (that runs to end of file) | 
|  | ('a\n//%PDDM-EXPAND foo()\nbody\n//%PDDM-EXPAND-END\n' | 
|  | '//%PDDM-DEFINE bar()\n//%body2\n', | 
|  | (1, 1, 2) ), | 
|  | # 6. One define ended with an expand. | 
|  | ('a\nb\n//%PDDM-DEFINE bar()\n//%body2\n' | 
|  | '//%PDDM-EXPAND bar()\nbody2\n//%PDDM-EXPAND-END\n', | 
|  | (2, 2, 1) ), | 
|  | # 7. Two expands (one end), one define. | 
|  | ('a\n//%PDDM-EXPAND foo(1)\nbody\n//%PDDM-EXPAND foo(2)\nbody2\n//%PDDM-EXPAND-END\n' | 
|  | '//%PDDM-DEFINE foo()\n//%body2\n', | 
|  | (1, 2, 2) ), | 
|  | ] | 
|  | for idx, (input_str, line_counts) in enumerate(test_list, 1): | 
|  | f = io.StringIO(input_str) | 
|  | sf = pddm.SourceFile(f) | 
|  | sf._ParseFile() | 
|  | self.assertEqual(len(sf._sections), len(line_counts), | 
|  | 'Entry %d -- %d != %d' % | 
|  | (idx, len(sf._sections), len(line_counts))) | 
|  | for idx2, (sec, expected) in enumerate(zip(sf._sections, line_counts), 1): | 
|  | self.assertEqual(sec.num_lines_captured, expected, | 
|  | 'Entry %d, section %d -- %d != %d' % | 
|  | (idx, idx2, sec.num_lines_captured, expected)) | 
|  |  | 
|  | def testErrors(self): | 
|  | test_list = [ | 
|  | # 1. Directive within expansion | 
|  | ('//%PDDM-EXPAND a()\n//%PDDM-BOGUS', | 
|  | 'Ran into directive ("//%PDDM-BOGUS", line 2) while in "//%PDDM-EXPAND a()".'), | 
|  | ('//%PDDM-EXPAND a()\n//%PDDM-DEFINE a()\n//%body\n', | 
|  | 'Ran into directive ("//%PDDM-DEFINE", line 2) while in "//%PDDM-EXPAND a()".'), | 
|  | # 3. Expansion ran off end of file | 
|  | ('//%PDDM-EXPAND a()\na\nb\n', | 
|  | 'Hit the end of the file while in "//%PDDM-EXPAND a()".'), | 
|  | # 4. Directive within define | 
|  | ('//%PDDM-DEFINE a()\n//%body\n//%PDDM-BOGUS', | 
|  | 'Ran into directive ("//%PDDM-BOGUS", line 3) while in "//%PDDM-DEFINE a()".'), | 
|  | ('//%PDDM-DEFINE a()\n//%body\n//%PDDM-EXPAND-END a()', | 
|  | 'Ran into directive ("//%PDDM-EXPAND-END", line 3) while in "//%PDDM-DEFINE a()".'), | 
|  | # 6. Directives that shouldn't start sections | 
|  | ('a\n//%PDDM-DEFINE-END a()\n//a\n', | 
|  | 'Unexpected line 2: "//%PDDM-DEFINE-END a()".'), | 
|  | ('a\n//%PDDM-EXPAND-END a()\n//a\n', | 
|  | 'Unexpected line 2: "//%PDDM-EXPAND-END a()".'), | 
|  | ('//%PDDM-BOGUS\n//a\n', | 
|  | 'Unexpected line 1: "//%PDDM-BOGUS".'), | 
|  | ] | 
|  | for idx, (input_str, expected_err) in enumerate(test_list, 1): | 
|  | f = io.StringIO(input_str) | 
|  | try: | 
|  | pddm.SourceFile(f)._ParseFile() | 
|  | self.fail('Should throw exception, entry %d' % idx) | 
|  | except pddm.PDDMError as e: | 
|  | self.assertEqual(e.message, expected_err, | 
|  | 'Entry %d failed: %r' % (idx, e)) | 
|  |  | 
|  | class TestProcessingSource(unittest.TestCase): | 
|  |  | 
|  | def testBasics(self): | 
|  | self.maxDiff = None | 
|  | input_str = """ | 
|  | //%PDDM-IMPORT-DEFINES ImportFile | 
|  | foo | 
|  | //%PDDM-EXPAND mumble(abc) | 
|  | //%PDDM-EXPAND-END | 
|  | bar | 
|  | //%PDDM-EXPAND mumble(def) | 
|  | //%PDDM-EXPAND mumble(ghi) | 
|  | //%PDDM-EXPAND-END | 
|  | baz | 
|  | //%PDDM-DEFINE mumble(a_) | 
|  | //%a_: getName(a_) | 
|  | """ | 
|  | input_str2 = """ | 
|  | //%PDDM-DEFINE getName(x_) | 
|  | //%do##x_$u##(int x_); | 
|  |  | 
|  | """ | 
|  | expected = """ | 
|  | //%PDDM-IMPORT-DEFINES ImportFile | 
|  | foo | 
|  | //%PDDM-EXPAND mumble(abc) | 
|  | // This block of code is generated, do not edit it directly. | 
|  |  | 
|  | abc: doAbc(int abc); | 
|  | //%PDDM-EXPAND-END mumble(abc) | 
|  | bar | 
|  | //%PDDM-EXPAND mumble(def) | 
|  | // This block of code is generated, do not edit it directly. | 
|  |  | 
|  | def: doDef(int def); | 
|  | //%PDDM-EXPAND mumble(ghi) | 
|  | // This block of code is generated, do not edit it directly. | 
|  |  | 
|  | ghi: doGhi(int ghi); | 
|  | //%PDDM-EXPAND-END (2 expansions) | 
|  | baz | 
|  | //%PDDM-DEFINE mumble(a_) | 
|  | //%a_: getName(a_) | 
|  | """ | 
|  | expected_stripped = """ | 
|  | //%PDDM-IMPORT-DEFINES ImportFile | 
|  | foo | 
|  | //%PDDM-EXPAND mumble(abc) | 
|  | //%PDDM-EXPAND-END mumble(abc) | 
|  | bar | 
|  | //%PDDM-EXPAND mumble(def) | 
|  | //%PDDM-EXPAND mumble(ghi) | 
|  | //%PDDM-EXPAND-END (2 expansions) | 
|  | baz | 
|  | //%PDDM-DEFINE mumble(a_) | 
|  | //%a_: getName(a_) | 
|  | """ | 
|  | def _Resolver(name): | 
|  | self.assertEqual(name, 'ImportFile') | 
|  | return io.StringIO(input_str2) | 
|  | f = io.StringIO(input_str) | 
|  | sf = pddm.SourceFile(f, _Resolver) | 
|  | sf.ProcessContent() | 
|  | self.assertEqual(sf.processed_content, expected) | 
|  | # Feed it through and nothing should change. | 
|  | f2 = io.StringIO(sf.processed_content) | 
|  | sf2 = pddm.SourceFile(f2, _Resolver) | 
|  | sf2.ProcessContent() | 
|  | self.assertEqual(sf2.processed_content, expected) | 
|  | self.assertEqual(sf2.processed_content, sf.processed_content) | 
|  | # Test stripping (with the original input and expanded version). | 
|  | f2 = io.StringIO(input_str) | 
|  | sf2 = pddm.SourceFile(f2) | 
|  | sf2.ProcessContent(strip_expansion=True) | 
|  | self.assertEqual(sf2.processed_content, expected_stripped) | 
|  | f2 = io.StringIO(sf.processed_content) | 
|  | sf2 = pddm.SourceFile(f2, _Resolver) | 
|  | sf2.ProcessContent(strip_expansion=True) | 
|  | self.assertEqual(sf2.processed_content, expected_stripped) | 
|  |  | 
|  | def testProcessFileWithMacroParseError(self): | 
|  | input_str = """ | 
|  | foo | 
|  | //%PDDM-DEFINE mumble(a_) | 
|  | //%body | 
|  | //%PDDM-DEFINE mumble(x_) | 
|  | //%body2 | 
|  |  | 
|  | """ | 
|  | f = io.StringIO(input_str) | 
|  | sf = pddm.SourceFile(f) | 
|  | try: | 
|  | sf.ProcessContent() | 
|  | self.fail('Should throw exception! Test failed to catch macro parsing error.') | 
|  | except pddm.PDDMError as e: | 
|  | self.assertEqual(e.message, | 
|  | 'Attempt to redefine macro: "PDDM-DEFINE mumble(x_)"\n' | 
|  | '...while parsing section that started:\n' | 
|  | '  Line 3: //%PDDM-DEFINE mumble(a_)') | 
|  |  | 
|  | def testProcessFileWithExpandError(self): | 
|  | input_str = """ | 
|  | foo | 
|  | //%PDDM-DEFINE mumble(a_) | 
|  | //%body | 
|  | //%PDDM-EXPAND foobar(x_) | 
|  | //%PDDM-EXPAND-END | 
|  |  | 
|  | """ | 
|  | f = io.StringIO(input_str) | 
|  | sf = pddm.SourceFile(f) | 
|  | try: | 
|  | sf.ProcessContent() | 
|  | self.fail('Should throw exception! Test failed to catch expand error.') | 
|  | except pddm.PDDMError as e: | 
|  | self.assertEqual(e.message, | 
|  | 'No macro named "foobar".\n' | 
|  | '...while expanding "foobar(x_)" from the section that' | 
|  | ' started:\n   Line 5: //%PDDM-EXPAND foobar(x_)') | 
|  |  | 
|  |  | 
|  | if __name__ == '__main__': | 
|  | unittest.main() |