~bzr-pqm/bzr/bzr.dev

3504.4.13 by John Arbash Meinel
Clean up according to review comments.
1
# Copyright (C) 2008 Canonical Ltd
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
2
#
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
7
#
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
12
#
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
4183.7.1 by Sabin Iacob
update FSF mailing address
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
16
17
"""Helper functions for Walkdirs on win32."""
18
19
3737.1.3 by John Arbash Meinel
Move more compatibility code into python-compat.h
20
cdef extern from "python-compat.h":
3557.2.8 by Alexander Belchenko
We don't need all the extra cdef statments in a cdef extern block
21
    struct _HANDLE:
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
22
        pass
23
    ctypedef _HANDLE *HANDLE
3504.4.8 by John Arbash Meinel
Some code cleanups.
24
    ctypedef unsigned long DWORD
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
25
    ctypedef long long __int64
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
26
    ctypedef unsigned short WCHAR
3557.2.8 by Alexander Belchenko
We don't need all the extra cdef statments in a cdef extern block
27
    struct _FILETIME:
3504.4.6 by John Arbash Meinel
Start exposing the times on the stat, this now seems to be a complete walkdirs implementation.
28
        DWORD dwHighDateTime
29
        DWORD dwLowDateTime
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
30
    ctypedef _FILETIME FILETIME
31
3557.2.8 by Alexander Belchenko
We don't need all the extra cdef statments in a cdef extern block
32
    struct _WIN32_FIND_DATAW:
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
33
        DWORD dwFileAttributes
34
        FILETIME ftCreationTime
35
        FILETIME ftLastAccessTime
36
        FILETIME ftLastWriteTime
37
        DWORD nFileSizeHigh
38
        DWORD nFileSizeLow
39
        # Some reserved stuff here
40
        WCHAR cFileName[260] # MAX_PATH
41
        WCHAR cAlternateFilename[14]
42
43
    # We have to use the typedef trick, otherwise pyrex uses:
44
    #  struct WIN32_FIND_DATAW
45
    # which fails due to 'incomplete type'
46
    ctypedef _WIN32_FIND_DATAW WIN32_FIND_DATAW
47
3557.2.8 by Alexander Belchenko
We don't need all the extra cdef statments in a cdef extern block
48
    HANDLE INVALID_HANDLE_VALUE
49
    HANDLE FindFirstFileW(WCHAR *path, WIN32_FIND_DATAW *data)
50
    int FindNextFileW(HANDLE search, WIN32_FIND_DATAW *data)
51
    int FindClose(HANDLE search)
52
53
    DWORD FILE_ATTRIBUTE_READONLY
54
    DWORD FILE_ATTRIBUTE_DIRECTORY
55
    int ERROR_NO_MORE_FILES
56
57
    int GetLastError()
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
58
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
59
    # Wide character functions
60
    DWORD wcslen(WCHAR *)
61
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
62
63
cdef extern from "Python.h":
64
    WCHAR *PyUnicode_AS_UNICODE(object)
65
    Py_ssize_t PyUnicode_GET_SIZE(object)
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
66
    object PyUnicode_FromUnicode(WCHAR *, Py_ssize_t)
3504.4.8 by John Arbash Meinel
Some code cleanups.
67
    int PyList_Append(object, object) except -1
68
    object PyUnicode_AsUTF8String(object)
69
70
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
71
import operator
72
import stat
73
3696.3.10 by Robert Collins
Review feedback.
74
from bzrlib import osutils, _readdir_py
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
75
76
3504.4.9 by John Arbash Meinel
Switch to using a cdef object with readonly attributes.
77
cdef class _Win32Stat:
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
78
    """Represent a 'stat' result generated from WIN32_FIND_DATA"""
79
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
80
    cdef readonly int st_mode
81
    cdef readonly double st_ctime
82
    cdef readonly double st_mtime
83
    cdef readonly double st_atime
3788.1.2 by John Arbash Meinel
Switch from using cdef readonly __int64 to using a property.
84
    # We can't just declare this as 'readonly' because python2.4 doesn't define
85
    # T_LONGLONG as a structure member. So instead we just use a property that
86
    # will convert it correctly anyway.
87
    cdef __int64 _st_size
88
89
    property st_size:
90
        def __get__(self):
91
            return self._st_size
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
92
93
    # os.stat always returns 0, so we hard code it here
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
94
    cdef readonly int st_dev
95
    cdef readonly int st_ino
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
96
97
    def __repr__(self):
98
        """Repr is the same as a Stat object.
99
100
        (mode, ino, dev, nlink, uid, gid, size, atime, mtime, ctime)
101
        """
