test_winreg.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. # Test the windows specific win32reg module.
  2. # Only win32reg functions not hit here: FlushKey, LoadKey and SaveKey
  3. import os, sys, errno
  4. import unittest
  5. from test.support import import_helper
  6. import threading
  7. from platform import machine, win32_edition
  8. # Do this first so test will be skipped if module doesn't exist
  9. import_helper.import_module('winreg', required_on=['win'])
  10. # Now import everything
  11. from winreg import *
  12. try:
  13. REMOTE_NAME = sys.argv[sys.argv.index("--remote")+1]
  14. except (IndexError, ValueError):
  15. REMOTE_NAME = None
  16. # tuple of (major, minor)
  17. WIN_VER = sys.getwindowsversion()[:2]
  18. # Some tests should only run on 64-bit architectures where WOW64 will be.
  19. WIN64_MACHINE = True if machine() == "AMD64" else False
  20. # Starting with Windows 7 and Windows Server 2008 R2, WOW64 no longer uses
  21. # registry reflection and formerly reflected keys are shared instead.
  22. # Windows 7 and Windows Server 2008 R2 are version 6.1. Due to this, some
  23. # tests are only valid up until 6.1
  24. HAS_REFLECTION = True if WIN_VER < (6, 1) else False
  25. # Use a per-process key to prevent concurrent test runs (buildbot!) from
  26. # stomping on each other.
  27. test_key_base = "Python Test Key [%d] - Delete Me" % (os.getpid(),)
  28. test_key_name = "SOFTWARE\\" + test_key_base
  29. # On OS'es that support reflection we should test with a reflected key
  30. test_reflect_key_name = "SOFTWARE\\Classes\\" + test_key_base
  31. test_data = [
  32. ("Int Value", 45, REG_DWORD),
  33. ("Qword Value", 0x1122334455667788, REG_QWORD),
  34. ("String Val", "A string value", REG_SZ),
  35. ("StringExpand", "The path is %path%", REG_EXPAND_SZ),
  36. ("Multi-string", ["Lots", "of", "string", "values"], REG_MULTI_SZ),
  37. ("Multi-nul", ["", "", "", ""], REG_MULTI_SZ),
  38. ("Raw Data", b"binary\x00data", REG_BINARY),
  39. ("Big String", "x"*(2**14-1), REG_SZ),
  40. ("Big Binary", b"x"*(2**14), REG_BINARY),
  41. # Two and three kanjis, meaning: "Japan" and "Japanese".
  42. ("Japanese 日本", "日本語", REG_SZ),
  43. ]
  44. class BaseWinregTests(unittest.TestCase):
  45. def setUp(self):
  46. # Make sure that the test key is absent when the test
  47. # starts.
  48. self.delete_tree(HKEY_CURRENT_USER, test_key_name)
  49. def delete_tree(self, root, subkey):
  50. try:
  51. hkey = OpenKey(root, subkey, 0, KEY_ALL_ACCESS)
  52. except OSError:
  53. # subkey does not exist
  54. return
  55. while True:
  56. try:
  57. subsubkey = EnumKey(hkey, 0)
  58. except OSError:
  59. # no more subkeys
  60. break
  61. self.delete_tree(hkey, subsubkey)
  62. CloseKey(hkey)
  63. DeleteKey(root, subkey)
  64. def _write_test_data(self, root_key, subkeystr="sub_key",
  65. CreateKey=CreateKey):
  66. # Set the default value for this key.
  67. SetValue(root_key, test_key_name, REG_SZ, "Default value")
  68. key = CreateKey(root_key, test_key_name)
  69. self.assertTrue(key.handle != 0)
  70. # Create a sub-key
  71. sub_key = CreateKey(key, subkeystr)
  72. # Give the sub-key some named values
  73. for value_name, value_data, value_type in test_data:
  74. SetValueEx(sub_key, value_name, 0, value_type, value_data)
  75. # Check we wrote as many items as we thought.
  76. nkeys, nvalues, since_mod = QueryInfoKey(key)
  77. self.assertEqual(nkeys, 1, "Not the correct number of sub keys")
  78. self.assertEqual(nvalues, 1, "Not the correct number of values")
  79. nkeys, nvalues, since_mod = QueryInfoKey(sub_key)
  80. self.assertEqual(nkeys, 0, "Not the correct number of sub keys")
  81. self.assertEqual(nvalues, len(test_data),
  82. "Not the correct number of values")
  83. # Close this key this way...
  84. # (but before we do, copy the key as an integer - this allows
  85. # us to test that the key really gets closed).
  86. int_sub_key = int(sub_key)
  87. CloseKey(sub_key)
  88. try:
  89. QueryInfoKey(int_sub_key)
  90. self.fail("It appears the CloseKey() function does "
  91. "not close the actual key!")
  92. except OSError:
  93. pass
  94. # ... and close that key that way :-)
  95. int_key = int(key)
  96. key.Close()
  97. try:
  98. QueryInfoKey(int_key)
  99. self.fail("It appears the key.Close() function "
  100. "does not close the actual key!")
  101. except OSError:
  102. pass
  103. def _read_test_data(self, root_key, subkeystr="sub_key", OpenKey=OpenKey):
  104. # Check we can get default value for this key.
  105. val = QueryValue(root_key, test_key_name)
  106. self.assertEqual(val, "Default value",
  107. "Registry didn't give back the correct value")
  108. key = OpenKey(root_key, test_key_name)
  109. # Read the sub-keys
  110. with OpenKey(key, subkeystr) as sub_key:
  111. # Check I can enumerate over the values.
  112. index = 0
  113. while 1:
  114. try:
  115. data = EnumValue(sub_key, index)
  116. except OSError:
  117. break
  118. self.assertEqual(data in test_data, True,
  119. "Didn't read back the correct test data")
  120. index = index + 1
  121. self.assertEqual(index, len(test_data),
  122. "Didn't read the correct number of items")
  123. # Check I can directly access each item
  124. for value_name, value_data, value_type in test_data:
  125. read_val, read_typ = QueryValueEx(sub_key, value_name)
  126. self.assertEqual(read_val, value_data,
  127. "Could not directly read the value")
  128. self.assertEqual(read_typ, value_type,
  129. "Could not directly read the value")
  130. sub_key.Close()
  131. # Enumerate our main key.
  132. read_val = EnumKey(key, 0)
  133. self.assertEqual(read_val, subkeystr, "Read subkey value wrong")
  134. try:
  135. EnumKey(key, 1)
  136. self.fail("Was able to get a second key when I only have one!")
  137. except OSError:
  138. pass
  139. key.Close()
  140. def _delete_test_data(self, root_key, subkeystr="sub_key"):
  141. key = OpenKey(root_key, test_key_name, 0, KEY_ALL_ACCESS)
  142. sub_key = OpenKey(key, subkeystr, 0, KEY_ALL_ACCESS)
  143. # It is not necessary to delete the values before deleting
  144. # the key (although subkeys must not exist). We delete them
  145. # manually just to prove we can :-)
  146. for value_name, value_data, value_type in test_data:
  147. DeleteValue(sub_key, value_name)
  148. nkeys, nvalues, since_mod = QueryInfoKey(sub_key)
  149. self.assertEqual(nkeys, 0, "subkey not empty before delete")
  150. self.assertEqual(nvalues, 0, "subkey not empty before delete")
  151. sub_key.Close()
  152. DeleteKey(key, subkeystr)
  153. try:
  154. # Shouldn't be able to delete it twice!
  155. DeleteKey(key, subkeystr)
  156. self.fail("Deleting the key twice succeeded")
  157. except OSError:
  158. pass
  159. key.Close()
  160. DeleteKey(root_key, test_key_name)
  161. # Opening should now fail!
  162. try:
  163. key = OpenKey(root_key, test_key_name)
  164. self.fail("Could open the non-existent key")
  165. except OSError: # Use this error name this time
  166. pass
  167. def _test_all(self, root_key, subkeystr="sub_key"):
  168. self._write_test_data(root_key, subkeystr)
  169. self._read_test_data(root_key, subkeystr)
  170. self._delete_test_data(root_key, subkeystr)
  171. def _test_named_args(self, key, sub_key):
  172. with CreateKeyEx(key=key, sub_key=sub_key, reserved=0,
  173. access=KEY_ALL_ACCESS) as ckey:
  174. self.assertTrue(ckey.handle != 0)
  175. with OpenKeyEx(key=key, sub_key=sub_key, reserved=0,
  176. access=KEY_ALL_ACCESS) as okey:
  177. self.assertTrue(okey.handle != 0)
  178. class LocalWinregTests(BaseWinregTests):
  179. def test_registry_works(self):
  180. self._test_all(HKEY_CURRENT_USER)
  181. self._test_all(HKEY_CURRENT_USER, "日本-subkey")
  182. def test_registry_works_extended_functions(self):
  183. # Substitute the regular CreateKey and OpenKey calls with their
  184. # extended counterparts.
  185. # Note: DeleteKeyEx is not used here because it is platform dependent
  186. cke = lambda key, sub_key: CreateKeyEx(key, sub_key, 0, KEY_ALL_ACCESS)
  187. self._write_test_data(HKEY_CURRENT_USER, CreateKey=cke)
  188. oke = lambda key, sub_key: OpenKeyEx(key, sub_key, 0, KEY_READ)
  189. self._read_test_data(HKEY_CURRENT_USER, OpenKey=oke)
  190. self._delete_test_data(HKEY_CURRENT_USER)
  191. def test_named_arguments(self):
  192. self._test_named_args(HKEY_CURRENT_USER, test_key_name)
  193. # Use the regular DeleteKey to clean up
  194. # DeleteKeyEx takes named args and is tested separately
  195. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  196. def test_connect_registry_to_local_machine_works(self):
  197. # perform minimal ConnectRegistry test which just invokes it
  198. h = ConnectRegistry(None, HKEY_LOCAL_MACHINE)
  199. self.assertNotEqual(h.handle, 0)
  200. h.Close()
  201. self.assertEqual(h.handle, 0)
  202. def test_nonexistent_remote_registry(self):
  203. connect = lambda: ConnectRegistry("abcdefghijkl", HKEY_CURRENT_USER)
  204. self.assertRaises(OSError, connect)
  205. def testExpandEnvironmentStrings(self):
  206. r = ExpandEnvironmentStrings("%windir%\\test")
  207. self.assertEqual(type(r), str)
  208. self.assertEqual(r, os.environ["windir"] + "\\test")
  209. def test_context_manager(self):
  210. # ensure that the handle is closed if an exception occurs
  211. try:
  212. with ConnectRegistry(None, HKEY_LOCAL_MACHINE) as h:
  213. self.assertNotEqual(h.handle, 0)
  214. raise OSError
  215. except OSError:
  216. self.assertEqual(h.handle, 0)
  217. def test_changing_value(self):
  218. # Issue2810: A race condition in 2.6 and 3.1 may cause
  219. # EnumValue or QueryValue to raise "WindowsError: More data is
  220. # available"
  221. done = False
  222. class VeryActiveThread(threading.Thread):
  223. def run(self):
  224. with CreateKey(HKEY_CURRENT_USER, test_key_name) as key:
  225. use_short = True
  226. long_string = 'x'*2000
  227. while not done:
  228. s = 'x' if use_short else long_string
  229. use_short = not use_short
  230. SetValue(key, 'changing_value', REG_SZ, s)
  231. thread = VeryActiveThread()
  232. thread.start()
  233. try:
  234. with CreateKey(HKEY_CURRENT_USER,
  235. test_key_name+'\\changing_value') as key:
  236. for _ in range(1000):
  237. num_subkeys, num_values, t = QueryInfoKey(key)
  238. for i in range(num_values):
  239. name = EnumValue(key, i)
  240. QueryValue(key, name[0])
  241. finally:
  242. done = True
  243. thread.join()
  244. DeleteKey(HKEY_CURRENT_USER, test_key_name+'\\changing_value')
  245. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  246. def test_long_key(self):
  247. # Issue2810, in 2.6 and 3.1 when the key name was exactly 256
  248. # characters, EnumKey raised "WindowsError: More data is
  249. # available"
  250. name = 'x'*256
  251. try:
  252. with CreateKey(HKEY_CURRENT_USER, test_key_name) as key:
  253. SetValue(key, name, REG_SZ, 'x')
  254. num_subkeys, num_values, t = QueryInfoKey(key)
  255. EnumKey(key, 0)
  256. finally:
  257. DeleteKey(HKEY_CURRENT_USER, '\\'.join((test_key_name, name)))
  258. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  259. def test_dynamic_key(self):
  260. # Issue2810, when the value is dynamically generated, these
  261. # raise "WindowsError: More data is available" in 2.6 and 3.1
  262. try:
  263. EnumValue(HKEY_PERFORMANCE_DATA, 0)
  264. except OSError as e:
  265. if e.errno in (errno.EPERM, errno.EACCES):
  266. self.skipTest("access denied to registry key "
  267. "(are you running in a non-interactive session?)")
  268. raise
  269. QueryValueEx(HKEY_PERFORMANCE_DATA, "")
  270. # Reflection requires XP x64/Vista at a minimum. XP doesn't have this stuff
  271. # or DeleteKeyEx so make sure their use raises NotImplementedError
  272. @unittest.skipUnless(WIN_VER < (5, 2), "Requires Windows XP")
  273. def test_reflection_unsupported(self):
  274. try:
  275. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  276. self.assertNotEqual(ck.handle, 0)
  277. key = OpenKey(HKEY_CURRENT_USER, test_key_name)
  278. self.assertNotEqual(key.handle, 0)
  279. with self.assertRaises(NotImplementedError):
  280. DisableReflectionKey(key)
  281. with self.assertRaises(NotImplementedError):
  282. EnableReflectionKey(key)
  283. with self.assertRaises(NotImplementedError):
  284. QueryReflectionKey(key)
  285. with self.assertRaises(NotImplementedError):
  286. DeleteKeyEx(HKEY_CURRENT_USER, test_key_name)
  287. finally:
  288. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  289. def test_setvalueex_value_range(self):
  290. # Test for Issue #14420, accept proper ranges for SetValueEx.
  291. # Py2Reg, which gets called by SetValueEx, was using PyLong_AsLong,
  292. # thus raising OverflowError. The implementation now uses
  293. # PyLong_AsUnsignedLong to match DWORD's size.
  294. try:
  295. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  296. self.assertNotEqual(ck.handle, 0)
  297. SetValueEx(ck, "test_name", None, REG_DWORD, 0x80000000)
  298. finally:
  299. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  300. def test_setvalueex_negative_one_check(self):
  301. # Test for Issue #43984, check -1 was not set by SetValueEx.
  302. # Py2Reg, which gets called by SetValueEx, wasn't checking return
  303. # value by PyLong_AsUnsignedLong, thus setting -1 as value in the registry.
  304. # The implementation now checks PyLong_AsUnsignedLong return value to assure
  305. # the value set was not -1.
  306. try:
  307. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  308. with self.assertRaises(OverflowError):
  309. SetValueEx(ck, "test_name_dword", None, REG_DWORD, -1)
  310. SetValueEx(ck, "test_name_qword", None, REG_QWORD, -1)
  311. self.assertRaises(FileNotFoundError, QueryValueEx, ck, "test_name_dword")
  312. self.assertRaises(FileNotFoundError, QueryValueEx, ck, "test_name_qword")
  313. finally:
  314. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  315. def test_queryvalueex_return_value(self):
  316. # Test for Issue #16759, return unsigned int from QueryValueEx.
  317. # Reg2Py, which gets called by QueryValueEx, was returning a value
  318. # generated by PyLong_FromLong. The implementation now uses
  319. # PyLong_FromUnsignedLong to match DWORD's size.
  320. try:
  321. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  322. self.assertNotEqual(ck.handle, 0)
  323. test_val = 0x80000000
  324. SetValueEx(ck, "test_name", None, REG_DWORD, test_val)
  325. ret_val, ret_type = QueryValueEx(ck, "test_name")
  326. self.assertEqual(ret_type, REG_DWORD)
  327. self.assertEqual(ret_val, test_val)
  328. finally:
  329. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  330. def test_setvalueex_crash_with_none_arg(self):
  331. # Test for Issue #21151, segfault when None is passed to SetValueEx
  332. try:
  333. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  334. self.assertNotEqual(ck.handle, 0)
  335. test_val = None
  336. SetValueEx(ck, "test_name", 0, REG_BINARY, test_val)
  337. ret_val, ret_type = QueryValueEx(ck, "test_name")
  338. self.assertEqual(ret_type, REG_BINARY)
  339. self.assertEqual(ret_val, test_val)
  340. finally:
  341. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  342. def test_read_string_containing_null(self):
  343. # Test for issue 25778: REG_SZ should not contain null characters
  344. try:
  345. with CreateKey(HKEY_CURRENT_USER, test_key_name) as ck:
  346. self.assertNotEqual(ck.handle, 0)
  347. test_val = "A string\x00 with a null"
  348. SetValueEx(ck, "test_name", 0, REG_SZ, test_val)
  349. ret_val, ret_type = QueryValueEx(ck, "test_name")
  350. self.assertEqual(ret_type, REG_SZ)
  351. self.assertEqual(ret_val, "A string")
  352. finally:
  353. DeleteKey(HKEY_CURRENT_USER, test_key_name)
  354. @unittest.skipUnless(REMOTE_NAME, "Skipping remote registry tests")
  355. class RemoteWinregTests(BaseWinregTests):
  356. def test_remote_registry_works(self):
  357. remote_key = ConnectRegistry(REMOTE_NAME, HKEY_CURRENT_USER)
  358. self._test_all(remote_key)
  359. @unittest.skipUnless(WIN64_MACHINE, "x64 specific registry tests")
  360. class Win64WinregTests(BaseWinregTests):
  361. def test_named_arguments(self):
  362. self._test_named_args(HKEY_CURRENT_USER, test_key_name)
  363. # Clean up and also exercise the named arguments
  364. DeleteKeyEx(key=HKEY_CURRENT_USER, sub_key=test_key_name,
  365. access=KEY_ALL_ACCESS, reserved=0)
  366. @unittest.skipIf(win32_edition() in ('WindowsCoreHeadless', 'IoTEdgeOS'), "APIs not available on WindowsCoreHeadless")
  367. def test_reflection_functions(self):
  368. # Test that we can call the query, enable, and disable functions
  369. # on a key which isn't on the reflection list with no consequences.
  370. with OpenKey(HKEY_LOCAL_MACHINE, "Software") as key:
  371. # HKLM\Software is redirected but not reflected in all OSes
  372. self.assertTrue(QueryReflectionKey(key))
  373. self.assertIsNone(EnableReflectionKey(key))
  374. self.assertIsNone(DisableReflectionKey(key))
  375. self.assertTrue(QueryReflectionKey(key))
  376. @unittest.skipUnless(HAS_REFLECTION, "OS doesn't support reflection")
  377. def test_reflection(self):
  378. # Test that we can create, open, and delete keys in the 32-bit
  379. # area. Because we are doing this in a key which gets reflected,
  380. # test the differences of 32 and 64-bit keys before and after the
  381. # reflection occurs (ie. when the created key is closed).
  382. try:
  383. with CreateKeyEx(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  384. KEY_ALL_ACCESS | KEY_WOW64_32KEY) as created_key:
  385. self.assertNotEqual(created_key.handle, 0)
  386. # The key should now be available in the 32-bit area
  387. with OpenKey(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  388. KEY_ALL_ACCESS | KEY_WOW64_32KEY) as key:
  389. self.assertNotEqual(key.handle, 0)
  390. # Write a value to what currently is only in the 32-bit area
  391. SetValueEx(created_key, "", 0, REG_SZ, "32KEY")
  392. # The key is not reflected until created_key is closed.
  393. # The 64-bit version of the key should not be available yet.
  394. open_fail = lambda: OpenKey(HKEY_CURRENT_USER,
  395. test_reflect_key_name, 0,
  396. KEY_READ | KEY_WOW64_64KEY)
  397. self.assertRaises(OSError, open_fail)
  398. # Now explicitly open the 64-bit version of the key
  399. with OpenKey(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  400. KEY_ALL_ACCESS | KEY_WOW64_64KEY) as key:
  401. self.assertNotEqual(key.handle, 0)
  402. # Make sure the original value we set is there
  403. self.assertEqual("32KEY", QueryValue(key, ""))
  404. # Set a new value, which will get reflected to 32-bit
  405. SetValueEx(key, "", 0, REG_SZ, "64KEY")
  406. # Reflection uses a "last-writer wins policy, so the value we set
  407. # on the 64-bit key should be the same on 32-bit
  408. with OpenKey(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  409. KEY_READ | KEY_WOW64_32KEY) as key:
  410. self.assertEqual("64KEY", QueryValue(key, ""))
  411. finally:
  412. DeleteKeyEx(HKEY_CURRENT_USER, test_reflect_key_name,
  413. KEY_WOW64_32KEY, 0)
  414. @unittest.skipUnless(HAS_REFLECTION, "OS doesn't support reflection")
  415. def test_disable_reflection(self):
  416. # Make use of a key which gets redirected and reflected
  417. try:
  418. with CreateKeyEx(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  419. KEY_ALL_ACCESS | KEY_WOW64_32KEY) as created_key:
  420. # QueryReflectionKey returns whether or not the key is disabled
  421. disabled = QueryReflectionKey(created_key)
  422. self.assertEqual(type(disabled), bool)
  423. # HKCU\Software\Classes is reflected by default
  424. self.assertFalse(disabled)
  425. DisableReflectionKey(created_key)
  426. self.assertTrue(QueryReflectionKey(created_key))
  427. # The key is now closed and would normally be reflected to the
  428. # 64-bit area, but let's make sure that didn't happen.
  429. open_fail = lambda: OpenKeyEx(HKEY_CURRENT_USER,
  430. test_reflect_key_name, 0,
  431. KEY_READ | KEY_WOW64_64KEY)
  432. self.assertRaises(OSError, open_fail)
  433. # Make sure the 32-bit key is actually there
  434. with OpenKeyEx(HKEY_CURRENT_USER, test_reflect_key_name, 0,
  435. KEY_READ | KEY_WOW64_32KEY) as key:
  436. self.assertNotEqual(key.handle, 0)
  437. finally:
  438. DeleteKeyEx(HKEY_CURRENT_USER, test_reflect_key_name,
  439. KEY_WOW64_32KEY, 0)
  440. def test_exception_numbers(self):
  441. with self.assertRaises(FileNotFoundError) as ctx:
  442. QueryValue(HKEY_CLASSES_ROOT, 'some_value_that_does_not_exist')
  443. if __name__ == "__main__":
  444. if not REMOTE_NAME:
  445. print("Remote registry calls can be tested using",
  446. "'test_winreg.py --remote \\\\machine_name'")
  447. unittest.main()