blob: 975a2d9c0a48a5bafe7cf9df7522754472b75c5f [file] [log] [blame]
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +00001
2"""
Barry Warsaw9a0d7792002-12-30 20:53:52 +00003File-like objects that read from or write to a bsddb record.
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +00004
5This implements (nearly) all stdio methods.
6
7f = DBRecIO(db, key, txn=None)
8f.close() # explicitly release resources held
9flag = f.isatty() # always false
10pos = f.tell() # get current position
11f.seek(pos) # set current position
12f.seek(pos, mode) # mode 0: absolute; 1: relative; 2: relative to EOF
13buf = f.read() # read until EOF
14buf = f.read(n) # read up to n bytes
15f.truncate([size]) # truncate file at to at most size (default: current pos)
16f.write(buf) # write at current position
17f.writelines(list) # for line in list: f.write(line)
18
19Notes:
20- fileno() is left unimplemented so that code which uses it triggers
21 an exception early.
22- There's a simple test set (see end of this file) - not yet updated
23 for DBRecIO.
24- readline() is not implemented yet.
25
26
27From:
28 Itamar Shtull-Trauring <itamar@maxnm.com>
29"""
30
31import errno
32import string
33
34class DBRecIO:
35 def __init__(self, db, key, txn=None):
36 self.db = db
37 self.key = key
38 self.txn = txn
39 self.len = None
40 self.pos = 0
41 self.closed = 0
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +000042
43 def close(self):
44 if not self.closed:
45 self.closed = 1
46 del self.db, self.txn
47
48 def isatty(self):
49 if self.closed:
50 raise ValueError, "I/O operation on closed file"
51 return 0
52
53 def seek(self, pos, mode = 0):
54 if self.closed:
55 raise ValueError, "I/O operation on closed file"
56 if mode == 1:
57 pos = pos + self.pos
58 elif mode == 2:
59 pos = pos + self.len
60 self.pos = max(0, pos)
61
62 def tell(self):
63 if self.closed:
64 raise ValueError, "I/O operation on closed file"
65 return self.pos
66
67 def read(self, n = -1):
68 if self.closed:
69 raise ValueError, "I/O operation on closed file"
70 if n < 0:
71 newpos = self.len
72 else:
73 newpos = min(self.pos+n, self.len)
74
75 dlen = newpos - self.pos
76
Thomas Wouters0e3f5912006-08-11 14:57:12 +000077 r = self.db.get(self.key, txn=self.txn, dlen=dlen, doff=self.pos)
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +000078 self.pos = newpos
79 return r
80
81 __fixme = """
82 def readline(self, length=None):
83 if self.closed:
84 raise ValueError, "I/O operation on closed file"
85 if self.buflist:
86 self.buf = self.buf + string.joinfields(self.buflist, '')
87 self.buflist = []
88 i = string.find(self.buf, '\n', self.pos)
89 if i < 0:
90 newpos = self.len
91 else:
92 newpos = i+1
93 if length is not None:
94 if self.pos + length < newpos:
95 newpos = self.pos + length
96 r = self.buf[self.pos:newpos]
97 self.pos = newpos
98 return r
99
100 def readlines(self, sizehint = 0):
101 total = 0
102 lines = []
103 line = self.readline()
104 while line:
105 lines.append(line)
106 total += len(line)
107 if 0 < sizehint <= total:
108 break
109 line = self.readline()
110 return lines
111 """
112
113 def truncate(self, size=None):
114 if self.closed:
115 raise ValueError, "I/O operation on closed file"
116 if size is None:
117 size = self.pos
118 elif size < 0:
119 raise IOError(errno.EINVAL,
120 "Negative size not allowed")
121 elif size < self.pos:
122 self.pos = size
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000123 self.db.put(self.key, "", txn=self.txn, dlen=self.len-size, doff=size)
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +0000124
125 def write(self, s):
126 if self.closed:
127 raise ValueError, "I/O operation on closed file"
128 if not s: return
129 if self.pos > self.len:
130 self.buflist.append('\0'*(self.pos - self.len))
131 self.len = self.pos
132 newpos = self.pos + len(s)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000133 self.db.put(self.key, s, txn=self.txn, dlen=len(s), doff=self.pos)
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +0000134 self.pos = newpos
135
136 def writelines(self, list):
137 self.write(string.joinfields(list, ''))
138
139 def flush(self):
140 if self.closed:
141 raise ValueError, "I/O operation on closed file"
142
143
144"""
145# A little test suite
146
147def _test():
148 import sys
149 if sys.argv[1:]:
150 file = sys.argv[1]
151 else:
152 file = '/etc/passwd'
153 lines = open(file, 'r').readlines()
154 text = open(file, 'r').read()
155 f = StringIO()
156 for line in lines[:-2]:
157 f.write(line)
158 f.writelines(lines[-2:])
159 if f.getvalue() != text:
160 raise RuntimeError, 'write failed'
161 length = f.tell()
162 print 'File length =', length
163 f.seek(len(lines[0]))
164 f.write(lines[1])
165 f.seek(0)
Walter Dörwald70a6b492004-02-12 17:35:32 +0000166 print 'First line =', repr(f.readline())
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +0000167 here = f.tell()
168 line = f.readline()
Walter Dörwald70a6b492004-02-12 17:35:32 +0000169 print 'Second line =', repr(line)
Martin v. Löwis6aa4a1f2002-11-19 08:09:52 +0000170 f.seek(-len(line), 1)
171 line2 = f.read(len(line))
172 if line != line2:
173 raise RuntimeError, 'bad result after seek back'
174 f.seek(len(line2), 1)
175 list = f.readlines()
176 line = list[-1]
177 f.seek(f.tell() - len(line))
178 line2 = f.read()
179 if line != line2:
180 raise RuntimeError, 'bad result after seek back from EOF'
181 print 'Read', len(list), 'more lines'
182 print 'File length =', f.tell()
183 if f.tell() != length:
184 raise RuntimeError, 'bad length'
185 f.close()
186
187if __name__ == '__main__':
188 _test()
189"""