##// END OF EJS Templates
Optimize diff and status in subdirectories...
mpm@selenic.com -
r312:09375250 default
parent child Browse files
Show More
@@ -1,670 +1,670 b''
1 # commands.py - command processing for mercurial
1 # commands.py - command processing for mercurial
2 #
2 #
3 # Copyright 2005 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005 Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms
5 # This software may be used and distributed according to the terms
6 # of the GNU General Public License, incorporated herein by reference.
6 # of the GNU General Public License, incorporated herein by reference.
7
7
8 import os, re, sys, signal
8 import os, re, sys, signal
9 import fancyopts, ui, hg
9 import fancyopts, ui, hg
10 from demandload import *
10 from demandload import *
11 demandload(globals(), "mdiff time hgweb traceback")
11 demandload(globals(), "mdiff time hgweb traceback")
12
12
13 class UnknownCommand(Exception): pass
13 class UnknownCommand(Exception): pass
14
14
15 def filterfiles(filters, files):
15 def filterfiles(filters, files):
16 l = [ x for x in files if x in filters ]
16 l = [ x for x in files if x in filters ]
17
17
18 for t in filters:
18 for t in filters:
19 if t and t[-1] != os.sep: t += os.sep
19 if t and t[-1] != os.sep: t += os.sep
20 l += [ x for x in files if x.startswith(t) ]
20 l += [ x for x in files if x.startswith(t) ]
21 return l
21 return l
22
22
23 def relfilter(repo, files):
23 def relfilter(repo, files):
24 if os.getcwd() != repo.root:
24 if os.getcwd() != repo.root:
25 p = os.getcwd()[len(repo.root) + 1: ]
25 p = os.getcwd()[len(repo.root) + 1: ]
26 return filterfiles([p], files)
26 return filterfiles([p], files)
27 return files
27 return files
28
28
29 def relpath(repo, args):
29 def relpath(repo, args):
30 if os.getcwd() != repo.root:
30 if os.getcwd() != repo.root:
31 p = os.getcwd()[len(repo.root) + 1: ]
31 p = os.getcwd()[len(repo.root) + 1: ]
32 return [ os.path.normpath(os.path.join(p, x)) for x in args ]
32 return [ os.path.normpath(os.path.join(p, x)) for x in args ]
33 return args
33 return args
34
34
35 def dodiff(repo, files = None, node1 = None, node2 = None):
35 def dodiff(repo, path, files = None, node1 = None, node2 = None):
36 def date(c):
36 def date(c):
37 return time.asctime(time.gmtime(float(c[2].split(' ')[0])))
37 return time.asctime(time.gmtime(float(c[2].split(' ')[0])))
38
38
39 if node2:
39 if node2:
40 change = repo.changelog.read(node2)
40 change = repo.changelog.read(node2)
41 mmap2 = repo.manifest.read(change[0])
41 mmap2 = repo.manifest.read(change[0])
42 (c, a, d) = repo.diffrevs(node1, node2)
42 (c, a, d) = repo.diffrevs(node1, node2)
43 def read(f): return repo.file(f).read(mmap2[f])
43 def read(f): return repo.file(f).read(mmap2[f])
44 date2 = date(change)
44 date2 = date(change)
45 else:
45 else:
46 date2 = time.asctime()
46 date2 = time.asctime()
47 (c, a, d, u) = repo.diffdir(repo.root, node1)
47 (c, a, d, u) = repo.diffdir(path, node1)
48 if not node1:
48 if not node1:
49 node1 = repo.dirstate.parents()[0]
49 node1 = repo.dirstate.parents()[0]
50 def read(f): return file(os.path.join(repo.root, f)).read()
50 def read(f): return file(os.path.join(repo.root, f)).read()
51
51
52 change = repo.changelog.read(node1)
52 change = repo.changelog.read(node1)
53 mmap = repo.manifest.read(change[0])
53 mmap = repo.manifest.read(change[0])
54 date1 = date(change)
54 date1 = date(change)
55
55
56 if files:
56 if files:
57 c, a, d = map(lambda x: filterfiles(files, x), (c, a, d))
57 c, a, d = map(lambda x: filterfiles(files, x), (c, a, d))
58
58
59 for f in c:
59 for f in c:
60 to = None
60 to = None
61 if f in mmap:
61 if f in mmap:
62 to = repo.file(f).read(mmap[f])
62 to = repo.file(f).read(mmap[f])
63 tn = read(f)
63 tn = read(f)
64 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
64 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
65 for f in a:
65 for f in a:
66 to = None
66 to = None
67 tn = read(f)
67 tn = read(f)
68 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
68 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
69 for f in d:
69 for f in d:
70 to = repo.file(f).read(mmap[f])
70 to = repo.file(f).read(mmap[f])
71 tn = None
71 tn = None
72 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
72 sys.stdout.write(mdiff.unidiff(to, date1, tn, date2, f))
73
73
74 def help(ui, cmd=None):
74 def help(ui, cmd=None):
75 '''show help for a given command or all commands'''
75 '''show help for a given command or all commands'''
76 if cmd:
76 if cmd:
77 try:
77 try:
78 i = find(cmd)
78 i = find(cmd)
79 ui.write("%s\n\n" % i[2])
79 ui.write("%s\n\n" % i[2])
80
80
81 if i[1]:
81 if i[1]:
82 for s, l, d, c in i[1]:
82 for s, l, d, c in i[1]:
83 opt=' '
83 opt=' '
84 if s: opt = opt + '-' + s + ' '
84 if s: opt = opt + '-' + s + ' '
85 if l: opt = opt + '--' + l + ' '
85 if l: opt = opt + '--' + l + ' '
86 if d: opt = opt + '(' + str(d) + ')'
86 if d: opt = opt + '(' + str(d) + ')'
87 ui.write(opt, "\n")
87 ui.write(opt, "\n")
88 if c: ui.write(' %s\n' % c)
88 if c: ui.write(' %s\n' % c)
89 ui.write("\n")
89 ui.write("\n")
90
90
91 ui.write(i[0].__doc__, "\n")
91 ui.write(i[0].__doc__, "\n")
92 except UnknownCommand:
92 except UnknownCommand:
93 ui.warn("hg: unknown command %s\n" % cmd)
93 ui.warn("hg: unknown command %s\n" % cmd)
94 sys.exit(0)
94 sys.exit(0)
95 else:
95 else:
96 ui.status('hg commands:\n\n')
96 ui.status('hg commands:\n\n')
97
97
98 h = {}
98 h = {}
99 for e in table.values():
99 for e in table.values():
100 f = e[0]
100 f = e[0]
101 if f.__name__.startswith("debug"): continue
101 if f.__name__.startswith("debug"): continue
102 d = ""
102 d = ""
103 if f.__doc__:
103 if f.__doc__:
104 d = f.__doc__.splitlines(0)[0].rstrip()
104 d = f.__doc__.splitlines(0)[0].rstrip()
105 h[f.__name__] = d
105 h[f.__name__] = d
106
106
107 fns = h.keys()
107 fns = h.keys()
108 fns.sort()
108 fns.sort()
109 m = max(map(len, fns))
109 m = max(map(len, fns))
110 for f in fns:
110 for f in fns:
111 ui.status(' %-*s %s\n' % (m, f, h[f]))
111 ui.status(' %-*s %s\n' % (m, f, h[f]))
112
112
113 # Commands start here, listed alphabetically
113 # Commands start here, listed alphabetically
114
114
115 def add(ui, repo, file, *files):
115 def add(ui, repo, file, *files):
116 '''add the specified files on the next commit'''
116 '''add the specified files on the next commit'''
117 repo.add(relpath(repo, (file,) + files))
117 repo.add(relpath(repo, (file,) + files))
118
118
119 def addremove(ui, repo):
119 def addremove(ui, repo):
120 """add all new files, delete all missing files"""
120 """add all new files, delete all missing files"""
121 (c, a, d, u) = repo.diffdir(repo.root)
121 (c, a, d, u) = repo.diffdir(repo.root)
122 repo.add(u)
122 repo.add(u)
123 repo.remove(d)
123 repo.remove(d)
124
124
125 def annotate(u, repo, file, *files, **ops):
125 def annotate(u, repo, file, *files, **ops):
126 """show changeset information per file line"""
126 """show changeset information per file line"""
127 def getnode(rev):
127 def getnode(rev):
128 return hg.short(repo.changelog.node(rev))
128 return hg.short(repo.changelog.node(rev))
129
129
130 def getname(rev):
130 def getname(rev):
131 try:
131 try:
132 return bcache[rev]
132 return bcache[rev]
133 except KeyError:
133 except KeyError:
134 cl = repo.changelog.read(repo.changelog.node(rev))
134 cl = repo.changelog.read(repo.changelog.node(rev))
135 name = cl[1]
135 name = cl[1]
136 f = name.find('@')
136 f = name.find('@')
137 if f >= 0:
137 if f >= 0:
138 name = name[:f]
138 name = name[:f]
139 bcache[rev] = name
139 bcache[rev] = name
140 return name
140 return name
141
141
142 bcache = {}
142 bcache = {}
143 opmap = [['user', getname], ['number', str], ['changeset', getnode]]
143 opmap = [['user', getname], ['number', str], ['changeset', getnode]]
144 if not ops['user'] and not ops['changeset']:
144 if not ops['user'] and not ops['changeset']:
145 ops['number'] = 1
145 ops['number'] = 1
146
146
147 node = repo.dirstate.parents()[0]
147 node = repo.dirstate.parents()[0]
148 if ops['revision']:
148 if ops['revision']:
149 node = repo.changelog.lookup(ops['revision'])
149 node = repo.changelog.lookup(ops['revision'])
150 change = repo.changelog.read(node)
150 change = repo.changelog.read(node)
151 mmap = repo.manifest.read(change[0])
151 mmap = repo.manifest.read(change[0])
152 maxuserlen = 0
152 maxuserlen = 0
153 maxchangelen = 0
153 maxchangelen = 0
154 for f in relpath(repo, (file,) + files):
154 for f in relpath(repo, (file,) + files):
155 lines = repo.file(f).annotate(mmap[f])
155 lines = repo.file(f).annotate(mmap[f])
156 pieces = []
156 pieces = []
157
157
158 for o, f in opmap:
158 for o, f in opmap:
159 if ops[o]:
159 if ops[o]:
160 l = [ f(n) for n,t in lines ]
160 l = [ f(n) for n,t in lines ]
161 m = max(map(len, l))
161 m = max(map(len, l))
162 pieces.append([ "%*s" % (m, x) for x in l])
162 pieces.append([ "%*s" % (m, x) for x in l])
163
163
164 for p,l in zip(zip(*pieces), lines):
164 for p,l in zip(zip(*pieces), lines):
165 u.write(" ".join(p) + ": " + l[1])
165 u.write(" ".join(p) + ": " + l[1])
166
166
167 def cat(ui, repo, file, rev = []):
167 def cat(ui, repo, file, rev = []):
168 """output the latest or given revision of a file"""
168 """output the latest or given revision of a file"""
169 r = repo.file(relpath(repo, [file])[0])
169 r = repo.file(relpath(repo, [file])[0])
170 n = r.tip()
170 n = r.tip()
171 if rev: n = r.lookup(rev)
171 if rev: n = r.lookup(rev)
172 sys.stdout.write(r.read(n))
172 sys.stdout.write(r.read(n))
173
173
174 def commit(ui, repo, *files, **opts):
174 def commit(ui, repo, *files, **opts):
175 """commit the specified files or all outstanding changes"""
175 """commit the specified files or all outstanding changes"""
176 text = opts['text']
176 text = opts['text']
177 if not text and opts['logfile']:
177 if not text and opts['logfile']:
178 try: text = open(opts['logfile']).read()
178 try: text = open(opts['logfile']).read()
179 except IOError: pass
179 except IOError: pass
180
180
181 repo.commit(relpath(repo, files), text)
181 repo.commit(relpath(repo, files), text)
182
182
183 def debugaddchangegroup(ui, repo):
183 def debugaddchangegroup(ui, repo):
184 data = sys.stdin.read()
184 data = sys.stdin.read()
185 repo.addchangegroup(data)
185 repo.addchangegroup(data)
186
186
187 def debugchangegroup(ui, repo, roots):
187 def debugchangegroup(ui, repo, roots):
188 newer = repo.newer(map(repo.lookup, roots))
188 newer = repo.newer(map(repo.lookup, roots))
189 for chunk in repo.changegroup(newer):
189 for chunk in repo.changegroup(newer):
190 sys.stdout.write(chunk)
190 sys.stdout.write(chunk)
191
191
192 def debugindex(ui, file):
192 def debugindex(ui, file):
193 r = hg.revlog(open, file, "")
193 r = hg.revlog(open, file, "")
194 print " rev offset length base linkrev"+\
194 print " rev offset length base linkrev"+\
195 " p1 p2 nodeid"
195 " p1 p2 nodeid"
196 for i in range(r.count()):
196 for i in range(r.count()):
197 e = r.index[i]
197 e = r.index[i]
198 print "% 6d % 9d % 7d % 6d % 7d %s.. %s.. %s.." % (
198 print "% 6d % 9d % 7d % 6d % 7d %s.. %s.. %s.." % (
199 i, e[0], e[1], e[2], e[3],
199 i, e[0], e[1], e[2], e[3],
200 hg.hex(e[4][:5]), hg.hex(e[5][:5]), hg.hex(e[6][:5]))
200 hg.hex(e[4][:5]), hg.hex(e[5][:5]), hg.hex(e[6][:5]))
201
201
202 def debugindexdot(ui, file):
202 def debugindexdot(ui, file):
203 r = hg.revlog(open, file, "")
203 r = hg.revlog(open, file, "")
204 print "digraph G {"
204 print "digraph G {"
205 for i in range(r.count()):
205 for i in range(r.count()):
206 e = r.index[i]
206 e = r.index[i]
207 print "\t%d -> %d" % (r.rev(e[4]), i)
207 print "\t%d -> %d" % (r.rev(e[4]), i)
208 if e[5] != hg.nullid:
208 if e[5] != hg.nullid:
209 print "\t%d -> %d" % (r.rev(e[5]), i)
209 print "\t%d -> %d" % (r.rev(e[5]), i)
210 print "}"
210 print "}"
211
211
212 def diff(ui, repo, *files, **opts):
212 def diff(ui, repo, *files, **opts):
213 """diff working directory (or selected files)"""
213 """diff working directory (or selected files)"""
214 revs = []
214 revs = []
215 if opts['rev']:
215 if opts['rev']:
216 revs = map(lambda x: repo.lookup(x), opts['rev'])
216 revs = map(lambda x: repo.lookup(x), opts['rev'])
217
217
218 if len(revs) > 2:
218 if len(revs) > 2:
219 self.ui.warn("too many revisions to diff\n")
219 self.ui.warn("too many revisions to diff\n")
220 sys.exit(1)
220 sys.exit(1)
221
221
222 if files:
222 if files:
223 files = relpath(repo, files)
223 files = relpath(repo, files)
224 else:
224 else:
225 files = relpath(repo, [""])
225 files = relpath(repo, [""])
226
226
227 dodiff(repo, files, *revs)
227 dodiff(repo, os.getcwd(), files, *revs)
228
228
229 def export(ui, repo, changeset):
229 def export(ui, repo, changeset):
230 """dump the changeset header and diffs for a revision"""
230 """dump the changeset header and diffs for a revision"""
231 node = repo.lookup(changeset)
231 node = repo.lookup(changeset)
232 prev, other = repo.changelog.parents(node)
232 prev, other = repo.changelog.parents(node)
233 change = repo.changelog.read(node)
233 change = repo.changelog.read(node)
234 print "# HG changeset patch"
234 print "# HG changeset patch"
235 print "# User %s" % change[1]
235 print "# User %s" % change[1]
236 print "# Node ID %s" % hg.hex(node)
236 print "# Node ID %s" % hg.hex(node)
237 print "# Parent %s" % hg.hex(prev)
237 print "# Parent %s" % hg.hex(prev)
238 print
238 print
239 if other != hg.nullid:
239 if other != hg.nullid:
240 print "# Parent %s" % hg.hex(other)
240 print "# Parent %s" % hg.hex(other)
241 print change[4].rstrip()
241 print change[4].rstrip()
242 print
242 print
243
243
244 dodiff(repo, None, prev, node)
244 dodiff(repo, "", None, prev, node)
245
245
246 def forget(ui, repo, file, *files):
246 def forget(ui, repo, file, *files):
247 """don't add the specified files on the next commit"""
247 """don't add the specified files on the next commit"""
248 repo.forget(relpath(repo, (file,) + files))
248 repo.forget(relpath(repo, (file,) + files))
249
249
250 def heads(ui, repo):
250 def heads(ui, repo):
251 '''show current repository heads'''
251 '''show current repository heads'''
252 for n in repo.changelog.heads():
252 for n in repo.changelog.heads():
253 i = repo.changelog.rev(n)
253 i = repo.changelog.rev(n)
254 changes = repo.changelog.read(n)
254 changes = repo.changelog.read(n)
255 (p1, p2) = repo.changelog.parents(n)
255 (p1, p2) = repo.changelog.parents(n)
256 (h, h1, h2) = map(hg.hex, (n, p1, p2))
256 (h, h1, h2) = map(hg.hex, (n, p1, p2))
257 (i1, i2) = map(repo.changelog.rev, (p1, p2))
257 (i1, i2) = map(repo.changelog.rev, (p1, p2))
258 print "rev: %4d:%s" % (i, h)
258 print "rev: %4d:%s" % (i, h)
259 print "parents: %4d:%s" % (i1, h1)
259 print "parents: %4d:%s" % (i1, h1)
260 if i2: print " %4d:%s" % (i2, h2)
260 if i2: print " %4d:%s" % (i2, h2)
261 print "manifest: %4d:%s" % (repo.manifest.rev(changes[0]),
261 print "manifest: %4d:%s" % (repo.manifest.rev(changes[0]),
262 hg.hex(changes[0]))
262 hg.hex(changes[0]))
263 print "user:", changes[1]
263 print "user:", changes[1]
264 print "date:", time.asctime(
264 print "date:", time.asctime(
265 time.localtime(float(changes[2].split(' ')[0])))
265 time.localtime(float(changes[2].split(' ')[0])))
266 if ui.verbose: print "files:", " ".join(changes[3])
266 if ui.verbose: print "files:", " ".join(changes[3])
267 print "description:"
267 print "description:"
268 print changes[4]
268 print changes[4]
269
269
270 def history(ui, repo):
270 def history(ui, repo):
271 """show the changelog history"""
271 """show the changelog history"""
272 for i in range(repo.changelog.count() - 1, -1, -1):
272 for i in range(repo.changelog.count() - 1, -1, -1):
273 n = repo.changelog.node(i)
273 n = repo.changelog.node(i)
274 changes = repo.changelog.read(n)
274 changes = repo.changelog.read(n)
275 (p1, p2) = repo.changelog.parents(n)
275 (p1, p2) = repo.changelog.parents(n)
276 (h, h1, h2) = map(hg.hex, (n, p1, p2))
276 (h, h1, h2) = map(hg.hex, (n, p1, p2))
277 (i1, i2) = map(repo.changelog.rev, (p1, p2))
277 (i1, i2) = map(repo.changelog.rev, (p1, p2))
278 print "rev: %4d:%s" % (i, h)
278 print "rev: %4d:%s" % (i, h)
279 print "parents: %4d:%s" % (i1, h1)
279 print "parents: %4d:%s" % (i1, h1)
280 if i2: print " %4d:%s" % (i2, h2)
280 if i2: print " %4d:%s" % (i2, h2)
281 print "manifest: %4d:%s" % (repo.manifest.rev(changes[0]),
281 print "manifest: %4d:%s" % (repo.manifest.rev(changes[0]),
282 hg.hex(changes[0]))
282 hg.hex(changes[0]))
283 print "user:", changes[1]
283 print "user:", changes[1]
284 print "date:", time.asctime(
284 print "date:", time.asctime(
285 time.localtime(float(changes[2].split(' ')[0])))
285 time.localtime(float(changes[2].split(' ')[0])))
286 if ui.verbose: print "files:", " ".join(changes[3])
286 if ui.verbose: print "files:", " ".join(changes[3])
287 print "description:"
287 print "description:"
288 print changes[4]
288 print changes[4]
289
289
290 def init(ui, source=None):
290 def init(ui, source=None):
291 """create a new repository or copy an existing one"""
291 """create a new repository or copy an existing one"""
292
292
293 if source:
293 if source:
294 paths = {}
294 paths = {}
295 for name, path in ui.configitems("paths"):
295 for name, path in ui.configitems("paths"):
296 paths[name] = path
296 paths[name] = path
297
297
298 if source in paths: source = paths[source]
298 if source in paths: source = paths[source]
299
299
300 link = 0
300 link = 0
301 if not source.startswith("http://"):
301 if not source.startswith("http://"):
302 d1 = os.stat(os.getcwd()).st_dev
302 d1 = os.stat(os.getcwd()).st_dev
303 d2 = os.stat(source).st_dev
303 d2 = os.stat(source).st_dev
304 if d1 == d2: link = 1
304 if d1 == d2: link = 1
305
305
306 if link:
306 if link:
307 ui.debug("copying by hardlink\n")
307 ui.debug("copying by hardlink\n")
308 os.system("cp -al %s/.hg .hg" % source)
308 os.system("cp -al %s/.hg .hg" % source)
309 try:
309 try:
310 os.remove(".hg/dirstate")
310 os.remove(".hg/dirstate")
311 except: pass
311 except: pass
312 else:
312 else:
313 repo = hg.repository(ui, ".", create=1)
313 repo = hg.repository(ui, ".", create=1)
314 other = hg.repository(ui, source)
314 other = hg.repository(ui, source)
315 cg = repo.getchangegroup(other)
315 cg = repo.getchangegroup(other)
316 repo.addchangegroup(cg)
316 repo.addchangegroup(cg)
317 else:
317 else:
318 hg.repository(ui, ".", create=1)
318 hg.repository(ui, ".", create=1)
319
319
320 def log(ui, repo, f):
320 def log(ui, repo, f):
321 """show the revision history of a single file"""
321 """show the revision history of a single file"""
322 f = relpath(repo, [f])[0]
322 f = relpath(repo, [f])[0]
323
323
324 r = repo.file(f)
324 r = repo.file(f)
325 for i in range(r.count() - 1, -1, -1):
325 for i in range(r.count() - 1, -1, -1):
326 n = r.node(i)
326 n = r.node(i)
327 (p1, p2) = r.parents(n)
327 (p1, p2) = r.parents(n)
328 (h, h1, h2) = map(hg.hex, (n, p1, p2))
328 (h, h1, h2) = map(hg.hex, (n, p1, p2))
329 (i1, i2) = map(r.rev, (p1, p2))
329 (i1, i2) = map(r.rev, (p1, p2))
330 cr = r.linkrev(n)
330 cr = r.linkrev(n)
331 cn = hg.hex(repo.changelog.node(cr))
331 cn = hg.hex(repo.changelog.node(cr))
332 print "rev: %4d:%s" % (i, h)
332 print "rev: %4d:%s" % (i, h)
333 print "changeset: %4d:%s" % (cr, cn)
333 print "changeset: %4d:%s" % (cr, cn)
334 print "parents: %4d:%s" % (i1, h1)
334 print "parents: %4d:%s" % (i1, h1)
335 if i2: print " %4d:%s" % (i2, h2)
335 if i2: print " %4d:%s" % (i2, h2)
336 changes = repo.changelog.read(repo.changelog.node(cr))
336 changes = repo.changelog.read(repo.changelog.node(cr))
337 print "user: %s" % changes[1]
337 print "user: %s" % changes[1]
338 print "date: %s" % time.asctime(
338 print "date: %s" % time.asctime(
339 time.localtime(float(changes[2].split(' ')[0])))
339 time.localtime(float(changes[2].split(' ')[0])))
340 print "description:"
340 print "description:"
341 print changes[4].rstrip()
341 print changes[4].rstrip()
342 print
342 print
343
343
344 def manifest(ui, repo, rev = []):
344 def manifest(ui, repo, rev = []):
345 """output the latest or given revision of the project manifest"""
345 """output the latest or given revision of the project manifest"""
346 n = repo.manifest.tip()
346 n = repo.manifest.tip()
347 if rev:
347 if rev:
348 n = repo.manifest.lookup(rev)
348 n = repo.manifest.lookup(rev)
349 m = repo.manifest.read(n)
349 m = repo.manifest.read(n)
350 mf = repo.manifest.readflags(n)
350 mf = repo.manifest.readflags(n)
351 files = m.keys()
351 files = m.keys()
352 files.sort()
352 files.sort()
353
353
354 for f in files:
354 for f in files:
355 ui.write("%40s %3s %s\n" % (hg.hex(m[f]), mf[f] and "755" or "644", f))
355 ui.write("%40s %3s %s\n" % (hg.hex(m[f]), mf[f] and "755" or "644", f))
356
356
357 def parents(ui, repo, node = None):
357 def parents(ui, repo, node = None):
358 '''show the parents of the current working dir'''
358 '''show the parents of the current working dir'''
359 if node:
359 if node:
360 p = repo.changelog.parents(repo.lookup(hg.bin(node)))
360 p = repo.changelog.parents(repo.lookup(hg.bin(node)))
361 else:
361 else:
362 p = repo.dirstate.parents()
362 p = repo.dirstate.parents()
363
363
364 for n in p:
364 for n in p:
365 if n != hg.nullid:
365 if n != hg.nullid:
366 ui.write("%d:%s\n" % (repo.changelog.rev(n), hg.hex(n)))
366 ui.write("%d:%s\n" % (repo.changelog.rev(n), hg.hex(n)))
367
367
368 def patch(ui, repo, patch1, *patches, **opts):
368 def patch(ui, repo, patch1, *patches, **opts):
369 """import an ordered set of patches"""
369 """import an ordered set of patches"""
370 try:
370 try:
371 import psyco
371 import psyco
372 psyco.full()
372 psyco.full()
373 except:
373 except:
374 pass
374 pass
375
375
376 patches = (patch1,) + patches
376 patches = (patch1,) + patches
377
377
378 d = opts["base"]
378 d = opts["base"]
379 strip = opts["strip"]
379 strip = opts["strip"]
380 quiet = opts["quiet"] and "> /dev/null" or ""
380 quiet = opts["quiet"] and "> /dev/null" or ""
381
381
382 for patch in patches:
382 for patch in patches:
383 ui.status("applying %s\n" % patch)
383 ui.status("applying %s\n" % patch)
384 pf = os.path.join(d, patch)
384 pf = os.path.join(d, patch)
385
385
386 text = ""
386 text = ""
387 for l in file(pf):
387 for l in file(pf):
388 if l[:4] == "--- ": break
388 if l[:4] == "--- ": break
389 text += l
389 text += l
390
390
391 # make sure text isn't empty
391 # make sure text isn't empty
392 if not text: text = "imported patch %s\n" % patch
392 if not text: text = "imported patch %s\n" % patch
393
393
394 f = os.popen("lsdiff --strip %d %s" % (strip, pf))
394 f = os.popen("lsdiff --strip %d %s" % (strip, pf))
395 files = filter(None, map(lambda x: x.rstrip(), f.read().splitlines()))
395 files = filter(None, map(lambda x: x.rstrip(), f.read().splitlines()))
396 f.close()
396 f.close()
397
397
398 if files:
398 if files:
399 if os.system("patch -p%d < %s %s" % (strip, pf, quiet)):
399 if os.system("patch -p%d < %s %s" % (strip, pf, quiet)):
400 raise "patch failed!"
400 raise "patch failed!"
401 repo.commit(files, text)
401 repo.commit(files, text)
402
402
403 def pull(ui, repo, source):
403 def pull(ui, repo, source):
404 """pull changes from the specified source"""
404 """pull changes from the specified source"""
405 paths = {}
405 paths = {}
406 for name, path in ui.configitems("paths"):
406 for name, path in ui.configitems("paths"):
407 paths[name] = path
407 paths[name] = path
408
408
409 if source in paths: source = paths[source]
409 if source in paths: source = paths[source]
410
410
411 other = hg.repository(ui, source)
411 other = hg.repository(ui, source)
412 cg = repo.getchangegroup(other)
412 cg = repo.getchangegroup(other)
413 repo.addchangegroup(cg)
413 repo.addchangegroup(cg)
414
414
415 def rawcommit(ui, repo, files, **rc):
415 def rawcommit(ui, repo, files, **rc):
416 "raw commit interface"
416 "raw commit interface"
417
417
418 text = rc['text']
418 text = rc['text']
419 if not text and rc['logfile']:
419 if not text and rc['logfile']:
420 try: text = open(rc['logfile']).read()
420 try: text = open(rc['logfile']).read()
421 except IOError: pass
421 except IOError: pass
422 if not text and not rc['logfile']:
422 if not text and not rc['logfile']:
423 print "missing commit text"
423 print "missing commit text"
424 return 1
424 return 1
425
425
426 files = relpath(repo, files)
426 files = relpath(repo, files)
427 if rc['files']:
427 if rc['files']:
428 files += open(rc['files']).read().splitlines()
428 files += open(rc['files']).read().splitlines()
429
429
430 repo.rawcommit(files, text, rc['user'], rc['date'], *rc['parent'])
430 repo.rawcommit(files, text, rc['user'], rc['date'], *rc['parent'])
431
431
432 def recover(ui, repo):
432 def recover(ui, repo):
433 """roll back an interrupted transaction"""
433 """roll back an interrupted transaction"""
434 repo.recover()
434 repo.recover()
435
435
436 def remove(ui, repo, file, *files):
436 def remove(ui, repo, file, *files):
437 """remove the specified files on the next commit"""
437 """remove the specified files on the next commit"""
438 repo.remove(relpath(repo, (file,) + files))
438 repo.remove(relpath(repo, (file,) + files))
439
439
440 def serve(ui, repo, **opts):
440 def serve(ui, repo, **opts):
441 """export the repository via HTTP"""
441 """export the repository via HTTP"""
442 hgweb.server(repo.root, opts["name"], opts["templates"],
442 hgweb.server(repo.root, opts["name"], opts["templates"],
443 opts["address"], opts["port"])
443 opts["address"], opts["port"])
444
444
445 def status(ui, repo):
445 def status(ui, repo):
446 '''show changed files in the working directory
446 '''show changed files in the working directory
447
447
448 C = changed
448 C = changed
449 A = added
449 A = added
450 R = removed
450 R = removed
451 ? = not tracked'''
451 ? = not tracked'''
452
452
453 (c, a, d, u) = repo.diffdir(repo.root)
453 (c, a, d, u) = repo.diffdir(os.getcwd())
454 (c, a, d, u) = map(lambda x: relfilter(repo, x), (c, a, d, u))
454 (c, a, d, u) = map(lambda x: relfilter(repo, x), (c, a, d, u))
455
455
456 for f in c: print "C", f
456 for f in c: print "C", f
457 for f in a: print "A", f
457 for f in a: print "A", f
458 for f in d: print "R", f
458 for f in d: print "R", f
459 for f in u: print "?", f
459 for f in u: print "?", f
460
460
461 def tags(ui, repo):
461 def tags(ui, repo):
462 """list repository tags"""
462 """list repository tags"""
463 repo.lookup(0) # prime the cache
463 repo.lookup(0) # prime the cache
464 i = repo.tags.items()
464 i = repo.tags.items()
465 n = []
465 n = []
466 for e in i:
466 for e in i:
467 try:
467 try:
468 l = repo.changelog.rev(e[1])
468 l = repo.changelog.rev(e[1])
469 except KeyError:
469 except KeyError:
470 l = -2
470 l = -2
471 n.append((l, e))
471 n.append((l, e))
472
472
473 n.sort()
473 n.sort()
474 n.reverse()
474 n.reverse()
475 i = [ e[1] for e in n ]
475 i = [ e[1] for e in n ]
476 for k, n in i:
476 for k, n in i:
477 try:
477 try:
478 r = repo.changelog.rev(n)
478 r = repo.changelog.rev(n)
479 except KeyError:
479 except KeyError:
480 r = "?"
480 r = "?"
481 print "%-30s %5d:%s" % (k, repo.changelog.rev(n), hg.hex(n))
481 print "%-30s %5d:%s" % (k, repo.changelog.rev(n), hg.hex(n))
482
482
483 def tip(ui, repo):
483 def tip(ui, repo):
484 """show the tip revision"""
484 """show the tip revision"""
485 n = repo.changelog.tip()
485 n = repo.changelog.tip()
486 t = repo.changelog.rev(n)
486 t = repo.changelog.rev(n)
487 ui.status("%d:%s\n" % (t, hg.hex(n)))
487 ui.status("%d:%s\n" % (t, hg.hex(n)))
488
488
489 def undo(ui, repo):
489 def undo(ui, repo):
490 """undo the last transaction"""
490 """undo the last transaction"""
491 repo.undo()
491 repo.undo()
492
492
493 def update(ui, repo, node=None, merge=False, clean=False):
493 def update(ui, repo, node=None, merge=False, clean=False):
494 '''update or merge working directory
494 '''update or merge working directory
495
495
496 If there are no outstanding changes in the working directory and
496 If there are no outstanding changes in the working directory and
497 there is a linear relationship between the current version and the
497 there is a linear relationship between the current version and the
498 requested version, the result is the requested version.
498 requested version, the result is the requested version.
499
499
500 Otherwise the result is a merge between the contents of the
500 Otherwise the result is a merge between the contents of the
501 current working directory and the requested version. Files that
501 current working directory and the requested version. Files that
502 changed between either parent are marked as changed for the next
502 changed between either parent are marked as changed for the next
503 commit and a commit must be performed before any further updates
503 commit and a commit must be performed before any further updates
504 are allowed.
504 are allowed.
505 '''
505 '''
506 node = node and repo.lookup(node) or repo.changelog.tip()
506 node = node and repo.lookup(node) or repo.changelog.tip()
507 return repo.update(node, allow=merge, force=clean)
507 return repo.update(node, allow=merge, force=clean)
508
508
509 def verify(ui, repo):
509 def verify(ui, repo):
510 """verify the integrity of the repository"""
510 """verify the integrity of the repository"""
511 return repo.verify()
511 return repo.verify()
512
512
513 # Command options and aliases are listed here, alphabetically
513 # Command options and aliases are listed here, alphabetically
514
514
515 table = {
515 table = {
516 "add": (add, [], "hg add [files]"),
516 "add": (add, [], "hg add [files]"),
517 "addremove": (addremove, [], "hg addremove"),
517 "addremove": (addremove, [], "hg addremove"),
518 "ann|annotate": (annotate,
518 "ann|annotate": (annotate,
519 [('r', 'revision', '', 'revision'),
519 [('r', 'revision', '', 'revision'),
520 ('u', 'user', None, 'show user'),
520 ('u', 'user', None, 'show user'),
521 ('n', 'number', None, 'show revision number'),
521 ('n', 'number', None, 'show revision number'),
522 ('c', 'changeset', None, 'show changeset')],
522 ('c', 'changeset', None, 'show changeset')],
523 'hg annotate [-u] [-c] [-n] [-r id] [files]'),
523 'hg annotate [-u] [-c] [-n] [-r id] [files]'),
524 "cat|dump": (cat, [], 'hg cat <file> [rev]'),
524 "cat|dump": (cat, [], 'hg cat <file> [rev]'),
525 "commit|ci": (commit,
525 "commit|ci": (commit,
526 [('t', 'text', "", 'commit text'),
526 [('t', 'text', "", 'commit text'),
527 ('l', 'logfile', "", 'commit text file')],
527 ('l', 'logfile', "", 'commit text file')],
528 'hg commit [files]'),
528 'hg commit [files]'),
529 "debugaddchangegroup": (debugaddchangegroup, [], 'debugaddchangegroup'),
529 "debugaddchangegroup": (debugaddchangegroup, [], 'debugaddchangegroup'),
530 "debugchangegroup": (debugchangegroup, [], 'debugchangegroup [roots]'),
530 "debugchangegroup": (debugchangegroup, [], 'debugchangegroup [roots]'),
531 "debugindex": (debugindex, [], 'debugindex <file>'),
531 "debugindex": (debugindex, [], 'debugindex <file>'),
532 "debugindexdot": (debugindexdot, [], 'debugindexdot <file>'),
532 "debugindexdot": (debugindexdot, [], 'debugindexdot <file>'),
533 "diff": (diff, [('r', 'rev', [], 'revision')],
533 "diff": (diff, [('r', 'rev', [], 'revision')],
534 'hg diff [-r A] [-r B] [files]'),
534 'hg diff [-r A] [-r B] [files]'),
535 "export": (export, [], "hg export <changeset>"),
535 "export": (export, [], "hg export <changeset>"),
536 "forget": (forget, [], "hg forget [files]"),
536 "forget": (forget, [], "hg forget [files]"),
537 "heads": (heads, [], 'hg heads'),
537 "heads": (heads, [], 'hg heads'),
538 "history": (history, [], 'hg history'),
538 "history": (history, [], 'hg history'),
539 "help": (help, [], 'hg help [command]'),
539 "help": (help, [], 'hg help [command]'),
540 "init": (init, [], 'hg init [url]'),
540 "init": (init, [], 'hg init [url]'),
541 "log": (log, [], 'hg log <file>'),
541 "log": (log, [], 'hg log <file>'),
542 "manifest|dumpmanifest": (manifest, [], 'hg manifest [rev]'),
542 "manifest|dumpmanifest": (manifest, [], 'hg manifest [rev]'),
543 "parents": (parents, [], 'hg parents [node]'),
543 "parents": (parents, [], 'hg parents [node]'),
544 "patch|import": (patch,
544 "patch|import": (patch,
545 [('p', 'strip', 1, 'path strip'),
545 [('p', 'strip', 1, 'path strip'),
546 ('b', 'base', "", 'base path'),
546 ('b', 'base', "", 'base path'),
547 ('q', 'quiet', "", 'silence diff')],
547 ('q', 'quiet', "", 'silence diff')],
548 "hg import [options] patches"),
548 "hg import [options] patches"),
549 "pull|merge": (pull, [], 'hg pull [source]'),
549 "pull|merge": (pull, [], 'hg pull [source]'),
550 "rawcommit": (rawcommit,
550 "rawcommit": (rawcommit,
551 [('p', 'parent', [], 'parent'),
551 [('p', 'parent', [], 'parent'),
552 ('d', 'date', "", 'data'),
552 ('d', 'date', "", 'data'),
553 ('u', 'user', "", 'user'),
553 ('u', 'user', "", 'user'),
554 ('F', 'files', "", 'file list'),
554 ('F', 'files', "", 'file list'),
555 ('t', 'text', "", 'commit text'),
555 ('t', 'text', "", 'commit text'),
556 ('l', 'logfile', "", 'commit text file')],
556 ('l', 'logfile', "", 'commit text file')],
557 'hg rawcommit [options] [files]'),
557 'hg rawcommit [options] [files]'),
558 "recover": (recover, [], "hg recover"),
558 "recover": (recover, [], "hg recover"),
559 "remove": (remove, [], "hg remove [files]"),
559 "remove": (remove, [], "hg remove [files]"),
560 "serve": (serve, [('p', 'port', 8000, 'listen port'),
560 "serve": (serve, [('p', 'port', 8000, 'listen port'),
561 ('a', 'address', '', 'interface address'),
561 ('a', 'address', '', 'interface address'),
562 ('n', 'name', os.getcwd(), 'repository name'),
562 ('n', 'name', os.getcwd(), 'repository name'),
563 ('t', 'templates', "", 'template map')],
563 ('t', 'templates', "", 'template map')],
564 "hg serve [options]"),
564 "hg serve [options]"),
565 "status": (status, [], 'hg status'),
565 "status": (status, [], 'hg status'),
566 "tags": (tags, [], 'hg tags'),
566 "tags": (tags, [], 'hg tags'),
567 "tip": (tip, [], 'hg tip'),
567 "tip": (tip, [], 'hg tip'),
568 "undo": (undo, [], 'hg undo'),
568 "undo": (undo, [], 'hg undo'),
569 "update|up|checkout|co|resolve": (update,
569 "update|up|checkout|co|resolve": (update,
570 [('m', 'merge', None,
570 [('m', 'merge', None,
571 'allow merging of conflicts'),
571 'allow merging of conflicts'),
572 ('C', 'clean', None,
572 ('C', 'clean', None,
573 'overwrite locally modified files')],
573 'overwrite locally modified files')],
574 'hg update [options] [node]'),
574 'hg update [options] [node]'),
575 "verify": (verify, [], 'hg verify'),
575 "verify": (verify, [], 'hg verify'),
576 }
576 }
577
577
578 norepo = "init branch help debugindex debugindexdot"
578 norepo = "init branch help debugindex debugindexdot"
579
579
580 def find(cmd):
580 def find(cmd):
581 i = None
581 i = None
582 for e in table.keys():
582 for e in table.keys():
583 if re.match(e + "$", cmd):
583 if re.match(e + "$", cmd):
584 return table[e]
584 return table[e]
585
585
586 raise UnknownCommand(cmd)
586 raise UnknownCommand(cmd)
587
587
588 class SignalInterrupt(Exception): pass
588 class SignalInterrupt(Exception): pass
589
589
590 def catchterm(*args):
590 def catchterm(*args):
591 raise SignalInterrupt
591 raise SignalInterrupt
592
592
593 def run():
593 def run():
594 sys.exit(dispatch(sys.argv[1:]))
594 sys.exit(dispatch(sys.argv[1:]))
595
595
596 def dispatch(args):
596 def dispatch(args):
597 options = {}
597 options = {}
598 opts = [('v', 'verbose', None, 'verbose'),
598 opts = [('v', 'verbose', None, 'verbose'),
599 ('d', 'debug', None, 'debug'),
599 ('d', 'debug', None, 'debug'),
600 ('q', 'quiet', None, 'quiet'),
600 ('q', 'quiet', None, 'quiet'),
601 ('p', 'profile', None, 'profile'),
601 ('p', 'profile', None, 'profile'),
602 ('y', 'noninteractive', None, 'run non-interactively'),
602 ('y', 'noninteractive', None, 'run non-interactively'),
603 ]
603 ]
604
604
605 args = fancyopts.fancyopts(args, opts, options,
605 args = fancyopts.fancyopts(args, opts, options,
606 'hg [options] <command> [options] [files]')
606 'hg [options] <command> [options] [files]')
607
607
608 if not args:
608 if not args:
609 cmd = "help"
609 cmd = "help"
610 else:
610 else:
611 cmd, args = args[0], args[1:]
611 cmd, args = args[0], args[1:]
612
612
613 u = ui.ui(options["verbose"], options["debug"], options["quiet"],
613 u = ui.ui(options["verbose"], options["debug"], options["quiet"],
614 not options["noninteractive"])
614 not options["noninteractive"])
615
615
616 try:
616 try:
617 i = find(cmd)
617 i = find(cmd)
618 except UnknownCommand:
618 except UnknownCommand:
619 u.warn("hg: unknown command '%s'\n" % cmd)
619 u.warn("hg: unknown command '%s'\n" % cmd)
620 help(u)
620 help(u)
621 sys.exit(1)
621 sys.exit(1)
622
622
623 signal.signal(signal.SIGTERM, catchterm)
623 signal.signal(signal.SIGTERM, catchterm)
624
624
625 cmdoptions = {}
625 cmdoptions = {}
626 try:
626 try:
627 args = fancyopts.fancyopts(args, i[1], cmdoptions, i[2])
627 args = fancyopts.fancyopts(args, i[1], cmdoptions, i[2])
628 except fancyopts.getopt.GetoptError, inst:
628 except fancyopts.getopt.GetoptError, inst:
629 u.warn("hg %s: %s\n" % (cmd, inst))
629 u.warn("hg %s: %s\n" % (cmd, inst))
630 help(u, cmd)
630 help(u, cmd)
631 sys.exit(-1)
631 sys.exit(-1)
632
632
633 if cmd not in norepo.split():
633 if cmd not in norepo.split():
634 repo = hg.repository(ui = u)
634 repo = hg.repository(ui = u)
635 d = lambda: i[0](u, repo, *args, **cmdoptions)
635 d = lambda: i[0](u, repo, *args, **cmdoptions)
636 else:
636 else:
637 d = lambda: i[0](u, *args, **cmdoptions)
637 d = lambda: i[0](u, *args, **cmdoptions)
638
638
639 try:
639 try:
640 if options['profile']:
640 if options['profile']:
641 import hotshot, hotshot.stats
641 import hotshot, hotshot.stats
642 prof = hotshot.Profile("hg.prof")
642 prof = hotshot.Profile("hg.prof")
643 r = prof.runcall(d)
643 r = prof.runcall(d)
644 prof.close()
644 prof.close()
645 stats = hotshot.stats.load("hg.prof")
645 stats = hotshot.stats.load("hg.prof")
646 stats.strip_dirs()
646 stats.strip_dirs()
647 stats.sort_stats('time', 'calls')
647 stats.sort_stats('time', 'calls')
648 stats.print_stats(40)
648 stats.print_stats(40)
649 return r
649 return r
650 else:
650 else:
651 return d()
651 return d()
652 except SignalInterrupt:
652 except SignalInterrupt:
653 u.warn("killed!\n")
653 u.warn("killed!\n")
654 except KeyboardInterrupt:
654 except KeyboardInterrupt:
655 u.warn("interrupted!\n")
655 u.warn("interrupted!\n")
656 except IOError, inst:
656 except IOError, inst:
657 if inst.errno == 32:
657 if inst.errno == 32:
658 u.warn("broken pipe\n")
658 u.warn("broken pipe\n")
659 else:
659 else:
660 raise
660 raise
661 except TypeError, inst:
661 except TypeError, inst:
662 # was this an argument error?
662 # was this an argument error?
663 tb = traceback.extract_tb(sys.exc_info()[2])
663 tb = traceback.extract_tb(sys.exc_info()[2])
664 if len(tb) > 2: # no
664 if len(tb) > 2: # no
665 raise
665 raise
666 u.debug(inst, "\n")
666 u.debug(inst, "\n")
667 u.warn("%s: invalid arguments\n" % i[0].__name__)
667 u.warn("%s: invalid arguments\n" % i[0].__name__)
668 help(u, cmd)
668 help(u, cmd)
669 sys.exit(-1)
669 sys.exit(-1)
670
670
@@ -1,1300 +1,1300 b''
1 # hg.py - repository classes for mercurial
1 # hg.py - repository classes for mercurial
2 #
2 #
3 # Copyright 2005 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005 Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms
5 # This software may be used and distributed according to the terms
6 # of the GNU General Public License, incorporated herein by reference.
6 # of the GNU General Public License, incorporated herein by reference.
7
7
8 import sys, struct, os
8 import sys, struct, os
9 from revlog import *
9 from revlog import *
10 from demandload import *
10 from demandload import *
11 demandload(globals(), "re lock urllib urllib2 transaction time socket")
11 demandload(globals(), "re lock urllib urllib2 transaction time socket")
12 demandload(globals(), "tempfile byterange difflib")
12 demandload(globals(), "tempfile byterange difflib")
13
13
14 def is_exec(f):
14 def is_exec(f):
15 return (os.stat(f).st_mode & 0100 != 0)
15 return (os.stat(f).st_mode & 0100 != 0)
16
16
17 def set_exec(f, mode):
17 def set_exec(f, mode):
18 s = os.stat(f).st_mode
18 s = os.stat(f).st_mode
19 if (s & 0100 != 0) == mode:
19 if (s & 0100 != 0) == mode:
20 return
20 return
21 if mode:
21 if mode:
22 # Turn on +x for every +r bit when making a file executable
22 # Turn on +x for every +r bit when making a file executable
23 # and obey umask.
23 # and obey umask.
24 umask = os.umask(0)
24 umask = os.umask(0)
25 os.umask(umask)
25 os.umask(umask)
26 os.chmod(f, s | (s & 0444) >> 2 & ~umask)
26 os.chmod(f, s | (s & 0444) >> 2 & ~umask)
27 else:
27 else:
28 os.chmod(f, s & 0666)
28 os.chmod(f, s & 0666)
29
29
30 class filelog(revlog):
30 class filelog(revlog):
31 def __init__(self, opener, path):
31 def __init__(self, opener, path):
32 revlog.__init__(self, opener,
32 revlog.__init__(self, opener,
33 os.path.join("data", path + ".i"),
33 os.path.join("data", path + ".i"),
34 os.path.join("data", path + ".d"))
34 os.path.join("data", path + ".d"))
35
35
36 def read(self, node):
36 def read(self, node):
37 return self.revision(node)
37 return self.revision(node)
38 def add(self, text, transaction, link, p1=None, p2=None):
38 def add(self, text, transaction, link, p1=None, p2=None):
39 return self.addrevision(text, transaction, link, p1, p2)
39 return self.addrevision(text, transaction, link, p1, p2)
40
40
41 def annotate(self, node):
41 def annotate(self, node):
42
42
43 def decorate(text, rev):
43 def decorate(text, rev):
44 return [(rev, l) for l in text.splitlines(1)]
44 return [(rev, l) for l in text.splitlines(1)]
45
45
46 def strip(annotation):
46 def strip(annotation):
47 return [e[1] for e in annotation]
47 return [e[1] for e in annotation]
48
48
49 def pair(parent, child):
49 def pair(parent, child):
50 new = []
50 new = []
51 sm = difflib.SequenceMatcher(None, strip(parent), strip(child))
51 sm = difflib.SequenceMatcher(None, strip(parent), strip(child))
52 for o, m, n, s, t in sm.get_opcodes():
52 for o, m, n, s, t in sm.get_opcodes():
53 if o == 'equal':
53 if o == 'equal':
54 new += parent[m:n]
54 new += parent[m:n]
55 else:
55 else:
56 new += child[s:t]
56 new += child[s:t]
57 return new
57 return new
58
58
59 # find all ancestors
59 # find all ancestors
60 needed = {node:1}
60 needed = {node:1}
61 visit = [node]
61 visit = [node]
62 while visit:
62 while visit:
63 n = visit.pop(0)
63 n = visit.pop(0)
64 for p in self.parents(n):
64 for p in self.parents(n):
65 if p not in needed:
65 if p not in needed:
66 needed[p] = 1
66 needed[p] = 1
67 visit.append(p)
67 visit.append(p)
68 else:
68 else:
69 # count how many times we'll use this
69 # count how many times we'll use this
70 needed[p] += 1
70 needed[p] += 1
71
71
72 # sort by revision which is a topological order
72 # sort by revision which is a topological order
73 visit = needed.keys()
73 visit = needed.keys()
74 visit = [ (self.rev(n), n) for n in visit ]
74 visit = [ (self.rev(n), n) for n in visit ]
75 visit.sort()
75 visit.sort()
76 visit = [ p[1] for p in visit ]
76 visit = [ p[1] for p in visit ]
77 hist = {}
77 hist = {}
78
78
79 for n in visit:
79 for n in visit:
80 curr = decorate(self.read(n), self.linkrev(n))
80 curr = decorate(self.read(n), self.linkrev(n))
81 for p in self.parents(n):
81 for p in self.parents(n):
82 if p != nullid:
82 if p != nullid:
83 curr = pair(hist[p], curr)
83 curr = pair(hist[p], curr)
84 # trim the history of unneeded revs
84 # trim the history of unneeded revs
85 needed[p] -= 1
85 needed[p] -= 1
86 if not needed[p]:
86 if not needed[p]:
87 del hist[p]
87 del hist[p]
88 hist[n] = curr
88 hist[n] = curr
89
89
90 return hist[n]
90 return hist[n]
91
91
92 class manifest(revlog):
92 class manifest(revlog):
93 def __init__(self, opener):
93 def __init__(self, opener):
94 self.mapcache = None
94 self.mapcache = None
95 self.listcache = None
95 self.listcache = None
96 self.addlist = None
96 self.addlist = None
97 revlog.__init__(self, opener, "00manifest.i", "00manifest.d")
97 revlog.__init__(self, opener, "00manifest.i", "00manifest.d")
98
98
99 def read(self, node):
99 def read(self, node):
100 if self.mapcache and self.mapcache[0] == node:
100 if self.mapcache and self.mapcache[0] == node:
101 return self.mapcache[1].copy()
101 return self.mapcache[1].copy()
102 text = self.revision(node)
102 text = self.revision(node)
103 map = {}
103 map = {}
104 flag = {}
104 flag = {}
105 self.listcache = (text, text.splitlines(1))
105 self.listcache = (text, text.splitlines(1))
106 for l in self.listcache[1]:
106 for l in self.listcache[1]:
107 (f, n) = l.split('\0')
107 (f, n) = l.split('\0')
108 map[f] = bin(n[:40])
108 map[f] = bin(n[:40])
109 flag[f] = (n[40:-1] == "x")
109 flag[f] = (n[40:-1] == "x")
110 self.mapcache = (node, map, flag)
110 self.mapcache = (node, map, flag)
111 return map
111 return map
112
112
113 def readflags(self, node):
113 def readflags(self, node):
114 if self.mapcache or self.mapcache[0] != node:
114 if self.mapcache or self.mapcache[0] != node:
115 self.read(node)
115 self.read(node)
116 return self.mapcache[2]
116 return self.mapcache[2]
117
117
118 def diff(self, a, b):
118 def diff(self, a, b):
119 # this is sneaky, as we're not actually using a and b
119 # this is sneaky, as we're not actually using a and b
120 if self.listcache and self.addlist and self.listcache[0] == a:
120 if self.listcache and self.addlist and self.listcache[0] == a:
121 d = mdiff.diff(self.listcache[1], self.addlist, 1)
121 d = mdiff.diff(self.listcache[1], self.addlist, 1)
122 if mdiff.patch(a, d) != b:
122 if mdiff.patch(a, d) != b:
123 sys.stderr.write("*** sortdiff failed, falling back ***\n")
123 sys.stderr.write("*** sortdiff failed, falling back ***\n")
124 return mdiff.textdiff(a, b)
124 return mdiff.textdiff(a, b)
125 return d
125 return d
126 else:
126 else:
127 return mdiff.textdiff(a, b)
127 return mdiff.textdiff(a, b)
128
128
129 def add(self, map, flags, transaction, link, p1=None, p2=None):
129 def add(self, map, flags, transaction, link, p1=None, p2=None):
130 files = map.keys()
130 files = map.keys()
131 files.sort()
131 files.sort()
132
132
133 self.addlist = ["%s\000%s%s\n" %
133 self.addlist = ["%s\000%s%s\n" %
134 (f, hex(map[f]), flags[f] and "x" or '')
134 (f, hex(map[f]), flags[f] and "x" or '')
135 for f in files]
135 for f in files]
136 text = "".join(self.addlist)
136 text = "".join(self.addlist)
137
137
138 n = self.addrevision(text, transaction, link, p1, p2)
138 n = self.addrevision(text, transaction, link, p1, p2)
139 self.mapcache = (n, map, flags)
139 self.mapcache = (n, map, flags)
140 self.listcache = (text, self.addlist)
140 self.listcache = (text, self.addlist)
141 self.addlist = None
141 self.addlist = None
142
142
143 return n
143 return n
144
144
145 class changelog(revlog):
145 class changelog(revlog):
146 def __init__(self, opener):
146 def __init__(self, opener):
147 revlog.__init__(self, opener, "00changelog.i", "00changelog.d")
147 revlog.__init__(self, opener, "00changelog.i", "00changelog.d")
148
148
149 def extract(self, text):
149 def extract(self, text):
150 if not text:
150 if not text:
151 return (nullid, "", "0", [], "")
151 return (nullid, "", "0", [], "")
152 last = text.index("\n\n")
152 last = text.index("\n\n")
153 desc = text[last + 2:]
153 desc = text[last + 2:]
154 l = text[:last].splitlines()
154 l = text[:last].splitlines()
155 manifest = bin(l[0])
155 manifest = bin(l[0])
156 user = l[1]
156 user = l[1]
157 date = l[2]
157 date = l[2]
158 files = l[3:]
158 files = l[3:]
159 return (manifest, user, date, files, desc)
159 return (manifest, user, date, files, desc)
160
160
161 def read(self, node):
161 def read(self, node):
162 return self.extract(self.revision(node))
162 return self.extract(self.revision(node))
163
163
164 def add(self, manifest, list, desc, transaction, p1=None, p2=None,
164 def add(self, manifest, list, desc, transaction, p1=None, p2=None,
165 user=None, date=None):
165 user=None, date=None):
166 user = (user or
166 user = (user or
167 os.environ.get("HGUSER") or
167 os.environ.get("HGUSER") or
168 os.environ.get("EMAIL") or
168 os.environ.get("EMAIL") or
169 os.environ.get("LOGNAME", "unknown") + '@' + socket.getfqdn())
169 os.environ.get("LOGNAME", "unknown") + '@' + socket.getfqdn())
170 date = date or "%d %d" % (time.time(), time.timezone)
170 date = date or "%d %d" % (time.time(), time.timezone)
171 list.sort()
171 list.sort()
172 l = [hex(manifest), user, date] + list + ["", desc]
172 l = [hex(manifest), user, date] + list + ["", desc]
173 text = "\n".join(l)
173 text = "\n".join(l)
174 return self.addrevision(text, transaction, self.count(), p1, p2)
174 return self.addrevision(text, transaction, self.count(), p1, p2)
175
175
176 class dirstate:
176 class dirstate:
177 def __init__(self, opener, ui, root):
177 def __init__(self, opener, ui, root):
178 self.opener = opener
178 self.opener = opener
179 self.root = root
179 self.root = root
180 self.dirty = 0
180 self.dirty = 0
181 self.ui = ui
181 self.ui = ui
182 self.map = None
182 self.map = None
183 self.pl = None
183 self.pl = None
184
184
185 def __del__(self):
185 def __del__(self):
186 if self.dirty:
186 if self.dirty:
187 self.write()
187 self.write()
188
188
189 def __getitem__(self, key):
189 def __getitem__(self, key):
190 try:
190 try:
191 return self.map[key]
191 return self.map[key]
192 except TypeError:
192 except TypeError:
193 self.read()
193 self.read()
194 return self[key]
194 return self[key]
195
195
196 def __contains__(self, key):
196 def __contains__(self, key):
197 if not self.map: self.read()
197 if not self.map: self.read()
198 return key in self.map
198 return key in self.map
199
199
200 def parents(self):
200 def parents(self):
201 if not self.pl:
201 if not self.pl:
202 self.read()
202 self.read()
203 return self.pl
203 return self.pl
204
204
205 def setparents(self, p1, p2 = nullid):
205 def setparents(self, p1, p2 = nullid):
206 self.dirty = 1
206 self.dirty = 1
207 self.pl = p1, p2
207 self.pl = p1, p2
208
208
209 def state(self, key):
209 def state(self, key):
210 try:
210 try:
211 return self[key][0]
211 return self[key][0]
212 except KeyError:
212 except KeyError:
213 return "?"
213 return "?"
214
214
215 def read(self):
215 def read(self):
216 if self.map is not None: return self.map
216 if self.map is not None: return self.map
217
217
218 self.map = {}
218 self.map = {}
219 self.pl = [nullid, nullid]
219 self.pl = [nullid, nullid]
220 try:
220 try:
221 st = self.opener("dirstate").read()
221 st = self.opener("dirstate").read()
222 if not st: return
222 if not st: return
223 except: return
223 except: return
224
224
225 self.pl = [st[:20], st[20: 40]]
225 self.pl = [st[:20], st[20: 40]]
226
226
227 pos = 40
227 pos = 40
228 while pos < len(st):
228 while pos < len(st):
229 e = struct.unpack(">cllll", st[pos:pos+17])
229 e = struct.unpack(">cllll", st[pos:pos+17])
230 l = e[4]
230 l = e[4]
231 pos += 17
231 pos += 17
232 f = st[pos:pos + l]
232 f = st[pos:pos + l]
233 self.map[f] = e[:4]
233 self.map[f] = e[:4]
234 pos += l
234 pos += l
235
235
236 def update(self, files, state):
236 def update(self, files, state):
237 ''' current states:
237 ''' current states:
238 n normal
238 n normal
239 m needs merging
239 m needs merging
240 r marked for removal
240 r marked for removal
241 a marked for addition'''
241 a marked for addition'''
242
242
243 if not files: return
243 if not files: return
244 self.read()
244 self.read()
245 self.dirty = 1
245 self.dirty = 1
246 for f in files:
246 for f in files:
247 if state == "r":
247 if state == "r":
248 self.map[f] = ('r', 0, 0, 0)
248 self.map[f] = ('r', 0, 0, 0)
249 else:
249 else:
250 s = os.stat(os.path.join(self.root, f))
250 s = os.stat(os.path.join(self.root, f))
251 self.map[f] = (state, s.st_mode, s.st_size, s.st_mtime)
251 self.map[f] = (state, s.st_mode, s.st_size, s.st_mtime)
252
252
253 def forget(self, files):
253 def forget(self, files):
254 if not files: return
254 if not files: return
255 self.read()
255 self.read()
256 self.dirty = 1
256 self.dirty = 1
257 for f in files:
257 for f in files:
258 try:
258 try:
259 del self.map[f]
259 del self.map[f]
260 except KeyError:
260 except KeyError:
261 self.ui.warn("not in dirstate: %s!\n" % f)
261 self.ui.warn("not in dirstate: %s!\n" % f)
262 pass
262 pass
263
263
264 def clear(self):
264 def clear(self):
265 self.map = {}
265 self.map = {}
266 self.dirty = 1
266 self.dirty = 1
267
267
268 def write(self):
268 def write(self):
269 st = self.opener("dirstate", "w")
269 st = self.opener("dirstate", "w")
270 st.write("".join(self.pl))
270 st.write("".join(self.pl))
271 for f, e in self.map.items():
271 for f, e in self.map.items():
272 e = struct.pack(">cllll", e[0], e[1], e[2], e[3], len(f))
272 e = struct.pack(">cllll", e[0], e[1], e[2], e[3], len(f))
273 st.write(e + f)
273 st.write(e + f)
274 self.dirty = 0
274 self.dirty = 0
275
275
276 def copy(self):
276 def copy(self):
277 self.read()
277 self.read()
278 return self.map.copy()
278 return self.map.copy()
279
279
280 # used to avoid circular references so destructors work
280 # used to avoid circular references so destructors work
281 def opener(base):
281 def opener(base):
282 p = base
282 p = base
283 def o(path, mode="r"):
283 def o(path, mode="r"):
284 if p[:7] == "http://":
284 if p[:7] == "http://":
285 f = os.path.join(p, urllib.quote(path))
285 f = os.path.join(p, urllib.quote(path))
286 return httprangereader(f)
286 return httprangereader(f)
287
287
288 f = os.path.join(p, path)
288 f = os.path.join(p, path)
289
289
290 mode += "b" # for that other OS
290 mode += "b" # for that other OS
291
291
292 if mode[0] != "r":
292 if mode[0] != "r":
293 try:
293 try:
294 s = os.stat(f)
294 s = os.stat(f)
295 except OSError:
295 except OSError:
296 d = os.path.dirname(f)
296 d = os.path.dirname(f)
297 if not os.path.isdir(d):
297 if not os.path.isdir(d):
298 os.makedirs(d)
298 os.makedirs(d)
299 else:
299 else:
300 if s.st_nlink > 1:
300 if s.st_nlink > 1:
301 file(f + ".tmp", "w").write(file(f).read())
301 file(f + ".tmp", "w").write(file(f).read())
302 os.rename(f+".tmp", f)
302 os.rename(f+".tmp", f)
303
303
304 return file(f, mode)
304 return file(f, mode)
305
305
306 return o
306 return o
307
307
308 class localrepository:
308 class localrepository:
309 def __init__(self, ui, path=None, create=0):
309 def __init__(self, ui, path=None, create=0):
310 self.remote = 0
310 self.remote = 0
311 if path and path[:7] == "http://":
311 if path and path[:7] == "http://":
312 self.remote = 1
312 self.remote = 1
313 self.path = path
313 self.path = path
314 else:
314 else:
315 if not path:
315 if not path:
316 p = os.getcwd()
316 p = os.getcwd()
317 while not os.path.isdir(os.path.join(p, ".hg")):
317 while not os.path.isdir(os.path.join(p, ".hg")):
318 p = os.path.dirname(p)
318 p = os.path.dirname(p)
319 if p == "/": raise "No repo found"
319 if p == "/": raise "No repo found"
320 path = p
320 path = p
321 self.path = os.path.join(path, ".hg")
321 self.path = os.path.join(path, ".hg")
322
322
323 self.root = path
323 self.root = path
324 self.ui = ui
324 self.ui = ui
325
325
326 if create:
326 if create:
327 os.mkdir(self.path)
327 os.mkdir(self.path)
328 os.mkdir(self.join("data"))
328 os.mkdir(self.join("data"))
329
329
330 self.opener = opener(self.path)
330 self.opener = opener(self.path)
331 self.wopener = opener(self.root)
331 self.wopener = opener(self.root)
332 self.manifest = manifest(self.opener)
332 self.manifest = manifest(self.opener)
333 self.changelog = changelog(self.opener)
333 self.changelog = changelog(self.opener)
334 self.ignorelist = None
334 self.ignorelist = None
335 self.tags = None
335 self.tags = None
336
336
337 if not self.remote:
337 if not self.remote:
338 self.dirstate = dirstate(self.opener, ui, self.root)
338 self.dirstate = dirstate(self.opener, ui, self.root)
339
339
340 def ignore(self, f):
340 def ignore(self, f):
341 if self.ignorelist is None:
341 if self.ignorelist is None:
342 self.ignorelist = []
342 self.ignorelist = []
343 try:
343 try:
344 l = self.wfile(".hgignore")
344 l = self.wfile(".hgignore")
345 for pat in l:
345 for pat in l:
346 if pat != "\n":
346 if pat != "\n":
347 self.ignorelist.append(re.compile(pat[:-1]))
347 self.ignorelist.append(re.compile(pat[:-1]))
348 except IOError: pass
348 except IOError: pass
349 for pat in self.ignorelist:
349 for pat in self.ignorelist:
350 if pat.search(f): return True
350 if pat.search(f): return True
351 return False
351 return False
352
352
353 def lookup(self, key):
353 def lookup(self, key):
354 if self.tags is None:
354 if self.tags is None:
355 self.tags = {}
355 self.tags = {}
356 try:
356 try:
357 # read each head of the tags file, ending with the tip
357 # read each head of the tags file, ending with the tip
358 # and add each tag found to the map, with "newer" ones
358 # and add each tag found to the map, with "newer" ones
359 # taking precedence
359 # taking precedence
360 fl = self.file(".hgtags")
360 fl = self.file(".hgtags")
361 h = fl.heads()
361 h = fl.heads()
362 h.reverse()
362 h.reverse()
363 for r in h:
363 for r in h:
364 for l in fl.revision(r).splitlines():
364 for l in fl.revision(r).splitlines():
365 if l:
365 if l:
366 n, k = l.split(" ")
366 n, k = l.split(" ")
367 self.tags[k] = bin(n)
367 self.tags[k] = bin(n)
368 except KeyError: pass
368 except KeyError: pass
369 self.tags['tip'] = self.changelog.tip()
369 self.tags['tip'] = self.changelog.tip()
370 try:
370 try:
371 return self.tags[key]
371 return self.tags[key]
372 except KeyError:
372 except KeyError:
373 return self.changelog.lookup(key)
373 return self.changelog.lookup(key)
374
374
375 def join(self, f):
375 def join(self, f):
376 return os.path.join(self.path, f)
376 return os.path.join(self.path, f)
377
377
378 def wjoin(self, f):
378 def wjoin(self, f):
379 return os.path.join(self.root, f)
379 return os.path.join(self.root, f)
380
380
381 def file(self, f):
381 def file(self, f):
382 if f[0] == '/': f = f[1:]
382 if f[0] == '/': f = f[1:]
383 return filelog(self.opener, f)
383 return filelog(self.opener, f)
384
384
385 def wfile(self, f, mode='r'):
385 def wfile(self, f, mode='r'):
386 return self.wopener(f, mode)
386 return self.wopener(f, mode)
387
387
388 def transaction(self):
388 def transaction(self):
389 # save dirstate for undo
389 # save dirstate for undo
390 try:
390 try:
391 ds = self.opener("dirstate").read()
391 ds = self.opener("dirstate").read()
392 except IOError:
392 except IOError:
393 ds = ""
393 ds = ""
394 self.opener("undo.dirstate", "w").write(ds)
394 self.opener("undo.dirstate", "w").write(ds)
395
395
396 return transaction.transaction(self.opener, self.join("journal"),
396 return transaction.transaction(self.opener, self.join("journal"),
397 self.join("undo"))
397 self.join("undo"))
398
398
399 def recover(self):
399 def recover(self):
400 lock = self.lock()
400 lock = self.lock()
401 if os.path.exists(self.join("recover")):
401 if os.path.exists(self.join("recover")):
402 self.ui.status("attempting to rollback interrupted transaction\n")
402 self.ui.status("attempting to rollback interrupted transaction\n")
403 return transaction.rollback(self.opener, self.join("recover"))
403 return transaction.rollback(self.opener, self.join("recover"))
404 else:
404 else:
405 self.ui.warn("no interrupted transaction available\n")
405 self.ui.warn("no interrupted transaction available\n")
406
406
407 def undo(self):
407 def undo(self):
408 lock = self.lock()
408 lock = self.lock()
409 if os.path.exists(self.join("undo")):
409 if os.path.exists(self.join("undo")):
410 self.ui.status("attempting to rollback last transaction\n")
410 self.ui.status("attempting to rollback last transaction\n")
411 transaction.rollback(self.opener, self.join("undo"))
411 transaction.rollback(self.opener, self.join("undo"))
412 self.dirstate = None
412 self.dirstate = None
413 os.rename(self.join("undo.dirstate"), self.join("dirstate"))
413 os.rename(self.join("undo.dirstate"), self.join("dirstate"))
414 self.dirstate = dirstate(self.opener, self.ui, self.root)
414 self.dirstate = dirstate(self.opener, self.ui, self.root)
415 else:
415 else:
416 self.ui.warn("no undo information available\n")
416 self.ui.warn("no undo information available\n")
417
417
418 def lock(self, wait = 1):
418 def lock(self, wait = 1):
419 try:
419 try:
420 return lock.lock(self.join("lock"), 0)
420 return lock.lock(self.join("lock"), 0)
421 except lock.LockHeld, inst:
421 except lock.LockHeld, inst:
422 if wait:
422 if wait:
423 self.ui.warn("waiting for lock held by %s\n" % inst.args[0])
423 self.ui.warn("waiting for lock held by %s\n" % inst.args[0])
424 return lock.lock(self.join("lock"), wait)
424 return lock.lock(self.join("lock"), wait)
425 raise inst
425 raise inst
426
426
427 def rawcommit(self, files, text, user, date, p1=None, p2=None):
427 def rawcommit(self, files, text, user, date, p1=None, p2=None):
428 p1 = p1 or self.dirstate.parents()[0] or nullid
428 p1 = p1 or self.dirstate.parents()[0] or nullid
429 p2 = p2 or self.dirstate.parents()[1] or nullid
429 p2 = p2 or self.dirstate.parents()[1] or nullid
430 c1 = self.changelog.read(p1)
430 c1 = self.changelog.read(p1)
431 c2 = self.changelog.read(p2)
431 c2 = self.changelog.read(p2)
432 m1 = self.manifest.read(c1[0])
432 m1 = self.manifest.read(c1[0])
433 mf1 = self.manifest.readflags(c1[0])
433 mf1 = self.manifest.readflags(c1[0])
434 m2 = self.manifest.read(c2[0])
434 m2 = self.manifest.read(c2[0])
435
435
436 tr = self.transaction()
436 tr = self.transaction()
437 mm = m1.copy()
437 mm = m1.copy()
438 mfm = mf1.copy()
438 mfm = mf1.copy()
439 linkrev = self.changelog.count()
439 linkrev = self.changelog.count()
440 for f in files:
440 for f in files:
441 try:
441 try:
442 t = self.wfile(f).read()
442 t = self.wfile(f).read()
443 tm = is_exec(self.wjoin(f))
443 tm = is_exec(self.wjoin(f))
444 r = self.file(f)
444 r = self.file(f)
445 mfm[f] = tm
445 mfm[f] = tm
446 mm[f] = r.add(t, tr, linkrev,
446 mm[f] = r.add(t, tr, linkrev,
447 m1.get(f, nullid), m2.get(f, nullid))
447 m1.get(f, nullid), m2.get(f, nullid))
448 except IOError:
448 except IOError:
449 del mm[f]
449 del mm[f]
450 del mfm[f]
450 del mfm[f]
451
451
452 mnode = self.manifest.add(mm, mfm, tr, linkrev, c1[0], c2[0])
452 mnode = self.manifest.add(mm, mfm, tr, linkrev, c1[0], c2[0])
453 n = self.changelog.add(mnode, files, text, tr, p1, p2, user, date)
453 n = self.changelog.add(mnode, files, text, tr, p1, p2, user, date)
454 tr.close()
454 tr.close()
455 self.dirstate.setparents(p1, p2)
455 self.dirstate.setparents(p1, p2)
456 self.dirstate.clear()
456 self.dirstate.clear()
457 self.dirstate.update(files, "n")
457 self.dirstate.update(files, "n")
458
458
459 def commit(self, files = None, text = ""):
459 def commit(self, files = None, text = ""):
460 commit = []
460 commit = []
461 remove = []
461 remove = []
462 if files:
462 if files:
463 for f in files:
463 for f in files:
464 s = self.dirstate.state(f)
464 s = self.dirstate.state(f)
465 if s in 'nmai':
465 if s in 'nmai':
466 commit.append(f)
466 commit.append(f)
467 elif s == 'r':
467 elif s == 'r':
468 remove.append(f)
468 remove.append(f)
469 else:
469 else:
470 self.ui.warn("%s not tracked!\n" % f)
470 self.ui.warn("%s not tracked!\n" % f)
471 else:
471 else:
472 (c, a, d, u) = self.diffdir(self.root)
472 (c, a, d, u) = self.diffdir(self.root)
473 commit = c + a
473 commit = c + a
474 remove = d
474 remove = d
475
475
476 if not commit and not remove:
476 if not commit and not remove:
477 self.ui.status("nothing changed\n")
477 self.ui.status("nothing changed\n")
478 return
478 return
479
479
480 p1, p2 = self.dirstate.parents()
480 p1, p2 = self.dirstate.parents()
481 c1 = self.changelog.read(p1)
481 c1 = self.changelog.read(p1)
482 c2 = self.changelog.read(p2)
482 c2 = self.changelog.read(p2)
483 m1 = self.manifest.read(c1[0])
483 m1 = self.manifest.read(c1[0])
484 mf1 = self.manifest.readflags(c1[0])
484 mf1 = self.manifest.readflags(c1[0])
485 m2 = self.manifest.read(c2[0])
485 m2 = self.manifest.read(c2[0])
486 lock = self.lock()
486 lock = self.lock()
487 tr = self.transaction()
487 tr = self.transaction()
488
488
489 # check in files
489 # check in files
490 new = {}
490 new = {}
491 linkrev = self.changelog.count()
491 linkrev = self.changelog.count()
492 commit.sort()
492 commit.sort()
493 for f in commit:
493 for f in commit:
494 self.ui.note(f + "\n")
494 self.ui.note(f + "\n")
495 try:
495 try:
496 fp = self.wjoin(f)
496 fp = self.wjoin(f)
497 mf1[f] = is_exec(fp)
497 mf1[f] = is_exec(fp)
498 t = file(fp).read()
498 t = file(fp).read()
499 except IOError:
499 except IOError:
500 self.warn("trouble committing %s!\n" % f)
500 self.warn("trouble committing %s!\n" % f)
501 raise
501 raise
502
502
503 r = self.file(f)
503 r = self.file(f)
504 fp1 = m1.get(f, nullid)
504 fp1 = m1.get(f, nullid)
505 fp2 = m2.get(f, nullid)
505 fp2 = m2.get(f, nullid)
506 new[f] = r.add(t, tr, linkrev, fp1, fp2)
506 new[f] = r.add(t, tr, linkrev, fp1, fp2)
507
507
508 # update manifest
508 # update manifest
509 m1.update(new)
509 m1.update(new)
510 for f in remove: del m1[f]
510 for f in remove: del m1[f]
511 mn = self.manifest.add(m1, mf1, tr, linkrev, c1[0], c2[0])
511 mn = self.manifest.add(m1, mf1, tr, linkrev, c1[0], c2[0])
512
512
513 # add changeset
513 # add changeset
514 new = new.keys()
514 new = new.keys()
515 new.sort()
515 new.sort()
516
516
517 if not text:
517 if not text:
518 edittext = "\n" + "HG: manifest hash %s\n" % hex(mn)
518 edittext = "\n" + "HG: manifest hash %s\n" % hex(mn)
519 edittext += "".join(["HG: changed %s\n" % f for f in new])
519 edittext += "".join(["HG: changed %s\n" % f for f in new])
520 edittext += "".join(["HG: removed %s\n" % f for f in remove])
520 edittext += "".join(["HG: removed %s\n" % f for f in remove])
521 edittext = self.ui.edit(edittext)
521 edittext = self.ui.edit(edittext)
522 if not edittext.rstrip():
522 if not edittext.rstrip():
523 return 1
523 return 1
524 text = edittext
524 text = edittext
525
525
526 n = self.changelog.add(mn, new, text, tr, p1, p2)
526 n = self.changelog.add(mn, new, text, tr, p1, p2)
527 tr.close()
527 tr.close()
528
528
529 self.dirstate.setparents(n)
529 self.dirstate.setparents(n)
530 self.dirstate.update(new, "n")
530 self.dirstate.update(new, "n")
531 self.dirstate.forget(remove)
531 self.dirstate.forget(remove)
532
532
533 def diffdir(self, path, changeset = None):
533 def diffdir(self, path, changeset = None):
534 changed = []
534 changed = []
535 added = []
535 added = []
536 unknown = []
536 unknown = []
537 mf = {}
537 mf = {}
538
538
539 if changeset:
539 if changeset:
540 change = self.changelog.read(changeset)
540 change = self.changelog.read(changeset)
541 mf = self.manifest.read(change[0])
541 mf = self.manifest.read(change[0])
542 dc = dict.fromkeys(mf)
542 dc = dict.fromkeys(mf)
543 else:
543 else:
544 changeset = self.dirstate.parents()[0]
544 changeset = self.dirstate.parents()[0]
545 change = self.changelog.read(changeset)
545 change = self.changelog.read(changeset)
546 mf = self.manifest.read(change[0])
546 mf = self.manifest.read(change[0])
547 dc = self.dirstate.copy()
547 dc = self.dirstate.copy()
548
548
549 def fcmp(fn):
549 def fcmp(fn):
550 t1 = self.wfile(fn).read()
550 t1 = self.wfile(fn).read()
551 t2 = self.file(fn).revision(mf[fn])
551 t2 = self.file(fn).revision(mf[fn])
552 return cmp(t1, t2)
552 return cmp(t1, t2)
553
553
554 for dir, subdirs, files in os.walk(self.root):
554 for dir, subdirs, files in os.walk(path):
555 d = dir[len(self.root)+1:]
555 d = dir[len(self.root)+1:]
556 if ".hg" in subdirs: subdirs.remove(".hg")
556 if ".hg" in subdirs: subdirs.remove(".hg")
557
557
558 for f in files:
558 for f in files:
559 fn = os.path.join(d, f)
559 fn = os.path.join(d, f)
560 try: s = os.stat(os.path.join(self.root, fn))
560 try: s = os.stat(os.path.join(self.root, fn))
561 except: continue
561 except: continue
562 if fn in dc:
562 if fn in dc:
563 c = dc[fn]
563 c = dc[fn]
564 del dc[fn]
564 del dc[fn]
565 if not c:
565 if not c:
566 if fcmp(fn):
566 if fcmp(fn):
567 changed.append(fn)
567 changed.append(fn)
568 elif c[0] == 'm':
568 elif c[0] == 'm':
569 changed.append(fn)
569 changed.append(fn)
570 elif c[0] == 'a':
570 elif c[0] == 'a':
571 added.append(fn)
571 added.append(fn)
572 elif c[0] == 'r':
572 elif c[0] == 'r':
573 unknown.append(fn)
573 unknown.append(fn)
574 elif c[2] != s.st_size or (c[1] ^ s.st_mode) & 0100:
574 elif c[2] != s.st_size or (c[1] ^ s.st_mode) & 0100:
575 changed.append(fn)
575 changed.append(fn)
576 elif c[1] != s.st_mode or c[3] != s.st_mtime:
576 elif c[1] != s.st_mode or c[3] != s.st_mtime:
577 if fcmp(fn):
577 if fcmp(fn):
578 changed.append(fn)
578 changed.append(fn)
579 else:
579 else:
580 if self.ignore(fn): continue
580 if self.ignore(fn): continue
581 unknown.append(fn)
581 unknown.append(fn)
582
582
583 deleted = dc.keys()
583 deleted = dc.keys()
584 deleted.sort()
584 deleted.sort()
585
585
586 return (changed, added, deleted, unknown)
586 return (changed, added, deleted, unknown)
587
587
588 def diffrevs(self, node1, node2):
588 def diffrevs(self, node1, node2):
589 changed, added = [], []
589 changed, added = [], []
590
590
591 change = self.changelog.read(node1)
591 change = self.changelog.read(node1)
592 mf1 = self.manifest.read(change[0])
592 mf1 = self.manifest.read(change[0])
593 change = self.changelog.read(node2)
593 change = self.changelog.read(node2)
594 mf2 = self.manifest.read(change[0])
594 mf2 = self.manifest.read(change[0])
595
595
596 for fn in mf2:
596 for fn in mf2:
597 if mf1.has_key(fn):
597 if mf1.has_key(fn):
598 if mf1[fn] != mf2[fn]:
598 if mf1[fn] != mf2[fn]:
599 changed.append(fn)
599 changed.append(fn)
600 del mf1[fn]
600 del mf1[fn]
601 else:
601 else:
602 added.append(fn)
602 added.append(fn)
603
603
604 deleted = mf1.keys()
604 deleted = mf1.keys()
605 deleted.sort()
605 deleted.sort()
606
606
607 return (changed, added, deleted)
607 return (changed, added, deleted)
608
608
609 def add(self, list):
609 def add(self, list):
610 for f in list:
610 for f in list:
611 p = self.wjoin(f)
611 p = self.wjoin(f)
612 if not os.path.isfile(p):
612 if not os.path.isfile(p):
613 self.ui.warn("%s does not exist!\n" % f)
613 self.ui.warn("%s does not exist!\n" % f)
614 elif self.dirstate.state(f) == 'n':
614 elif self.dirstate.state(f) == 'n':
615 self.ui.warn("%s already tracked!\n" % f)
615 self.ui.warn("%s already tracked!\n" % f)
616 else:
616 else:
617 self.dirstate.update([f], "a")
617 self.dirstate.update([f], "a")
618
618
619 def forget(self, list):
619 def forget(self, list):
620 for f in list:
620 for f in list:
621 if self.dirstate.state(f) not in 'ai':
621 if self.dirstate.state(f) not in 'ai':
622 self.ui.warn("%s not added!\n" % f)
622 self.ui.warn("%s not added!\n" % f)
623 else:
623 else:
624 self.dirstate.forget([f])
624 self.dirstate.forget([f])
625
625
626 def remove(self, list):
626 def remove(self, list):
627 for f in list:
627 for f in list:
628 p = self.wjoin(f)
628 p = self.wjoin(f)
629 if os.path.isfile(p):
629 if os.path.isfile(p):
630 self.ui.warn("%s still exists!\n" % f)
630 self.ui.warn("%s still exists!\n" % f)
631 elif f not in self.dirstate:
631 elif f not in self.dirstate:
632 self.ui.warn("%s not tracked!\n" % f)
632 self.ui.warn("%s not tracked!\n" % f)
633 else:
633 else:
634 self.dirstate.update([f], "r")
634 self.dirstate.update([f], "r")
635
635
636 def heads(self):
636 def heads(self):
637 return self.changelog.heads()
637 return self.changelog.heads()
638
638
639 def branches(self, nodes):
639 def branches(self, nodes):
640 if not nodes: nodes = [self.changelog.tip()]
640 if not nodes: nodes = [self.changelog.tip()]
641 b = []
641 b = []
642 for n in nodes:
642 for n in nodes:
643 t = n
643 t = n
644 while n:
644 while n:
645 p = self.changelog.parents(n)
645 p = self.changelog.parents(n)
646 if p[1] != nullid or p[0] == nullid:
646 if p[1] != nullid or p[0] == nullid:
647 b.append((t, n, p[0], p[1]))
647 b.append((t, n, p[0], p[1]))
648 break
648 break
649 n = p[0]
649 n = p[0]
650 return b
650 return b
651
651
652 def between(self, pairs):
652 def between(self, pairs):
653 r = []
653 r = []
654
654
655 for top, bottom in pairs:
655 for top, bottom in pairs:
656 n, l, i = top, [], 0
656 n, l, i = top, [], 0
657 f = 1
657 f = 1
658
658
659 while n != bottom:
659 while n != bottom:
660 p = self.changelog.parents(n)[0]
660 p = self.changelog.parents(n)[0]
661 if i == f:
661 if i == f:
662 l.append(n)
662 l.append(n)
663 f = f * 2
663 f = f * 2
664 n = p
664 n = p
665 i += 1
665 i += 1
666
666
667 r.append(l)
667 r.append(l)
668
668
669 return r
669 return r
670
670
671 def newer(self, nodes):
671 def newer(self, nodes):
672 m = {}
672 m = {}
673 nl = []
673 nl = []
674 pm = {}
674 pm = {}
675 cl = self.changelog
675 cl = self.changelog
676 t = l = cl.count()
676 t = l = cl.count()
677
677
678 # find the lowest numbered node
678 # find the lowest numbered node
679 for n in nodes:
679 for n in nodes:
680 l = min(l, cl.rev(n))
680 l = min(l, cl.rev(n))
681 m[n] = 1
681 m[n] = 1
682
682
683 for i in xrange(l, t):
683 for i in xrange(l, t):
684 n = cl.node(i)
684 n = cl.node(i)
685 if n in m: # explicitly listed
685 if n in m: # explicitly listed
686 pm[n] = 1
686 pm[n] = 1
687 nl.append(n)
687 nl.append(n)
688 continue
688 continue
689 for p in cl.parents(n):
689 for p in cl.parents(n):
690 if p in pm: # parent listed
690 if p in pm: # parent listed
691 pm[n] = 1
691 pm[n] = 1
692 nl.append(n)
692 nl.append(n)
693 break
693 break
694
694
695 return nl
695 return nl
696
696
697 def getchangegroup(self, remote):
697 def getchangegroup(self, remote):
698 m = self.changelog.nodemap
698 m = self.changelog.nodemap
699 search = []
699 search = []
700 fetch = []
700 fetch = []
701 seen = {}
701 seen = {}
702 seenbranch = {}
702 seenbranch = {}
703
703
704 # if we have an empty repo, fetch everything
704 # if we have an empty repo, fetch everything
705 if self.changelog.tip() == nullid:
705 if self.changelog.tip() == nullid:
706 self.ui.status("requesting all changes\n")
706 self.ui.status("requesting all changes\n")
707 return remote.changegroup([nullid])
707 return remote.changegroup([nullid])
708
708
709 # otherwise, assume we're closer to the tip than the root
709 # otherwise, assume we're closer to the tip than the root
710 self.ui.status("searching for changes\n")
710 self.ui.status("searching for changes\n")
711 heads = remote.heads()
711 heads = remote.heads()
712 unknown = []
712 unknown = []
713 for h in heads:
713 for h in heads:
714 if h not in m:
714 if h not in m:
715 unknown.append(h)
715 unknown.append(h)
716
716
717 if not unknown:
717 if not unknown:
718 self.ui.status("nothing to do!\n")
718 self.ui.status("nothing to do!\n")
719 return None
719 return None
720
720
721 unknown = remote.branches(unknown)
721 unknown = remote.branches(unknown)
722 while unknown:
722 while unknown:
723 n = unknown.pop(0)
723 n = unknown.pop(0)
724 seen[n[0]] = 1
724 seen[n[0]] = 1
725
725
726 self.ui.debug("examining %s:%s\n" % (short(n[0]), short(n[1])))
726 self.ui.debug("examining %s:%s\n" % (short(n[0]), short(n[1])))
727 if n == nullid: break
727 if n == nullid: break
728 if n in seenbranch:
728 if n in seenbranch:
729 self.ui.debug("branch already found\n")
729 self.ui.debug("branch already found\n")
730 continue
730 continue
731 if n[1] and n[1] in m: # do we know the base?
731 if n[1] and n[1] in m: # do we know the base?
732 self.ui.debug("found incomplete branch %s:%s\n"
732 self.ui.debug("found incomplete branch %s:%s\n"
733 % (short(n[0]), short(n[1])))
733 % (short(n[0]), short(n[1])))
734 search.append(n) # schedule branch range for scanning
734 search.append(n) # schedule branch range for scanning
735 seenbranch[n] = 1
735 seenbranch[n] = 1
736 else:
736 else:
737 if n[2] in m and n[3] in m:
737 if n[2] in m and n[3] in m:
738 if n[1] not in fetch:
738 if n[1] not in fetch:
739 self.ui.debug("found new changeset %s\n" %
739 self.ui.debug("found new changeset %s\n" %
740 short(n[1]))
740 short(n[1]))
741 fetch.append(n[1]) # earliest unknown
741 fetch.append(n[1]) # earliest unknown
742 continue
742 continue
743
743
744 r = []
744 r = []
745 for a in n[2:4]:
745 for a in n[2:4]:
746 if a not in seen: r.append(a)
746 if a not in seen: r.append(a)
747
747
748 if r:
748 if r:
749 self.ui.debug("requesting %s\n" %
749 self.ui.debug("requesting %s\n" %
750 " ".join(map(short, r)))
750 " ".join(map(short, r)))
751 for b in remote.branches(r):
751 for b in remote.branches(r):
752 self.ui.debug("received %s:%s\n" %
752 self.ui.debug("received %s:%s\n" %
753 (short(b[0]), short(b[1])))
753 (short(b[0]), short(b[1])))
754 if b[0] not in m and b[0] not in seen:
754 if b[0] not in m and b[0] not in seen:
755 unknown.append(b)
755 unknown.append(b)
756
756
757 while search:
757 while search:
758 n = search.pop(0)
758 n = search.pop(0)
759 l = remote.between([(n[0], n[1])])[0]
759 l = remote.between([(n[0], n[1])])[0]
760 p = n[0]
760 p = n[0]
761 f = 1
761 f = 1
762 for i in l + [n[1]]:
762 for i in l + [n[1]]:
763 if i in m:
763 if i in m:
764 if f <= 2:
764 if f <= 2:
765 self.ui.debug("found new branch changeset %s\n" %
765 self.ui.debug("found new branch changeset %s\n" %
766 short(p))
766 short(p))
767 fetch.append(p)
767 fetch.append(p)
768 else:
768 else:
769 self.ui.debug("narrowed branch search to %s:%s\n"
769 self.ui.debug("narrowed branch search to %s:%s\n"
770 % (short(p), short(i)))
770 % (short(p), short(i)))
771 search.append((p, i))
771 search.append((p, i))
772 break
772 break
773 p, f = i, f * 2
773 p, f = i, f * 2
774
774
775 for f in fetch:
775 for f in fetch:
776 if f in m:
776 if f in m:
777 raise "already have", short(f[:4])
777 raise "already have", short(f[:4])
778
778
779 self.ui.note("adding new changesets starting at " +
779 self.ui.note("adding new changesets starting at " +
780 " ".join([short(f) for f in fetch]) + "\n")
780 " ".join([short(f) for f in fetch]) + "\n")
781
781
782 return remote.changegroup(fetch)
782 return remote.changegroup(fetch)
783
783
784 def changegroup(self, basenodes):
784 def changegroup(self, basenodes):
785 nodes = self.newer(basenodes)
785 nodes = self.newer(basenodes)
786
786
787 # construct the link map
787 # construct the link map
788 linkmap = {}
788 linkmap = {}
789 for n in nodes:
789 for n in nodes:
790 linkmap[self.changelog.rev(n)] = n
790 linkmap[self.changelog.rev(n)] = n
791
791
792 # construct a list of all changed files
792 # construct a list of all changed files
793 changed = {}
793 changed = {}
794 for n in nodes:
794 for n in nodes:
795 c = self.changelog.read(n)
795 c = self.changelog.read(n)
796 for f in c[3]:
796 for f in c[3]:
797 changed[f] = 1
797 changed[f] = 1
798 changed = changed.keys()
798 changed = changed.keys()
799 changed.sort()
799 changed.sort()
800
800
801 # the changegroup is changesets + manifests + all file revs
801 # the changegroup is changesets + manifests + all file revs
802 revs = [ self.changelog.rev(n) for n in nodes ]
802 revs = [ self.changelog.rev(n) for n in nodes ]
803
803
804 for y in self.changelog.group(linkmap): yield y
804 for y in self.changelog.group(linkmap): yield y
805 for y in self.manifest.group(linkmap): yield y
805 for y in self.manifest.group(linkmap): yield y
806 for f in changed:
806 for f in changed:
807 yield struct.pack(">l", len(f) + 4) + f
807 yield struct.pack(">l", len(f) + 4) + f
808 g = self.file(f).group(linkmap)
808 g = self.file(f).group(linkmap)
809 for y in g:
809 for y in g:
810 yield y
810 yield y
811
811
812 def addchangegroup(self, generator):
812 def addchangegroup(self, generator):
813
813
814 class genread:
814 class genread:
815 def __init__(self, generator):
815 def __init__(self, generator):
816 self.g = generator
816 self.g = generator
817 self.buf = ""
817 self.buf = ""
818 def read(self, l):
818 def read(self, l):
819 while l > len(self.buf):
819 while l > len(self.buf):
820 try:
820 try:
821 self.buf += self.g.next()
821 self.buf += self.g.next()
822 except StopIteration:
822 except StopIteration:
823 break
823 break
824 d, self.buf = self.buf[:l], self.buf[l:]
824 d, self.buf = self.buf[:l], self.buf[l:]
825 return d
825 return d
826
826
827 def getchunk():
827 def getchunk():
828 d = source.read(4)
828 d = source.read(4)
829 if not d: return ""
829 if not d: return ""
830 l = struct.unpack(">l", d)[0]
830 l = struct.unpack(">l", d)[0]
831 if l <= 4: return ""
831 if l <= 4: return ""
832 return source.read(l - 4)
832 return source.read(l - 4)
833
833
834 def getgroup():
834 def getgroup():
835 while 1:
835 while 1:
836 c = getchunk()
836 c = getchunk()
837 if not c: break
837 if not c: break
838 yield c
838 yield c
839
839
840 def csmap(x):
840 def csmap(x):
841 self.ui.debug("add changeset %s\n" % short(x))
841 self.ui.debug("add changeset %s\n" % short(x))
842 return self.changelog.count()
842 return self.changelog.count()
843
843
844 def revmap(x):
844 def revmap(x):
845 return self.changelog.rev(x)
845 return self.changelog.rev(x)
846
846
847 if not generator: return
847 if not generator: return
848 changesets = files = revisions = 0
848 changesets = files = revisions = 0
849
849
850 source = genread(generator)
850 source = genread(generator)
851 lock = self.lock()
851 lock = self.lock()
852 tr = self.transaction()
852 tr = self.transaction()
853
853
854 # pull off the changeset group
854 # pull off the changeset group
855 self.ui.status("adding changesets\n")
855 self.ui.status("adding changesets\n")
856 co = self.changelog.tip()
856 co = self.changelog.tip()
857 cn = self.changelog.addgroup(getgroup(), csmap, tr, 1) # unique
857 cn = self.changelog.addgroup(getgroup(), csmap, tr, 1) # unique
858 changesets = self.changelog.rev(cn) - self.changelog.rev(co)
858 changesets = self.changelog.rev(cn) - self.changelog.rev(co)
859
859
860 # pull off the manifest group
860 # pull off the manifest group
861 self.ui.status("adding manifests\n")
861 self.ui.status("adding manifests\n")
862 mm = self.manifest.tip()
862 mm = self.manifest.tip()
863 mo = self.manifest.addgroup(getgroup(), revmap, tr)
863 mo = self.manifest.addgroup(getgroup(), revmap, tr)
864
864
865 # process the files
865 # process the files
866 self.ui.status("adding file revisions\n")
866 self.ui.status("adding file revisions\n")
867 while 1:
867 while 1:
868 f = getchunk()
868 f = getchunk()
869 if not f: break
869 if not f: break
870 self.ui.debug("adding %s revisions\n" % f)
870 self.ui.debug("adding %s revisions\n" % f)
871 fl = self.file(f)
871 fl = self.file(f)
872 o = fl.tip()
872 o = fl.tip()
873 n = fl.addgroup(getgroup(), revmap, tr)
873 n = fl.addgroup(getgroup(), revmap, tr)
874 revisions += fl.rev(n) - fl.rev(o)
874 revisions += fl.rev(n) - fl.rev(o)
875 files += 1
875 files += 1
876
876
877 self.ui.status(("modified %d files, added %d changesets" +
877 self.ui.status(("modified %d files, added %d changesets" +
878 " and %d new revisions\n")
878 " and %d new revisions\n")
879 % (files, changesets, revisions))
879 % (files, changesets, revisions))
880
880
881 tr.close()
881 tr.close()
882 return
882 return
883
883
884 def update(self, node, allow=False, force=False):
884 def update(self, node, allow=False, force=False):
885 pl = self.dirstate.parents()
885 pl = self.dirstate.parents()
886 if not force and pl[1] != nullid:
886 if not force and pl[1] != nullid:
887 self.ui.warn("aborting: outstanding uncommitted merges\n")
887 self.ui.warn("aborting: outstanding uncommitted merges\n")
888 return
888 return
889
889
890 p1, p2 = pl[0], node
890 p1, p2 = pl[0], node
891 pa = self.changelog.ancestor(p1, p2)
891 pa = self.changelog.ancestor(p1, p2)
892 m1n = self.changelog.read(p1)[0]
892 m1n = self.changelog.read(p1)[0]
893 m2n = self.changelog.read(p2)[0]
893 m2n = self.changelog.read(p2)[0]
894 man = self.manifest.ancestor(m1n, m2n)
894 man = self.manifest.ancestor(m1n, m2n)
895 m1 = self.manifest.read(m1n)
895 m1 = self.manifest.read(m1n)
896 mf1 = self.manifest.readflags(m1n)
896 mf1 = self.manifest.readflags(m1n)
897 m2 = self.manifest.read(m2n)
897 m2 = self.manifest.read(m2n)
898 mf2 = self.manifest.readflags(m2n)
898 mf2 = self.manifest.readflags(m2n)
899 ma = self.manifest.read(man)
899 ma = self.manifest.read(man)
900 mfa = self.manifest.readflags(m2n)
900 mfa = self.manifest.readflags(m2n)
901
901
902 (c, a, d, u) = self.diffdir(self.root)
902 (c, a, d, u) = self.diffdir(self.root)
903
903
904 # resolve the manifest to determine which files
904 # resolve the manifest to determine which files
905 # we care about merging
905 # we care about merging
906 self.ui.note("resolving manifests\n")
906 self.ui.note("resolving manifests\n")
907 self.ui.debug(" ancestor %s local %s remote %s\n" %
907 self.ui.debug(" ancestor %s local %s remote %s\n" %
908 (short(man), short(m1n), short(m2n)))
908 (short(man), short(m1n), short(m2n)))
909
909
910 merge = {}
910 merge = {}
911 get = {}
911 get = {}
912 remove = []
912 remove = []
913 mark = {}
913 mark = {}
914
914
915 # construct a working dir manifest
915 # construct a working dir manifest
916 mw = m1.copy()
916 mw = m1.copy()
917 mfw = mf1.copy()
917 mfw = mf1.copy()
918 for f in a + c + u:
918 for f in a + c + u:
919 mw[f] = ""
919 mw[f] = ""
920 mfw[f] = is_exec(self.wjoin(f))
920 mfw[f] = is_exec(self.wjoin(f))
921 for f in d:
921 for f in d:
922 if f in mw: del mw[f]
922 if f in mw: del mw[f]
923
923
924 for f, n in mw.iteritems():
924 for f, n in mw.iteritems():
925 if f in m2:
925 if f in m2:
926 s = 0
926 s = 0
927
927
928 # are files different?
928 # are files different?
929 if n != m2[f]:
929 if n != m2[f]:
930 a = ma.get(f, nullid)
930 a = ma.get(f, nullid)
931 # are both different from the ancestor?
931 # are both different from the ancestor?
932 if n != a and m2[f] != a:
932 if n != a and m2[f] != a:
933 self.ui.debug(" %s versions differ, resolve\n" % f)
933 self.ui.debug(" %s versions differ, resolve\n" % f)
934 merge[f] = (m1.get(f, nullid), m2[f])
934 merge[f] = (m1.get(f, nullid), m2[f])
935 # merge executable bits
935 # merge executable bits
936 # "if we changed or they changed, change in merge"
936 # "if we changed or they changed, change in merge"
937 a, b, c = mfa.get(f, 0), mfw[f], mf2[f]
937 a, b, c = mfa.get(f, 0), mfw[f], mf2[f]
938 mode = ((a^b) | (a^c)) ^ a
938 mode = ((a^b) | (a^c)) ^ a
939 merge[f] = (m1.get(f, nullid), m2[f], mode)
939 merge[f] = (m1.get(f, nullid), m2[f], mode)
940 s = 1
940 s = 1
941 # are we clobbering?
941 # are we clobbering?
942 # is remote's version newer?
942 # is remote's version newer?
943 # or are we going back in time?
943 # or are we going back in time?
944 elif force or m2[f] != a or (p2 == pa and mw[f] == m1[f]):
944 elif force or m2[f] != a or (p2 == pa and mw[f] == m1[f]):
945 self.ui.debug(" remote %s is newer, get\n" % f)
945 self.ui.debug(" remote %s is newer, get\n" % f)
946 get[f] = m2[f]
946 get[f] = m2[f]
947 s = 1
947 s = 1
948 else:
948 else:
949 mark[f] = 1
949 mark[f] = 1
950
950
951 if not s and mfw[f] != mf2[f]:
951 if not s and mfw[f] != mf2[f]:
952 if force:
952 if force:
953 self.ui.debug(" updating permissions for %s\n" % f)
953 self.ui.debug(" updating permissions for %s\n" % f)
954 set_exec(self.wjoin(f), mf2[f])
954 set_exec(self.wjoin(f), mf2[f])
955 else:
955 else:
956 a, b, c = mfa.get(f, 0), mfw[f], mf2[f]
956 a, b, c = mfa.get(f, 0), mfw[f], mf2[f]
957 mode = ((a^b) | (a^c)) ^ a
957 mode = ((a^b) | (a^c)) ^ a
958 if mode != b:
958 if mode != b:
959 self.ui.debug(" updating permissions for %s\n" % f)
959 self.ui.debug(" updating permissions for %s\n" % f)
960 set_exec(self.wjoin(f), mode)
960 set_exec(self.wjoin(f), mode)
961 mark[f] = 1
961 mark[f] = 1
962 del m2[f]
962 del m2[f]
963 elif f in ma:
963 elif f in ma:
964 if not force and n != ma[f]:
964 if not force and n != ma[f]:
965 r = self.ui.prompt(
965 r = self.ui.prompt(
966 (" local changed %s which remote deleted\n" % f) +
966 (" local changed %s which remote deleted\n" % f) +
967 "(k)eep or (d)elete?", "[kd]", "k")
967 "(k)eep or (d)elete?", "[kd]", "k")
968 if r == "d":
968 if r == "d":
969 remove.append(f)
969 remove.append(f)
970 else:
970 else:
971 self.ui.debug("other deleted %s\n" % f)
971 self.ui.debug("other deleted %s\n" % f)
972 remove.append(f) # other deleted it
972 remove.append(f) # other deleted it
973 else:
973 else:
974 if n == m1.get(f, nullid): # same as parent
974 if n == m1.get(f, nullid): # same as parent
975 self.ui.debug("remote deleted %s\n" % f)
975 self.ui.debug("remote deleted %s\n" % f)
976 remove.append(f)
976 remove.append(f)
977 else:
977 else:
978 self.ui.debug("working dir created %s, keeping\n" % f)
978 self.ui.debug("working dir created %s, keeping\n" % f)
979
979
980 for f, n in m2.iteritems():
980 for f, n in m2.iteritems():
981 if f[0] == "/": continue
981 if f[0] == "/": continue
982 if not force and f in ma and n != ma[f]:
982 if not force and f in ma and n != ma[f]:
983 r = self.ui.prompt(
983 r = self.ui.prompt(
984 ("remote changed %s which local deleted\n" % f) +
984 ("remote changed %s which local deleted\n" % f) +
985 "(k)eep or (d)elete?", "[kd]", "k")
985 "(k)eep or (d)elete?", "[kd]", "k")
986 if r == "d": remove.append(f)
986 if r == "d": remove.append(f)
987 else:
987 else:
988 self.ui.debug("remote created %s\n" % f)
988 self.ui.debug("remote created %s\n" % f)
989 get[f] = n
989 get[f] = n
990
990
991 del mw, m1, m2, ma
991 del mw, m1, m2, ma
992
992
993 if force:
993 if force:
994 for f in merge:
994 for f in merge:
995 get[f] = merge[f][1]
995 get[f] = merge[f][1]
996 merge = {}
996 merge = {}
997
997
998 if pa == p1 or pa == p2:
998 if pa == p1 or pa == p2:
999 # we don't need to do any magic, just jump to the new rev
999 # we don't need to do any magic, just jump to the new rev
1000 mode = 'n'
1000 mode = 'n'
1001 p1, p2 = p2, nullid
1001 p1, p2 = p2, nullid
1002 else:
1002 else:
1003 if not allow:
1003 if not allow:
1004 self.ui.status("this update spans a branch" +
1004 self.ui.status("this update spans a branch" +
1005 " affecting the following files:\n")
1005 " affecting the following files:\n")
1006 fl = merge.keys() + get.keys()
1006 fl = merge.keys() + get.keys()
1007 fl.sort()
1007 fl.sort()
1008 for f in fl:
1008 for f in fl:
1009 cf = ""
1009 cf = ""
1010 if f in merge: cf = " (resolve)"
1010 if f in merge: cf = " (resolve)"
1011 self.ui.status(" %s%s\n" % (f, cf))
1011 self.ui.status(" %s%s\n" % (f, cf))
1012 self.ui.warn("aborting update spanning branches!\n")
1012 self.ui.warn("aborting update spanning branches!\n")
1013 self.ui.status("(use update -m to perform a branch merge)\n")
1013 self.ui.status("(use update -m to perform a branch merge)\n")
1014 return 1
1014 return 1
1015 # we have to remember what files we needed to get/change
1015 # we have to remember what files we needed to get/change
1016 # because any file that's different from either one of its
1016 # because any file that's different from either one of its
1017 # parents must be in the changeset
1017 # parents must be in the changeset
1018 mode = 'm'
1018 mode = 'm'
1019 self.dirstate.update(mark.keys(), "m")
1019 self.dirstate.update(mark.keys(), "m")
1020
1020
1021 self.dirstate.setparents(p1, p2)
1021 self.dirstate.setparents(p1, p2)
1022
1022
1023 # get the files we don't need to change
1023 # get the files we don't need to change
1024 files = get.keys()
1024 files = get.keys()
1025 files.sort()
1025 files.sort()
1026 for f in files:
1026 for f in files:
1027 if f[0] == "/": continue
1027 if f[0] == "/": continue
1028 self.ui.note("getting %s\n" % f)
1028 self.ui.note("getting %s\n" % f)
1029 t = self.file(f).read(get[f])
1029 t = self.file(f).read(get[f])
1030 try:
1030 try:
1031 self.wfile(f, "w").write(t)
1031 self.wfile(f, "w").write(t)
1032 except IOError:
1032 except IOError:
1033 os.makedirs(os.path.dirname(self.wjoin(f)))
1033 os.makedirs(os.path.dirname(self.wjoin(f)))
1034 self.wfile(f, "w").write(t)
1034 self.wfile(f, "w").write(t)
1035 set_exec(self.wjoin(f), mf2[f])
1035 set_exec(self.wjoin(f), mf2[f])
1036 self.dirstate.update([f], mode)
1036 self.dirstate.update([f], mode)
1037
1037
1038 # merge the tricky bits
1038 # merge the tricky bits
1039 files = merge.keys()
1039 files = merge.keys()
1040 files.sort()
1040 files.sort()
1041 for f in files:
1041 for f in files:
1042 self.ui.status("merging %s\n" % f)
1042 self.ui.status("merging %s\n" % f)
1043 m, o, flag = merge[f]
1043 m, o, flag = merge[f]
1044 self.merge3(f, m, o)
1044 self.merge3(f, m, o)
1045 set_exec(self.wjoin(f), flag)
1045 set_exec(self.wjoin(f), flag)
1046 self.dirstate.update([f], 'm')
1046 self.dirstate.update([f], 'm')
1047
1047
1048 for f in remove:
1048 for f in remove:
1049 self.ui.note("removing %s\n" % f)
1049 self.ui.note("removing %s\n" % f)
1050 os.unlink(f)
1050 os.unlink(f)
1051 if mode == 'n':
1051 if mode == 'n':
1052 self.dirstate.forget(remove)
1052 self.dirstate.forget(remove)
1053 else:
1053 else:
1054 self.dirstate.update(remove, 'r')
1054 self.dirstate.update(remove, 'r')
1055
1055
1056 def merge3(self, fn, my, other):
1056 def merge3(self, fn, my, other):
1057 """perform a 3-way merge in the working directory"""
1057 """perform a 3-way merge in the working directory"""
1058
1058
1059 def temp(prefix, node):
1059 def temp(prefix, node):
1060 pre = "%s~%s." % (os.path.basename(fn), prefix)
1060 pre = "%s~%s." % (os.path.basename(fn), prefix)
1061 (fd, name) = tempfile.mkstemp("", pre)
1061 (fd, name) = tempfile.mkstemp("", pre)
1062 f = os.fdopen(fd, "w")
1062 f = os.fdopen(fd, "w")
1063 f.write(fl.revision(node))
1063 f.write(fl.revision(node))
1064 f.close()
1064 f.close()
1065 return name
1065 return name
1066
1066
1067 fl = self.file(fn)
1067 fl = self.file(fn)
1068 base = fl.ancestor(my, other)
1068 base = fl.ancestor(my, other)
1069 a = self.wjoin(fn)
1069 a = self.wjoin(fn)
1070 b = temp("other", other)
1070 b = temp("other", other)
1071 c = temp("base", base)
1071 c = temp("base", base)
1072
1072
1073 self.ui.note("resolving %s\n" % fn)
1073 self.ui.note("resolving %s\n" % fn)
1074 self.ui.debug("file %s: other %s ancestor %s\n" %
1074 self.ui.debug("file %s: other %s ancestor %s\n" %
1075 (fn, short(other), short(base)))
1075 (fn, short(other), short(base)))
1076
1076
1077 cmd = os.environ.get("HGMERGE", "hgmerge")
1077 cmd = os.environ.get("HGMERGE", "hgmerge")
1078 r = os.system("%s %s %s %s" % (cmd, a, b, c))
1078 r = os.system("%s %s %s %s" % (cmd, a, b, c))
1079 if r:
1079 if r:
1080 self.ui.warn("merging %s failed!\n" % fn)
1080 self.ui.warn("merging %s failed!\n" % fn)
1081
1081
1082 os.unlink(b)
1082 os.unlink(b)
1083 os.unlink(c)
1083 os.unlink(c)
1084
1084
1085 def verify(self):
1085 def verify(self):
1086 filelinkrevs = {}
1086 filelinkrevs = {}
1087 filenodes = {}
1087 filenodes = {}
1088 changesets = revisions = files = 0
1088 changesets = revisions = files = 0
1089 errors = 0
1089 errors = 0
1090
1090
1091 seen = {}
1091 seen = {}
1092 self.ui.status("checking changesets\n")
1092 self.ui.status("checking changesets\n")
1093 for i in range(self.changelog.count()):
1093 for i in range(self.changelog.count()):
1094 changesets += 1
1094 changesets += 1
1095 n = self.changelog.node(i)
1095 n = self.changelog.node(i)
1096 if n in seen:
1096 if n in seen:
1097 self.ui.warn("duplicate changeset at revision %d\n" % i)
1097 self.ui.warn("duplicate changeset at revision %d\n" % i)
1098 errors += 1
1098 errors += 1
1099 seen[n] = 1
1099 seen[n] = 1
1100
1100
1101 for p in self.changelog.parents(n):
1101 for p in self.changelog.parents(n):
1102 if p not in self.changelog.nodemap:
1102 if p not in self.changelog.nodemap:
1103 self.ui.warn("changeset %s has unknown parent %s\n" %
1103 self.ui.warn("changeset %s has unknown parent %s\n" %
1104 (short(n), short(p)))
1104 (short(n), short(p)))
1105 errors += 1
1105 errors += 1
1106 try:
1106 try:
1107 changes = self.changelog.read(n)
1107 changes = self.changelog.read(n)
1108 except Exception, inst:
1108 except Exception, inst:
1109 self.ui.warn("unpacking changeset %s: %s\n" % (short(n), inst))
1109 self.ui.warn("unpacking changeset %s: %s\n" % (short(n), inst))
1110 errors += 1
1110 errors += 1
1111
1111
1112 for f in changes[3]:
1112 for f in changes[3]:
1113 filelinkrevs.setdefault(f, []).append(i)
1113 filelinkrevs.setdefault(f, []).append(i)
1114
1114
1115 seen = {}
1115 seen = {}
1116 self.ui.status("checking manifests\n")
1116 self.ui.status("checking manifests\n")
1117 for i in range(self.manifest.count()):
1117 for i in range(self.manifest.count()):
1118 n = self.manifest.node(i)
1118 n = self.manifest.node(i)
1119 if n in seen:
1119 if n in seen:
1120 self.ui.warn("duplicate manifest at revision %d\n" % i)
1120 self.ui.warn("duplicate manifest at revision %d\n" % i)
1121 errors += 1
1121 errors += 1
1122 seen[n] = 1
1122 seen[n] = 1
1123
1123
1124 for p in self.manifest.parents(n):
1124 for p in self.manifest.parents(n):
1125 if p not in self.manifest.nodemap:
1125 if p not in self.manifest.nodemap:
1126 self.ui.warn("manifest %s has unknown parent %s\n" %
1126 self.ui.warn("manifest %s has unknown parent %s\n" %
1127 (short(n), short(p)))
1127 (short(n), short(p)))
1128 errors += 1
1128 errors += 1
1129
1129
1130 try:
1130 try:
1131 delta = mdiff.patchtext(self.manifest.delta(n))
1131 delta = mdiff.patchtext(self.manifest.delta(n))
1132 except KeyboardInterrupt:
1132 except KeyboardInterrupt:
1133 print "aborted"
1133 print "aborted"
1134 sys.exit(0)
1134 sys.exit(0)
1135 except Exception, inst:
1135 except Exception, inst:
1136 self.ui.warn("unpacking manifest %s: %s\n"
1136 self.ui.warn("unpacking manifest %s: %s\n"
1137 % (short(n), inst))
1137 % (short(n), inst))
1138 errors += 1
1138 errors += 1
1139
1139
1140 ff = [ l.split('\0') for l in delta.splitlines() ]
1140 ff = [ l.split('\0') for l in delta.splitlines() ]
1141 for f, fn in ff:
1141 for f, fn in ff:
1142 filenodes.setdefault(f, {})[bin(fn[:40])] = 1
1142 filenodes.setdefault(f, {})[bin(fn[:40])] = 1
1143
1143
1144 self.ui.status("crosschecking files in changesets and manifests\n")
1144 self.ui.status("crosschecking files in changesets and manifests\n")
1145 for f in filenodes:
1145 for f in filenodes:
1146 if f not in filelinkrevs:
1146 if f not in filelinkrevs:
1147 self.ui.warn("file %s in manifest but not in changesets\n" % f)
1147 self.ui.warn("file %s in manifest but not in changesets\n" % f)
1148 errors += 1
1148 errors += 1
1149
1149
1150 for f in filelinkrevs:
1150 for f in filelinkrevs:
1151 if f not in filenodes:
1151 if f not in filenodes:
1152 self.ui.warn("file %s in changeset but not in manifest\n" % f)
1152 self.ui.warn("file %s in changeset but not in manifest\n" % f)
1153 errors += 1
1153 errors += 1
1154
1154
1155 self.ui.status("checking files\n")
1155 self.ui.status("checking files\n")
1156 ff = filenodes.keys()
1156 ff = filenodes.keys()
1157 ff.sort()
1157 ff.sort()
1158 for f in ff:
1158 for f in ff:
1159 if f == "/dev/null": continue
1159 if f == "/dev/null": continue
1160 files += 1
1160 files += 1
1161 fl = self.file(f)
1161 fl = self.file(f)
1162 nodes = { nullid: 1 }
1162 nodes = { nullid: 1 }
1163 seen = {}
1163 seen = {}
1164 for i in range(fl.count()):
1164 for i in range(fl.count()):
1165 revisions += 1
1165 revisions += 1
1166 n = fl.node(i)
1166 n = fl.node(i)
1167
1167
1168 if n in seen:
1168 if n in seen:
1169 self.ui.warn("%s: duplicate revision %d\n" % (f, i))
1169 self.ui.warn("%s: duplicate revision %d\n" % (f, i))
1170 errors += 1
1170 errors += 1
1171
1171
1172 if n not in filenodes[f]:
1172 if n not in filenodes[f]:
1173 self.ui.warn("%s: %d:%s not in manifests\n"
1173 self.ui.warn("%s: %d:%s not in manifests\n"
1174 % (f, i, short(n)))
1174 % (f, i, short(n)))
1175 print len(filenodes[f].keys()), fl.count(), f
1175 print len(filenodes[f].keys()), fl.count(), f
1176 errors += 1
1176 errors += 1
1177 else:
1177 else:
1178 del filenodes[f][n]
1178 del filenodes[f][n]
1179
1179
1180 flr = fl.linkrev(n)
1180 flr = fl.linkrev(n)
1181 if flr not in filelinkrevs[f]:
1181 if flr not in filelinkrevs[f]:
1182 self.ui.warn("%s:%s points to unexpected changeset %d\n"
1182 self.ui.warn("%s:%s points to unexpected changeset %d\n"
1183 % (f, short(n), fl.linkrev(n)))
1183 % (f, short(n), fl.linkrev(n)))
1184 errors += 1
1184 errors += 1
1185 else:
1185 else:
1186 filelinkrevs[f].remove(flr)
1186 filelinkrevs[f].remove(flr)
1187
1187
1188 # verify contents
1188 # verify contents
1189 try:
1189 try:
1190 t = fl.read(n)
1190 t = fl.read(n)
1191 except Exception, inst:
1191 except Exception, inst:
1192 self.ui.warn("unpacking file %s %s: %s\n"
1192 self.ui.warn("unpacking file %s %s: %s\n"
1193 % (f, short(n), inst))
1193 % (f, short(n), inst))
1194 errors += 1
1194 errors += 1
1195
1195
1196 # verify parents
1196 # verify parents
1197 (p1, p2) = fl.parents(n)
1197 (p1, p2) = fl.parents(n)
1198 if p1 not in nodes:
1198 if p1 not in nodes:
1199 self.ui.warn("file %s:%s unknown parent 1 %s" %
1199 self.ui.warn("file %s:%s unknown parent 1 %s" %
1200 (f, short(n), short(p1)))
1200 (f, short(n), short(p1)))
1201 errors += 1
1201 errors += 1
1202 if p2 not in nodes:
1202 if p2 not in nodes:
1203 self.ui.warn("file %s:%s unknown parent 2 %s" %
1203 self.ui.warn("file %s:%s unknown parent 2 %s" %
1204 (f, short(n), short(p1)))
1204 (f, short(n), short(p1)))
1205 errors += 1
1205 errors += 1
1206 nodes[n] = 1
1206 nodes[n] = 1
1207
1207
1208 # cross-check
1208 # cross-check
1209 for node in filenodes[f]:
1209 for node in filenodes[f]:
1210 self.ui.warn("node %s in manifests not in %s\n"
1210 self.ui.warn("node %s in manifests not in %s\n"
1211 % (hex(n), f))
1211 % (hex(n), f))
1212 errors += 1
1212 errors += 1
1213
1213
1214 self.ui.status("%d files, %d changesets, %d total revisions\n" %
1214 self.ui.status("%d files, %d changesets, %d total revisions\n" %
1215 (files, changesets, revisions))
1215 (files, changesets, revisions))
1216
1216
1217 if errors:
1217 if errors:
1218 self.ui.warn("%d integrity errors encountered!\n" % errors)
1218 self.ui.warn("%d integrity errors encountered!\n" % errors)
1219 return 1
1219 return 1
1220
1220
1221 class remoterepository:
1221 class remoterepository:
1222 def __init__(self, ui, path):
1222 def __init__(self, ui, path):
1223 self.url = path
1223 self.url = path
1224 self.ui = ui
1224 self.ui = ui
1225
1225
1226 def do_cmd(self, cmd, **args):
1226 def do_cmd(self, cmd, **args):
1227 self.ui.debug("sending %s command\n" % cmd)
1227 self.ui.debug("sending %s command\n" % cmd)
1228 q = {"cmd": cmd}
1228 q = {"cmd": cmd}
1229 q.update(args)
1229 q.update(args)
1230 qs = urllib.urlencode(q)
1230 qs = urllib.urlencode(q)
1231 cu = "%s?%s" % (self.url, qs)
1231 cu = "%s?%s" % (self.url, qs)
1232 return urllib.urlopen(cu)
1232 return urllib.urlopen(cu)
1233
1233
1234 def heads(self):
1234 def heads(self):
1235 d = self.do_cmd("heads").read()
1235 d = self.do_cmd("heads").read()
1236 try:
1236 try:
1237 return map(bin, d[:-1].split(" "))
1237 return map(bin, d[:-1].split(" "))
1238 except:
1238 except:
1239 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1239 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1240 raise
1240 raise
1241
1241
1242 def branches(self, nodes):
1242 def branches(self, nodes):
1243 n = " ".join(map(hex, nodes))
1243 n = " ".join(map(hex, nodes))
1244 d = self.do_cmd("branches", nodes=n).read()
1244 d = self.do_cmd("branches", nodes=n).read()
1245 try:
1245 try:
1246 br = [ tuple(map(bin, b.split(" "))) for b in d.splitlines() ]
1246 br = [ tuple(map(bin, b.split(" "))) for b in d.splitlines() ]
1247 return br
1247 return br
1248 except:
1248 except:
1249 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1249 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1250 raise
1250 raise
1251
1251
1252 def between(self, pairs):
1252 def between(self, pairs):
1253 n = "\n".join(["-".join(map(hex, p)) for p in pairs])
1253 n = "\n".join(["-".join(map(hex, p)) for p in pairs])
1254 d = self.do_cmd("between", pairs=n).read()
1254 d = self.do_cmd("between", pairs=n).read()
1255 try:
1255 try:
1256 p = [ l and map(bin, l.split(" ")) or [] for l in d.splitlines() ]
1256 p = [ l and map(bin, l.split(" ")) or [] for l in d.splitlines() ]
1257 return p
1257 return p
1258 except:
1258 except:
1259 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1259 self.ui.warn("unexpected response:\n" + d[:400] + "\n...\n")
1260 raise
1260 raise
1261
1261
1262 def changegroup(self, nodes):
1262 def changegroup(self, nodes):
1263 n = " ".join(map(hex, nodes))
1263 n = " ".join(map(hex, nodes))
1264 zd = zlib.decompressobj()
1264 zd = zlib.decompressobj()
1265 f = self.do_cmd("changegroup", roots=n)
1265 f = self.do_cmd("changegroup", roots=n)
1266 bytes = 0
1266 bytes = 0
1267 while 1:
1267 while 1:
1268 d = f.read(4096)
1268 d = f.read(4096)
1269 bytes += len(d)
1269 bytes += len(d)
1270 if not d:
1270 if not d:
1271 yield zd.flush()
1271 yield zd.flush()
1272 break
1272 break
1273 yield zd.decompress(d)
1273 yield zd.decompress(d)
1274 self.ui.note("%d bytes of data transfered\n" % bytes)
1274 self.ui.note("%d bytes of data transfered\n" % bytes)
1275
1275
1276 def repository(ui, path=None, create=0):
1276 def repository(ui, path=None, create=0):
1277 if path and path[:7] == "http://":
1277 if path and path[:7] == "http://":
1278 return remoterepository(ui, path)
1278 return remoterepository(ui, path)
1279 if path and path[:5] == "hg://":
1279 if path and path[:5] == "hg://":
1280 return remoterepository(ui, path.replace("hg://", "http://"))
1280 return remoterepository(ui, path.replace("hg://", "http://"))
1281 if path and path[:11] == "old-http://":
1281 if path and path[:11] == "old-http://":
1282 return localrepository(ui, path.replace("old-http://", "http://"))
1282 return localrepository(ui, path.replace("old-http://", "http://"))
1283 else:
1283 else:
1284 return localrepository(ui, path, create)
1284 return localrepository(ui, path, create)
1285
1285
1286 class httprangereader:
1286 class httprangereader:
1287 def __init__(self, url):
1287 def __init__(self, url):
1288 self.url = url
1288 self.url = url
1289 self.pos = 0
1289 self.pos = 0
1290 def seek(self, pos):
1290 def seek(self, pos):
1291 self.pos = pos
1291 self.pos = pos
1292 def read(self, bytes=None):
1292 def read(self, bytes=None):
1293 opener = urllib2.build_opener(byterange.HTTPRangeHandler())
1293 opener = urllib2.build_opener(byterange.HTTPRangeHandler())
1294 urllib2.install_opener(opener)
1294 urllib2.install_opener(opener)
1295 req = urllib2.Request(self.url)
1295 req = urllib2.Request(self.url)
1296 end = ''
1296 end = ''
1297 if bytes: end = self.pos + bytes
1297 if bytes: end = self.pos + bytes
1298 req.add_header('Range', 'bytes=%d-%s' % (self.pos, end))
1298 req.add_header('Range', 'bytes=%d-%s' % (self.pos, end))
1299 f = urllib2.urlopen(req)
1299 f = urllib2.urlopen(req)
1300 return f.read()
1300 return f.read()
General Comments 0
You need to be logged in to leave comments. Login now