~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/multiparent.py

Merge up bzr.dev.

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2007-2011 Canonical Ltd
 
1
# Copyright (C) 2007 Canonical Ltd
2
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
12
12
#
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
 
 
17
 
from __future__ import absolute_import
 
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
18
16
 
19
17
from bzrlib.lazy_import import lazy_import
20
18
 
21
19
lazy_import(globals(), """
22
20
import errno
23
 
import gzip
24
21
import itertools
25
22
import os
26
23
from StringIO import StringIO
27
24
 
28
25
from bzrlib import (
29
 
    bencode,
30
26
    errors,
31
27
    patiencediff,
 
28
    trace,
32
29
    ui,
33
30
    )
 
31
from bzrlib.util import bencode
34
32
""")
35
 
 
36
 
 
37
 
def topo_iter_keys(vf, keys=None):
38
 
    if keys is None:
39
 
        keys = vf.keys()
40
 
    parents = vf.get_parent_map(keys)
41
 
    return _topo_iter(parents, keys)
 
33
from bzrlib.tuned_gzip import GzipFile
 
34
 
42
35
 
43
36
def topo_iter(vf, versions=None):
 
37
    seen = set()
 
38
    descendants = {}
44
39
    if versions is None:
45
40
        versions = vf.versions()
46
41
    parents = vf.get_parent_map(versions)
47
 
    return _topo_iter(parents, versions)
48
 
 
49
 
def _topo_iter(parents, versions):
50
 
    seen = set()
51
 
    descendants = {}
52
42
    def pending_parents(version):
53
 
        if parents[version] is None:
54
 
            return []
55
43
        return [v for v in parents[version] if v in versions and
56
44
                v not in seen]
57
45
    for version_id in versions:
58
 
        if parents[version_id] is None:
59
 
            # parentless
60
 
            continue
61
46
        for parent_id in parents[version_id]:
62
47
            descendants.setdefault(parent_id, []).append(version_id)
63
48
    cur = [v for v in versions if len(pending_parents(v)) == 0]
77
62
class MultiParent(object):
78
63
    """A multi-parent diff"""
79
64
 
80
 
    __slots__ = ['hunks']
81
 
 
82
65
    def __init__(self, hunks=None):
83
66
        if hunks is not None:
84
67
            self.hunks = hunks
261
244
class NewText(object):
262
245
    """The contents of text that is introduced by this text"""
263
246
 
264
 
    __slots__ = ['lines']
265
 
 
266
247
    def __init__(self, lines):
267
248
        self.lines = lines
268
249
 
284
265
class ParentText(object):
285
266
    """A reference to text present in a parent text"""
286
267
 
287
 
    __slots__ = ['parent', 'parent_pos', 'child_pos', 'num_lines']
288
 
 
289
268
    def __init__(self, parent, parent_pos, child_pos, num_lines):
290
269
        self.parent = parent
291
270
        self.parent_pos = parent_pos
292
271
        self.child_pos = child_pos
293
272
        self.num_lines = num_lines
294
273
 
295
 
    def _as_dict(self):
296
 
        return dict(parent=self.parent, parent_pos=self.parent_pos,
297
 
                    child_pos=self.child_pos, num_lines=self.num_lines)
298
 
 
299
274
    def __repr__(self):
300
 
        return ('ParentText(%(parent)r, %(parent_pos)r, %(child_pos)r,'
301
 
                ' %(num_lines)r)' % self._as_dict())
 
275
        return 'ParentText(%(parent)r, %(parent_pos)r, %(child_pos)r,'\
 
276
            ' %(num_lines)r)' % self.__dict__
302
277
 
303
278
    def __eq__(self, other):
304
 
        if self.__class__ is not other.__class__:
 
279
        if self.__class__ != other.__class__:
305
280
            return False
306
 
        return self._as_dict() == other._as_dict()
 
281
        return (self.__dict__ == other.__dict__)
307
282
 
308
283
    def to_patch(self):
309
 
        yield ('c %(parent)d %(parent_pos)d %(child_pos)d %(num_lines)d\n'
310
 
               % self._as_dict())
 
284
        yield 'c %(parent)d %(parent_pos)d %(child_pos)d %(num_lines)d\n'\
 
285
            % self.__dict__
311
286
 
312
287
 
313
288
class BaseVersionedFile(object):
327
302
        return version in self._parents
328
303
 
329
304
    def do_snapshot(self, version_id, parent_ids):
330
 
        """Determine whether to perform a snapshot for this version"""
 
305
        """Determine whether to perform a a snapshot for this version"""
331
306
        if self.snapshot_interval is None:
332
307
            return False
333
308
        if self.max_snapshots is not None and\
423
398
                            if not (lines == self.get_line_list([revision])[0]):
424
399
                                raise AssertionError()
425
400
                            self.clear_cache()
426
 
                    pb.update(gettext('Importing revisions'),
 
401
                    pb.update('Importing revisions',
427
402
                              (total - len(revisions)) + len(added), total)
428
403
                revisions = [r for r in revisions if r not in added]
429
404
        finally:
562
537
            sio = StringIO(infile.read(count))
563
538
        finally:
564
539
            infile.close()
565
 
        zip_file = gzip.GzipFile(None, mode='rb', fileobj=sio)
 
540
        zip_file = GzipFile(None, mode='rb', fileobj=sio)
566
541
        try:
567
542
            file_version_id = zip_file.readline()
568
 
            content = zip_file.read()
569
 
            return MultiParent.from_patch(content)
 
543
            return MultiParent.from_patch(zip_file.read())
570
544
        finally:
571
545
            zip_file.close()
572
546
 
578
552
                                    # before any write returns 0
579
553
            start = outfile.tell()
580
554
            try:
581
 
                zipfile = gzip.GzipFile(None, mode='ab', fileobj=outfile)
 
555
                zipfile = GzipFile(None, mode='ab', fileobj=outfile)
582
556
                zipfile.writelines(itertools.chain(
583
557
                    ['version %s\n' % version_id], diff.to_patch()))
584
558
            finally:
675
649
 
676
650
def gzip_string(lines):
677
651
    sio = StringIO()
678
 
    data_file = gzip.GzipFile(None, mode='wb', fileobj=sio)
 
652
    data_file = GzipFile(None, mode='wb', fileobj=sio)
679
653
    data_file.writelines(lines)
680
654
    data_file.close()
681
655
    return sio.getvalue()