##// END OF EJS Templates
py3: convert keys of kwargs back to bytes using pycompat.byteskwargs()
Pulkit Goyal -
r32897:799db2af default
parent child Browse files
Show More
@@ -1,243 +1,245
1 """strip changesets and their descendants from history
1 """strip changesets and their descendants from history
2
2
3 This extension allows you to strip changesets and all their descendants from the
3 This extension allows you to strip changesets and all their descendants from the
4 repository. See the command help for details.
4 repository. See the command help for details.
5 """
5 """
6 from __future__ import absolute_import
6 from __future__ import absolute_import
7
7
8 from mercurial.i18n import _
8 from mercurial.i18n import _
9 from mercurial import (
9 from mercurial import (
10 bookmarks as bookmarksmod,
10 bookmarks as bookmarksmod,
11 cmdutil,
11 cmdutil,
12 error,
12 error,
13 hg,
13 hg,
14 lock as lockmod,
14 lock as lockmod,
15 merge,
15 merge,
16 node as nodemod,
16 node as nodemod,
17 pycompat,
17 registrar,
18 registrar,
18 repair,
19 repair,
19 scmutil,
20 scmutil,
20 util,
21 util,
21 )
22 )
22 nullid = nodemod.nullid
23 nullid = nodemod.nullid
23 release = lockmod.release
24 release = lockmod.release
24
25
25 cmdtable = {}
26 cmdtable = {}
26 command = registrar.command(cmdtable)
27 command = registrar.command(cmdtable)
27 # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for
28 # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for
28 # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should
29 # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should
29 # be specifying the version(s) of Mercurial they are tested with, or
30 # be specifying the version(s) of Mercurial they are tested with, or
30 # leave the attribute unspecified.
31 # leave the attribute unspecified.
31 testedwith = 'ships-with-hg-core'
32 testedwith = 'ships-with-hg-core'
32
33
33 def checksubstate(repo, baserev=None):
34 def checksubstate(repo, baserev=None):
34 '''return list of subrepos at a different revision than substate.
35 '''return list of subrepos at a different revision than substate.
35 Abort if any subrepos have uncommitted changes.'''
36 Abort if any subrepos have uncommitted changes.'''
36 inclsubs = []
37 inclsubs = []
37 wctx = repo[None]
38 wctx = repo[None]
38 if baserev:
39 if baserev:
39 bctx = repo[baserev]
40 bctx = repo[baserev]
40 else:
41 else:
41 bctx = wctx.parents()[0]
42 bctx = wctx.parents()[0]
42 for s in sorted(wctx.substate):
43 for s in sorted(wctx.substate):
43 wctx.sub(s).bailifchanged(True)
44 wctx.sub(s).bailifchanged(True)
44 if s not in bctx.substate or bctx.sub(s).dirty():
45 if s not in bctx.substate or bctx.sub(s).dirty():
45 inclsubs.append(s)
46 inclsubs.append(s)
46 return inclsubs
47 return inclsubs
47
48
48 def checklocalchanges(repo, force=False, excsuffix=''):
49 def checklocalchanges(repo, force=False, excsuffix=''):
49 cmdutil.checkunfinished(repo)
50 cmdutil.checkunfinished(repo)
50 s = repo.status()
51 s = repo.status()
51 if not force:
52 if not force:
52 if s.modified or s.added or s.removed or s.deleted:
53 if s.modified or s.added or s.removed or s.deleted:
53 _("local changes found") # i18n tool detection
54 _("local changes found") # i18n tool detection
54 raise error.Abort(_("local changes found" + excsuffix))
55 raise error.Abort(_("local changes found" + excsuffix))
55 if checksubstate(repo):
56 if checksubstate(repo):
56 _("local changed subrepos found") # i18n tool detection
57 _("local changed subrepos found") # i18n tool detection
57 raise error.Abort(_("local changed subrepos found" + excsuffix))
58 raise error.Abort(_("local changed subrepos found" + excsuffix))
58 return s
59 return s
59
60
60 def strip(ui, repo, revs, update=True, backup=True, force=None, bookmarks=None):
61 def strip(ui, repo, revs, update=True, backup=True, force=None, bookmarks=None):
61 wlock = lock = None
62 wlock = lock = None
62 try:
63 try:
63 wlock = repo.wlock()
64 wlock = repo.wlock()
64 lock = repo.lock()
65 lock = repo.lock()
65
66
66 if update:
67 if update:
67 checklocalchanges(repo, force=force)
68 checklocalchanges(repo, force=force)
68 urev, p2 = repo.changelog.parents(revs[0])
69 urev, p2 = repo.changelog.parents(revs[0])
69 if (util.safehasattr(repo, 'mq') and
70 if (util.safehasattr(repo, 'mq') and
70 p2 != nullid
71 p2 != nullid
71 and p2 in [x.node for x in repo.mq.applied]):
72 and p2 in [x.node for x in repo.mq.applied]):
72 urev = p2
73 urev = p2
73 hg.clean(repo, urev)
74 hg.clean(repo, urev)
74 repo.dirstate.write(repo.currenttransaction())
75 repo.dirstate.write(repo.currenttransaction())
75
76
76 repair.strip(ui, repo, revs, backup)
77 repair.strip(ui, repo, revs, backup)
77
78
78 repomarks = repo._bookmarks
79 repomarks = repo._bookmarks
79 if bookmarks:
80 if bookmarks:
80 with repo.transaction('strip') as tr:
81 with repo.transaction('strip') as tr:
81 if repo._activebookmark in bookmarks:
82 if repo._activebookmark in bookmarks:
82 bookmarksmod.deactivate(repo)
83 bookmarksmod.deactivate(repo)
83 for bookmark in bookmarks:
84 for bookmark in bookmarks:
84 del repomarks[bookmark]
85 del repomarks[bookmark]
85 repomarks.recordchange(tr)
86 repomarks.recordchange(tr)
86 for bookmark in sorted(bookmarks):
87 for bookmark in sorted(bookmarks):
87 ui.write(_("bookmark '%s' deleted\n") % bookmark)
88 ui.write(_("bookmark '%s' deleted\n") % bookmark)
88 finally:
89 finally:
89 release(lock, wlock)
90 release(lock, wlock)
90
91
91
92
92 @command("strip",
93 @command("strip",
93 [
94 [
94 ('r', 'rev', [], _('strip specified revision (optional, '
95 ('r', 'rev', [], _('strip specified revision (optional, '
95 'can specify revisions without this '
96 'can specify revisions without this '
96 'option)'), _('REV')),
97 'option)'), _('REV')),
97 ('f', 'force', None, _('force removal of changesets, discard '
98 ('f', 'force', None, _('force removal of changesets, discard '
98 'uncommitted changes (no backup)')),
99 'uncommitted changes (no backup)')),
99 ('', 'no-backup', None, _('no backups')),
100 ('', 'no-backup', None, _('no backups')),
100 ('', 'nobackup', None, _('no backups (DEPRECATED)')),
101 ('', 'nobackup', None, _('no backups (DEPRECATED)')),
101 ('n', '', None, _('ignored (DEPRECATED)')),
102 ('n', '', None, _('ignored (DEPRECATED)')),
102 ('k', 'keep', None, _("do not modify working directory during "
103 ('k', 'keep', None, _("do not modify working directory during "
103 "strip")),
104 "strip")),
104 ('B', 'bookmark', [], _("remove revs only reachable from given"
105 ('B', 'bookmark', [], _("remove revs only reachable from given"
105 " bookmark"))],
106 " bookmark"))],
106 _('hg strip [-k] [-f] [-B bookmark] [-r] REV...'))
107 _('hg strip [-k] [-f] [-B bookmark] [-r] REV...'))
107 def stripcmd(ui, repo, *revs, **opts):
108 def stripcmd(ui, repo, *revs, **opts):
108 """strip changesets and all their descendants from the repository
109 """strip changesets and all their descendants from the repository
109
110
110 The strip command removes the specified changesets and all their
111 The strip command removes the specified changesets and all their
111 descendants. If the working directory has uncommitted changes, the
112 descendants. If the working directory has uncommitted changes, the
112 operation is aborted unless the --force flag is supplied, in which
113 operation is aborted unless the --force flag is supplied, in which
113 case changes will be discarded.
114 case changes will be discarded.
114
115
115 If a parent of the working directory is stripped, then the working
116 If a parent of the working directory is stripped, then the working
116 directory will automatically be updated to the most recent
117 directory will automatically be updated to the most recent
117 available ancestor of the stripped parent after the operation
118 available ancestor of the stripped parent after the operation
118 completes.
119 completes.
119
120
120 Any stripped changesets are stored in ``.hg/strip-backup`` as a
121 Any stripped changesets are stored in ``.hg/strip-backup`` as a
121 bundle (see :hg:`help bundle` and :hg:`help unbundle`). They can
122 bundle (see :hg:`help bundle` and :hg:`help unbundle`). They can
122 be restored by running :hg:`unbundle .hg/strip-backup/BUNDLE`,
123 be restored by running :hg:`unbundle .hg/strip-backup/BUNDLE`,
123 where BUNDLE is the bundle file created by the strip. Note that
124 where BUNDLE is the bundle file created by the strip. Note that
124 the local revision numbers will in general be different after the
125 the local revision numbers will in general be different after the
125 restore.
126 restore.
126
127
127 Use the --no-backup option to discard the backup bundle once the
128 Use the --no-backup option to discard the backup bundle once the
128 operation completes.
129 operation completes.
129
130
130 Strip is not a history-rewriting operation and can be used on
131 Strip is not a history-rewriting operation and can be used on
131 changesets in the public phase. But if the stripped changesets have
132 changesets in the public phase. But if the stripped changesets have
132 been pushed to a remote repository you will likely pull them again.
133 been pushed to a remote repository you will likely pull them again.
133
134
134 Return 0 on success.
135 Return 0 on success.
135 """
136 """
137 opts = pycompat.byteskwargs(opts)
136 backup = True
138 backup = True
137 if opts.get('no_backup') or opts.get('nobackup'):
139 if opts.get('no_backup') or opts.get('nobackup'):
138 backup = False
140 backup = False
139
141
140 cl = repo.changelog
142 cl = repo.changelog
141 revs = list(revs) + opts.get('rev')
143 revs = list(revs) + opts.get('rev')
142 revs = set(scmutil.revrange(repo, revs))
144 revs = set(scmutil.revrange(repo, revs))
143
145
144 with repo.wlock():
146 with repo.wlock():
145 bookmarks = set(opts.get('bookmark'))
147 bookmarks = set(opts.get('bookmark'))
146 if bookmarks:
148 if bookmarks:
147 repomarks = repo._bookmarks
149 repomarks = repo._bookmarks
148 if not bookmarks.issubset(repomarks):
150 if not bookmarks.issubset(repomarks):
149 raise error.Abort(_("bookmark '%s' not found") %
151 raise error.Abort(_("bookmark '%s' not found") %
150 ','.join(sorted(bookmarks - set(repomarks.keys()))))
152 ','.join(sorted(bookmarks - set(repomarks.keys()))))
151
153
152 # If the requested bookmark is not the only one pointing to a
154 # If the requested bookmark is not the only one pointing to a
153 # a revision we have to only delete the bookmark and not strip
155 # a revision we have to only delete the bookmark and not strip
154 # anything. revsets cannot detect that case.
156 # anything. revsets cannot detect that case.
155 nodetobookmarks = {}
157 nodetobookmarks = {}
156 for mark, node in repomarks.iteritems():
158 for mark, node in repomarks.iteritems():
157 nodetobookmarks.setdefault(node, []).append(mark)
159 nodetobookmarks.setdefault(node, []).append(mark)
158 for marks in nodetobookmarks.values():
160 for marks in nodetobookmarks.values():
159 if bookmarks.issuperset(marks):
161 if bookmarks.issuperset(marks):
160 rsrevs = repair.stripbmrevset(repo, marks[0])
162 rsrevs = repair.stripbmrevset(repo, marks[0])
161 revs.update(set(rsrevs))
163 revs.update(set(rsrevs))
162 if not revs:
164 if not revs:
163 lock = tr = None
165 lock = tr = None
164 try:
166 try:
165 lock = repo.lock()
167 lock = repo.lock()
166 tr = repo.transaction('bookmark')
168 tr = repo.transaction('bookmark')
167 for bookmark in bookmarks:
169 for bookmark in bookmarks:
168 del repomarks[bookmark]
170 del repomarks[bookmark]
169 repomarks.recordchange(tr)
171 repomarks.recordchange(tr)
170 tr.close()
172 tr.close()
171 for bookmark in sorted(bookmarks):
173 for bookmark in sorted(bookmarks):
172 ui.write(_("bookmark '%s' deleted\n") % bookmark)
174 ui.write(_("bookmark '%s' deleted\n") % bookmark)
173 finally:
175 finally:
174 release(lock, tr)
176 release(lock, tr)
175
177
176 if not revs:
178 if not revs:
177 raise error.Abort(_('empty revision set'))
179 raise error.Abort(_('empty revision set'))
178
180
179 descendants = set(cl.descendants(revs))
181 descendants = set(cl.descendants(revs))
180 strippedrevs = revs.union(descendants)
182 strippedrevs = revs.union(descendants)
181 roots = revs.difference(descendants)
183 roots = revs.difference(descendants)
182
184
183 update = False
185 update = False
184 # if one of the wdir parent is stripped we'll need
186 # if one of the wdir parent is stripped we'll need
185 # to update away to an earlier revision
187 # to update away to an earlier revision
186 for p in repo.dirstate.parents():
188 for p in repo.dirstate.parents():
187 if p != nullid and cl.rev(p) in strippedrevs:
189 if p != nullid and cl.rev(p) in strippedrevs:
188 update = True
190 update = True
189 break
191 break
190
192
191 rootnodes = set(cl.node(r) for r in roots)
193 rootnodes = set(cl.node(r) for r in roots)
192
194
193 q = getattr(repo, 'mq', None)
195 q = getattr(repo, 'mq', None)
194 if q is not None and q.applied:
196 if q is not None and q.applied:
195 # refresh queue state if we're about to strip
197 # refresh queue state if we're about to strip
196 # applied patches
198 # applied patches
197 if cl.rev(repo.lookup('qtip')) in strippedrevs:
199 if cl.rev(repo.lookup('qtip')) in strippedrevs:
198 q.applieddirty = True
200 q.applieddirty = True
199 start = 0
201 start = 0
200 end = len(q.applied)
202 end = len(q.applied)
201 for i, statusentry in enumerate(q.applied):
203 for i, statusentry in enumerate(q.applied):
202 if statusentry.node in rootnodes:
204 if statusentry.node in rootnodes:
203 # if one of the stripped roots is an applied
205 # if one of the stripped roots is an applied
204 # patch, only part of the queue is stripped
206 # patch, only part of the queue is stripped
205 start = i
207 start = i
206 break
208 break
207 del q.applied[start:end]
209 del q.applied[start:end]
208 q.savedirty()
210 q.savedirty()
209
211
210 revs = sorted(rootnodes)
212 revs = sorted(rootnodes)
211 if update and opts.get('keep'):
213 if update and opts.get('keep'):
212 urev, p2 = repo.changelog.parents(revs[0])
214 urev, p2 = repo.changelog.parents(revs[0])
213 if (util.safehasattr(repo, 'mq') and p2 != nullid
215 if (util.safehasattr(repo, 'mq') and p2 != nullid
214 and p2 in [x.node for x in repo.mq.applied]):
216 and p2 in [x.node for x in repo.mq.applied]):
215 urev = p2
217 urev = p2
216 uctx = repo[urev]
218 uctx = repo[urev]
217
219
218 # only reset the dirstate for files that would actually change
220 # only reset the dirstate for files that would actually change
219 # between the working context and uctx
221 # between the working context and uctx
220 descendantrevs = repo.revs("%s::." % uctx.rev())
222 descendantrevs = repo.revs("%s::." % uctx.rev())
221 changedfiles = []
223 changedfiles = []
222 for rev in descendantrevs:
224 for rev in descendantrevs:
223 # blindly reset the files, regardless of what actually changed
225 # blindly reset the files, regardless of what actually changed
224 changedfiles.extend(repo[rev].files())
226 changedfiles.extend(repo[rev].files())
225
227
226 # reset files that only changed in the dirstate too
228 # reset files that only changed in the dirstate too
227 dirstate = repo.dirstate
229 dirstate = repo.dirstate
228 dirchanges = [f for f in dirstate if dirstate[f] != 'n']
230 dirchanges = [f for f in dirstate if dirstate[f] != 'n']
229 changedfiles.extend(dirchanges)
231 changedfiles.extend(dirchanges)
230
232
231 repo.dirstate.rebuild(urev, uctx.manifest(), changedfiles)
233 repo.dirstate.rebuild(urev, uctx.manifest(), changedfiles)
232 repo.dirstate.write(repo.currenttransaction())
234 repo.dirstate.write(repo.currenttransaction())
233
235
234 # clear resolve state
236 # clear resolve state
235 merge.mergestate.clean(repo, repo['.'].node())
237 merge.mergestate.clean(repo, repo['.'].node())
236
238
237 update = False
239 update = False
238
240
239
241
240 strip(ui, repo, revs, backup=backup, update=update,
242 strip(ui, repo, revs, backup=backup, update=update,
241 force=opts.get('force'), bookmarks=bookmarks)
243 force=opts.get('force'), bookmarks=bookmarks)
242
244
243 return 0
245 return 0
@@ -1,491 +1,492
1 # formatter.py - generic output formatting for mercurial
1 # formatter.py - generic output formatting for mercurial
2 #
2 #
3 # Copyright 2012 Matt Mackall <mpm@selenic.com>
3 # Copyright 2012 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 """Generic output formatting for Mercurial
8 """Generic output formatting for Mercurial
9
9
10 The formatter provides API to show data in various ways. The following
10 The formatter provides API to show data in various ways. The following
11 functions should be used in place of ui.write():
11 functions should be used in place of ui.write():
12
12
13 - fm.write() for unconditional output
13 - fm.write() for unconditional output
14 - fm.condwrite() to show some extra data conditionally in plain output
14 - fm.condwrite() to show some extra data conditionally in plain output
15 - fm.context() to provide changectx to template output
15 - fm.context() to provide changectx to template output
16 - fm.data() to provide extra data to JSON or template output
16 - fm.data() to provide extra data to JSON or template output
17 - fm.plain() to show raw text that isn't provided to JSON or template output
17 - fm.plain() to show raw text that isn't provided to JSON or template output
18
18
19 To show structured data (e.g. date tuples, dicts, lists), apply fm.format*()
19 To show structured data (e.g. date tuples, dicts, lists), apply fm.format*()
20 beforehand so the data is converted to the appropriate data type. Use
20 beforehand so the data is converted to the appropriate data type. Use
21 fm.isplain() if you need to convert or format data conditionally which isn't
21 fm.isplain() if you need to convert or format data conditionally which isn't
22 supported by the formatter API.
22 supported by the formatter API.
23
23
24 To build nested structure (i.e. a list of dicts), use fm.nested().
24 To build nested structure (i.e. a list of dicts), use fm.nested().
25
25
26 See also https://www.mercurial-scm.org/wiki/GenericTemplatingPlan
26 See also https://www.mercurial-scm.org/wiki/GenericTemplatingPlan
27
27
28 fm.condwrite() vs 'if cond:':
28 fm.condwrite() vs 'if cond:':
29
29
30 In most cases, use fm.condwrite() so users can selectively show the data
30 In most cases, use fm.condwrite() so users can selectively show the data
31 in template output. If it's costly to build data, use plain 'if cond:' with
31 in template output. If it's costly to build data, use plain 'if cond:' with
32 fm.write().
32 fm.write().
33
33
34 fm.nested() vs fm.formatdict() (or fm.formatlist()):
34 fm.nested() vs fm.formatdict() (or fm.formatlist()):
35
35
36 fm.nested() should be used to form a tree structure (a list of dicts of
36 fm.nested() should be used to form a tree structure (a list of dicts of
37 lists of dicts...) which can be accessed through template keywords, e.g.
37 lists of dicts...) which can be accessed through template keywords, e.g.
38 "{foo % "{bar % {...}} {baz % {...}}"}". On the other hand, fm.formatdict()
38 "{foo % "{bar % {...}} {baz % {...}}"}". On the other hand, fm.formatdict()
39 exports a dict-type object to template, which can be accessed by e.g.
39 exports a dict-type object to template, which can be accessed by e.g.
40 "{get(foo, key)}" function.
40 "{get(foo, key)}" function.
41
41
42 Doctest helper:
42 Doctest helper:
43
43
44 >>> def show(fn, verbose=False, **opts):
44 >>> def show(fn, verbose=False, **opts):
45 ... import sys
45 ... import sys
46 ... from . import ui as uimod
46 ... from . import ui as uimod
47 ... ui = uimod.ui()
47 ... ui = uimod.ui()
48 ... ui.fout = sys.stdout # redirect to doctest
48 ... ui.fout = sys.stdout # redirect to doctest
49 ... ui.verbose = verbose
49 ... ui.verbose = verbose
50 ... return fn(ui, ui.formatter(fn.__name__, opts))
50 ... return fn(ui, ui.formatter(fn.__name__, opts))
51
51
52 Basic example:
52 Basic example:
53
53
54 >>> def files(ui, fm):
54 >>> def files(ui, fm):
55 ... files = [('foo', 123, (0, 0)), ('bar', 456, (1, 0))]
55 ... files = [('foo', 123, (0, 0)), ('bar', 456, (1, 0))]
56 ... for f in files:
56 ... for f in files:
57 ... fm.startitem()
57 ... fm.startitem()
58 ... fm.write('path', '%s', f[0])
58 ... fm.write('path', '%s', f[0])
59 ... fm.condwrite(ui.verbose, 'date', ' %s',
59 ... fm.condwrite(ui.verbose, 'date', ' %s',
60 ... fm.formatdate(f[2], '%Y-%m-%d %H:%M:%S'))
60 ... fm.formatdate(f[2], '%Y-%m-%d %H:%M:%S'))
61 ... fm.data(size=f[1])
61 ... fm.data(size=f[1])
62 ... fm.plain('\\n')
62 ... fm.plain('\\n')
63 ... fm.end()
63 ... fm.end()
64 >>> show(files)
64 >>> show(files)
65 foo
65 foo
66 bar
66 bar
67 >>> show(files, verbose=True)
67 >>> show(files, verbose=True)
68 foo 1970-01-01 00:00:00
68 foo 1970-01-01 00:00:00
69 bar 1970-01-01 00:00:01
69 bar 1970-01-01 00:00:01
70 >>> show(files, template='json')
70 >>> show(files, template='json')
71 [
71 [
72 {
72 {
73 "date": [0, 0],
73 "date": [0, 0],
74 "path": "foo",
74 "path": "foo",
75 "size": 123
75 "size": 123
76 },
76 },
77 {
77 {
78 "date": [1, 0],
78 "date": [1, 0],
79 "path": "bar",
79 "path": "bar",
80 "size": 456
80 "size": 456
81 }
81 }
82 ]
82 ]
83 >>> show(files, template='path: {path}\\ndate: {date|rfc3339date}\\n')
83 >>> show(files, template='path: {path}\\ndate: {date|rfc3339date}\\n')
84 path: foo
84 path: foo
85 date: 1970-01-01T00:00:00+00:00
85 date: 1970-01-01T00:00:00+00:00
86 path: bar
86 path: bar
87 date: 1970-01-01T00:00:01+00:00
87 date: 1970-01-01T00:00:01+00:00
88
88
89 Nested example:
89 Nested example:
90
90
91 >>> def subrepos(ui, fm):
91 >>> def subrepos(ui, fm):
92 ... fm.startitem()
92 ... fm.startitem()
93 ... fm.write('repo', '[%s]\\n', 'baz')
93 ... fm.write('repo', '[%s]\\n', 'baz')
94 ... files(ui, fm.nested('files'))
94 ... files(ui, fm.nested('files'))
95 ... fm.end()
95 ... fm.end()
96 >>> show(subrepos)
96 >>> show(subrepos)
97 [baz]
97 [baz]
98 foo
98 foo
99 bar
99 bar
100 >>> show(subrepos, template='{repo}: {join(files % "{path}", ", ")}\\n')
100 >>> show(subrepos, template='{repo}: {join(files % "{path}", ", ")}\\n')
101 baz: foo, bar
101 baz: foo, bar
102 """
102 """
103
103
104 from __future__ import absolute_import
104 from __future__ import absolute_import
105
105
106 import collections
106 import collections
107 import contextlib
107 import contextlib
108 import itertools
108 import itertools
109 import os
109 import os
110
110
111 from .i18n import _
111 from .i18n import _
112 from .node import (
112 from .node import (
113 hex,
113 hex,
114 short,
114 short,
115 )
115 )
116
116
117 from . import (
117 from . import (
118 error,
118 error,
119 pycompat,
119 pycompat,
120 templatefilters,
120 templatefilters,
121 templatekw,
121 templatekw,
122 templater,
122 templater,
123 util,
123 util,
124 )
124 )
125
125
126 pickle = util.pickle
126 pickle = util.pickle
127
127
128 class _nullconverter(object):
128 class _nullconverter(object):
129 '''convert non-primitive data types to be processed by formatter'''
129 '''convert non-primitive data types to be processed by formatter'''
130 @staticmethod
130 @staticmethod
131 def formatdate(date, fmt):
131 def formatdate(date, fmt):
132 '''convert date tuple to appropriate format'''
132 '''convert date tuple to appropriate format'''
133 return date
133 return date
134 @staticmethod
134 @staticmethod
135 def formatdict(data, key, value, fmt, sep):
135 def formatdict(data, key, value, fmt, sep):
136 '''convert dict or key-value pairs to appropriate dict format'''
136 '''convert dict or key-value pairs to appropriate dict format'''
137 # use plain dict instead of util.sortdict so that data can be
137 # use plain dict instead of util.sortdict so that data can be
138 # serialized as a builtin dict in pickle output
138 # serialized as a builtin dict in pickle output
139 return dict(data)
139 return dict(data)
140 @staticmethod
140 @staticmethod
141 def formatlist(data, name, fmt, sep):
141 def formatlist(data, name, fmt, sep):
142 '''convert iterable to appropriate list format'''
142 '''convert iterable to appropriate list format'''
143 return list(data)
143 return list(data)
144
144
145 class baseformatter(object):
145 class baseformatter(object):
146 def __init__(self, ui, topic, opts, converter):
146 def __init__(self, ui, topic, opts, converter):
147 self._ui = ui
147 self._ui = ui
148 self._topic = topic
148 self._topic = topic
149 self._style = opts.get("style")
149 self._style = opts.get("style")
150 self._template = opts.get("template")
150 self._template = opts.get("template")
151 self._converter = converter
151 self._converter = converter
152 self._item = None
152 self._item = None
153 # function to convert node to string suitable for this output
153 # function to convert node to string suitable for this output
154 self.hexfunc = hex
154 self.hexfunc = hex
155 def __enter__(self):
155 def __enter__(self):
156 return self
156 return self
157 def __exit__(self, exctype, excvalue, traceback):
157 def __exit__(self, exctype, excvalue, traceback):
158 if exctype is None:
158 if exctype is None:
159 self.end()
159 self.end()
160 def _showitem(self):
160 def _showitem(self):
161 '''show a formatted item once all data is collected'''
161 '''show a formatted item once all data is collected'''
162 pass
162 pass
163 def startitem(self):
163 def startitem(self):
164 '''begin an item in the format list'''
164 '''begin an item in the format list'''
165 if self._item is not None:
165 if self._item is not None:
166 self._showitem()
166 self._showitem()
167 self._item = {}
167 self._item = {}
168 def formatdate(self, date, fmt='%a %b %d %H:%M:%S %Y %1%2'):
168 def formatdate(self, date, fmt='%a %b %d %H:%M:%S %Y %1%2'):
169 '''convert date tuple to appropriate format'''
169 '''convert date tuple to appropriate format'''
170 return self._converter.formatdate(date, fmt)
170 return self._converter.formatdate(date, fmt)
171 def formatdict(self, data, key='key', value='value', fmt='%s=%s', sep=' '):
171 def formatdict(self, data, key='key', value='value', fmt='%s=%s', sep=' '):
172 '''convert dict or key-value pairs to appropriate dict format'''
172 '''convert dict or key-value pairs to appropriate dict format'''
173 return self._converter.formatdict(data, key, value, fmt, sep)
173 return self._converter.formatdict(data, key, value, fmt, sep)
174 def formatlist(self, data, name, fmt='%s', sep=' '):
174 def formatlist(self, data, name, fmt='%s', sep=' '):
175 '''convert iterable to appropriate list format'''
175 '''convert iterable to appropriate list format'''
176 # name is mandatory argument for now, but it could be optional if
176 # name is mandatory argument for now, but it could be optional if
177 # we have default template keyword, e.g. {item}
177 # we have default template keyword, e.g. {item}
178 return self._converter.formatlist(data, name, fmt, sep)
178 return self._converter.formatlist(data, name, fmt, sep)
179 def context(self, **ctxs):
179 def context(self, **ctxs):
180 '''insert context objects to be used to render template keywords'''
180 '''insert context objects to be used to render template keywords'''
181 pass
181 pass
182 def data(self, **data):
182 def data(self, **data):
183 '''insert data into item that's not shown in default output'''
183 '''insert data into item that's not shown in default output'''
184 data = pycompat.byteskwargs(data)
184 data = pycompat.byteskwargs(data)
185 self._item.update(data)
185 self._item.update(data)
186 def write(self, fields, deftext, *fielddata, **opts):
186 def write(self, fields, deftext, *fielddata, **opts):
187 '''do default text output while assigning data to item'''
187 '''do default text output while assigning data to item'''
188 fieldkeys = fields.split()
188 fieldkeys = fields.split()
189 assert len(fieldkeys) == len(fielddata)
189 assert len(fieldkeys) == len(fielddata)
190 self._item.update(zip(fieldkeys, fielddata))
190 self._item.update(zip(fieldkeys, fielddata))
191 def condwrite(self, cond, fields, deftext, *fielddata, **opts):
191 def condwrite(self, cond, fields, deftext, *fielddata, **opts):
192 '''do conditional write (primarily for plain formatter)'''
192 '''do conditional write (primarily for plain formatter)'''
193 fieldkeys = fields.split()
193 fieldkeys = fields.split()
194 assert len(fieldkeys) == len(fielddata)
194 assert len(fieldkeys) == len(fielddata)
195 self._item.update(zip(fieldkeys, fielddata))
195 self._item.update(zip(fieldkeys, fielddata))
196 def plain(self, text, **opts):
196 def plain(self, text, **opts):
197 '''show raw text for non-templated mode'''
197 '''show raw text for non-templated mode'''
198 pass
198 pass
199 def isplain(self):
199 def isplain(self):
200 '''check for plain formatter usage'''
200 '''check for plain formatter usage'''
201 return False
201 return False
202 def nested(self, field):
202 def nested(self, field):
203 '''sub formatter to store nested data in the specified field'''
203 '''sub formatter to store nested data in the specified field'''
204 self._item[field] = data = []
204 self._item[field] = data = []
205 return _nestedformatter(self._ui, self._converter, data)
205 return _nestedformatter(self._ui, self._converter, data)
206 def end(self):
206 def end(self):
207 '''end output for the formatter'''
207 '''end output for the formatter'''
208 if self._item is not None:
208 if self._item is not None:
209 self._showitem()
209 self._showitem()
210
210
211 def nullformatter(ui, topic):
211 def nullformatter(ui, topic):
212 '''formatter that prints nothing'''
212 '''formatter that prints nothing'''
213 return baseformatter(ui, topic, opts={}, converter=_nullconverter)
213 return baseformatter(ui, topic, opts={}, converter=_nullconverter)
214
214
215 class _nestedformatter(baseformatter):
215 class _nestedformatter(baseformatter):
216 '''build sub items and store them in the parent formatter'''
216 '''build sub items and store them in the parent formatter'''
217 def __init__(self, ui, converter, data):
217 def __init__(self, ui, converter, data):
218 baseformatter.__init__(self, ui, topic='', opts={}, converter=converter)
218 baseformatter.__init__(self, ui, topic='', opts={}, converter=converter)
219 self._data = data
219 self._data = data
220 def _showitem(self):
220 def _showitem(self):
221 self._data.append(self._item)
221 self._data.append(self._item)
222
222
223 def _iteritems(data):
223 def _iteritems(data):
224 '''iterate key-value pairs in stable order'''
224 '''iterate key-value pairs in stable order'''
225 if isinstance(data, dict):
225 if isinstance(data, dict):
226 return sorted(data.iteritems())
226 return sorted(data.iteritems())
227 return data
227 return data
228
228
229 class _plainconverter(object):
229 class _plainconverter(object):
230 '''convert non-primitive data types to text'''
230 '''convert non-primitive data types to text'''
231 @staticmethod
231 @staticmethod
232 def formatdate(date, fmt):
232 def formatdate(date, fmt):
233 '''stringify date tuple in the given format'''
233 '''stringify date tuple in the given format'''
234 return util.datestr(date, fmt)
234 return util.datestr(date, fmt)
235 @staticmethod
235 @staticmethod
236 def formatdict(data, key, value, fmt, sep):
236 def formatdict(data, key, value, fmt, sep):
237 '''stringify key-value pairs separated by sep'''
237 '''stringify key-value pairs separated by sep'''
238 return sep.join(fmt % (k, v) for k, v in _iteritems(data))
238 return sep.join(fmt % (k, v) for k, v in _iteritems(data))
239 @staticmethod
239 @staticmethod
240 def formatlist(data, name, fmt, sep):
240 def formatlist(data, name, fmt, sep):
241 '''stringify iterable separated by sep'''
241 '''stringify iterable separated by sep'''
242 return sep.join(fmt % e for e in data)
242 return sep.join(fmt % e for e in data)
243
243
244 class plainformatter(baseformatter):
244 class plainformatter(baseformatter):
245 '''the default text output scheme'''
245 '''the default text output scheme'''
246 def __init__(self, ui, out, topic, opts):
246 def __init__(self, ui, out, topic, opts):
247 baseformatter.__init__(self, ui, topic, opts, _plainconverter)
247 baseformatter.__init__(self, ui, topic, opts, _plainconverter)
248 if ui.debugflag:
248 if ui.debugflag:
249 self.hexfunc = hex
249 self.hexfunc = hex
250 else:
250 else:
251 self.hexfunc = short
251 self.hexfunc = short
252 if ui is out:
252 if ui is out:
253 self._write = ui.write
253 self._write = ui.write
254 else:
254 else:
255 self._write = lambda s, **opts: out.write(s)
255 self._write = lambda s, **opts: out.write(s)
256 def startitem(self):
256 def startitem(self):
257 pass
257 pass
258 def data(self, **data):
258 def data(self, **data):
259 pass
259 pass
260 def write(self, fields, deftext, *fielddata, **opts):
260 def write(self, fields, deftext, *fielddata, **opts):
261 self._write(deftext % fielddata, **opts)
261 self._write(deftext % fielddata, **opts)
262 def condwrite(self, cond, fields, deftext, *fielddata, **opts):
262 def condwrite(self, cond, fields, deftext, *fielddata, **opts):
263 '''do conditional write'''
263 '''do conditional write'''
264 if cond:
264 if cond:
265 self._write(deftext % fielddata, **opts)
265 self._write(deftext % fielddata, **opts)
266 def plain(self, text, **opts):
266 def plain(self, text, **opts):
267 self._write(text, **opts)
267 self._write(text, **opts)
268 def isplain(self):
268 def isplain(self):
269 return True
269 return True
270 def nested(self, field):
270 def nested(self, field):
271 # nested data will be directly written to ui
271 # nested data will be directly written to ui
272 return self
272 return self
273 def end(self):
273 def end(self):
274 pass
274 pass
275
275
276 class debugformatter(baseformatter):
276 class debugformatter(baseformatter):
277 def __init__(self, ui, out, topic, opts):
277 def __init__(self, ui, out, topic, opts):
278 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
278 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
279 self._out = out
279 self._out = out
280 self._out.write("%s = [\n" % self._topic)
280 self._out.write("%s = [\n" % self._topic)
281 def _showitem(self):
281 def _showitem(self):
282 self._out.write(" " + repr(self._item) + ",\n")
282 self._out.write(" " + repr(self._item) + ",\n")
283 def end(self):
283 def end(self):
284 baseformatter.end(self)
284 baseformatter.end(self)
285 self._out.write("]\n")
285 self._out.write("]\n")
286
286
287 class pickleformatter(baseformatter):
287 class pickleformatter(baseformatter):
288 def __init__(self, ui, out, topic, opts):
288 def __init__(self, ui, out, topic, opts):
289 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
289 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
290 self._out = out
290 self._out = out
291 self._data = []
291 self._data = []
292 def _showitem(self):
292 def _showitem(self):
293 self._data.append(self._item)
293 self._data.append(self._item)
294 def end(self):
294 def end(self):
295 baseformatter.end(self)
295 baseformatter.end(self)
296 self._out.write(pickle.dumps(self._data))
296 self._out.write(pickle.dumps(self._data))
297
297
298 class jsonformatter(baseformatter):
298 class jsonformatter(baseformatter):
299 def __init__(self, ui, out, topic, opts):
299 def __init__(self, ui, out, topic, opts):
300 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
300 baseformatter.__init__(self, ui, topic, opts, _nullconverter)
301 self._out = out
301 self._out = out
302 self._out.write("[")
302 self._out.write("[")
303 self._first = True
303 self._first = True
304 def _showitem(self):
304 def _showitem(self):
305 if self._first:
305 if self._first:
306 self._first = False
306 self._first = False
307 else:
307 else:
308 self._out.write(",")
308 self._out.write(",")
309
309
310 self._out.write("\n {\n")
310 self._out.write("\n {\n")
311 first = True
311 first = True
312 for k, v in sorted(self._item.items()):
312 for k, v in sorted(self._item.items()):
313 if first:
313 if first:
314 first = False
314 first = False
315 else:
315 else:
316 self._out.write(",\n")
316 self._out.write(",\n")
317 u = templatefilters.json(v, paranoid=False)
317 u = templatefilters.json(v, paranoid=False)
318 self._out.write(' "%s": %s' % (k, u))
318 self._out.write(' "%s": %s' % (k, u))
319 self._out.write("\n }")
319 self._out.write("\n }")
320 def end(self):
320 def end(self):
321 baseformatter.end(self)
321 baseformatter.end(self)
322 self._out.write("\n]\n")
322 self._out.write("\n]\n")
323
323
324 class _templateconverter(object):
324 class _templateconverter(object):
325 '''convert non-primitive data types to be processed by templater'''
325 '''convert non-primitive data types to be processed by templater'''
326 @staticmethod
326 @staticmethod
327 def formatdate(date, fmt):
327 def formatdate(date, fmt):
328 '''return date tuple'''
328 '''return date tuple'''
329 return date
329 return date
330 @staticmethod
330 @staticmethod
331 def formatdict(data, key, value, fmt, sep):
331 def formatdict(data, key, value, fmt, sep):
332 '''build object that can be evaluated as either plain string or dict'''
332 '''build object that can be evaluated as either plain string or dict'''
333 data = util.sortdict(_iteritems(data))
333 data = util.sortdict(_iteritems(data))
334 def f():
334 def f():
335 yield _plainconverter.formatdict(data, key, value, fmt, sep)
335 yield _plainconverter.formatdict(data, key, value, fmt, sep)
336 return templatekw.hybriddict(data, key=key, value=value, fmt=fmt,
336 return templatekw.hybriddict(data, key=key, value=value, fmt=fmt,
337 gen=f())
337 gen=f())
338 @staticmethod
338 @staticmethod
339 def formatlist(data, name, fmt, sep):
339 def formatlist(data, name, fmt, sep):
340 '''build object that can be evaluated as either plain string or list'''
340 '''build object that can be evaluated as either plain string or list'''
341 data = list(data)
341 data = list(data)
342 def f():
342 def f():
343 yield _plainconverter.formatlist(data, name, fmt, sep)
343 yield _plainconverter.formatlist(data, name, fmt, sep)
344 return templatekw.hybridlist(data, name=name, fmt=fmt, gen=f())
344 return templatekw.hybridlist(data, name=name, fmt=fmt, gen=f())
345
345
346 class templateformatter(baseformatter):
346 class templateformatter(baseformatter):
347 def __init__(self, ui, out, topic, opts):
347 def __init__(self, ui, out, topic, opts):
348 baseformatter.__init__(self, ui, topic, opts, _templateconverter)
348 baseformatter.__init__(self, ui, topic, opts, _templateconverter)
349 self._out = out
349 self._out = out
350 spec = lookuptemplate(ui, topic, opts.get('template', ''))
350 spec = lookuptemplate(ui, topic, opts.get('template', ''))
351 self._tref = spec.ref
351 self._tref = spec.ref
352 self._t = loadtemplater(ui, spec, cache=templatekw.defaulttempl)
352 self._t = loadtemplater(ui, spec, cache=templatekw.defaulttempl)
353 self._counter = itertools.count()
353 self._counter = itertools.count()
354 self._cache = {} # for templatekw/funcs to store reusable data
354 self._cache = {} # for templatekw/funcs to store reusable data
355 def context(self, **ctxs):
355 def context(self, **ctxs):
356 '''insert context objects to be used to render template keywords'''
356 '''insert context objects to be used to render template keywords'''
357 ctxs = pycompat.byteskwargs(ctxs)
357 assert all(k == 'ctx' for k in ctxs)
358 assert all(k == 'ctx' for k in ctxs)
358 self._item.update(ctxs)
359 self._item.update(ctxs)
359 def _showitem(self):
360 def _showitem(self):
360 # TODO: add support for filectx. probably each template keyword or
361 # TODO: add support for filectx. probably each template keyword or
361 # function will have to declare dependent resources. e.g.
362 # function will have to declare dependent resources. e.g.
362 # @templatekeyword(..., requires=('ctx',))
363 # @templatekeyword(..., requires=('ctx',))
363 props = {}
364 props = {}
364 if 'ctx' in self._item:
365 if 'ctx' in self._item:
365 props.update(templatekw.keywords)
366 props.update(templatekw.keywords)
366 props['index'] = next(self._counter)
367 props['index'] = next(self._counter)
367 # explicitly-defined fields precede templatekw
368 # explicitly-defined fields precede templatekw
368 props.update(self._item)
369 props.update(self._item)
369 if 'ctx' in self._item:
370 if 'ctx' in self._item:
370 # but template resources must be always available
371 # but template resources must be always available
371 props['templ'] = self._t
372 props['templ'] = self._t
372 props['repo'] = props['ctx'].repo()
373 props['repo'] = props['ctx'].repo()
373 props['revcache'] = {}
374 props['revcache'] = {}
374 props = pycompat.strkwargs(props)
375 props = pycompat.strkwargs(props)
375 g = self._t(self._tref, ui=self._ui, cache=self._cache, **props)
376 g = self._t(self._tref, ui=self._ui, cache=self._cache, **props)
376 self._out.write(templater.stringify(g))
377 self._out.write(templater.stringify(g))
377
378
378 templatespec = collections.namedtuple(r'templatespec',
379 templatespec = collections.namedtuple(r'templatespec',
379 r'ref tmpl mapfile')
380 r'ref tmpl mapfile')
380
381
381 def lookuptemplate(ui, topic, tmpl):
382 def lookuptemplate(ui, topic, tmpl):
382 """Find the template matching the given -T/--template spec 'tmpl'
383 """Find the template matching the given -T/--template spec 'tmpl'
383
384
384 'tmpl' can be any of the following:
385 'tmpl' can be any of the following:
385
386
386 - a literal template (e.g. '{rev}')
387 - a literal template (e.g. '{rev}')
387 - a map-file name or path (e.g. 'changelog')
388 - a map-file name or path (e.g. 'changelog')
388 - a reference to [templates] in config file
389 - a reference to [templates] in config file
389 - a path to raw template file
390 - a path to raw template file
390
391
391 A map file defines a stand-alone template environment. If a map file
392 A map file defines a stand-alone template environment. If a map file
392 selected, all templates defined in the file will be loaded, and the
393 selected, all templates defined in the file will be loaded, and the
393 template matching the given topic will be rendered. No aliases will be
394 template matching the given topic will be rendered. No aliases will be
394 loaded from user config.
395 loaded from user config.
395
396
396 If no map file selected, all templates in [templates] section will be
397 If no map file selected, all templates in [templates] section will be
397 available as well as aliases in [templatealias].
398 available as well as aliases in [templatealias].
398 """
399 """
399
400
400 # looks like a literal template?
401 # looks like a literal template?
401 if '{' in tmpl:
402 if '{' in tmpl:
402 return templatespec('', tmpl, None)
403 return templatespec('', tmpl, None)
403
404
404 # perhaps a stock style?
405 # perhaps a stock style?
405 if not os.path.split(tmpl)[0]:
406 if not os.path.split(tmpl)[0]:
406 mapname = (templater.templatepath('map-cmdline.' + tmpl)
407 mapname = (templater.templatepath('map-cmdline.' + tmpl)
407 or templater.templatepath(tmpl))
408 or templater.templatepath(tmpl))
408 if mapname and os.path.isfile(mapname):
409 if mapname and os.path.isfile(mapname):
409 return templatespec(topic, None, mapname)
410 return templatespec(topic, None, mapname)
410
411
411 # perhaps it's a reference to [templates]
412 # perhaps it's a reference to [templates]
412 if ui.config('templates', tmpl):
413 if ui.config('templates', tmpl):
413 return templatespec(tmpl, None, None)
414 return templatespec(tmpl, None, None)
414
415
415 if tmpl == 'list':
416 if tmpl == 'list':
416 ui.write(_("available styles: %s\n") % templater.stylelist())
417 ui.write(_("available styles: %s\n") % templater.stylelist())
417 raise error.Abort(_("specify a template"))
418 raise error.Abort(_("specify a template"))
418
419
419 # perhaps it's a path to a map or a template
420 # perhaps it's a path to a map or a template
420 if ('/' in tmpl or '\\' in tmpl) and os.path.isfile(tmpl):
421 if ('/' in tmpl or '\\' in tmpl) and os.path.isfile(tmpl):
421 # is it a mapfile for a style?
422 # is it a mapfile for a style?
422 if os.path.basename(tmpl).startswith("map-"):
423 if os.path.basename(tmpl).startswith("map-"):
423 return templatespec(topic, None, os.path.realpath(tmpl))
424 return templatespec(topic, None, os.path.realpath(tmpl))
424 with util.posixfile(tmpl, 'rb') as f:
425 with util.posixfile(tmpl, 'rb') as f:
425 tmpl = f.read()
426 tmpl = f.read()
426 return templatespec('', tmpl, None)
427 return templatespec('', tmpl, None)
427
428
428 # constant string?
429 # constant string?
429 return templatespec('', tmpl, None)
430 return templatespec('', tmpl, None)
430
431
431 def loadtemplater(ui, spec, cache=None):
432 def loadtemplater(ui, spec, cache=None):
432 """Create a templater from either a literal template or loading from
433 """Create a templater from either a literal template or loading from
433 a map file"""
434 a map file"""
434 assert not (spec.tmpl and spec.mapfile)
435 assert not (spec.tmpl and spec.mapfile)
435 if spec.mapfile:
436 if spec.mapfile:
436 return templater.templater.frommapfile(spec.mapfile, cache=cache)
437 return templater.templater.frommapfile(spec.mapfile, cache=cache)
437 return maketemplater(ui, spec.tmpl, cache=cache)
438 return maketemplater(ui, spec.tmpl, cache=cache)
438
439
439 def maketemplater(ui, tmpl, cache=None):
440 def maketemplater(ui, tmpl, cache=None):
440 """Create a templater from a string template 'tmpl'"""
441 """Create a templater from a string template 'tmpl'"""
441 aliases = ui.configitems('templatealias')
442 aliases = ui.configitems('templatealias')
442 t = templater.templater(cache=cache, aliases=aliases)
443 t = templater.templater(cache=cache, aliases=aliases)
443 t.cache.update((k, templater.unquotestring(v))
444 t.cache.update((k, templater.unquotestring(v))
444 for k, v in ui.configitems('templates'))
445 for k, v in ui.configitems('templates'))
445 if tmpl:
446 if tmpl:
446 t.cache[''] = tmpl
447 t.cache[''] = tmpl
447 return t
448 return t
448
449
449 def formatter(ui, out, topic, opts):
450 def formatter(ui, out, topic, opts):
450 template = opts.get("template", "")
451 template = opts.get("template", "")
451 if template == "json":
452 if template == "json":
452 return jsonformatter(ui, out, topic, opts)
453 return jsonformatter(ui, out, topic, opts)
453 elif template == "pickle":
454 elif template == "pickle":
454 return pickleformatter(ui, out, topic, opts)
455 return pickleformatter(ui, out, topic, opts)
455 elif template == "debug":
456 elif template == "debug":
456 return debugformatter(ui, out, topic, opts)
457 return debugformatter(ui, out, topic, opts)
457 elif template != "":
458 elif template != "":
458 return templateformatter(ui, out, topic, opts)
459 return templateformatter(ui, out, topic, opts)
459 # developer config: ui.formatdebug
460 # developer config: ui.formatdebug
460 elif ui.configbool('ui', 'formatdebug'):
461 elif ui.configbool('ui', 'formatdebug'):
461 return debugformatter(ui, out, topic, opts)
462 return debugformatter(ui, out, topic, opts)
462 # deprecated config: ui.formatjson
463 # deprecated config: ui.formatjson
463 elif ui.configbool('ui', 'formatjson'):
464 elif ui.configbool('ui', 'formatjson'):
464 return jsonformatter(ui, out, topic, opts)
465 return jsonformatter(ui, out, topic, opts)
465 return plainformatter(ui, out, topic, opts)
466 return plainformatter(ui, out, topic, opts)
466
467
467 @contextlib.contextmanager
468 @contextlib.contextmanager
468 def openformatter(ui, filename, topic, opts):
469 def openformatter(ui, filename, topic, opts):
469 """Create a formatter that writes outputs to the specified file
470 """Create a formatter that writes outputs to the specified file
470
471
471 Must be invoked using the 'with' statement.
472 Must be invoked using the 'with' statement.
472 """
473 """
473 with util.posixfile(filename, 'wb') as out:
474 with util.posixfile(filename, 'wb') as out:
474 with formatter(ui, out, topic, opts) as fm:
475 with formatter(ui, out, topic, opts) as fm:
475 yield fm
476 yield fm
476
477
477 @contextlib.contextmanager
478 @contextlib.contextmanager
478 def _neverending(fm):
479 def _neverending(fm):
479 yield fm
480 yield fm
480
481
481 def maybereopen(fm, filename, opts):
482 def maybereopen(fm, filename, opts):
482 """Create a formatter backed by file if filename specified, else return
483 """Create a formatter backed by file if filename specified, else return
483 the given formatter
484 the given formatter
484
485
485 Must be invoked using the 'with' statement. This will never call fm.end()
486 Must be invoked using the 'with' statement. This will never call fm.end()
486 of the given formatter.
487 of the given formatter.
487 """
488 """
488 if filename:
489 if filename:
489 return openformatter(fm._ui, filename, fm._topic, opts)
490 return openformatter(fm._ui, filename, fm._topic, opts)
490 else:
491 else:
491 return _neverending(fm)
492 return _neverending(fm)
@@ -1,269 +1,270
1 # hook.py - hook support for mercurial
1 # hook.py - hook support for mercurial
2 #
2 #
3 # Copyright 2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 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 __future__ import absolute_import
8 from __future__ import absolute_import
9
9
10 import os
10 import os
11 import sys
11 import sys
12
12
13 from .i18n import _
13 from .i18n import _
14 from . import (
14 from . import (
15 demandimport,
15 demandimport,
16 encoding,
16 encoding,
17 error,
17 error,
18 extensions,
18 extensions,
19 pycompat,
19 pycompat,
20 util,
20 util,
21 )
21 )
22
22
23 def _pythonhook(ui, repo, htype, hname, funcname, args, throw):
23 def _pythonhook(ui, repo, htype, hname, funcname, args, throw):
24 '''call python hook. hook is callable object, looked up as
24 '''call python hook. hook is callable object, looked up as
25 name in python module. if callable returns "true", hook
25 name in python module. if callable returns "true", hook
26 fails, else passes. if hook raises exception, treated as
26 fails, else passes. if hook raises exception, treated as
27 hook failure. exception propagates if throw is "true".
27 hook failure. exception propagates if throw is "true".
28
28
29 reason for "true" meaning "hook failed" is so that
29 reason for "true" meaning "hook failed" is so that
30 unmodified commands (e.g. mercurial.commands.update) can
30 unmodified commands (e.g. mercurial.commands.update) can
31 be run as hooks without wrappers to convert return values.'''
31 be run as hooks without wrappers to convert return values.'''
32
32
33 if callable(funcname):
33 if callable(funcname):
34 obj = funcname
34 obj = funcname
35 funcname = pycompat.sysbytes(obj.__module__ + r"." + obj.__name__)
35 funcname = pycompat.sysbytes(obj.__module__ + r"." + obj.__name__)
36 else:
36 else:
37 d = funcname.rfind('.')
37 d = funcname.rfind('.')
38 if d == -1:
38 if d == -1:
39 raise error.HookLoadError(
39 raise error.HookLoadError(
40 _('%s hook is invalid: "%s" not in a module')
40 _('%s hook is invalid: "%s" not in a module')
41 % (hname, funcname))
41 % (hname, funcname))
42 modname = funcname[:d]
42 modname = funcname[:d]
43 oldpaths = sys.path
43 oldpaths = sys.path
44 if util.mainfrozen():
44 if util.mainfrozen():
45 # binary installs require sys.path manipulation
45 # binary installs require sys.path manipulation
46 modpath, modfile = os.path.split(modname)
46 modpath, modfile = os.path.split(modname)
47 if modpath and modfile:
47 if modpath and modfile:
48 sys.path = sys.path[:] + [modpath]
48 sys.path = sys.path[:] + [modpath]
49 modname = modfile
49 modname = modfile
50 with demandimport.deactivated():
50 with demandimport.deactivated():
51 try:
51 try:
52 obj = __import__(modname)
52 obj = __import__(modname)
53 except (ImportError, SyntaxError):
53 except (ImportError, SyntaxError):
54 e1 = sys.exc_info()
54 e1 = sys.exc_info()
55 try:
55 try:
56 # extensions are loaded with hgext_ prefix
56 # extensions are loaded with hgext_ prefix
57 obj = __import__("hgext_%s" % modname)
57 obj = __import__("hgext_%s" % modname)
58 except (ImportError, SyntaxError):
58 except (ImportError, SyntaxError):
59 e2 = sys.exc_info()
59 e2 = sys.exc_info()
60 if ui.tracebackflag:
60 if ui.tracebackflag:
61 ui.warn(_('exception from first failed import '
61 ui.warn(_('exception from first failed import '
62 'attempt:\n'))
62 'attempt:\n'))
63 ui.traceback(e1)
63 ui.traceback(e1)
64 if ui.tracebackflag:
64 if ui.tracebackflag:
65 ui.warn(_('exception from second failed import '
65 ui.warn(_('exception from second failed import '
66 'attempt:\n'))
66 'attempt:\n'))
67 ui.traceback(e2)
67 ui.traceback(e2)
68
68
69 if not ui.tracebackflag:
69 if not ui.tracebackflag:
70 tracebackhint = _(
70 tracebackhint = _(
71 'run with --traceback for stack trace')
71 'run with --traceback for stack trace')
72 else:
72 else:
73 tracebackhint = None
73 tracebackhint = None
74 raise error.HookLoadError(
74 raise error.HookLoadError(
75 _('%s hook is invalid: import of "%s" failed') %
75 _('%s hook is invalid: import of "%s" failed') %
76 (hname, modname), hint=tracebackhint)
76 (hname, modname), hint=tracebackhint)
77 sys.path = oldpaths
77 sys.path = oldpaths
78 try:
78 try:
79 for p in funcname.split('.')[1:]:
79 for p in funcname.split('.')[1:]:
80 obj = getattr(obj, p)
80 obj = getattr(obj, p)
81 except AttributeError:
81 except AttributeError:
82 raise error.HookLoadError(
82 raise error.HookLoadError(
83 _('%s hook is invalid: "%s" is not defined')
83 _('%s hook is invalid: "%s" is not defined')
84 % (hname, funcname))
84 % (hname, funcname))
85 if not callable(obj):
85 if not callable(obj):
86 raise error.HookLoadError(
86 raise error.HookLoadError(
87 _('%s hook is invalid: "%s" is not callable')
87 _('%s hook is invalid: "%s" is not callable')
88 % (hname, funcname))
88 % (hname, funcname))
89
89
90 ui.note(_("calling hook %s: %s\n") % (hname, funcname))
90 ui.note(_("calling hook %s: %s\n") % (hname, funcname))
91 starttime = util.timer()
91 starttime = util.timer()
92
92
93 try:
93 try:
94 r = obj(ui=ui, repo=repo, hooktype=htype, **args)
94 r = obj(ui=ui, repo=repo, hooktype=htype, **args)
95 except Exception as exc:
95 except Exception as exc:
96 if isinstance(exc, error.Abort):
96 if isinstance(exc, error.Abort):
97 ui.warn(_('error: %s hook failed: %s\n') %
97 ui.warn(_('error: %s hook failed: %s\n') %
98 (hname, exc.args[0]))
98 (hname, exc.args[0]))
99 else:
99 else:
100 ui.warn(_('error: %s hook raised an exception: '
100 ui.warn(_('error: %s hook raised an exception: '
101 '%s\n') % (hname, encoding.strtolocal(str(exc))))
101 '%s\n') % (hname, encoding.strtolocal(str(exc))))
102 if throw:
102 if throw:
103 raise
103 raise
104 if not ui.tracebackflag:
104 if not ui.tracebackflag:
105 ui.warn(_('(run with --traceback for stack trace)\n'))
105 ui.warn(_('(run with --traceback for stack trace)\n'))
106 ui.traceback()
106 ui.traceback()
107 return True, True
107 return True, True
108 finally:
108 finally:
109 duration = util.timer() - starttime
109 duration = util.timer() - starttime
110 ui.log('pythonhook', 'pythonhook-%s: %s finished in %0.2f seconds\n',
110 ui.log('pythonhook', 'pythonhook-%s: %s finished in %0.2f seconds\n',
111 htype, funcname, duration)
111 htype, funcname, duration)
112 if r:
112 if r:
113 if throw:
113 if throw:
114 raise error.HookAbort(_('%s hook failed') % hname)
114 raise error.HookAbort(_('%s hook failed') % hname)
115 ui.warn(_('warning: %s hook failed\n') % hname)
115 ui.warn(_('warning: %s hook failed\n') % hname)
116 return r, False
116 return r, False
117
117
118 def _exthook(ui, repo, htype, name, cmd, args, throw):
118 def _exthook(ui, repo, htype, name, cmd, args, throw):
119 ui.note(_("running hook %s: %s\n") % (name, cmd))
119 ui.note(_("running hook %s: %s\n") % (name, cmd))
120
120
121 starttime = util.timer()
121 starttime = util.timer()
122 env = {}
122 env = {}
123
123
124 # make in-memory changes visible to external process
124 # make in-memory changes visible to external process
125 if repo is not None:
125 if repo is not None:
126 tr = repo.currenttransaction()
126 tr = repo.currenttransaction()
127 repo.dirstate.write(tr)
127 repo.dirstate.write(tr)
128 if tr and tr.writepending():
128 if tr and tr.writepending():
129 env['HG_PENDING'] = repo.root
129 env['HG_PENDING'] = repo.root
130 env['HG_HOOKTYPE'] = htype
130 env['HG_HOOKTYPE'] = htype
131 env['HG_HOOKNAME'] = name
131 env['HG_HOOKNAME'] = name
132
132
133 for k, v in args.iteritems():
133 for k, v in args.iteritems():
134 if callable(v):
134 if callable(v):
135 v = v()
135 v = v()
136 if isinstance(v, dict):
136 if isinstance(v, dict):
137 # make the dictionary element order stable across Python
137 # make the dictionary element order stable across Python
138 # implementations
138 # implementations
139 v = ('{' +
139 v = ('{' +
140 ', '.join('%r: %r' % i for i in sorted(v.iteritems())) +
140 ', '.join('%r: %r' % i for i in sorted(v.iteritems())) +
141 '}')
141 '}')
142 env['HG_' + k.upper()] = v
142 env['HG_' + k.upper()] = v
143
143
144 if repo:
144 if repo:
145 cwd = repo.root
145 cwd = repo.root
146 else:
146 else:
147 cwd = pycompat.getcwd()
147 cwd = pycompat.getcwd()
148 r = ui.system(cmd, environ=env, cwd=cwd, blockedtag='exthook-%s' % (name,))
148 r = ui.system(cmd, environ=env, cwd=cwd, blockedtag='exthook-%s' % (name,))
149
149
150 duration = util.timer() - starttime
150 duration = util.timer() - starttime
151 ui.log('exthook', 'exthook-%s: %s finished in %0.2f seconds\n',
151 ui.log('exthook', 'exthook-%s: %s finished in %0.2f seconds\n',
152 name, cmd, duration)
152 name, cmd, duration)
153 if r:
153 if r:
154 desc, r = util.explainexit(r)
154 desc, r = util.explainexit(r)
155 if throw:
155 if throw:
156 raise error.HookAbort(_('%s hook %s') % (name, desc))
156 raise error.HookAbort(_('%s hook %s') % (name, desc))
157 ui.warn(_('warning: %s hook %s\n') % (name, desc))
157 ui.warn(_('warning: %s hook %s\n') % (name, desc))
158 return r
158 return r
159
159
160 # represent an untrusted hook command
160 # represent an untrusted hook command
161 _fromuntrusted = object()
161 _fromuntrusted = object()
162
162
163 def _allhooks(ui):
163 def _allhooks(ui):
164 """return a list of (hook-id, cmd) pairs sorted by priority"""
164 """return a list of (hook-id, cmd) pairs sorted by priority"""
165 hooks = _hookitems(ui)
165 hooks = _hookitems(ui)
166 # Be careful in this section, propagating the real commands from untrusted
166 # Be careful in this section, propagating the real commands from untrusted
167 # sources would create a security vulnerability, make sure anything altered
167 # sources would create a security vulnerability, make sure anything altered
168 # in that section uses "_fromuntrusted" as its command.
168 # in that section uses "_fromuntrusted" as its command.
169 untrustedhooks = _hookitems(ui, _untrusted=True)
169 untrustedhooks = _hookitems(ui, _untrusted=True)
170 for name, value in untrustedhooks.items():
170 for name, value in untrustedhooks.items():
171 trustedvalue = hooks.get(name, (None, None, name, _fromuntrusted))
171 trustedvalue = hooks.get(name, (None, None, name, _fromuntrusted))
172 if value != trustedvalue:
172 if value != trustedvalue:
173 (lp, lo, lk, lv) = trustedvalue
173 (lp, lo, lk, lv) = trustedvalue
174 hooks[name] = (lp, lo, lk, _fromuntrusted)
174 hooks[name] = (lp, lo, lk, _fromuntrusted)
175 # (end of the security sensitive section)
175 # (end of the security sensitive section)
176 return [(k, v) for p, o, k, v in sorted(hooks.values())]
176 return [(k, v) for p, o, k, v in sorted(hooks.values())]
177
177
178 def _hookitems(ui, _untrusted=False):
178 def _hookitems(ui, _untrusted=False):
179 """return all hooks items ready to be sorted"""
179 """return all hooks items ready to be sorted"""
180 hooks = {}
180 hooks = {}
181 for name, cmd in ui.configitems('hooks', untrusted=_untrusted):
181 for name, cmd in ui.configitems('hooks', untrusted=_untrusted):
182 if not name.startswith('priority'):
182 if not name.startswith('priority'):
183 priority = ui.configint('hooks', 'priority.%s' % name, 0)
183 priority = ui.configint('hooks', 'priority.%s' % name, 0)
184 hooks[name] = (-priority, len(hooks), name, cmd)
184 hooks[name] = (-priority, len(hooks), name, cmd)
185 return hooks
185 return hooks
186
186
187 _redirect = False
187 _redirect = False
188 def redirect(state):
188 def redirect(state):
189 global _redirect
189 global _redirect
190 _redirect = state
190 _redirect = state
191
191
192 def hook(ui, repo, htype, throw=False, **args):
192 def hook(ui, repo, htype, throw=False, **args):
193 if not ui.callhooks:
193 if not ui.callhooks:
194 return False
194 return False
195
195
196 hooks = []
196 hooks = []
197 for hname, cmd in _allhooks(ui):
197 for hname, cmd in _allhooks(ui):
198 if hname.split('.')[0] == htype and cmd:
198 if hname.split('.')[0] == htype and cmd:
199 hooks.append((hname, cmd))
199 hooks.append((hname, cmd))
200
200
201 res = runhooks(ui, repo, htype, hooks, throw=throw, **args)
201 res = runhooks(ui, repo, htype, hooks, throw=throw, **args)
202 r = False
202 r = False
203 for hname, cmd in hooks:
203 for hname, cmd in hooks:
204 r = res[hname][0] or r
204 r = res[hname][0] or r
205 return r
205 return r
206
206
207 def runhooks(ui, repo, htype, hooks, throw=False, **args):
207 def runhooks(ui, repo, htype, hooks, throw=False, **args):
208 args = pycompat.byteskwargs(args)
208 res = {}
209 res = {}
209 oldstdout = -1
210 oldstdout = -1
210
211
211 try:
212 try:
212 for hname, cmd in hooks:
213 for hname, cmd in hooks:
213 if oldstdout == -1 and _redirect:
214 if oldstdout == -1 and _redirect:
214 try:
215 try:
215 stdoutno = util.stdout.fileno()
216 stdoutno = util.stdout.fileno()
216 stderrno = util.stderr.fileno()
217 stderrno = util.stderr.fileno()
217 # temporarily redirect stdout to stderr, if possible
218 # temporarily redirect stdout to stderr, if possible
218 if stdoutno >= 0 and stderrno >= 0:
219 if stdoutno >= 0 and stderrno >= 0:
219 util.stdout.flush()
220 util.stdout.flush()
220 oldstdout = os.dup(stdoutno)
221 oldstdout = os.dup(stdoutno)
221 os.dup2(stderrno, stdoutno)
222 os.dup2(stderrno, stdoutno)
222 except (OSError, AttributeError):
223 except (OSError, AttributeError):
223 # files seem to be bogus, give up on redirecting (WSGI, etc)
224 # files seem to be bogus, give up on redirecting (WSGI, etc)
224 pass
225 pass
225
226
226 if cmd is _fromuntrusted:
227 if cmd is _fromuntrusted:
227 if throw:
228 if throw:
228 raise error.HookAbort(
229 raise error.HookAbort(
229 _('untrusted hook %s not executed') % hname,
230 _('untrusted hook %s not executed') % hname,
230 hint = _("see 'hg help config.trusted'"))
231 hint = _("see 'hg help config.trusted'"))
231 ui.warn(_('warning: untrusted hook %s not executed\n') % hname)
232 ui.warn(_('warning: untrusted hook %s not executed\n') % hname)
232 r = 1
233 r = 1
233 raised = False
234 raised = False
234 elif callable(cmd):
235 elif callable(cmd):
235 r, raised = _pythonhook(ui, repo, htype, hname, cmd, args,
236 r, raised = _pythonhook(ui, repo, htype, hname, cmd, args,
236 throw)
237 throw)
237 elif cmd.startswith('python:'):
238 elif cmd.startswith('python:'):
238 if cmd.count(':') >= 2:
239 if cmd.count(':') >= 2:
239 path, cmd = cmd[7:].rsplit(':', 1)
240 path, cmd = cmd[7:].rsplit(':', 1)
240 path = util.expandpath(path)
241 path = util.expandpath(path)
241 if repo:
242 if repo:
242 path = os.path.join(repo.root, path)
243 path = os.path.join(repo.root, path)
243 try:
244 try:
244 mod = extensions.loadpath(path, 'hghook.%s' % hname)
245 mod = extensions.loadpath(path, 'hghook.%s' % hname)
245 except Exception:
246 except Exception:
246 ui.write(_("loading %s hook failed:\n") % hname)
247 ui.write(_("loading %s hook failed:\n") % hname)
247 raise
248 raise
248 hookfn = getattr(mod, cmd)
249 hookfn = getattr(mod, cmd)
249 else:
250 else:
250 hookfn = cmd[7:].strip()
251 hookfn = cmd[7:].strip()
251 r, raised = _pythonhook(ui, repo, htype, hname, hookfn, args,
252 r, raised = _pythonhook(ui, repo, htype, hname, hookfn, args,
252 throw)
253 throw)
253 else:
254 else:
254 r = _exthook(ui, repo, htype, hname, cmd, args, throw)
255 r = _exthook(ui, repo, htype, hname, cmd, args, throw)
255 raised = False
256 raised = False
256
257
257 res[hname] = r, raised
258 res[hname] = r, raised
258
259
259 # The stderr is fully buffered on Windows when connected to a pipe.
260 # The stderr is fully buffered on Windows when connected to a pipe.
260 # A forcible flush is required to make small stderr data in the
261 # A forcible flush is required to make small stderr data in the
261 # remote side available to the client immediately.
262 # remote side available to the client immediately.
262 util.stderr.flush()
263 util.stderr.flush()
263 finally:
264 finally:
264 if _redirect and oldstdout >= 0:
265 if _redirect and oldstdout >= 0:
265 util.stdout.flush() # write hook output to stderr fd
266 util.stdout.flush() # write hook output to stderr fd
266 os.dup2(oldstdout, stdoutno)
267 os.dup2(oldstdout, stdoutno)
267 os.close(oldstdout)
268 os.close(oldstdout)
268
269
269 return res
270 return res
General Comments 0
You need to be logged in to leave comments. Login now