~abentley/bzrtools/bzrtools.dev

16 by abentley
Got is_clean under test, added setters/getters for pull data
1
import bzrlib
19 by abentley
librified most of the pull script
2
import os
16 by abentley
Got is_clean under test, added setters/getters for pull data
3
import os.path
19 by abentley
librified most of the pull script
4
import sys
16 by abentley
Got is_clean under test, added setters/getters for pull data
5
import tempfile
6
import shutil
7
8
def temp_branch():
9
    dirname = tempfile.mkdtemp("temp-branch")
10
    return bzrlib.Branch(dirname, init=True)
11
12
def rm_branch(br):
13
    shutil.rmtree(br.base)
14
15
def is_clean(cur_branch):
16
    """
17
    Return true if no files are modifed or unknown
18
    >>> br = temp_branch()
19
    >>> is_clean(br)
20
    True
21
    >>> fooname = os.path.join(br.base, "foo")
22
    >>> file(fooname, "wb").write("bar")
23
    >>> is_clean(br)
24
    False
25
    >>> bzrlib.add.smart_add([fooname])
26
    >>> is_clean(br)
27
    False
28
    >>> br.commit("added file")
29
    >>> is_clean(br)
30
    True
31
    >>> rm_branch(br)
32
    """
33
    old_tree = cur_branch.basis_tree()
34
    new_tree = cur_branch.working_tree()
35
    for file_state, fid, old_name, new_name, kind in \
36
        bzrlib.diff_trees(old_tree, new_tree):
37
        if file_state not in ('I', '.'):
38
            return False
39
    return True
40
41
def set_pull_data(br, location, rev_id):
19 by abentley
librified most of the pull script
42
    pull_file = file (br.controlfilename("x-pull-data"), "wb")
16 by abentley
Got is_clean under test, added setters/getters for pull data
43
    pull_file.write("%s\n%s\n" % (location, rev_id))
44
45
def get_pull_data(br):
46
    """
47
    >>> br = temp_branch()
48
    >>> get_pull_data(br)
49
    (None, None)
50
    >>> set_pull_data(br, 'http://somewhere', '888-777')
51
    >>> get_pull_data(br)
52
    ('http://somewhere', '888-777')
53
    >>> rm_branch(br)
54
    """
19 by abentley
librified most of the pull script
55
    filename = br.controlfilename("x-pull-data")
16 by abentley
Got is_clean under test, added setters/getters for pull data
56
    if not os.path.exists(filename):
57
        return (None, None)
58
    pull_file = file (filename, "rb")
59
    location, rev_id = [f.rstrip('\n') for f in pull_file]
60
    return location, rev_id
61
20 by abentley
added bzr-push command
62
def set_push_data(br, location):
63
    push_file = file (br.controlfilename("x-push-data"), "wb")
64
    push_file.write("%s\n" % location)
65
66
def get_push_data(br):
67
    """
68
    >>> br = temp_branch()
69
    >>> get_push_data(br) is None
70
    True
71
    >>> set_push_data(br, 'http://somewhere')
72
    >>> get_push_data(br)
73
    'http://somewhere'
74
    >>> rm_branch(br)
75
    """
76
    filename = br.controlfilename("x-push-data")
77
    if not os.path.exists(filename):
78
        return None
79
    push_file = file (filename, "rb")
80
    (location,) = [f.rstrip('\n') for f in push_file]
81
    return location
82
19 by abentley
librified most of the pull script
83
"""
84
>>> shell_escape('hello')
85
'\h\e\l\l\o'
86
"""
87
def shell_escape(arg):
88
    return "".join(['\\'+c for c in arg])
89
90
def safe_system(args):
91
    """
92
    >>> real_system = os.system
93
    >>> os.system = sys.stdout.write
94
    >>> safe_system(['a', 'b', 'cd'])
95
    \\a \\b \\c\\d
96
    >>> os.system = real_system
97
    """
98
    arg_str = " ".join([shell_escape(a) for a in args])
