1#Copyright ReportLab Europe Ltd. 2000-2017 2#see license.txt for license details 3"""Tests for reportlab.lib.utils 4""" 5__version__='3.3.0' 6from reportlab.lib.testutils import setOutDir,makeSuiteForClasses, printLocation 7setOutDir(__name__) 8import os, time, sys 9import reportlab 10from reportlab import rl_config 11import unittest 12from reportlab.lib import colors 13from reportlab.lib.utils import recursiveImport, recursiveGetAttr, recursiveSetAttr, rl_isfile, \ 14 isCompactDistro, isPy3, isPyPy, TimeStamp, rl_get_module, \ 15 recursiveGetAttr, recursiveSetAttr, recursiveDelAttr, \ 16 asUnicode, asUnicodeEx, asBytes 17 18def _rel_open_and_read(fn): 19 from reportlab.lib.utils import open_and_read 20 from reportlab.lib.testutils import testsFolder 21 cwd = os.getcwd() 22 os.chdir(testsFolder) 23 try: 24 return open_and_read(fn) 25 finally: 26 os.chdir(cwd) 27 28class ImporterTestCase(unittest.TestCase): 29 "Test import utilities" 30 31 @classmethod 32 def setUpClass(cls): 33 from reportlab.lib.utils import get_rl_tempdir 34 cls._value = float(repr(time.time())) 35 s = int(cls._value) 36 cls._tempdir = get_rl_tempdir('reportlab_test','tmp_%s' % s) 37 if not os.path.isdir(cls._tempdir): 38 os.makedirs(cls._tempdir,0o700) 39 _testmodulename = os.path.join(cls._tempdir,'test_module_%s.py' % s) 40 with open(_testmodulename,'w') as f: 41 f.write('__all__=[]\nvalue=%s\n' % repr(cls._value)) 42 if sys.platform=='darwin' and isPy3: 43 time.sleep(0.3) 44 cls._testmodulename = os.path.splitext(os.path.basename(_testmodulename))[0] 45 46 @classmethod 47 def tearDownClass(cls): 48 from shutil import rmtree 49 rmtree(cls._tempdir,1) 50 51 def myAssertRaisesRegex(self,*args,**kwds): 52 return getattr(self,'assertRaisesRegex' if isPy3 else 'assertRaisesRegexp')(*args,**kwds) 53 54 def test1(self): 55 "try stuff known to be in the path" 56 m1 = recursiveImport('reportlab.pdfgen.canvas') 57 import reportlab.pdfgen.canvas 58 assert m1 == reportlab.pdfgen.canvas 59 60 def test2(self): 61 "try under a well known directory NOT on the path" 62 from reportlab.lib.testutils import testsFolder 63 D = os.path.join(testsFolder,'..','tools','pythonpoint') 64 fn = os.path.join(D,'stdparser.py') 65 if rl_isfile(fn) or rl_isfile(fn+'c') or rl_isfile(fn+'o'): 66 m1 = recursiveImport('stdparser', baseDir=D) 67 68 def test3(self): 69 "ensure CWD is on the path" 70 try: 71 cwd = os.getcwd() 72 os.chdir(self._tempdir) 73 m1 = recursiveImport(self._testmodulename) 74 finally: 75 os.chdir(cwd) 76 77 def test4(self): 78 "ensure noCWD removes current dir from path" 79 try: 80 cwd = os.getcwd() 81 os.chdir(self._tempdir) 82 import sys 83 try: 84 del sys.modules[self._testmodulename] 85 except KeyError: 86 pass 87 self.assertRaises(ImportError, 88 recursiveImport, 89 self._testmodulename, 90 noCWD=1) 91 finally: 92 os.chdir(cwd) 93 94 def test5(self): 95 "recursive attribute setting/getting on modules" 96 import reportlab.lib.units 97 inch = recursiveGetAttr(reportlab, 'lib.units.inch') 98 assert inch == 72 99 100 recursiveSetAttr(reportlab, 'lib.units.cubit', 18*inch) 101 cubit = recursiveGetAttr(reportlab, 'lib.units.cubit') 102 assert cubit == 18*inch 103 104 def test6(self): 105 "recursive attribute setting/getting on drawings" 106 from reportlab.graphics.charts.barcharts import sampleH1 107 drawing = sampleH1() 108 recursiveSetAttr(drawing, 'barchart.valueAxis.valueMax', 72) 109 theMax = recursiveGetAttr(drawing, 'barchart.valueAxis.valueMax') 110 assert theMax == 72 111 112 def test7(self): 113 "test open and read of a simple relative file" 114 b = _rel_open_and_read('../docs/images/Edit_Prefs.gif') 115 116 def test8(self): 117 "test open and read of a relative file: URL" 118 b = _rel_open_and_read('file:../docs/images/Edit_Prefs.gif') 119 120 def test9(self): 121 "test open and read of an http: URL" 122 from reportlab.lib.utils import open_and_read 123 b = open_and_read('http://www.reportlab.com/rsrc/encryption.gif') 124 125 def test10(self): 126 "test open and read of a simple relative file" 127 from reportlab.lib.utils import open_and_read, getBytesIO 128 b = getBytesIO(_rel_open_and_read('../docs/images/Edit_Prefs.gif')) 129 b = open_and_read(b) 130 131 def test11(self): 132 "test open and read of an RFC 2397 data URI with base64 encoding" 133 result = _rel_open_and_read('') 134 self.assertEqual(result,b'GIF87a\x01\x00\x01\x00\x80\x00\x00\xff\xff\xff\xff\xff\xff,\x00\x00\x00\x00\x01\x00\x01\x00\x00\x02\x02D\x01\x00;') 135 136 def test12(self): 137 "test open and read of an RFC 2397 data URI without an encoding" 138 result = _rel_open_and_read('data:text/plain;,Hello%20World') 139 self.assertEqual(result,b'Hello World') 140 141 def testRecursiveImportErrors(self): 142 "check we get useful error messages" 143 try: 144 m1 = recursiveImport('reportlab.pdfgen.brush') 145 self.fail("Imported a nonexistent module") 146 except ImportError as e: 147 self.assertIn('reportlab.pdfgen.brush',str(e)) 148 149 try: 150 m1 = recursiveImport('totally.non.existent') 151 self.fail("Imported a nonexistent module") 152 except ImportError as e: 153 self.assertIn('totally',str(e)) 154 155 try: 156 #import a module in the 'tests' directory with a bug 157 m1 = recursiveImport('unimportable') 158 self.fail("Imported a buggy module") 159 except Exception as e: 160 self.assertIn(("integer division by zeroException raised while importing 'unimportable': integer division by zero" 161 if isPyPy 162 else ('division by zero' if isPy3 163 else 'integer division or modulo by zero')) 164 ,str(e)) 165 def test14(self): 166 "test the TimeStamp behaviour" 167 oinvariant = rl_config.invariant 168 sden = 'SOURCE_DATE_EPOCH' 169 if sden in os.environ: 170 sde = os.environ[sden] 171 del os.environ[sden] 172 else: 173 sde = self 174 175 try: 176 rl_config.invariant = False 177 t = time.time() 178 ts = TimeStamp() 179 self.assertTrue(abs(t-ts.t)<1) 180 ts = TimeStamp(invariant=True) 181 self.assertEqual(ts.t,946684800.0) 182 self.assertEqual(ts.YMDhms,(2000, 1, 1, 0, 0, 0)) 183 self.assertEqual(ts.tzname,'UTC') 184 os.environ[sden] = '1490003100' 185 ts = TimeStamp(invariant=True) #debian variable takes precedence here 186 self.assertEqual(ts.t,1490003100) 187 self.assertEqual(ts.YMDhms,(2017, 3, 20, 9, 45, 0)) 188 self.assertEqual(ts.tzname,'UTC') 189 rl_config.invariant = True 190 ts = TimeStamp() #still takes precedence 191 self.assertEqual(ts.t,1490003100) 192 self.assertEqual(ts.YMDhms,(2017, 3, 20, 9, 45, 0)) 193 self.assertEqual(ts.tzname,'UTC') 194 del os.environ[sden] 195 ts = TimeStamp() #now rl_config takes precedence 196 self.assertEqual(ts.t,946684800.0) 197 self.assertEqual(ts.YMDhms,(2000, 1, 1, 0, 0, 0)) 198 self.assertEqual(ts.tzname,'UTC') 199 finally: 200 if sde is not self: 201 os.environ[sden] = sde 202 rl_config.invariant = oinvariant 203 204 def test15(self): 205 m = rl_get_module(self._testmodulename,self._tempdir) 206 self.assertEqual(self._value,m.value) 207 208 def test16(self): 209 from reportlab.lib import fontfinder 210 ff = fontfinder.FontFinder(useCache=False,recur=True) 211 ff.addDirectories(rl_config.T1SearchPath + rl_config.TTFSearchPath) 212 ff.search() 213 ff.getFamilyNames() 214 215 def test17(self): 216 self.assertEqual(asUnicode(u'abc'),u'abc') 217 self.assertEqual(asUnicode(b'abc'),u'abc') 218 self.assertRaises(AttributeError,asUnicode,['abc']) 219 self.myAssertRaisesRegex(AttributeError,r"asUnicode\(.*'list' object has no attribute 'decode'", asUnicode,['abc']) 220 self.assertEqual(asUnicodeEx(u'abc'),u'abc') 221 self.assertEqual(asUnicodeEx(b'abc'),u'abc') 222 self.assertEqual(asUnicodeEx(123),u'123') 223 self.assertEqual(asBytes(u'abc'),b'abc') 224 self.assertEqual(asBytes(b'abc'),b'abc') 225 self.assertRaises(AttributeError,asBytes,['abc']) 226 self.myAssertRaisesRegex(AttributeError,"asBytes\(.*'list' object has no attribute 'encode'", asBytes,['abc']) 227 228class RaccessTest: 229 l = [1, 2] 230 a = [0, [1, 2, [3,4]]] 231 b = {'x': {'y': 'y'}, 'z': [1, 2]} 232 z = 'z' 233 234class RaccessPerson: 235 settings = { 236 'autosave': True, 237 'style': { 238 'height': 30, 239 'width': 200 240 }, 241 'themes': ['light', 'dark'] 242 } 243 def __init__(self, name, age, friends): 244 self.name = name 245 self.age = age 246 self.friends = friends 247 248class RaccessTestCase(unittest.TestCase): 249 "Test recursive access functions" 250 def test1(self): 251 def innerTest(k,v): 252 obj = RaccessTest() 253 obj.t = obj 254 obj.a.append(obj) 255 obj.b['w'] = obj 256 self.assertEqual(recursiveGetAttr(obj,k),v,"error getattr(obj,%r)==%r" % (k,v)) 257 for k,v in [ 258 ('l', RaccessTest.l), 259 ('t.t.t.t.z', 'z'), 260 ('a[0]', 0), 261 ('a[1][0]', 1), 262 ('a[1][2]', [3,4]), 263 ('b["x"]', {'y': 'y'}), 264 ('b["x"]["y"]', 'y'), 265 ('b["z"]', [1,2]), 266 ('b["z"][1]', 2), 267 ('b["w"].z', 'z'), 268 ('b["w"].t.l', [1, 2]), 269 ('a[-1].z', 'z'), 270 ('l[-1]', 2), 271 ('a[2].t.a[-1].z', 'z'), 272 ('a[2].t.b["z"][0]', 1), 273 ('a[-1].t.z', 'z'), 274 ]: 275 innerTest(k,v) 276 277 def test_person_example(self): 278 bob = RaccessPerson(name="Bob", age=31, friends=[]) 279 jill = RaccessPerson(name="Jill", age=29, friends=[bob]) 280 jack = RaccessPerson(name="Jack", age=28, friends=[bob, jill]) 281 282 # Nothing new 283 self.assertEqual(recursiveGetAttr(bob, 'age') ,31) 284 285 # Lists 286 self.assertEqual(recursiveGetAttr(jill, 'friends[0].name') ,'Bob') 287 self.assertEqual(recursiveGetAttr(jack, 'friends[-1].age') ,29) 288 289 # Dict lookups 290 self.assertEqual(recursiveGetAttr(jack, 'settings["style"]["width"]') ,200) 291 292 # Combination of lookups 293 self.assertEqual(recursiveGetAttr(jack, 'settings["themes"][-2]') ,'light') 294 self.assertEqual(recursiveGetAttr(jack, 'friends[-1].settings["themes"][1]') ,'dark') 295 296 # Setattr 297 #recursiveSetAttr(bob, 'settings["style"]["width"]', 400) 298 #self.assertEqual(recursiveGetAttr(bob, 'settings["style"]["width"]') ,400) 299 300 # Nested objects 301 recursiveSetAttr(bob, 'friends', [jack, jill]) 302 self.assertEqual(recursiveGetAttr(jack, 'friends[0].friends[0]') ,jack) 303 304 recursiveSetAttr(jill, 'friends[0].age', 32) 305 self.assertEqual(bob.age ,32) 306 307 # Deletion 308 #recursiveDelAttr(jill, 'friends[0]') 309 #self.assertEqual(len(jill.friends) ,0) 310 311 recursiveDelAttr(jill, 'age') 312 assert not hasattr(jill, 'age') 313 314 recursiveDelAttr(bob, 'friends[0].age') 315 assert not hasattr(jack, 'age') 316 317 # Unsupported 318 #with self.assertRaises(NotImplementedError) as e: 319 # recursiveGetAttr(bob, 'friends[0+1]') 320 321 # Nice try, function calls are not allowed 322 #with self.assertRaises(ValueError): 323 # recursiveGetAttr(bob, 'friends.pop(0)') 324 325 # Must be an expression 326 with self.assertRaises(SyntaxError): 327 recursiveGetAttr(bob, 'friends = []') 328 329 # Must be an expression 330 with self.assertRaises(SyntaxError): 331 recursiveGetAttr(bob, 'friends..') 332 333 # Must be an expression 334 with self.assertRaises(KeyError): 335 recursiveGetAttr(bob, 'settings["DoesNotExist"]') 336 337 # Must be an expression 338 with self.assertRaises(IndexError): 339 recursiveGetAttr(bob, 'friends[100]') 340 341 def test_empty(self): 342 obj = RaccessTest() 343 with self.assertRaises(ValueError): 344 recursiveGetAttr(obj," ") 345 346 with self.assertRaises(ValueError): 347 recursiveGetAttr(obj,"") 348 349 with self.assertRaises(TypeError): 350 recursiveGetAttr(obj, 0) 351 352 with self.assertRaises(TypeError): 353 recursiveGetAttr(obj, None) 354 355 with self.assertRaises(TypeError): 356 recursiveGetAttr(obj, obj) 357 358def makeSuite(): 359 return makeSuiteForClasses(ImporterTestCase,RaccessTestCase) 360 361if __name__ == "__main__": #noruntests 362 unittest.TextTestRunner().run(makeSuite()) 363 printLocation() 364