|
@@
-1,2965
+1,2967
b''
|
|
1
|
# cmdutil.py - help for command processing in mercurial
|
|
1
|
# cmdutil.py - help for command processing in mercurial
|
|
2
|
#
|
|
2
|
#
|
|
3
|
# Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
|
|
3
|
# Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
|
|
4
|
#
|
|
4
|
#
|
|
5
|
# This software may be used and distributed according to the terms of the
|
|
5
|
# This software may be used and distributed according to the terms of the
|
|
6
|
# GNU General Public License version 2 or any later version.
|
|
6
|
# GNU General Public License version 2 or any later version.
|
|
7
|
|
|
7
|
|
|
8
|
from node import hex, nullid, nullrev, short
|
|
8
|
from node import hex, nullid, nullrev, short
|
|
9
|
from i18n import _
|
|
9
|
from i18n import _
|
|
10
|
import os, sys, errno, re, tempfile
|
|
10
|
import os, sys, errno, re, tempfile
|
|
11
|
import util, scmutil, templater, patch, error, templatekw, revlog, copies
|
|
11
|
import util, scmutil, templater, patch, error, templatekw, revlog, copies
|
|
12
|
import match as matchmod
|
|
12
|
import match as matchmod
|
|
13
|
import context, repair, graphmod, revset, phases, obsolete, pathutil
|
|
13
|
import context, repair, graphmod, revset, phases, obsolete, pathutil
|
|
14
|
import changelog
|
|
14
|
import changelog
|
|
15
|
import bookmarks
|
|
15
|
import bookmarks
|
|
16
|
import encoding
|
|
16
|
import encoding
|
|
17
|
import lock as lockmod
|
|
17
|
import lock as lockmod
|
|
18
|
|
|
18
|
|
|
19
|
def parsealiases(cmd):
|
|
19
|
def parsealiases(cmd):
|
|
20
|
return cmd.lstrip("^").split("|")
|
|
20
|
return cmd.lstrip("^").split("|")
|
|
21
|
|
|
21
|
|
|
22
|
def findpossible(cmd, table, strict=False):
|
|
22
|
def findpossible(cmd, table, strict=False):
|
|
23
|
"""
|
|
23
|
"""
|
|
24
|
Return cmd -> (aliases, command table entry)
|
|
24
|
Return cmd -> (aliases, command table entry)
|
|
25
|
for each matching command.
|
|
25
|
for each matching command.
|
|
26
|
Return debug commands (or their aliases) only if no normal command matches.
|
|
26
|
Return debug commands (or their aliases) only if no normal command matches.
|
|
27
|
"""
|
|
27
|
"""
|
|
28
|
choice = {}
|
|
28
|
choice = {}
|
|
29
|
debugchoice = {}
|
|
29
|
debugchoice = {}
|
|
30
|
|
|
30
|
|
|
31
|
if cmd in table:
|
|
31
|
if cmd in table:
|
|
32
|
# short-circuit exact matches, "log" alias beats "^log|history"
|
|
32
|
# short-circuit exact matches, "log" alias beats "^log|history"
|
|
33
|
keys = [cmd]
|
|
33
|
keys = [cmd]
|
|
34
|
else:
|
|
34
|
else:
|
|
35
|
keys = table.keys()
|
|
35
|
keys = table.keys()
|
|
36
|
|
|
36
|
|
|
37
|
for e in keys:
|
|
37
|
for e in keys:
|
|
38
|
aliases = parsealiases(e)
|
|
38
|
aliases = parsealiases(e)
|
|
39
|
found = None
|
|
39
|
found = None
|
|
40
|
if cmd in aliases:
|
|
40
|
if cmd in aliases:
|
|
41
|
found = cmd
|
|
41
|
found = cmd
|
|
42
|
elif not strict:
|
|
42
|
elif not strict:
|
|
43
|
for a in aliases:
|
|
43
|
for a in aliases:
|
|
44
|
if a.startswith(cmd):
|
|
44
|
if a.startswith(cmd):
|
|
45
|
found = a
|
|
45
|
found = a
|
|
46
|
break
|
|
46
|
break
|
|
47
|
if found is not None:
|
|
47
|
if found is not None:
|
|
48
|
if aliases[0].startswith("debug") or found.startswith("debug"):
|
|
48
|
if aliases[0].startswith("debug") or found.startswith("debug"):
|
|
49
|
debugchoice[found] = (aliases, table[e])
|
|
49
|
debugchoice[found] = (aliases, table[e])
|
|
50
|
else:
|
|
50
|
else:
|
|
51
|
choice[found] = (aliases, table[e])
|
|
51
|
choice[found] = (aliases, table[e])
|
|
52
|
|
|
52
|
|
|
53
|
if not choice and debugchoice:
|
|
53
|
if not choice and debugchoice:
|
|
54
|
choice = debugchoice
|
|
54
|
choice = debugchoice
|
|
55
|
|
|
55
|
|
|
56
|
return choice
|
|
56
|
return choice
|
|
57
|
|
|
57
|
|
|
58
|
def findcmd(cmd, table, strict=True):
|
|
58
|
def findcmd(cmd, table, strict=True):
|
|
59
|
"""Return (aliases, command table entry) for command string."""
|
|
59
|
"""Return (aliases, command table entry) for command string."""
|
|
60
|
choice = findpossible(cmd, table, strict)
|
|
60
|
choice = findpossible(cmd, table, strict)
|
|
61
|
|
|
61
|
|
|
62
|
if cmd in choice:
|
|
62
|
if cmd in choice:
|
|
63
|
return choice[cmd]
|
|
63
|
return choice[cmd]
|
|
64
|
|
|
64
|
|
|
65
|
if len(choice) > 1:
|
|
65
|
if len(choice) > 1:
|
|
66
|
clist = choice.keys()
|
|
66
|
clist = choice.keys()
|
|
67
|
clist.sort()
|
|
67
|
clist.sort()
|
|
68
|
raise error.AmbiguousCommand(cmd, clist)
|
|
68
|
raise error.AmbiguousCommand(cmd, clist)
|
|
69
|
|
|
69
|
|
|
70
|
if choice:
|
|
70
|
if choice:
|
|
71
|
return choice.values()[0]
|
|
71
|
return choice.values()[0]
|
|
72
|
|
|
72
|
|
|
73
|
raise error.UnknownCommand(cmd)
|
|
73
|
raise error.UnknownCommand(cmd)
|
|
74
|
|
|
74
|
|
|
75
|
def findrepo(p):
|
|
75
|
def findrepo(p):
|
|
76
|
while not os.path.isdir(os.path.join(p, ".hg")):
|
|
76
|
while not os.path.isdir(os.path.join(p, ".hg")):
|
|
77
|
oldp, p = p, os.path.dirname(p)
|
|
77
|
oldp, p = p, os.path.dirname(p)
|
|
78
|
if p == oldp:
|
|
78
|
if p == oldp:
|
|
79
|
return None
|
|
79
|
return None
|
|
80
|
|
|
80
|
|
|
81
|
return p
|
|
81
|
return p
|
|
82
|
|
|
82
|
|
|
83
|
def bailifchanged(repo):
|
|
83
|
def bailifchanged(repo):
|
|
84
|
if repo.dirstate.p2() != nullid:
|
|
84
|
if repo.dirstate.p2() != nullid:
|
|
85
|
raise util.Abort(_('outstanding uncommitted merge'))
|
|
85
|
raise util.Abort(_('outstanding uncommitted merge'))
|
|
86
|
modified, added, removed, deleted = repo.status()[:4]
|
|
86
|
modified, added, removed, deleted = repo.status()[:4]
|
|
87
|
if modified or added or removed or deleted:
|
|
87
|
if modified or added or removed or deleted:
|
|
88
|
raise util.Abort(_('uncommitted changes'))
|
|
88
|
raise util.Abort(_('uncommitted changes'))
|
|
89
|
ctx = repo[None]
|
|
89
|
ctx = repo[None]
|
|
90
|
for s in sorted(ctx.substate):
|
|
90
|
for s in sorted(ctx.substate):
|
|
91
|
if ctx.sub(s).dirty():
|
|
91
|
if ctx.sub(s).dirty():
|
|
92
|
raise util.Abort(_("uncommitted changes in subrepo %s") % s)
|
|
92
|
raise util.Abort(_("uncommitted changes in subrepo %s") % s)
|
|
93
|
|
|
93
|
|
|
94
|
def logmessage(ui, opts):
|
|
94
|
def logmessage(ui, opts):
|
|
95
|
""" get the log message according to -m and -l option """
|
|
95
|
""" get the log message according to -m and -l option """
|
|
96
|
message = opts.get('message')
|
|
96
|
message = opts.get('message')
|
|
97
|
logfile = opts.get('logfile')
|
|
97
|
logfile = opts.get('logfile')
|
|
98
|
|
|
98
|
|
|
99
|
if message and logfile:
|
|
99
|
if message and logfile:
|
|
100
|
raise util.Abort(_('options --message and --logfile are mutually '
|
|
100
|
raise util.Abort(_('options --message and --logfile are mutually '
|
|
101
|
'exclusive'))
|
|
101
|
'exclusive'))
|
|
102
|
if not message and logfile:
|
|
102
|
if not message and logfile:
|
|
103
|
try:
|
|
103
|
try:
|
|
104
|
if logfile == '-':
|
|
104
|
if logfile == '-':
|
|
105
|
message = ui.fin.read()
|
|
105
|
message = ui.fin.read()
|
|
106
|
else:
|
|
106
|
else:
|
|
107
|
message = '\n'.join(util.readfile(logfile).splitlines())
|
|
107
|
message = '\n'.join(util.readfile(logfile).splitlines())
|
|
108
|
except IOError, inst:
|
|
108
|
except IOError, inst:
|
|
109
|
raise util.Abort(_("can't read commit message '%s': %s") %
|
|
109
|
raise util.Abort(_("can't read commit message '%s': %s") %
|
|
110
|
(logfile, inst.strerror))
|
|
110
|
(logfile, inst.strerror))
|
|
111
|
return message
|
|
111
|
return message
|
|
112
|
|
|
112
|
|
|
113
|
def mergeeditform(ctxorbool, baseformname):
|
|
113
|
def mergeeditform(ctxorbool, baseformname):
|
|
114
|
"""return appropriate editform name (referencing a committemplate)
|
|
114
|
"""return appropriate editform name (referencing a committemplate)
|
|
115
|
|
|
115
|
|
|
116
|
'ctxorbool' is either a ctx to be committed, or a bool indicating whether
|
|
116
|
'ctxorbool' is either a ctx to be committed, or a bool indicating whether
|
|
117
|
merging is committed.
|
|
117
|
merging is committed.
|
|
118
|
|
|
118
|
|
|
119
|
This returns baseformname with '.merge' appended if it is a merge,
|
|
119
|
This returns baseformname with '.merge' appended if it is a merge,
|
|
120
|
otherwise '.normal' is appended.
|
|
120
|
otherwise '.normal' is appended.
|
|
121
|
"""
|
|
121
|
"""
|
|
122
|
if isinstance(ctxorbool, bool):
|
|
122
|
if isinstance(ctxorbool, bool):
|
|
123
|
if ctxorbool:
|
|
123
|
if ctxorbool:
|
|
124
|
return baseformname + ".merge"
|
|
124
|
return baseformname + ".merge"
|
|
125
|
elif 1 < len(ctxorbool.parents()):
|
|
125
|
elif 1 < len(ctxorbool.parents()):
|
|
126
|
return baseformname + ".merge"
|
|
126
|
return baseformname + ".merge"
|
|
127
|
|
|
127
|
|
|
128
|
return baseformname + ".normal"
|
|
128
|
return baseformname + ".normal"
|
|
129
|
|
|
129
|
|
|
130
|
def getcommiteditor(edit=False, finishdesc=None, extramsg=None,
|
|
130
|
def getcommiteditor(edit=False, finishdesc=None, extramsg=None,
|
|
131
|
editform='', **opts):
|
|
131
|
editform='', **opts):
|
|
132
|
"""get appropriate commit message editor according to '--edit' option
|
|
132
|
"""get appropriate commit message editor according to '--edit' option
|
|
133
|
|
|
133
|
|
|
134
|
'finishdesc' is a function to be called with edited commit message
|
|
134
|
'finishdesc' is a function to be called with edited commit message
|
|
135
|
(= 'description' of the new changeset) just after editing, but
|
|
135
|
(= 'description' of the new changeset) just after editing, but
|
|
136
|
before checking empty-ness. It should return actual text to be
|
|
136
|
before checking empty-ness. It should return actual text to be
|
|
137
|
stored into history. This allows to change description before
|
|
137
|
stored into history. This allows to change description before
|
|
138
|
storing.
|
|
138
|
storing.
|
|
139
|
|
|
139
|
|
|
140
|
'extramsg' is a extra message to be shown in the editor instead of
|
|
140
|
'extramsg' is a extra message to be shown in the editor instead of
|
|
141
|
'Leave message empty to abort commit' line. 'HG: ' prefix and EOL
|
|
141
|
'Leave message empty to abort commit' line. 'HG: ' prefix and EOL
|
|
142
|
is automatically added.
|
|
142
|
is automatically added.
|
|
143
|
|
|
143
|
|
|
144
|
'editform' is a dot-separated list of names, to distinguish
|
|
144
|
'editform' is a dot-separated list of names, to distinguish
|
|
145
|
the purpose of commit text editing.
|
|
145
|
the purpose of commit text editing.
|
|
146
|
|
|
146
|
|
|
147
|
'getcommiteditor' returns 'commitforceeditor' regardless of
|
|
147
|
'getcommiteditor' returns 'commitforceeditor' regardless of
|
|
148
|
'edit', if one of 'finishdesc' or 'extramsg' is specified, because
|
|
148
|
'edit', if one of 'finishdesc' or 'extramsg' is specified, because
|
|
149
|
they are specific for usage in MQ.
|
|
149
|
they are specific for usage in MQ.
|
|
150
|
"""
|
|
150
|
"""
|
|
151
|
if edit or finishdesc or extramsg:
|
|
151
|
if edit or finishdesc or extramsg:
|
|
152
|
return lambda r, c, s: commitforceeditor(r, c, s,
|
|
152
|
return lambda r, c, s: commitforceeditor(r, c, s,
|
|
153
|
finishdesc=finishdesc,
|
|
153
|
finishdesc=finishdesc,
|
|
154
|
extramsg=extramsg,
|
|
154
|
extramsg=extramsg,
|
|
155
|
editform=editform)
|
|
155
|
editform=editform)
|
|
156
|
elif editform:
|
|
156
|
elif editform:
|
|
157
|
return lambda r, c, s: commiteditor(r, c, s, editform=editform)
|
|
157
|
return lambda r, c, s: commiteditor(r, c, s, editform=editform)
|
|
158
|
else:
|
|
158
|
else:
|
|
159
|
return commiteditor
|
|
159
|
return commiteditor
|
|
160
|
|
|
160
|
|
|
161
|
def loglimit(opts):
|
|
161
|
def loglimit(opts):
|
|
162
|
"""get the log limit according to option -l/--limit"""
|
|
162
|
"""get the log limit according to option -l/--limit"""
|
|
163
|
limit = opts.get('limit')
|
|
163
|
limit = opts.get('limit')
|
|
164
|
if limit:
|
|
164
|
if limit:
|
|
165
|
try:
|
|
165
|
try:
|
|
166
|
limit = int(limit)
|
|
166
|
limit = int(limit)
|
|
167
|
except ValueError:
|
|
167
|
except ValueError:
|
|
168
|
raise util.Abort(_('limit must be a positive integer'))
|
|
168
|
raise util.Abort(_('limit must be a positive integer'))
|
|
169
|
if limit <= 0:
|
|
169
|
if limit <= 0:
|
|
170
|
raise util.Abort(_('limit must be positive'))
|
|
170
|
raise util.Abort(_('limit must be positive'))
|
|
171
|
else:
|
|
171
|
else:
|
|
172
|
limit = None
|
|
172
|
limit = None
|
|
173
|
return limit
|
|
173
|
return limit
|
|
174
|
|
|
174
|
|
|
175
|
def makefilename(repo, pat, node, desc=None,
|
|
175
|
def makefilename(repo, pat, node, desc=None,
|
|
176
|
total=None, seqno=None, revwidth=None, pathname=None):
|
|
176
|
total=None, seqno=None, revwidth=None, pathname=None):
|
|
177
|
node_expander = {
|
|
177
|
node_expander = {
|
|
178
|
'H': lambda: hex(node),
|
|
178
|
'H': lambda: hex(node),
|
|
179
|
'R': lambda: str(repo.changelog.rev(node)),
|
|
179
|
'R': lambda: str(repo.changelog.rev(node)),
|
|
180
|
'h': lambda: short(node),
|
|
180
|
'h': lambda: short(node),
|
|
181
|
'm': lambda: re.sub('[^\w]', '_', str(desc))
|
|
181
|
'm': lambda: re.sub('[^\w]', '_', str(desc))
|
|
182
|
}
|
|
182
|
}
|
|
183
|
expander = {
|
|
183
|
expander = {
|
|
184
|
'%': lambda: '%',
|
|
184
|
'%': lambda: '%',
|
|
185
|
'b': lambda: os.path.basename(repo.root),
|
|
185
|
'b': lambda: os.path.basename(repo.root),
|
|
186
|
}
|
|
186
|
}
|
|
187
|
|
|
187
|
|
|
188
|
try:
|
|
188
|
try:
|
|
189
|
if node:
|
|
189
|
if node:
|
|
190
|
expander.update(node_expander)
|
|
190
|
expander.update(node_expander)
|
|
191
|
if node:
|
|
191
|
if node:
|
|
192
|
expander['r'] = (lambda:
|
|
192
|
expander['r'] = (lambda:
|
|
193
|
str(repo.changelog.rev(node)).zfill(revwidth or 0))
|
|
193
|
str(repo.changelog.rev(node)).zfill(revwidth or 0))
|
|
194
|
if total is not None:
|
|
194
|
if total is not None:
|
|
195
|
expander['N'] = lambda: str(total)
|
|
195
|
expander['N'] = lambda: str(total)
|
|
196
|
if seqno is not None:
|
|
196
|
if seqno is not None:
|
|
197
|
expander['n'] = lambda: str(seqno)
|
|
197
|
expander['n'] = lambda: str(seqno)
|
|
198
|
if total is not None and seqno is not None:
|
|
198
|
if total is not None and seqno is not None:
|
|
199
|
expander['n'] = lambda: str(seqno).zfill(len(str(total)))
|
|
199
|
expander['n'] = lambda: str(seqno).zfill(len(str(total)))
|
|
200
|
if pathname is not None:
|
|
200
|
if pathname is not None:
|
|
201
|
expander['s'] = lambda: os.path.basename(pathname)
|
|
201
|
expander['s'] = lambda: os.path.basename(pathname)
|
|
202
|
expander['d'] = lambda: os.path.dirname(pathname) or '.'
|
|
202
|
expander['d'] = lambda: os.path.dirname(pathname) or '.'
|
|
203
|
expander['p'] = lambda: pathname
|
|
203
|
expander['p'] = lambda: pathname
|
|
204
|
|
|
204
|
|
|
205
|
newname = []
|
|
205
|
newname = []
|
|
206
|
patlen = len(pat)
|
|
206
|
patlen = len(pat)
|
|
207
|
i = 0
|
|
207
|
i = 0
|
|
208
|
while i < patlen:
|
|
208
|
while i < patlen:
|
|
209
|
c = pat[i]
|
|
209
|
c = pat[i]
|
|
210
|
if c == '%':
|
|
210
|
if c == '%':
|
|
211
|
i += 1
|
|
211
|
i += 1
|
|
212
|
c = pat[i]
|
|
212
|
c = pat[i]
|
|
213
|
c = expander[c]()
|
|
213
|
c = expander[c]()
|
|
214
|
newname.append(c)
|
|
214
|
newname.append(c)
|
|
215
|
i += 1
|
|
215
|
i += 1
|
|
216
|
return ''.join(newname)
|
|
216
|
return ''.join(newname)
|
|
217
|
except KeyError, inst:
|
|
217
|
except KeyError, inst:
|
|
218
|
raise util.Abort(_("invalid format spec '%%%s' in output filename") %
|
|
218
|
raise util.Abort(_("invalid format spec '%%%s' in output filename") %
|
|
219
|
inst.args[0])
|
|
219
|
inst.args[0])
|
|
220
|
|
|
220
|
|
|
221
|
def makefileobj(repo, pat, node=None, desc=None, total=None,
|
|
221
|
def makefileobj(repo, pat, node=None, desc=None, total=None,
|
|
222
|
seqno=None, revwidth=None, mode='wb', modemap=None,
|
|
222
|
seqno=None, revwidth=None, mode='wb', modemap=None,
|
|
223
|
pathname=None):
|
|
223
|
pathname=None):
|
|
224
|
|
|
224
|
|
|
225
|
writable = mode not in ('r', 'rb')
|
|
225
|
writable = mode not in ('r', 'rb')
|
|
226
|
|
|
226
|
|
|
227
|
if not pat or pat == '-':
|
|
227
|
if not pat or pat == '-':
|
|
228
|
fp = writable and repo.ui.fout or repo.ui.fin
|
|
228
|
fp = writable and repo.ui.fout or repo.ui.fin
|
|
229
|
if util.safehasattr(fp, 'fileno'):
|
|
229
|
if util.safehasattr(fp, 'fileno'):
|
|
230
|
return os.fdopen(os.dup(fp.fileno()), mode)
|
|
230
|
return os.fdopen(os.dup(fp.fileno()), mode)
|
|
231
|
else:
|
|
231
|
else:
|
|
232
|
# if this fp can't be duped properly, return
|
|
232
|
# if this fp can't be duped properly, return
|
|
233
|
# a dummy object that can be closed
|
|
233
|
# a dummy object that can be closed
|
|
234
|
class wrappedfileobj(object):
|
|
234
|
class wrappedfileobj(object):
|
|
235
|
noop = lambda x: None
|
|
235
|
noop = lambda x: None
|
|
236
|
def __init__(self, f):
|
|
236
|
def __init__(self, f):
|
|
237
|
self.f = f
|
|
237
|
self.f = f
|
|
238
|
def __getattr__(self, attr):
|
|
238
|
def __getattr__(self, attr):
|
|
239
|
if attr == 'close':
|
|
239
|
if attr == 'close':
|
|
240
|
return self.noop
|
|
240
|
return self.noop
|
|
241
|
else:
|
|
241
|
else:
|
|
242
|
return getattr(self.f, attr)
|
|
242
|
return getattr(self.f, attr)
|
|
243
|
|
|
243
|
|
|
244
|
return wrappedfileobj(fp)
|
|
244
|
return wrappedfileobj(fp)
|
|
245
|
if util.safehasattr(pat, 'write') and writable:
|
|
245
|
if util.safehasattr(pat, 'write') and writable:
|
|
246
|
return pat
|
|
246
|
return pat
|
|
247
|
if util.safehasattr(pat, 'read') and 'r' in mode:
|
|
247
|
if util.safehasattr(pat, 'read') and 'r' in mode:
|
|
248
|
return pat
|
|
248
|
return pat
|
|
249
|
fn = makefilename(repo, pat, node, desc, total, seqno, revwidth, pathname)
|
|
249
|
fn = makefilename(repo, pat, node, desc, total, seqno, revwidth, pathname)
|
|
250
|
if modemap is not None:
|
|
250
|
if modemap is not None:
|
|
251
|
mode = modemap.get(fn, mode)
|
|
251
|
mode = modemap.get(fn, mode)
|
|
252
|
if mode == 'wb':
|
|
252
|
if mode == 'wb':
|
|
253
|
modemap[fn] = 'ab'
|
|
253
|
modemap[fn] = 'ab'
|
|
254
|
return open(fn, mode)
|
|
254
|
return open(fn, mode)
|
|
255
|
|
|
255
|
|
|
256
|
def openrevlog(repo, cmd, file_, opts):
|
|
256
|
def openrevlog(repo, cmd, file_, opts):
|
|
257
|
"""opens the changelog, manifest, a filelog or a given revlog"""
|
|
257
|
"""opens the changelog, manifest, a filelog or a given revlog"""
|
|
258
|
cl = opts['changelog']
|
|
258
|
cl = opts['changelog']
|
|
259
|
mf = opts['manifest']
|
|
259
|
mf = opts['manifest']
|
|
260
|
msg = None
|
|
260
|
msg = None
|
|
261
|
if cl and mf:
|
|
261
|
if cl and mf:
|
|
262
|
msg = _('cannot specify --changelog and --manifest at the same time')
|
|
262
|
msg = _('cannot specify --changelog and --manifest at the same time')
|
|
263
|
elif cl or mf:
|
|
263
|
elif cl or mf:
|
|
264
|
if file_:
|
|
264
|
if file_:
|
|
265
|
msg = _('cannot specify filename with --changelog or --manifest')
|
|
265
|
msg = _('cannot specify filename with --changelog or --manifest')
|
|
266
|
elif not repo:
|
|
266
|
elif not repo:
|
|
267
|
msg = _('cannot specify --changelog or --manifest '
|
|
267
|
msg = _('cannot specify --changelog or --manifest '
|
|
268
|
'without a repository')
|
|
268
|
'without a repository')
|
|
269
|
if msg:
|
|
269
|
if msg:
|
|
270
|
raise util.Abort(msg)
|
|
270
|
raise util.Abort(msg)
|
|
271
|
|
|
271
|
|
|
272
|
r = None
|
|
272
|
r = None
|
|
273
|
if repo:
|
|
273
|
if repo:
|
|
274
|
if cl:
|
|
274
|
if cl:
|
|
275
|
r = repo.unfiltered().changelog
|
|
275
|
r = repo.unfiltered().changelog
|
|
276
|
elif mf:
|
|
276
|
elif mf:
|
|
277
|
r = repo.manifest
|
|
277
|
r = repo.manifest
|
|
278
|
elif file_:
|
|
278
|
elif file_:
|
|
279
|
filelog = repo.file(file_)
|
|
279
|
filelog = repo.file(file_)
|
|
280
|
if len(filelog):
|
|
280
|
if len(filelog):
|
|
281
|
r = filelog
|
|
281
|
r = filelog
|
|
282
|
if not r:
|
|
282
|
if not r:
|
|
283
|
if not file_:
|
|
283
|
if not file_:
|
|
284
|
raise error.CommandError(cmd, _('invalid arguments'))
|
|
284
|
raise error.CommandError(cmd, _('invalid arguments'))
|
|
285
|
if not os.path.isfile(file_):
|
|
285
|
if not os.path.isfile(file_):
|
|
286
|
raise util.Abort(_("revlog '%s' not found") % file_)
|
|
286
|
raise util.Abort(_("revlog '%s' not found") % file_)
|
|
287
|
r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False),
|
|
287
|
r = revlog.revlog(scmutil.opener(os.getcwd(), audit=False),
|
|
288
|
file_[:-2] + ".i")
|
|
288
|
file_[:-2] + ".i")
|
|
289
|
return r
|
|
289
|
return r
|
|
290
|
|
|
290
|
|
|
291
|
def copy(ui, repo, pats, opts, rename=False):
|
|
291
|
def copy(ui, repo, pats, opts, rename=False):
|
|
292
|
# called with the repo lock held
|
|
292
|
# called with the repo lock held
|
|
293
|
#
|
|
293
|
#
|
|
294
|
# hgsep => pathname that uses "/" to separate directories
|
|
294
|
# hgsep => pathname that uses "/" to separate directories
|
|
295
|
# ossep => pathname that uses os.sep to separate directories
|
|
295
|
# ossep => pathname that uses os.sep to separate directories
|
|
296
|
cwd = repo.getcwd()
|
|
296
|
cwd = repo.getcwd()
|
|
297
|
targets = {}
|
|
297
|
targets = {}
|
|
298
|
after = opts.get("after")
|
|
298
|
after = opts.get("after")
|
|
299
|
dryrun = opts.get("dry_run")
|
|
299
|
dryrun = opts.get("dry_run")
|
|
300
|
wctx = repo[None]
|
|
300
|
wctx = repo[None]
|
|
301
|
|
|
301
|
|
|
302
|
def walkpat(pat):
|
|
302
|
def walkpat(pat):
|
|
303
|
srcs = []
|
|
303
|
srcs = []
|
|
304
|
badstates = after and '?' or '?r'
|
|
304
|
badstates = after and '?' or '?r'
|
|
305
|
m = scmutil.match(repo[None], [pat], opts, globbed=True)
|
|
305
|
m = scmutil.match(repo[None], [pat], opts, globbed=True)
|
|
306
|
for abs in repo.walk(m):
|
|
306
|
for abs in repo.walk(m):
|
|
307
|
state = repo.dirstate[abs]
|
|
307
|
state = repo.dirstate[abs]
|
|
308
|
rel = m.rel(abs)
|
|
308
|
rel = m.rel(abs)
|
|
309
|
exact = m.exact(abs)
|
|
309
|
exact = m.exact(abs)
|
|
310
|
if state in badstates:
|
|
310
|
if state in badstates:
|
|
311
|
if exact and state == '?':
|
|
311
|
if exact and state == '?':
|
|
312
|
ui.warn(_('%s: not copying - file is not managed\n') % rel)
|
|
312
|
ui.warn(_('%s: not copying - file is not managed\n') % rel)
|
|
313
|
if exact and state == 'r':
|
|
313
|
if exact and state == 'r':
|
|
314
|
ui.warn(_('%s: not copying - file has been marked for'
|
|
314
|
ui.warn(_('%s: not copying - file has been marked for'
|
|
315
|
' remove\n') % rel)
|
|
315
|
' remove\n') % rel)
|
|
316
|
continue
|
|
316
|
continue
|
|
317
|
# abs: hgsep
|
|
317
|
# abs: hgsep
|
|
318
|
# rel: ossep
|
|
318
|
# rel: ossep
|
|
319
|
srcs.append((abs, rel, exact))
|
|
319
|
srcs.append((abs, rel, exact))
|
|
320
|
return srcs
|
|
320
|
return srcs
|
|
321
|
|
|
321
|
|
|
322
|
# abssrc: hgsep
|
|
322
|
# abssrc: hgsep
|
|
323
|
# relsrc: ossep
|
|
323
|
# relsrc: ossep
|
|
324
|
# otarget: ossep
|
|
324
|
# otarget: ossep
|
|
325
|
def copyfile(abssrc, relsrc, otarget, exact):
|
|
325
|
def copyfile(abssrc, relsrc, otarget, exact):
|
|
326
|
abstarget = pathutil.canonpath(repo.root, cwd, otarget)
|
|
326
|
abstarget = pathutil.canonpath(repo.root, cwd, otarget)
|
|
327
|
if '/' in abstarget:
|
|
327
|
if '/' in abstarget:
|
|
328
|
# We cannot normalize abstarget itself, this would prevent
|
|
328
|
# We cannot normalize abstarget itself, this would prevent
|
|
329
|
# case only renames, like a => A.
|
|
329
|
# case only renames, like a => A.
|
|
330
|
abspath, absname = abstarget.rsplit('/', 1)
|
|
330
|
abspath, absname = abstarget.rsplit('/', 1)
|
|
331
|
abstarget = repo.dirstate.normalize(abspath) + '/' + absname
|
|
331
|
abstarget = repo.dirstate.normalize(abspath) + '/' + absname
|
|
332
|
reltarget = repo.pathto(abstarget, cwd)
|
|
332
|
reltarget = repo.pathto(abstarget, cwd)
|
|
333
|
target = repo.wjoin(abstarget)
|
|
333
|
target = repo.wjoin(abstarget)
|
|
334
|
src = repo.wjoin(abssrc)
|
|
334
|
src = repo.wjoin(abssrc)
|
|
335
|
state = repo.dirstate[abstarget]
|
|
335
|
state = repo.dirstate[abstarget]
|
|
336
|
|
|
336
|
|
|
337
|
scmutil.checkportable(ui, abstarget)
|
|
337
|
scmutil.checkportable(ui, abstarget)
|
|
338
|
|
|
338
|
|
|
339
|
# check for collisions
|
|
339
|
# check for collisions
|
|
340
|
prevsrc = targets.get(abstarget)
|
|
340
|
prevsrc = targets.get(abstarget)
|
|
341
|
if prevsrc is not None:
|
|
341
|
if prevsrc is not None:
|
|
342
|
ui.warn(_('%s: not overwriting - %s collides with %s\n') %
|
|
342
|
ui.warn(_('%s: not overwriting - %s collides with %s\n') %
|
|
343
|
(reltarget, repo.pathto(abssrc, cwd),
|
|
343
|
(reltarget, repo.pathto(abssrc, cwd),
|
|
344
|
repo.pathto(prevsrc, cwd)))
|
|
344
|
repo.pathto(prevsrc, cwd)))
|
|
345
|
return
|
|
345
|
return
|
|
346
|
|
|
346
|
|
|
347
|
# check for overwrites
|
|
347
|
# check for overwrites
|
|
348
|
exists = os.path.lexists(target)
|
|
348
|
exists = os.path.lexists(target)
|
|
349
|
samefile = False
|
|
349
|
samefile = False
|
|
350
|
if exists and abssrc != abstarget:
|
|
350
|
if exists and abssrc != abstarget:
|
|
351
|
if (repo.dirstate.normalize(abssrc) ==
|
|
351
|
if (repo.dirstate.normalize(abssrc) ==
|
|
352
|
repo.dirstate.normalize(abstarget)):
|
|
352
|
repo.dirstate.normalize(abstarget)):
|
|
353
|
if not rename:
|
|
353
|
if not rename:
|
|
354
|
ui.warn(_("%s: can't copy - same file\n") % reltarget)
|
|
354
|
ui.warn(_("%s: can't copy - same file\n") % reltarget)
|
|
355
|
return
|
|
355
|
return
|
|
356
|
exists = False
|
|
356
|
exists = False
|
|
357
|
samefile = True
|
|
357
|
samefile = True
|
|
358
|
|
|
358
|
|
|
359
|
if not after and exists or after and state in 'mn':
|
|
359
|
if not after and exists or after and state in 'mn':
|
|
360
|
if not opts['force']:
|
|
360
|
if not opts['force']:
|
|
361
|
ui.warn(_('%s: not overwriting - file exists\n') %
|
|
361
|
ui.warn(_('%s: not overwriting - file exists\n') %
|
|
362
|
reltarget)
|
|
362
|
reltarget)
|
|
363
|
return
|
|
363
|
return
|
|
364
|
|
|
364
|
|
|
365
|
if after:
|
|
365
|
if after:
|
|
366
|
if not exists:
|
|
366
|
if not exists:
|
|
367
|
if rename:
|
|
367
|
if rename:
|
|
368
|
ui.warn(_('%s: not recording move - %s does not exist\n') %
|
|
368
|
ui.warn(_('%s: not recording move - %s does not exist\n') %
|
|
369
|
(relsrc, reltarget))
|
|
369
|
(relsrc, reltarget))
|
|
370
|
else:
|
|
370
|
else:
|
|
371
|
ui.warn(_('%s: not recording copy - %s does not exist\n') %
|
|
371
|
ui.warn(_('%s: not recording copy - %s does not exist\n') %
|
|
372
|
(relsrc, reltarget))
|
|
372
|
(relsrc, reltarget))
|
|
373
|
return
|
|
373
|
return
|
|
374
|
elif not dryrun:
|
|
374
|
elif not dryrun:
|
|
375
|
try:
|
|
375
|
try:
|
|
376
|
if exists:
|
|
376
|
if exists:
|
|
377
|
os.unlink(target)
|
|
377
|
os.unlink(target)
|
|
378
|
targetdir = os.path.dirname(target) or '.'
|
|
378
|
targetdir = os.path.dirname(target) or '.'
|
|
379
|
if not os.path.isdir(targetdir):
|
|
379
|
if not os.path.isdir(targetdir):
|
|
380
|
os.makedirs(targetdir)
|
|
380
|
os.makedirs(targetdir)
|
|
381
|
if samefile:
|
|
381
|
if samefile:
|
|
382
|
tmp = target + "~hgrename"
|
|
382
|
tmp = target + "~hgrename"
|
|
383
|
os.rename(src, tmp)
|
|
383
|
os.rename(src, tmp)
|
|
384
|
os.rename(tmp, target)
|
|
384
|
os.rename(tmp, target)
|
|
385
|
else:
|
|
385
|
else:
|
|
386
|
util.copyfile(src, target)
|
|
386
|
util.copyfile(src, target)
|
|
387
|
srcexists = True
|
|
387
|
srcexists = True
|
|
388
|
except IOError, inst:
|
|
388
|
except IOError, inst:
|
|
389
|
if inst.errno == errno.ENOENT:
|
|
389
|
if inst.errno == errno.ENOENT:
|
|
390
|
ui.warn(_('%s: deleted in working copy\n') % relsrc)
|
|
390
|
ui.warn(_('%s: deleted in working copy\n') % relsrc)
|
|
391
|
srcexists = False
|
|
391
|
srcexists = False
|
|
392
|
else:
|
|
392
|
else:
|
|
393
|
ui.warn(_('%s: cannot copy - %s\n') %
|
|
393
|
ui.warn(_('%s: cannot copy - %s\n') %
|
|
394
|
(relsrc, inst.strerror))
|
|
394
|
(relsrc, inst.strerror))
|
|
395
|
return True # report a failure
|
|
395
|
return True # report a failure
|
|
396
|
|
|
396
|
|
|
397
|
if ui.verbose or not exact:
|
|
397
|
if ui.verbose or not exact:
|
|
398
|
if rename:
|
|
398
|
if rename:
|
|
399
|
ui.status(_('moving %s to %s\n') % (relsrc, reltarget))
|
|
399
|
ui.status(_('moving %s to %s\n') % (relsrc, reltarget))
|
|
400
|
else:
|
|
400
|
else:
|
|
401
|
ui.status(_('copying %s to %s\n') % (relsrc, reltarget))
|
|
401
|
ui.status(_('copying %s to %s\n') % (relsrc, reltarget))
|
|
402
|
|
|
402
|
|
|
403
|
targets[abstarget] = abssrc
|
|
403
|
targets[abstarget] = abssrc
|
|
404
|
|
|
404
|
|
|
405
|
# fix up dirstate
|
|
405
|
# fix up dirstate
|
|
406
|
scmutil.dirstatecopy(ui, repo, wctx, abssrc, abstarget,
|
|
406
|
scmutil.dirstatecopy(ui, repo, wctx, abssrc, abstarget,
|
|
407
|
dryrun=dryrun, cwd=cwd)
|
|
407
|
dryrun=dryrun, cwd=cwd)
|
|
408
|
if rename and not dryrun:
|
|
408
|
if rename and not dryrun:
|
|
409
|
if not after and srcexists and not samefile:
|
|
409
|
if not after and srcexists and not samefile:
|
|
410
|
util.unlinkpath(repo.wjoin(abssrc))
|
|
410
|
util.unlinkpath(repo.wjoin(abssrc))
|
|
411
|
wctx.forget([abssrc])
|
|
411
|
wctx.forget([abssrc])
|
|
412
|
|
|
412
|
|
|
413
|
# pat: ossep
|
|
413
|
# pat: ossep
|
|
414
|
# dest ossep
|
|
414
|
# dest ossep
|
|
415
|
# srcs: list of (hgsep, hgsep, ossep, bool)
|
|
415
|
# srcs: list of (hgsep, hgsep, ossep, bool)
|
|
416
|
# return: function that takes hgsep and returns ossep
|
|
416
|
# return: function that takes hgsep and returns ossep
|
|
417
|
def targetpathfn(pat, dest, srcs):
|
|
417
|
def targetpathfn(pat, dest, srcs):
|
|
418
|
if os.path.isdir(pat):
|
|
418
|
if os.path.isdir(pat):
|
|
419
|
abspfx = pathutil.canonpath(repo.root, cwd, pat)
|
|
419
|
abspfx = pathutil.canonpath(repo.root, cwd, pat)
|
|
420
|
abspfx = util.localpath(abspfx)
|
|
420
|
abspfx = util.localpath(abspfx)
|
|
421
|
if destdirexists:
|
|
421
|
if destdirexists:
|
|
422
|
striplen = len(os.path.split(abspfx)[0])
|
|
422
|
striplen = len(os.path.split(abspfx)[0])
|
|
423
|
else:
|
|
423
|
else:
|
|
424
|
striplen = len(abspfx)
|
|
424
|
striplen = len(abspfx)
|
|
425
|
if striplen:
|
|
425
|
if striplen:
|
|
426
|
striplen += len(os.sep)
|
|
426
|
striplen += len(os.sep)
|
|
427
|
res = lambda p: os.path.join(dest, util.localpath(p)[striplen:])
|
|
427
|
res = lambda p: os.path.join(dest, util.localpath(p)[striplen:])
|
|
428
|
elif destdirexists:
|
|
428
|
elif destdirexists:
|
|
429
|
res = lambda p: os.path.join(dest,
|
|
429
|
res = lambda p: os.path.join(dest,
|
|
430
|
os.path.basename(util.localpath(p)))
|
|
430
|
os.path.basename(util.localpath(p)))
|
|
431
|
else:
|
|
431
|
else:
|
|
432
|
res = lambda p: dest
|
|
432
|
res = lambda p: dest
|
|
433
|
return res
|
|
433
|
return res
|
|
434
|
|
|
434
|
|
|
435
|
# pat: ossep
|
|
435
|
# pat: ossep
|
|
436
|
# dest ossep
|
|
436
|
# dest ossep
|
|
437
|
# srcs: list of (hgsep, hgsep, ossep, bool)
|
|
437
|
# srcs: list of (hgsep, hgsep, ossep, bool)
|
|
438
|
# return: function that takes hgsep and returns ossep
|
|
438
|
# return: function that takes hgsep and returns ossep
|
|
439
|
def targetpathafterfn(pat, dest, srcs):
|
|
439
|
def targetpathafterfn(pat, dest, srcs):
|
|
440
|
if matchmod.patkind(pat):
|
|
440
|
if matchmod.patkind(pat):
|
|
441
|
# a mercurial pattern
|
|
441
|
# a mercurial pattern
|
|
442
|
res = lambda p: os.path.join(dest,
|
|
442
|
res = lambda p: os.path.join(dest,
|
|
443
|
os.path.basename(util.localpath(p)))
|
|
443
|
os.path.basename(util.localpath(p)))
|
|
444
|
else:
|
|
444
|
else:
|
|
445
|
abspfx = pathutil.canonpath(repo.root, cwd, pat)
|
|
445
|
abspfx = pathutil.canonpath(repo.root, cwd, pat)
|
|
446
|
if len(abspfx) < len(srcs[0][0]):
|
|
446
|
if len(abspfx) < len(srcs[0][0]):
|
|
447
|
# A directory. Either the target path contains the last
|
|
447
|
# A directory. Either the target path contains the last
|
|
448
|
# component of the source path or it does not.
|
|
448
|
# component of the source path or it does not.
|
|
449
|
def evalpath(striplen):
|
|
449
|
def evalpath(striplen):
|
|
450
|
score = 0
|
|
450
|
score = 0
|
|
451
|
for s in srcs:
|
|
451
|
for s in srcs:
|
|
452
|
t = os.path.join(dest, util.localpath(s[0])[striplen:])
|
|
452
|
t = os.path.join(dest, util.localpath(s[0])[striplen:])
|
|
453
|
if os.path.lexists(t):
|
|
453
|
if os.path.lexists(t):
|
|
454
|
score += 1
|
|
454
|
score += 1
|
|
455
|
return score
|
|
455
|
return score
|
|
456
|
|
|
456
|
|
|
457
|
abspfx = util.localpath(abspfx)
|
|
457
|
abspfx = util.localpath(abspfx)
|
|
458
|
striplen = len(abspfx)
|
|
458
|
striplen = len(abspfx)
|
|
459
|
if striplen:
|
|
459
|
if striplen:
|
|
460
|
striplen += len(os.sep)
|
|
460
|
striplen += len(os.sep)
|
|
461
|
if os.path.isdir(os.path.join(dest, os.path.split(abspfx)[1])):
|
|
461
|
if os.path.isdir(os.path.join(dest, os.path.split(abspfx)[1])):
|
|
462
|
score = evalpath(striplen)
|
|
462
|
score = evalpath(striplen)
|
|
463
|
striplen1 = len(os.path.split(abspfx)[0])
|
|
463
|
striplen1 = len(os.path.split(abspfx)[0])
|
|
464
|
if striplen1:
|
|
464
|
if striplen1:
|
|
465
|
striplen1 += len(os.sep)
|
|
465
|
striplen1 += len(os.sep)
|
|
466
|
if evalpath(striplen1) > score:
|
|
466
|
if evalpath(striplen1) > score:
|
|
467
|
striplen = striplen1
|
|
467
|
striplen = striplen1
|
|
468
|
res = lambda p: os.path.join(dest,
|
|
468
|
res = lambda p: os.path.join(dest,
|
|
469
|
util.localpath(p)[striplen:])
|
|
469
|
util.localpath(p)[striplen:])
|
|
470
|
else:
|
|
470
|
else:
|
|
471
|
# a file
|
|
471
|
# a file
|
|
472
|
if destdirexists:
|
|
472
|
if destdirexists:
|
|
473
|
res = lambda p: os.path.join(dest,
|
|
473
|
res = lambda p: os.path.join(dest,
|
|
474
|
os.path.basename(util.localpath(p)))
|
|
474
|
os.path.basename(util.localpath(p)))
|
|
475
|
else:
|
|
475
|
else:
|
|
476
|
res = lambda p: dest
|
|
476
|
res = lambda p: dest
|
|
477
|
return res
|
|
477
|
return res
|
|
478
|
|
|
478
|
|
|
479
|
|
|
479
|
|
|
480
|
pats = scmutil.expandpats(pats)
|
|
480
|
pats = scmutil.expandpats(pats)
|
|
481
|
if not pats:
|
|
481
|
if not pats:
|
|
482
|
raise util.Abort(_('no source or destination specified'))
|
|
482
|
raise util.Abort(_('no source or destination specified'))
|
|
483
|
if len(pats) == 1:
|
|
483
|
if len(pats) == 1:
|
|
484
|
raise util.Abort(_('no destination specified'))
|
|
484
|
raise util.Abort(_('no destination specified'))
|
|
485
|
dest = pats.pop()
|
|
485
|
dest = pats.pop()
|
|
486
|
destdirexists = os.path.isdir(dest) and not os.path.islink(dest)
|
|
486
|
destdirexists = os.path.isdir(dest) and not os.path.islink(dest)
|
|
487
|
if not destdirexists:
|
|
487
|
if not destdirexists:
|
|
488
|
if len(pats) > 1 or matchmod.patkind(pats[0]):
|
|
488
|
if len(pats) > 1 or matchmod.patkind(pats[0]):
|
|
489
|
raise util.Abort(_('with multiple sources, destination must be an '
|
|
489
|
raise util.Abort(_('with multiple sources, destination must be an '
|
|
490
|
'existing directory'))
|
|
490
|
'existing directory'))
|
|
491
|
if util.endswithsep(dest):
|
|
491
|
if util.endswithsep(dest):
|
|
492
|
raise util.Abort(_('destination %s is not a directory') % dest)
|
|
492
|
raise util.Abort(_('destination %s is not a directory') % dest)
|
|
493
|
|
|
493
|
|
|
494
|
tfn = targetpathfn
|
|
494
|
tfn = targetpathfn
|
|
495
|
if after:
|
|
495
|
if after:
|
|
496
|
tfn = targetpathafterfn
|
|
496
|
tfn = targetpathafterfn
|
|
497
|
copylist = []
|
|
497
|
copylist = []
|
|
498
|
for pat in pats:
|
|
498
|
for pat in pats:
|
|
499
|
srcs = walkpat(pat)
|
|
499
|
srcs = walkpat(pat)
|
|
500
|
if not srcs:
|
|
500
|
if not srcs:
|
|
501
|
continue
|
|
501
|
continue
|
|
502
|
copylist.append((tfn(pat, dest, srcs), srcs))
|
|
502
|
copylist.append((tfn(pat, dest, srcs), srcs))
|
|
503
|
if not copylist:
|
|
503
|
if not copylist:
|
|
504
|
raise util.Abort(_('no files to copy'))
|
|
504
|
raise util.Abort(_('no files to copy'))
|
|
505
|
|
|
505
|
|
|
506
|
errors = 0
|
|
506
|
errors = 0
|
|
507
|
for targetpath, srcs in copylist:
|
|
507
|
for targetpath, srcs in copylist:
|
|
508
|
for abssrc, relsrc, exact in srcs:
|
|
508
|
for abssrc, relsrc, exact in srcs:
|
|
509
|
if copyfile(abssrc, relsrc, targetpath(abssrc), exact):
|
|
509
|
if copyfile(abssrc, relsrc, targetpath(abssrc), exact):
|
|
510
|
errors += 1
|
|
510
|
errors += 1
|
|
511
|
|
|
511
|
|
|
512
|
if errors:
|
|
512
|
if errors:
|
|
513
|
ui.warn(_('(consider using --after)\n'))
|
|
513
|
ui.warn(_('(consider using --after)\n'))
|
|
514
|
|
|
514
|
|
|
515
|
return errors != 0
|
|
515
|
return errors != 0
|
|
516
|
|
|
516
|
|
|
517
|
def service(opts, parentfn=None, initfn=None, runfn=None, logfile=None,
|
|
517
|
def service(opts, parentfn=None, initfn=None, runfn=None, logfile=None,
|
|
518
|
runargs=None, appendpid=False):
|
|
518
|
runargs=None, appendpid=False):
|
|
519
|
'''Run a command as a service.'''
|
|
519
|
'''Run a command as a service.'''
|
|
520
|
|
|
520
|
|
|
521
|
def writepid(pid):
|
|
521
|
def writepid(pid):
|
|
522
|
if opts['pid_file']:
|
|
522
|
if opts['pid_file']:
|
|
523
|
mode = appendpid and 'a' or 'w'
|
|
523
|
mode = appendpid and 'a' or 'w'
|
|
524
|
fp = open(opts['pid_file'], mode)
|
|
524
|
fp = open(opts['pid_file'], mode)
|
|
525
|
fp.write(str(pid) + '\n')
|
|
525
|
fp.write(str(pid) + '\n')
|
|
526
|
fp.close()
|
|
526
|
fp.close()
|
|
527
|
|
|
527
|
|
|
528
|
if opts['daemon'] and not opts['daemon_pipefds']:
|
|
528
|
if opts['daemon'] and not opts['daemon_pipefds']:
|
|
529
|
# Signal child process startup with file removal
|
|
529
|
# Signal child process startup with file removal
|
|
530
|
lockfd, lockpath = tempfile.mkstemp(prefix='hg-service-')
|
|
530
|
lockfd, lockpath = tempfile.mkstemp(prefix='hg-service-')
|
|
531
|
os.close(lockfd)
|
|
531
|
os.close(lockfd)
|
|
532
|
try:
|
|
532
|
try:
|
|
533
|
if not runargs:
|
|
533
|
if not runargs:
|
|
534
|
runargs = util.hgcmd() + sys.argv[1:]
|
|
534
|
runargs = util.hgcmd() + sys.argv[1:]
|
|
535
|
runargs.append('--daemon-pipefds=%s' % lockpath)
|
|
535
|
runargs.append('--daemon-pipefds=%s' % lockpath)
|
|
536
|
# Don't pass --cwd to the child process, because we've already
|
|
536
|
# Don't pass --cwd to the child process, because we've already
|
|
537
|
# changed directory.
|
|
537
|
# changed directory.
|
|
538
|
for i in xrange(1, len(runargs)):
|
|
538
|
for i in xrange(1, len(runargs)):
|
|
539
|
if runargs[i].startswith('--cwd='):
|
|
539
|
if runargs[i].startswith('--cwd='):
|
|
540
|
del runargs[i]
|
|
540
|
del runargs[i]
|
|
541
|
break
|
|
541
|
break
|
|
542
|
elif runargs[i].startswith('--cwd'):
|
|
542
|
elif runargs[i].startswith('--cwd'):
|
|
543
|
del runargs[i:i + 2]
|
|
543
|
del runargs[i:i + 2]
|
|
544
|
break
|
|
544
|
break
|
|
545
|
def condfn():
|
|
545
|
def condfn():
|
|
546
|
return not os.path.exists(lockpath)
|
|
546
|
return not os.path.exists(lockpath)
|
|
547
|
pid = util.rundetached(runargs, condfn)
|
|
547
|
pid = util.rundetached(runargs, condfn)
|
|
548
|
if pid < 0:
|
|
548
|
if pid < 0:
|
|
549
|
raise util.Abort(_('child process failed to start'))
|
|
549
|
raise util.Abort(_('child process failed to start'))
|
|
550
|
writepid(pid)
|
|
550
|
writepid(pid)
|
|
551
|
finally:
|
|
551
|
finally:
|
|
552
|
try:
|
|
552
|
try:
|
|
553
|
os.unlink(lockpath)
|
|
553
|
os.unlink(lockpath)
|
|
554
|
except OSError, e:
|
|
554
|
except OSError, e:
|
|
555
|
if e.errno != errno.ENOENT:
|
|
555
|
if e.errno != errno.ENOENT:
|
|
556
|
raise
|
|
556
|
raise
|
|
557
|
if parentfn:
|
|
557
|
if parentfn:
|
|
558
|
return parentfn(pid)
|
|
558
|
return parentfn(pid)
|
|
559
|
else:
|
|
559
|
else:
|
|
560
|
return
|
|
560
|
return
|
|
561
|
|
|
561
|
|
|
562
|
if initfn:
|
|
562
|
if initfn:
|
|
563
|
initfn()
|
|
563
|
initfn()
|
|
564
|
|
|
564
|
|
|
565
|
if not opts['daemon']:
|
|
565
|
if not opts['daemon']:
|
|
566
|
writepid(os.getpid())
|
|
566
|
writepid(os.getpid())
|
|
567
|
|
|
567
|
|
|
568
|
if opts['daemon_pipefds']:
|
|
568
|
if opts['daemon_pipefds']:
|
|
569
|
lockpath = opts['daemon_pipefds']
|
|
569
|
lockpath = opts['daemon_pipefds']
|
|
570
|
try:
|
|
570
|
try:
|
|
571
|
os.setsid()
|
|
571
|
os.setsid()
|
|
572
|
except AttributeError:
|
|
572
|
except AttributeError:
|
|
573
|
pass
|
|
573
|
pass
|
|
574
|
os.unlink(lockpath)
|
|
574
|
os.unlink(lockpath)
|
|
575
|
util.hidewindow()
|
|
575
|
util.hidewindow()
|
|
576
|
sys.stdout.flush()
|
|
576
|
sys.stdout.flush()
|
|
577
|
sys.stderr.flush()
|
|
577
|
sys.stderr.flush()
|
|
578
|
|
|
578
|
|
|
579
|
nullfd = os.open(os.devnull, os.O_RDWR)
|
|
579
|
nullfd = os.open(os.devnull, os.O_RDWR)
|
|
580
|
logfilefd = nullfd
|
|
580
|
logfilefd = nullfd
|
|
581
|
if logfile:
|
|
581
|
if logfile:
|
|
582
|
logfilefd = os.open(logfile, os.O_RDWR | os.O_CREAT | os.O_APPEND)
|
|
582
|
logfilefd = os.open(logfile, os.O_RDWR | os.O_CREAT | os.O_APPEND)
|
|
583
|
os.dup2(nullfd, 0)
|
|
583
|
os.dup2(nullfd, 0)
|
|
584
|
os.dup2(logfilefd, 1)
|
|
584
|
os.dup2(logfilefd, 1)
|
|
585
|
os.dup2(logfilefd, 2)
|
|
585
|
os.dup2(logfilefd, 2)
|
|
586
|
if nullfd not in (0, 1, 2):
|
|
586
|
if nullfd not in (0, 1, 2):
|
|
587
|
os.close(nullfd)
|
|
587
|
os.close(nullfd)
|
|
588
|
if logfile and logfilefd not in (0, 1, 2):
|
|
588
|
if logfile and logfilefd not in (0, 1, 2):
|
|
589
|
os.close(logfilefd)
|
|
589
|
os.close(logfilefd)
|
|
590
|
|
|
590
|
|
|
591
|
if runfn:
|
|
591
|
if runfn:
|
|
592
|
return runfn()
|
|
592
|
return runfn()
|
|
593
|
|
|
593
|
|
|
594
|
def tryimportone(ui, repo, hunk, parents, opts, msgs, updatefunc):
|
|
594
|
def tryimportone(ui, repo, hunk, parents, opts, msgs, updatefunc):
|
|
595
|
"""Utility function used by commands.import to import a single patch
|
|
595
|
"""Utility function used by commands.import to import a single patch
|
|
596
|
|
|
596
|
|
|
597
|
This function is explicitly defined here to help the evolve extension to
|
|
597
|
This function is explicitly defined here to help the evolve extension to
|
|
598
|
wrap this part of the import logic.
|
|
598
|
wrap this part of the import logic.
|
|
599
|
|
|
599
|
|
|
600
|
The API is currently a bit ugly because it a simple code translation from
|
|
600
|
The API is currently a bit ugly because it a simple code translation from
|
|
601
|
the import command. Feel free to make it better.
|
|
601
|
the import command. Feel free to make it better.
|
|
602
|
|
|
602
|
|
|
603
|
:hunk: a patch (as a binary string)
|
|
603
|
:hunk: a patch (as a binary string)
|
|
604
|
:parents: nodes that will be parent of the created commit
|
|
604
|
:parents: nodes that will be parent of the created commit
|
|
605
|
:opts: the full dict of option passed to the import command
|
|
605
|
:opts: the full dict of option passed to the import command
|
|
606
|
:msgs: list to save commit message to.
|
|
606
|
:msgs: list to save commit message to.
|
|
607
|
(used in case we need to save it when failing)
|
|
607
|
(used in case we need to save it when failing)
|
|
608
|
:updatefunc: a function that update a repo to a given node
|
|
608
|
:updatefunc: a function that update a repo to a given node
|
|
609
|
updatefunc(<repo>, <node>)
|
|
609
|
updatefunc(<repo>, <node>)
|
|
610
|
"""
|
|
610
|
"""
|
|
611
|
tmpname, message, user, date, branch, nodeid, p1, p2 = \
|
|
611
|
tmpname, message, user, date, branch, nodeid, p1, p2 = \
|
|
612
|
patch.extract(ui, hunk)
|
|
612
|
patch.extract(ui, hunk)
|
|
613
|
|
|
613
|
|
|
614
|
update = not opts.get('bypass')
|
|
614
|
update = not opts.get('bypass')
|
|
615
|
strip = opts["strip"]
|
|
615
|
strip = opts["strip"]
|
|
616
|
sim = float(opts.get('similarity') or 0)
|
|
616
|
sim = float(opts.get('similarity') or 0)
|
|
617
|
if not tmpname:
|
|
617
|
if not tmpname:
|
|
618
|
return (None, None, False)
|
|
618
|
return (None, None, False)
|
|
619
|
msg = _('applied to working directory')
|
|
619
|
msg = _('applied to working directory')
|
|
620
|
|
|
620
|
|
|
621
|
rejects = False
|
|
621
|
rejects = False
|
|
622
|
|
|
622
|
|
|
623
|
try:
|
|
623
|
try:
|
|
624
|
cmdline_message = logmessage(ui, opts)
|
|
624
|
cmdline_message = logmessage(ui, opts)
|
|
625
|
if cmdline_message:
|
|
625
|
if cmdline_message:
|
|
626
|
# pickup the cmdline msg
|
|
626
|
# pickup the cmdline msg
|
|
627
|
message = cmdline_message
|
|
627
|
message = cmdline_message
|
|
628
|
elif message:
|
|
628
|
elif message:
|
|
629
|
# pickup the patch msg
|
|
629
|
# pickup the patch msg
|
|
630
|
message = message.strip()
|
|
630
|
message = message.strip()
|
|
631
|
else:
|
|
631
|
else:
|
|
632
|
# launch the editor
|
|
632
|
# launch the editor
|
|
633
|
message = None
|
|
633
|
message = None
|
|
634
|
ui.debug('message:\n%s\n' % message)
|
|
634
|
ui.debug('message:\n%s\n' % message)
|
|
635
|
|
|
635
|
|
|
636
|
if len(parents) == 1:
|
|
636
|
if len(parents) == 1:
|
|
637
|
parents.append(repo[nullid])
|
|
637
|
parents.append(repo[nullid])
|
|
638
|
if opts.get('exact'):
|
|
638
|
if opts.get('exact'):
|
|
639
|
if not nodeid or not p1:
|
|
639
|
if not nodeid or not p1:
|
|
640
|
raise util.Abort(_('not a Mercurial patch'))
|
|
640
|
raise util.Abort(_('not a Mercurial patch'))
|
|
641
|
p1 = repo[p1]
|
|
641
|
p1 = repo[p1]
|
|
642
|
p2 = repo[p2 or nullid]
|
|
642
|
p2 = repo[p2 or nullid]
|
|
643
|
elif p2:
|
|
643
|
elif p2:
|
|
644
|
try:
|
|
644
|
try:
|
|
645
|
p1 = repo[p1]
|
|
645
|
p1 = repo[p1]
|
|
646
|
p2 = repo[p2]
|
|
646
|
p2 = repo[p2]
|
|
647
|
# Without any options, consider p2 only if the
|
|
647
|
# Without any options, consider p2 only if the
|
|
648
|
# patch is being applied on top of the recorded
|
|
648
|
# patch is being applied on top of the recorded
|
|
649
|
# first parent.
|
|
649
|
# first parent.
|
|
650
|
if p1 != parents[0]:
|
|
650
|
if p1 != parents[0]:
|
|
651
|
p1 = parents[0]
|
|
651
|
p1 = parents[0]
|
|
652
|
p2 = repo[nullid]
|
|
652
|
p2 = repo[nullid]
|
|
653
|
except error.RepoError:
|
|
653
|
except error.RepoError:
|
|
654
|
p1, p2 = parents
|
|
654
|
p1, p2 = parents
|
|
655
|
if p2.node() == nullid:
|
|
655
|
if p2.node() == nullid:
|
|
656
|
ui.warn(_("warning: import the patch as a normal revision\n"
|
|
656
|
ui.warn(_("warning: import the patch as a normal revision\n"
|
|
657
|
"(use --exact to import the patch as a merge)\n"))
|
|
657
|
"(use --exact to import the patch as a merge)\n"))
|
|
658
|
else:
|
|
658
|
else:
|
|
659
|
p1, p2 = parents
|
|
659
|
p1, p2 = parents
|
|
660
|
|
|
660
|
|
|
661
|
n = None
|
|
661
|
n = None
|
|
662
|
if update:
|
|
662
|
if update:
|
|
663
|
repo.dirstate.beginparentchange()
|
|
663
|
repo.dirstate.beginparentchange()
|
|
664
|
if p1 != parents[0]:
|
|
664
|
if p1 != parents[0]:
|
|
665
|
updatefunc(repo, p1.node())
|
|
665
|
updatefunc(repo, p1.node())
|
|
666
|
if p2 != parents[1]:
|
|
666
|
if p2 != parents[1]:
|
|
667
|
repo.setparents(p1.node(), p2.node())
|
|
667
|
repo.setparents(p1.node(), p2.node())
|
|
668
|
|
|
668
|
|
|
669
|
if opts.get('exact') or opts.get('import_branch'):
|
|
669
|
if opts.get('exact') or opts.get('import_branch'):
|
|
670
|
repo.dirstate.setbranch(branch or 'default')
|
|
670
|
repo.dirstate.setbranch(branch or 'default')
|
|
671
|
|
|
671
|
|
|
672
|
partial = opts.get('partial', False)
|
|
672
|
partial = opts.get('partial', False)
|
|
673
|
files = set()
|
|
673
|
files = set()
|
|
674
|
try:
|
|
674
|
try:
|
|
675
|
patch.patch(ui, repo, tmpname, strip=strip, files=files,
|
|
675
|
patch.patch(ui, repo, tmpname, strip=strip, files=files,
|
|
676
|
eolmode=None, similarity=sim / 100.0)
|
|
676
|
eolmode=None, similarity=sim / 100.0)
|
|
677
|
except patch.PatchError, e:
|
|
677
|
except patch.PatchError, e:
|
|
678
|
if not partial:
|
|
678
|
if not partial:
|
|
679
|
raise util.Abort(str(e))
|
|
679
|
raise util.Abort(str(e))
|
|
680
|
if partial:
|
|
680
|
if partial:
|
|
681
|
rejects = True
|
|
681
|
rejects = True
|
|
682
|
|
|
682
|
|
|
683
|
files = list(files)
|
|
683
|
files = list(files)
|
|
684
|
if opts.get('no_commit'):
|
|
684
|
if opts.get('no_commit'):
|
|
685
|
if message:
|
|
685
|
if message:
|
|
686
|
msgs.append(message)
|
|
686
|
msgs.append(message)
|
|
687
|
else:
|
|
687
|
else:
|
|
688
|
if opts.get('exact') or p2:
|
|
688
|
if opts.get('exact') or p2:
|
|
689
|
# If you got here, you either use --force and know what
|
|
689
|
# If you got here, you either use --force and know what
|
|
690
|
# you are doing or used --exact or a merge patch while
|
|
690
|
# you are doing or used --exact or a merge patch while
|
|
691
|
# being updated to its first parent.
|
|
691
|
# being updated to its first parent.
|
|
692
|
m = None
|
|
692
|
m = None
|
|
693
|
else:
|
|
693
|
else:
|
|
694
|
m = scmutil.matchfiles(repo, files or [])
|
|
694
|
m = scmutil.matchfiles(repo, files or [])
|
|
695
|
editform = mergeeditform(repo[None], 'import.normal')
|
|
695
|
editform = mergeeditform(repo[None], 'import.normal')
|
|
696
|
if opts.get('exact'):
|
|
696
|
if opts.get('exact'):
|
|
697
|
editor = None
|
|
697
|
editor = None
|
|
698
|
else:
|
|
698
|
else:
|
|
699
|
editor = getcommiteditor(editform=editform, **opts)
|
|
699
|
editor = getcommiteditor(editform=editform, **opts)
|
|
700
|
n = repo.commit(message, opts.get('user') or user,
|
|
700
|
n = repo.commit(message, opts.get('user') or user,
|
|
701
|
opts.get('date') or date, match=m,
|
|
701
|
opts.get('date') or date, match=m,
|
|
702
|
editor=editor, force=partial)
|
|
702
|
editor=editor, force=partial)
|
|
703
|
repo.dirstate.endparentchange()
|
|
703
|
repo.dirstate.endparentchange()
|
|
704
|
else:
|
|
704
|
else:
|
|
705
|
if opts.get('exact') or opts.get('import_branch'):
|
|
705
|
if opts.get('exact') or opts.get('import_branch'):
|
|
706
|
branch = branch or 'default'
|
|
706
|
branch = branch or 'default'
|
|
707
|
else:
|
|
707
|
else:
|
|
708
|
branch = p1.branch()
|
|
708
|
branch = p1.branch()
|
|
709
|
store = patch.filestore()
|
|
709
|
store = patch.filestore()
|
|
710
|
try:
|
|
710
|
try:
|
|
711
|
files = set()
|
|
711
|
files = set()
|
|
712
|
try:
|
|
712
|
try:
|
|
713
|
patch.patchrepo(ui, repo, p1, store, tmpname, strip,
|
|
713
|
patch.patchrepo(ui, repo, p1, store, tmpname, strip,
|
|
714
|
files, eolmode=None)
|
|
714
|
files, eolmode=None)
|
|
715
|
except patch.PatchError, e:
|
|
715
|
except patch.PatchError, e:
|
|
716
|
raise util.Abort(str(e))
|
|
716
|
raise util.Abort(str(e))
|
|
717
|
if opts.get('exact'):
|
|
717
|
if opts.get('exact'):
|
|
718
|
editor = None
|
|
718
|
editor = None
|
|
719
|
else:
|
|
719
|
else:
|
|
720
|
editor = getcommiteditor(editform='import.bypass')
|
|
720
|
editor = getcommiteditor(editform='import.bypass')
|
|
721
|
memctx = context.makememctx(repo, (p1.node(), p2.node()),
|
|
721
|
memctx = context.makememctx(repo, (p1.node(), p2.node()),
|
|
722
|
message,
|
|
722
|
message,
|
|
723
|
opts.get('user') or user,
|
|
723
|
opts.get('user') or user,
|
|
724
|
opts.get('date') or date,
|
|
724
|
opts.get('date') or date,
|
|
725
|
branch, files, store,
|
|
725
|
branch, files, store,
|
|
726
|
editor=editor)
|
|
726
|
editor=editor)
|
|
727
|
n = memctx.commit()
|
|
727
|
n = memctx.commit()
|
|
728
|
finally:
|
|
728
|
finally:
|
|
729
|
store.close()
|
|
729
|
store.close()
|
|
730
|
if opts.get('exact') and opts.get('no_commit'):
|
|
730
|
if opts.get('exact') and opts.get('no_commit'):
|
|
731
|
# --exact with --no-commit is still useful in that it does merge
|
|
731
|
# --exact with --no-commit is still useful in that it does merge
|
|
732
|
# and branch bits
|
|
732
|
# and branch bits
|
|
733
|
ui.warn(_("warning: can't check exact import with --no-commit\n"))
|
|
733
|
ui.warn(_("warning: can't check exact import with --no-commit\n"))
|
|
734
|
elif opts.get('exact') and hex(n) != nodeid:
|
|
734
|
elif opts.get('exact') and hex(n) != nodeid:
|
|
735
|
raise util.Abort(_('patch is damaged or loses information'))
|
|
735
|
raise util.Abort(_('patch is damaged or loses information'))
|
|
736
|
if n:
|
|
736
|
if n:
|
|
737
|
# i18n: refers to a short changeset id
|
|
737
|
# i18n: refers to a short changeset id
|
|
738
|
msg = _('created %s') % short(n)
|
|
738
|
msg = _('created %s') % short(n)
|
|
739
|
return (msg, n, rejects)
|
|
739
|
return (msg, n, rejects)
|
|
740
|
finally:
|
|
740
|
finally:
|
|
741
|
os.unlink(tmpname)
|
|
741
|
os.unlink(tmpname)
|
|
742
|
|
|
742
|
|
|
743
|
def export(repo, revs, template='hg-%h.patch', fp=None, switch_parent=False,
|
|
743
|
def export(repo, revs, template='hg-%h.patch', fp=None, switch_parent=False,
|
|
744
|
opts=None):
|
|
744
|
opts=None):
|
|
745
|
'''export changesets as hg patches.'''
|
|
745
|
'''export changesets as hg patches.'''
|
|
746
|
|
|
746
|
|
|
747
|
total = len(revs)
|
|
747
|
total = len(revs)
|
|
748
|
revwidth = max([len(str(rev)) for rev in revs])
|
|
748
|
revwidth = max([len(str(rev)) for rev in revs])
|
|
749
|
filemode = {}
|
|
749
|
filemode = {}
|
|
750
|
|
|
750
|
|
|
751
|
def single(rev, seqno, fp):
|
|
751
|
def single(rev, seqno, fp):
|
|
752
|
ctx = repo[rev]
|
|
752
|
ctx = repo[rev]
|
|
753
|
node = ctx.node()
|
|
753
|
node = ctx.node()
|
|
754
|
parents = [p.node() for p in ctx.parents() if p]
|
|
754
|
parents = [p.node() for p in ctx.parents() if p]
|
|
755
|
branch = ctx.branch()
|
|
755
|
branch = ctx.branch()
|
|
756
|
if switch_parent:
|
|
756
|
if switch_parent:
|
|
757
|
parents.reverse()
|
|
757
|
parents.reverse()
|
|
758
|
prev = (parents and parents[0]) or nullid
|
|
758
|
prev = (parents and parents[0]) or nullid
|
|
759
|
|
|
759
|
|
|
760
|
shouldclose = False
|
|
760
|
shouldclose = False
|
|
761
|
if not fp and len(template) > 0:
|
|
761
|
if not fp and len(template) > 0:
|
|
762
|
desc_lines = ctx.description().rstrip().split('\n')
|
|
762
|
desc_lines = ctx.description().rstrip().split('\n')
|
|
763
|
desc = desc_lines[0] #Commit always has a first line.
|
|
763
|
desc = desc_lines[0] #Commit always has a first line.
|
|
764
|
fp = makefileobj(repo, template, node, desc=desc, total=total,
|
|
764
|
fp = makefileobj(repo, template, node, desc=desc, total=total,
|
|
765
|
seqno=seqno, revwidth=revwidth, mode='wb',
|
|
765
|
seqno=seqno, revwidth=revwidth, mode='wb',
|
|
766
|
modemap=filemode)
|
|
766
|
modemap=filemode)
|
|
767
|
if fp != template:
|
|
767
|
if fp != template:
|
|
768
|
shouldclose = True
|
|
768
|
shouldclose = True
|
|
769
|
if fp and fp != sys.stdout and util.safehasattr(fp, 'name'):
|
|
769
|
if fp and fp != sys.stdout and util.safehasattr(fp, 'name'):
|
|
770
|
repo.ui.note("%s\n" % fp.name)
|
|
770
|
repo.ui.note("%s\n" % fp.name)
|
|
771
|
|
|
771
|
|
|
772
|
if not fp:
|
|
772
|
if not fp:
|
|
773
|
write = repo.ui.write
|
|
773
|
write = repo.ui.write
|
|
774
|
else:
|
|
774
|
else:
|
|
775
|
def write(s, **kw):
|
|
775
|
def write(s, **kw):
|
|
776
|
fp.write(s)
|
|
776
|
fp.write(s)
|
|
777
|
|
|
777
|
|
|
778
|
|
|
778
|
|
|
779
|
write("# HG changeset patch\n")
|
|
779
|
write("# HG changeset patch\n")
|
|
780
|
write("# User %s\n" % ctx.user())
|
|
780
|
write("# User %s\n" % ctx.user())
|
|
781
|
write("# Date %d %d\n" % ctx.date())
|
|
781
|
write("# Date %d %d\n" % ctx.date())
|
|
782
|
write("# %s\n" % util.datestr(ctx.date()))
|
|
782
|
write("# %s\n" % util.datestr(ctx.date()))
|
|
783
|
if branch and branch != 'default':
|
|
783
|
if branch and branch != 'default':
|
|
784
|
write("# Branch %s\n" % branch)
|
|
784
|
write("# Branch %s\n" % branch)
|
|
785
|
write("# Node ID %s\n" % hex(node))
|
|
785
|
write("# Node ID %s\n" % hex(node))
|
|
786
|
write("# Parent %s\n" % hex(prev))
|
|
786
|
write("# Parent %s\n" % hex(prev))
|
|
787
|
if len(parents) > 1:
|
|
787
|
if len(parents) > 1:
|
|
788
|
write("# Parent %s\n" % hex(parents[1]))
|
|
788
|
write("# Parent %s\n" % hex(parents[1]))
|
|
789
|
write(ctx.description().rstrip())
|
|
789
|
write(ctx.description().rstrip())
|
|
790
|
write("\n\n")
|
|
790
|
write("\n\n")
|
|
791
|
|
|
791
|
|
|
792
|
for chunk, label in patch.diffui(repo, prev, node, opts=opts):
|
|
792
|
for chunk, label in patch.diffui(repo, prev, node, opts=opts):
|
|
793
|
write(chunk, label=label)
|
|
793
|
write(chunk, label=label)
|
|
794
|
|
|
794
|
|
|
795
|
if shouldclose:
|
|
795
|
if shouldclose:
|
|
796
|
fp.close()
|
|
796
|
fp.close()
|
|
797
|
|
|
797
|
|
|
798
|
for seqno, rev in enumerate(revs):
|
|
798
|
for seqno, rev in enumerate(revs):
|
|
799
|
single(rev, seqno + 1, fp)
|
|
799
|
single(rev, seqno + 1, fp)
|
|
800
|
|
|
800
|
|
|
801
|
def diffordiffstat(ui, repo, diffopts, node1, node2, match,
|
|
801
|
def diffordiffstat(ui, repo, diffopts, node1, node2, match,
|
|
802
|
changes=None, stat=False, fp=None, prefix='',
|
|
802
|
changes=None, stat=False, fp=None, prefix='',
|
|
803
|
listsubrepos=False):
|
|
803
|
listsubrepos=False):
|
|
804
|
'''show diff or diffstat.'''
|
|
804
|
'''show diff or diffstat.'''
|
|
805
|
if fp is None:
|
|
805
|
if fp is None:
|
|
806
|
write = ui.write
|
|
806
|
write = ui.write
|
|
807
|
else:
|
|
807
|
else:
|
|
808
|
def write(s, **kw):
|
|
808
|
def write(s, **kw):
|
|
809
|
fp.write(s)
|
|
809
|
fp.write(s)
|
|
810
|
|
|
810
|
|
|
811
|
if stat:
|
|
811
|
if stat:
|
|
812
|
diffopts = diffopts.copy(context=0)
|
|
812
|
diffopts = diffopts.copy(context=0)
|
|
813
|
width = 80
|
|
813
|
width = 80
|
|
814
|
if not ui.plain():
|
|
814
|
if not ui.plain():
|
|
815
|
width = ui.termwidth()
|
|
815
|
width = ui.termwidth()
|
|
816
|
chunks = patch.diff(repo, node1, node2, match, changes, diffopts,
|
|
816
|
chunks = patch.diff(repo, node1, node2, match, changes, diffopts,
|
|
817
|
prefix=prefix)
|
|
817
|
prefix=prefix)
|
|
818
|
for chunk, label in patch.diffstatui(util.iterlines(chunks),
|
|
818
|
for chunk, label in patch.diffstatui(util.iterlines(chunks),
|
|
819
|
width=width,
|
|
819
|
width=width,
|
|
820
|
git=diffopts.git):
|
|
820
|
git=diffopts.git):
|
|
821
|
write(chunk, label=label)
|
|
821
|
write(chunk, label=label)
|
|
822
|
else:
|
|
822
|
else:
|
|
823
|
for chunk, label in patch.diffui(repo, node1, node2, match,
|
|
823
|
for chunk, label in patch.diffui(repo, node1, node2, match,
|
|
824
|
changes, diffopts, prefix=prefix):
|
|
824
|
changes, diffopts, prefix=prefix):
|
|
825
|
write(chunk, label=label)
|
|
825
|
write(chunk, label=label)
|
|
826
|
|
|
826
|
|
|
827
|
if listsubrepos:
|
|
827
|
if listsubrepos:
|
|
828
|
ctx1 = repo[node1]
|
|
828
|
ctx1 = repo[node1]
|
|
829
|
ctx2 = repo[node2]
|
|
829
|
ctx2 = repo[node2]
|
|
830
|
for subpath, sub in scmutil.itersubrepos(ctx1, ctx2):
|
|
830
|
for subpath, sub in scmutil.itersubrepos(ctx1, ctx2):
|
|
831
|
tempnode2 = node2
|
|
831
|
tempnode2 = node2
|
|
832
|
try:
|
|
832
|
try:
|
|
833
|
if node2 is not None:
|
|
833
|
if node2 is not None:
|
|
834
|
tempnode2 = ctx2.substate[subpath][1]
|
|
834
|
tempnode2 = ctx2.substate[subpath][1]
|
|
835
|
except KeyError:
|
|
835
|
except KeyError:
|
|
836
|
# A subrepo that existed in node1 was deleted between node1 and
|
|
836
|
# A subrepo that existed in node1 was deleted between node1 and
|
|
837
|
# node2 (inclusive). Thus, ctx2's substate won't contain that
|
|
837
|
# node2 (inclusive). Thus, ctx2's substate won't contain that
|
|
838
|
# subpath. The best we can do is to ignore it.
|
|
838
|
# subpath. The best we can do is to ignore it.
|
|
839
|
tempnode2 = None
|
|
839
|
tempnode2 = None
|
|
840
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
840
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
841
|
sub.diff(ui, diffopts, tempnode2, submatch, changes=changes,
|
|
841
|
sub.diff(ui, diffopts, tempnode2, submatch, changes=changes,
|
|
842
|
stat=stat, fp=fp, prefix=prefix)
|
|
842
|
stat=stat, fp=fp, prefix=prefix)
|
|
843
|
|
|
843
|
|
|
844
|
class changeset_printer(object):
|
|
844
|
class changeset_printer(object):
|
|
845
|
'''show changeset information when templating not requested.'''
|
|
845
|
'''show changeset information when templating not requested.'''
|
|
846
|
|
|
846
|
|
|
847
|
def __init__(self, ui, repo, matchfn, diffopts, buffered):
|
|
847
|
def __init__(self, ui, repo, matchfn, diffopts, buffered):
|
|
848
|
self.ui = ui
|
|
848
|
self.ui = ui
|
|
849
|
self.repo = repo
|
|
849
|
self.repo = repo
|
|
850
|
self.buffered = buffered
|
|
850
|
self.buffered = buffered
|
|
851
|
self.matchfn = matchfn
|
|
851
|
self.matchfn = matchfn
|
|
852
|
self.diffopts = diffopts
|
|
852
|
self.diffopts = diffopts
|
|
853
|
self.header = {}
|
|
853
|
self.header = {}
|
|
854
|
self.hunk = {}
|
|
854
|
self.hunk = {}
|
|
855
|
self.lastheader = None
|
|
855
|
self.lastheader = None
|
|
856
|
self.footer = None
|
|
856
|
self.footer = None
|
|
857
|
|
|
857
|
|
|
858
|
def flush(self, rev):
|
|
858
|
def flush(self, rev):
|
|
859
|
if rev in self.header:
|
|
859
|
if rev in self.header:
|
|
860
|
h = self.header[rev]
|
|
860
|
h = self.header[rev]
|
|
861
|
if h != self.lastheader:
|
|
861
|
if h != self.lastheader:
|
|
862
|
self.lastheader = h
|
|
862
|
self.lastheader = h
|
|
863
|
self.ui.write(h)
|
|
863
|
self.ui.write(h)
|
|
864
|
del self.header[rev]
|
|
864
|
del self.header[rev]
|
|
865
|
if rev in self.hunk:
|
|
865
|
if rev in self.hunk:
|
|
866
|
self.ui.write(self.hunk[rev])
|
|
866
|
self.ui.write(self.hunk[rev])
|
|
867
|
del self.hunk[rev]
|
|
867
|
del self.hunk[rev]
|
|
868
|
return 1
|
|
868
|
return 1
|
|
869
|
return 0
|
|
869
|
return 0
|
|
870
|
|
|
870
|
|
|
871
|
def close(self):
|
|
871
|
def close(self):
|
|
872
|
if self.footer:
|
|
872
|
if self.footer:
|
|
873
|
self.ui.write(self.footer)
|
|
873
|
self.ui.write(self.footer)
|
|
874
|
|
|
874
|
|
|
875
|
def show(self, ctx, copies=None, matchfn=None, **props):
|
|
875
|
def show(self, ctx, copies=None, matchfn=None, **props):
|
|
876
|
if self.buffered:
|
|
876
|
if self.buffered:
|
|
877
|
self.ui.pushbuffer()
|
|
877
|
self.ui.pushbuffer()
|
|
878
|
self._show(ctx, copies, matchfn, props)
|
|
878
|
self._show(ctx, copies, matchfn, props)
|
|
879
|
self.hunk[ctx.rev()] = self.ui.popbuffer(labeled=True)
|
|
879
|
self.hunk[ctx.rev()] = self.ui.popbuffer(labeled=True)
|
|
880
|
else:
|
|
880
|
else:
|
|
881
|
self._show(ctx, copies, matchfn, props)
|
|
881
|
self._show(ctx, copies, matchfn, props)
|
|
882
|
|
|
882
|
|
|
883
|
def _show(self, ctx, copies, matchfn, props):
|
|
883
|
def _show(self, ctx, copies, matchfn, props):
|
|
884
|
'''show a single changeset or file revision'''
|
|
884
|
'''show a single changeset or file revision'''
|
|
885
|
changenode = ctx.node()
|
|
885
|
changenode = ctx.node()
|
|
886
|
rev = ctx.rev()
|
|
886
|
rev = ctx.rev()
|
|
887
|
|
|
887
|
|
|
888
|
if self.ui.quiet:
|
|
888
|
if self.ui.quiet:
|
|
889
|
self.ui.write("%d:%s\n" % (rev, short(changenode)),
|
|
889
|
self.ui.write("%d:%s\n" % (rev, short(changenode)),
|
|
890
|
label='log.node')
|
|
890
|
label='log.node')
|
|
891
|
return
|
|
891
|
return
|
|
892
|
|
|
892
|
|
|
893
|
log = self.repo.changelog
|
|
893
|
log = self.repo.changelog
|
|
894
|
date = util.datestr(ctx.date())
|
|
894
|
date = util.datestr(ctx.date())
|
|
895
|
|
|
895
|
|
|
896
|
hexfunc = self.ui.debugflag and hex or short
|
|
896
|
hexfunc = self.ui.debugflag and hex or short
|
|
897
|
|
|
897
|
|
|
898
|
parents = [(p, hexfunc(log.node(p)))
|
|
898
|
parents = [(p, hexfunc(log.node(p)))
|
|
899
|
for p in self._meaningful_parentrevs(log, rev)]
|
|
899
|
for p in self._meaningful_parentrevs(log, rev)]
|
|
900
|
|
|
900
|
|
|
901
|
# i18n: column positioning for "hg log"
|
|
901
|
# i18n: column positioning for "hg log"
|
|
902
|
self.ui.write(_("changeset: %d:%s\n") % (rev, hexfunc(changenode)),
|
|
902
|
self.ui.write(_("changeset: %d:%s\n") % (rev, hexfunc(changenode)),
|
|
903
|
label='log.changeset changeset.%s' % ctx.phasestr())
|
|
903
|
label='log.changeset changeset.%s' % ctx.phasestr())
|
|
904
|
|
|
904
|
|
|
905
|
# branches are shown first before any other names due to backwards
|
|
905
|
# branches are shown first before any other names due to backwards
|
|
906
|
# compatibility
|
|
906
|
# compatibility
|
|
907
|
branch = ctx.branch()
|
|
907
|
branch = ctx.branch()
|
|
908
|
# don't show the default branch name
|
|
908
|
# don't show the default branch name
|
|
909
|
if branch != 'default':
|
|
909
|
if branch != 'default':
|
|
910
|
# i18n: column positioning for "hg log"
|
|
910
|
# i18n: column positioning for "hg log"
|
|
911
|
self.ui.write(_("branch: %s\n") % branch,
|
|
911
|
self.ui.write(_("branch: %s\n") % branch,
|
|
912
|
label='log.branch')
|
|
912
|
label='log.branch')
|
|
913
|
|
|
913
|
|
|
914
|
for name, ns in self.repo.names.iteritems():
|
|
914
|
for name, ns in self.repo.names.iteritems():
|
|
915
|
# branches has special logic already handled above, so here we just
|
|
915
|
# branches has special logic already handled above, so here we just
|
|
916
|
# skip it
|
|
916
|
# skip it
|
|
917
|
if name == 'branches':
|
|
917
|
if name == 'branches':
|
|
918
|
continue
|
|
918
|
continue
|
|
919
|
# we will use the templatename as the color name since those two
|
|
919
|
# we will use the templatename as the color name since those two
|
|
920
|
# should be the same
|
|
920
|
# should be the same
|
|
921
|
for name in ns.names(self.repo, changenode):
|
|
921
|
for name in ns.names(self.repo, changenode):
|
|
922
|
self.ui.write(ns.logfmt % name,
|
|
922
|
self.ui.write(ns.logfmt % name,
|
|
923
|
label='log.%s' % ns.colorname)
|
|
923
|
label='log.%s' % ns.colorname)
|
|
924
|
if self.ui.debugflag:
|
|
924
|
if self.ui.debugflag:
|
|
925
|
# i18n: column positioning for "hg log"
|
|
925
|
# i18n: column positioning for "hg log"
|
|
926
|
self.ui.write(_("phase: %s\n") % _(ctx.phasestr()),
|
|
926
|
self.ui.write(_("phase: %s\n") % _(ctx.phasestr()),
|
|
927
|
label='log.phase')
|
|
927
|
label='log.phase')
|
|
928
|
for parent in parents:
|
|
928
|
for parent in parents:
|
|
929
|
label = 'log.parent changeset.%s' % self.repo[parent[0]].phasestr()
|
|
929
|
label = 'log.parent changeset.%s' % self.repo[parent[0]].phasestr()
|
|
930
|
# i18n: column positioning for "hg log"
|
|
930
|
# i18n: column positioning for "hg log"
|
|
931
|
self.ui.write(_("parent: %d:%s\n") % parent,
|
|
931
|
self.ui.write(_("parent: %d:%s\n") % parent,
|
|
932
|
label=label)
|
|
932
|
label=label)
|
|
933
|
|
|
933
|
|
|
934
|
if self.ui.debugflag:
|
|
934
|
if self.ui.debugflag:
|
|
935
|
mnode = ctx.manifestnode()
|
|
935
|
mnode = ctx.manifestnode()
|
|
936
|
# i18n: column positioning for "hg log"
|
|
936
|
# i18n: column positioning for "hg log"
|
|
937
|
self.ui.write(_("manifest: %d:%s\n") %
|
|
937
|
self.ui.write(_("manifest: %d:%s\n") %
|
|
938
|
(self.repo.manifest.rev(mnode), hex(mnode)),
|
|
938
|
(self.repo.manifest.rev(mnode), hex(mnode)),
|
|
939
|
label='ui.debug log.manifest')
|
|
939
|
label='ui.debug log.manifest')
|
|
940
|
# i18n: column positioning for "hg log"
|
|
940
|
# i18n: column positioning for "hg log"
|
|
941
|
self.ui.write(_("user: %s\n") % ctx.user(),
|
|
941
|
self.ui.write(_("user: %s\n") % ctx.user(),
|
|
942
|
label='log.user')
|
|
942
|
label='log.user')
|
|
943
|
# i18n: column positioning for "hg log"
|
|
943
|
# i18n: column positioning for "hg log"
|
|
944
|
self.ui.write(_("date: %s\n") % date,
|
|
944
|
self.ui.write(_("date: %s\n") % date,
|
|
945
|
label='log.date')
|
|
945
|
label='log.date')
|
|
946
|
|
|
946
|
|
|
947
|
if self.ui.debugflag:
|
|
947
|
if self.ui.debugflag:
|
|
948
|
files = self.repo.status(log.parents(changenode)[0], changenode)[:3]
|
|
948
|
files = self.repo.status(log.parents(changenode)[0], changenode)[:3]
|
|
949
|
for key, value in zip([# i18n: column positioning for "hg log"
|
|
949
|
for key, value in zip([# i18n: column positioning for "hg log"
|
|
950
|
_("files:"),
|
|
950
|
_("files:"),
|
|
951
|
# i18n: column positioning for "hg log"
|
|
951
|
# i18n: column positioning for "hg log"
|
|
952
|
_("files+:"),
|
|
952
|
_("files+:"),
|
|
953
|
# i18n: column positioning for "hg log"
|
|
953
|
# i18n: column positioning for "hg log"
|
|
954
|
_("files-:")], files):
|
|
954
|
_("files-:")], files):
|
|
955
|
if value:
|
|
955
|
if value:
|
|
956
|
self.ui.write("%-12s %s\n" % (key, " ".join(value)),
|
|
956
|
self.ui.write("%-12s %s\n" % (key, " ".join(value)),
|
|
957
|
label='ui.debug log.files')
|
|
957
|
label='ui.debug log.files')
|
|
958
|
elif ctx.files() and self.ui.verbose:
|
|
958
|
elif ctx.files() and self.ui.verbose:
|
|
959
|
# i18n: column positioning for "hg log"
|
|
959
|
# i18n: column positioning for "hg log"
|
|
960
|
self.ui.write(_("files: %s\n") % " ".join(ctx.files()),
|
|
960
|
self.ui.write(_("files: %s\n") % " ".join(ctx.files()),
|
|
961
|
label='ui.note log.files')
|
|
961
|
label='ui.note log.files')
|
|
962
|
if copies and self.ui.verbose:
|
|
962
|
if copies and self.ui.verbose:
|
|
963
|
copies = ['%s (%s)' % c for c in copies]
|
|
963
|
copies = ['%s (%s)' % c for c in copies]
|
|
964
|
# i18n: column positioning for "hg log"
|
|
964
|
# i18n: column positioning for "hg log"
|
|
965
|
self.ui.write(_("copies: %s\n") % ' '.join(copies),
|
|
965
|
self.ui.write(_("copies: %s\n") % ' '.join(copies),
|
|
966
|
label='ui.note log.copies')
|
|
966
|
label='ui.note log.copies')
|
|
967
|
|
|
967
|
|
|
968
|
extra = ctx.extra()
|
|
968
|
extra = ctx.extra()
|
|
969
|
if extra and self.ui.debugflag:
|
|
969
|
if extra and self.ui.debugflag:
|
|
970
|
for key, value in sorted(extra.items()):
|
|
970
|
for key, value in sorted(extra.items()):
|
|
971
|
# i18n: column positioning for "hg log"
|
|
971
|
# i18n: column positioning for "hg log"
|
|
972
|
self.ui.write(_("extra: %s=%s\n")
|
|
972
|
self.ui.write(_("extra: %s=%s\n")
|
|
973
|
% (key, value.encode('string_escape')),
|
|
973
|
% (key, value.encode('string_escape')),
|
|
974
|
label='ui.debug log.extra')
|
|
974
|
label='ui.debug log.extra')
|
|
975
|
|
|
975
|
|
|
976
|
description = ctx.description().strip()
|
|
976
|
description = ctx.description().strip()
|
|
977
|
if description:
|
|
977
|
if description:
|
|
978
|
if self.ui.verbose:
|
|
978
|
if self.ui.verbose:
|
|
979
|
self.ui.write(_("description:\n"),
|
|
979
|
self.ui.write(_("description:\n"),
|
|
980
|
label='ui.note log.description')
|
|
980
|
label='ui.note log.description')
|
|
981
|
self.ui.write(description,
|
|
981
|
self.ui.write(description,
|
|
982
|
label='ui.note log.description')
|
|
982
|
label='ui.note log.description')
|
|
983
|
self.ui.write("\n\n")
|
|
983
|
self.ui.write("\n\n")
|
|
984
|
else:
|
|
984
|
else:
|
|
985
|
# i18n: column positioning for "hg log"
|
|
985
|
# i18n: column positioning for "hg log"
|
|
986
|
self.ui.write(_("summary: %s\n") %
|
|
986
|
self.ui.write(_("summary: %s\n") %
|
|
987
|
description.splitlines()[0],
|
|
987
|
description.splitlines()[0],
|
|
988
|
label='log.summary')
|
|
988
|
label='log.summary')
|
|
989
|
self.ui.write("\n")
|
|
989
|
self.ui.write("\n")
|
|
990
|
|
|
990
|
|
|
991
|
self.showpatch(changenode, matchfn)
|
|
991
|
self.showpatch(changenode, matchfn)
|
|
992
|
|
|
992
|
|
|
993
|
def showpatch(self, node, matchfn):
|
|
993
|
def showpatch(self, node, matchfn):
|
|
994
|
if not matchfn:
|
|
994
|
if not matchfn:
|
|
995
|
matchfn = self.matchfn
|
|
995
|
matchfn = self.matchfn
|
|
996
|
if matchfn:
|
|
996
|
if matchfn:
|
|
997
|
stat = self.diffopts.get('stat')
|
|
997
|
stat = self.diffopts.get('stat')
|
|
998
|
diff = self.diffopts.get('patch')
|
|
998
|
diff = self.diffopts.get('patch')
|
|
999
|
diffopts = patch.diffallopts(self.ui, self.diffopts)
|
|
999
|
diffopts = patch.diffallopts(self.ui, self.diffopts)
|
|
1000
|
prev = self.repo.changelog.parents(node)[0]
|
|
1000
|
prev = self.repo.changelog.parents(node)[0]
|
|
1001
|
if stat:
|
|
1001
|
if stat:
|
|
1002
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1002
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1003
|
match=matchfn, stat=True)
|
|
1003
|
match=matchfn, stat=True)
|
|
1004
|
if diff:
|
|
1004
|
if diff:
|
|
1005
|
if stat:
|
|
1005
|
if stat:
|
|
1006
|
self.ui.write("\n")
|
|
1006
|
self.ui.write("\n")
|
|
1007
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1007
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1008
|
match=matchfn, stat=False)
|
|
1008
|
match=matchfn, stat=False)
|
|
1009
|
self.ui.write("\n")
|
|
1009
|
self.ui.write("\n")
|
|
1010
|
|
|
1010
|
|
|
1011
|
def _meaningful_parentrevs(self, log, rev):
|
|
1011
|
def _meaningful_parentrevs(self, log, rev):
|
|
1012
|
"""Return list of meaningful (or all if debug) parentrevs for rev.
|
|
1012
|
"""Return list of meaningful (or all if debug) parentrevs for rev.
|
|
1013
|
|
|
1013
|
|
|
1014
|
For merges (two non-nullrev revisions) both parents are meaningful.
|
|
1014
|
For merges (two non-nullrev revisions) both parents are meaningful.
|
|
1015
|
Otherwise the first parent revision is considered meaningful if it
|
|
1015
|
Otherwise the first parent revision is considered meaningful if it
|
|
1016
|
is not the preceding revision.
|
|
1016
|
is not the preceding revision.
|
|
1017
|
"""
|
|
1017
|
"""
|
|
1018
|
parents = log.parentrevs(rev)
|
|
1018
|
parents = log.parentrevs(rev)
|
|
1019
|
if not self.ui.debugflag and parents[1] == nullrev:
|
|
1019
|
if not self.ui.debugflag and parents[1] == nullrev:
|
|
1020
|
if parents[0] >= rev - 1:
|
|
1020
|
if parents[0] >= rev - 1:
|
|
1021
|
parents = []
|
|
1021
|
parents = []
|
|
1022
|
else:
|
|
1022
|
else:
|
|
1023
|
parents = [parents[0]]
|
|
1023
|
parents = [parents[0]]
|
|
1024
|
return parents
|
|
1024
|
return parents
|
|
1025
|
|
|
1025
|
|
|
1026
|
class jsonchangeset(changeset_printer):
|
|
1026
|
class jsonchangeset(changeset_printer):
|
|
1027
|
'''format changeset information.'''
|
|
1027
|
'''format changeset information.'''
|
|
1028
|
|
|
1028
|
|
|
1029
|
def __init__(self, ui, repo, matchfn, diffopts, buffered):
|
|
1029
|
def __init__(self, ui, repo, matchfn, diffopts, buffered):
|
|
1030
|
changeset_printer.__init__(self, ui, repo, matchfn, diffopts, buffered)
|
|
1030
|
changeset_printer.__init__(self, ui, repo, matchfn, diffopts, buffered)
|
|
1031
|
self.cache = {}
|
|
1031
|
self.cache = {}
|
|
1032
|
self._first = True
|
|
1032
|
self._first = True
|
|
1033
|
|
|
1033
|
|
|
1034
|
def close(self):
|
|
1034
|
def close(self):
|
|
1035
|
if not self._first:
|
|
1035
|
if not self._first:
|
|
1036
|
self.ui.write("\n]\n")
|
|
1036
|
self.ui.write("\n]\n")
|
|
1037
|
else:
|
|
1037
|
else:
|
|
1038
|
self.ui.write("[]\n")
|
|
1038
|
self.ui.write("[]\n")
|
|
1039
|
|
|
1039
|
|
|
1040
|
def _show(self, ctx, copies, matchfn, props):
|
|
1040
|
def _show(self, ctx, copies, matchfn, props):
|
|
1041
|
'''show a single changeset or file revision'''
|
|
1041
|
'''show a single changeset or file revision'''
|
|
1042
|
hexnode = hex(ctx.node())
|
|
1042
|
hexnode = hex(ctx.node())
|
|
1043
|
rev = ctx.rev()
|
|
1043
|
rev = ctx.rev()
|
|
1044
|
j = encoding.jsonescape
|
|
1044
|
j = encoding.jsonescape
|
|
1045
|
|
|
1045
|
|
|
1046
|
if self._first:
|
|
1046
|
if self._first:
|
|
1047
|
self.ui.write("[\n {")
|
|
1047
|
self.ui.write("[\n {")
|
|
1048
|
self._first = False
|
|
1048
|
self._first = False
|
|
1049
|
else:
|
|
1049
|
else:
|
|
1050
|
self.ui.write(",\n {")
|
|
1050
|
self.ui.write(",\n {")
|
|
1051
|
|
|
1051
|
|
|
1052
|
if self.ui.quiet:
|
|
1052
|
if self.ui.quiet:
|
|
1053
|
self.ui.write('\n "rev": %d' % rev)
|
|
1053
|
self.ui.write('\n "rev": %d' % rev)
|
|
1054
|
self.ui.write(',\n "node": "%s"' % hexnode)
|
|
1054
|
self.ui.write(',\n "node": "%s"' % hexnode)
|
|
1055
|
self.ui.write('\n }')
|
|
1055
|
self.ui.write('\n }')
|
|
1056
|
return
|
|
1056
|
return
|
|
1057
|
|
|
1057
|
|
|
1058
|
self.ui.write('\n "rev": %d' % rev)
|
|
1058
|
self.ui.write('\n "rev": %d' % rev)
|
|
1059
|
self.ui.write(',\n "node": "%s"' % hexnode)
|
|
1059
|
self.ui.write(',\n "node": "%s"' % hexnode)
|
|
1060
|
self.ui.write(',\n "branch": "%s"' % j(ctx.branch()))
|
|
1060
|
self.ui.write(',\n "branch": "%s"' % j(ctx.branch()))
|
|
1061
|
self.ui.write(',\n "phase": "%s"' % ctx.phasestr())
|
|
1061
|
self.ui.write(',\n "phase": "%s"' % ctx.phasestr())
|
|
1062
|
self.ui.write(',\n "user": "%s"' % j(ctx.user()))
|
|
1062
|
self.ui.write(',\n "user": "%s"' % j(ctx.user()))
|
|
1063
|
self.ui.write(',\n "date": [%d, %d]' % ctx.date())
|
|
1063
|
self.ui.write(',\n "date": [%d, %d]' % ctx.date())
|
|
1064
|
self.ui.write(',\n "desc": "%s"' % j(ctx.description()))
|
|
1064
|
self.ui.write(',\n "desc": "%s"' % j(ctx.description()))
|
|
1065
|
|
|
1065
|
|
|
1066
|
self.ui.write(',\n "bookmarks": [%s]' %
|
|
1066
|
self.ui.write(',\n "bookmarks": [%s]' %
|
|
1067
|
", ".join('"%s"' % j(b) for b in ctx.bookmarks()))
|
|
1067
|
", ".join('"%s"' % j(b) for b in ctx.bookmarks()))
|
|
1068
|
self.ui.write(',\n "tags": [%s]' %
|
|
1068
|
self.ui.write(',\n "tags": [%s]' %
|
|
1069
|
", ".join('"%s"' % j(t) for t in ctx.tags()))
|
|
1069
|
", ".join('"%s"' % j(t) for t in ctx.tags()))
|
|
1070
|
self.ui.write(',\n "parents": [%s]' %
|
|
1070
|
self.ui.write(',\n "parents": [%s]' %
|
|
1071
|
", ".join('"%s"' % c.hex() for c in ctx.parents()))
|
|
1071
|
", ".join('"%s"' % c.hex() for c in ctx.parents()))
|
|
1072
|
|
|
1072
|
|
|
1073
|
if self.ui.debugflag:
|
|
1073
|
if self.ui.debugflag:
|
|
1074
|
self.ui.write(',\n "manifest": "%s"' % hex(ctx.manifestnode()))
|
|
1074
|
self.ui.write(',\n "manifest": "%s"' % hex(ctx.manifestnode()))
|
|
1075
|
|
|
1075
|
|
|
1076
|
self.ui.write(',\n "extra": {%s}' %
|
|
1076
|
self.ui.write(',\n "extra": {%s}' %
|
|
1077
|
", ".join('"%s": "%s"' % (j(k), j(v))
|
|
1077
|
", ".join('"%s": "%s"' % (j(k), j(v))
|
|
1078
|
for k, v in ctx.extra().items()))
|
|
1078
|
for k, v in ctx.extra().items()))
|
|
1079
|
|
|
1079
|
|
|
1080
|
files = ctx.p1().status(ctx)
|
|
1080
|
files = ctx.p1().status(ctx)
|
|
1081
|
self.ui.write(',\n "modified": [%s]' %
|
|
1081
|
self.ui.write(',\n "modified": [%s]' %
|
|
1082
|
", ".join('"%s"' % j(f) for f in files[0]))
|
|
1082
|
", ".join('"%s"' % j(f) for f in files[0]))
|
|
1083
|
self.ui.write(',\n "added": [%s]' %
|
|
1083
|
self.ui.write(',\n "added": [%s]' %
|
|
1084
|
", ".join('"%s"' % j(f) for f in files[1]))
|
|
1084
|
", ".join('"%s"' % j(f) for f in files[1]))
|
|
1085
|
self.ui.write(',\n "removed": [%s]' %
|
|
1085
|
self.ui.write(',\n "removed": [%s]' %
|
|
1086
|
", ".join('"%s"' % j(f) for f in files[2]))
|
|
1086
|
", ".join('"%s"' % j(f) for f in files[2]))
|
|
1087
|
|
|
1087
|
|
|
1088
|
elif self.ui.verbose:
|
|
1088
|
elif self.ui.verbose:
|
|
1089
|
self.ui.write(',\n "files": [%s]' %
|
|
1089
|
self.ui.write(',\n "files": [%s]' %
|
|
1090
|
", ".join('"%s"' % j(f) for f in ctx.files()))
|
|
1090
|
", ".join('"%s"' % j(f) for f in ctx.files()))
|
|
1091
|
|
|
1091
|
|
|
1092
|
if copies:
|
|
1092
|
if copies:
|
|
1093
|
self.ui.write(',\n "copies": {%s}' %
|
|
1093
|
self.ui.write(',\n "copies": {%s}' %
|
|
1094
|
", ".join('"%s": "%s"' % (j(k), j(v))
|
|
1094
|
", ".join('"%s": "%s"' % (j(k), j(v))
|
|
1095
|
for k, v in copies))
|
|
1095
|
for k, v in copies))
|
|
1096
|
|
|
1096
|
|
|
1097
|
matchfn = self.matchfn
|
|
1097
|
matchfn = self.matchfn
|
|
1098
|
if matchfn:
|
|
1098
|
if matchfn:
|
|
1099
|
stat = self.diffopts.get('stat')
|
|
1099
|
stat = self.diffopts.get('stat')
|
|
1100
|
diff = self.diffopts.get('patch')
|
|
1100
|
diff = self.diffopts.get('patch')
|
|
1101
|
diffopts = patch.difffeatureopts(self.ui, self.diffopts, git=True)
|
|
1101
|
diffopts = patch.difffeatureopts(self.ui, self.diffopts, git=True)
|
|
1102
|
node, prev = ctx.node(), ctx.p1().node()
|
|
1102
|
node, prev = ctx.node(), ctx.p1().node()
|
|
1103
|
if stat:
|
|
1103
|
if stat:
|
|
1104
|
self.ui.pushbuffer()
|
|
1104
|
self.ui.pushbuffer()
|
|
1105
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1105
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1106
|
match=matchfn, stat=True)
|
|
1106
|
match=matchfn, stat=True)
|
|
1107
|
self.ui.write(',\n "diffstat": "%s"' % j(self.ui.popbuffer()))
|
|
1107
|
self.ui.write(',\n "diffstat": "%s"' % j(self.ui.popbuffer()))
|
|
1108
|
if diff:
|
|
1108
|
if diff:
|
|
1109
|
self.ui.pushbuffer()
|
|
1109
|
self.ui.pushbuffer()
|
|
1110
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1110
|
diffordiffstat(self.ui, self.repo, diffopts, prev, node,
|
|
1111
|
match=matchfn, stat=False)
|
|
1111
|
match=matchfn, stat=False)
|
|
1112
|
self.ui.write(',\n "diff": "%s"' % j(self.ui.popbuffer()))
|
|
1112
|
self.ui.write(',\n "diff": "%s"' % j(self.ui.popbuffer()))
|
|
1113
|
|
|
1113
|
|
|
1114
|
self.ui.write("\n }")
|
|
1114
|
self.ui.write("\n }")
|
|
1115
|
|
|
1115
|
|
|
1116
|
class changeset_templater(changeset_printer):
|
|
1116
|
class changeset_templater(changeset_printer):
|
|
1117
|
'''format changeset information.'''
|
|
1117
|
'''format changeset information.'''
|
|
1118
|
|
|
1118
|
|
|
1119
|
def __init__(self, ui, repo, matchfn, diffopts, tmpl, mapfile, buffered):
|
|
1119
|
def __init__(self, ui, repo, matchfn, diffopts, tmpl, mapfile, buffered):
|
|
1120
|
changeset_printer.__init__(self, ui, repo, matchfn, diffopts, buffered)
|
|
1120
|
changeset_printer.__init__(self, ui, repo, matchfn, diffopts, buffered)
|
|
1121
|
formatnode = ui.debugflag and (lambda x: x) or (lambda x: x[:12])
|
|
1121
|
formatnode = ui.debugflag and (lambda x: x) or (lambda x: x[:12])
|
|
1122
|
defaulttempl = {
|
|
1122
|
defaulttempl = {
|
|
1123
|
'parent': '{rev}:{node|formatnode} ',
|
|
1123
|
'parent': '{rev}:{node|formatnode} ',
|
|
1124
|
'manifest': '{rev}:{node|formatnode}',
|
|
1124
|
'manifest': '{rev}:{node|formatnode}',
|
|
1125
|
'file_copy': '{name} ({source})',
|
|
1125
|
'file_copy': '{name} ({source})',
|
|
1126
|
'extra': '{key}={value|stringescape}'
|
|
1126
|
'extra': '{key}={value|stringescape}'
|
|
1127
|
}
|
|
1127
|
}
|
|
1128
|
# filecopy is preserved for compatibility reasons
|
|
1128
|
# filecopy is preserved for compatibility reasons
|
|
1129
|
defaulttempl['filecopy'] = defaulttempl['file_copy']
|
|
1129
|
defaulttempl['filecopy'] = defaulttempl['file_copy']
|
|
1130
|
self.t = templater.templater(mapfile, {'formatnode': formatnode},
|
|
1130
|
self.t = templater.templater(mapfile, {'formatnode': formatnode},
|
|
1131
|
cache=defaulttempl)
|
|
1131
|
cache=defaulttempl)
|
|
1132
|
if tmpl:
|
|
1132
|
if tmpl:
|
|
1133
|
self.t.cache['changeset'] = tmpl
|
|
1133
|
self.t.cache['changeset'] = tmpl
|
|
1134
|
|
|
1134
|
|
|
1135
|
self.cache = {}
|
|
1135
|
self.cache = {}
|
|
1136
|
|
|
1136
|
|
|
1137
|
def _meaningful_parentrevs(self, ctx):
|
|
1137
|
def _meaningful_parentrevs(self, ctx):
|
|
1138
|
"""Return list of meaningful (or all if debug) parentrevs for rev.
|
|
1138
|
"""Return list of meaningful (or all if debug) parentrevs for rev.
|
|
1139
|
"""
|
|
1139
|
"""
|
|
1140
|
parents = ctx.parents()
|
|
1140
|
parents = ctx.parents()
|
|
1141
|
if len(parents) > 1:
|
|
1141
|
if len(parents) > 1:
|
|
1142
|
return parents
|
|
1142
|
return parents
|
|
1143
|
if self.ui.debugflag:
|
|
1143
|
if self.ui.debugflag:
|
|
1144
|
return [parents[0], self.repo['null']]
|
|
1144
|
return [parents[0], self.repo['null']]
|
|
1145
|
if parents[0].rev() >= ctx.rev() - 1:
|
|
1145
|
if parents[0].rev() >= ctx.rev() - 1:
|
|
1146
|
return []
|
|
1146
|
return []
|
|
1147
|
return parents
|
|
1147
|
return parents
|
|
1148
|
|
|
1148
|
|
|
1149
|
def _show(self, ctx, copies, matchfn, props):
|
|
1149
|
def _show(self, ctx, copies, matchfn, props):
|
|
1150
|
'''show a single changeset or file revision'''
|
|
1150
|
'''show a single changeset or file revision'''
|
|
1151
|
|
|
1151
|
|
|
1152
|
showlist = templatekw.showlist
|
|
1152
|
showlist = templatekw.showlist
|
|
1153
|
|
|
1153
|
|
|
1154
|
# showparents() behaviour depends on ui trace level which
|
|
1154
|
# showparents() behaviour depends on ui trace level which
|
|
1155
|
# causes unexpected behaviours at templating level and makes
|
|
1155
|
# causes unexpected behaviours at templating level and makes
|
|
1156
|
# it harder to extract it in a standalone function. Its
|
|
1156
|
# it harder to extract it in a standalone function. Its
|
|
1157
|
# behaviour cannot be changed so leave it here for now.
|
|
1157
|
# behaviour cannot be changed so leave it here for now.
|
|
1158
|
def showparents(**args):
|
|
1158
|
def showparents(**args):
|
|
1159
|
ctx = args['ctx']
|
|
1159
|
ctx = args['ctx']
|
|
1160
|
parents = [[('rev', p.rev()),
|
|
1160
|
parents = [[('rev', p.rev()),
|
|
1161
|
('node', p.hex()),
|
|
1161
|
('node', p.hex()),
|
|
1162
|
('phase', p.phasestr())]
|
|
1162
|
('phase', p.phasestr())]
|
|
1163
|
for p in self._meaningful_parentrevs(ctx)]
|
|
1163
|
for p in self._meaningful_parentrevs(ctx)]
|
|
1164
|
return showlist('parent', parents, **args)
|
|
1164
|
return showlist('parent', parents, **args)
|
|
1165
|
|
|
1165
|
|
|
1166
|
props = props.copy()
|
|
1166
|
props = props.copy()
|
|
1167
|
props.update(templatekw.keywords)
|
|
1167
|
props.update(templatekw.keywords)
|
|
1168
|
props['parents'] = showparents
|
|
1168
|
props['parents'] = showparents
|
|
1169
|
props['templ'] = self.t
|
|
1169
|
props['templ'] = self.t
|
|
1170
|
props['ctx'] = ctx
|
|
1170
|
props['ctx'] = ctx
|
|
1171
|
props['repo'] = self.repo
|
|
1171
|
props['repo'] = self.repo
|
|
1172
|
props['revcache'] = {'copies': copies}
|
|
1172
|
props['revcache'] = {'copies': copies}
|
|
1173
|
props['cache'] = self.cache
|
|
1173
|
props['cache'] = self.cache
|
|
1174
|
|
|
1174
|
|
|
1175
|
# find correct templates for current mode
|
|
1175
|
# find correct templates for current mode
|
|
1176
|
|
|
1176
|
|
|
1177
|
tmplmodes = [
|
|
1177
|
tmplmodes = [
|
|
1178
|
(True, None),
|
|
1178
|
(True, None),
|
|
1179
|
(self.ui.verbose, 'verbose'),
|
|
1179
|
(self.ui.verbose, 'verbose'),
|
|
1180
|
(self.ui.quiet, 'quiet'),
|
|
1180
|
(self.ui.quiet, 'quiet'),
|
|
1181
|
(self.ui.debugflag, 'debug'),
|
|
1181
|
(self.ui.debugflag, 'debug'),
|
|
1182
|
]
|
|
1182
|
]
|
|
1183
|
|
|
1183
|
|
|
1184
|
types = {'header': '', 'footer':'', 'changeset': 'changeset'}
|
|
1184
|
types = {'header': '', 'footer':'', 'changeset': 'changeset'}
|
|
1185
|
for mode, postfix in tmplmodes:
|
|
1185
|
for mode, postfix in tmplmodes:
|
|
1186
|
for type in types:
|
|
1186
|
for type in types:
|
|
1187
|
cur = postfix and ('%s_%s' % (type, postfix)) or type
|
|
1187
|
cur = postfix and ('%s_%s' % (type, postfix)) or type
|
|
1188
|
if mode and cur in self.t:
|
|
1188
|
if mode and cur in self.t:
|
|
1189
|
types[type] = cur
|
|
1189
|
types[type] = cur
|
|
1190
|
|
|
1190
|
|
|
1191
|
try:
|
|
1191
|
try:
|
|
1192
|
|
|
1192
|
|
|
1193
|
# write header
|
|
1193
|
# write header
|
|
1194
|
if types['header']:
|
|
1194
|
if types['header']:
|
|
1195
|
h = templater.stringify(self.t(types['header'], **props))
|
|
1195
|
h = templater.stringify(self.t(types['header'], **props))
|
|
1196
|
if self.buffered:
|
|
1196
|
if self.buffered:
|
|
1197
|
self.header[ctx.rev()] = h
|
|
1197
|
self.header[ctx.rev()] = h
|
|
1198
|
else:
|
|
1198
|
else:
|
|
1199
|
if self.lastheader != h:
|
|
1199
|
if self.lastheader != h:
|
|
1200
|
self.lastheader = h
|
|
1200
|
self.lastheader = h
|
|
1201
|
self.ui.write(h)
|
|
1201
|
self.ui.write(h)
|
|
1202
|
|
|
1202
|
|
|
1203
|
# write changeset metadata, then patch if requested
|
|
1203
|
# write changeset metadata, then patch if requested
|
|
1204
|
key = types['changeset']
|
|
1204
|
key = types['changeset']
|
|
1205
|
self.ui.write(templater.stringify(self.t(key, **props)))
|
|
1205
|
self.ui.write(templater.stringify(self.t(key, **props)))
|
|
1206
|
self.showpatch(ctx.node(), matchfn)
|
|
1206
|
self.showpatch(ctx.node(), matchfn)
|
|
1207
|
|
|
1207
|
|
|
1208
|
if types['footer']:
|
|
1208
|
if types['footer']:
|
|
1209
|
if not self.footer:
|
|
1209
|
if not self.footer:
|
|
1210
|
self.footer = templater.stringify(self.t(types['footer'],
|
|
1210
|
self.footer = templater.stringify(self.t(types['footer'],
|
|
1211
|
**props))
|
|
1211
|
**props))
|
|
1212
|
|
|
1212
|
|
|
1213
|
except KeyError, inst:
|
|
1213
|
except KeyError, inst:
|
|
1214
|
msg = _("%s: no key named '%s'")
|
|
1214
|
msg = _("%s: no key named '%s'")
|
|
1215
|
raise util.Abort(msg % (self.t.mapfile, inst.args[0]))
|
|
1215
|
raise util.Abort(msg % (self.t.mapfile, inst.args[0]))
|
|
1216
|
except SyntaxError, inst:
|
|
1216
|
except SyntaxError, inst:
|
|
1217
|
raise util.Abort('%s: %s' % (self.t.mapfile, inst.args[0]))
|
|
1217
|
raise util.Abort('%s: %s' % (self.t.mapfile, inst.args[0]))
|
|
1218
|
|
|
1218
|
|
|
1219
|
def gettemplate(ui, tmpl, style):
|
|
1219
|
def gettemplate(ui, tmpl, style):
|
|
1220
|
"""
|
|
1220
|
"""
|
|
1221
|
Find the template matching the given template spec or style.
|
|
1221
|
Find the template matching the given template spec or style.
|
|
1222
|
"""
|
|
1222
|
"""
|
|
1223
|
|
|
1223
|
|
|
1224
|
# ui settings
|
|
1224
|
# ui settings
|
|
1225
|
if not tmpl and not style: # template are stronger than style
|
|
1225
|
if not tmpl and not style: # template are stronger than style
|
|
1226
|
tmpl = ui.config('ui', 'logtemplate')
|
|
1226
|
tmpl = ui.config('ui', 'logtemplate')
|
|
1227
|
if tmpl:
|
|
1227
|
if tmpl:
|
|
1228
|
try:
|
|
1228
|
try:
|
|
1229
|
tmpl = templater.parsestring(tmpl)
|
|
1229
|
tmpl = templater.parsestring(tmpl)
|
|
1230
|
except SyntaxError:
|
|
1230
|
except SyntaxError:
|
|
1231
|
tmpl = templater.parsestring(tmpl, quoted=False)
|
|
1231
|
tmpl = templater.parsestring(tmpl, quoted=False)
|
|
1232
|
return tmpl, None
|
|
1232
|
return tmpl, None
|
|
1233
|
else:
|
|
1233
|
else:
|
|
1234
|
style = util.expandpath(ui.config('ui', 'style', ''))
|
|
1234
|
style = util.expandpath(ui.config('ui', 'style', ''))
|
|
1235
|
|
|
1235
|
|
|
1236
|
if not tmpl and style:
|
|
1236
|
if not tmpl and style:
|
|
1237
|
mapfile = style
|
|
1237
|
mapfile = style
|
|
1238
|
if not os.path.split(mapfile)[0]:
|
|
1238
|
if not os.path.split(mapfile)[0]:
|
|
1239
|
mapname = (templater.templatepath('map-cmdline.' + mapfile)
|
|
1239
|
mapname = (templater.templatepath('map-cmdline.' + mapfile)
|
|
1240
|
or templater.templatepath(mapfile))
|
|
1240
|
or templater.templatepath(mapfile))
|
|
1241
|
if mapname:
|
|
1241
|
if mapname:
|
|
1242
|
mapfile = mapname
|
|
1242
|
mapfile = mapname
|
|
1243
|
return None, mapfile
|
|
1243
|
return None, mapfile
|
|
1244
|
|
|
1244
|
|
|
1245
|
if not tmpl:
|
|
1245
|
if not tmpl:
|
|
1246
|
return None, None
|
|
1246
|
return None, None
|
|
1247
|
|
|
1247
|
|
|
1248
|
# looks like a literal template?
|
|
1248
|
# looks like a literal template?
|
|
1249
|
if '{' in tmpl:
|
|
1249
|
if '{' in tmpl:
|
|
1250
|
return tmpl, None
|
|
1250
|
return tmpl, None
|
|
1251
|
|
|
1251
|
|
|
1252
|
# perhaps a stock style?
|
|
1252
|
# perhaps a stock style?
|
|
1253
|
if not os.path.split(tmpl)[0]:
|
|
1253
|
if not os.path.split(tmpl)[0]:
|
|
1254
|
mapname = (templater.templatepath('map-cmdline.' + tmpl)
|
|
1254
|
mapname = (templater.templatepath('map-cmdline.' + tmpl)
|
|
1255
|
or templater.templatepath(tmpl))
|
|
1255
|
or templater.templatepath(tmpl))
|
|
1256
|
if mapname and os.path.isfile(mapname):
|
|
1256
|
if mapname and os.path.isfile(mapname):
|
|
1257
|
return None, mapname
|
|
1257
|
return None, mapname
|
|
1258
|
|
|
1258
|
|
|
1259
|
# perhaps it's a reference to [templates]
|
|
1259
|
# perhaps it's a reference to [templates]
|
|
1260
|
t = ui.config('templates', tmpl)
|
|
1260
|
t = ui.config('templates', tmpl)
|
|
1261
|
if t:
|
|
1261
|
if t:
|
|
1262
|
try:
|
|
1262
|
try:
|
|
1263
|
tmpl = templater.parsestring(t)
|
|
1263
|
tmpl = templater.parsestring(t)
|
|
1264
|
except SyntaxError:
|
|
1264
|
except SyntaxError:
|
|
1265
|
tmpl = templater.parsestring(t, quoted=False)
|
|
1265
|
tmpl = templater.parsestring(t, quoted=False)
|
|
1266
|
return tmpl, None
|
|
1266
|
return tmpl, None
|
|
1267
|
|
|
1267
|
|
|
1268
|
if tmpl == 'list':
|
|
1268
|
if tmpl == 'list':
|
|
1269
|
ui.write(_("available styles: %s\n") % templater.stylelist())
|
|
1269
|
ui.write(_("available styles: %s\n") % templater.stylelist())
|
|
1270
|
raise util.Abort(_("specify a template"))
|
|
1270
|
raise util.Abort(_("specify a template"))
|
|
1271
|
|
|
1271
|
|
|
1272
|
# perhaps it's a path to a map or a template
|
|
1272
|
# perhaps it's a path to a map or a template
|
|
1273
|
if ('/' in tmpl or '\\' in tmpl) and os.path.isfile(tmpl):
|
|
1273
|
if ('/' in tmpl or '\\' in tmpl) and os.path.isfile(tmpl):
|
|
1274
|
# is it a mapfile for a style?
|
|
1274
|
# is it a mapfile for a style?
|
|
1275
|
if os.path.basename(tmpl).startswith("map-"):
|
|
1275
|
if os.path.basename(tmpl).startswith("map-"):
|
|
1276
|
return None, os.path.realpath(tmpl)
|
|
1276
|
return None, os.path.realpath(tmpl)
|
|
1277
|
tmpl = open(tmpl).read()
|
|
1277
|
tmpl = open(tmpl).read()
|
|
1278
|
return tmpl, None
|
|
1278
|
return tmpl, None
|
|
1279
|
|
|
1279
|
|
|
1280
|
# constant string?
|
|
1280
|
# constant string?
|
|
1281
|
return tmpl, None
|
|
1281
|
return tmpl, None
|
|
1282
|
|
|
1282
|
|
|
1283
|
def show_changeset(ui, repo, opts, buffered=False):
|
|
1283
|
def show_changeset(ui, repo, opts, buffered=False):
|
|
1284
|
"""show one changeset using template or regular display.
|
|
1284
|
"""show one changeset using template or regular display.
|
|
1285
|
|
|
1285
|
|
|
1286
|
Display format will be the first non-empty hit of:
|
|
1286
|
Display format will be the first non-empty hit of:
|
|
1287
|
1. option 'template'
|
|
1287
|
1. option 'template'
|
|
1288
|
2. option 'style'
|
|
1288
|
2. option 'style'
|
|
1289
|
3. [ui] setting 'logtemplate'
|
|
1289
|
3. [ui] setting 'logtemplate'
|
|
1290
|
4. [ui] setting 'style'
|
|
1290
|
4. [ui] setting 'style'
|
|
1291
|
If all of these values are either the unset or the empty string,
|
|
1291
|
If all of these values are either the unset or the empty string,
|
|
1292
|
regular display via changeset_printer() is done.
|
|
1292
|
regular display via changeset_printer() is done.
|
|
1293
|
"""
|
|
1293
|
"""
|
|
1294
|
# options
|
|
1294
|
# options
|
|
1295
|
matchfn = None
|
|
1295
|
matchfn = None
|
|
1296
|
if opts.get('patch') or opts.get('stat'):
|
|
1296
|
if opts.get('patch') or opts.get('stat'):
|
|
1297
|
matchfn = scmutil.matchall(repo)
|
|
1297
|
matchfn = scmutil.matchall(repo)
|
|
1298
|
|
|
1298
|
|
|
1299
|
if opts.get('template') == 'json':
|
|
1299
|
if opts.get('template') == 'json':
|
|
1300
|
return jsonchangeset(ui, repo, matchfn, opts, buffered)
|
|
1300
|
return jsonchangeset(ui, repo, matchfn, opts, buffered)
|
|
1301
|
|
|
1301
|
|
|
1302
|
tmpl, mapfile = gettemplate(ui, opts.get('template'), opts.get('style'))
|
|
1302
|
tmpl, mapfile = gettemplate(ui, opts.get('template'), opts.get('style'))
|
|
1303
|
|
|
1303
|
|
|
1304
|
if not tmpl and not mapfile:
|
|
1304
|
if not tmpl and not mapfile:
|
|
1305
|
return changeset_printer(ui, repo, matchfn, opts, buffered)
|
|
1305
|
return changeset_printer(ui, repo, matchfn, opts, buffered)
|
|
1306
|
|
|
1306
|
|
|
1307
|
try:
|
|
1307
|
try:
|
|
1308
|
t = changeset_templater(ui, repo, matchfn, opts, tmpl, mapfile,
|
|
1308
|
t = changeset_templater(ui, repo, matchfn, opts, tmpl, mapfile,
|
|
1309
|
buffered)
|
|
1309
|
buffered)
|
|
1310
|
except SyntaxError, inst:
|
|
1310
|
except SyntaxError, inst:
|
|
1311
|
raise util.Abort(inst.args[0])
|
|
1311
|
raise util.Abort(inst.args[0])
|
|
1312
|
return t
|
|
1312
|
return t
|
|
1313
|
|
|
1313
|
|
|
1314
|
def showmarker(ui, marker):
|
|
1314
|
def showmarker(ui, marker):
|
|
1315
|
"""utility function to display obsolescence marker in a readable way
|
|
1315
|
"""utility function to display obsolescence marker in a readable way
|
|
1316
|
|
|
1316
|
|
|
1317
|
To be used by debug function."""
|
|
1317
|
To be used by debug function."""
|
|
1318
|
ui.write(hex(marker.precnode()))
|
|
1318
|
ui.write(hex(marker.precnode()))
|
|
1319
|
for repl in marker.succnodes():
|
|
1319
|
for repl in marker.succnodes():
|
|
1320
|
ui.write(' ')
|
|
1320
|
ui.write(' ')
|
|
1321
|
ui.write(hex(repl))
|
|
1321
|
ui.write(hex(repl))
|
|
1322
|
ui.write(' %X ' % marker.flags())
|
|
1322
|
ui.write(' %X ' % marker.flags())
|
|
1323
|
parents = marker.parentnodes()
|
|
1323
|
parents = marker.parentnodes()
|
|
1324
|
if parents is not None:
|
|
1324
|
if parents is not None:
|
|
1325
|
ui.write('{%s} ' % ', '.join(hex(p) for p in parents))
|
|
1325
|
ui.write('{%s} ' % ', '.join(hex(p) for p in parents))
|
|
1326
|
ui.write('(%s) ' % util.datestr(marker.date()))
|
|
1326
|
ui.write('(%s) ' % util.datestr(marker.date()))
|
|
1327
|
ui.write('{%s}' % (', '.join('%r: %r' % t for t in
|
|
1327
|
ui.write('{%s}' % (', '.join('%r: %r' % t for t in
|
|
1328
|
sorted(marker.metadata().items())
|
|
1328
|
sorted(marker.metadata().items())
|
|
1329
|
if t[0] != 'date')))
|
|
1329
|
if t[0] != 'date')))
|
|
1330
|
ui.write('\n')
|
|
1330
|
ui.write('\n')
|
|
1331
|
|
|
1331
|
|
|
1332
|
def finddate(ui, repo, date):
|
|
1332
|
def finddate(ui, repo, date):
|
|
1333
|
"""Find the tipmost changeset that matches the given date spec"""
|
|
1333
|
"""Find the tipmost changeset that matches the given date spec"""
|
|
1334
|
|
|
1334
|
|
|
1335
|
df = util.matchdate(date)
|
|
1335
|
df = util.matchdate(date)
|
|
1336
|
m = scmutil.matchall(repo)
|
|
1336
|
m = scmutil.matchall(repo)
|
|
1337
|
results = {}
|
|
1337
|
results = {}
|
|
1338
|
|
|
1338
|
|
|
1339
|
def prep(ctx, fns):
|
|
1339
|
def prep(ctx, fns):
|
|
1340
|
d = ctx.date()
|
|
1340
|
d = ctx.date()
|
|
1341
|
if df(d[0]):
|
|
1341
|
if df(d[0]):
|
|
1342
|
results[ctx.rev()] = d
|
|
1342
|
results[ctx.rev()] = d
|
|
1343
|
|
|
1343
|
|
|
1344
|
for ctx in walkchangerevs(repo, m, {'rev': None}, prep):
|
|
1344
|
for ctx in walkchangerevs(repo, m, {'rev': None}, prep):
|
|
1345
|
rev = ctx.rev()
|
|
1345
|
rev = ctx.rev()
|
|
1346
|
if rev in results:
|
|
1346
|
if rev in results:
|
|
1347
|
ui.status(_("found revision %s from %s\n") %
|
|
1347
|
ui.status(_("found revision %s from %s\n") %
|
|
1348
|
(rev, util.datestr(results[rev])))
|
|
1348
|
(rev, util.datestr(results[rev])))
|
|
1349
|
return str(rev)
|
|
1349
|
return str(rev)
|
|
1350
|
|
|
1350
|
|
|
1351
|
raise util.Abort(_("revision matching date not found"))
|
|
1351
|
raise util.Abort(_("revision matching date not found"))
|
|
1352
|
|
|
1352
|
|
|
1353
|
def increasingwindows(windowsize=8, sizelimit=512):
|
|
1353
|
def increasingwindows(windowsize=8, sizelimit=512):
|
|
1354
|
while True:
|
|
1354
|
while True:
|
|
1355
|
yield windowsize
|
|
1355
|
yield windowsize
|
|
1356
|
if windowsize < sizelimit:
|
|
1356
|
if windowsize < sizelimit:
|
|
1357
|
windowsize *= 2
|
|
1357
|
windowsize *= 2
|
|
1358
|
|
|
1358
|
|
|
1359
|
class FileWalkError(Exception):
|
|
1359
|
class FileWalkError(Exception):
|
|
1360
|
pass
|
|
1360
|
pass
|
|
1361
|
|
|
1361
|
|
|
1362
|
def walkfilerevs(repo, match, follow, revs, fncache):
|
|
1362
|
def walkfilerevs(repo, match, follow, revs, fncache):
|
|
1363
|
'''Walks the file history for the matched files.
|
|
1363
|
'''Walks the file history for the matched files.
|
|
1364
|
|
|
1364
|
|
|
1365
|
Returns the changeset revs that are involved in the file history.
|
|
1365
|
Returns the changeset revs that are involved in the file history.
|
|
1366
|
|
|
1366
|
|
|
1367
|
Throws FileWalkError if the file history can't be walked using
|
|
1367
|
Throws FileWalkError if the file history can't be walked using
|
|
1368
|
filelogs alone.
|
|
1368
|
filelogs alone.
|
|
1369
|
'''
|
|
1369
|
'''
|
|
1370
|
wanted = set()
|
|
1370
|
wanted = set()
|
|
1371
|
copies = []
|
|
1371
|
copies = []
|
|
1372
|
minrev, maxrev = min(revs), max(revs)
|
|
1372
|
minrev, maxrev = min(revs), max(revs)
|
|
1373
|
def filerevgen(filelog, last):
|
|
1373
|
def filerevgen(filelog, last):
|
|
1374
|
"""
|
|
1374
|
"""
|
|
1375
|
Only files, no patterns. Check the history of each file.
|
|
1375
|
Only files, no patterns. Check the history of each file.
|
|
1376
|
|
|
1376
|
|
|
1377
|
Examines filelog entries within minrev, maxrev linkrev range
|
|
1377
|
Examines filelog entries within minrev, maxrev linkrev range
|
|
1378
|
Returns an iterator yielding (linkrev, parentlinkrevs, copied)
|
|
1378
|
Returns an iterator yielding (linkrev, parentlinkrevs, copied)
|
|
1379
|
tuples in backwards order
|
|
1379
|
tuples in backwards order
|
|
1380
|
"""
|
|
1380
|
"""
|
|
1381
|
cl_count = len(repo)
|
|
1381
|
cl_count = len(repo)
|
|
1382
|
revs = []
|
|
1382
|
revs = []
|
|
1383
|
for j in xrange(0, last + 1):
|
|
1383
|
for j in xrange(0, last + 1):
|
|
1384
|
linkrev = filelog.linkrev(j)
|
|
1384
|
linkrev = filelog.linkrev(j)
|
|
1385
|
if linkrev < minrev:
|
|
1385
|
if linkrev < minrev:
|
|
1386
|
continue
|
|
1386
|
continue
|
|
1387
|
# only yield rev for which we have the changelog, it can
|
|
1387
|
# only yield rev for which we have the changelog, it can
|
|
1388
|
# happen while doing "hg log" during a pull or commit
|
|
1388
|
# happen while doing "hg log" during a pull or commit
|
|
1389
|
if linkrev >= cl_count:
|
|
1389
|
if linkrev >= cl_count:
|
|
1390
|
break
|
|
1390
|
break
|
|
1391
|
|
|
1391
|
|
|
1392
|
parentlinkrevs = []
|
|
1392
|
parentlinkrevs = []
|
|
1393
|
for p in filelog.parentrevs(j):
|
|
1393
|
for p in filelog.parentrevs(j):
|
|
1394
|
if p != nullrev:
|
|
1394
|
if p != nullrev:
|
|
1395
|
parentlinkrevs.append(filelog.linkrev(p))
|
|
1395
|
parentlinkrevs.append(filelog.linkrev(p))
|
|
1396
|
n = filelog.node(j)
|
|
1396
|
n = filelog.node(j)
|
|
1397
|
revs.append((linkrev, parentlinkrevs,
|
|
1397
|
revs.append((linkrev, parentlinkrevs,
|
|
1398
|
follow and filelog.renamed(n)))
|
|
1398
|
follow and filelog.renamed(n)))
|
|
1399
|
|
|
1399
|
|
|
1400
|
return reversed(revs)
|
|
1400
|
return reversed(revs)
|
|
1401
|
def iterfiles():
|
|
1401
|
def iterfiles():
|
|
1402
|
pctx = repo['.']
|
|
1402
|
pctx = repo['.']
|
|
1403
|
for filename in match.files():
|
|
1403
|
for filename in match.files():
|
|
1404
|
if follow:
|
|
1404
|
if follow:
|
|
1405
|
if filename not in pctx:
|
|
1405
|
if filename not in pctx:
|
|
1406
|
raise util.Abort(_('cannot follow file not in parent '
|
|
1406
|
raise util.Abort(_('cannot follow file not in parent '
|
|
1407
|
'revision: "%s"') % filename)
|
|
1407
|
'revision: "%s"') % filename)
|
|
1408
|
yield filename, pctx[filename].filenode()
|
|
1408
|
yield filename, pctx[filename].filenode()
|
|
1409
|
else:
|
|
1409
|
else:
|
|
1410
|
yield filename, None
|
|
1410
|
yield filename, None
|
|
1411
|
for filename_node in copies:
|
|
1411
|
for filename_node in copies:
|
|
1412
|
yield filename_node
|
|
1412
|
yield filename_node
|
|
1413
|
|
|
1413
|
|
|
1414
|
for file_, node in iterfiles():
|
|
1414
|
for file_, node in iterfiles():
|
|
1415
|
filelog = repo.file(file_)
|
|
1415
|
filelog = repo.file(file_)
|
|
1416
|
if not len(filelog):
|
|
1416
|
if not len(filelog):
|
|
1417
|
if node is None:
|
|
1417
|
if node is None:
|
|
1418
|
# A zero count may be a directory or deleted file, so
|
|
1418
|
# A zero count may be a directory or deleted file, so
|
|
1419
|
# try to find matching entries on the slow path.
|
|
1419
|
# try to find matching entries on the slow path.
|
|
1420
|
if follow:
|
|
1420
|
if follow:
|
|
1421
|
raise util.Abort(
|
|
1421
|
raise util.Abort(
|
|
1422
|
_('cannot follow nonexistent file: "%s"') % file_)
|
|
1422
|
_('cannot follow nonexistent file: "%s"') % file_)
|
|
1423
|
raise FileWalkError("Cannot walk via filelog")
|
|
1423
|
raise FileWalkError("Cannot walk via filelog")
|
|
1424
|
else:
|
|
1424
|
else:
|
|
1425
|
continue
|
|
1425
|
continue
|
|
1426
|
|
|
1426
|
|
|
1427
|
if node is None:
|
|
1427
|
if node is None:
|
|
1428
|
last = len(filelog) - 1
|
|
1428
|
last = len(filelog) - 1
|
|
1429
|
else:
|
|
1429
|
else:
|
|
1430
|
last = filelog.rev(node)
|
|
1430
|
last = filelog.rev(node)
|
|
1431
|
|
|
1431
|
|
|
1432
|
|
|
1432
|
|
|
1433
|
# keep track of all ancestors of the file
|
|
1433
|
# keep track of all ancestors of the file
|
|
1434
|
ancestors = set([filelog.linkrev(last)])
|
|
1434
|
ancestors = set([filelog.linkrev(last)])
|
|
1435
|
|
|
1435
|
|
|
1436
|
# iterate from latest to oldest revision
|
|
1436
|
# iterate from latest to oldest revision
|
|
1437
|
for rev, flparentlinkrevs, copied in filerevgen(filelog, last):
|
|
1437
|
for rev, flparentlinkrevs, copied in filerevgen(filelog, last):
|
|
1438
|
if not follow:
|
|
1438
|
if not follow:
|
|
1439
|
if rev > maxrev:
|
|
1439
|
if rev > maxrev:
|
|
1440
|
continue
|
|
1440
|
continue
|
|
1441
|
else:
|
|
1441
|
else:
|
|
1442
|
# Note that last might not be the first interesting
|
|
1442
|
# Note that last might not be the first interesting
|
|
1443
|
# rev to us:
|
|
1443
|
# rev to us:
|
|
1444
|
# if the file has been changed after maxrev, we'll
|
|
1444
|
# if the file has been changed after maxrev, we'll
|
|
1445
|
# have linkrev(last) > maxrev, and we still need
|
|
1445
|
# have linkrev(last) > maxrev, and we still need
|
|
1446
|
# to explore the file graph
|
|
1446
|
# to explore the file graph
|
|
1447
|
if rev not in ancestors:
|
|
1447
|
if rev not in ancestors:
|
|
1448
|
continue
|
|
1448
|
continue
|
|
1449
|
# XXX insert 1327 fix here
|
|
1449
|
# XXX insert 1327 fix here
|
|
1450
|
if flparentlinkrevs:
|
|
1450
|
if flparentlinkrevs:
|
|
1451
|
ancestors.update(flparentlinkrevs)
|
|
1451
|
ancestors.update(flparentlinkrevs)
|
|
1452
|
|
|
1452
|
|
|
1453
|
fncache.setdefault(rev, []).append(file_)
|
|
1453
|
fncache.setdefault(rev, []).append(file_)
|
|
1454
|
wanted.add(rev)
|
|
1454
|
wanted.add(rev)
|
|
1455
|
if copied:
|
|
1455
|
if copied:
|
|
1456
|
copies.append(copied)
|
|
1456
|
copies.append(copied)
|
|
1457
|
|
|
1457
|
|
|
1458
|
return wanted
|
|
1458
|
return wanted
|
|
1459
|
|
|
1459
|
|
|
1460
|
def walkchangerevs(repo, match, opts, prepare):
|
|
1460
|
def walkchangerevs(repo, match, opts, prepare):
|
|
1461
|
'''Iterate over files and the revs in which they changed.
|
|
1461
|
'''Iterate over files and the revs in which they changed.
|
|
1462
|
|
|
1462
|
|
|
1463
|
Callers most commonly need to iterate backwards over the history
|
|
1463
|
Callers most commonly need to iterate backwards over the history
|
|
1464
|
in which they are interested. Doing so has awful (quadratic-looking)
|
|
1464
|
in which they are interested. Doing so has awful (quadratic-looking)
|
|
1465
|
performance, so we use iterators in a "windowed" way.
|
|
1465
|
performance, so we use iterators in a "windowed" way.
|
|
1466
|
|
|
1466
|
|
|
1467
|
We walk a window of revisions in the desired order. Within the
|
|
1467
|
We walk a window of revisions in the desired order. Within the
|
|
1468
|
window, we first walk forwards to gather data, then in the desired
|
|
1468
|
window, we first walk forwards to gather data, then in the desired
|
|
1469
|
order (usually backwards) to display it.
|
|
1469
|
order (usually backwards) to display it.
|
|
1470
|
|
|
1470
|
|
|
1471
|
This function returns an iterator yielding contexts. Before
|
|
1471
|
This function returns an iterator yielding contexts. Before
|
|
1472
|
yielding each context, the iterator will first call the prepare
|
|
1472
|
yielding each context, the iterator will first call the prepare
|
|
1473
|
function on each context in the window in forward order.'''
|
|
1473
|
function on each context in the window in forward order.'''
|
|
1474
|
|
|
1474
|
|
|
1475
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1475
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1476
|
revs = _logrevs(repo, opts)
|
|
1476
|
revs = _logrevs(repo, opts)
|
|
1477
|
if not revs:
|
|
1477
|
if not revs:
|
|
1478
|
return []
|
|
1478
|
return []
|
|
1479
|
wanted = set()
|
|
1479
|
wanted = set()
|
|
1480
|
slowpath = match.anypats() or (match.files() and opts.get('removed'))
|
|
1480
|
slowpath = match.anypats() or (match.files() and opts.get('removed'))
|
|
1481
|
fncache = {}
|
|
1481
|
fncache = {}
|
|
1482
|
change = repo.changectx
|
|
1482
|
change = repo.changectx
|
|
1483
|
|
|
1483
|
|
|
1484
|
# First step is to fill wanted, the set of revisions that we want to yield.
|
|
1484
|
# First step is to fill wanted, the set of revisions that we want to yield.
|
|
1485
|
# When it does not induce extra cost, we also fill fncache for revisions in
|
|
1485
|
# When it does not induce extra cost, we also fill fncache for revisions in
|
|
1486
|
# wanted: a cache of filenames that were changed (ctx.files()) and that
|
|
1486
|
# wanted: a cache of filenames that were changed (ctx.files()) and that
|
|
1487
|
# match the file filtering conditions.
|
|
1487
|
# match the file filtering conditions.
|
|
1488
|
|
|
1488
|
|
|
1489
|
if not slowpath and not match.files():
|
|
1489
|
if not slowpath and not match.files():
|
|
1490
|
# No files, no patterns. Display all revs.
|
|
1490
|
# No files, no patterns. Display all revs.
|
|
1491
|
wanted = revs
|
|
1491
|
wanted = revs
|
|
1492
|
|
|
1492
|
|
|
1493
|
if not slowpath and match.files():
|
|
1493
|
if not slowpath and match.files():
|
|
1494
|
# We only have to read through the filelog to find wanted revisions
|
|
1494
|
# We only have to read through the filelog to find wanted revisions
|
|
1495
|
|
|
1495
|
|
|
1496
|
try:
|
|
1496
|
try:
|
|
1497
|
wanted = walkfilerevs(repo, match, follow, revs, fncache)
|
|
1497
|
wanted = walkfilerevs(repo, match, follow, revs, fncache)
|
|
1498
|
except FileWalkError:
|
|
1498
|
except FileWalkError:
|
|
1499
|
slowpath = True
|
|
1499
|
slowpath = True
|
|
1500
|
|
|
1500
|
|
|
1501
|
# We decided to fall back to the slowpath because at least one
|
|
1501
|
# We decided to fall back to the slowpath because at least one
|
|
1502
|
# of the paths was not a file. Check to see if at least one of them
|
|
1502
|
# of the paths was not a file. Check to see if at least one of them
|
|
1503
|
# existed in history, otherwise simply return
|
|
1503
|
# existed in history, otherwise simply return
|
|
1504
|
for path in match.files():
|
|
1504
|
for path in match.files():
|
|
1505
|
if path == '.' or path in repo.store:
|
|
1505
|
if path == '.' or path in repo.store:
|
|
1506
|
break
|
|
1506
|
break
|
|
1507
|
else:
|
|
1507
|
else:
|
|
1508
|
return []
|
|
1508
|
return []
|
|
1509
|
|
|
1509
|
|
|
1510
|
if slowpath:
|
|
1510
|
if slowpath:
|
|
1511
|
# We have to read the changelog to match filenames against
|
|
1511
|
# We have to read the changelog to match filenames against
|
|
1512
|
# changed files
|
|
1512
|
# changed files
|
|
1513
|
|
|
1513
|
|
|
1514
|
if follow:
|
|
1514
|
if follow:
|
|
1515
|
raise util.Abort(_('can only follow copies/renames for explicit '
|
|
1515
|
raise util.Abort(_('can only follow copies/renames for explicit '
|
|
1516
|
'filenames'))
|
|
1516
|
'filenames'))
|
|
1517
|
|
|
1517
|
|
|
1518
|
# The slow path checks files modified in every changeset.
|
|
1518
|
# The slow path checks files modified in every changeset.
|
|
1519
|
# This is really slow on large repos, so compute the set lazily.
|
|
1519
|
# This is really slow on large repos, so compute the set lazily.
|
|
1520
|
class lazywantedset(object):
|
|
1520
|
class lazywantedset(object):
|
|
1521
|
def __init__(self):
|
|
1521
|
def __init__(self):
|
|
1522
|
self.set = set()
|
|
1522
|
self.set = set()
|
|
1523
|
self.revs = set(revs)
|
|
1523
|
self.revs = set(revs)
|
|
1524
|
|
|
1524
|
|
|
1525
|
# No need to worry about locality here because it will be accessed
|
|
1525
|
# No need to worry about locality here because it will be accessed
|
|
1526
|
# in the same order as the increasing window below.
|
|
1526
|
# in the same order as the increasing window below.
|
|
1527
|
def __contains__(self, value):
|
|
1527
|
def __contains__(self, value):
|
|
1528
|
if value in self.set:
|
|
1528
|
if value in self.set:
|
|
1529
|
return True
|
|
1529
|
return True
|
|
1530
|
elif not value in self.revs:
|
|
1530
|
elif not value in self.revs:
|
|
1531
|
return False
|
|
1531
|
return False
|
|
1532
|
else:
|
|
1532
|
else:
|
|
1533
|
self.revs.discard(value)
|
|
1533
|
self.revs.discard(value)
|
|
1534
|
ctx = change(value)
|
|
1534
|
ctx = change(value)
|
|
1535
|
matches = filter(match, ctx.files())
|
|
1535
|
matches = filter(match, ctx.files())
|
|
1536
|
if matches:
|
|
1536
|
if matches:
|
|
1537
|
fncache[value] = matches
|
|
1537
|
fncache[value] = matches
|
|
1538
|
self.set.add(value)
|
|
1538
|
self.set.add(value)
|
|
1539
|
return True
|
|
1539
|
return True
|
|
1540
|
return False
|
|
1540
|
return False
|
|
1541
|
|
|
1541
|
|
|
1542
|
def discard(self, value):
|
|
1542
|
def discard(self, value):
|
|
1543
|
self.revs.discard(value)
|
|
1543
|
self.revs.discard(value)
|
|
1544
|
self.set.discard(value)
|
|
1544
|
self.set.discard(value)
|
|
1545
|
|
|
1545
|
|
|
1546
|
wanted = lazywantedset()
|
|
1546
|
wanted = lazywantedset()
|
|
1547
|
|
|
1547
|
|
|
1548
|
class followfilter(object):
|
|
1548
|
class followfilter(object):
|
|
1549
|
def __init__(self, onlyfirst=False):
|
|
1549
|
def __init__(self, onlyfirst=False):
|
|
1550
|
self.startrev = nullrev
|
|
1550
|
self.startrev = nullrev
|
|
1551
|
self.roots = set()
|
|
1551
|
self.roots = set()
|
|
1552
|
self.onlyfirst = onlyfirst
|
|
1552
|
self.onlyfirst = onlyfirst
|
|
1553
|
|
|
1553
|
|
|
1554
|
def match(self, rev):
|
|
1554
|
def match(self, rev):
|
|
1555
|
def realparents(rev):
|
|
1555
|
def realparents(rev):
|
|
1556
|
if self.onlyfirst:
|
|
1556
|
if self.onlyfirst:
|
|
1557
|
return repo.changelog.parentrevs(rev)[0:1]
|
|
1557
|
return repo.changelog.parentrevs(rev)[0:1]
|
|
1558
|
else:
|
|
1558
|
else:
|
|
1559
|
return filter(lambda x: x != nullrev,
|
|
1559
|
return filter(lambda x: x != nullrev,
|
|
1560
|
repo.changelog.parentrevs(rev))
|
|
1560
|
repo.changelog.parentrevs(rev))
|
|
1561
|
|
|
1561
|
|
|
1562
|
if self.startrev == nullrev:
|
|
1562
|
if self.startrev == nullrev:
|
|
1563
|
self.startrev = rev
|
|
1563
|
self.startrev = rev
|
|
1564
|
return True
|
|
1564
|
return True
|
|
1565
|
|
|
1565
|
|
|
1566
|
if rev > self.startrev:
|
|
1566
|
if rev > self.startrev:
|
|
1567
|
# forward: all descendants
|
|
1567
|
# forward: all descendants
|
|
1568
|
if not self.roots:
|
|
1568
|
if not self.roots:
|
|
1569
|
self.roots.add(self.startrev)
|
|
1569
|
self.roots.add(self.startrev)
|
|
1570
|
for parent in realparents(rev):
|
|
1570
|
for parent in realparents(rev):
|
|
1571
|
if parent in self.roots:
|
|
1571
|
if parent in self.roots:
|
|
1572
|
self.roots.add(rev)
|
|
1572
|
self.roots.add(rev)
|
|
1573
|
return True
|
|
1573
|
return True
|
|
1574
|
else:
|
|
1574
|
else:
|
|
1575
|
# backwards: all parents
|
|
1575
|
# backwards: all parents
|
|
1576
|
if not self.roots:
|
|
1576
|
if not self.roots:
|
|
1577
|
self.roots.update(realparents(self.startrev))
|
|
1577
|
self.roots.update(realparents(self.startrev))
|
|
1578
|
if rev in self.roots:
|
|
1578
|
if rev in self.roots:
|
|
1579
|
self.roots.remove(rev)
|
|
1579
|
self.roots.remove(rev)
|
|
1580
|
self.roots.update(realparents(rev))
|
|
1580
|
self.roots.update(realparents(rev))
|
|
1581
|
return True
|
|
1581
|
return True
|
|
1582
|
|
|
1582
|
|
|
1583
|
return False
|
|
1583
|
return False
|
|
1584
|
|
|
1584
|
|
|
1585
|
# it might be worthwhile to do this in the iterator if the rev range
|
|
1585
|
# it might be worthwhile to do this in the iterator if the rev range
|
|
1586
|
# is descending and the prune args are all within that range
|
|
1586
|
# is descending and the prune args are all within that range
|
|
1587
|
for rev in opts.get('prune', ()):
|
|
1587
|
for rev in opts.get('prune', ()):
|
|
1588
|
rev = repo[rev].rev()
|
|
1588
|
rev = repo[rev].rev()
|
|
1589
|
ff = followfilter()
|
|
1589
|
ff = followfilter()
|
|
1590
|
stop = min(revs[0], revs[-1])
|
|
1590
|
stop = min(revs[0], revs[-1])
|
|
1591
|
for x in xrange(rev, stop - 1, -1):
|
|
1591
|
for x in xrange(rev, stop - 1, -1):
|
|
1592
|
if ff.match(x):
|
|
1592
|
if ff.match(x):
|
|
1593
|
wanted = wanted - [x]
|
|
1593
|
wanted = wanted - [x]
|
|
1594
|
|
|
1594
|
|
|
1595
|
# Now that wanted is correctly initialized, we can iterate over the
|
|
1595
|
# Now that wanted is correctly initialized, we can iterate over the
|
|
1596
|
# revision range, yielding only revisions in wanted.
|
|
1596
|
# revision range, yielding only revisions in wanted.
|
|
1597
|
def iterate():
|
|
1597
|
def iterate():
|
|
1598
|
if follow and not match.files():
|
|
1598
|
if follow and not match.files():
|
|
1599
|
ff = followfilter(onlyfirst=opts.get('follow_first'))
|
|
1599
|
ff = followfilter(onlyfirst=opts.get('follow_first'))
|
|
1600
|
def want(rev):
|
|
1600
|
def want(rev):
|
|
1601
|
return ff.match(rev) and rev in wanted
|
|
1601
|
return ff.match(rev) and rev in wanted
|
|
1602
|
else:
|
|
1602
|
else:
|
|
1603
|
def want(rev):
|
|
1603
|
def want(rev):
|
|
1604
|
return rev in wanted
|
|
1604
|
return rev in wanted
|
|
1605
|
|
|
1605
|
|
|
1606
|
it = iter(revs)
|
|
1606
|
it = iter(revs)
|
|
1607
|
stopiteration = False
|
|
1607
|
stopiteration = False
|
|
1608
|
for windowsize in increasingwindows():
|
|
1608
|
for windowsize in increasingwindows():
|
|
1609
|
nrevs = []
|
|
1609
|
nrevs = []
|
|
1610
|
for i in xrange(windowsize):
|
|
1610
|
for i in xrange(windowsize):
|
|
1611
|
try:
|
|
1611
|
try:
|
|
1612
|
rev = it.next()
|
|
1612
|
rev = it.next()
|
|
1613
|
if want(rev):
|
|
1613
|
if want(rev):
|
|
1614
|
nrevs.append(rev)
|
|
1614
|
nrevs.append(rev)
|
|
1615
|
except (StopIteration):
|
|
1615
|
except (StopIteration):
|
|
1616
|
stopiteration = True
|
|
1616
|
stopiteration = True
|
|
1617
|
break
|
|
1617
|
break
|
|
1618
|
for rev in sorted(nrevs):
|
|
1618
|
for rev in sorted(nrevs):
|
|
1619
|
fns = fncache.get(rev)
|
|
1619
|
fns = fncache.get(rev)
|
|
1620
|
ctx = change(rev)
|
|
1620
|
ctx = change(rev)
|
|
1621
|
if not fns:
|
|
1621
|
if not fns:
|
|
1622
|
def fns_generator():
|
|
1622
|
def fns_generator():
|
|
1623
|
for f in ctx.files():
|
|
1623
|
for f in ctx.files():
|
|
1624
|
if match(f):
|
|
1624
|
if match(f):
|
|
1625
|
yield f
|
|
1625
|
yield f
|
|
1626
|
fns = fns_generator()
|
|
1626
|
fns = fns_generator()
|
|
1627
|
prepare(ctx, fns)
|
|
1627
|
prepare(ctx, fns)
|
|
1628
|
for rev in nrevs:
|
|
1628
|
for rev in nrevs:
|
|
1629
|
yield change(rev)
|
|
1629
|
yield change(rev)
|
|
1630
|
|
|
1630
|
|
|
1631
|
if stopiteration:
|
|
1631
|
if stopiteration:
|
|
1632
|
break
|
|
1632
|
break
|
|
1633
|
|
|
1633
|
|
|
1634
|
return iterate()
|
|
1634
|
return iterate()
|
|
1635
|
|
|
1635
|
|
|
1636
|
def _makefollowlogfilematcher(repo, files, followfirst):
|
|
1636
|
def _makefollowlogfilematcher(repo, files, followfirst):
|
|
1637
|
# When displaying a revision with --patch --follow FILE, we have
|
|
1637
|
# When displaying a revision with --patch --follow FILE, we have
|
|
1638
|
# to know which file of the revision must be diffed. With
|
|
1638
|
# to know which file of the revision must be diffed. With
|
|
1639
|
# --follow, we want the names of the ancestors of FILE in the
|
|
1639
|
# --follow, we want the names of the ancestors of FILE in the
|
|
1640
|
# revision, stored in "fcache". "fcache" is populated by
|
|
1640
|
# revision, stored in "fcache". "fcache" is populated by
|
|
1641
|
# reproducing the graph traversal already done by --follow revset
|
|
1641
|
# reproducing the graph traversal already done by --follow revset
|
|
1642
|
# and relating linkrevs to file names (which is not "correct" but
|
|
1642
|
# and relating linkrevs to file names (which is not "correct" but
|
|
1643
|
# good enough).
|
|
1643
|
# good enough).
|
|
1644
|
fcache = {}
|
|
1644
|
fcache = {}
|
|
1645
|
fcacheready = [False]
|
|
1645
|
fcacheready = [False]
|
|
1646
|
pctx = repo['.']
|
|
1646
|
pctx = repo['.']
|
|
1647
|
|
|
1647
|
|
|
1648
|
def populate():
|
|
1648
|
def populate():
|
|
1649
|
for fn in files:
|
|
1649
|
for fn in files:
|
|
1650
|
for i in ((pctx[fn],), pctx[fn].ancestors(followfirst=followfirst)):
|
|
1650
|
for i in ((pctx[fn],), pctx[fn].ancestors(followfirst=followfirst)):
|
|
1651
|
for c in i:
|
|
1651
|
for c in i:
|
|
1652
|
fcache.setdefault(c.linkrev(), set()).add(c.path())
|
|
1652
|
fcache.setdefault(c.linkrev(), set()).add(c.path())
|
|
1653
|
|
|
1653
|
|
|
1654
|
def filematcher(rev):
|
|
1654
|
def filematcher(rev):
|
|
1655
|
if not fcacheready[0]:
|
|
1655
|
if not fcacheready[0]:
|
|
1656
|
# Lazy initialization
|
|
1656
|
# Lazy initialization
|
|
1657
|
fcacheready[0] = True
|
|
1657
|
fcacheready[0] = True
|
|
1658
|
populate()
|
|
1658
|
populate()
|
|
1659
|
return scmutil.matchfiles(repo, fcache.get(rev, []))
|
|
1659
|
return scmutil.matchfiles(repo, fcache.get(rev, []))
|
|
1660
|
|
|
1660
|
|
|
1661
|
return filematcher
|
|
1661
|
return filematcher
|
|
1662
|
|
|
1662
|
|
|
1663
|
def _makenofollowlogfilematcher(repo, pats, opts):
|
|
1663
|
def _makenofollowlogfilematcher(repo, pats, opts):
|
|
1664
|
'''hook for extensions to override the filematcher for non-follow cases'''
|
|
1664
|
'''hook for extensions to override the filematcher for non-follow cases'''
|
|
1665
|
return None
|
|
1665
|
return None
|
|
1666
|
|
|
1666
|
|
|
1667
|
def _makelogrevset(repo, pats, opts, revs):
|
|
1667
|
def _makelogrevset(repo, pats, opts, revs):
|
|
1668
|
"""Return (expr, filematcher) where expr is a revset string built
|
|
1668
|
"""Return (expr, filematcher) where expr is a revset string built
|
|
1669
|
from log options and file patterns or None. If --stat or --patch
|
|
1669
|
from log options and file patterns or None. If --stat or --patch
|
|
1670
|
are not passed filematcher is None. Otherwise it is a callable
|
|
1670
|
are not passed filematcher is None. Otherwise it is a callable
|
|
1671
|
taking a revision number and returning a match objects filtering
|
|
1671
|
taking a revision number and returning a match objects filtering
|
|
1672
|
the files to be detailed when displaying the revision.
|
|
1672
|
the files to be detailed when displaying the revision.
|
|
1673
|
"""
|
|
1673
|
"""
|
|
1674
|
opt2revset = {
|
|
1674
|
opt2revset = {
|
|
1675
|
'no_merges': ('not merge()', None),
|
|
1675
|
'no_merges': ('not merge()', None),
|
|
1676
|
'only_merges': ('merge()', None),
|
|
1676
|
'only_merges': ('merge()', None),
|
|
1677
|
'_ancestors': ('ancestors(%(val)s)', None),
|
|
1677
|
'_ancestors': ('ancestors(%(val)s)', None),
|
|
1678
|
'_fancestors': ('_firstancestors(%(val)s)', None),
|
|
1678
|
'_fancestors': ('_firstancestors(%(val)s)', None),
|
|
1679
|
'_descendants': ('descendants(%(val)s)', None),
|
|
1679
|
'_descendants': ('descendants(%(val)s)', None),
|
|
1680
|
'_fdescendants': ('_firstdescendants(%(val)s)', None),
|
|
1680
|
'_fdescendants': ('_firstdescendants(%(val)s)', None),
|
|
1681
|
'_matchfiles': ('_matchfiles(%(val)s)', None),
|
|
1681
|
'_matchfiles': ('_matchfiles(%(val)s)', None),
|
|
1682
|
'date': ('date(%(val)r)', None),
|
|
1682
|
'date': ('date(%(val)r)', None),
|
|
1683
|
'branch': ('branch(%(val)r)', ' or '),
|
|
1683
|
'branch': ('branch(%(val)r)', ' or '),
|
|
1684
|
'_patslog': ('filelog(%(val)r)', ' or '),
|
|
1684
|
'_patslog': ('filelog(%(val)r)', ' or '),
|
|
1685
|
'_patsfollow': ('follow(%(val)r)', ' or '),
|
|
1685
|
'_patsfollow': ('follow(%(val)r)', ' or '),
|
|
1686
|
'_patsfollowfirst': ('_followfirst(%(val)r)', ' or '),
|
|
1686
|
'_patsfollowfirst': ('_followfirst(%(val)r)', ' or '),
|
|
1687
|
'keyword': ('keyword(%(val)r)', ' or '),
|
|
1687
|
'keyword': ('keyword(%(val)r)', ' or '),
|
|
1688
|
'prune': ('not (%(val)r or ancestors(%(val)r))', ' and '),
|
|
1688
|
'prune': ('not (%(val)r or ancestors(%(val)r))', ' and '),
|
|
1689
|
'user': ('user(%(val)r)', ' or '),
|
|
1689
|
'user': ('user(%(val)r)', ' or '),
|
|
1690
|
}
|
|
1690
|
}
|
|
1691
|
|
|
1691
|
|
|
1692
|
opts = dict(opts)
|
|
1692
|
opts = dict(opts)
|
|
1693
|
# follow or not follow?
|
|
1693
|
# follow or not follow?
|
|
1694
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1694
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1695
|
followfirst = opts.get('follow_first') and 1 or 0
|
|
1695
|
followfirst = opts.get('follow_first') and 1 or 0
|
|
1696
|
# --follow with FILE behaviour depends on revs...
|
|
1696
|
# --follow with FILE behaviour depends on revs...
|
|
1697
|
it = iter(revs)
|
|
1697
|
it = iter(revs)
|
|
1698
|
startrev = it.next()
|
|
1698
|
startrev = it.next()
|
|
1699
|
try:
|
|
1699
|
try:
|
|
1700
|
followdescendants = startrev < it.next()
|
|
1700
|
followdescendants = startrev < it.next()
|
|
1701
|
except (StopIteration):
|
|
1701
|
except (StopIteration):
|
|
1702
|
followdescendants = False
|
|
1702
|
followdescendants = False
|
|
1703
|
|
|
1703
|
|
|
1704
|
# branch and only_branch are really aliases and must be handled at
|
|
1704
|
# branch and only_branch are really aliases and must be handled at
|
|
1705
|
# the same time
|
|
1705
|
# the same time
|
|
1706
|
opts['branch'] = opts.get('branch', []) + opts.get('only_branch', [])
|
|
1706
|
opts['branch'] = opts.get('branch', []) + opts.get('only_branch', [])
|
|
1707
|
opts['branch'] = [repo.lookupbranch(b) for b in opts['branch']]
|
|
1707
|
opts['branch'] = [repo.lookupbranch(b) for b in opts['branch']]
|
|
1708
|
# pats/include/exclude are passed to match.match() directly in
|
|
1708
|
# pats/include/exclude are passed to match.match() directly in
|
|
1709
|
# _matchfiles() revset but walkchangerevs() builds its matcher with
|
|
1709
|
# _matchfiles() revset but walkchangerevs() builds its matcher with
|
|
1710
|
# scmutil.match(). The difference is input pats are globbed on
|
|
1710
|
# scmutil.match(). The difference is input pats are globbed on
|
|
1711
|
# platforms without shell expansion (windows).
|
|
1711
|
# platforms without shell expansion (windows).
|
|
1712
|
pctx = repo[None]
|
|
1712
|
pctx = repo[None]
|
|
1713
|
match, pats = scmutil.matchandpats(pctx, pats, opts)
|
|
1713
|
match, pats = scmutil.matchandpats(pctx, pats, opts)
|
|
1714
|
slowpath = match.anypats() or (match.files() and opts.get('removed'))
|
|
1714
|
slowpath = match.anypats() or (match.files() and opts.get('removed'))
|
|
1715
|
if not slowpath:
|
|
1715
|
if not slowpath:
|
|
1716
|
for f in match.files():
|
|
1716
|
for f in match.files():
|
|
1717
|
if follow and f not in pctx:
|
|
1717
|
if follow and f not in pctx:
|
|
1718
|
# If the file exists, it may be a directory, so let it
|
|
1718
|
# If the file exists, it may be a directory, so let it
|
|
1719
|
# take the slow path.
|
|
1719
|
# take the slow path.
|
|
1720
|
if os.path.exists(repo.wjoin(f)):
|
|
1720
|
if os.path.exists(repo.wjoin(f)):
|
|
1721
|
slowpath = True
|
|
1721
|
slowpath = True
|
|
1722
|
continue
|
|
1722
|
continue
|
|
1723
|
else:
|
|
1723
|
else:
|
|
1724
|
raise util.Abort(_('cannot follow file not in parent '
|
|
1724
|
raise util.Abort(_('cannot follow file not in parent '
|
|
1725
|
'revision: "%s"') % f)
|
|
1725
|
'revision: "%s"') % f)
|
|
1726
|
filelog = repo.file(f)
|
|
1726
|
filelog = repo.file(f)
|
|
1727
|
if not filelog:
|
|
1727
|
if not filelog:
|
|
1728
|
# A zero count may be a directory or deleted file, so
|
|
1728
|
# A zero count may be a directory or deleted file, so
|
|
1729
|
# try to find matching entries on the slow path.
|
|
1729
|
# try to find matching entries on the slow path.
|
|
1730
|
if follow:
|
|
1730
|
if follow:
|
|
1731
|
raise util.Abort(
|
|
1731
|
raise util.Abort(
|
|
1732
|
_('cannot follow nonexistent file: "%s"') % f)
|
|
1732
|
_('cannot follow nonexistent file: "%s"') % f)
|
|
1733
|
slowpath = True
|
|
1733
|
slowpath = True
|
|
1734
|
|
|
1734
|
|
|
1735
|
# We decided to fall back to the slowpath because at least one
|
|
1735
|
# We decided to fall back to the slowpath because at least one
|
|
1736
|
# of the paths was not a file. Check to see if at least one of them
|
|
1736
|
# of the paths was not a file. Check to see if at least one of them
|
|
1737
|
# existed in history - in that case, we'll continue down the
|
|
1737
|
# existed in history - in that case, we'll continue down the
|
|
1738
|
# slowpath; otherwise, we can turn off the slowpath
|
|
1738
|
# slowpath; otherwise, we can turn off the slowpath
|
|
1739
|
if slowpath:
|
|
1739
|
if slowpath:
|
|
1740
|
for path in match.files():
|
|
1740
|
for path in match.files():
|
|
1741
|
if path == '.' or path in repo.store:
|
|
1741
|
if path == '.' or path in repo.store:
|
|
1742
|
break
|
|
1742
|
break
|
|
1743
|
else:
|
|
1743
|
else:
|
|
1744
|
slowpath = False
|
|
1744
|
slowpath = False
|
|
1745
|
|
|
1745
|
|
|
1746
|
fpats = ('_patsfollow', '_patsfollowfirst')
|
|
1746
|
fpats = ('_patsfollow', '_patsfollowfirst')
|
|
1747
|
fnopats = (('_ancestors', '_fancestors'),
|
|
1747
|
fnopats = (('_ancestors', '_fancestors'),
|
|
1748
|
('_descendants', '_fdescendants'))
|
|
1748
|
('_descendants', '_fdescendants'))
|
|
1749
|
if slowpath:
|
|
1749
|
if slowpath:
|
|
1750
|
# See walkchangerevs() slow path.
|
|
1750
|
# See walkchangerevs() slow path.
|
|
1751
|
#
|
|
1751
|
#
|
|
1752
|
# pats/include/exclude cannot be represented as separate
|
|
1752
|
# pats/include/exclude cannot be represented as separate
|
|
1753
|
# revset expressions as their filtering logic applies at file
|
|
1753
|
# revset expressions as their filtering logic applies at file
|
|
1754
|
# level. For instance "-I a -X a" matches a revision touching
|
|
1754
|
# level. For instance "-I a -X a" matches a revision touching
|
|
1755
|
# "a" and "b" while "file(a) and not file(b)" does
|
|
1755
|
# "a" and "b" while "file(a) and not file(b)" does
|
|
1756
|
# not. Besides, filesets are evaluated against the working
|
|
1756
|
# not. Besides, filesets are evaluated against the working
|
|
1757
|
# directory.
|
|
1757
|
# directory.
|
|
1758
|
matchargs = ['r:', 'd:relpath']
|
|
1758
|
matchargs = ['r:', 'd:relpath']
|
|
1759
|
for p in pats:
|
|
1759
|
for p in pats:
|
|
1760
|
matchargs.append('p:' + p)
|
|
1760
|
matchargs.append('p:' + p)
|
|
1761
|
for p in opts.get('include', []):
|
|
1761
|
for p in opts.get('include', []):
|
|
1762
|
matchargs.append('i:' + p)
|
|
1762
|
matchargs.append('i:' + p)
|
|
1763
|
for p in opts.get('exclude', []):
|
|
1763
|
for p in opts.get('exclude', []):
|
|
1764
|
matchargs.append('x:' + p)
|
|
1764
|
matchargs.append('x:' + p)
|
|
1765
|
matchargs = ','.join(('%r' % p) for p in matchargs)
|
|
1765
|
matchargs = ','.join(('%r' % p) for p in matchargs)
|
|
1766
|
opts['_matchfiles'] = matchargs
|
|
1766
|
opts['_matchfiles'] = matchargs
|
|
1767
|
if follow:
|
|
1767
|
if follow:
|
|
1768
|
opts[fnopats[0][followfirst]] = '.'
|
|
1768
|
opts[fnopats[0][followfirst]] = '.'
|
|
1769
|
else:
|
|
1769
|
else:
|
|
1770
|
if follow:
|
|
1770
|
if follow:
|
|
1771
|
if pats:
|
|
1771
|
if pats:
|
|
1772
|
# follow() revset interprets its file argument as a
|
|
1772
|
# follow() revset interprets its file argument as a
|
|
1773
|
# manifest entry, so use match.files(), not pats.
|
|
1773
|
# manifest entry, so use match.files(), not pats.
|
|
1774
|
opts[fpats[followfirst]] = list(match.files())
|
|
1774
|
opts[fpats[followfirst]] = list(match.files())
|
|
1775
|
else:
|
|
1775
|
else:
|
|
1776
|
op = fnopats[followdescendants][followfirst]
|
|
1776
|
op = fnopats[followdescendants][followfirst]
|
|
1777
|
opts[op] = 'rev(%d)' % startrev
|
|
1777
|
opts[op] = 'rev(%d)' % startrev
|
|
1778
|
else:
|
|
1778
|
else:
|
|
1779
|
opts['_patslog'] = list(pats)
|
|
1779
|
opts['_patslog'] = list(pats)
|
|
1780
|
|
|
1780
|
|
|
1781
|
filematcher = None
|
|
1781
|
filematcher = None
|
|
1782
|
if opts.get('patch') or opts.get('stat'):
|
|
1782
|
if opts.get('patch') or opts.get('stat'):
|
|
1783
|
# When following files, track renames via a special matcher.
|
|
1783
|
# When following files, track renames via a special matcher.
|
|
1784
|
# If we're forced to take the slowpath it means we're following
|
|
1784
|
# If we're forced to take the slowpath it means we're following
|
|
1785
|
# at least one pattern/directory, so don't bother with rename tracking.
|
|
1785
|
# at least one pattern/directory, so don't bother with rename tracking.
|
|
1786
|
if follow and not match.always() and not slowpath:
|
|
1786
|
if follow and not match.always() and not slowpath:
|
|
1787
|
# _makefollowlogfilematcher expects its files argument to be
|
|
1787
|
# _makefollowlogfilematcher expects its files argument to be
|
|
1788
|
# relative to the repo root, so use match.files(), not pats.
|
|
1788
|
# relative to the repo root, so use match.files(), not pats.
|
|
1789
|
filematcher = _makefollowlogfilematcher(repo, match.files(),
|
|
1789
|
filematcher = _makefollowlogfilematcher(repo, match.files(),
|
|
1790
|
followfirst)
|
|
1790
|
followfirst)
|
|
1791
|
else:
|
|
1791
|
else:
|
|
1792
|
filematcher = _makenofollowlogfilematcher(repo, pats, opts)
|
|
1792
|
filematcher = _makenofollowlogfilematcher(repo, pats, opts)
|
|
1793
|
if filematcher is None:
|
|
1793
|
if filematcher is None:
|
|
1794
|
filematcher = lambda rev: match
|
|
1794
|
filematcher = lambda rev: match
|
|
1795
|
|
|
1795
|
|
|
1796
|
expr = []
|
|
1796
|
expr = []
|
|
1797
|
for op, val in sorted(opts.iteritems()):
|
|
1797
|
for op, val in sorted(opts.iteritems()):
|
|
1798
|
if not val:
|
|
1798
|
if not val:
|
|
1799
|
continue
|
|
1799
|
continue
|
|
1800
|
if op not in opt2revset:
|
|
1800
|
if op not in opt2revset:
|
|
1801
|
continue
|
|
1801
|
continue
|
|
1802
|
revop, andor = opt2revset[op]
|
|
1802
|
revop, andor = opt2revset[op]
|
|
1803
|
if '%(val)' not in revop:
|
|
1803
|
if '%(val)' not in revop:
|
|
1804
|
expr.append(revop)
|
|
1804
|
expr.append(revop)
|
|
1805
|
else:
|
|
1805
|
else:
|
|
1806
|
if not isinstance(val, list):
|
|
1806
|
if not isinstance(val, list):
|
|
1807
|
e = revop % {'val': val}
|
|
1807
|
e = revop % {'val': val}
|
|
1808
|
else:
|
|
1808
|
else:
|
|
1809
|
e = '(' + andor.join((revop % {'val': v}) for v in val) + ')'
|
|
1809
|
e = '(' + andor.join((revop % {'val': v}) for v in val) + ')'
|
|
1810
|
expr.append(e)
|
|
1810
|
expr.append(e)
|
|
1811
|
|
|
1811
|
|
|
1812
|
if expr:
|
|
1812
|
if expr:
|
|
1813
|
expr = '(' + ' and '.join(expr) + ')'
|
|
1813
|
expr = '(' + ' and '.join(expr) + ')'
|
|
1814
|
else:
|
|
1814
|
else:
|
|
1815
|
expr = None
|
|
1815
|
expr = None
|
|
1816
|
return expr, filematcher
|
|
1816
|
return expr, filematcher
|
|
1817
|
|
|
1817
|
|
|
1818
|
def _logrevs(repo, opts):
|
|
1818
|
def _logrevs(repo, opts):
|
|
1819
|
# Default --rev value depends on --follow but --follow behaviour
|
|
1819
|
# Default --rev value depends on --follow but --follow behaviour
|
|
1820
|
# depends on revisions resolved from --rev...
|
|
1820
|
# depends on revisions resolved from --rev...
|
|
1821
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1821
|
follow = opts.get('follow') or opts.get('follow_first')
|
|
1822
|
if opts.get('rev'):
|
|
1822
|
if opts.get('rev'):
|
|
1823
|
revs = scmutil.revrange(repo, opts['rev'])
|
|
1823
|
revs = scmutil.revrange(repo, opts['rev'])
|
|
1824
|
elif follow and repo.dirstate.p1() == nullid:
|
|
1824
|
elif follow and repo.dirstate.p1() == nullid:
|
|
1825
|
revs = revset.baseset()
|
|
1825
|
revs = revset.baseset()
|
|
1826
|
elif follow:
|
|
1826
|
elif follow:
|
|
1827
|
revs = repo.revs('reverse(:.)')
|
|
1827
|
revs = repo.revs('reverse(:.)')
|
|
1828
|
else:
|
|
1828
|
else:
|
|
1829
|
revs = revset.spanset(repo)
|
|
1829
|
revs = revset.spanset(repo)
|
|
1830
|
revs.reverse()
|
|
1830
|
revs.reverse()
|
|
1831
|
return revs
|
|
1831
|
return revs
|
|
1832
|
|
|
1832
|
|
|
1833
|
def getgraphlogrevs(repo, pats, opts):
|
|
1833
|
def getgraphlogrevs(repo, pats, opts):
|
|
1834
|
"""Return (revs, expr, filematcher) where revs is an iterable of
|
|
1834
|
"""Return (revs, expr, filematcher) where revs is an iterable of
|
|
1835
|
revision numbers, expr is a revset string built from log options
|
|
1835
|
revision numbers, expr is a revset string built from log options
|
|
1836
|
and file patterns or None, and used to filter 'revs'. If --stat or
|
|
1836
|
and file patterns or None, and used to filter 'revs'. If --stat or
|
|
1837
|
--patch are not passed filematcher is None. Otherwise it is a
|
|
1837
|
--patch are not passed filematcher is None. Otherwise it is a
|
|
1838
|
callable taking a revision number and returning a match objects
|
|
1838
|
callable taking a revision number and returning a match objects
|
|
1839
|
filtering the files to be detailed when displaying the revision.
|
|
1839
|
filtering the files to be detailed when displaying the revision.
|
|
1840
|
"""
|
|
1840
|
"""
|
|
1841
|
limit = loglimit(opts)
|
|
1841
|
limit = loglimit(opts)
|
|
1842
|
revs = _logrevs(repo, opts)
|
|
1842
|
revs = _logrevs(repo, opts)
|
|
1843
|
if not revs:
|
|
1843
|
if not revs:
|
|
1844
|
return revset.baseset(), None, None
|
|
1844
|
return revset.baseset(), None, None
|
|
1845
|
expr, filematcher = _makelogrevset(repo, pats, opts, revs)
|
|
1845
|
expr, filematcher = _makelogrevset(repo, pats, opts, revs)
|
|
1846
|
if opts.get('rev'):
|
|
1846
|
if opts.get('rev'):
|
|
1847
|
# User-specified revs might be unsorted, but don't sort before
|
|
1847
|
# User-specified revs might be unsorted, but don't sort before
|
|
1848
|
# _makelogrevset because it might depend on the order of revs
|
|
1848
|
# _makelogrevset because it might depend on the order of revs
|
|
1849
|
revs.sort(reverse=True)
|
|
1849
|
revs.sort(reverse=True)
|
|
1850
|
if expr:
|
|
1850
|
if expr:
|
|
1851
|
# Revset matchers often operate faster on revisions in changelog
|
|
1851
|
# Revset matchers often operate faster on revisions in changelog
|
|
1852
|
# order, because most filters deal with the changelog.
|
|
1852
|
# order, because most filters deal with the changelog.
|
|
1853
|
revs.reverse()
|
|
1853
|
revs.reverse()
|
|
1854
|
matcher = revset.match(repo.ui, expr)
|
|
1854
|
matcher = revset.match(repo.ui, expr)
|
|
1855
|
# Revset matches can reorder revisions. "A or B" typically returns
|
|
1855
|
# Revset matches can reorder revisions. "A or B" typically returns
|
|
1856
|
# returns the revision matching A then the revision matching B. Sort
|
|
1856
|
# returns the revision matching A then the revision matching B. Sort
|
|
1857
|
# again to fix that.
|
|
1857
|
# again to fix that.
|
|
1858
|
revs = matcher(repo, revs)
|
|
1858
|
revs = matcher(repo, revs)
|
|
1859
|
revs.sort(reverse=True)
|
|
1859
|
revs.sort(reverse=True)
|
|
1860
|
if limit is not None:
|
|
1860
|
if limit is not None:
|
|
1861
|
limitedrevs = []
|
|
1861
|
limitedrevs = []
|
|
1862
|
for idx, rev in enumerate(revs):
|
|
1862
|
for idx, rev in enumerate(revs):
|
|
1863
|
if idx >= limit:
|
|
1863
|
if idx >= limit:
|
|
1864
|
break
|
|
1864
|
break
|
|
1865
|
limitedrevs.append(rev)
|
|
1865
|
limitedrevs.append(rev)
|
|
1866
|
revs = revset.baseset(limitedrevs)
|
|
1866
|
revs = revset.baseset(limitedrevs)
|
|
1867
|
|
|
1867
|
|
|
1868
|
return revs, expr, filematcher
|
|
1868
|
return revs, expr, filematcher
|
|
1869
|
|
|
1869
|
|
|
1870
|
def getlogrevs(repo, pats, opts):
|
|
1870
|
def getlogrevs(repo, pats, opts):
|
|
1871
|
"""Return (revs, expr, filematcher) where revs is an iterable of
|
|
1871
|
"""Return (revs, expr, filematcher) where revs is an iterable of
|
|
1872
|
revision numbers, expr is a revset string built from log options
|
|
1872
|
revision numbers, expr is a revset string built from log options
|
|
1873
|
and file patterns or None, and used to filter 'revs'. If --stat or
|
|
1873
|
and file patterns or None, and used to filter 'revs'. If --stat or
|
|
1874
|
--patch are not passed filematcher is None. Otherwise it is a
|
|
1874
|
--patch are not passed filematcher is None. Otherwise it is a
|
|
1875
|
callable taking a revision number and returning a match objects
|
|
1875
|
callable taking a revision number and returning a match objects
|
|
1876
|
filtering the files to be detailed when displaying the revision.
|
|
1876
|
filtering the files to be detailed when displaying the revision.
|
|
1877
|
"""
|
|
1877
|
"""
|
|
1878
|
limit = loglimit(opts)
|
|
1878
|
limit = loglimit(opts)
|
|
1879
|
revs = _logrevs(repo, opts)
|
|
1879
|
revs = _logrevs(repo, opts)
|
|
1880
|
if not revs:
|
|
1880
|
if not revs:
|
|
1881
|
return revset.baseset([]), None, None
|
|
1881
|
return revset.baseset([]), None, None
|
|
1882
|
expr, filematcher = _makelogrevset(repo, pats, opts, revs)
|
|
1882
|
expr, filematcher = _makelogrevset(repo, pats, opts, revs)
|
|
1883
|
if expr:
|
|
1883
|
if expr:
|
|
1884
|
# Revset matchers often operate faster on revisions in changelog
|
|
1884
|
# Revset matchers often operate faster on revisions in changelog
|
|
1885
|
# order, because most filters deal with the changelog.
|
|
1885
|
# order, because most filters deal with the changelog.
|
|
1886
|
if not opts.get('rev'):
|
|
1886
|
if not opts.get('rev'):
|
|
1887
|
revs.reverse()
|
|
1887
|
revs.reverse()
|
|
1888
|
matcher = revset.match(repo.ui, expr)
|
|
1888
|
matcher = revset.match(repo.ui, expr)
|
|
1889
|
# Revset matches can reorder revisions. "A or B" typically returns
|
|
1889
|
# Revset matches can reorder revisions. "A or B" typically returns
|
|
1890
|
# returns the revision matching A then the revision matching B. Sort
|
|
1890
|
# returns the revision matching A then the revision matching B. Sort
|
|
1891
|
# again to fix that.
|
|
1891
|
# again to fix that.
|
|
1892
|
revs = matcher(repo, revs)
|
|
1892
|
revs = matcher(repo, revs)
|
|
1893
|
if not opts.get('rev'):
|
|
1893
|
if not opts.get('rev'):
|
|
1894
|
revs.sort(reverse=True)
|
|
1894
|
revs.sort(reverse=True)
|
|
1895
|
if limit is not None:
|
|
1895
|
if limit is not None:
|
|
1896
|
count = 0
|
|
1896
|
count = 0
|
|
1897
|
limitedrevs = []
|
|
1897
|
limitedrevs = []
|
|
1898
|
it = iter(revs)
|
|
1898
|
it = iter(revs)
|
|
1899
|
while count < limit:
|
|
1899
|
while count < limit:
|
|
1900
|
try:
|
|
1900
|
try:
|
|
1901
|
limitedrevs.append(it.next())
|
|
1901
|
limitedrevs.append(it.next())
|
|
1902
|
except (StopIteration):
|
|
1902
|
except (StopIteration):
|
|
1903
|
break
|
|
1903
|
break
|
|
1904
|
count += 1
|
|
1904
|
count += 1
|
|
1905
|
revs = revset.baseset(limitedrevs)
|
|
1905
|
revs = revset.baseset(limitedrevs)
|
|
1906
|
|
|
1906
|
|
|
1907
|
return revs, expr, filematcher
|
|
1907
|
return revs, expr, filematcher
|
|
1908
|
|
|
1908
|
|
|
1909
|
def displaygraph(ui, dag, displayer, showparents, edgefn, getrenamed=None,
|
|
1909
|
def displaygraph(ui, dag, displayer, showparents, edgefn, getrenamed=None,
|
|
1910
|
filematcher=None):
|
|
1910
|
filematcher=None):
|
|
1911
|
seen, state = [], graphmod.asciistate()
|
|
1911
|
seen, state = [], graphmod.asciistate()
|
|
1912
|
for rev, type, ctx, parents in dag:
|
|
1912
|
for rev, type, ctx, parents in dag:
|
|
1913
|
char = 'o'
|
|
1913
|
char = 'o'
|
|
1914
|
if ctx.node() in showparents:
|
|
1914
|
if ctx.node() in showparents:
|
|
1915
|
char = '@'
|
|
1915
|
char = '@'
|
|
1916
|
elif ctx.obsolete():
|
|
1916
|
elif ctx.obsolete():
|
|
1917
|
char = 'x'
|
|
1917
|
char = 'x'
|
|
|
|
|
1918
|
elif ctx.closesbranch():
|
|
|
|
|
1919
|
char = '_'
|
|
1918
|
copies = None
|
|
1920
|
copies = None
|
|
1919
|
if getrenamed and ctx.rev():
|
|
1921
|
if getrenamed and ctx.rev():
|
|
1920
|
copies = []
|
|
1922
|
copies = []
|
|
1921
|
for fn in ctx.files():
|
|
1923
|
for fn in ctx.files():
|
|
1922
|
rename = getrenamed(fn, ctx.rev())
|
|
1924
|
rename = getrenamed(fn, ctx.rev())
|
|
1923
|
if rename:
|
|
1925
|
if rename:
|
|
1924
|
copies.append((fn, rename[0]))
|
|
1926
|
copies.append((fn, rename[0]))
|
|
1925
|
revmatchfn = None
|
|
1927
|
revmatchfn = None
|
|
1926
|
if filematcher is not None:
|
|
1928
|
if filematcher is not None:
|
|
1927
|
revmatchfn = filematcher(ctx.rev())
|
|
1929
|
revmatchfn = filematcher(ctx.rev())
|
|
1928
|
displayer.show(ctx, copies=copies, matchfn=revmatchfn)
|
|
1930
|
displayer.show(ctx, copies=copies, matchfn=revmatchfn)
|
|
1929
|
lines = displayer.hunk.pop(rev).split('\n')
|
|
1931
|
lines = displayer.hunk.pop(rev).split('\n')
|
|
1930
|
if not lines[-1]:
|
|
1932
|
if not lines[-1]:
|
|
1931
|
del lines[-1]
|
|
1933
|
del lines[-1]
|
|
1932
|
displayer.flush(rev)
|
|
1934
|
displayer.flush(rev)
|
|
1933
|
edges = edgefn(type, char, lines, seen, rev, parents)
|
|
1935
|
edges = edgefn(type, char, lines, seen, rev, parents)
|
|
1934
|
for type, char, lines, coldata in edges:
|
|
1936
|
for type, char, lines, coldata in edges:
|
|
1935
|
graphmod.ascii(ui, state, type, char, lines, coldata)
|
|
1937
|
graphmod.ascii(ui, state, type, char, lines, coldata)
|
|
1936
|
displayer.close()
|
|
1938
|
displayer.close()
|
|
1937
|
|
|
1939
|
|
|
1938
|
def graphlog(ui, repo, *pats, **opts):
|
|
1940
|
def graphlog(ui, repo, *pats, **opts):
|
|
1939
|
# Parameters are identical to log command ones
|
|
1941
|
# Parameters are identical to log command ones
|
|
1940
|
revs, expr, filematcher = getgraphlogrevs(repo, pats, opts)
|
|
1942
|
revs, expr, filematcher = getgraphlogrevs(repo, pats, opts)
|
|
1941
|
revdag = graphmod.dagwalker(repo, revs)
|
|
1943
|
revdag = graphmod.dagwalker(repo, revs)
|
|
1942
|
|
|
1944
|
|
|
1943
|
getrenamed = None
|
|
1945
|
getrenamed = None
|
|
1944
|
if opts.get('copies'):
|
|
1946
|
if opts.get('copies'):
|
|
1945
|
endrev = None
|
|
1947
|
endrev = None
|
|
1946
|
if opts.get('rev'):
|
|
1948
|
if opts.get('rev'):
|
|
1947
|
endrev = scmutil.revrange(repo, opts.get('rev')).max() + 1
|
|
1949
|
endrev = scmutil.revrange(repo, opts.get('rev')).max() + 1
|
|
1948
|
getrenamed = templatekw.getrenamedfn(repo, endrev=endrev)
|
|
1950
|
getrenamed = templatekw.getrenamedfn(repo, endrev=endrev)
|
|
1949
|
displayer = show_changeset(ui, repo, opts, buffered=True)
|
|
1951
|
displayer = show_changeset(ui, repo, opts, buffered=True)
|
|
1950
|
showparents = [ctx.node() for ctx in repo[None].parents()]
|
|
1952
|
showparents = [ctx.node() for ctx in repo[None].parents()]
|
|
1951
|
displaygraph(ui, revdag, displayer, showparents,
|
|
1953
|
displaygraph(ui, revdag, displayer, showparents,
|
|
1952
|
graphmod.asciiedges, getrenamed, filematcher)
|
|
1954
|
graphmod.asciiedges, getrenamed, filematcher)
|
|
1953
|
|
|
1955
|
|
|
1954
|
def checkunsupportedgraphflags(pats, opts):
|
|
1956
|
def checkunsupportedgraphflags(pats, opts):
|
|
1955
|
for op in ["newest_first"]:
|
|
1957
|
for op in ["newest_first"]:
|
|
1956
|
if op in opts and opts[op]:
|
|
1958
|
if op in opts and opts[op]:
|
|
1957
|
raise util.Abort(_("-G/--graph option is incompatible with --%s")
|
|
1959
|
raise util.Abort(_("-G/--graph option is incompatible with --%s")
|
|
1958
|
% op.replace("_", "-"))
|
|
1960
|
% op.replace("_", "-"))
|
|
1959
|
|
|
1961
|
|
|
1960
|
def graphrevs(repo, nodes, opts):
|
|
1962
|
def graphrevs(repo, nodes, opts):
|
|
1961
|
limit = loglimit(opts)
|
|
1963
|
limit = loglimit(opts)
|
|
1962
|
nodes.reverse()
|
|
1964
|
nodes.reverse()
|
|
1963
|
if limit is not None:
|
|
1965
|
if limit is not None:
|
|
1964
|
nodes = nodes[:limit]
|
|
1966
|
nodes = nodes[:limit]
|
|
1965
|
return graphmod.nodes(repo, nodes)
|
|
1967
|
return graphmod.nodes(repo, nodes)
|
|
1966
|
|
|
1968
|
|
|
1967
|
def add(ui, repo, match, prefix, explicitonly, **opts):
|
|
1969
|
def add(ui, repo, match, prefix, explicitonly, **opts):
|
|
1968
|
join = lambda f: os.path.join(prefix, f)
|
|
1970
|
join = lambda f: os.path.join(prefix, f)
|
|
1969
|
bad = []
|
|
1971
|
bad = []
|
|
1970
|
oldbad = match.bad
|
|
1972
|
oldbad = match.bad
|
|
1971
|
match.bad = lambda x, y: bad.append(x) or oldbad(x, y)
|
|
1973
|
match.bad = lambda x, y: bad.append(x) or oldbad(x, y)
|
|
1972
|
names = []
|
|
1974
|
names = []
|
|
1973
|
wctx = repo[None]
|
|
1975
|
wctx = repo[None]
|
|
1974
|
cca = None
|
|
1976
|
cca = None
|
|
1975
|
abort, warn = scmutil.checkportabilityalert(ui)
|
|
1977
|
abort, warn = scmutil.checkportabilityalert(ui)
|
|
1976
|
if abort or warn:
|
|
1978
|
if abort or warn:
|
|
1977
|
cca = scmutil.casecollisionauditor(ui, abort, repo.dirstate)
|
|
1979
|
cca = scmutil.casecollisionauditor(ui, abort, repo.dirstate)
|
|
1978
|
for f in wctx.walk(match):
|
|
1980
|
for f in wctx.walk(match):
|
|
1979
|
exact = match.exact(f)
|
|
1981
|
exact = match.exact(f)
|
|
1980
|
if exact or not explicitonly and f not in wctx and repo.wvfs.lexists(f):
|
|
1982
|
if exact or not explicitonly and f not in wctx and repo.wvfs.lexists(f):
|
|
1981
|
if cca:
|
|
1983
|
if cca:
|
|
1982
|
cca(f)
|
|
1984
|
cca(f)
|
|
1983
|
names.append(f)
|
|
1985
|
names.append(f)
|
|
1984
|
if ui.verbose or not exact:
|
|
1986
|
if ui.verbose or not exact:
|
|
1985
|
ui.status(_('adding %s\n') % match.rel(f))
|
|
1987
|
ui.status(_('adding %s\n') % match.rel(f))
|
|
1986
|
|
|
1988
|
|
|
1987
|
for subpath in sorted(wctx.substate):
|
|
1989
|
for subpath in sorted(wctx.substate):
|
|
1988
|
sub = wctx.sub(subpath)
|
|
1990
|
sub = wctx.sub(subpath)
|
|
1989
|
try:
|
|
1991
|
try:
|
|
1990
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
1992
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
1991
|
if opts.get('subrepos'):
|
|
1993
|
if opts.get('subrepos'):
|
|
1992
|
bad.extend(sub.add(ui, submatch, prefix, False, **opts))
|
|
1994
|
bad.extend(sub.add(ui, submatch, prefix, False, **opts))
|
|
1993
|
else:
|
|
1995
|
else:
|
|
1994
|
bad.extend(sub.add(ui, submatch, prefix, True, **opts))
|
|
1996
|
bad.extend(sub.add(ui, submatch, prefix, True, **opts))
|
|
1995
|
except error.LookupError:
|
|
1997
|
except error.LookupError:
|
|
1996
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
1998
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
1997
|
% join(subpath))
|
|
1999
|
% join(subpath))
|
|
1998
|
|
|
2000
|
|
|
1999
|
if not opts.get('dry_run'):
|
|
2001
|
if not opts.get('dry_run'):
|
|
2000
|
rejected = wctx.add(names, prefix)
|
|
2002
|
rejected = wctx.add(names, prefix)
|
|
2001
|
bad.extend(f for f in rejected if f in match.files())
|
|
2003
|
bad.extend(f for f in rejected if f in match.files())
|
|
2002
|
return bad
|
|
2004
|
return bad
|
|
2003
|
|
|
2005
|
|
|
2004
|
def forget(ui, repo, match, prefix, explicitonly):
|
|
2006
|
def forget(ui, repo, match, prefix, explicitonly):
|
|
2005
|
join = lambda f: os.path.join(prefix, f)
|
|
2007
|
join = lambda f: os.path.join(prefix, f)
|
|
2006
|
bad = []
|
|
2008
|
bad = []
|
|
2007
|
oldbad = match.bad
|
|
2009
|
oldbad = match.bad
|
|
2008
|
match.bad = lambda x, y: bad.append(x) or oldbad(x, y)
|
|
2010
|
match.bad = lambda x, y: bad.append(x) or oldbad(x, y)
|
|
2009
|
wctx = repo[None]
|
|
2011
|
wctx = repo[None]
|
|
2010
|
forgot = []
|
|
2012
|
forgot = []
|
|
2011
|
s = repo.status(match=match, clean=True)
|
|
2013
|
s = repo.status(match=match, clean=True)
|
|
2012
|
forget = sorted(s[0] + s[1] + s[3] + s[6])
|
|
2014
|
forget = sorted(s[0] + s[1] + s[3] + s[6])
|
|
2013
|
if explicitonly:
|
|
2015
|
if explicitonly:
|
|
2014
|
forget = [f for f in forget if match.exact(f)]
|
|
2016
|
forget = [f for f in forget if match.exact(f)]
|
|
2015
|
|
|
2017
|
|
|
2016
|
for subpath in sorted(wctx.substate):
|
|
2018
|
for subpath in sorted(wctx.substate):
|
|
2017
|
sub = wctx.sub(subpath)
|
|
2019
|
sub = wctx.sub(subpath)
|
|
2018
|
try:
|
|
2020
|
try:
|
|
2019
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
2021
|
submatch = matchmod.narrowmatcher(subpath, match)
|
|
2020
|
subbad, subforgot = sub.forget(submatch, prefix)
|
|
2022
|
subbad, subforgot = sub.forget(submatch, prefix)
|
|
2021
|
bad.extend([subpath + '/' + f for f in subbad])
|
|
2023
|
bad.extend([subpath + '/' + f for f in subbad])
|
|
2022
|
forgot.extend([subpath + '/' + f for f in subforgot])
|
|
2024
|
forgot.extend([subpath + '/' + f for f in subforgot])
|
|
2023
|
except error.LookupError:
|
|
2025
|
except error.LookupError:
|
|
2024
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2026
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2025
|
% join(subpath))
|
|
2027
|
% join(subpath))
|
|
2026
|
|
|
2028
|
|
|
2027
|
if not explicitonly:
|
|
2029
|
if not explicitonly:
|
|
2028
|
for f in match.files():
|
|
2030
|
for f in match.files():
|
|
2029
|
if f not in repo.dirstate and not repo.wvfs.isdir(f):
|
|
2031
|
if f not in repo.dirstate and not repo.wvfs.isdir(f):
|
|
2030
|
if f not in forgot:
|
|
2032
|
if f not in forgot:
|
|
2031
|
if repo.wvfs.exists(f):
|
|
2033
|
if repo.wvfs.exists(f):
|
|
2032
|
ui.warn(_('not removing %s: '
|
|
2034
|
ui.warn(_('not removing %s: '
|
|
2033
|
'file is already untracked\n')
|
|
2035
|
'file is already untracked\n')
|
|
2034
|
% match.rel(f))
|
|
2036
|
% match.rel(f))
|
|
2035
|
bad.append(f)
|
|
2037
|
bad.append(f)
|
|
2036
|
|
|
2038
|
|
|
2037
|
for f in forget:
|
|
2039
|
for f in forget:
|
|
2038
|
if ui.verbose or not match.exact(f):
|
|
2040
|
if ui.verbose or not match.exact(f):
|
|
2039
|
ui.status(_('removing %s\n') % match.rel(f))
|
|
2041
|
ui.status(_('removing %s\n') % match.rel(f))
|
|
2040
|
|
|
2042
|
|
|
2041
|
rejected = wctx.forget(forget, prefix)
|
|
2043
|
rejected = wctx.forget(forget, prefix)
|
|
2042
|
bad.extend(f for f in rejected if f in match.files())
|
|
2044
|
bad.extend(f for f in rejected if f in match.files())
|
|
2043
|
forgot.extend(f for f in forget if f not in rejected)
|
|
2045
|
forgot.extend(f for f in forget if f not in rejected)
|
|
2044
|
return bad, forgot
|
|
2046
|
return bad, forgot
|
|
2045
|
|
|
2047
|
|
|
2046
|
def remove(ui, repo, m, prefix, after, force, subrepos):
|
|
2048
|
def remove(ui, repo, m, prefix, after, force, subrepos):
|
|
2047
|
join = lambda f: os.path.join(prefix, f)
|
|
2049
|
join = lambda f: os.path.join(prefix, f)
|
|
2048
|
ret = 0
|
|
2050
|
ret = 0
|
|
2049
|
s = repo.status(match=m, clean=True)
|
|
2051
|
s = repo.status(match=m, clean=True)
|
|
2050
|
modified, added, deleted, clean = s[0], s[1], s[3], s[6]
|
|
2052
|
modified, added, deleted, clean = s[0], s[1], s[3], s[6]
|
|
2051
|
|
|
2053
|
|
|
2052
|
wctx = repo[None]
|
|
2054
|
wctx = repo[None]
|
|
2053
|
|
|
2055
|
|
|
2054
|
for subpath in sorted(wctx.substate):
|
|
2056
|
for subpath in sorted(wctx.substate):
|
|
2055
|
def matchessubrepo(matcher, subpath):
|
|
2057
|
def matchessubrepo(matcher, subpath):
|
|
2056
|
if matcher.exact(subpath):
|
|
2058
|
if matcher.exact(subpath):
|
|
2057
|
return True
|
|
2059
|
return True
|
|
2058
|
for f in matcher.files():
|
|
2060
|
for f in matcher.files():
|
|
2059
|
if f.startswith(subpath):
|
|
2061
|
if f.startswith(subpath):
|
|
2060
|
return True
|
|
2062
|
return True
|
|
2061
|
return False
|
|
2063
|
return False
|
|
2062
|
|
|
2064
|
|
|
2063
|
if subrepos or matchessubrepo(m, subpath):
|
|
2065
|
if subrepos or matchessubrepo(m, subpath):
|
|
2064
|
sub = wctx.sub(subpath)
|
|
2066
|
sub = wctx.sub(subpath)
|
|
2065
|
try:
|
|
2067
|
try:
|
|
2066
|
submatch = matchmod.narrowmatcher(subpath, m)
|
|
2068
|
submatch = matchmod.narrowmatcher(subpath, m)
|
|
2067
|
if sub.removefiles(submatch, prefix, after, force, subrepos):
|
|
2069
|
if sub.removefiles(submatch, prefix, after, force, subrepos):
|
|
2068
|
ret = 1
|
|
2070
|
ret = 1
|
|
2069
|
except error.LookupError:
|
|
2071
|
except error.LookupError:
|
|
2070
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2072
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2071
|
% join(subpath))
|
|
2073
|
% join(subpath))
|
|
2072
|
|
|
2074
|
|
|
2073
|
# warn about failure to delete explicit files/dirs
|
|
2075
|
# warn about failure to delete explicit files/dirs
|
|
2074
|
deleteddirs = scmutil.dirs(deleted)
|
|
2076
|
deleteddirs = scmutil.dirs(deleted)
|
|
2075
|
for f in m.files():
|
|
2077
|
for f in m.files():
|
|
2076
|
def insubrepo():
|
|
2078
|
def insubrepo():
|
|
2077
|
for subpath in wctx.substate:
|
|
2079
|
for subpath in wctx.substate:
|
|
2078
|
if f.startswith(subpath):
|
|
2080
|
if f.startswith(subpath):
|
|
2079
|
return True
|
|
2081
|
return True
|
|
2080
|
return False
|
|
2082
|
return False
|
|
2081
|
|
|
2083
|
|
|
2082
|
isdir = f in deleteddirs or f in wctx.dirs()
|
|
2084
|
isdir = f in deleteddirs or f in wctx.dirs()
|
|
2083
|
if f in repo.dirstate or isdir or f == '.' or insubrepo():
|
|
2085
|
if f in repo.dirstate or isdir or f == '.' or insubrepo():
|
|
2084
|
continue
|
|
2086
|
continue
|
|
2085
|
|
|
2087
|
|
|
2086
|
if repo.wvfs.exists(f):
|
|
2088
|
if repo.wvfs.exists(f):
|
|
2087
|
if repo.wvfs.isdir(f):
|
|
2089
|
if repo.wvfs.isdir(f):
|
|
2088
|
ui.warn(_('not removing %s: no tracked files\n')
|
|
2090
|
ui.warn(_('not removing %s: no tracked files\n')
|
|
2089
|
% m.rel(f))
|
|
2091
|
% m.rel(f))
|
|
2090
|
else:
|
|
2092
|
else:
|
|
2091
|
ui.warn(_('not removing %s: file is untracked\n')
|
|
2093
|
ui.warn(_('not removing %s: file is untracked\n')
|
|
2092
|
% m.rel(f))
|
|
2094
|
% m.rel(f))
|
|
2093
|
# missing files will generate a warning elsewhere
|
|
2095
|
# missing files will generate a warning elsewhere
|
|
2094
|
ret = 1
|
|
2096
|
ret = 1
|
|
2095
|
|
|
2097
|
|
|
2096
|
if force:
|
|
2098
|
if force:
|
|
2097
|
list = modified + deleted + clean + added
|
|
2099
|
list = modified + deleted + clean + added
|
|
2098
|
elif after:
|
|
2100
|
elif after:
|
|
2099
|
list = deleted
|
|
2101
|
list = deleted
|
|
2100
|
for f in modified + added + clean:
|
|
2102
|
for f in modified + added + clean:
|
|
2101
|
ui.warn(_('not removing %s: file still exists\n') % m.rel(f))
|
|
2103
|
ui.warn(_('not removing %s: file still exists\n') % m.rel(f))
|
|
2102
|
ret = 1
|
|
2104
|
ret = 1
|
|
2103
|
else:
|
|
2105
|
else:
|
|
2104
|
list = deleted + clean
|
|
2106
|
list = deleted + clean
|
|
2105
|
for f in modified:
|
|
2107
|
for f in modified:
|
|
2106
|
ui.warn(_('not removing %s: file is modified (use -f'
|
|
2108
|
ui.warn(_('not removing %s: file is modified (use -f'
|
|
2107
|
' to force removal)\n') % m.rel(f))
|
|
2109
|
' to force removal)\n') % m.rel(f))
|
|
2108
|
ret = 1
|
|
2110
|
ret = 1
|
|
2109
|
for f in added:
|
|
2111
|
for f in added:
|
|
2110
|
ui.warn(_('not removing %s: file has been marked for add'
|
|
2112
|
ui.warn(_('not removing %s: file has been marked for add'
|
|
2111
|
' (use forget to undo)\n') % m.rel(f))
|
|
2113
|
' (use forget to undo)\n') % m.rel(f))
|
|
2112
|
ret = 1
|
|
2114
|
ret = 1
|
|
2113
|
|
|
2115
|
|
|
2114
|
for f in sorted(list):
|
|
2116
|
for f in sorted(list):
|
|
2115
|
if ui.verbose or not m.exact(f):
|
|
2117
|
if ui.verbose or not m.exact(f):
|
|
2116
|
ui.status(_('removing %s\n') % m.rel(f))
|
|
2118
|
ui.status(_('removing %s\n') % m.rel(f))
|
|
2117
|
|
|
2119
|
|
|
2118
|
wlock = repo.wlock()
|
|
2120
|
wlock = repo.wlock()
|
|
2119
|
try:
|
|
2121
|
try:
|
|
2120
|
if not after:
|
|
2122
|
if not after:
|
|
2121
|
for f in list:
|
|
2123
|
for f in list:
|
|
2122
|
if f in added:
|
|
2124
|
if f in added:
|
|
2123
|
continue # we never unlink added files on remove
|
|
2125
|
continue # we never unlink added files on remove
|
|
2124
|
util.unlinkpath(repo.wjoin(f), ignoremissing=True)
|
|
2126
|
util.unlinkpath(repo.wjoin(f), ignoremissing=True)
|
|
2125
|
repo[None].forget(list)
|
|
2127
|
repo[None].forget(list)
|
|
2126
|
finally:
|
|
2128
|
finally:
|
|
2127
|
wlock.release()
|
|
2129
|
wlock.release()
|
|
2128
|
|
|
2130
|
|
|
2129
|
return ret
|
|
2131
|
return ret
|
|
2130
|
|
|
2132
|
|
|
2131
|
def cat(ui, repo, ctx, matcher, prefix, **opts):
|
|
2133
|
def cat(ui, repo, ctx, matcher, prefix, **opts):
|
|
2132
|
err = 1
|
|
2134
|
err = 1
|
|
2133
|
|
|
2135
|
|
|
2134
|
def write(path):
|
|
2136
|
def write(path):
|
|
2135
|
fp = makefileobj(repo, opts.get('output'), ctx.node(),
|
|
2137
|
fp = makefileobj(repo, opts.get('output'), ctx.node(),
|
|
2136
|
pathname=os.path.join(prefix, path))
|
|
2138
|
pathname=os.path.join(prefix, path))
|
|
2137
|
data = ctx[path].data()
|
|
2139
|
data = ctx[path].data()
|
|
2138
|
if opts.get('decode'):
|
|
2140
|
if opts.get('decode'):
|
|
2139
|
data = repo.wwritedata(path, data)
|
|
2141
|
data = repo.wwritedata(path, data)
|
|
2140
|
fp.write(data)
|
|
2142
|
fp.write(data)
|
|
2141
|
fp.close()
|
|
2143
|
fp.close()
|
|
2142
|
|
|
2144
|
|
|
2143
|
# Automation often uses hg cat on single files, so special case it
|
|
2145
|
# Automation often uses hg cat on single files, so special case it
|
|
2144
|
# for performance to avoid the cost of parsing the manifest.
|
|
2146
|
# for performance to avoid the cost of parsing the manifest.
|
|
2145
|
if len(matcher.files()) == 1 and not matcher.anypats():
|
|
2147
|
if len(matcher.files()) == 1 and not matcher.anypats():
|
|
2146
|
file = matcher.files()[0]
|
|
2148
|
file = matcher.files()[0]
|
|
2147
|
mf = repo.manifest
|
|
2149
|
mf = repo.manifest
|
|
2148
|
mfnode = ctx._changeset[0]
|
|
2150
|
mfnode = ctx._changeset[0]
|
|
2149
|
if mf.find(mfnode, file)[0]:
|
|
2151
|
if mf.find(mfnode, file)[0]:
|
|
2150
|
write(file)
|
|
2152
|
write(file)
|
|
2151
|
return 0
|
|
2153
|
return 0
|
|
2152
|
|
|
2154
|
|
|
2153
|
# Don't warn about "missing" files that are really in subrepos
|
|
2155
|
# Don't warn about "missing" files that are really in subrepos
|
|
2154
|
bad = matcher.bad
|
|
2156
|
bad = matcher.bad
|
|
2155
|
|
|
2157
|
|
|
2156
|
def badfn(path, msg):
|
|
2158
|
def badfn(path, msg):
|
|
2157
|
for subpath in ctx.substate:
|
|
2159
|
for subpath in ctx.substate:
|
|
2158
|
if path.startswith(subpath):
|
|
2160
|
if path.startswith(subpath):
|
|
2159
|
return
|
|
2161
|
return
|
|
2160
|
bad(path, msg)
|
|
2162
|
bad(path, msg)
|
|
2161
|
|
|
2163
|
|
|
2162
|
matcher.bad = badfn
|
|
2164
|
matcher.bad = badfn
|
|
2163
|
|
|
2165
|
|
|
2164
|
for abs in ctx.walk(matcher):
|
|
2166
|
for abs in ctx.walk(matcher):
|
|
2165
|
write(abs)
|
|
2167
|
write(abs)
|
|
2166
|
err = 0
|
|
2168
|
err = 0
|
|
2167
|
|
|
2169
|
|
|
2168
|
matcher.bad = bad
|
|
2170
|
matcher.bad = bad
|
|
2169
|
|
|
2171
|
|
|
2170
|
for subpath in sorted(ctx.substate):
|
|
2172
|
for subpath in sorted(ctx.substate):
|
|
2171
|
sub = ctx.sub(subpath)
|
|
2173
|
sub = ctx.sub(subpath)
|
|
2172
|
try:
|
|
2174
|
try:
|
|
2173
|
submatch = matchmod.narrowmatcher(subpath, matcher)
|
|
2175
|
submatch = matchmod.narrowmatcher(subpath, matcher)
|
|
2174
|
|
|
2176
|
|
|
2175
|
if not sub.cat(submatch, os.path.join(prefix, sub._path),
|
|
2177
|
if not sub.cat(submatch, os.path.join(prefix, sub._path),
|
|
2176
|
**opts):
|
|
2178
|
**opts):
|
|
2177
|
err = 0
|
|
2179
|
err = 0
|
|
2178
|
except error.RepoLookupError:
|
|
2180
|
except error.RepoLookupError:
|
|
2179
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2181
|
ui.status(_("skipping missing subrepository: %s\n")
|
|
2180
|
% os.path.join(prefix, subpath))
|
|
2182
|
% os.path.join(prefix, subpath))
|
|
2181
|
|
|
2183
|
|
|
2182
|
return err
|
|
2184
|
return err
|
|
2183
|
|
|
2185
|
|
|
2184
|
def commit(ui, repo, commitfunc, pats, opts):
|
|
2186
|
def commit(ui, repo, commitfunc, pats, opts):
|
|
2185
|
'''commit the specified files or all outstanding changes'''
|
|
2187
|
'''commit the specified files or all outstanding changes'''
|
|
2186
|
date = opts.get('date')
|
|
2188
|
date = opts.get('date')
|
|
2187
|
if date:
|
|
2189
|
if date:
|
|
2188
|
opts['date'] = util.parsedate(date)
|
|
2190
|
opts['date'] = util.parsedate(date)
|
|
2189
|
message = logmessage(ui, opts)
|
|
2191
|
message = logmessage(ui, opts)
|
|
2190
|
matcher = scmutil.match(repo[None], pats, opts)
|
|
2192
|
matcher = scmutil.match(repo[None], pats, opts)
|
|
2191
|
|
|
2193
|
|
|
2192
|
# extract addremove carefully -- this function can be called from a command
|
|
2194
|
# extract addremove carefully -- this function can be called from a command
|
|
2193
|
# that doesn't support addremove
|
|
2195
|
# that doesn't support addremove
|
|
2194
|
if opts.get('addremove'):
|
|
2196
|
if opts.get('addremove'):
|
|
2195
|
if scmutil.addremove(repo, matcher, "", opts) != 0:
|
|
2197
|
if scmutil.addremove(repo, matcher, "", opts) != 0:
|
|
2196
|
raise util.Abort(
|
|
2198
|
raise util.Abort(
|
|
2197
|
_("failed to mark all new/missing files as added/removed"))
|
|
2199
|
_("failed to mark all new/missing files as added/removed"))
|
|
2198
|
|
|
2200
|
|
|
2199
|
return commitfunc(ui, repo, message, matcher, opts)
|
|
2201
|
return commitfunc(ui, repo, message, matcher, opts)
|
|
2200
|
|
|
2202
|
|
|
2201
|
def amend(ui, repo, commitfunc, old, extra, pats, opts):
|
|
2203
|
def amend(ui, repo, commitfunc, old, extra, pats, opts):
|
|
2202
|
# amend will reuse the existing user if not specified, but the obsolete
|
|
2204
|
# amend will reuse the existing user if not specified, but the obsolete
|
|
2203
|
# marker creation requires that the current user's name is specified.
|
|
2205
|
# marker creation requires that the current user's name is specified.
|
|
2204
|
if obsolete._enabled:
|
|
2206
|
if obsolete._enabled:
|
|
2205
|
ui.username() # raise exception if username not set
|
|
2207
|
ui.username() # raise exception if username not set
|
|
2206
|
|
|
2208
|
|
|
2207
|
ui.note(_('amending changeset %s\n') % old)
|
|
2209
|
ui.note(_('amending changeset %s\n') % old)
|
|
2208
|
base = old.p1()
|
|
2210
|
base = old.p1()
|
|
2209
|
|
|
2211
|
|
|
2210
|
wlock = lock = newid = None
|
|
2212
|
wlock = lock = newid = None
|
|
2211
|
try:
|
|
2213
|
try:
|
|
2212
|
wlock = repo.wlock()
|
|
2214
|
wlock = repo.wlock()
|
|
2213
|
lock = repo.lock()
|
|
2215
|
lock = repo.lock()
|
|
2214
|
tr = repo.transaction('amend')
|
|
2216
|
tr = repo.transaction('amend')
|
|
2215
|
try:
|
|
2217
|
try:
|
|
2216
|
# See if we got a message from -m or -l, if not, open the editor
|
|
2218
|
# See if we got a message from -m or -l, if not, open the editor
|
|
2217
|
# with the message of the changeset to amend
|
|
2219
|
# with the message of the changeset to amend
|
|
2218
|
message = logmessage(ui, opts)
|
|
2220
|
message = logmessage(ui, opts)
|
|
2219
|
# ensure logfile does not conflict with later enforcement of the
|
|
2221
|
# ensure logfile does not conflict with later enforcement of the
|
|
2220
|
# message. potential logfile content has been processed by
|
|
2222
|
# message. potential logfile content has been processed by
|
|
2221
|
# `logmessage` anyway.
|
|
2223
|
# `logmessage` anyway.
|
|
2222
|
opts.pop('logfile')
|
|
2224
|
opts.pop('logfile')
|
|
2223
|
# First, do a regular commit to record all changes in the working
|
|
2225
|
# First, do a regular commit to record all changes in the working
|
|
2224
|
# directory (if there are any)
|
|
2226
|
# directory (if there are any)
|
|
2225
|
ui.callhooks = False
|
|
2227
|
ui.callhooks = False
|
|
2226
|
currentbookmark = repo._bookmarkcurrent
|
|
2228
|
currentbookmark = repo._bookmarkcurrent
|
|
2227
|
try:
|
|
2229
|
try:
|
|
2228
|
repo._bookmarkcurrent = None
|
|
2230
|
repo._bookmarkcurrent = None
|
|
2229
|
opts['message'] = 'temporary amend commit for %s' % old
|
|
2231
|
opts['message'] = 'temporary amend commit for %s' % old
|
|
2230
|
node = commit(ui, repo, commitfunc, pats, opts)
|
|
2232
|
node = commit(ui, repo, commitfunc, pats, opts)
|
|
2231
|
finally:
|
|
2233
|
finally:
|
|
2232
|
repo._bookmarkcurrent = currentbookmark
|
|
2234
|
repo._bookmarkcurrent = currentbookmark
|
|
2233
|
ui.callhooks = True
|
|
2235
|
ui.callhooks = True
|
|
2234
|
ctx = repo[node]
|
|
2236
|
ctx = repo[node]
|
|
2235
|
|
|
2237
|
|
|
2236
|
# Participating changesets:
|
|
2238
|
# Participating changesets:
|
|
2237
|
#
|
|
2239
|
#
|
|
2238
|
# node/ctx o - new (intermediate) commit that contains changes
|
|
2240
|
# node/ctx o - new (intermediate) commit that contains changes
|
|
2239
|
# | from working dir to go into amending commit
|
|
2241
|
# | from working dir to go into amending commit
|
|
2240
|
# | (or a workingctx if there were no changes)
|
|
2242
|
# | (or a workingctx if there were no changes)
|
|
2241
|
# |
|
|
2243
|
# |
|
|
2242
|
# old o - changeset to amend
|
|
2244
|
# old o - changeset to amend
|
|
2243
|
# |
|
|
2245
|
# |
|
|
2244
|
# base o - parent of amending changeset
|
|
2246
|
# base o - parent of amending changeset
|
|
2245
|
|
|
2247
|
|
|
2246
|
# Update extra dict from amended commit (e.g. to preserve graft
|
|
2248
|
# Update extra dict from amended commit (e.g. to preserve graft
|
|
2247
|
# source)
|
|
2249
|
# source)
|
|
2248
|
extra.update(old.extra())
|
|
2250
|
extra.update(old.extra())
|
|
2249
|
|
|
2251
|
|
|
2250
|
# Also update it from the intermediate commit or from the wctx
|
|
2252
|
# Also update it from the intermediate commit or from the wctx
|
|
2251
|
extra.update(ctx.extra())
|
|
2253
|
extra.update(ctx.extra())
|
|
2252
|
|
|
2254
|
|
|
2253
|
if len(old.parents()) > 1:
|
|
2255
|
if len(old.parents()) > 1:
|
|
2254
|
# ctx.files() isn't reliable for merges, so fall back to the
|
|
2256
|
# ctx.files() isn't reliable for merges, so fall back to the
|
|
2255
|
# slower repo.status() method
|
|
2257
|
# slower repo.status() method
|
|
2256
|
files = set([fn for st in repo.status(base, old)[:3]
|
|
2258
|
files = set([fn for st in repo.status(base, old)[:3]
|
|
2257
|
for fn in st])
|
|
2259
|
for fn in st])
|
|
2258
|
else:
|
|
2260
|
else:
|
|
2259
|
files = set(old.files())
|
|
2261
|
files = set(old.files())
|
|
2260
|
|
|
2262
|
|
|
2261
|
# Second, we use either the commit we just did, or if there were no
|
|
2263
|
# Second, we use either the commit we just did, or if there were no
|
|
2262
|
# changes the parent of the working directory as the version of the
|
|
2264
|
# changes the parent of the working directory as the version of the
|
|
2263
|
# files in the final amend commit
|
|
2265
|
# files in the final amend commit
|
|
2264
|
if node:
|
|
2266
|
if node:
|
|
2265
|
ui.note(_('copying changeset %s to %s\n') % (ctx, base))
|
|
2267
|
ui.note(_('copying changeset %s to %s\n') % (ctx, base))
|
|
2266
|
|
|
2268
|
|
|
2267
|
user = ctx.user()
|
|
2269
|
user = ctx.user()
|
|
2268
|
date = ctx.date()
|
|
2270
|
date = ctx.date()
|
|
2269
|
# Recompute copies (avoid recording a -> b -> a)
|
|
2271
|
# Recompute copies (avoid recording a -> b -> a)
|
|
2270
|
copied = copies.pathcopies(base, ctx)
|
|
2272
|
copied = copies.pathcopies(base, ctx)
|
|
2271
|
if old.p2:
|
|
2273
|
if old.p2:
|
|
2272
|
copied.update(copies.pathcopies(old.p2(), ctx))
|
|
2274
|
copied.update(copies.pathcopies(old.p2(), ctx))
|
|
2273
|
|
|
2275
|
|
|
2274
|
# Prune files which were reverted by the updates: if old
|
|
2276
|
# Prune files which were reverted by the updates: if old
|
|
2275
|
# introduced file X and our intermediate commit, node,
|
|
2277
|
# introduced file X and our intermediate commit, node,
|
|
2276
|
# renamed that file, then those two files are the same and
|
|
2278
|
# renamed that file, then those two files are the same and
|
|
2277
|
# we can discard X from our list of files. Likewise if X
|
|
2279
|
# we can discard X from our list of files. Likewise if X
|
|
2278
|
# was deleted, it's no longer relevant
|
|
2280
|
# was deleted, it's no longer relevant
|
|
2279
|
files.update(ctx.files())
|
|
2281
|
files.update(ctx.files())
|
|
2280
|
|
|
2282
|
|
|
2281
|
def samefile(f):
|
|
2283
|
def samefile(f):
|
|
2282
|
if f in ctx.manifest():
|
|
2284
|
if f in ctx.manifest():
|
|
2283
|
a = ctx.filectx(f)
|
|
2285
|
a = ctx.filectx(f)
|
|
2284
|
if f in base.manifest():
|
|
2286
|
if f in base.manifest():
|
|
2285
|
b = base.filectx(f)
|
|
2287
|
b = base.filectx(f)
|
|
2286
|
return (not a.cmp(b)
|
|
2288
|
return (not a.cmp(b)
|
|
2287
|
and a.flags() == b.flags())
|
|
2289
|
and a.flags() == b.flags())
|
|
2288
|
else:
|
|
2290
|
else:
|
|
2289
|
return False
|
|
2291
|
return False
|
|
2290
|
else:
|
|
2292
|
else:
|
|
2291
|
return f not in base.manifest()
|
|
2293
|
return f not in base.manifest()
|
|
2292
|
files = [f for f in files if not samefile(f)]
|
|
2294
|
files = [f for f in files if not samefile(f)]
|
|
2293
|
|
|
2295
|
|
|
2294
|
def filectxfn(repo, ctx_, path):
|
|
2296
|
def filectxfn(repo, ctx_, path):
|
|
2295
|
try:
|
|
2297
|
try:
|
|
2296
|
fctx = ctx[path]
|
|
2298
|
fctx = ctx[path]
|
|
2297
|
flags = fctx.flags()
|
|
2299
|
flags = fctx.flags()
|
|
2298
|
mctx = context.memfilectx(repo,
|
|
2300
|
mctx = context.memfilectx(repo,
|
|
2299
|
fctx.path(), fctx.data(),
|
|
2301
|
fctx.path(), fctx.data(),
|
|
2300
|
islink='l' in flags,
|
|
2302
|
islink='l' in flags,
|
|
2301
|
isexec='x' in flags,
|
|
2303
|
isexec='x' in flags,
|
|
2302
|
copied=copied.get(path))
|
|
2304
|
copied=copied.get(path))
|
|
2303
|
return mctx
|
|
2305
|
return mctx
|
|
2304
|
except KeyError:
|
|
2306
|
except KeyError:
|
|
2305
|
return None
|
|
2307
|
return None
|
|
2306
|
else:
|
|
2308
|
else:
|
|
2307
|
ui.note(_('copying changeset %s to %s\n') % (old, base))
|
|
2309
|
ui.note(_('copying changeset %s to %s\n') % (old, base))
|
|
2308
|
|
|
2310
|
|
|
2309
|
# Use version of files as in the old cset
|
|
2311
|
# Use version of files as in the old cset
|
|
2310
|
def filectxfn(repo, ctx_, path):
|
|
2312
|
def filectxfn(repo, ctx_, path):
|
|
2311
|
try:
|
|
2313
|
try:
|
|
2312
|
return old.filectx(path)
|
|
2314
|
return old.filectx(path)
|
|
2313
|
except KeyError:
|
|
2315
|
except KeyError:
|
|
2314
|
return None
|
|
2316
|
return None
|
|
2315
|
|
|
2317
|
|
|
2316
|
user = opts.get('user') or old.user()
|
|
2318
|
user = opts.get('user') or old.user()
|
|
2317
|
date = opts.get('date') or old.date()
|
|
2319
|
date = opts.get('date') or old.date()
|
|
2318
|
editform = mergeeditform(old, 'commit.amend')
|
|
2320
|
editform = mergeeditform(old, 'commit.amend')
|
|
2319
|
editor = getcommiteditor(editform=editform, **opts)
|
|
2321
|
editor = getcommiteditor(editform=editform, **opts)
|
|
2320
|
if not message:
|
|
2322
|
if not message:
|
|
2321
|
editor = getcommiteditor(edit=True, editform=editform)
|
|
2323
|
editor = getcommiteditor(edit=True, editform=editform)
|
|
2322
|
message = old.description()
|
|
2324
|
message = old.description()
|
|
2323
|
|
|
2325
|
|
|
2324
|
pureextra = extra.copy()
|
|
2326
|
pureextra = extra.copy()
|
|
2325
|
extra['amend_source'] = old.hex()
|
|
2327
|
extra['amend_source'] = old.hex()
|
|
2326
|
|
|
2328
|
|
|
2327
|
new = context.memctx(repo,
|
|
2329
|
new = context.memctx(repo,
|
|
2328
|
parents=[base.node(), old.p2().node()],
|
|
2330
|
parents=[base.node(), old.p2().node()],
|
|
2329
|
text=message,
|
|
2331
|
text=message,
|
|
2330
|
files=files,
|
|
2332
|
files=files,
|
|
2331
|
filectxfn=filectxfn,
|
|
2333
|
filectxfn=filectxfn,
|
|
2332
|
user=user,
|
|
2334
|
user=user,
|
|
2333
|
date=date,
|
|
2335
|
date=date,
|
|
2334
|
extra=extra,
|
|
2336
|
extra=extra,
|
|
2335
|
editor=editor)
|
|
2337
|
editor=editor)
|
|
2336
|
|
|
2338
|
|
|
2337
|
newdesc = changelog.stripdesc(new.description())
|
|
2339
|
newdesc = changelog.stripdesc(new.description())
|
|
2338
|
if ((not node)
|
|
2340
|
if ((not node)
|
|
2339
|
and newdesc == old.description()
|
|
2341
|
and newdesc == old.description()
|
|
2340
|
and user == old.user()
|
|
2342
|
and user == old.user()
|
|
2341
|
and date == old.date()
|
|
2343
|
and date == old.date()
|
|
2342
|
and pureextra == old.extra()):
|
|
2344
|
and pureextra == old.extra()):
|
|
2343
|
# nothing changed. continuing here would create a new node
|
|
2345
|
# nothing changed. continuing here would create a new node
|
|
2344
|
# anyway because of the amend_source noise.
|
|
2346
|
# anyway because of the amend_source noise.
|
|
2345
|
#
|
|
2347
|
#
|
|
2346
|
# This not what we expect from amend.
|
|
2348
|
# This not what we expect from amend.
|
|
2347
|
return old.node()
|
|
2349
|
return old.node()
|
|
2348
|
|
|
2350
|
|
|
2349
|
ph = repo.ui.config('phases', 'new-commit', phases.draft)
|
|
2351
|
ph = repo.ui.config('phases', 'new-commit', phases.draft)
|
|
2350
|
try:
|
|
2352
|
try:
|
|
2351
|
if opts.get('secret'):
|
|
2353
|
if opts.get('secret'):
|
|
2352
|
commitphase = 'secret'
|
|
2354
|
commitphase = 'secret'
|
|
2353
|
else:
|
|
2355
|
else:
|
|
2354
|
commitphase = old.phase()
|
|
2356
|
commitphase = old.phase()
|
|
2355
|
repo.ui.setconfig('phases', 'new-commit', commitphase, 'amend')
|
|
2357
|
repo.ui.setconfig('phases', 'new-commit', commitphase, 'amend')
|
|
2356
|
newid = repo.commitctx(new)
|
|
2358
|
newid = repo.commitctx(new)
|
|
2357
|
finally:
|
|
2359
|
finally:
|
|
2358
|
repo.ui.setconfig('phases', 'new-commit', ph, 'amend')
|
|
2360
|
repo.ui.setconfig('phases', 'new-commit', ph, 'amend')
|
|
2359
|
if newid != old.node():
|
|
2361
|
if newid != old.node():
|
|
2360
|
# Reroute the working copy parent to the new changeset
|
|
2362
|
# Reroute the working copy parent to the new changeset
|
|
2361
|
repo.setparents(newid, nullid)
|
|
2363
|
repo.setparents(newid, nullid)
|
|
2362
|
|
|
2364
|
|
|
2363
|
# Move bookmarks from old parent to amend commit
|
|
2365
|
# Move bookmarks from old parent to amend commit
|
|
2364
|
bms = repo.nodebookmarks(old.node())
|
|
2366
|
bms = repo.nodebookmarks(old.node())
|
|
2365
|
if bms:
|
|
2367
|
if bms:
|
|
2366
|
marks = repo._bookmarks
|
|
2368
|
marks = repo._bookmarks
|
|
2367
|
for bm in bms:
|
|
2369
|
for bm in bms:
|
|
2368
|
marks[bm] = newid
|
|
2370
|
marks[bm] = newid
|
|
2369
|
marks.write()
|
|
2371
|
marks.write()
|
|
2370
|
#commit the whole amend process
|
|
2372
|
#commit the whole amend process
|
|
2371
|
createmarkers = obsolete.isenabled(repo, obsolete.createmarkersopt)
|
|
2373
|
createmarkers = obsolete.isenabled(repo, obsolete.createmarkersopt)
|
|
2372
|
if createmarkers and newid != old.node():
|
|
2374
|
if createmarkers and newid != old.node():
|
|
2373
|
# mark the new changeset as successor of the rewritten one
|
|
2375
|
# mark the new changeset as successor of the rewritten one
|
|
2374
|
new = repo[newid]
|
|
2376
|
new = repo[newid]
|
|
2375
|
obs = [(old, (new,))]
|
|
2377
|
obs = [(old, (new,))]
|
|
2376
|
if node:
|
|
2378
|
if node:
|
|
2377
|
obs.append((ctx, ()))
|
|
2379
|
obs.append((ctx, ()))
|
|
2378
|
|
|
2380
|
|
|
2379
|
obsolete.createmarkers(repo, obs)
|
|
2381
|
obsolete.createmarkers(repo, obs)
|
|
2380
|
tr.close()
|
|
2382
|
tr.close()
|
|
2381
|
finally:
|
|
2383
|
finally:
|
|
2382
|
tr.release()
|
|
2384
|
tr.release()
|
|
2383
|
if not createmarkers and newid != old.node():
|
|
2385
|
if not createmarkers and newid != old.node():
|
|
2384
|
# Strip the intermediate commit (if there was one) and the amended
|
|
2386
|
# Strip the intermediate commit (if there was one) and the amended
|
|
2385
|
# commit
|
|
2387
|
# commit
|
|
2386
|
if node:
|
|
2388
|
if node:
|
|
2387
|
ui.note(_('stripping intermediate changeset %s\n') % ctx)
|
|
2389
|
ui.note(_('stripping intermediate changeset %s\n') % ctx)
|
|
2388
|
ui.note(_('stripping amended changeset %s\n') % old)
|
|
2390
|
ui.note(_('stripping amended changeset %s\n') % old)
|
|
2389
|
repair.strip(ui, repo, old.node(), topic='amend-backup')
|
|
2391
|
repair.strip(ui, repo, old.node(), topic='amend-backup')
|
|
2390
|
finally:
|
|
2392
|
finally:
|
|
2391
|
if newid is None:
|
|
2393
|
if newid is None:
|
|
2392
|
repo.dirstate.invalidate()
|
|
2394
|
repo.dirstate.invalidate()
|
|
2393
|
lockmod.release(lock, wlock)
|
|
2395
|
lockmod.release(lock, wlock)
|
|
2394
|
return newid
|
|
2396
|
return newid
|
|
2395
|
|
|
2397
|
|
|
2396
|
def commiteditor(repo, ctx, subs, editform=''):
|
|
2398
|
def commiteditor(repo, ctx, subs, editform=''):
|
|
2397
|
if ctx.description():
|
|
2399
|
if ctx.description():
|
|
2398
|
return ctx.description()
|
|
2400
|
return ctx.description()
|
|
2399
|
return commitforceeditor(repo, ctx, subs, editform=editform)
|
|
2401
|
return commitforceeditor(repo, ctx, subs, editform=editform)
|
|
2400
|
|
|
2402
|
|
|
2401
|
def commitforceeditor(repo, ctx, subs, finishdesc=None, extramsg=None,
|
|
2403
|
def commitforceeditor(repo, ctx, subs, finishdesc=None, extramsg=None,
|
|
2402
|
editform=''):
|
|
2404
|
editform=''):
|
|
2403
|
if not extramsg:
|
|
2405
|
if not extramsg:
|
|
2404
|
extramsg = _("Leave message empty to abort commit.")
|
|
2406
|
extramsg = _("Leave message empty to abort commit.")
|
|
2405
|
|
|
2407
|
|
|
2406
|
forms = [e for e in editform.split('.') if e]
|
|
2408
|
forms = [e for e in editform.split('.') if e]
|
|
2407
|
forms.insert(0, 'changeset')
|
|
2409
|
forms.insert(0, 'changeset')
|
|
2408
|
while forms:
|
|
2410
|
while forms:
|
|
2409
|
tmpl = repo.ui.config('committemplate', '.'.join(forms))
|
|
2411
|
tmpl = repo.ui.config('committemplate', '.'.join(forms))
|
|
2410
|
if tmpl:
|
|
2412
|
if tmpl:
|
|
2411
|
committext = buildcommittemplate(repo, ctx, subs, extramsg, tmpl)
|
|
2413
|
committext = buildcommittemplate(repo, ctx, subs, extramsg, tmpl)
|
|
2412
|
break
|
|
2414
|
break
|
|
2413
|
forms.pop()
|
|
2415
|
forms.pop()
|
|
2414
|
else:
|
|
2416
|
else:
|
|
2415
|
committext = buildcommittext(repo, ctx, subs, extramsg)
|
|
2417
|
committext = buildcommittext(repo, ctx, subs, extramsg)
|
|
2416
|
|
|
2418
|
|
|
2417
|
# run editor in the repository root
|
|
2419
|
# run editor in the repository root
|
|
2418
|
olddir = os.getcwd()
|
|
2420
|
olddir = os.getcwd()
|
|
2419
|
os.chdir(repo.root)
|
|
2421
|
os.chdir(repo.root)
|
|
2420
|
text = repo.ui.edit(committext, ctx.user(), ctx.extra(), editform=editform)
|
|
2422
|
text = repo.ui.edit(committext, ctx.user(), ctx.extra(), editform=editform)
|
|
2421
|
text = re.sub("(?m)^HG:.*(\n|$)", "", text)
|
|
2423
|
text = re.sub("(?m)^HG:.*(\n|$)", "", text)
|
|
2422
|
os.chdir(olddir)
|
|
2424
|
os.chdir(olddir)
|
|
2423
|
|
|
2425
|
|
|
2424
|
if finishdesc:
|
|
2426
|
if finishdesc:
|
|
2425
|
text = finishdesc(text)
|
|
2427
|
text = finishdesc(text)
|
|
2426
|
if not text.strip():
|
|
2428
|
if not text.strip():
|
|
2427
|
raise util.Abort(_("empty commit message"))
|
|
2429
|
raise util.Abort(_("empty commit message"))
|
|
2428
|
|
|
2430
|
|
|
2429
|
return text
|
|
2431
|
return text
|
|
2430
|
|
|
2432
|
|
|
2431
|
def buildcommittemplate(repo, ctx, subs, extramsg, tmpl):
|
|
2433
|
def buildcommittemplate(repo, ctx, subs, extramsg, tmpl):
|
|
2432
|
ui = repo.ui
|
|
2434
|
ui = repo.ui
|
|
2433
|
tmpl, mapfile = gettemplate(ui, tmpl, None)
|
|
2435
|
tmpl, mapfile = gettemplate(ui, tmpl, None)
|
|
2434
|
|
|
2436
|
|
|
2435
|
try:
|
|
2437
|
try:
|
|
2436
|
t = changeset_templater(ui, repo, None, {}, tmpl, mapfile, False)
|
|
2438
|
t = changeset_templater(ui, repo, None, {}, tmpl, mapfile, False)
|
|
2437
|
except SyntaxError, inst:
|
|
2439
|
except SyntaxError, inst:
|
|
2438
|
raise util.Abort(inst.args[0])
|
|
2440
|
raise util.Abort(inst.args[0])
|
|
2439
|
|
|
2441
|
|
|
2440
|
for k, v in repo.ui.configitems('committemplate'):
|
|
2442
|
for k, v in repo.ui.configitems('committemplate'):
|
|
2441
|
if k != 'changeset':
|
|
2443
|
if k != 'changeset':
|
|
2442
|
t.t.cache[k] = v
|
|
2444
|
t.t.cache[k] = v
|
|
2443
|
|
|
2445
|
|
|
2444
|
if not extramsg:
|
|
2446
|
if not extramsg:
|
|
2445
|
extramsg = '' # ensure that extramsg is string
|
|
2447
|
extramsg = '' # ensure that extramsg is string
|
|
2446
|
|
|
2448
|
|
|
2447
|
ui.pushbuffer()
|
|
2449
|
ui.pushbuffer()
|
|
2448
|
t.show(ctx, extramsg=extramsg)
|
|
2450
|
t.show(ctx, extramsg=extramsg)
|
|
2449
|
return ui.popbuffer()
|
|
2451
|
return ui.popbuffer()
|
|
2450
|
|
|
2452
|
|
|
2451
|
def buildcommittext(repo, ctx, subs, extramsg):
|
|
2453
|
def buildcommittext(repo, ctx, subs, extramsg):
|
|
2452
|
edittext = []
|
|
2454
|
edittext = []
|
|
2453
|
modified, added, removed = ctx.modified(), ctx.added(), ctx.removed()
|
|
2455
|
modified, added, removed = ctx.modified(), ctx.added(), ctx.removed()
|
|
2454
|
if ctx.description():
|
|
2456
|
if ctx.description():
|
|
2455
|
edittext.append(ctx.description())
|
|
2457
|
edittext.append(ctx.description())
|
|
2456
|
edittext.append("")
|
|
2458
|
edittext.append("")
|
|
2457
|
edittext.append("") # Empty line between message and comments.
|
|
2459
|
edittext.append("") # Empty line between message and comments.
|
|
2458
|
edittext.append(_("HG: Enter commit message."
|
|
2460
|
edittext.append(_("HG: Enter commit message."
|
|
2459
|
" Lines beginning with 'HG:' are removed."))
|
|
2461
|
" Lines beginning with 'HG:' are removed."))
|
|
2460
|
edittext.append("HG: %s" % extramsg)
|
|
2462
|
edittext.append("HG: %s" % extramsg)
|
|
2461
|
edittext.append("HG: --")
|
|
2463
|
edittext.append("HG: --")
|
|
2462
|
edittext.append(_("HG: user: %s") % ctx.user())
|
|
2464
|
edittext.append(_("HG: user: %s") % ctx.user())
|
|
2463
|
if ctx.p2():
|
|
2465
|
if ctx.p2():
|
|
2464
|
edittext.append(_("HG: branch merge"))
|
|
2466
|
edittext.append(_("HG: branch merge"))
|
|
2465
|
if ctx.branch():
|
|
2467
|
if ctx.branch():
|
|
2466
|
edittext.append(_("HG: branch '%s'") % ctx.branch())
|
|
2468
|
edittext.append(_("HG: branch '%s'") % ctx.branch())
|
|
2467
|
if bookmarks.iscurrent(repo):
|
|
2469
|
if bookmarks.iscurrent(repo):
|
|
2468
|
edittext.append(_("HG: bookmark '%s'") % repo._bookmarkcurrent)
|
|
2470
|
edittext.append(_("HG: bookmark '%s'") % repo._bookmarkcurrent)
|
|
2469
|
edittext.extend([_("HG: subrepo %s") % s for s in subs])
|
|
2471
|
edittext.extend([_("HG: subrepo %s") % s for s in subs])
|
|
2470
|
edittext.extend([_("HG: added %s") % f for f in added])
|
|
2472
|
edittext.extend([_("HG: added %s") % f for f in added])
|
|
2471
|
edittext.extend([_("HG: changed %s") % f for f in modified])
|
|
2473
|
edittext.extend([_("HG: changed %s") % f for f in modified])
|
|
2472
|
edittext.extend([_("HG: removed %s") % f for f in removed])
|
|
2474
|
edittext.extend([_("HG: removed %s") % f for f in removed])
|
|
2473
|
if not added and not modified and not removed:
|
|
2475
|
if not added and not modified and not removed:
|
|
2474
|
edittext.append(_("HG: no files changed"))
|
|
2476
|
edittext.append(_("HG: no files changed"))
|
|
2475
|
edittext.append("")
|
|
2477
|
edittext.append("")
|
|
2476
|
|
|
2478
|
|
|
2477
|
return "\n".join(edittext)
|
|
2479
|
return "\n".join(edittext)
|
|
2478
|
|
|
2480
|
|
|
2479
|
def commitstatus(repo, node, branch, bheads=None, opts={}):
|
|
2481
|
def commitstatus(repo, node, branch, bheads=None, opts={}):
|
|
2480
|
ctx = repo[node]
|
|
2482
|
ctx = repo[node]
|
|
2481
|
parents = ctx.parents()
|
|
2483
|
parents = ctx.parents()
|
|
2482
|
|
|
2484
|
|
|
2483
|
if (not opts.get('amend') and bheads and node not in bheads and not
|
|
2485
|
if (not opts.get('amend') and bheads and node not in bheads and not
|
|
2484
|
[x for x in parents if x.node() in bheads and x.branch() == branch]):
|
|
2486
|
[x for x in parents if x.node() in bheads and x.branch() == branch]):
|
|
2485
|
repo.ui.status(_('created new head\n'))
|
|
2487
|
repo.ui.status(_('created new head\n'))
|
|
2486
|
# The message is not printed for initial roots. For the other
|
|
2488
|
# The message is not printed for initial roots. For the other
|
|
2487
|
# changesets, it is printed in the following situations:
|
|
2489
|
# changesets, it is printed in the following situations:
|
|
2488
|
#
|
|
2490
|
#
|
|
2489
|
# Par column: for the 2 parents with ...
|
|
2491
|
# Par column: for the 2 parents with ...
|
|
2490
|
# N: null or no parent
|
|
2492
|
# N: null or no parent
|
|
2491
|
# B: parent is on another named branch
|
|
2493
|
# B: parent is on another named branch
|
|
2492
|
# C: parent is a regular non head changeset
|
|
2494
|
# C: parent is a regular non head changeset
|
|
2493
|
# H: parent was a branch head of the current branch
|
|
2495
|
# H: parent was a branch head of the current branch
|
|
2494
|
# Msg column: whether we print "created new head" message
|
|
2496
|
# Msg column: whether we print "created new head" message
|
|
2495
|
# In the following, it is assumed that there already exists some
|
|
2497
|
# In the following, it is assumed that there already exists some
|
|
2496
|
# initial branch heads of the current branch, otherwise nothing is
|
|
2498
|
# initial branch heads of the current branch, otherwise nothing is
|
|
2497
|
# printed anyway.
|
|
2499
|
# printed anyway.
|
|
2498
|
#
|
|
2500
|
#
|
|
2499
|
# Par Msg Comment
|
|
2501
|
# Par Msg Comment
|
|
2500
|
# N N y additional topo root
|
|
2502
|
# N N y additional topo root
|
|
2501
|
#
|
|
2503
|
#
|
|
2502
|
# B N y additional branch root
|
|
2504
|
# B N y additional branch root
|
|
2503
|
# C N y additional topo head
|
|
2505
|
# C N y additional topo head
|
|
2504
|
# H N n usual case
|
|
2506
|
# H N n usual case
|
|
2505
|
#
|
|
2507
|
#
|
|
2506
|
# B B y weird additional branch root
|
|
2508
|
# B B y weird additional branch root
|
|
2507
|
# C B y branch merge
|
|
2509
|
# C B y branch merge
|
|
2508
|
# H B n merge with named branch
|
|
2510
|
# H B n merge with named branch
|
|
2509
|
#
|
|
2511
|
#
|
|
2510
|
# C C y additional head from merge
|
|
2512
|
# C C y additional head from merge
|
|
2511
|
# C H n merge with a head
|
|
2513
|
# C H n merge with a head
|
|
2512
|
#
|
|
2514
|
#
|
|
2513
|
# H H n head merge: head count decreases
|
|
2515
|
# H H n head merge: head count decreases
|
|
2514
|
|
|
2516
|
|
|
2515
|
if not opts.get('close_branch'):
|
|
2517
|
if not opts.get('close_branch'):
|
|
2516
|
for r in parents:
|
|
2518
|
for r in parents:
|
|
2517
|
if r.closesbranch() and r.branch() == branch:
|
|
2519
|
if r.closesbranch() and r.branch() == branch:
|
|
2518
|
repo.ui.status(_('reopening closed branch head %d\n') % r)
|
|
2520
|
repo.ui.status(_('reopening closed branch head %d\n') % r)
|
|
2519
|
|
|
2521
|
|
|
2520
|
if repo.ui.debugflag:
|
|
2522
|
if repo.ui.debugflag:
|
|
2521
|
repo.ui.write(_('committed changeset %d:%s\n') % (int(ctx), ctx.hex()))
|
|
2523
|
repo.ui.write(_('committed changeset %d:%s\n') % (int(ctx), ctx.hex()))
|
|
2522
|
elif repo.ui.verbose:
|
|
2524
|
elif repo.ui.verbose:
|
|
2523
|
repo.ui.write(_('committed changeset %d:%s\n') % (int(ctx), ctx))
|
|
2525
|
repo.ui.write(_('committed changeset %d:%s\n') % (int(ctx), ctx))
|
|
2524
|
|
|
2526
|
|
|
2525
|
def revert(ui, repo, ctx, parents, *pats, **opts):
|
|
2527
|
def revert(ui, repo, ctx, parents, *pats, **opts):
|
|
2526
|
parent, p2 = parents
|
|
2528
|
parent, p2 = parents
|
|
2527
|
node = ctx.node()
|
|
2529
|
node = ctx.node()
|
|
2528
|
|
|
2530
|
|
|
2529
|
mf = ctx.manifest()
|
|
2531
|
mf = ctx.manifest()
|
|
2530
|
if node == p2:
|
|
2532
|
if node == p2:
|
|
2531
|
parent = p2
|
|
2533
|
parent = p2
|
|
2532
|
if node == parent:
|
|
2534
|
if node == parent:
|
|
2533
|
pmf = mf
|
|
2535
|
pmf = mf
|
|
2534
|
else:
|
|
2536
|
else:
|
|
2535
|
pmf = None
|
|
2537
|
pmf = None
|
|
2536
|
|
|
2538
|
|
|
2537
|
# need all matching names in dirstate and manifest of target rev,
|
|
2539
|
# need all matching names in dirstate and manifest of target rev,
|
|
2538
|
# so have to walk both. do not print errors if files exist in one
|
|
2540
|
# so have to walk both. do not print errors if files exist in one
|
|
2539
|
# but not other.
|
|
2541
|
# but not other.
|
|
2540
|
|
|
2542
|
|
|
2541
|
# `names` is a mapping for all elements in working copy and target revision
|
|
2543
|
# `names` is a mapping for all elements in working copy and target revision
|
|
2542
|
# The mapping is in the form:
|
|
2544
|
# The mapping is in the form:
|
|
2543
|
# <asb path in repo> -> (<path from CWD>, <exactly specified by matcher?>)
|
|
2545
|
# <asb path in repo> -> (<path from CWD>, <exactly specified by matcher?>)
|
|
2544
|
names = {}
|
|
2546
|
names = {}
|
|
2545
|
|
|
2547
|
|
|
2546
|
wlock = repo.wlock()
|
|
2548
|
wlock = repo.wlock()
|
|
2547
|
try:
|
|
2549
|
try:
|
|
2548
|
## filling of the `names` mapping
|
|
2550
|
## filling of the `names` mapping
|
|
2549
|
# walk dirstate to fill `names`
|
|
2551
|
# walk dirstate to fill `names`
|
|
2550
|
|
|
2552
|
|
|
2551
|
m = scmutil.match(repo[None], pats, opts)
|
|
2553
|
m = scmutil.match(repo[None], pats, opts)
|
|
2552
|
if not m.always() or node != parent:
|
|
2554
|
if not m.always() or node != parent:
|
|
2553
|
m.bad = lambda x, y: False
|
|
2555
|
m.bad = lambda x, y: False
|
|
2554
|
for abs in repo.walk(m):
|
|
2556
|
for abs in repo.walk(m):
|
|
2555
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2557
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2556
|
|
|
2558
|
|
|
2557
|
# walk target manifest to fill `names`
|
|
2559
|
# walk target manifest to fill `names`
|
|
2558
|
|
|
2560
|
|
|
2559
|
def badfn(path, msg):
|
|
2561
|
def badfn(path, msg):
|
|
2560
|
if path in names:
|
|
2562
|
if path in names:
|
|
2561
|
return
|
|
2563
|
return
|
|
2562
|
if path in ctx.substate:
|
|
2564
|
if path in ctx.substate:
|
|
2563
|
return
|
|
2565
|
return
|
|
2564
|
path_ = path + '/'
|
|
2566
|
path_ = path + '/'
|
|
2565
|
for f in names:
|
|
2567
|
for f in names:
|
|
2566
|
if f.startswith(path_):
|
|
2568
|
if f.startswith(path_):
|
|
2567
|
return
|
|
2569
|
return
|
|
2568
|
ui.warn("%s: %s\n" % (m.rel(path), msg))
|
|
2570
|
ui.warn("%s: %s\n" % (m.rel(path), msg))
|
|
2569
|
|
|
2571
|
|
|
2570
|
m = scmutil.match(ctx, pats, opts)
|
|
2572
|
m = scmutil.match(ctx, pats, opts)
|
|
2571
|
m.bad = badfn
|
|
2573
|
m.bad = badfn
|
|
2572
|
for abs in ctx.walk(m):
|
|
2574
|
for abs in ctx.walk(m):
|
|
2573
|
if abs not in names:
|
|
2575
|
if abs not in names:
|
|
2574
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2576
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2575
|
|
|
2577
|
|
|
2576
|
# Find status of all file in `names`.
|
|
2578
|
# Find status of all file in `names`.
|
|
2577
|
m = scmutil.matchfiles(repo, names)
|
|
2579
|
m = scmutil.matchfiles(repo, names)
|
|
2578
|
|
|
2580
|
|
|
2579
|
changes = repo.status(node1=node, match=m,
|
|
2581
|
changes = repo.status(node1=node, match=m,
|
|
2580
|
unknown=True, ignored=True, clean=True)
|
|
2582
|
unknown=True, ignored=True, clean=True)
|
|
2581
|
else:
|
|
2583
|
else:
|
|
2582
|
changes = repo.status(match=m)
|
|
2584
|
changes = repo.status(match=m)
|
|
2583
|
for kind in changes:
|
|
2585
|
for kind in changes:
|
|
2584
|
for abs in kind:
|
|
2586
|
for abs in kind:
|
|
2585
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2587
|
names[abs] = m.rel(abs), m.exact(abs)
|
|
2586
|
|
|
2588
|
|
|
2587
|
m = scmutil.matchfiles(repo, names)
|
|
2589
|
m = scmutil.matchfiles(repo, names)
|
|
2588
|
|
|
2590
|
|
|
2589
|
modified = set(changes.modified)
|
|
2591
|
modified = set(changes.modified)
|
|
2590
|
added = set(changes.added)
|
|
2592
|
added = set(changes.added)
|
|
2591
|
removed = set(changes.removed)
|
|
2593
|
removed = set(changes.removed)
|
|
2592
|
_deleted = set(changes.deleted)
|
|
2594
|
_deleted = set(changes.deleted)
|
|
2593
|
unknown = set(changes.unknown)
|
|
2595
|
unknown = set(changes.unknown)
|
|
2594
|
unknown.update(changes.ignored)
|
|
2596
|
unknown.update(changes.ignored)
|
|
2595
|
clean = set(changes.clean)
|
|
2597
|
clean = set(changes.clean)
|
|
2596
|
modadded = set()
|
|
2598
|
modadded = set()
|
|
2597
|
|
|
2599
|
|
|
2598
|
# split between files known in target manifest and the others
|
|
2600
|
# split between files known in target manifest and the others
|
|
2599
|
smf = set(mf)
|
|
2601
|
smf = set(mf)
|
|
2600
|
|
|
2602
|
|
|
2601
|
# determine the exact nature of the deleted changesets
|
|
2603
|
# determine the exact nature of the deleted changesets
|
|
2602
|
deladded = _deleted - smf
|
|
2604
|
deladded = _deleted - smf
|
|
2603
|
deleted = _deleted - deladded
|
|
2605
|
deleted = _deleted - deladded
|
|
2604
|
|
|
2606
|
|
|
2605
|
# We need to account for the state of the file in the dirstate,
|
|
2607
|
# We need to account for the state of the file in the dirstate,
|
|
2606
|
# even when we revert against something else than parent. This will
|
|
2608
|
# even when we revert against something else than parent. This will
|
|
2607
|
# slightly alter the behavior of revert (doing back up or not, delete
|
|
2609
|
# slightly alter the behavior of revert (doing back up or not, delete
|
|
2608
|
# or just forget etc).
|
|
2610
|
# or just forget etc).
|
|
2609
|
if parent == node:
|
|
2611
|
if parent == node:
|
|
2610
|
dsmodified = modified
|
|
2612
|
dsmodified = modified
|
|
2611
|
dsadded = added
|
|
2613
|
dsadded = added
|
|
2612
|
dsremoved = removed
|
|
2614
|
dsremoved = removed
|
|
2613
|
# store all local modifications, useful later for rename detection
|
|
2615
|
# store all local modifications, useful later for rename detection
|
|
2614
|
localchanges = dsmodified | dsadded
|
|
2616
|
localchanges = dsmodified | dsadded
|
|
2615
|
modified, added, removed = set(), set(), set()
|
|
2617
|
modified, added, removed = set(), set(), set()
|
|
2616
|
else:
|
|
2618
|
else:
|
|
2617
|
changes = repo.status(node1=parent, match=m)
|
|
2619
|
changes = repo.status(node1=parent, match=m)
|
|
2618
|
dsmodified = set(changes.modified)
|
|
2620
|
dsmodified = set(changes.modified)
|
|
2619
|
dsadded = set(changes.added)
|
|
2621
|
dsadded = set(changes.added)
|
|
2620
|
dsremoved = set(changes.removed)
|
|
2622
|
dsremoved = set(changes.removed)
|
|
2621
|
# store all local modifications, useful later for rename detection
|
|
2623
|
# store all local modifications, useful later for rename detection
|
|
2622
|
localchanges = dsmodified | dsadded
|
|
2624
|
localchanges = dsmodified | dsadded
|
|
2623
|
|
|
2625
|
|
|
2624
|
# only take into account for removes between wc and target
|
|
2626
|
# only take into account for removes between wc and target
|
|
2625
|
clean |= dsremoved - removed
|
|
2627
|
clean |= dsremoved - removed
|
|
2626
|
dsremoved &= removed
|
|
2628
|
dsremoved &= removed
|
|
2627
|
# distinct between dirstate remove and other
|
|
2629
|
# distinct between dirstate remove and other
|
|
2628
|
removed -= dsremoved
|
|
2630
|
removed -= dsremoved
|
|
2629
|
|
|
2631
|
|
|
2630
|
modadded = added & dsmodified
|
|
2632
|
modadded = added & dsmodified
|
|
2631
|
added -= modadded
|
|
2633
|
added -= modadded
|
|
2632
|
|
|
2634
|
|
|
2633
|
# tell newly modified apart.
|
|
2635
|
# tell newly modified apart.
|
|
2634
|
dsmodified &= modified
|
|
2636
|
dsmodified &= modified
|
|
2635
|
dsmodified |= modified & dsadded # dirstate added may needs backup
|
|
2637
|
dsmodified |= modified & dsadded # dirstate added may needs backup
|
|
2636
|
modified -= dsmodified
|
|
2638
|
modified -= dsmodified
|
|
2637
|
|
|
2639
|
|
|
2638
|
# We need to wait for some post-processing to update this set
|
|
2640
|
# We need to wait for some post-processing to update this set
|
|
2639
|
# before making the distinction. The dirstate will be used for
|
|
2641
|
# before making the distinction. The dirstate will be used for
|
|
2640
|
# that purpose.
|
|
2642
|
# that purpose.
|
|
2641
|
dsadded = added
|
|
2643
|
dsadded = added
|
|
2642
|
|
|
2644
|
|
|
2643
|
# in case of merge, files that are actually added can be reported as
|
|
2645
|
# in case of merge, files that are actually added can be reported as
|
|
2644
|
# modified, we need to post process the result
|
|
2646
|
# modified, we need to post process the result
|
|
2645
|
if p2 != nullid:
|
|
2647
|
if p2 != nullid:
|
|
2646
|
if pmf is None:
|
|
2648
|
if pmf is None:
|
|
2647
|
# only need parent manifest in the merge case,
|
|
2649
|
# only need parent manifest in the merge case,
|
|
2648
|
# so do not read by default
|
|
2650
|
# so do not read by default
|
|
2649
|
pmf = repo[parent].manifest()
|
|
2651
|
pmf = repo[parent].manifest()
|
|
2650
|
mergeadd = dsmodified - set(pmf)
|
|
2652
|
mergeadd = dsmodified - set(pmf)
|
|
2651
|
dsadded |= mergeadd
|
|
2653
|
dsadded |= mergeadd
|
|
2652
|
dsmodified -= mergeadd
|
|
2654
|
dsmodified -= mergeadd
|
|
2653
|
|
|
2655
|
|
|
2654
|
# if f is a rename, update `names` to also revert the source
|
|
2656
|
# if f is a rename, update `names` to also revert the source
|
|
2655
|
cwd = repo.getcwd()
|
|
2657
|
cwd = repo.getcwd()
|
|
2656
|
for f in localchanges:
|
|
2658
|
for f in localchanges:
|
|
2657
|
src = repo.dirstate.copied(f)
|
|
2659
|
src = repo.dirstate.copied(f)
|
|
2658
|
# XXX should we check for rename down to target node?
|
|
2660
|
# XXX should we check for rename down to target node?
|
|
2659
|
if src and src not in names and repo.dirstate[src] == 'r':
|
|
2661
|
if src and src not in names and repo.dirstate[src] == 'r':
|
|
2660
|
dsremoved.add(src)
|
|
2662
|
dsremoved.add(src)
|
|
2661
|
names[src] = (repo.pathto(src, cwd), True)
|
|
2663
|
names[src] = (repo.pathto(src, cwd), True)
|
|
2662
|
|
|
2664
|
|
|
2663
|
# distinguish between file to forget and the other
|
|
2665
|
# distinguish between file to forget and the other
|
|
2664
|
added = set()
|
|
2666
|
added = set()
|
|
2665
|
for abs in dsadded:
|
|
2667
|
for abs in dsadded:
|
|
2666
|
if repo.dirstate[abs] != 'a':
|
|
2668
|
if repo.dirstate[abs] != 'a':
|
|
2667
|
added.add(abs)
|
|
2669
|
added.add(abs)
|
|
2668
|
dsadded -= added
|
|
2670
|
dsadded -= added
|
|
2669
|
|
|
2671
|
|
|
2670
|
for abs in deladded:
|
|
2672
|
for abs in deladded:
|
|
2671
|
if repo.dirstate[abs] == 'a':
|
|
2673
|
if repo.dirstate[abs] == 'a':
|
|
2672
|
dsadded.add(abs)
|
|
2674
|
dsadded.add(abs)
|
|
2673
|
deladded -= dsadded
|
|
2675
|
deladded -= dsadded
|
|
2674
|
|
|
2676
|
|
|
2675
|
# For files marked as removed, we check if an unknown file is present at
|
|
2677
|
# For files marked as removed, we check if an unknown file is present at
|
|
2676
|
# the same path. If a such file exists it may need to be backed up.
|
|
2678
|
# the same path. If a such file exists it may need to be backed up.
|
|
2677
|
# Making the distinction at this stage helps have simpler backup
|
|
2679
|
# Making the distinction at this stage helps have simpler backup
|
|
2678
|
# logic.
|
|
2680
|
# logic.
|
|
2679
|
removunk = set()
|
|
2681
|
removunk = set()
|
|
2680
|
for abs in removed:
|
|
2682
|
for abs in removed:
|
|
2681
|
target = repo.wjoin(abs)
|
|
2683
|
target = repo.wjoin(abs)
|
|
2682
|
if os.path.lexists(target):
|
|
2684
|
if os.path.lexists(target):
|
|
2683
|
removunk.add(abs)
|
|
2685
|
removunk.add(abs)
|
|
2684
|
removed -= removunk
|
|
2686
|
removed -= removunk
|
|
2685
|
|
|
2687
|
|
|
2686
|
dsremovunk = set()
|
|
2688
|
dsremovunk = set()
|
|
2687
|
for abs in dsremoved:
|
|
2689
|
for abs in dsremoved:
|
|
2688
|
target = repo.wjoin(abs)
|
|
2690
|
target = repo.wjoin(abs)
|
|
2689
|
if os.path.lexists(target):
|
|
2691
|
if os.path.lexists(target):
|
|
2690
|
dsremovunk.add(abs)
|
|
2692
|
dsremovunk.add(abs)
|
|
2691
|
dsremoved -= dsremovunk
|
|
2693
|
dsremoved -= dsremovunk
|
|
2692
|
|
|
2694
|
|
|
2693
|
# action to be actually performed by revert
|
|
2695
|
# action to be actually performed by revert
|
|
2694
|
# (<list of file>, message>) tuple
|
|
2696
|
# (<list of file>, message>) tuple
|
|
2695
|
actions = {'revert': ([], _('reverting %s\n')),
|
|
2697
|
actions = {'revert': ([], _('reverting %s\n')),
|
|
2696
|
'add': ([], _('adding %s\n')),
|
|
2698
|
'add': ([], _('adding %s\n')),
|
|
2697
|
'remove': ([], _('removing %s\n')),
|
|
2699
|
'remove': ([], _('removing %s\n')),
|
|
2698
|
'drop': ([], _('removing %s\n')),
|
|
2700
|
'drop': ([], _('removing %s\n')),
|
|
2699
|
'forget': ([], _('forgetting %s\n')),
|
|
2701
|
'forget': ([], _('forgetting %s\n')),
|
|
2700
|
'undelete': ([], _('undeleting %s\n')),
|
|
2702
|
'undelete': ([], _('undeleting %s\n')),
|
|
2701
|
'noop': (None, _('no changes needed to %s\n')),
|
|
2703
|
'noop': (None, _('no changes needed to %s\n')),
|
|
2702
|
'unknown': (None, _('file not managed: %s\n')),
|
|
2704
|
'unknown': (None, _('file not managed: %s\n')),
|
|
2703
|
}
|
|
2705
|
}
|
|
2704
|
|
|
2706
|
|
|
2705
|
# "constant" that convey the backup strategy.
|
|
2707
|
# "constant" that convey the backup strategy.
|
|
2706
|
# All set to `discard` if `no-backup` is set do avoid checking
|
|
2708
|
# All set to `discard` if `no-backup` is set do avoid checking
|
|
2707
|
# no_backup lower in the code.
|
|
2709
|
# no_backup lower in the code.
|
|
2708
|
# These values are ordered for comparison purposes
|
|
2710
|
# These values are ordered for comparison purposes
|
|
2709
|
backup = 2 # unconditionally do backup
|
|
2711
|
backup = 2 # unconditionally do backup
|
|
2710
|
check = 1 # check if the existing file differs from target
|
|
2712
|
check = 1 # check if the existing file differs from target
|
|
2711
|
discard = 0 # never do backup
|
|
2713
|
discard = 0 # never do backup
|
|
2712
|
if opts.get('no_backup'):
|
|
2714
|
if opts.get('no_backup'):
|
|
2713
|
backup = check = discard
|
|
2715
|
backup = check = discard
|
|
2714
|
|
|
2716
|
|
|
2715
|
backupanddel = actions['remove']
|
|
2717
|
backupanddel = actions['remove']
|
|
2716
|
if not opts.get('no_backup'):
|
|
2718
|
if not opts.get('no_backup'):
|
|
2717
|
backupanddel = actions['drop']
|
|
2719
|
backupanddel = actions['drop']
|
|
2718
|
|
|
2720
|
|
|
2719
|
disptable = (
|
|
2721
|
disptable = (
|
|
2720
|
# dispatch table:
|
|
2722
|
# dispatch table:
|
|
2721
|
# file state
|
|
2723
|
# file state
|
|
2722
|
# action
|
|
2724
|
# action
|
|
2723
|
# make backup
|
|
2725
|
# make backup
|
|
2724
|
|
|
2726
|
|
|
2725
|
## Sets that results that will change file on disk
|
|
2727
|
## Sets that results that will change file on disk
|
|
2726
|
# Modified compared to target, no local change
|
|
2728
|
# Modified compared to target, no local change
|
|
2727
|
(modified, actions['revert'], discard),
|
|
2729
|
(modified, actions['revert'], discard),
|
|
2728
|
# Modified compared to target, but local file is deleted
|
|
2730
|
# Modified compared to target, but local file is deleted
|
|
2729
|
(deleted, actions['revert'], discard),
|
|
2731
|
(deleted, actions['revert'], discard),
|
|
2730
|
# Modified compared to target, local change
|
|
2732
|
# Modified compared to target, local change
|
|
2731
|
(dsmodified, actions['revert'], backup),
|
|
2733
|
(dsmodified, actions['revert'], backup),
|
|
2732
|
# Added since target
|
|
2734
|
# Added since target
|
|
2733
|
(added, actions['remove'], discard),
|
|
2735
|
(added, actions['remove'], discard),
|
|
2734
|
# Added in working directory
|
|
2736
|
# Added in working directory
|
|
2735
|
(dsadded, actions['forget'], discard),
|
|
2737
|
(dsadded, actions['forget'], discard),
|
|
2736
|
# Added since target, have local modification
|
|
2738
|
# Added since target, have local modification
|
|
2737
|
(modadded, backupanddel, backup),
|
|
2739
|
(modadded, backupanddel, backup),
|
|
2738
|
# Added since target but file is missing in working directory
|
|
2740
|
# Added since target but file is missing in working directory
|
|
2739
|
(deladded, actions['drop'], discard),
|
|
2741
|
(deladded, actions['drop'], discard),
|
|
2740
|
# Removed since target, before working copy parent
|
|
2742
|
# Removed since target, before working copy parent
|
|
2741
|
(removed, actions['add'], discard),
|
|
2743
|
(removed, actions['add'], discard),
|
|
2742
|
# Same as `removed` but an unknown file exists at the same path
|
|
2744
|
# Same as `removed` but an unknown file exists at the same path
|
|
2743
|
(removunk, actions['add'], check),
|
|
2745
|
(removunk, actions['add'], check),
|
|
2744
|
# Removed since targe, marked as such in working copy parent
|
|
2746
|
# Removed since targe, marked as such in working copy parent
|
|
2745
|
(dsremoved, actions['undelete'], discard),
|
|
2747
|
(dsremoved, actions['undelete'], discard),
|
|
2746
|
# Same as `dsremoved` but an unknown file exists at the same path
|
|
2748
|
# Same as `dsremoved` but an unknown file exists at the same path
|
|
2747
|
(dsremovunk, actions['undelete'], check),
|
|
2749
|
(dsremovunk, actions['undelete'], check),
|
|
2748
|
## the following sets does not result in any file changes
|
|
2750
|
## the following sets does not result in any file changes
|
|
2749
|
# File with no modification
|
|
2751
|
# File with no modification
|
|
2750
|
(clean, actions['noop'], discard),
|
|
2752
|
(clean, actions['noop'], discard),
|
|
2751
|
# Existing file, not tracked anywhere
|
|
2753
|
# Existing file, not tracked anywhere
|
|
2752
|
(unknown, actions['unknown'], discard),
|
|
2754
|
(unknown, actions['unknown'], discard),
|
|
2753
|
)
|
|
2755
|
)
|
|
2754
|
|
|
2756
|
|
|
2755
|
wctx = repo[None]
|
|
2757
|
wctx = repo[None]
|
|
2756
|
for abs, (rel, exact) in sorted(names.items()):
|
|
2758
|
for abs, (rel, exact) in sorted(names.items()):
|
|
2757
|
# target file to be touch on disk (relative to cwd)
|
|
2759
|
# target file to be touch on disk (relative to cwd)
|
|
2758
|
target = repo.wjoin(abs)
|
|
2760
|
target = repo.wjoin(abs)
|
|
2759
|
# search the entry in the dispatch table.
|
|
2761
|
# search the entry in the dispatch table.
|
|
2760
|
# if the file is in any of these sets, it was touched in the working
|
|
2762
|
# if the file is in any of these sets, it was touched in the working
|
|
2761
|
# directory parent and we are sure it needs to be reverted.
|
|
2763
|
# directory parent and we are sure it needs to be reverted.
|
|
2762
|
for table, (xlist, msg), dobackup in disptable:
|
|
2764
|
for table, (xlist, msg), dobackup in disptable:
|
|
2763
|
if abs not in table:
|
|
2765
|
if abs not in table:
|
|
2764
|
continue
|
|
2766
|
continue
|
|
2765
|
if xlist is not None:
|
|
2767
|
if xlist is not None:
|
|
2766
|
xlist.append(abs)
|
|
2768
|
xlist.append(abs)
|
|
2767
|
if dobackup and (backup <= dobackup
|
|
2769
|
if dobackup and (backup <= dobackup
|
|
2768
|
or wctx[abs].cmp(ctx[abs])):
|
|
2770
|
or wctx[abs].cmp(ctx[abs])):
|
|
2769
|
bakname = "%s.orig" % rel
|
|
2771
|
bakname = "%s.orig" % rel
|
|
2770
|
ui.note(_('saving current version of %s as %s\n') %
|
|
2772
|
ui.note(_('saving current version of %s as %s\n') %
|
|
2771
|
(rel, bakname))
|
|
2773
|
(rel, bakname))
|
|
2772
|
if not opts.get('dry_run'):
|
|
2774
|
if not opts.get('dry_run'):
|
|
2773
|
util.rename(target, bakname)
|
|
2775
|
util.rename(target, bakname)
|
|
2774
|
if ui.verbose or not exact:
|
|
2776
|
if ui.verbose or not exact:
|
|
2775
|
if not isinstance(msg, basestring):
|
|
2777
|
if not isinstance(msg, basestring):
|
|
2776
|
msg = msg(abs)
|
|
2778
|
msg = msg(abs)
|
|
2777
|
ui.status(msg % rel)
|
|
2779
|
ui.status(msg % rel)
|
|
2778
|
elif exact:
|
|
2780
|
elif exact:
|
|
2779
|
ui.warn(msg % rel)
|
|
2781
|
ui.warn(msg % rel)
|
|
2780
|
break
|
|
2782
|
break
|
|
2781
|
|
|
2783
|
|
|
2782
|
|
|
2784
|
|
|
2783
|
if not opts.get('dry_run'):
|
|
2785
|
if not opts.get('dry_run'):
|
|
2784
|
needdata = ('revert', 'add', 'undelete')
|
|
2786
|
needdata = ('revert', 'add', 'undelete')
|
|
2785
|
_revertprefetch(repo, ctx, *[actions[name][0] for name in needdata])
|
|
2787
|
_revertprefetch(repo, ctx, *[actions[name][0] for name in needdata])
|
|
2786
|
|
|
2788
|
|
|
2787
|
_performrevert(repo, parents, ctx, actions)
|
|
2789
|
_performrevert(repo, parents, ctx, actions)
|
|
2788
|
|
|
2790
|
|
|
2789
|
# get the list of subrepos that must be reverted
|
|
2791
|
# get the list of subrepos that must be reverted
|
|
2790
|
subrepomatch = scmutil.match(ctx, pats, opts)
|
|
2792
|
subrepomatch = scmutil.match(ctx, pats, opts)
|
|
2791
|
targetsubs = sorted(s for s in ctx.substate if subrepomatch(s))
|
|
2793
|
targetsubs = sorted(s for s in ctx.substate if subrepomatch(s))
|
|
2792
|
|
|
2794
|
|
|
2793
|
if targetsubs:
|
|
2795
|
if targetsubs:
|
|
2794
|
# Revert the subrepos on the revert list
|
|
2796
|
# Revert the subrepos on the revert list
|
|
2795
|
for sub in targetsubs:
|
|
2797
|
for sub in targetsubs:
|
|
2796
|
ctx.sub(sub).revert(ctx.substate[sub], *pats, **opts)
|
|
2798
|
ctx.sub(sub).revert(ctx.substate[sub], *pats, **opts)
|
|
2797
|
finally:
|
|
2799
|
finally:
|
|
2798
|
wlock.release()
|
|
2800
|
wlock.release()
|
|
2799
|
|
|
2801
|
|
|
2800
|
def _revertprefetch(repo, ctx, *files):
|
|
2802
|
def _revertprefetch(repo, ctx, *files):
|
|
2801
|
"""Let extension changing the storage layer prefetch content"""
|
|
2803
|
"""Let extension changing the storage layer prefetch content"""
|
|
2802
|
pass
|
|
2804
|
pass
|
|
2803
|
|
|
2805
|
|
|
2804
|
def _performrevert(repo, parents, ctx, actions):
|
|
2806
|
def _performrevert(repo, parents, ctx, actions):
|
|
2805
|
"""function that actually perform all the actions computed for revert
|
|
2807
|
"""function that actually perform all the actions computed for revert
|
|
2806
|
|
|
2808
|
|
|
2807
|
This is an independent function to let extension to plug in and react to
|
|
2809
|
This is an independent function to let extension to plug in and react to
|
|
2808
|
the imminent revert.
|
|
2810
|
the imminent revert.
|
|
2809
|
|
|
2811
|
|
|
2810
|
Make sure you have the working directory locked when calling this function.
|
|
2812
|
Make sure you have the working directory locked when calling this function.
|
|
2811
|
"""
|
|
2813
|
"""
|
|
2812
|
parent, p2 = parents
|
|
2814
|
parent, p2 = parents
|
|
2813
|
node = ctx.node()
|
|
2815
|
node = ctx.node()
|
|
2814
|
def checkout(f):
|
|
2816
|
def checkout(f):
|
|
2815
|
fc = ctx[f]
|
|
2817
|
fc = ctx[f]
|
|
2816
|
repo.wwrite(f, fc.data(), fc.flags())
|
|
2818
|
repo.wwrite(f, fc.data(), fc.flags())
|
|
2817
|
|
|
2819
|
|
|
2818
|
audit_path = pathutil.pathauditor(repo.root)
|
|
2820
|
audit_path = pathutil.pathauditor(repo.root)
|
|
2819
|
for f in actions['forget'][0]:
|
|
2821
|
for f in actions['forget'][0]:
|
|
2820
|
repo.dirstate.drop(f)
|
|
2822
|
repo.dirstate.drop(f)
|
|
2821
|
for f in actions['remove'][0]:
|
|
2823
|
for f in actions['remove'][0]:
|
|
2822
|
audit_path(f)
|
|
2824
|
audit_path(f)
|
|
2823
|
util.unlinkpath(repo.wjoin(f))
|
|
2825
|
util.unlinkpath(repo.wjoin(f))
|
|
2824
|
repo.dirstate.remove(f)
|
|
2826
|
repo.dirstate.remove(f)
|
|
2825
|
for f in actions['drop'][0]:
|
|
2827
|
for f in actions['drop'][0]:
|
|
2826
|
audit_path(f)
|
|
2828
|
audit_path(f)
|
|
2827
|
repo.dirstate.remove(f)
|
|
2829
|
repo.dirstate.remove(f)
|
|
2828
|
|
|
2830
|
|
|
2829
|
normal = None
|
|
2831
|
normal = None
|
|
2830
|
if node == parent:
|
|
2832
|
if node == parent:
|
|
2831
|
# We're reverting to our parent. If possible, we'd like status
|
|
2833
|
# We're reverting to our parent. If possible, we'd like status
|
|
2832
|
# to report the file as clean. We have to use normallookup for
|
|
2834
|
# to report the file as clean. We have to use normallookup for
|
|
2833
|
# merges to avoid losing information about merged/dirty files.
|
|
2835
|
# merges to avoid losing information about merged/dirty files.
|
|
2834
|
if p2 != nullid:
|
|
2836
|
if p2 != nullid:
|
|
2835
|
normal = repo.dirstate.normallookup
|
|
2837
|
normal = repo.dirstate.normallookup
|
|
2836
|
else:
|
|
2838
|
else:
|
|
2837
|
normal = repo.dirstate.normal
|
|
2839
|
normal = repo.dirstate.normal
|
|
2838
|
for f in actions['revert'][0]:
|
|
2840
|
for f in actions['revert'][0]:
|
|
2839
|
checkout(f)
|
|
2841
|
checkout(f)
|
|
2840
|
if normal:
|
|
2842
|
if normal:
|
|
2841
|
normal(f)
|
|
2843
|
normal(f)
|
|
2842
|
|
|
2844
|
|
|
2843
|
for f in actions['add'][0]:
|
|
2845
|
for f in actions['add'][0]:
|
|
2844
|
checkout(f)
|
|
2846
|
checkout(f)
|
|
2845
|
repo.dirstate.add(f)
|
|
2847
|
repo.dirstate.add(f)
|
|
2846
|
|
|
2848
|
|
|
2847
|
normal = repo.dirstate.normallookup
|
|
2849
|
normal = repo.dirstate.normallookup
|
|
2848
|
if node == parent and p2 == nullid:
|
|
2850
|
if node == parent and p2 == nullid:
|
|
2849
|
normal = repo.dirstate.normal
|
|
2851
|
normal = repo.dirstate.normal
|
|
2850
|
for f in actions['undelete'][0]:
|
|
2852
|
for f in actions['undelete'][0]:
|
|
2851
|
checkout(f)
|
|
2853
|
checkout(f)
|
|
2852
|
normal(f)
|
|
2854
|
normal(f)
|
|
2853
|
|
|
2855
|
|
|
2854
|
copied = copies.pathcopies(repo[parent], ctx)
|
|
2856
|
copied = copies.pathcopies(repo[parent], ctx)
|
|
2855
|
|
|
2857
|
|
|
2856
|
for f in actions['add'][0] + actions['undelete'][0] + actions['revert'][0]:
|
|
2858
|
for f in actions['add'][0] + actions['undelete'][0] + actions['revert'][0]:
|
|
2857
|
if f in copied:
|
|
2859
|
if f in copied:
|
|
2858
|
repo.dirstate.copy(copied[f], f)
|
|
2860
|
repo.dirstate.copy(copied[f], f)
|
|
2859
|
|
|
2861
|
|
|
2860
|
def command(table):
|
|
2862
|
def command(table):
|
|
2861
|
"""Returns a function object to be used as a decorator for making commands.
|
|
2863
|
"""Returns a function object to be used as a decorator for making commands.
|
|
2862
|
|
|
2864
|
|
|
2863
|
This function receives a command table as its argument. The table should
|
|
2865
|
This function receives a command table as its argument. The table should
|
|
2864
|
be a dict.
|
|
2866
|
be a dict.
|
|
2865
|
|
|
2867
|
|
|
2866
|
The returned function can be used as a decorator for adding commands
|
|
2868
|
The returned function can be used as a decorator for adding commands
|
|
2867
|
to that command table. This function accepts multiple arguments to define
|
|
2869
|
to that command table. This function accepts multiple arguments to define
|
|
2868
|
a command.
|
|
2870
|
a command.
|
|
2869
|
|
|
2871
|
|
|
2870
|
The first argument is the command name.
|
|
2872
|
The first argument is the command name.
|
|
2871
|
|
|
2873
|
|
|
2872
|
The options argument is an iterable of tuples defining command arguments.
|
|
2874
|
The options argument is an iterable of tuples defining command arguments.
|
|
2873
|
See ``mercurial.fancyopts.fancyopts()`` for the format of each tuple.
|
|
2875
|
See ``mercurial.fancyopts.fancyopts()`` for the format of each tuple.
|
|
2874
|
|
|
2876
|
|
|
2875
|
The synopsis argument defines a short, one line summary of how to use the
|
|
2877
|
The synopsis argument defines a short, one line summary of how to use the
|
|
2876
|
command. This shows up in the help output.
|
|
2878
|
command. This shows up in the help output.
|
|
2877
|
|
|
2879
|
|
|
2878
|
The norepo argument defines whether the command does not require a
|
|
2880
|
The norepo argument defines whether the command does not require a
|
|
2879
|
local repository. Most commands operate against a repository, thus the
|
|
2881
|
local repository. Most commands operate against a repository, thus the
|
|
2880
|
default is False.
|
|
2882
|
default is False.
|
|
2881
|
|
|
2883
|
|
|
2882
|
The optionalrepo argument defines whether the command optionally requires
|
|
2884
|
The optionalrepo argument defines whether the command optionally requires
|
|
2883
|
a local repository.
|
|
2885
|
a local repository.
|
|
2884
|
|
|
2886
|
|
|
2885
|
The inferrepo argument defines whether to try to find a repository from the
|
|
2887
|
The inferrepo argument defines whether to try to find a repository from the
|
|
2886
|
command line arguments. If True, arguments will be examined for potential
|
|
2888
|
command line arguments. If True, arguments will be examined for potential
|
|
2887
|
repository locations. See ``findrepo()``. If a repository is found, it
|
|
2889
|
repository locations. See ``findrepo()``. If a repository is found, it
|
|
2888
|
will be used.
|
|
2890
|
will be used.
|
|
2889
|
"""
|
|
2891
|
"""
|
|
2890
|
def cmd(name, options=(), synopsis=None, norepo=False, optionalrepo=False,
|
|
2892
|
def cmd(name, options=(), synopsis=None, norepo=False, optionalrepo=False,
|
|
2891
|
inferrepo=False):
|
|
2893
|
inferrepo=False):
|
|
2892
|
def decorator(func):
|
|
2894
|
def decorator(func):
|
|
2893
|
if synopsis:
|
|
2895
|
if synopsis:
|
|
2894
|
table[name] = func, list(options), synopsis
|
|
2896
|
table[name] = func, list(options), synopsis
|
|
2895
|
else:
|
|
2897
|
else:
|
|
2896
|
table[name] = func, list(options)
|
|
2898
|
table[name] = func, list(options)
|
|
2897
|
|
|
2899
|
|
|
2898
|
if norepo:
|
|
2900
|
if norepo:
|
|
2899
|
# Avoid import cycle.
|
|
2901
|
# Avoid import cycle.
|
|
2900
|
import commands
|
|
2902
|
import commands
|
|
2901
|
commands.norepo += ' %s' % ' '.join(parsealiases(name))
|
|
2903
|
commands.norepo += ' %s' % ' '.join(parsealiases(name))
|
|
2902
|
|
|
2904
|
|
|
2903
|
if optionalrepo:
|
|
2905
|
if optionalrepo:
|
|
2904
|
import commands
|
|
2906
|
import commands
|
|
2905
|
commands.optionalrepo += ' %s' % ' '.join(parsealiases(name))
|
|
2907
|
commands.optionalrepo += ' %s' % ' '.join(parsealiases(name))
|
|
2906
|
|
|
2908
|
|
|
2907
|
if inferrepo:
|
|
2909
|
if inferrepo:
|
|
2908
|
import commands
|
|
2910
|
import commands
|
|
2909
|
commands.inferrepo += ' %s' % ' '.join(parsealiases(name))
|
|
2911
|
commands.inferrepo += ' %s' % ' '.join(parsealiases(name))
|
|
2910
|
|
|
2912
|
|
|
2911
|
return func
|
|
2913
|
return func
|
|
2912
|
return decorator
|
|
2914
|
return decorator
|
|
2913
|
|
|
2915
|
|
|
2914
|
return cmd
|
|
2916
|
return cmd
|
|
2915
|
|
|
2917
|
|
|
2916
|
# a list of (ui, repo, otherpeer, opts, missing) functions called by
|
|
2918
|
# a list of (ui, repo, otherpeer, opts, missing) functions called by
|
|
2917
|
# commands.outgoing. "missing" is "missing" of the result of
|
|
2919
|
# commands.outgoing. "missing" is "missing" of the result of
|
|
2918
|
# "findcommonoutgoing()"
|
|
2920
|
# "findcommonoutgoing()"
|
|
2919
|
outgoinghooks = util.hooks()
|
|
2921
|
outgoinghooks = util.hooks()
|
|
2920
|
|
|
2922
|
|
|
2921
|
# a list of (ui, repo) functions called by commands.summary
|
|
2923
|
# a list of (ui, repo) functions called by commands.summary
|
|
2922
|
summaryhooks = util.hooks()
|
|
2924
|
summaryhooks = util.hooks()
|
|
2923
|
|
|
2925
|
|
|
2924
|
# a list of (ui, repo, opts, changes) functions called by commands.summary.
|
|
2926
|
# a list of (ui, repo, opts, changes) functions called by commands.summary.
|
|
2925
|
#
|
|
2927
|
#
|
|
2926
|
# functions should return tuple of booleans below, if 'changes' is None:
|
|
2928
|
# functions should return tuple of booleans below, if 'changes' is None:
|
|
2927
|
# (whether-incomings-are-needed, whether-outgoings-are-needed)
|
|
2929
|
# (whether-incomings-are-needed, whether-outgoings-are-needed)
|
|
2928
|
#
|
|
2930
|
#
|
|
2929
|
# otherwise, 'changes' is a tuple of tuples below:
|
|
2931
|
# otherwise, 'changes' is a tuple of tuples below:
|
|
2930
|
# - (sourceurl, sourcebranch, sourcepeer, incoming)
|
|
2932
|
# - (sourceurl, sourcebranch, sourcepeer, incoming)
|
|
2931
|
# - (desturl, destbranch, destpeer, outgoing)
|
|
2933
|
# - (desturl, destbranch, destpeer, outgoing)
|
|
2932
|
summaryremotehooks = util.hooks()
|
|
2934
|
summaryremotehooks = util.hooks()
|
|
2933
|
|
|
2935
|
|
|
2934
|
# A list of state files kept by multistep operations like graft.
|
|
2936
|
# A list of state files kept by multistep operations like graft.
|
|
2935
|
# Since graft cannot be aborted, it is considered 'clearable' by update.
|
|
2937
|
# Since graft cannot be aborted, it is considered 'clearable' by update.
|
|
2936
|
# note: bisect is intentionally excluded
|
|
2938
|
# note: bisect is intentionally excluded
|
|
2937
|
# (state file, clearable, allowcommit, error, hint)
|
|
2939
|
# (state file, clearable, allowcommit, error, hint)
|
|
2938
|
unfinishedstates = [
|
|
2940
|
unfinishedstates = [
|
|
2939
|
('graftstate', True, False, _('graft in progress'),
|
|
2941
|
('graftstate', True, False, _('graft in progress'),
|
|
2940
|
_("use 'hg graft --continue' or 'hg update' to abort")),
|
|
2942
|
_("use 'hg graft --continue' or 'hg update' to abort")),
|
|
2941
|
('updatestate', True, False, _('last update was interrupted'),
|
|
2943
|
('updatestate', True, False, _('last update was interrupted'),
|
|
2942
|
_("use 'hg update' to get a consistent checkout"))
|
|
2944
|
_("use 'hg update' to get a consistent checkout"))
|
|
2943
|
]
|
|
2945
|
]
|
|
2944
|
|
|
2946
|
|
|
2945
|
def checkunfinished(repo, commit=False):
|
|
2947
|
def checkunfinished(repo, commit=False):
|
|
2946
|
'''Look for an unfinished multistep operation, like graft, and abort
|
|
2948
|
'''Look for an unfinished multistep operation, like graft, and abort
|
|
2947
|
if found. It's probably good to check this right before
|
|
2949
|
if found. It's probably good to check this right before
|
|
2948
|
bailifchanged().
|
|
2950
|
bailifchanged().
|
|
2949
|
'''
|
|
2951
|
'''
|
|
2950
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2952
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2951
|
if commit and allowcommit:
|
|
2953
|
if commit and allowcommit:
|
|
2952
|
continue
|
|
2954
|
continue
|
|
2953
|
if repo.vfs.exists(f):
|
|
2955
|
if repo.vfs.exists(f):
|
|
2954
|
raise util.Abort(msg, hint=hint)
|
|
2956
|
raise util.Abort(msg, hint=hint)
|
|
2955
|
|
|
2957
|
|
|
2956
|
def clearunfinished(repo):
|
|
2958
|
def clearunfinished(repo):
|
|
2957
|
'''Check for unfinished operations (as above), and clear the ones
|
|
2959
|
'''Check for unfinished operations (as above), and clear the ones
|
|
2958
|
that are clearable.
|
|
2960
|
that are clearable.
|
|
2959
|
'''
|
|
2961
|
'''
|
|
2960
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2962
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2961
|
if not clearable and repo.vfs.exists(f):
|
|
2963
|
if not clearable and repo.vfs.exists(f):
|
|
2962
|
raise util.Abort(msg, hint=hint)
|
|
2964
|
raise util.Abort(msg, hint=hint)
|
|
2963
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2965
|
for f, clearable, allowcommit, msg, hint in unfinishedstates:
|
|
2964
|
if clearable and repo.vfs.exists(f):
|
|
2966
|
if clearable and repo.vfs.exists(f):
|
|
2965
|
util.unlink(repo.join(f))
|
|
2967
|
util.unlink(repo.join(f))
|