~abentley/bzrtools/bzrtools.dev

147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
1
# Copyright (C) 2004 Aaron Bentley
2
# <aaron.bentley@utoronto.ca>
3
#
4
#    This program is free software; you can redistribute it and/or modify
5
#    it under the terms of the GNU General Public License as published by
6
#    the Free Software Foundation; either version 2 of the License, or
7
#    (at your option) any later version.
8
#
9
#    This program is distributed in the hope that it will be useful,
10
#    but WITHOUT ANY WARRANTY; without even the implied warranty of
11
#    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12
#    GNU General Public License for more details.
13
#
14
#    You should have received a copy of the GNU General Public License
15
#    along with this program; if not, write to the Free Software
16
#    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
17
147.2.3 by Aaron Bentley
Fixed import issues w/ PyBaz
18
import pybaz
19
from pybaz.backends.baz import sequence_cmd
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
20
import re
21
22
__docformat__ = "restructuredtext"
23
__doc__ = "Utility functions to be used by commands"
147.1.24 by Robert Collins
trim fai cribbage
24
147.4.6 by Robert Collins
Fix continuation direct_merges output, and allow reusing history in a version import
25
def direct_merges(merges, excludes=[]):
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
26
    """Get a list of direct merges, from a list of direct and indirect
27
    
28
    :param merges: Iterator of merge patchlogs
147.1.26 by Robert Collins
be consistent about pybaz vs arch naming in the fai derived code
29
    :type merges: iter of `pybaz.Patchlog`
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
30
    :return: The direct merges
147.1.26 by Robert Collins
be consistent about pybaz vs arch naming in the fai derived code
31
    :rtype: list of `pybaz.Patchlog`
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
32
    """
147.4.7 by Robert Collins
use sets and strings to optimise finding indirect merges somewhat
33
    indirect = set()
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
34
    logs = list(merges)
147.1.5 by Robert Collins
test and implement direct merge support of 2-in-a-row patches
35
    if not logs:
36
        return []
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
37
    for log in logs:
147.4.4 by Robert Collins
when we encounter an obviously corrupt patchlog, work around it by using the archive.
38
        try:
147.4.7 by Robert Collins
use sets and strings to optimise finding indirect merges somewhat
39
            this_indirect = set([str(f) for f in log.new_patches 
40
                                 if f != log.revision and 
41
                                     str(f) not in indirect])
147.4.4 by Robert Collins
when we encounter an obviously corrupt patchlog, work around it by using the archive.
42
        except pybaz.errors.NamespaceError:
43
            print
44
            print "log ", log, " unusable, attempting to use archive copy."
45
            log = pybaz.Revision(str(log.revision)).patchlog
147.4.7 by Robert Collins
use sets and strings to optimise finding indirect merges somewhat
46
            this_indirect = set([str(f) for f in log.new_patches 
47
                                 if f != log.revision and 
48
                                     str(f) not in indirect])
49
        indirect.update(this_indirect)
147.1.7 by Robert Collins
handle inaccessible sibling archives somewhat - note version-0 is still not handled
50
        if log.continuation_of is not None:
51
            # continuations list everything in new_patches
52
            continue
53
        try:
54
            # ask baz if we can
55
            ancestor = log.revision.ancestor
56
        except pybaz.errors.ExecProblem:
57
            # baz does not know
58
            # guess
147.1.9 by Robert Collins
determine version-0 ancestors from a tree when possible
59
            if log.revision.patchlevel != 'version-0': 
60
                ancestor = namespace_previous(log.revision)
61
            else:
62
                found = False
63
                ancestor = None
64
                # ancestor might be in the tree
65
                for tree_log in pybaz.WorkingTree(log.tree).iter_logs(
66
                    log.revision.version, reverse=True):
67
                    if found:
68
                        ancestor = tree_log.revision
69
                        break
70
                    if tree_log.revision == log.revision:
71
                        found = True
