David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 1 | import unittest |
Hirokazu Yamamoto | dda7fdf | 2010-09-23 15:28:35 +0000 | [diff] [blame] | 2 | import sys |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 3 | import os |
Serhiy Storchaka | 9e6b975 | 2013-08-21 21:38:21 +0300 | [diff] [blame] | 4 | import _testcapi |
Benjamin Peterson | ee8712c | 2008-05-20 21:35:26 +0000 | [diff] [blame] | 5 | from test import support |
R. David Murray | a21e4ca | 2009-03-31 23:16:50 +0000 | [diff] [blame] | 6 | |
| 7 | # Skip this test if the _tkinter module wasn't built. |
| 8 | _tkinter = support.import_module('_tkinter') |
| 9 | |
Terry Jan Reedy | 743c85a | 2013-03-16 02:37:06 -0400 | [diff] [blame] | 10 | # Make sure tkinter._fix runs to set up the environment |
| 11 | support.import_fresh_module('tkinter') |
| 12 | |
Guilherme Polo | fa8fba9 | 2009-02-07 02:33:47 +0000 | [diff] [blame] | 13 | from tkinter import Tcl |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 14 | from _tkinter import TclError |
| 15 | |
Serhiy Storchaka | dfa689b | 2013-09-08 20:29:37 +0300 | [diff] [blame] | 16 | tcl_version = _tkinter.TCL_VERSION.split('.') |
| 17 | try: |
| 18 | for i in range(len(tcl_version)): |
| 19 | tcl_version[i] = int(tcl_version[i]) |
| 20 | except ValueError: |
| 21 | pass |
| 22 | tcl_version = tuple(tcl_version) |
| 23 | |
Benjamin Peterson | c4bbc8d | 2009-01-30 03:39:35 +0000 | [diff] [blame] | 24 | |
| 25 | class TkinterTest(unittest.TestCase): |
| 26 | |
| 27 | def testFlattenLen(self): |
| 28 | # flatten(<object with no length>) |
| 29 | self.assertRaises(TypeError, _tkinter._flatten, True) |
| 30 | |
| 31 | |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 32 | class TclTest(unittest.TestCase): |
| 33 | |
| 34 | def setUp(self): |
| 35 | self.interp = Tcl() |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 36 | self.wantobjects = self.interp.tk.wantobjects() |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 37 | |
| 38 | def testEval(self): |
| 39 | tcl = self.interp |
| 40 | tcl.eval('set a 1') |
| 41 | self.assertEqual(tcl.eval('set a'),'1') |
| 42 | |
| 43 | def testEvalException(self): |
| 44 | tcl = self.interp |
| 45 | self.assertRaises(TclError,tcl.eval,'set a') |
| 46 | |
| 47 | def testEvalException2(self): |
| 48 | tcl = self.interp |
| 49 | self.assertRaises(TclError,tcl.eval,'this is wrong') |
| 50 | |
| 51 | def testCall(self): |
| 52 | tcl = self.interp |
| 53 | tcl.call('set','a','1') |
| 54 | self.assertEqual(tcl.call('set','a'),'1') |
| 55 | |
| 56 | def testCallException(self): |
| 57 | tcl = self.interp |
| 58 | self.assertRaises(TclError,tcl.call,'set','a') |
| 59 | |
| 60 | def testCallException2(self): |
| 61 | tcl = self.interp |
| 62 | self.assertRaises(TclError,tcl.call,'this','is','wrong') |
| 63 | |
| 64 | def testSetVar(self): |
| 65 | tcl = self.interp |
| 66 | tcl.setvar('a','1') |
| 67 | self.assertEqual(tcl.eval('set a'),'1') |
| 68 | |
| 69 | def testSetVarArray(self): |
| 70 | tcl = self.interp |
| 71 | tcl.setvar('a(1)','1') |
| 72 | self.assertEqual(tcl.eval('set a(1)'),'1') |
| 73 | |
| 74 | def testGetVar(self): |
| 75 | tcl = self.interp |
| 76 | tcl.eval('set a 1') |
| 77 | self.assertEqual(tcl.getvar('a'),'1') |
| 78 | |
| 79 | def testGetVarArray(self): |
| 80 | tcl = self.interp |
| 81 | tcl.eval('set a(1) 1') |
| 82 | self.assertEqual(tcl.getvar('a(1)'),'1') |
| 83 | |
| 84 | def testGetVarException(self): |
| 85 | tcl = self.interp |
| 86 | self.assertRaises(TclError,tcl.getvar,'a') |
| 87 | |
| 88 | def testGetVarArrayException(self): |
| 89 | tcl = self.interp |
| 90 | self.assertRaises(TclError,tcl.getvar,'a(1)') |
| 91 | |
| 92 | def testUnsetVar(self): |
| 93 | tcl = self.interp |
| 94 | tcl.setvar('a',1) |
| 95 | self.assertEqual(tcl.eval('info exists a'),'1') |
| 96 | tcl.unsetvar('a') |
| 97 | self.assertEqual(tcl.eval('info exists a'),'0') |
| 98 | |
| 99 | def testUnsetVarArray(self): |
| 100 | tcl = self.interp |
| 101 | tcl.setvar('a(1)',1) |
| 102 | tcl.setvar('a(2)',2) |
| 103 | self.assertEqual(tcl.eval('info exists a(1)'),'1') |
| 104 | self.assertEqual(tcl.eval('info exists a(2)'),'1') |
| 105 | tcl.unsetvar('a(1)') |
| 106 | self.assertEqual(tcl.eval('info exists a(1)'),'0') |
| 107 | self.assertEqual(tcl.eval('info exists a(2)'),'1') |
| 108 | |
| 109 | def testUnsetVarException(self): |
| 110 | tcl = self.interp |
| 111 | self.assertRaises(TclError,tcl.unsetvar,'a') |
Tim Peters | 27f8836 | 2004-07-08 04:22:35 +0000 | [diff] [blame] | 112 | |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 113 | def testEvalFile(self): |
| 114 | tcl = self.interp |
| 115 | filename = "testEvalFile.tcl" |
| 116 | fd = open(filename,'w') |
| 117 | script = """set a 1 |
| 118 | set b 2 |
| 119 | set c [ expr $a + $b ] |
| 120 | """ |
| 121 | fd.write(script) |
| 122 | fd.close() |
| 123 | tcl.evalfile(filename) |
Neal Norwitz | 9a8d55e | 2004-02-29 15:37:50 +0000 | [diff] [blame] | 124 | os.remove(filename) |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 125 | self.assertEqual(tcl.eval('set a'),'1') |
| 126 | self.assertEqual(tcl.eval('set b'),'2') |
| 127 | self.assertEqual(tcl.eval('set c'),'3') |
| 128 | |
| 129 | def testEvalFileException(self): |
| 130 | tcl = self.interp |
| 131 | filename = "doesnotexists" |
| 132 | try: |
| 133 | os.remove(filename) |
Guido van Rossum | b940e11 | 2007-01-10 16:19:56 +0000 | [diff] [blame] | 134 | except Exception as e: |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 135 | pass |
| 136 | self.assertRaises(TclError,tcl.evalfile,filename) |
| 137 | |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 138 | def testPackageRequireException(self): |
| 139 | tcl = self.interp |
| 140 | self.assertRaises(TclError,tcl.eval,'package require DNE') |
| 141 | |
Hirokazu Yamamoto | dda7fdf | 2010-09-23 15:28:35 +0000 | [diff] [blame] | 142 | @unittest.skipUnless(sys.platform == 'win32', 'Requires Windows') |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 143 | def testLoadWithUNC(self): |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 144 | # Build a UNC path from the regular path. |
| 145 | # Something like |
| 146 | # \\%COMPUTERNAME%\c$\python27\python.exe |
| 147 | |
| 148 | fullname = os.path.abspath(sys.executable) |
| 149 | if fullname[1] != ':': |
Hirokazu Yamamoto | dda7fdf | 2010-09-23 15:28:35 +0000 | [diff] [blame] | 150 | raise unittest.SkipTest('Absolute path should have drive part') |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 151 | unc_name = r'\\%s\%s$\%s' % (os.environ['COMPUTERNAME'], |
| 152 | fullname[0], |
| 153 | fullname[3:]) |
Hirokazu Yamamoto | dda7fdf | 2010-09-23 15:28:35 +0000 | [diff] [blame] | 154 | if not os.path.exists(unc_name): |
| 155 | raise unittest.SkipTest('Cannot connect to UNC Path') |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 156 | |
Georg Brandl | ec5ae3b | 2010-07-31 19:17:11 +0000 | [diff] [blame] | 157 | with support.EnvironmentVarGuard() as env: |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 158 | env.unset("TCL_LIBRARY") |
Hirokazu Yamamoto | f2dec8d | 2010-09-18 04:42:41 +0000 | [diff] [blame] | 159 | f = os.popen('%s -c "import tkinter; print(tkinter)"' % (unc_name,)) |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 160 | |
Hirokazu Yamamoto | f2dec8d | 2010-09-18 04:42:41 +0000 | [diff] [blame] | 161 | self.assertIn('tkinter', f.read()) |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 162 | # exit code must be zero |
| 163 | self.assertEqual(f.close(), None) |
| 164 | |
Serhiy Storchaka | df4bb46 | 2013-02-07 15:40:03 +0200 | [diff] [blame] | 165 | def test_passing_values(self): |
| 166 | def passValue(value): |
| 167 | return self.interp.call('set', '_', value) |
| 168 | |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 169 | self.assertEqual(passValue(True), True if self.wantobjects else '1') |
| 170 | self.assertEqual(passValue(False), False if self.wantobjects else '0') |
Serhiy Storchaka | df4bb46 | 2013-02-07 15:40:03 +0200 | [diff] [blame] | 171 | self.assertEqual(passValue('string'), 'string') |
| 172 | self.assertEqual(passValue('string\u20ac'), 'string\u20ac') |
| 173 | for i in (0, 1, -1, 2**31-1, -2**31): |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 174 | self.assertEqual(passValue(i), i if self.wantobjects else str(i)) |
Serhiy Storchaka | df4bb46 | 2013-02-07 15:40:03 +0200 | [diff] [blame] | 175 | for f in (0.0, 1.0, -1.0, 1/3, |
| 176 | sys.float_info.min, sys.float_info.max, |
| 177 | -sys.float_info.min, -sys.float_info.max): |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 178 | if self.wantobjects: |
Serhiy Storchaka | df4bb46 | 2013-02-07 15:40:03 +0200 | [diff] [blame] | 179 | self.assertEqual(passValue(f), f) |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 180 | else: |
| 181 | self.assertEqual(float(passValue(f)), f) |
| 182 | if self.wantobjects: |
| 183 | f = passValue(float('nan')) |
| 184 | self.assertNotEqual(f, f) |
| 185 | self.assertEqual(passValue(float('inf')), float('inf')) |
| 186 | self.assertEqual(passValue(-float('inf')), -float('inf')) |
| 187 | else: |
| 188 | f = float(passValue(float('nan'))) |
| 189 | self.assertNotEqual(f, f) |
| 190 | self.assertEqual(float(passValue(float('inf'))), float('inf')) |
| 191 | self.assertEqual(float(passValue(-float('inf'))), -float('inf')) |
| 192 | self.assertEqual(passValue((1, '2', (3.4,))), |
| 193 | (1, '2', (3.4,)) if self.wantobjects else '1 2 3.4') |
Martin v. Löwis | 2d5157e | 2010-06-04 19:50:26 +0000 | [diff] [blame] | 194 | |
Serhiy Storchaka | 4b73016 | 2014-01-23 09:44:05 +0200 | [diff] [blame] | 195 | def test_user_command(self): |
| 196 | result = None |
| 197 | def testfunc(arg): |
| 198 | nonlocal result |
| 199 | result = arg |
| 200 | return arg |
| 201 | self.interp.createcommand('testfunc', testfunc) |
Serhiy Storchaka | bdf0cb5 | 2014-01-23 16:48:44 +0200 | [diff] [blame] | 202 | def check(value, expected, eq=self.assertEqual): |
| 203 | r = self.interp.call('testfunc', value) |
| 204 | self.assertIsInstance(result, str) |
| 205 | eq(result, expected) |
| 206 | self.assertIsInstance(r, str) |
| 207 | eq(r, expected) |
| 208 | def float_eq(actual, expected): |
| 209 | expected = float(expected) |
| 210 | self.assertAlmostEqual(float(actual), expected, |
| 211 | delta=abs(expected) * 1e-10) |
| 212 | def nan_eq(actual, expected): |
| 213 | actual = float(actual) |
| 214 | self.assertNotEqual(actual, actual) |
Serhiy Storchaka | 4b73016 | 2014-01-23 09:44:05 +0200 | [diff] [blame] | 215 | |
| 216 | check(True, '1') |
| 217 | check(False, '0') |
| 218 | check('string', 'string') |
| 219 | check('string\xbd', 'string\xbd') |
| 220 | check('string\u20ac', 'string\u20ac') |
| 221 | for i in (0, 1, -1, 2**31-1, -2**31): |
| 222 | check(i, str(i)) |
Serhiy Storchaka | bdf0cb5 | 2014-01-23 16:48:44 +0200 | [diff] [blame] | 223 | for f in (0.0, 1.0, -1.0): |
| 224 | check(f, repr(f)) |
| 225 | for f in (1/3.0, sys.float_info.min, sys.float_info.max, |
Serhiy Storchaka | 4b73016 | 2014-01-23 09:44:05 +0200 | [diff] [blame] | 226 | -sys.float_info.min, -sys.float_info.max): |
Serhiy Storchaka | bdf0cb5 | 2014-01-23 16:48:44 +0200 | [diff] [blame] | 227 | check(f, f, eq=float_eq) |
| 228 | check(float('inf'), 'Inf', eq=float_eq) |
| 229 | check(-float('inf'), '-Inf', eq=float_eq) |
| 230 | check(float('nan'), 'NaN', eq=nan_eq) |
Serhiy Storchaka | 4b73016 | 2014-01-23 09:44:05 +0200 | [diff] [blame] | 231 | check((), '') |
| 232 | check((1, (2,), (3, 4), '5 6', ()), '1 2 {3 4} {5 6} {}') |
| 233 | |
Serhiy Storchaka | 275d5fd | 2013-07-11 20:34:47 +0300 | [diff] [blame] | 234 | def test_splitlist(self): |
| 235 | splitlist = self.interp.tk.splitlist |
| 236 | call = self.interp.tk.call |
| 237 | self.assertRaises(TypeError, splitlist) |
| 238 | self.assertRaises(TypeError, splitlist, 'a', 'b') |
| 239 | self.assertRaises(TypeError, splitlist, 2) |
| 240 | testcases = [ |
| 241 | ('2', ('2',)), |
| 242 | ('', ()), |
| 243 | ('{}', ('',)), |
| 244 | ('""', ('',)), |
| 245 | ('a\n b\t\r c\n ', ('a', 'b', 'c')), |
| 246 | (b'a\n b\t\r c\n ', ('a', 'b', 'c')), |
| 247 | ('a \u20ac', ('a', '\u20ac')), |
| 248 | (b'a \xe2\x82\xac', ('a', '\u20ac')), |
| 249 | ('a {b c}', ('a', 'b c')), |
| 250 | (r'a b\ c', ('a', 'b c')), |
| 251 | (('a', 'b c'), ('a', 'b c')), |
| 252 | ('a 2', ('a', '2')), |
| 253 | (('a', 2), ('a', 2)), |
| 254 | ('a 3.4', ('a', '3.4')), |
| 255 | (('a', 3.4), ('a', 3.4)), |
| 256 | ((), ()), |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 257 | (call('list', 1, '2', (3.4,)), |
| 258 | (1, '2', (3.4,)) if self.wantobjects else |
| 259 | ('1', '2', '3.4')), |
Serhiy Storchaka | 275d5fd | 2013-07-11 20:34:47 +0300 | [diff] [blame] | 260 | ] |
Serhiy Storchaka | dfa689b | 2013-09-08 20:29:37 +0300 | [diff] [blame] | 261 | if tcl_version >= (8, 5): |
| 262 | testcases += [ |
| 263 | (call('dict', 'create', 1, '\u20ac', b'\xe2\x82\xac', (3.4,)), |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 264 | (1, '\u20ac', '\u20ac', (3.4,)) if self.wantobjects else |
| 265 | ('1', '\u20ac', '\u20ac', '3.4')), |
Serhiy Storchaka | dfa689b | 2013-09-08 20:29:37 +0300 | [diff] [blame] | 266 | ] |
Serhiy Storchaka | 275d5fd | 2013-07-11 20:34:47 +0300 | [diff] [blame] | 267 | for arg, res in testcases: |
| 268 | self.assertEqual(splitlist(arg), res, msg=arg) |
| 269 | self.assertRaises(TclError, splitlist, '{') |
| 270 | |
| 271 | def test_split(self): |
| 272 | split = self.interp.tk.split |
| 273 | call = self.interp.tk.call |
| 274 | self.assertRaises(TypeError, split) |
| 275 | self.assertRaises(TypeError, split, 'a', 'b') |
| 276 | self.assertRaises(TypeError, split, 2) |
| 277 | testcases = [ |
| 278 | ('2', '2'), |
| 279 | ('', ''), |
| 280 | ('{}', ''), |
| 281 | ('""', ''), |
| 282 | ('{', '{'), |
| 283 | ('a\n b\t\r c\n ', ('a', 'b', 'c')), |
| 284 | (b'a\n b\t\r c\n ', ('a', 'b', 'c')), |
| 285 | ('a \u20ac', ('a', '\u20ac')), |
| 286 | (b'a \xe2\x82\xac', ('a', '\u20ac')), |
| 287 | ('a {b c}', ('a', ('b', 'c'))), |
| 288 | (r'a b\ c', ('a', ('b', 'c'))), |
| 289 | (('a', b'b c'), ('a', ('b', 'c'))), |
| 290 | (('a', 'b c'), ('a', ('b', 'c'))), |
| 291 | ('a 2', ('a', '2')), |
| 292 | (('a', 2), ('a', 2)), |
| 293 | ('a 3.4', ('a', '3.4')), |
| 294 | (('a', 3.4), ('a', 3.4)), |
| 295 | (('a', (2, 3.4)), ('a', (2, 3.4))), |
| 296 | ((), ()), |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 297 | (call('list', 1, '2', (3.4,)), |
| 298 | (1, '2', (3.4,)) if self.wantobjects else |
| 299 | ('1', '2', '3.4')), |
Serhiy Storchaka | 275d5fd | 2013-07-11 20:34:47 +0300 | [diff] [blame] | 300 | ] |
Serhiy Storchaka | dfa689b | 2013-09-08 20:29:37 +0300 | [diff] [blame] | 301 | if tcl_version >= (8, 5): |
| 302 | testcases += [ |
| 303 | (call('dict', 'create', 12, '\u20ac', b'\xe2\x82\xac', (3.4,)), |
Serhiy Storchaka | cc4290b | 2013-12-25 17:29:01 +0200 | [diff] [blame] | 304 | (12, '\u20ac', '\u20ac', (3.4,)) if self.wantobjects else |
| 305 | ('12', '\u20ac', '\u20ac', '3.4')), |
Serhiy Storchaka | dfa689b | 2013-09-08 20:29:37 +0300 | [diff] [blame] | 306 | ] |
Serhiy Storchaka | 275d5fd | 2013-07-11 20:34:47 +0300 | [diff] [blame] | 307 | for arg, res in testcases: |
| 308 | self.assertEqual(split(arg), res, msg=arg) |
| 309 | |
Neal Norwitz | 63dfece | 2004-02-19 02:37:29 +0000 | [diff] [blame] | 310 | |
Serhiy Storchaka | 9e6b975 | 2013-08-21 21:38:21 +0300 | [diff] [blame] | 311 | class BigmemTclTest(unittest.TestCase): |
| 312 | |
| 313 | def setUp(self): |
| 314 | self.interp = Tcl() |
| 315 | |
| 316 | @unittest.skipUnless(_testcapi.INT_MAX < _testcapi.PY_SSIZE_T_MAX, |
| 317 | "needs UINT_MAX < SIZE_MAX") |
| 318 | @support.bigmemtest(size=_testcapi.INT_MAX + 1, memuse=5, dry_run=False) |
| 319 | def test_huge_string(self, size): |
| 320 | value = ' ' * size |
| 321 | self.assertRaises(OverflowError, self.interp.call, 'set', '_', value) |
| 322 | |
| 323 | |
Serhiy Storchaka | 8db1823 | 2013-11-20 17:43:49 +0200 | [diff] [blame] | 324 | def setUpModule(): |
| 325 | if support.verbose: |
| 326 | tcl = Tcl() |
| 327 | print('patchlevel =', tcl.call('info', 'patchlevel')) |
| 328 | |
| 329 | |
Neal Norwitz | 63dfece | 2004-02-19 02:37:29 +0000 | [diff] [blame] | 330 | def test_main(): |
Serhiy Storchaka | 9e6b975 | 2013-08-21 21:38:21 +0300 | [diff] [blame] | 331 | support.run_unittest(TclTest, TkinterTest, BigmemTclTest) |
Neal Norwitz | 63dfece | 2004-02-19 02:37:29 +0000 | [diff] [blame] | 332 | |
David Ascher | e2b4b32 | 2004-02-18 05:59:53 +0000 | [diff] [blame] | 333 | if __name__ == "__main__": |
Neal Norwitz | 63dfece | 2004-02-19 02:37:29 +0000 | [diff] [blame] | 334 | test_main() |