102
        return repr((self.st_mode, 0, 0, 0, 0, 0, self.st_size, self.st_atime,
103
                     self.st_mtime, self.st_ctime))
104
105
3504.4.11 by John Arbash Meinel
A bit more reorganizing.
106
cdef object _get_name(WIN32_FIND_DATAW *data):
107
    """Extract the Unicode name for this file/dir."""
108
    return PyUnicode_FromUnicode(data.cFileName,
109
                                 wcslen(data.cFileName))
110
111
4634.117.10 by John Arbash Meinel
Change 'no except' to 'cannot_raise'
112
cdef int _get_mode_bits(WIN32_FIND_DATAW *data): # cannot_raise
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
113
    cdef int mode_bits
114
115
    mode_bits = 0100666 # writeable file, the most common
116
    if data.dwFileAttributes & FILE_ATTRIBUTE_READONLY == FILE_ATTRIBUTE_READONLY:
3557.2.5 by John Arbash Meinel
Test that the empty-directory logic for all _walkdirs implementations is correct.
117
        mode_bits = mode_bits ^ 0222 # remove the write bits
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
118
    if data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY == FILE_ATTRIBUTE_DIRECTORY:
119
        # Remove the FILE bit, set the DIR bit, and set the EXEC bits
3557.2.5 by John Arbash Meinel
Test that the empty-directory logic for all _walkdirs implementations is correct.
120
        mode_bits = mode_bits ^ 0140111
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
121
    return mode_bits
122
123
4634.117.10 by John Arbash Meinel
Change 'no except' to 'cannot_raise'
124
cdef __int64 _get_size(WIN32_FIND_DATAW *data): # cannot_raise
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
125
    # Pyrex casts a DWORD into a PyLong anyway, so it is safe to do << 32
126
    # on a DWORD
127
    return ((<__int64>data.nFileSizeHigh) << 32) + data.nFileSizeLow
128
129
4634.117.10 by John Arbash Meinel
Change 'no except' to 'cannot_raise'
130
cdef double _ftime_to_timestamp(FILETIME *ft): # cannot_raise
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
131
    """Convert from a FILETIME struct into a floating point timestamp.
132
133
    The fields of a FILETIME structure are the hi and lo part
134
    of a 64-bit value expressed in 100 nanosecond units.
135
    1e7 is one second in such units; 1e-7 the inverse.
136
    429.4967296 is 2**32 / 1e7 or 2**32 * 1e-7.
137
    It also uses the epoch 1601-01-01 rather than 1970-01-01
138
    (taken from posixmodule.c)
139
    """
140
    cdef __int64 val
141
    # NB: This gives slightly different results versus casting to a 64-bit
142
    #     integer and doing integer math before casting into a floating
143
    #     point number. But the difference is in the sub millisecond range,
144
    #     which doesn't seem critical here.
145
    # secs between epochs: 11,644,473,600
146
    val = ((<__int64>ft.dwHighDateTime) << 32) + ft.dwLowDateTime
147
    return (val * 1.0e-7) - 11644473600.0
148
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
149
4634.117.10 by John Arbash Meinel
Change 'no except' to 'cannot_raise'
150
cdef int _should_skip(WIN32_FIND_DATAW *data): # cannot_raise
3504.4.11 by John Arbash Meinel
A bit more reorganizing.
151
    """Is this '.' or '..' so we should skip it?"""
152
    if (data.cFileName[0] != c'.'):
153
        return 0
154
    if data.cFileName[1] == c'\0':
155
        return 1
156
    if data.cFileName[1] == c'.' and data.cFileName[2] == c'\0':
157
        return 1
158
    return 0
159
160
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
161
cdef class Win32ReadDir:
162
    """Read directories on win32."""
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
163
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
164
    cdef object _directory_kind
165
    cdef object _file_kind
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
166
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
167
    def __init__(self):
3696.3.10 by Robert Collins
Review feedback.
168
        self._directory_kind = _readdir_py._directory
169
        self._file_kind = _readdir_py._file
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
170
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
171
    def top_prefix_to_starting_dir(self, top, prefix=""):
172
        """See DirReader.top_prefix_to_starting_dir."""
3696.3.2 by John Arbash Meinel
Fix up some of the path joining logic.
173
        return (osutils.safe_utf8(prefix), None, None, None,
174
                osutils.safe_unicode(top))
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
175
3504.4.12 by John Arbash Meinel
A couple small cleanups, make test_osutils more correct
176
    cdef object _get_kind(self, WIN32_FIND_DATAW *data):
177
        if data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY:
178
            return self._directory_kind
179
        return self._file_kind
