PageRenderTime 47ms CodeModel.GetById 18ms RepoModel.GetById 1ms app.codeStats 0ms

/Lib/test/test_dbm_dumb.py

https://gitlab.com/unofficial-mirrors/cpython
Python | 299 lines | 268 code | 25 blank | 6 comment | 32 complexity | fa18ef440331a8830c10c4a0c37daa3b MD5 | raw file
  1. """Test script for the dumbdbm module
  2. Original by Roger E. Masse
  3. """
  4. import io
  5. import operator
  6. import os
  7. import stat
  8. import unittest
  9. import dbm.dumb as dumbdbm
  10. from test import support
  11. from functools import partial
  12. _fname = support.TESTFN
  13. def _delete_files():
  14. for ext in [".dir", ".dat", ".bak"]:
  15. try:
  16. os.unlink(_fname + ext)
  17. except OSError:
  18. pass
  19. class DumbDBMTestCase(unittest.TestCase):
  20. _dict = {b'0': b'',
  21. b'a': b'Python:',
  22. b'b': b'Programming',
  23. b'c': b'the',
  24. b'd': b'way',
  25. b'f': b'Guido',
  26. b'g': b'intended',
  27. '\u00fc'.encode('utf-8') : b'!',
  28. }
  29. def test_dumbdbm_creation(self):
  30. f = dumbdbm.open(_fname, 'c')
  31. self.assertEqual(list(f.keys()), [])
  32. for key in self._dict:
  33. f[key] = self._dict[key]
  34. self.read_helper(f)
  35. f.close()
  36. @unittest.skipUnless(hasattr(os, 'umask'), 'test needs os.umask()')
  37. @unittest.skipUnless(hasattr(os, 'chmod'), 'test needs os.chmod()')
  38. def test_dumbdbm_creation_mode(self):
  39. try:
  40. old_umask = os.umask(0o002)
  41. f = dumbdbm.open(_fname, 'c', 0o637)
  42. f.close()
  43. finally:
  44. os.umask(old_umask)
  45. expected_mode = 0o635
  46. if os.name != 'posix':
  47. # Windows only supports setting the read-only attribute.
  48. # This shouldn't fail, but doesn't work like Unix either.
  49. expected_mode = 0o666
  50. import stat
  51. st = os.stat(_fname + '.dat')
  52. self.assertEqual(stat.S_IMODE(st.st_mode), expected_mode)
  53. st = os.stat(_fname + '.dir')
  54. self.assertEqual(stat.S_IMODE(st.st_mode), expected_mode)
  55. def test_close_twice(self):
  56. f = dumbdbm.open(_fname)
  57. f[b'a'] = b'b'
  58. self.assertEqual(f[b'a'], b'b')
  59. f.close()
  60. f.close()
  61. def test_dumbdbm_modification(self):
  62. self.init_db()
  63. f = dumbdbm.open(_fname, 'w')
  64. self._dict[b'g'] = f[b'g'] = b"indented"
  65. self.read_helper(f)
  66. f.close()
  67. def test_dumbdbm_read(self):
  68. self.init_db()
  69. f = dumbdbm.open(_fname, 'r')
  70. self.read_helper(f)
  71. with self.assertWarnsRegex(DeprecationWarning,
  72. 'The database is opened for reading only'):
  73. f[b'g'] = b'x'
  74. with self.assertWarnsRegex(DeprecationWarning,
  75. 'The database is opened for reading only'):
  76. del f[b'a']
  77. f.close()
  78. def test_dumbdbm_keys(self):
  79. self.init_db()
  80. f = dumbdbm.open(_fname)
  81. keys = self.keys_helper(f)
  82. f.close()
  83. def test_write_contains(self):
  84. f = dumbdbm.open(_fname)
  85. f[b'1'] = b'hello'
  86. self.assertIn(b'1', f)
  87. f.close()
  88. def test_write_write_read(self):
  89. # test for bug #482460
  90. f = dumbdbm.open(_fname)
  91. f[b'1'] = b'hello'
  92. f[b'1'] = b'hello2'
  93. f.close()
  94. f = dumbdbm.open(_fname)
  95. self.assertEqual(f[b'1'], b'hello2')
  96. f.close()
  97. def test_str_read(self):
  98. self.init_db()
  99. f = dumbdbm.open(_fname, 'r')
  100. self.assertEqual(f['\u00fc'], self._dict['\u00fc'.encode('utf-8')])
  101. def test_str_write_contains(self):
  102. self.init_db()
  103. f = dumbdbm.open(_fname)
  104. f['\u00fc'] = b'!'
  105. f['1'] = 'a'
  106. f.close()
  107. f = dumbdbm.open(_fname, 'r')
  108. self.assertIn('\u00fc', f)
  109. self.assertEqual(f['\u00fc'.encode('utf-8')],
  110. self._dict['\u00fc'.encode('utf-8')])
  111. self.assertEqual(f[b'1'], b'a')
  112. def test_line_endings(self):
  113. # test for bug #1172763: dumbdbm would die if the line endings
  114. # weren't what was expected.
  115. f = dumbdbm.open(_fname)
  116. f[b'1'] = b'hello'
  117. f[b'2'] = b'hello2'
  118. f.close()
  119. # Mangle the file by changing the line separator to Windows or Unix
  120. with io.open(_fname + '.dir', 'rb') as file:
  121. data = file.read()
  122. if os.linesep == '\n':
  123. data = data.replace(b'\n', b'\r\n')
  124. else:
  125. data = data.replace(b'\r\n', b'\n')
  126. with io.open(_fname + '.dir', 'wb') as file:
  127. file.write(data)
  128. f = dumbdbm.open(_fname)
  129. self.assertEqual(f[b'1'], b'hello')
  130. self.assertEqual(f[b'2'], b'hello2')
  131. def read_helper(self, f):
  132. keys = self.keys_helper(f)
  133. for key in self._dict:
  134. self.assertEqual(self._dict[key], f[key])
  135. def init_db(self):
  136. f = dumbdbm.open(_fname, 'n')
  137. for k in self._dict:
  138. f[k] = self._dict[k]
  139. f.close()
  140. def keys_helper(self, f):
  141. keys = sorted(f.keys())
  142. dkeys = sorted(self._dict.keys())
  143. self.assertEqual(keys, dkeys)
  144. return keys
  145. # Perform randomized operations. This doesn't make assumptions about
  146. # what *might* fail.
  147. def test_random(self):
  148. import random
  149. d = {} # mirror the database
  150. for dummy in range(5):
  151. f = dumbdbm.open(_fname)
  152. for dummy in range(100):
  153. k = random.choice('abcdefghijklm')
  154. if random.random() < 0.2:
  155. if k in d:
  156. del d[k]
  157. del f[k]
  158. else:
  159. v = random.choice((b'a', b'b', b'c')) * random.randrange(10000)
  160. d[k] = v
  161. f[k] = v
  162. self.assertEqual(f[k], v)
  163. f.close()
  164. f = dumbdbm.open(_fname)
  165. expected = sorted((k.encode("latin-1"), v) for k, v in d.items())
  166. got = sorted(f.items())
  167. self.assertEqual(expected, got)
  168. f.close()
  169. def test_context_manager(self):
  170. with dumbdbm.open(_fname, 'c') as db:
  171. db["dumbdbm context manager"] = "context manager"
  172. with dumbdbm.open(_fname, 'r') as db:
  173. self.assertEqual(list(db.keys()), [b"dumbdbm context manager"])
  174. with self.assertRaises(dumbdbm.error):
  175. db.keys()
  176. def test_check_closed(self):
  177. f = dumbdbm.open(_fname, 'c')
  178. f.close()
  179. for meth in (partial(operator.delitem, f),
  180. partial(operator.setitem, f, 'b'),
  181. partial(operator.getitem, f),
  182. partial(operator.contains, f)):
  183. with self.assertRaises(dumbdbm.error) as cm:
  184. meth('test')
  185. self.assertEqual(str(cm.exception),
  186. "DBM object has already been closed")
  187. for meth in (operator.methodcaller('keys'),
  188. operator.methodcaller('iterkeys'),
  189. operator.methodcaller('items'),
  190. len):
  191. with self.assertRaises(dumbdbm.error) as cm:
  192. meth(f)
  193. self.assertEqual(str(cm.exception),
  194. "DBM object has already been closed")
  195. def test_create_new(self):
  196. with dumbdbm.open(_fname, 'n') as f:
  197. for k in self._dict:
  198. f[k] = self._dict[k]
  199. with dumbdbm.open(_fname, 'n') as f:
  200. self.assertEqual(f.keys(), [])
  201. def test_eval(self):
  202. with open(_fname + '.dir', 'w') as stream:
  203. stream.write("str(print('Hacked!')), 0\n")
  204. with support.captured_stdout() as stdout:
  205. with self.assertRaises(ValueError):
  206. with dumbdbm.open(_fname) as f:
  207. pass
  208. self.assertEqual(stdout.getvalue(), '')
  209. def test_warn_on_ignored_flags(self):
  210. for value in ('r', 'w'):
  211. _delete_files()
  212. with self.assertWarnsRegex(DeprecationWarning,
  213. "The database file is missing, the "
  214. "semantics of the 'c' flag will "
  215. "be used."):
  216. f = dumbdbm.open(_fname, value)
  217. f.close()
  218. def test_missing_index(self):
  219. with dumbdbm.open(_fname, 'n') as f:
  220. pass
  221. os.unlink(_fname + '.dir')
  222. for value in ('r', 'w'):
  223. with self.assertWarnsRegex(DeprecationWarning,
  224. "The index file is missing, the "
  225. "semantics of the 'c' flag will "
  226. "be used."):
  227. f = dumbdbm.open(_fname, value)
  228. f.close()
  229. self.assertEqual(os.path.exists(_fname + '.dir'), value == 'w')
  230. self.assertFalse(os.path.exists(_fname + '.bak'))
  231. def test_invalid_flag(self):
  232. for flag in ('x', 'rf', None):
  233. with self.assertWarnsRegex(DeprecationWarning,
  234. "Flag must be one of "
  235. "'r', 'w', 'c', or 'n'"):
  236. f = dumbdbm.open(_fname, flag)
  237. f.close()
  238. @unittest.skipUnless(hasattr(os, 'chmod'), 'test needs os.chmod()')
  239. def test_readonly_files(self):
  240. with support.temp_dir() as dir:
  241. fname = os.path.join(dir, 'db')
  242. with dumbdbm.open(fname, 'n') as f:
  243. self.assertEqual(list(f.keys()), [])
  244. for key in self._dict:
  245. f[key] = self._dict[key]
  246. os.chmod(fname + ".dir", stat.S_IRUSR)
  247. os.chmod(fname + ".dat", stat.S_IRUSR)
  248. os.chmod(dir, stat.S_IRUSR|stat.S_IXUSR)
  249. with dumbdbm.open(fname, 'r') as f:
  250. self.assertEqual(sorted(f.keys()), sorted(self._dict))
  251. f.close() # don't write
  252. def tearDown(self):
  253. _delete_files()
  254. def setUp(self):
  255. _delete_files()
  256. if __name__ == "__main__":
  257. unittest.main()