/brz/remove-bazaar

To get this branch, use:
bzr branch http://gegoxaren.bato24.eu/bzr/brz/remove-bazaar
0.3.1 by John Arbash Meinel
Creating a plugin for uncommitting the last revision.
1
#!/usr/bin/env python
2
"""\
3
Remove the last revision from the history of the current branch.
4
"""
5
6
import os
7
import bzrlib
8
9
try:
10
    set
11
except NameError:
12
    from sets import Set as set
13
14
def test_remove(filename):
15
    if os.path.exists(filename):
16
        os.remove(filename)
17
    else:
18
        print '* file does not exist: %r' % filename
19
20
21
def uncommit(branch, remove_files=False,
22
        dry_run=False, verbose=False):
23
    """Remove the last revision from the supplied branch.
24
25
    :param remove_files: If True, remove files from the stores
26
        as well.
27
    """
28
    from bzrlib.atomicfile import AtomicFile
29
    rh = branch.revision_history()
30
    rev_id = rh.pop()
31
    rev = branch.get_revision(rev_id)
32
    inv = branch.get_inventory(rev.inventory_id)
33
    inv_prev = []
34
    for p in rev.parents:
35
        inv_prev.append(branch.get_revision_inventory(p.revision_id))
36
37
    new_rev_history = AtomicFile(branch.controlfilename('revision-history'))
38
    new_rev_history.write('\n'.join(rh))
39
    # Committing now, because even if we fail to remove all files
40
    # once we have removed at least one, all the rest are invalid.
41
    if not dry_run:
42
        new_rev_history.commit()
43
    else:
44
        new_rev_history.abort()
45
46
    if remove_files:
47
        # Figure out what text-store entries are new
48
        files_to_remove = []
49
        for file_id in inv:
50
            ie = inv[file_id]
51
            if not hasattr(ie, 'text_id'):
52
                continue
53
            for other_inv in inv_prev:
54
                if file_id in other_inv:
55
                    other_ie = other_inv[file_id]
56
                    if other_ie.text_id == ie.text_id:
57
                        break
58
            else:
59
                # None of the previous ancestors used
60
                # the same inventory
61
                files_to_remove.append(branch.controlfilename(['text-store',
62
                    ie.text_id + '.gz']))
63
        rev_file = branch.controlfilename(['revision-store',
64
                rev_id + '.gz'])
65
        files_to_remove.append(rev_file)
66
        inv_file = branch.controlfilename(['inventory-store',
67
                rev.inventory_id + '.gz'])
68
        files_to_remove.append(inv_file)
69
70
        if verbose:
71
            print 'Removing files:'
72
            for f in files_to_remove:
73
                print '\t%s' % branch.relpath(f)
74
75
        if not dry_run:
76
            # Actually start removing files
77
            for f in files_to_remove:
78
                test_remove(f)
79