99
    return os.system(arg_str)
100
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
101
def rsync(source, target, ssh=False, exclude_globs=()):
19 by abentley
librified most of the pull script
102
    """
103
    >>> real_system = os.system
104
    >>> os.system = sys.stdout.write
105
    >>> rsync("a", "b")
106
    \\r\\s\\y\\n\\c \\-\\a\\v \\-\\-\\d\\e\\l\\e\\t\\e \\a \\b
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
107
    >>> rsync("a", "b", exclude_globs=("*.py",))
108
    \\r\\s\\y\\n\\c \\-\\a\\v \\-\\-\\d\\e\\l\\e\\t\\e\
109
 \\-\\-\\e\\x\\c\\l\\u\\d\\e \\*\\.\\p\\y \\a \\b
19 by abentley
librified most of the pull script
110
    >>> os.system = real_system
111
    """
20 by abentley
added bzr-push command
112
    cmd = ["rsync", "-av", "--delete"]
113
    if ssh:
114
        cmd.extend(('-e', 'ssh'))
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
115
    for exclude in exclude_globs:
116
        cmd.extend(('--exclude', exclude))
20 by abentley
added bzr-push command
117
    cmd.extend((source, target))
118
    safe_system(cmd)
19 by abentley
librified most of the pull script
119
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
120
exclusions = ('x-push-data', 'x-pull-data')
121
122
19 by abentley
librified most of the pull script
123
def pull(cur_branch, location=None):
124
    pull_location, pull_revision = get_pull_data(cur_branch)
125
    if pull_location is not None:
126
        if cur_branch.last_patch() != pull_revision:
127
            print "Aborting: This branch has had commits, so pull would lose data."
128
            sys.exit(1)
21 by abentley
removed references to sys.argv
129
    if location is not None:
130
        pull_location = location
19 by abentley
librified most of the pull script
131
        if not pull_location.endswith('/'):
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
132
            pull_location+='/'
19 by abentley
librified most of the pull script
133
134
    if pull_location is None:
135
        print "No pull location saved.  Please specify one on the command line."
136
        sys.exit(1)
137
138
    if not is_clean(cur_branch):
139
        print "Error: This tree has uncommitted changes or unknown (?) files."
140
        sys.exit(1)
141
142
    print "Synchronizing with %s" % pull_location
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
143
    rsync (pull_location, cur_branch.base+'/', exclude_globs=exclusions)
19 by abentley
librified most of the pull script
144
145
    set_pull_data(cur_branch, pull_location, cur_branch.last_patch())
146
20 by abentley
added bzr-push command
147
148
def push(cur_branch, location=None):
149
    push_location = get_push_data(cur_branch)
150
    if location is not None:
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
151
        if not location.endswith('/'):
152
            location += '/'
20 by abentley
added bzr-push command
153
        push_location = location
154
    
155
    if push_location is None:
156
        print "No push location saved.  Please specify one on the command line."
157
        sys.exit(1)
158
159
    if not is_clean(cur_branch):
160
        print "Error: This tree has uncommitted changes or unknown (?) files."
161
        sys.exit(1)
162
163
    print "Pushing to %s" % push_location
25 by abentley
fixed push for x files, tracefile, push/pull miscommunication
164
    rsync(cur_branch.base+'/', push_location, ssh=True,
165
          exclude_globs=exclusions)
20 by abentley
added bzr-push command
166
167
    set_push_data(cur_branch, push_location)
168
19 by abentley
librified most of the pull script
169
def run_tests():
16 by abentley
Got is_clean under test, added setters/getters for pull data
170
    import doctest
18 by abentley
Finished implementing bzr-pull
171
    result = doctest.testmod()
19 by abentley
librified most of the pull script
172
    if result[1] > 0:
173
        if result[0] == 0:
174
            print "All tests passed"
175
    else:
176
        print "No tests to run"
177
if __name__ == "__main__":
178
    run_tests()