Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 1 | """Test script for the dbm.open function based on testdumbdbm.py""" |
| 2 | |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 3 | import unittest |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 4 | import glob |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 5 | from test.support import import_helper |
| 6 | from test.support import os_helper |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 7 | |
R. David Murray | a21e4ca | 2009-03-31 23:16:50 +0000 | [diff] [blame] | 8 | # Skip tests if dbm module doesn't exist. |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 9 | dbm = import_helper.import_module('dbm') |
R. David Murray | a21e4ca | 2009-03-31 23:16:50 +0000 | [diff] [blame] | 10 | |
Ezio Melotti | b08495b | 2013-07-07 13:15:08 +0200 | [diff] [blame] | 11 | try: |
| 12 | from dbm import ndbm |
| 13 | except ImportError: |
| 14 | ndbm = None |
| 15 | |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 16 | _fname = os_helper.TESTFN |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 17 | |
| 18 | # |
| 19 | # Iterates over every database module supported by dbm currently available, |
| 20 | # setting dbm to use each in turn, and yielding that module |
| 21 | # |
| 22 | def dbm_iterator(): |
Georg Brandl | b17acad | 2008-05-28 08:43:17 +0000 | [diff] [blame] | 23 | for name in dbm._names: |
| 24 | try: |
| 25 | mod = __import__(name, fromlist=['open']) |
| 26 | except ImportError: |
| 27 | continue |
| 28 | dbm._modules[name] = mod |
| 29 | yield mod |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 30 | |
| 31 | # |
| 32 | # Clean up all scratch databases we might have created during testing |
| 33 | # |
| 34 | def delete_files(): |
| 35 | # we don't know the precise name the underlying database uses |
| 36 | # so we use glob to locate all names |
Serhiy Storchaka | 9355868 | 2020-06-20 11:10:31 +0300 | [diff] [blame] | 37 | for f in glob.glob(glob.escape(_fname) + "*"): |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 38 | os_helper.unlink(f) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 39 | |
| 40 | |
Ezio Melotti | f79493b | 2013-03-01 11:23:28 +0200 | [diff] [blame] | 41 | class AnyDBMTestCase: |
Serhiy Storchaka | 70af06c | 2018-04-29 15:45:03 +0300 | [diff] [blame] | 42 | _dict = {'a': b'Python:', |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 43 | 'b': b'Programming', |
| 44 | 'c': b'the', |
| 45 | 'd': b'way', |
| 46 | 'f': b'Guido', |
| 47 | 'g': b'intended', |
| 48 | } |
| 49 | |
Georg Brandl | b17acad | 2008-05-28 08:43:17 +0000 | [diff] [blame] | 50 | def init_db(self): |
| 51 | f = dbm.open(_fname, 'n') |
| 52 | for k in self._dict: |
| 53 | f[k.encode("ascii")] = self._dict[k] |
| 54 | f.close() |
| 55 | |
| 56 | def keys_helper(self, f): |
| 57 | keys = sorted(k.decode("ascii") for k in f.keys()) |
| 58 | dkeys = sorted(self._dict.keys()) |
| 59 | self.assertEqual(keys, dkeys) |
| 60 | return keys |
| 61 | |
| 62 | def test_error(self): |
Andrew Svetlov | f7a17b4 | 2012-12-25 16:47:37 +0200 | [diff] [blame] | 63 | self.assertTrue(issubclass(self.module.error, OSError)) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 64 | |
Amaury Forgeot d'Arc | b5cf301 | 2008-09-25 22:27:43 +0000 | [diff] [blame] | 65 | def test_anydbm_not_existing(self): |
| 66 | self.assertRaises(dbm.error, dbm.open, _fname) |
| 67 | |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 68 | def test_anydbm_creation(self): |
| 69 | f = dbm.open(_fname, 'c') |
| 70 | self.assertEqual(list(f.keys()), []) |
| 71 | for key in self._dict: |
| 72 | f[key.encode("ascii")] = self._dict[key] |
| 73 | self.read_helper(f) |
| 74 | f.close() |
| 75 | |
briancurtin | 525c25d | 2011-03-14 16:03:54 -0400 | [diff] [blame] | 76 | def test_anydbm_creation_n_file_exists_with_invalid_contents(self): |
Victor Stinner | bf81622 | 2011-06-30 23:25:47 +0200 | [diff] [blame] | 77 | # create an empty file |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 78 | os_helper.create_empty_file(_fname) |
Serhiy Storchaka | 6592d7f | 2018-06-05 16:03:00 +0300 | [diff] [blame] | 79 | with dbm.open(_fname, 'n') as f: |
| 80 | self.assertEqual(len(f), 0) |
briancurtin | 525c25d | 2011-03-14 16:03:54 -0400 | [diff] [blame] | 81 | |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 82 | def test_anydbm_modification(self): |
| 83 | self.init_db() |
| 84 | f = dbm.open(_fname, 'c') |
| 85 | self._dict['g'] = f[b'g'] = b"indented" |
| 86 | self.read_helper(f) |
Serhiy Storchaka | 2e38cc3 | 2018-04-29 12:38:06 +0300 | [diff] [blame] | 87 | # setdefault() works as in the dict interface |
| 88 | self.assertEqual(f.setdefault(b'xxx', b'foo'), b'foo') |
| 89 | self.assertEqual(f[b'xxx'], b'foo') |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 90 | f.close() |
| 91 | |
| 92 | def test_anydbm_read(self): |
| 93 | self.init_db() |
| 94 | f = dbm.open(_fname, 'r') |
| 95 | self.read_helper(f) |
Serhiy Storchaka | 2e38cc3 | 2018-04-29 12:38:06 +0300 | [diff] [blame] | 96 | # get() works as in the dict interface |
| 97 | self.assertEqual(f.get(b'a'), self._dict['a']) |
| 98 | self.assertEqual(f.get(b'xxx', b'foo'), b'foo') |
| 99 | self.assertIsNone(f.get(b'xxx')) |
| 100 | with self.assertRaises(KeyError): |
| 101 | f[b'xxx'] |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 102 | f.close() |
| 103 | |
| 104 | def test_anydbm_keys(self): |
| 105 | self.init_db() |
| 106 | f = dbm.open(_fname, 'r') |
| 107 | keys = self.keys_helper(f) |
| 108 | f.close() |
| 109 | |
Serhiy Storchaka | 70af06c | 2018-04-29 15:45:03 +0300 | [diff] [blame] | 110 | def test_empty_value(self): |
| 111 | if getattr(dbm._defaultmod, 'library', None) == 'Berkeley DB': |
| 112 | self.skipTest("Berkeley DB doesn't distinguish the empty value " |
| 113 | "from the absent one") |
| 114 | f = dbm.open(_fname, 'c') |
| 115 | self.assertEqual(f.keys(), []) |
| 116 | f[b'empty'] = b'' |
| 117 | self.assertEqual(f.keys(), [b'empty']) |
| 118 | self.assertIn(b'empty', f) |
| 119 | self.assertEqual(f[b'empty'], b'') |
| 120 | self.assertEqual(f.get(b'empty'), b'') |
| 121 | self.assertEqual(f.setdefault(b'empty'), b'') |
| 122 | f.close() |
| 123 | |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 124 | def test_anydbm_access(self): |
| 125 | self.init_db() |
| 126 | f = dbm.open(_fname, 'r') |
| 127 | key = "a".encode("ascii") |
Ezio Melotti | b58e0bd | 2010-01-23 15:40:09 +0000 | [diff] [blame] | 128 | self.assertIn(key, f) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 129 | assert(f[key] == b"Python:") |
| 130 | f.close() |
| 131 | |
| 132 | def read_helper(self, f): |
| 133 | keys = self.keys_helper(f) |
| 134 | for key in self._dict: |
| 135 | self.assertEqual(self._dict[key], f[key.encode("ascii")]) |
| 136 | |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 137 | def tearDown(self): |
| 138 | delete_files() |
| 139 | |
| 140 | def setUp(self): |
Georg Brandl | b17acad | 2008-05-28 08:43:17 +0000 | [diff] [blame] | 141 | dbm._defaultmod = self.module |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 142 | delete_files() |
| 143 | |
| 144 | |
| 145 | class WhichDBTestCase(unittest.TestCase): |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 146 | def test_whichdb(self): |
| 147 | for module in dbm_iterator(): |
| 148 | # Check whether whichdb correctly guesses module name |
| 149 | # for databases opened with "module" module. |
| 150 | # Try with empty files first |
| 151 | name = module.__name__ |
| 152 | if name == 'dbm.dumb': |
| 153 | continue # whichdb can't support dbm.dumb |
Senthil Kumaran | 725c2b9 | 2011-02-26 07:11:27 +0000 | [diff] [blame] | 154 | delete_files() |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 155 | f = module.open(_fname, 'c') |
| 156 | f.close() |
Ezio Melotti | b08495b | 2013-07-07 13:15:08 +0200 | [diff] [blame] | 157 | self.assertEqual(name, self.dbm.whichdb(_fname)) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 158 | # Now add a key |
| 159 | f = module.open(_fname, 'w') |
| 160 | f[b"1"] = b"1" |
| 161 | # and test that we can find it |
Benjamin Peterson | 577473f | 2010-01-19 00:09:57 +0000 | [diff] [blame] | 162 | self.assertIn(b"1", f) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 163 | # and read it |
Serhiy Storchaka | 22525de | 2018-06-19 13:31:48 +0300 | [diff] [blame] | 164 | self.assertEqual(f[b"1"], b"1") |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 165 | f.close() |
Ezio Melotti | b08495b | 2013-07-07 13:15:08 +0200 | [diff] [blame] | 166 | self.assertEqual(name, self.dbm.whichdb(_fname)) |
| 167 | |
| 168 | @unittest.skipUnless(ndbm, reason='Test requires ndbm') |
| 169 | def test_whichdb_ndbm(self): |
| 170 | # Issue 17198: check that ndbm which is referenced in whichdb is defined |
| 171 | db_file = '{}_ndbm.db'.format(_fname) |
| 172 | with open(db_file, 'w'): |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 173 | self.addCleanup(os_helper.unlink, db_file) |
Ezio Melotti | b08495b | 2013-07-07 13:15:08 +0200 | [diff] [blame] | 174 | self.assertIsNone(self.dbm.whichdb(db_file[:-3])) |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 175 | |
| 176 | def tearDown(self): |
| 177 | delete_files() |
| 178 | |
| 179 | def setUp(self): |
| 180 | delete_files() |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 181 | self.filename = os_helper.TESTFN |
Benjamin Peterson | f10a79a | 2008-10-11 00:49:57 +0000 | [diff] [blame] | 182 | self.d = dbm.open(self.filename, 'c') |
| 183 | self.d.close() |
Hai Shi | a089d21 | 2020-07-06 17:15:08 +0800 | [diff] [blame] | 184 | self.dbm = import_helper.import_fresh_module('dbm') |
Benjamin Peterson | f10a79a | 2008-10-11 00:49:57 +0000 | [diff] [blame] | 185 | |
| 186 | def test_keys(self): |
| 187 | self.d = dbm.open(self.filename, 'c') |
| 188 | self.assertEqual(self.d.keys(), []) |
| 189 | a = [(b'a', b'b'), (b'12345678910', b'019237410982340912840198242')] |
| 190 | for k, v in a: |
| 191 | self.d[k] = v |
| 192 | self.assertEqual(sorted(self.d.keys()), sorted(k for (k, v) in a)) |
| 193 | for k, v in a: |
Benjamin Peterson | 577473f | 2010-01-19 00:09:57 +0000 | [diff] [blame] | 194 | self.assertIn(k, self.d) |
Benjamin Peterson | f10a79a | 2008-10-11 00:49:57 +0000 | [diff] [blame] | 195 | self.assertEqual(self.d[k], v) |
Benjamin Peterson | 577473f | 2010-01-19 00:09:57 +0000 | [diff] [blame] | 196 | self.assertNotIn(b'xxx', self.d) |
Benjamin Peterson | 1bc15c2 | 2008-10-11 02:19:18 +0000 | [diff] [blame] | 197 | self.assertRaises(KeyError, lambda: self.d[b'xxx']) |
Benjamin Peterson | f10a79a | 2008-10-11 00:49:57 +0000 | [diff] [blame] | 198 | self.d.close() |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 199 | |
| 200 | |
Ezio Melotti | f79493b | 2013-03-01 11:23:28 +0200 | [diff] [blame] | 201 | def load_tests(loader, tests, pattern): |
| 202 | classes = [] |
Georg Brandl | b17acad | 2008-05-28 08:43:17 +0000 | [diff] [blame] | 203 | for mod in dbm_iterator(): |
Ezio Melotti | f79493b | 2013-03-01 11:23:28 +0200 | [diff] [blame] | 204 | classes.append(type("TestCase-" + mod.__name__, |
| 205 | (AnyDBMTestCase, unittest.TestCase), |
Georg Brandl | b17acad | 2008-05-28 08:43:17 +0000 | [diff] [blame] | 206 | {'module': mod})) |
Ezio Melotti | f79493b | 2013-03-01 11:23:28 +0200 | [diff] [blame] | 207 | suites = [unittest.makeSuite(c) for c in classes] |
| 208 | |
| 209 | tests.addTests(suites) |
| 210 | return tests |
Georg Brandl | 7f986ac | 2008-05-26 10:30:20 +0000 | [diff] [blame] | 211 | |
| 212 | if __name__ == "__main__": |
Ezio Melotti | f79493b | 2013-03-01 11:23:28 +0200 | [diff] [blame] | 213 | unittest.main() |