180
3504.4.9 by John Arbash Meinel
Switch to using a cdef object with readonly attributes.
181
    cdef _Win32Stat _get_stat_value(self, WIN32_FIND_DATAW *data):
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
182
        """Get the filename and the stat information."""
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
183
        cdef _Win32Stat statvalue
184
185
        statvalue = _Win32Stat()
186
        statvalue.st_mode = _get_mode_bits(data)
187
        statvalue.st_ctime = _ftime_to_timestamp(&data.ftCreationTime)
188
        statvalue.st_mtime = _ftime_to_timestamp(&data.ftLastWriteTime)
189
        statvalue.st_atime = _ftime_to_timestamp(&data.ftLastAccessTime)
3788.1.6 by John Arbash Meinel
Quick fix for win32 extension.
190
        statvalue._st_size = _get_size(data)
3504.4.10 by John Arbash Meinel
Move the helpers to be standalone, rather than members
191
        statvalue.st_ino = 0
192
        statvalue.st_dev = 0
193
        return statvalue
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
194
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
195
    def read_dir(self, prefix, top):
196
        """Win32 implementation of DirReader.read_dir.
3557.2.2 by John Arbash Meinel
Include better documentation about what is going on. (suggested by Bialix)
197
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
198
        :seealso: DirReader.read_dir
3557.2.2 by John Arbash Meinel
Include better documentation about what is going on. (suggested by Bialix)
199
        """
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
200
        cdef WIN32_FIND_DATAW search_data
201
        cdef HANDLE hFindFile
202
        cdef int last_err
203
        cdef WCHAR *query
204
        cdef int result
205
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
206
        if prefix:
207
            relprefix = prefix + '/'
208
        else:
209
            relprefix = ''
210
        top_slash = top + '/'
211
3696.3.2 by John Arbash Meinel
Fix up some of the path joining logic.
212
        top_star = top_slash + '*'
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
213
214
        dirblock = []
215
216
        query = PyUnicode_AS_UNICODE(top_star)
217
        hFindFile = FindFirstFileW(query, &search_data)
218
        if hFindFile == INVALID_HANDLE_VALUE:
219
            # Raise an exception? This path doesn't seem to exist
3504.4.8 by John Arbash Meinel
Some code cleanups.
220
            raise WindowsError(GetLastError(), top_star)
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
221
222
        try:
223
            result = 1
224
            while result:
225
                # Skip '.' and '..'
3504.4.11 by John Arbash Meinel
A bit more reorganizing.
226
                if _should_skip(&search_data):
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
227
                    result = FindNextFileW(hFindFile, &search_data)
228
                    continue
3504.4.12 by John Arbash Meinel
A couple small cleanups, make test_osutils more correct
229
                name_unicode = _get_name(&search_data)
3504.4.8 by John Arbash Meinel
Some code cleanups.
230
                name_utf8 = PyUnicode_AsUTF8String(name_unicode)
3696.3.2 by John Arbash Meinel
Fix up some of the path joining logic.
231
                PyList_Append(dirblock,
232
                    (relprefix + name_utf8, name_utf8,
3504.4.8 by John Arbash Meinel
Some code cleanups.
233
                     self._get_kind(&search_data),
234
                     self._get_stat_value(&search_data),
3696.3.2 by John Arbash Meinel
Fix up some of the path joining logic.
235
                     top_slash + name_unicode))
3504.4.4 by John Arbash Meinel
We have walkdirs basically working, only without timestamps
236
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
237
                result = FindNextFileW(hFindFile, &search_data)
3504.4.8 by John Arbash Meinel
Some code cleanups.
238
            # FindNextFileW sets GetLastError() == ERROR_NO_MORE_FILES when it
239
            # actually finishes. If we have anything else, then we have a
240
            # genuine problem
241
            last_err = GetLastError()
242
            if last_err != ERROR_NO_MORE_FILES:
243
                raise WindowsError(last_err)
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
244
        finally:
245
            result = FindClose(hFindFile)
246
            if result == 0:
247
                last_err = GetLastError()
3557.2.2 by John Arbash Meinel
Include better documentation about what is going on. (suggested by Bialix)
248
                # TODO: We should probably raise an exception if FindClose
249
                #       returns an error, however, I don't want to supress an
250
                #       earlier Exception, so for now, I'm ignoring this
3696.3.1 by Robert Collins
Refactor bzrlib.osutils._walkdirs_utf8 to aid API migration in future.
251
        dirblock.sort(key=operator.itemgetter(1))
3504.4.3 by John Arbash Meinel
Start working on an extension specifically for win32,
252
        return dirblock