osutil.py
304 lines
| 9.2 KiB
| text/x-python
|
PythonLexer
Martin Geisler
|
r8232 | # osutil.py - pure Python version of osutil.c | ||
# | ||||
# Copyright 2009 Matt Mackall <mpm@selenic.com> and others | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Martin Geisler
|
r8232 | |||
Rodrigo Damazio Bovendorp
|
r42724 | from __future__ import absolute_import, division | ||
Gregory Szorc
|
r27338 | |||
Yuya Nishihara
|
r27474 | import ctypes | ||
import ctypes.util | ||||
Martin Geisler
|
r7704 | import os | ||
Yuya Nishihara
|
r27474 | import socket | ||
Benoit Boissinot
|
r10651 | import stat as statmod | ||
Martin Geisler
|
r7704 | |||
Gregory Szorc
|
r43359 | from ..pycompat import getattr | ||
Yuya Nishihara
|
r32367 | from .. import ( | ||
Matt Harbison
|
r39680 | encoding, | ||
Pulkit Goyal
|
r30304 | pycompat, | ||
) | ||||
Augie Fackler
|
r43346 | |||
Martin Geisler
|
r7704 | def _mode_to_kind(mode): | ||
Benoit Boissinot
|
r10651 | if statmod.S_ISREG(mode): | ||
return statmod.S_IFREG | ||||
if statmod.S_ISDIR(mode): | ||||
return statmod.S_IFDIR | ||||
if statmod.S_ISLNK(mode): | ||||
return statmod.S_IFLNK | ||||
if statmod.S_ISBLK(mode): | ||||
return statmod.S_IFBLK | ||||
if statmod.S_ISCHR(mode): | ||||
return statmod.S_IFCHR | ||||
if statmod.S_ISFIFO(mode): | ||||
return statmod.S_IFIFO | ||||
if statmod.S_ISSOCK(mode): | ||||
return statmod.S_IFSOCK | ||||
Martin Geisler
|
r7704 | return mode | ||
Augie Fackler
|
r43346 | |||
Yuya Nishihara
|
r32512 | def listdir(path, stat=False, skip=None): | ||
Martin Geisler
|
r7704 | '''listdir(path, stat=False) -> list_of_tuples | ||
Return a sorted list containing information about the entries | ||||
in the directory. | ||||
If stat is True, each element is a 3-tuple: | ||||
(name, type, stat object) | ||||
Otherwise, each element is a 2-tuple: | ||||
(name, type) | ||||
''' | ||||
result = [] | ||||
prefix = path | ||||
Pulkit Goyal
|
r30304 | if not prefix.endswith(pycompat.ossep): | ||
prefix += pycompat.ossep | ||||
Martin Geisler
|
r7704 | names = os.listdir(path) | ||
names.sort() | ||||
for fn in names: | ||||
st = os.lstat(prefix + fn) | ||||
Benoit Boissinot
|
r10651 | if fn == skip and statmod.S_ISDIR(st.st_mode): | ||
Martin Geisler
|
r7704 | return [] | ||
if stat: | ||||
result.append((fn, _mode_to_kind(st.st_mode), st)) | ||||
else: | ||||
result.append((fn, _mode_to_kind(st.st_mode))) | ||||
return result | ||||
Sune Foldager
|
r8421 | |||
Augie Fackler
|
r43346 | |||
Jun Wu
|
r34646 | if not pycompat.iswindows: | ||
Adrian Buehlmann
|
r14413 | posixfile = open | ||
Yuya Nishihara
|
r27474 | |||
_SCM_RIGHTS = 0x01 | ||||
_socklen_t = ctypes.c_uint | ||||
Augie Fackler
|
r43347 | if pycompat.sysplatform.startswith(b'linux'): | ||
Yuya Nishihara
|
r27474 | # socket.h says "the type should be socklen_t but the definition of | ||
# the kernel is incompatible with this." | ||||
_cmsg_len_t = ctypes.c_size_t | ||||
_msg_controllen_t = ctypes.c_size_t | ||||
_msg_iovlen_t = ctypes.c_size_t | ||||
else: | ||||
_cmsg_len_t = _socklen_t | ||||
_msg_controllen_t = _socklen_t | ||||
_msg_iovlen_t = ctypes.c_int | ||||
class _iovec(ctypes.Structure): | ||||
_fields_ = [ | ||||
Pulkit Goyal
|
r29698 | (u'iov_base', ctypes.c_void_p), | ||
(u'iov_len', ctypes.c_size_t), | ||||
Yuya Nishihara
|
r27474 | ] | ||
class _msghdr(ctypes.Structure): | ||||
_fields_ = [ | ||||
Pulkit Goyal
|
r29698 | (u'msg_name', ctypes.c_void_p), | ||
(u'msg_namelen', _socklen_t), | ||||
(u'msg_iov', ctypes.POINTER(_iovec)), | ||||
(u'msg_iovlen', _msg_iovlen_t), | ||||
(u'msg_control', ctypes.c_void_p), | ||||
(u'msg_controllen', _msg_controllen_t), | ||||
(u'msg_flags', ctypes.c_int), | ||||
Yuya Nishihara
|
r27474 | ] | ||
class _cmsghdr(ctypes.Structure): | ||||
_fields_ = [ | ||||
Pulkit Goyal
|
r29698 | (u'cmsg_len', _cmsg_len_t), | ||
(u'cmsg_level', ctypes.c_int), | ||||
(u'cmsg_type', ctypes.c_int), | ||||
(u'cmsg_data', ctypes.c_ubyte * 0), | ||||
Yuya Nishihara
|
r27474 | ] | ||
Pulkit Goyal
|
r29698 | _libc = ctypes.CDLL(ctypes.util.find_library(u'c'), use_errno=True) | ||
Yuya Nishihara
|
r27971 | _recvmsg = getattr(_libc, 'recvmsg', None) | ||
if _recvmsg: | ||||
_recvmsg.restype = getattr(ctypes, 'c_ssize_t', ctypes.c_long) | ||||
Augie Fackler
|
r43346 | _recvmsg.argtypes = ( | ||
ctypes.c_int, | ||||
ctypes.POINTER(_msghdr), | ||||
ctypes.c_int, | ||||
) | ||||
Yuya Nishihara
|
r27971 | else: | ||
# recvmsg isn't always provided by libc; such systems are unsupported | ||||
def _recvmsg(sockfd, msg, flags): | ||||
Augie Fackler
|
r43347 | raise NotImplementedError(b'unsupported platform') | ||
Yuya Nishihara
|
r27474 | |||
def _CMSG_FIRSTHDR(msgh): | ||||
if msgh.msg_controllen < ctypes.sizeof(_cmsghdr): | ||||
return | ||||
cmsgptr = ctypes.cast(msgh.msg_control, ctypes.POINTER(_cmsghdr)) | ||||
return cmsgptr.contents | ||||
# The pure version is less portable than the native version because the | ||||
# handling of socket ancillary data heavily depends on C preprocessor. | ||||
# Also, some length fields are wrongly typed in Linux kernel. | ||||
def recvfds(sockfd): | ||||
"""receive list of file descriptors via socket""" | ||||
dummy = (ctypes.c_ubyte * 1)() | ||||
iov = _iovec(ctypes.cast(dummy, ctypes.c_void_p), ctypes.sizeof(dummy)) | ||||
cbuf = ctypes.create_string_buffer(256) | ||||
Augie Fackler
|
r43346 | msgh = _msghdr( | ||
None, | ||||
0, | ||||
ctypes.pointer(iov), | ||||
1, | ||||
ctypes.cast(cbuf, ctypes.c_void_p), | ||||
ctypes.sizeof(cbuf), | ||||
0, | ||||
) | ||||
Yuya Nishihara
|
r27474 | r = _recvmsg(sockfd, ctypes.byref(msgh), 0) | ||
if r < 0: | ||||
e = ctypes.get_errno() | ||||
raise OSError(e, os.strerror(e)) | ||||
# assumes that the first cmsg has fds because it isn't easy to write | ||||
# portable CMSG_NXTHDR() with ctypes. | ||||
cmsg = _CMSG_FIRSTHDR(msgh) | ||||
if not cmsg: | ||||
return [] | ||||
Augie Fackler
|
r43346 | if ( | ||
cmsg.cmsg_level != socket.SOL_SOCKET | ||||
or cmsg.cmsg_type != _SCM_RIGHTS | ||||
): | ||||
Yuya Nishihara
|
r27474 | return [] | ||
rfds = ctypes.cast(cmsg.cmsg_data, ctypes.POINTER(ctypes.c_int)) | ||||
Augie Fackler
|
r43346 | rfdscount = ( | ||
cmsg.cmsg_len - _cmsghdr.cmsg_data.offset | ||||
) // ctypes.sizeof(ctypes.c_int) | ||||
Gregory Szorc
|
r38806 | return [rfds[i] for i in pycompat.xrange(rfdscount)] | ||
Yuya Nishihara
|
r27474 | |||
Augie Fackler
|
r43346 | |||
Adrian Buehlmann
|
r14413 | else: | ||
Gregory Szorc
|
r27338 | import msvcrt | ||
Adrian Buehlmann
|
r14413 | |||
_kernel32 = ctypes.windll.kernel32 | ||||
_DWORD = ctypes.c_ulong | ||||
_LPCSTR = _LPSTR = ctypes.c_char_p | ||||
_HANDLE = ctypes.c_void_p | ||||
_INVALID_HANDLE_VALUE = _HANDLE(-1).value | ||||
Mads Kiilerich
|
r18959 | # CreateFile | ||
Adrian Buehlmann
|
r14413 | _FILE_SHARE_READ = 0x00000001 | ||
_FILE_SHARE_WRITE = 0x00000002 | ||||
_FILE_SHARE_DELETE = 0x00000004 | ||||
_CREATE_ALWAYS = 2 | ||||
_OPEN_EXISTING = 3 | ||||
_OPEN_ALWAYS = 4 | ||||
_GENERIC_READ = 0x80000000 | ||||
_GENERIC_WRITE = 0x40000000 | ||||
_FILE_ATTRIBUTE_NORMAL = 0x80 | ||||
Mads Kiilerich
|
r17429 | # open_osfhandle flags | ||
Adrian Buehlmann
|
r14413 | _O_RDONLY = 0x0000 | ||
_O_RDWR = 0x0002 | ||||
_O_APPEND = 0x0008 | ||||
_O_TEXT = 0x4000 | ||||
_O_BINARY = 0x8000 | ||||
# types of parameters of C functions used (required by pypy) | ||||
Augie Fackler
|
r43346 | _kernel32.CreateFileA.argtypes = [ | ||
_LPCSTR, | ||||
_DWORD, | ||||
_DWORD, | ||||
ctypes.c_void_p, | ||||
_DWORD, | ||||
_DWORD, | ||||
_HANDLE, | ||||
] | ||||
Adrian Buehlmann
|
r14413 | _kernel32.CreateFileA.restype = _HANDLE | ||
def _raiseioerror(name): | ||||
err = ctypes.WinError() | ||||
Augie Fackler
|
r43346 | raise IOError( | ||
err.errno, r'%s: %s' % (encoding.strfromlocal(name), err.strerror) | ||||
) | ||||
Adrian Buehlmann
|
r14413 | |||
class posixfile(object): | ||||
'''a file object aiming for POSIX-like semantics | ||||
CPython's open() returns a file that was opened *without* setting the | ||||
_FILE_SHARE_DELETE flag, which causes rename and unlink to abort. | ||||
This even happens if any hardlinked copy of the file is in open state. | ||||
We set _FILE_SHARE_DELETE here, so files opened with posixfile can be | ||||
renamed and deleted while they are held open. | ||||
Note that if a file opened with posixfile is unlinked, the file | ||||
remains but cannot be opened again or be recreated under the same name, | ||||
until all reading processes have closed the file.''' | ||||
Matt Harbison
|
r39680 | def __init__(self, name, mode=b'r', bufsize=-1): | ||
if b'b' in mode: | ||||
Adrian Buehlmann
|
r14413 | flags = _O_BINARY | ||
else: | ||||
flags = _O_TEXT | ||||
Matt Harbison
|
r39680 | m0 = mode[0:1] | ||
if m0 == b'r' and b'+' not in mode: | ||||
Adrian Buehlmann
|
r14413 | flags |= _O_RDONLY | ||
access = _GENERIC_READ | ||||
else: | ||||
# work around http://support.microsoft.com/kb/899149 and | ||||
# set _O_RDWR for 'w' and 'a', even if mode has no '+' | ||||
flags |= _O_RDWR | ||||
access = _GENERIC_READ | _GENERIC_WRITE | ||||
Matt Harbison
|
r39680 | if m0 == b'r': | ||
Adrian Buehlmann
|
r14413 | creation = _OPEN_EXISTING | ||
Matt Harbison
|
r39680 | elif m0 == b'w': | ||
Adrian Buehlmann
|
r14413 | creation = _CREATE_ALWAYS | ||
Matt Harbison
|
r39680 | elif m0 == b'a': | ||
Adrian Buehlmann
|
r14413 | creation = _OPEN_ALWAYS | ||
flags |= _O_APPEND | ||||
else: | ||||
Matt Harbison
|
r39680 | raise ValueError(r"invalid mode: %s" % pycompat.sysstr(mode)) | ||
Adrian Buehlmann
|
r14413 | |||
Augie Fackler
|
r43346 | fh = _kernel32.CreateFileA( | ||
name, | ||||
access, | ||||
_FILE_SHARE_READ | _FILE_SHARE_WRITE | _FILE_SHARE_DELETE, | ||||
None, | ||||
creation, | ||||
_FILE_ATTRIBUTE_NORMAL, | ||||
None, | ||||
) | ||||
Adrian Buehlmann
|
r14413 | if fh == _INVALID_HANDLE_VALUE: | ||
_raiseioerror(name) | ||||
Adrian Buehlmann
|
r16474 | fd = msvcrt.open_osfhandle(fh, flags) | ||
Adrian Buehlmann
|
r14413 | if fd == -1: | ||
_kernel32.CloseHandle(fh) | ||||
_raiseioerror(name) | ||||
Pulkit Goyal
|
r30925 | f = os.fdopen(fd, pycompat.sysstr(mode), bufsize) | ||
Adrian Buehlmann
|
r14413 | # unfortunately, f.name is '<fdopen>' at this point -- so we store | ||
# the name on this wrapper. We cannot just assign to f.name, | ||||
# because that attribute is read-only. | ||||
Yuya Nishihara
|
r31644 | object.__setattr__(self, r'name', name) | ||
object.__setattr__(self, r'_file', f) | ||||
Adrian Buehlmann
|
r14413 | |||
def __iter__(self): | ||||
return self._file | ||||
def __getattr__(self, name): | ||||
return getattr(self._file, name) | ||||
def __setattr__(self, name, value): | ||||
'''mimics the read-only attributes of Python file objects | ||||
by raising 'TypeError: readonly attribute' if someone tries: | ||||
f = posixfile('foo.txt') | ||||
f.name = 'bla' ''' | ||||
return self._file.__setattr__(name, value) | ||||
Gregory Szorc
|
r27704 | |||
def __enter__(self): | ||||
Matt Harbison
|
r40976 | self._file.__enter__() | ||
return self | ||||
Gregory Szorc
|
r27704 | |||
def __exit__(self, exc_type, exc_value, exc_tb): | ||||
return self._file.__exit__(exc_type, exc_value, exc_tb) | ||||