72
                print "ancestor of %s is %s" % (log.revision, ancestor)
147.1.8 by Robert Collins
handle version-xxx and base-0 correctly in namespace_previous
73
        if ancestor is not None:
147.4.7 by Robert Collins
use sets and strings to optimise finding indirect merges somewhat
74
            indirect.add(str(ancestor))
75
    return [log.revision for log in logs if not str(log.revision) in indirect
76
            and log.revision not in excludes]
77
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
78
147.1.7 by Robert Collins
handle inaccessible sibling archives somewhat - note version-0 is still not handled
79
def namespace_previous(revision):
147.1.8 by Robert Collins
handle version-xxx and base-0 correctly in namespace_previous
80
    if revision.patchlevel == 'base-0':
81
        return None
147.1.7 by Robert Collins
handle inaccessible sibling archives somewhat - note version-0 is still not handled
82
    if revision.patchlevel == 'patch-1':
83
        return revision.version['base-0']
84
    if revision.patchlevel.startswith('patch'):
85
        level = int(revision.patchlevel[len('patch-'):]) -1
86
        return revision.version['patch-%d' % level]
87
    if revision.patchlevel == 'version-0':
88
        raise RuntimeError("cannot determine prior namespace level for a "
147.1.9 by Robert Collins
determine version-0 ancestors from a tree when possible
89
                           "version-0 patch ('%s')" % revision)
147.1.29 by Robert Collins
update to latest bzr api
90
    if revision.patchlevel == 'versionfix-1':
147.1.7 by Robert Collins
handle inaccessible sibling archives somewhat - note version-0 is still not handled
91
        return revision.version['version-0']
147.1.29 by Robert Collins
update to latest bzr api
92
    if revision.patchlevel.startswith('versionfix'):
93
        level = int(revision.patchlevel[len('versionfix-'):]) -1
94
        return revision.version['versionfix-%d' % level]
147.1.8 by Robert Collins
handle version-xxx and base-0 correctly in namespace_previous
95
    raise NotImplementedError
147.1.3 by Robert Collins
test and deliver basic pending-merges into bzr so that merging is recorded
96
147.1.24 by Robert Collins
trim fai cribbage
97
def iter_new_merges(tree, version, reverse=False):
98
    """List patchlogs that are new in this tree since the last commit.
99
100
    :param tree: The working tree to calculate new revisions in
101
    :type tree: str
102
    :param version: The version to use when determining new logs
103
    :type version: str
104
    :param reverse: If true, list backwards, from newest to oldest
105
    :type reverse: bool
106
    :return: An iterator for new revision logs in this tree
147.1.26 by Robert Collins
be consistent about pybaz vs arch naming in the fai derived code
107
    :rtype: Iterator of `pybaz.Patchlog`
147.1.24 by Robert Collins
trim fai cribbage
108
    """
147.1.26 by Robert Collins
be consistent about pybaz vs arch naming in the fai derived code
109
    assert (isinstance(version, pybaz.Version))
147.1.24 by Robert Collins
trim fai cribbage
110
    for line in _iter_new_merges(tree, version.fullname, reverse):
147.1.26 by Robert Collins
be consistent about pybaz vs arch naming in the fai derived code
111
        yield pybaz.Patchlog(line, tree)
147.1.24 by Robert Collins
trim fai cribbage
112
113
def _iter_new_merges(directory, version, reverse):
114
    """List patchlogs that are new in this tree since the last commit.
115
116
    :param directory: The working tree to calculate new revisions in
117
    :type directory: str
118
    :param version: The version to use when determining new logs
119
    :type version: str
120
    :param reverse: If true, list backwards, from newest to oldest
121
    :type reverse: bool
122
    :return: An iterator for names of revisions new in this tree
123
    :rtype: Iterator of str
124
    """
125
    args = [ 'new-merges', '--dir', directory ]
126
    if reverse: 
127
        args.append('--reverse')
128
    args.append(version)
129
    return sequence_cmd(args)