1#! /usr/bin/python 2# 3# Protocol Buffers - Google's data interchange format 4# Copyright 2015 Google Inc. All rights reserved. 5# https://developers.google.com/protocol-buffers/ 6# 7# Redistribution and use in source and binary forms, with or without 8# modification, are permitted provided that the following conditions are 9# met: 10# 11# * Redistributions of source code must retain the above copyright 12# notice, this list of conditions and the following disclaimer. 13# * Redistributions in binary form must reproduce the above 14# copyright notice, this list of conditions and the following disclaimer 15# in the documentation and/or other materials provided with the 16# distribution. 17# * Neither the name of Google Inc. nor the names of its 18# contributors may be used to endorse or promote products derived from 19# this software without specific prior written permission. 20# 21# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 22# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 23# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 24# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 25# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 26# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 27# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 28# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 29# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 30# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 31# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 32 33"""Tests for pddm.py.""" 34 35import io 36import unittest 37 38import pddm 39 40 41class TestParsingMacros(unittest.TestCase): 42 43 def testParseEmpty(self): 44 f = io.StringIO(u'') 45 result = pddm.MacroCollection(f) 46 self.assertEqual(len(result._macros), 0) 47 48 def testParseOne(self): 49 f = io.StringIO(u"""PDDM-DEFINE foo( ) 50body""") 51 result = pddm.MacroCollection(f) 52 self.assertEqual(len(result._macros), 1) 53 macro = result._macros.get('foo') 54 self.assertIsNotNone(macro) 55 self.assertEquals(macro.name, 'foo') 56 self.assertEquals(macro.args, tuple()) 57 self.assertEquals(macro.body, 'body') 58 59 def testParseGeneral(self): 60 # Tests multiple defines, spaces in all places, etc. 61 f = io.StringIO(u""" 62PDDM-DEFINE noArgs( ) 63body1 64body2 65 66PDDM-DEFINE-END 67 68PDDM-DEFINE oneArg(foo) 69body3 70PDDM-DEFINE twoArgs( bar_ , baz ) 71body4 72body5""") 73 result = pddm.MacroCollection(f) 74 self.assertEqual(len(result._macros), 3) 75 macro = result._macros.get('noArgs') 76 self.assertIsNotNone(macro) 77 self.assertEquals(macro.name, 'noArgs') 78 self.assertEquals(macro.args, tuple()) 79 self.assertEquals(macro.body, 'body1\nbody2\n') 80 macro = result._macros.get('oneArg') 81 self.assertIsNotNone(macro) 82 self.assertEquals(macro.name, 'oneArg') 83 self.assertEquals(macro.args, ('foo',)) 84 self.assertEquals(macro.body, 'body3') 85 macro = result._macros.get('twoArgs') 86 self.assertIsNotNone(macro) 87 self.assertEquals(macro.name, 'twoArgs') 88 self.assertEquals(macro.args, ('bar_', 'baz')) 89 self.assertEquals(macro.body, 'body4\nbody5') 90 # Add into existing collection 91 f = io.StringIO(u""" 92PDDM-DEFINE another(a,b,c) 93body1 94body2""") 95 result.ParseInput(f) 96 self.assertEqual(len(result._macros), 4) 97 macro = result._macros.get('another') 98 self.assertIsNotNone(macro) 99 self.assertEquals(macro.name, 'another') 100 self.assertEquals(macro.args, ('a', 'b', 'c')) 101 self.assertEquals(macro.body, 'body1\nbody2') 102 103 def testParseDirectiveIssues(self): 104 test_list = [ 105 # Unknown directive 106 (u'PDDM-DEFINE foo()\nbody\nPDDM-DEFINED foo\nbaz', 107 'Hit a line with an unknown directive: '), 108 # End without begin 109 (u'PDDM-DEFINE foo()\nbody\nPDDM-DEFINE-END\nPDDM-DEFINE-END\n', 110 'Got DEFINE-END directive without an active macro: '), 111 # Line not in macro block 112 (u'PDDM-DEFINE foo()\nbody\nPDDM-DEFINE-END\nmumble\n', 113 'Hit a line that wasn\'t a directive and no open macro definition: '), 114 # Redefine macro 115 (u'PDDM-DEFINE foo()\nbody\nPDDM-DEFINE foo(a)\nmumble\n', 116 'Attempt to redefine macro: '), 117 ] 118 for idx, (input_str, expected_prefix) in enumerate(test_list, 1): 119 f = io.StringIO(input_str) 120 try: 121 result = pddm.MacroCollection(f) 122 self.fail('Should throw exception, entry %d' % idx) 123 except pddm.PDDMError as e: 124 self.assertTrue(e.message.startswith(expected_prefix), 125 'Entry %d failed: %r' % (idx, e)) 126 127 def testParseBeginIssues(self): 128 test_list = [ 129 # 1. No name 130 (u'PDDM-DEFINE\nmumble', 131 'Failed to parse macro definition: '), 132 # 2. No name (with spaces) 133 (u'PDDM-DEFINE \nmumble', 134 'Failed to parse macro definition: '), 135 # 3. No open paren 136 (u'PDDM-DEFINE foo\nmumble', 137 'Failed to parse macro definition: '), 138 # 4. No close paren 139 (u'PDDM-DEFINE foo(\nmumble', 140 'Failed to parse macro definition: '), 141 # 5. No close paren (with args) 142 (u'PDDM-DEFINE foo(a, b\nmumble', 143 'Failed to parse macro definition: '), 144 # 6. No name before args 145 (u'PDDM-DEFINE (a, b)\nmumble', 146 'Failed to parse macro definition: '), 147 # 7. No name before args 148 (u'PDDM-DEFINE foo bar(a, b)\nmumble', 149 'Failed to parse macro definition: '), 150 # 8. Empty arg name 151 (u'PDDM-DEFINE foo(a, ,b)\nmumble', 152 'Empty arg name in macro definition: '), 153 (u'PDDM-DEFINE foo(a,,b)\nmumble', 154 'Empty arg name in macro definition: '), 155 # 10. Duplicate name 156 (u'PDDM-DEFINE foo(a,b,a,c)\nmumble', 157 'Arg name "a" used more than once in macro definition: '), 158 # 11. Invalid arg name 159 (u'PDDM-DEFINE foo(a b,c)\nmumble', 160 'Invalid arg name "a b" in macro definition: '), 161 (u'PDDM-DEFINE foo(a.b,c)\nmumble', 162 'Invalid arg name "a.b" in macro definition: '), 163 (u'PDDM-DEFINE foo(a-b,c)\nmumble', 164 'Invalid arg name "a-b" in macro definition: '), 165 (u'PDDM-DEFINE foo(a,b,c.)\nmumble', 166 'Invalid arg name "c." in macro definition: '), 167 # 15. Extra stuff after the name 168 (u'PDDM-DEFINE foo(a,c) foo\nmumble', 169 'Failed to parse macro definition: '), 170 (u'PDDM-DEFINE foo(a,c) foo)\nmumble', 171 'Failed to parse macro definition: '), 172 ] 173 for idx, (input_str, expected_prefix) in enumerate(test_list, 1): 174 f = io.StringIO(input_str) 175 try: 176 result = pddm.MacroCollection(f) 177 self.fail('Should throw exception, entry %d' % idx) 178 except pddm.PDDMError as e: 179 self.assertTrue(e.message.startswith(expected_prefix), 180 'Entry %d failed: %r' % (idx, e)) 181 182 183class TestExpandingMacros(unittest.TestCase): 184 185 def testExpandBasics(self): 186 f = io.StringIO(u""" 187PDDM-DEFINE noArgs( ) 188body1 189body2 190 191PDDM-DEFINE-END 192 193PDDM-DEFINE oneArg(a) 194body3 a 195 196PDDM-DEFINE-END 197 198PDDM-DEFINE twoArgs(b,c) 199body4 b c 200body5 201PDDM-DEFINE-END 202 203""") 204 mc = pddm.MacroCollection(f) 205 test_list = [ 206 (u'noArgs()', 207 'body1\nbody2\n'), 208 (u'oneArg(wee)', 209 'body3 wee\n'), 210 (u'twoArgs(having some, fun)', 211 'body4 having some fun\nbody5'), 212 # One arg, pass empty. 213 (u'oneArg()', 214 'body3 \n'), 215 # Two args, gets empty in each slot. 216 (u'twoArgs(, empty)', 217 'body4 empty\nbody5'), 218 (u'twoArgs(empty, )', 219 'body4 empty \nbody5'), 220 (u'twoArgs(, )', 221 'body4 \nbody5'), 222 ] 223 for idx, (input_str, expected) in enumerate(test_list, 1): 224 result = mc.Expand(input_str) 225 self.assertEqual(result, expected, 226 'Entry %d --\n Result: %r\n Expected: %r' % 227 (idx, result, expected)) 228 229 def testExpandArgOptions(self): 230 f = io.StringIO(u""" 231PDDM-DEFINE bar(a) 232a-a$S-a$l-a$L-a$u-a$U 233PDDM-DEFINE-END 234""") 235 mc = pddm.MacroCollection(f) 236 237 self.assertEqual(mc.Expand('bar(xYz)'), 'xYz- -xYz-xyz-XYz-XYZ') 238 self.assertEqual(mc.Expand('bar(MnoP)'), 'MnoP- -mnoP-mnop-MnoP-MNOP') 239 # Test empty 240 self.assertEqual(mc.Expand('bar()'), '-----') 241 242 def testExpandSimpleMacroErrors(self): 243 f = io.StringIO(u""" 244PDDM-DEFINE foo(a, b) 245<a-z> 246PDDM-DEFINE baz(a) 247a - a$z 248""") 249 mc = pddm.MacroCollection(f) 250 test_list = [ 251 # 1. Unknown macro 252 (u'bar()', 253 'No macro named "bar".'), 254 (u'bar(a)', 255 'No macro named "bar".'), 256 # 3. Arg mismatch 257 (u'foo()', 258 'Expected 2 args, got: "foo()".'), 259 (u'foo(a b)', 260 'Expected 2 args, got: "foo(a b)".'), 261 (u'foo(a,b,c)', 262 'Expected 2 args, got: "foo(a,b,c)".'), 263 # 6. Unknown option in expansion 264 (u'baz(mumble)', 265 'Unknown arg option "a$z" while expanding "baz(mumble)".'), 266 ] 267 for idx, (input_str, expected_err) in enumerate(test_list, 1): 268 try: 269 result = mc.Expand(input_str) 270 self.fail('Should throw exception, entry %d' % idx) 271 except pddm.PDDMError as e: 272 self.assertEqual(e.message, expected_err, 273 'Entry %d failed: %r' % (idx, e)) 274 275 def testExpandReferences(self): 276 f = io.StringIO(u""" 277PDDM-DEFINE StartIt() 278foo(abc, def) 279foo(ghi, jkl) 280PDDM-DEFINE foo(a, b) 281bar(a, int) 282bar(b, NSString *) 283PDDM-DEFINE bar(n, t) 284- (t)n; 285- (void)set##n$u##:(t)value; 286 287""") 288 mc = pddm.MacroCollection(f) 289 expected = """- (int)abc; 290- (void)setAbc:(int)value; 291 292- (NSString *)def; 293- (void)setDef:(NSString *)value; 294 295- (int)ghi; 296- (void)setGhi:(int)value; 297 298- (NSString *)jkl; 299- (void)setJkl:(NSString *)value; 300""" 301 self.assertEqual(mc.Expand('StartIt()'), expected) 302 303 def testCatchRecursion(self): 304 f = io.StringIO(u""" 305PDDM-DEFINE foo(a, b) 306bar(1, a) 307bar(2, b) 308PDDM-DEFINE bar(x, y) 309foo(x, y) 310""") 311 mc = pddm.MacroCollection(f) 312 try: 313 result = mc.Expand('foo(A,B)') 314 self.fail('Should throw exception! Test failed to catch recursion.') 315 except pddm.PDDMError as e: 316 self.assertEqual(e.message, 317 'Found macro recursion, invoking "foo(1, A)":\n...while expanding "bar(1, A)".\n...while expanding "foo(A,B)".') 318 319 320class TestParsingSource(unittest.TestCase): 321 322 def testBasicParse(self): 323 test_list = [ 324 # 1. no directives 325 (u'a\nb\nc', 326 (3,) ), 327 # 2. One define 328 (u'a\n//%PDDM-DEFINE foo()\n//%body\nc', 329 (1, 2, 1) ), 330 # 3. Two defines 331 (u'a\n//%PDDM-DEFINE foo()\n//%body\n//%PDDM-DEFINE bar()\n//%body2\nc', 332 (1, 4, 1) ), 333 # 4. Two defines with ends 334 (u'a\n//%PDDM-DEFINE foo()\n//%body\n//%PDDM-DEFINE-END\n' 335 u'//%PDDM-DEFINE bar()\n//%body2\n//%PDDM-DEFINE-END\nc', 336 (1, 6, 1) ), 337 # 5. One expand, one define (that runs to end of file) 338 (u'a\n//%PDDM-EXPAND foo()\nbody\n//%PDDM-EXPAND-END\n' 339 u'//%PDDM-DEFINE bar()\n//%body2\n', 340 (1, 1, 2) ), 341 # 6. One define ended with an expand. 342 (u'a\nb\n//%PDDM-DEFINE bar()\n//%body2\n' 343 u'//%PDDM-EXPAND bar()\nbody2\n//%PDDM-EXPAND-END\n', 344 (2, 2, 1) ), 345 # 7. Two expands (one end), one define. 346 (u'a\n//%PDDM-EXPAND foo(1)\nbody\n//%PDDM-EXPAND foo(2)\nbody2\n//%PDDM-EXPAND-END\n' 347 u'//%PDDM-DEFINE foo()\n//%body2\n', 348 (1, 2, 2) ), 349 ] 350 for idx, (input_str, line_counts) in enumerate(test_list, 1): 351 f = io.StringIO(input_str) 352 sf = pddm.SourceFile(f) 353 sf._ParseFile() 354 self.assertEqual(len(sf._sections), len(line_counts), 355 'Entry %d -- %d != %d' % 356 (idx, len(sf._sections), len(line_counts))) 357 for idx2, (sec, expected) in enumerate(zip(sf._sections, line_counts), 1): 358 self.assertEqual(sec.num_lines_captured, expected, 359 'Entry %d, section %d -- %d != %d' % 360 (idx, idx2, sec.num_lines_captured, expected)) 361 362 def testErrors(self): 363 test_list = [ 364 # 1. Directive within expansion 365 (u'//%PDDM-EXPAND a()\n//%PDDM-BOGUS', 366 'Ran into directive ("//%PDDM-BOGUS", line 2) while in "//%PDDM-EXPAND a()".'), 367 (u'//%PDDM-EXPAND a()\n//%PDDM-DEFINE a()\n//%body\n', 368 'Ran into directive ("//%PDDM-DEFINE", line 2) while in "//%PDDM-EXPAND a()".'), 369 # 3. Expansion ran off end of file 370 (u'//%PDDM-EXPAND a()\na\nb\n', 371 'Hit the end of the file while in "//%PDDM-EXPAND a()".'), 372 # 4. Directive within define 373 (u'//%PDDM-DEFINE a()\n//%body\n//%PDDM-BOGUS', 374 'Ran into directive ("//%PDDM-BOGUS", line 3) while in "//%PDDM-DEFINE a()".'), 375 (u'//%PDDM-DEFINE a()\n//%body\n//%PDDM-EXPAND-END a()', 376 'Ran into directive ("//%PDDM-EXPAND-END", line 3) while in "//%PDDM-DEFINE a()".'), 377 # 6. Directives that shouldn't start sections 378 (u'a\n//%PDDM-DEFINE-END a()\n//a\n', 379 'Unexpected line 2: "//%PDDM-DEFINE-END a()".'), 380 (u'a\n//%PDDM-EXPAND-END a()\n//a\n', 381 'Unexpected line 2: "//%PDDM-EXPAND-END a()".'), 382 (u'//%PDDM-BOGUS\n//a\n', 383 'Unexpected line 1: "//%PDDM-BOGUS".'), 384 ] 385 for idx, (input_str, expected_err) in enumerate(test_list, 1): 386 f = io.StringIO(input_str) 387 try: 388 pddm.SourceFile(f)._ParseFile() 389 self.fail('Should throw exception, entry %d' % idx) 390 except pddm.PDDMError as e: 391 self.assertEqual(e.message, expected_err, 392 'Entry %d failed: %r' % (idx, e)) 393 394class TestProcessingSource(unittest.TestCase): 395 396 def testBasics(self): 397 self.maxDiff = None 398 input_str = u""" 399//%PDDM-IMPORT-DEFINES ImportFile 400foo 401//%PDDM-EXPAND mumble(abc) 402//%PDDM-EXPAND-END 403bar 404//%PDDM-EXPAND mumble(def) 405//%PDDM-EXPAND mumble(ghi) 406//%PDDM-EXPAND-END 407baz 408//%PDDM-DEFINE mumble(a_) 409//%a_: getName(a_) 410""" 411 input_str2 = u""" 412//%PDDM-DEFINE getName(x_) 413//%do##x_$u##(int x_); 414 415""" 416 expected = u""" 417//%PDDM-IMPORT-DEFINES ImportFile 418foo 419//%PDDM-EXPAND mumble(abc) 420// This block of code is generated, do not edit it directly. 421// clang-format off 422 423abc: doAbc(int abc); 424// clang-format on 425//%PDDM-EXPAND-END mumble(abc) 426bar 427//%PDDM-EXPAND mumble(def) 428// This block of code is generated, do not edit it directly. 429// clang-format off 430 431def: doDef(int def); 432// clang-format on 433//%PDDM-EXPAND mumble(ghi) 434// This block of code is generated, do not edit it directly. 435// clang-format off 436 437ghi: doGhi(int ghi); 438// clang-format on 439//%PDDM-EXPAND-END (2 expansions) 440baz 441//%PDDM-DEFINE mumble(a_) 442//%a_: getName(a_) 443""" 444 expected_stripped = u""" 445//%PDDM-IMPORT-DEFINES ImportFile 446foo 447//%PDDM-EXPAND mumble(abc) 448//%PDDM-EXPAND-END mumble(abc) 449bar 450//%PDDM-EXPAND mumble(def) 451//%PDDM-EXPAND mumble(ghi) 452//%PDDM-EXPAND-END (2 expansions) 453baz 454//%PDDM-DEFINE mumble(a_) 455//%a_: getName(a_) 456""" 457 def _Resolver(name): 458 self.assertEqual(name, 'ImportFile') 459 return io.StringIO(input_str2) 460 f = io.StringIO(input_str) 461 sf = pddm.SourceFile(f, _Resolver) 462 sf.ProcessContent() 463 self.assertEqual(sf.processed_content, expected) 464 # Feed it through and nothing should change. 465 f2 = io.StringIO(sf.processed_content) 466 sf2 = pddm.SourceFile(f2, _Resolver) 467 sf2.ProcessContent() 468 self.assertEqual(sf2.processed_content, expected) 469 self.assertEqual(sf2.processed_content, sf.processed_content) 470 # Test stripping (with the original input and expanded version). 471 f2 = io.StringIO(input_str) 472 sf2 = pddm.SourceFile(f2) 473 sf2.ProcessContent(strip_expansion=True) 474 self.assertEqual(sf2.processed_content, expected_stripped) 475 f2 = io.StringIO(sf.processed_content) 476 sf2 = pddm.SourceFile(f2, _Resolver) 477 sf2.ProcessContent(strip_expansion=True) 478 self.assertEqual(sf2.processed_content, expected_stripped) 479 480 def testProcessFileWithMacroParseError(self): 481 input_str = u""" 482foo 483//%PDDM-DEFINE mumble(a_) 484//%body 485//%PDDM-DEFINE mumble(x_) 486//%body2 487 488""" 489 f = io.StringIO(input_str) 490 sf = pddm.SourceFile(f) 491 try: 492 sf.ProcessContent() 493 self.fail('Should throw exception! Test failed to catch macro parsing error.') 494 except pddm.PDDMError as e: 495 self.assertEqual(e.message, 496 'Attempt to redefine macro: "PDDM-DEFINE mumble(x_)"\n' 497 '...while parsing section that started:\n' 498 ' Line 3: //%PDDM-DEFINE mumble(a_)') 499 500 def testProcessFileWithExpandError(self): 501 input_str = u""" 502foo 503//%PDDM-DEFINE mumble(a_) 504//%body 505//%PDDM-EXPAND foobar(x_) 506//%PDDM-EXPAND-END 507 508""" 509 f = io.StringIO(input_str) 510 sf = pddm.SourceFile(f) 511 try: 512 sf.ProcessContent() 513 self.fail('Should throw exception! Test failed to catch expand error.') 514 except pddm.PDDMError as e: 515 self.assertEqual(e.message, 516 'No macro named "foobar".\n' 517 '...while expanding "foobar(x_)" from the section that' 518 ' started:\n Line 5: //%PDDM-EXPAND foobar(x_)') 519 520 521if __name__ == '__main__': 522 unittest.main() 523