Christian Heimes added the comment:
PATCH:
* remove the analogy fd < 0 -> file is closed from _fileio.FileIO
* added new flag closed to _fileio.FileIO
* renamed closefd to close_fd to distinguish it from closed
* make it impossible to instantiate another stdprinter
* added repr and fileno methods to stdprinter
Guido:
Are you fine with the changes? The patch doesn't fix the problem (yet)
but it's the first step towards a solution.
----------
nosy: +gvanrossum
Added file: http://bugs.python.org/file8730/py3k_fileio_closed.patch
__________________________________
Tracker <[EMAIL PROTECTED]>
<http://bugs.python.org/issue1415>
__________________________________
Index: Modules/_fileio.c
===================================================================
--- Modules/_fileio.c (Revision 58929)
+++ Modules/_fileio.c (Arbeitskopie)
@@ -30,10 +30,11 @@
typedef struct {
PyObject_HEAD
int fd;
+ int closed : 1;
unsigned readable : 1;
unsigned writable : 1;
int seekable : 2; /* -1 means unknown */
- int closefd : 1;
+ int close_fd : 1;
PyObject *weakreflist;
} PyFileIOObject;
@@ -46,6 +47,12 @@
internal_close(PyFileIOObject *self)
{
int save_errno = 0;
+
+ if (self->closed) {
+ return save_errno;
+ }
+ self->closed = 1;
+
if (self->fd >= 0) {
int fd = self->fd;
self->fd = -1;
@@ -60,7 +67,7 @@
static PyObject *
fileio_close(PyFileIOObject *self)
{
- if (!self->closefd) {
+ if (!self->close_fd) {
if (PyErr_WarnEx(PyExc_RuntimeWarning,
"Trying to close unclosable fd!", 3) < 0) {
return NULL;
@@ -86,6 +93,7 @@
self = (PyFileIOObject *) type->tp_alloc(type, 0);
if (self != NULL) {
self->fd = -1;
+ self->closed = 0;
self->weakreflist = NULL;
}
@@ -127,7 +135,7 @@
fileio_init(PyObject *oself, PyObject *args, PyObject *kwds)
{
PyFileIOObject *self = (PyFileIOObject *) oself;
- static char *kwlist[] = {"file", "mode", "closefd", NULL};
+ static char *kwlist[] = {"file", "mode", "close_fd", NULL};
char *name = NULL;
char *mode = "r";
char *s;
@@ -138,7 +146,7 @@
int rwa = 0, plus = 0, append = 0;
int flags = 0;
int fd = -1;
- int closefd = 1;
+ int close_fd = 1;
assert(PyFileIO_Check(oself));
if (self->fd >= 0) {
@@ -148,7 +156,7 @@
}
if (PyArg_ParseTupleAndKeywords(args, kwds, "i|si:fileio",
- kwlist, &fd, &mode, &closefd)) {
+ kwlist, &fd, &mode, &close_fd)) {
if (fd < 0) {
PyErr_SetString(PyExc_ValueError,
"Negative filedescriptor");
@@ -163,7 +171,7 @@
/* On NT, so wide API available */
PyObject *po;
if (PyArg_ParseTupleAndKeywords(args, kwds, "U|si:fileio",
- kwlist, &po, &mode, &closefd)
+ kwlist, &po, &mode, &close_fd)
) {
widename = PyUnicode_AS_UNICODE(po);
} else {
@@ -178,7 +186,7 @@
if (!PyArg_ParseTupleAndKeywords(args, kwds, "et|si:fileio",
kwlist,
Py_FileSystemDefaultEncoding,
- &name, &mode, &closefd))
+ &name, &mode, &close_fd))
goto error;
}
}
@@ -247,13 +255,13 @@
if (fd >= 0) {
self->fd = fd;
- self->closefd = closefd;
+ self->close_fd = close_fd;
}
else {
- self->closefd = 1;
- if (!closefd) {
+ self->close_fd = 1;
+ if (!close_fd) {
PyErr_SetString(PyExc_ValueError,
- "Cannot use closefd=True with file name");
+ "Cannot use close_fd=True with file name");
goto error;
}
@@ -292,7 +300,7 @@
if (self->weakreflist != NULL)
PyObject_ClearWeakRefs((PyObject *) self);
- if (self->fd >= 0 && self->closefd) {
+ if (self->fd >= 0 && self->close_fd) {
errno = internal_close(self);
if (errno < 0) {
#ifdef HAVE_STRERROR
@@ -315,12 +323,39 @@
}
static PyObject *
+err_invalid(void)
+{
+#ifdef EBADF
+ errno = EBADF;
+ PyErr_SetFromErrno(PyExc_IOError);
+ return NULL;
+#else
+ PyErr_SetString(PyExc_IOError, "Bad file descriptor");
+ return NULL;
+#endif
+}
+
+static PyObject *
err_mode(char *action)
{
PyErr_Format(PyExc_ValueError, "File not open for %s", action);
return NULL;
}
+int
+err_check(PyFileIOObject *self)
+{
+ if (self->closed) {
+ err_closed();
+ return 1;
+ }
+ if (self->fd < 0) {
+ err_invalid();
+ return 1;
+ }
+ return 0;
+}
+
static PyObject *
fileio_fileno(PyFileIOObject *self)
{
@@ -332,24 +367,24 @@
static PyObject *
fileio_readable(PyFileIOObject *self)
{
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
return PyBool_FromLong((long) self->readable);
}
static PyObject *
fileio_writable(PyFileIOObject *self)
{
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
return PyBool_FromLong((long) self->writable);
}
static PyObject *
fileio_seekable(PyFileIOObject *self)
{
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (self->seekable < 0) {
int ret;
Py_BEGIN_ALLOW_THREADS
@@ -369,8 +404,8 @@
char *ptr;
Py_ssize_t n;
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (!self->readable)
return err_mode("reading");
@@ -456,8 +491,8 @@
Py_ssize_t size = -1;
PyObject *bytes;
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (!self->readable)
return err_mode("reading");
@@ -501,8 +536,8 @@
Py_ssize_t n;
char *ptr;
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (!self->writable)
return err_mode("writing");
@@ -593,8 +628,8 @@
PyObject *posobj;
int whence = 0;
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (!PyArg_ParseTuple(args, "O|i", &posobj, &whence))
return NULL;
@@ -605,8 +640,8 @@
static PyObject *
fileio_tell(PyFileIOObject *self, PyObject *args)
{
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
return portable_lseek(self->fd, NULL, 1);
}
@@ -618,11 +653,9 @@
PyObject *posobj = NULL;
Py_off_t pos;
int ret;
- int fd;
- fd = self->fd;
- if (fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
if (!self->writable)
return err_mode("writing");
@@ -630,7 +663,7 @@
return NULL;
if (posobj == Py_None || posobj == NULL) {
- posobj = portable_lseek(fd, NULL, 1);
+ posobj = portable_lseek(self->fd, NULL, 1);
if (posobj == NULL)
return NULL;
}
@@ -665,7 +698,7 @@
/* Have to move current pos to desired endpoint on Windows. */
errno = 0;
- pos2 = portable_lseek(fd, posobj, SEEK_SET);
+ pos2 = portable_lseek(self->fd, posobj, SEEK_SET);
if (pos2 == NULL) {
Py_DECREF(posobj);
Py_DECREF(oldposobj);
@@ -676,7 +709,7 @@
/* Truncate. Note that this may grow the file! */
Py_BEGIN_ALLOW_THREADS
errno = 0;
- hFile = (HANDLE)_get_osfhandle(fd);
+ hFile = (HANDLE)_get_osfhandle(self->fd);
ret = hFile == (HANDLE)-1;
if (ret == 0) {
ret = SetEndOfFile(hFile) == 0;
@@ -687,7 +720,7 @@
if (ret == 0) {
/* Move to the previous position in the file */
- pos2 = portable_lseek(fd, oldposobj, SEEK_SET);
+ pos2 = portable_lseek(self->fd, oldposobj, SEEK_SET);
if (pos2 == NULL) {
Py_DECREF(posobj);
Py_DECREF(oldposobj);
@@ -700,7 +733,7 @@
#else
Py_BEGIN_ALLOW_THREADS
errno = 0;
- ret = ftruncate(fd, pos);
+ ret = ftruncate(self->fd, pos);
Py_END_ALLOW_THREADS
#endif /* !MS_WINDOWS */
@@ -742,8 +775,9 @@
{
long res;
- if (self->fd < 0)
- return err_closed();
+ if (err_check(self))
+ return NULL;
+
Py_BEGIN_ALLOW_THREADS
res = isatty(self->fd);
Py_END_ALLOW_THREADS
@@ -850,7 +884,7 @@
static PyObject *
get_closed(PyFileIOObject *self, void *closure)
{
- return PyBool_FromLong((long)(self->fd < 0));
+ return PyBool_FromLong((long)(self->closed));
}
static PyObject *
Index: Objects/fileobject.c
===================================================================
--- Objects/fileobject.c (Revision 58929)
+++ Objects/fileobject.c (Arbeitskopie)
@@ -347,12 +347,20 @@
return (PyObject *) self;
}
+PyObject*
+fileio_init(PyObject *self, PyObject *args, PyObject *kwds)
+{
+ PyErr_SetString(PyExc_TypeError,
+ "cannot create 'stderrprinter' instances");
+ return NULL;
+}
+
PyObject *
PyFile_NewStdPrinter(int fd)
{
PyStdPrinter_Object *self;
- if ((fd != fileno(stdout) && fd != fileno(stderr)) || fd < 0) {
+ if (fd != fileno(stdout) && fd != fileno(stderr)) {
/* not enough infrastructure for PyErr_BadInternalCall() */
return NULL;
}
@@ -372,14 +380,17 @@
Py_ssize_t n;
if (self->fd < 0) {
- PyErr_SetString(PyExc_ValueError,
- "I/O operation on closed file");
- return NULL;
+ /* fd might be invalid on Windows
+ * I can't raise an exception here. It may lead to an
+ * unlimited recursion in the case stderr is invalid.
+ */
+ return PyLong_FromLong((long)-1);
}
- if (!PyArg_ParseTuple(args, "s#", &c, &n)) {
+ if (!PyArg_ParseTuple(args, "s", &c)) {
return NULL;
}
+ n = strlen(c);
Py_BEGIN_ALLOW_THREADS
errno = 0;
@@ -393,10 +404,24 @@
return NULL;
}
- return PyInt_FromSsize_t(n);
+ return PyLong_FromSsize_t(n);
}
+static PyObject *
+stdprinter_fileno(PyStdPrinter_Object *self)
+{
+ return PyInt_FromLong((long) self->fd);
+}
+
+static PyObject *
+stdprinter_repr(PyStdPrinter_Object *self)
+{
+ return PyUnicode_FromFormat("<stdprinter(fd=%d) object at 0x%x>",
+ self->fd, self);
+}
+
static PyMethodDef stdprinter_methods[] = {
+ {"fileno", (PyCFunction)stdprinter_fileno, METH_NOARGS, ""},
{"write", (PyCFunction)stdprinter_write, METH_VARARGS, ""},
{NULL, NULL} /* sentinel */
};
@@ -412,7 +437,7 @@
0, /* tp_getattr */
0, /* tp_setattr */
0, /* tp_compare */
- 0, /* tp_repr */
+ (reprfunc)stdprinter_repr, /* tp_repr */
0, /* tp_as_number */
0, /* tp_as_sequence */
0, /* tp_as_mapping */
@@ -438,7 +463,7 @@
0, /* tp_descr_get */
0, /* tp_descr_set */
0, /* tp_dictoffset */
- 0, /* tp_init */
+ fileio_init, /* tp_init */
PyType_GenericAlloc, /* tp_alloc */
stdprinter_new, /* tp_new */
PyObject_Del, /* tp_free */
_______________________________________________
Python-bugs-list mailing list
Unsubscribe:
http://mail.python.org/mailman/options/python-bugs-list/archive%40mail-archive.com