test_dumbdbm.py 5.15 KB
Newer Older
1 2 3 4 5
#! /usr/bin/env python
"""Test script for the dumbdbm module
   Original by Roger E. Masse
"""

6
import io
7
import os
8 9
import unittest
import dumbdbm
10
from test import test_support
11

12
_fname = test_support.TESTFN
13 14 15 16 17 18 19 20

def _delete_files():
    for ext in [".dir", ".dat", ".bak"]:
        try:
            os.unlink(_fname + ext)
        except OSError:
            pass

21
class DumbDBMTestCase(unittest.TestCase):
22 23 24 25 26 27 28
    _dict = {'0': b'',
             'a': b'Python:',
             'b': b'Programming',
             'c': b'the',
             'd': b'way',
             'f': b'Guido',
             'g': b'intended',
29 30 31 32
             }

    def __init__(self, *args):
        unittest.TestCase.__init__(self, *args)
Tim Peters's avatar
Tim Peters committed
33

34
    def test_dumbdbm_creation(self):
35
        f = dumbdbm.open(_fname, 'c')
36
        self.assertEqual(list(f.keys()), [])
37
        for key in self._dict:
38
            f[key.encode("ascii")] = self._dict[key]
39 40 41
        self.read_helper(f)
        f.close()

42 43 44 45 46 47
    def test_dumbdbm_creation_mode(self):
        # On platforms without chmod, don't do anything.
        if not (hasattr(os, 'chmod') and hasattr(os, 'umask')):
            return

        try:
48 49
            old_umask = os.umask(0o002)
            f = dumbdbm.open(_fname, 'c', 0o637)
50 51 52
            f.close()
        finally:
            os.umask(old_umask)
53

54
        expected_mode = 0o635
55 56 57
        if os.name != 'posix':
            # Windows only supports setting the read-only attribute.
            # This shouldn't fail, but doesn't work like Unix either.
58
            expected_mode = 0o666
59

60 61
        import stat
        st = os.stat(_fname + '.dat')
62
        self.assertEqual(stat.S_IMODE(st.st_mode), expected_mode)
63
        st = os.stat(_fname + '.dir')
64
        self.assertEqual(stat.S_IMODE(st.st_mode), expected_mode)
65

66 67
    def test_close_twice(self):
        f = dumbdbm.open(_fname)
68 69
        f[b'a'] = b'b'
        self.assertEqual(f[b'a'], b'b')
70 71 72
        f.close()
        f.close()

73
    def test_dumbdbm_modification(self):
74
        self.init_db()
75
        f = dumbdbm.open(_fname, 'w')
76
        self._dict['g'] = f[b'g'] = b"indented"
77 78 79 80
        self.read_helper(f)
        f.close()

    def test_dumbdbm_read(self):
81
        self.init_db()
82
        f = dumbdbm.open(_fname, 'r')
83 84 85 86
        self.read_helper(f)
        f.close()

    def test_dumbdbm_keys(self):
87
        self.init_db()
88
        f = dumbdbm.open(_fname)
89 90 91
        keys = self.keys_helper(f)
        f.close()

92 93 94
    def test_write_contains(self):
        f = dumbdbm.open(_fname)
        f[b'1'] = b'hello'
95
        self.assertTrue(b'1' in f)
96 97
        f.close()

98 99 100
    def test_write_write_read(self):
        # test for bug #482460
        f = dumbdbm.open(_fname)
101 102
        f[b'1'] = b'hello'
        f[b'1'] = b'hello2'
103 104
        f.close()
        f = dumbdbm.open(_fname)
105
        self.assertEqual(f[b'1'], b'hello2')
106 107
        f.close()

108 109 110 111
    def test_line_endings(self):
        # test for bug #1172763: dumbdbm would die if the line endings
        # weren't what was expected.
        f = dumbdbm.open(_fname)
112 113
        f[b'1'] = b'hello'
        f[b'2'] = b'hello2'
114 115
        f.close()

Christian Heimes's avatar
Christian Heimes committed
116
        # Mangle the file by changing the line separator to Windows or Unix
117
        data = io.open(_fname + '.dir', 'rb').read()
118
        if os.linesep == '\n':
Christian Heimes's avatar
Christian Heimes committed
119 120 121
            data = data.replace(b'\n', b'\r\n')
        else:
            data = data.replace(b'\r\n', b'\n')
122
        io.open(_fname + '.dir', 'wb').write(data)
123

124
        f = dumbdbm.open(_fname)
125 126
        self.assertEqual(f[b'1'], b'hello')
        self.assertEqual(f[b'2'], b'hello2')
127 128


129 130 131
    def read_helper(self, f):
        keys = self.keys_helper(f)
        for key in self._dict:
132
            self.assertEqual(self._dict[key], f[key.encode("ascii")])
Tim Peters's avatar
Tim Peters committed
133

134 135 136
    def init_db(self):
        f = dumbdbm.open(_fname, 'w')
        for k in self._dict:
137
            f[k.encode("ascii")] = self._dict[k]
138 139
        f.close()

140
    def keys_helper(self, f):
141
        keys = sorted(k.decode("ascii") for k in f.keys())
142
        dkeys = sorted(self._dict.keys())
143
        self.assertEqual(keys, dkeys)
144 145
        return keys

146 147 148 149 150 151 152 153 154 155 156 157
    # Perform randomized operations.  This doesn't make assumptions about
    # what *might* fail.
    def test_random(self):
        import random
        d = {}  # mirror the database
        for dummy in range(5):
            f = dumbdbm.open(_fname)
            for dummy in range(100):
                k = random.choice('abcdefghijklm')
                if random.random() < 0.2:
                    if k in d:
                        del d[k]
158
                        del f[k.encode("ascii")]
159
                else:
160
                    v = random.choice((b'a', b'b', b'c')) * random.randrange(10000)
161
                    d[k] = v
162
                    f[k.encode("ascii")] = v
163
                    self.assertEqual(f[k.encode("ascii")], v)
164 165 166
            f.close()

            f = dumbdbm.open(_fname)
167
            expected = sorted((k.encode("latin-1"), v) for k, v in d.items())
168
            got = sorted(f.items())
169 170 171
            self.assertEqual(expected, got)
            f.close()

172 173 174 175 176 177
    def tearDown(self):
        _delete_files()

    def setUp(self):
        _delete_files()

178
def test_main():
179 180 181 182
    try:
        test_support.run_unittest(DumbDBMTestCase)
    finally:
        _delete_files()
183 184 185

if __name__ == "__main__":
    test_main()