Kaydet (Commit) 9c669ccc authored tarafından Brian Curtin's avatar Brian Curtin

Fix #11583. Changed os.path.isdir to use GetFileAttributes instead of os.stat.

By changing to the Windows GetFileAttributes API in nt._isdir we can figure
out if the path is a directory without opening the file via os.stat. This has
the minor benefit of speeding up os.path.isdir by at least 2x for regular
files and 10-15x improvements were seen on symbolic links (which opened the
file multiple times during os.stat). Since os.path.isdir is used in
several places on interpreter startup, we get a minor speedup in startup time.
üst 41c1910b
...@@ -672,3 +672,16 @@ except ImportError: ...@@ -672,3 +672,16 @@ except ImportError:
def sameopenfile(f1, f2): def sameopenfile(f1, f2):
"""Test whether two file objects reference the same file""" """Test whether two file objects reference the same file"""
return _getfileinformation(f1) == _getfileinformation(f2) return _getfileinformation(f1) == _getfileinformation(f2)
try:
# The genericpath.isdir implementation uses os.stat and checks the mode
# attribute to tell whether or not the path is a directory.
# This is overkill on Windows - just pass the path to GetFileAttributes
# and check the attribute from there.
from nt import _isdir
except ImportError:
from genericpath import isdir as _isdir
def isdir(path):
return _isdir(path)
...@@ -22,6 +22,9 @@ Core and Builtins ...@@ -22,6 +22,9 @@ Core and Builtins
Library Library
------- -------
- Issue #11583: Speed up os.path.isdir on Windows by using GetFileAttributes
instead of os.stat.
- Named tuples now work correctly with vars(). - Named tuples now work correctly with vars().
- Issue #12085: Fix an attribute error in subprocess.Popen destructor if the - Issue #12085: Fix an attribute error in subprocess.Popen destructor if the
......
...@@ -2819,6 +2819,42 @@ posix__getfileinformation(PyObject *self, PyObject *args) ...@@ -2819,6 +2819,42 @@ posix__getfileinformation(PyObject *self, PyObject *args)
info.nFileIndexHigh, info.nFileIndexHigh,
info.nFileIndexLow); info.nFileIndexLow);
} }
static PyObject *
posix__isdir(PyObject *self, PyObject *args)
{
PyObject *opath;
char *path;
PyUnicodeObject *po;
DWORD attributes;
if (PyArg_ParseTuple(args, "U|:_isdir", &po)) {
Py_UNICODE *wpath = PyUnicode_AS_UNICODE(po);
attributes = GetFileAttributesW(wpath);
if (attributes == INVALID_FILE_ATTRIBUTES)
Py_RETURN_FALSE;
goto check;
}
/* Drop the argument parsing error as narrow strings
are also valid. */
PyErr_Clear();
if (!PyArg_ParseTuple(args, "O&:_isdir",
PyUnicode_FSConverter, &opath))
return NULL;
path = PyBytes_AsString(opath);
attributes = GetFileAttributesA(path);
if (attributes == INVALID_FILE_ATTRIBUTES)
Py_RETURN_FALSE;
check:
if (attributes & FILE_ATTRIBUTE_DIRECTORY)
Py_RETURN_TRUE;
else
Py_RETURN_FALSE;
}
#endif /* MS_WINDOWS */ #endif /* MS_WINDOWS */
PyDoc_STRVAR(posix_mkdir__doc__, PyDoc_STRVAR(posix_mkdir__doc__,
...@@ -8055,6 +8091,7 @@ static PyMethodDef posix_methods[] = { ...@@ -8055,6 +8091,7 @@ static PyMethodDef posix_methods[] = {
{"_getfullpathname", posix__getfullpathname, METH_VARARGS, NULL}, {"_getfullpathname", posix__getfullpathname, METH_VARARGS, NULL},
{"_getfinalpathname", posix__getfinalpathname, METH_VARARGS, NULL}, {"_getfinalpathname", posix__getfinalpathname, METH_VARARGS, NULL},
{"_getfileinformation", posix__getfileinformation, METH_VARARGS, NULL}, {"_getfileinformation", posix__getfileinformation, METH_VARARGS, NULL},
{"_isdir", posix__isdir, METH_VARARGS, NULL},
#endif #endif
#ifdef HAVE_GETLOADAVG #ifdef HAVE_GETLOADAVG
{"getloadavg", posix_getloadavg, METH_NOARGS, posix_getloadavg__doc__}, {"getloadavg", posix_getloadavg, METH_NOARGS, posix_getloadavg__doc__},
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment