1 """File-like objects that read from or write to a string buffer.
3 This implements (nearly) all stdio methods.
5 f = StringIO() # ready for writing
6 f = StringIO(buf) # ready for reading
7 f.close() # explicitly release resources held
8 flag = f.isatty() # always false
9 pos = f.tell() # get current position
10 f.seek(pos) # set current position
11 f.seek(pos, mode) # mode 0: absolute; 1: relative; 2: relative to EOF
12 buf = f.read() # read until EOF
13 buf = f.read(n) # read up to n bytes
14 buf = f.readline() # read until end of line ('\n') or EOF
15 list = f.readlines()# list of f.readline() results until EOF
16 f.truncate([size]) # truncate file at to at most size (default: current pos)
17 f.write(buf) # write at current position
18 f.writelines(list) # for line in list: f.write(line)
19 f.getvalue() # return whole file's contents as a string
22 - Using a real file is often faster (but less convenient).
23 - There's also a much faster implementation in C, called cStringIO, but
24 it's not subclassable.
25 - fileno() is left unimplemented so that code which uses it triggers
27 - Seeking far beyond EOF and then writing will insert real null
28 bytes that occupy space in the buffer.
29 - There's a simple test set (see end of this file).
33 from errno
import EINVAL
37 __all__
= ["StringIO"]
40 def __init__(self
, buf
= ''):
41 # Force self.buf to be a string
50 return iter(self
.readline
, '')
55 del self
.buf
, self
.pos
59 raise ValueError, "I/O operation on closed file"
62 def seek(self
, pos
, mode
= 0):
64 raise ValueError, "I/O operation on closed file"
66 self
.buf
+= ''.join(self
.buflist
)
72 self
.pos
= max(0, pos
)
76 raise ValueError, "I/O operation on closed file"
79 def read(self
, n
= -1):
81 raise ValueError, "I/O operation on closed file"
83 self
.buf
+= ''.join(self
.buflist
)
88 newpos
= min(self
.pos
+n
, self
.len)
89 r
= self
.buf
[self
.pos
:newpos
]
93 def readline(self
, length
=None):
95 raise ValueError, "I/O operation on closed file"
97 self
.buf
+= ''.join(self
.buflist
)
99 i
= self
.buf
.find('\n', self
.pos
)
104 if length
is not None:
105 if self
.pos
+ length
< newpos
:
106 newpos
= self
.pos
+ length
107 r
= self
.buf
[self
.pos
:newpos
]
111 def readlines(self
, sizehint
= 0):
114 line
= self
.readline()
118 if 0 < sizehint
<= total
:
120 line
= self
.readline()
123 def truncate(self
, size
=None):
125 raise ValueError, "I/O operation on closed file"
129 raise IOError(EINVAL
, "Negative size not allowed")
130 elif size
< self
.pos
:
132 self
.buf
= self
.getvalue()[:size
]
136 raise ValueError, "I/O operation on closed file"
138 # Force s to be a string
140 if self
.pos
> self
.len:
141 self
.buflist
.append('\0'*(self
.pos
- self
.len))
143 newpos
= self
.pos
+ len(s
)
144 if self
.pos
< self
.len:
146 self
.buf
+= ''.join(self
.buflist
)
148 self
.buflist
= [self
.buf
[:self
.pos
], s
, self
.buf
[newpos
:]]
150 if newpos
> self
.len:
153 self
.buflist
.append(s
)
157 def writelines(self
, list):
158 self
.write(''.join(list))
162 raise ValueError, "I/O operation on closed file"
166 self
.buf
+= ''.join(self
.buflist
)
171 # A little test suite
179 lines
= open(file, 'r').readlines()
180 text
= open(file, 'r').read()
182 for line
in lines
[:-2]:
184 f
.writelines(lines
[-2:])
185 if f
.getvalue() != text
:
186 raise RuntimeError, 'write failed'
188 print 'File length =', length
189 f
.seek(len(lines
[0]))
192 print 'First line =', `f
.readline()`
195 print 'Second line =', `line`
196 f
.seek(-len(line
), 1)
197 line2
= f
.read(len(line
))
199 raise RuntimeError, 'bad result after seek back'
200 f
.seek(len(line2
), 1)
203 f
.seek(f
.tell() - len(line
))
206 raise RuntimeError, 'bad result after seek back from EOF'
207 print 'Read', len(list), 'more lines'
208 print 'File length =', f
.tell()
209 if f
.tell() != length
:
210 raise RuntimeError, 'bad length'
213 if __name__
== '__main__':