13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17
17
"""Python implementations of Dirstate Helper functions."""
19
from __future__ import absolute_import
21
25
# We cannot import the dirstate module, because it loads this module
22
26
# All we really need is the IN_MEMORY_MODIFIED constant
24
28
from bzrlib.dirstate import DirState
27
def _bisect_path_left_py(paths, path):
31
def pack_stat(st, _b64=binascii.b2a_base64, _pack=struct.Struct('>6L').pack):
32
"""Convert stat values into a packed representation
34
Not all of the fields from the stat included are strictly needed, and by
35
just encoding the mtime and mode a slight speed increase could be gained.
36
However, using the pyrex version instead is a bigger win.
38
# base64 encoding always adds a final newline, so strip it off
39
return _b64(_pack(st.st_size & 0xFFFFFFFF, int(st.st_mtime) & 0xFFFFFFFF,
40
int(st.st_ctime) & 0xFFFFFFFF, st.st_dev & 0xFFFFFFFF,
41
st.st_ino & 0xFFFFFFFF, st.st_mode))[:-1]
44
def _unpack_stat(packed_stat):
45
"""Turn a packed_stat back into the stat fields.
47
This is meant as a debugging tool, should not be used in real code.
49
(st_size, st_mtime, st_ctime, st_dev, st_ino,
50
st_mode) = struct.unpack('>6L', binascii.a2b_base64(packed_stat))
51
return dict(st_size=st_size, st_mtime=st_mtime, st_ctime=st_ctime,
52
st_dev=st_dev, st_ino=st_ino, st_mode=st_mode)
55
def _bisect_path_left(paths, path):
28
56
"""Return the index where to insert path into paths.
30
58
This uses the dirblock sorting. So all children in a directory come before
63
91
mid = (lo + hi) // 2
64
92
# Grab the dirname for the current dirblock
66
if _cmp_path_by_dirblock_py(cur, path) < 0:
94
if _cmp_path_by_dirblock(cur, path) < 0:
73
def _bisect_path_right_py(paths, path):
101
def _bisect_path_right(paths, path):
74
102
"""Return the index where to insert path into paths.
76
104
This uses a path-wise comparison so we get::
95
123
# Grab the dirname for the current dirblock
97
if _cmp_path_by_dirblock_py(path, cur) < 0:
125
if _cmp_path_by_dirblock(path, cur) < 0:
104
def bisect_dirblock_py(dirblocks, dirname, lo=0, hi=None, cache={}):
132
def bisect_dirblock(dirblocks, dirname, lo=0, hi=None, cache={}):
105
133
"""Return the index where to insert dirname into the dirblocks.
107
135
The return value idx is such that all directories blocks in dirblock[:idx]
135
def cmp_by_dirs_py(path1, path2):
163
def cmp_by_dirs(path1, path2):
136
164
"""Compare two paths directory by directory.
138
166
This is equivalent to doing::
158
186
return cmp(path1.split('/'), path2.split('/'))
161
def _cmp_path_by_dirblock_py(path1, path2):
189
def _cmp_path_by_dirblock(path1, path2):
162
190
"""Compare two paths based on what directory they are in.
164
192
This generates a sort order, such that all children of a directory are