keyword.py
543 lines
| 20.4 KiB
| text/x-python
|
PythonLexer
/ hgext / keyword.py
Christian Ebert
|
r5815 | # keyword.py - $Keyword$ expansion for Mercurial | ||
# | ||||
Christian Ebert
|
r5831 | # Copyright 2007, 2008 Christian Ebert <blacktrash@gmx.net> | ||
Christian Ebert
|
r5815 | # | ||
# This software may be used and distributed according to the terms | ||||
# of the GNU General Public License, incorporated herein by reference. | ||||
# | ||||
# $Id$ | ||||
# | ||||
# Keyword expansion hack against the grain of a DSCM | ||||
# | ||||
# There are many good reasons why this is not needed in a distributed | ||||
# SCM, still it may be useful in very small projects based on single | ||||
# files (like LaTeX packages), that are mostly addressed to an audience | ||||
# not running a version control system. | ||||
# | ||||
# For in-depth discussion refer to | ||||
# <http://www.selenic.com/mercurial/wiki/index.cgi/KeywordPlan>. | ||||
# | ||||
# Keyword expansion is based on Mercurial's changeset template mappings. | ||||
# | ||||
# Binary files are not touched. | ||||
# | ||||
# Setup in hgrc: | ||||
# | ||||
# [extensions] | ||||
# # enable extension | ||||
# hgext.keyword = | ||||
# | ||||
# Files to act upon/ignore are specified in the [keyword] section. | ||||
# Customized keyword template mappings in the [keywordmaps] section. | ||||
# | ||||
# Run "hg help keyword" and "hg kwdemo" to get info on configuration. | ||||
'''keyword expansion in local repositories | ||||
This extension expands RCS/CVS-like or self-customized $Keywords$ | ||||
in tracked text files selected by your configuration. | ||||
Keywords are only expanded in local repositories and not stored in | ||||
the change history. The mechanism can be regarded as a convenience | ||||
for the current user or for archive distribution. | ||||
Configuration is done in the [keyword] and [keywordmaps] sections | ||||
of hgrc files. | ||||
Example: | ||||
[keyword] | ||||
# expand keywords in every python file except those matching "x*" | ||||
**.py = | ||||
x* = ignore | ||||
Note: the more specific you are in your filename patterns | ||||
the less you lose speed in huge repos. | ||||
For [keywordmaps] template mapping and expansion demonstration and | ||||
control run "hg kwdemo". | ||||
An additional date template filter {date|utcdate} is provided. | ||||
The default template mappings (view with "hg kwdemo -d") can be replaced | ||||
with customized keywords and templates. | ||||
Again, run "hg kwdemo" to control the results of your config changes. | ||||
Before changing/disabling active keywords, run "hg kwshrink" to avoid | ||||
the risk of inadvertedly storing expanded keywords in the change history. | ||||
To force expansion after enabling it, or a configuration change, run | ||||
"hg kwexpand". | ||||
Christian Ebert
|
r5884 | Also, when committing with the record extension or using mq's qrecord, be aware | ||
that keywords cannot be updated. Again, run "hg kwexpand" on the files in | ||||
question to update keyword expansions after all changes have been checked in. | ||||
Christian Ebert
|
r5815 | Expansions spanning more than one line and incremental expansions, | ||
like CVS' $Log$, are not supported. A keyword template map | ||||
"Log = {desc}" expands to the first line of the changeset description. | ||||
''' | ||||
Matt Mackall
|
r7216 | from mercurial import commands, cmdutil, dispatch, filelog, revlog, extensions | ||
Matt Mackall
|
r5976 | from mercurial import patch, localrepo, templater, templatefilters, util | ||
Christian Ebert
|
r6072 | from mercurial.hgweb import webcommands | ||
Joel Rosdahl
|
r6211 | from mercurial.node import nullid, hex | ||
Christian Ebert
|
r5815 | from mercurial.i18n import _ | ||
Christian Ebert
|
r6069 | import re, shutil, tempfile, time | ||
Christian Ebert
|
r5815 | |||
commands.optionalrepo += ' kwdemo' | ||||
Christian Ebert
|
r6024 | # hg commands that do not act on keywords | ||
Christian Ebert
|
r6667 | nokwcommands = ('add addremove annotate bundle copy export grep incoming init' | ||
Christian Ebert
|
r6867 | ' log outgoing push rename rollback tip verify' | ||
Christian Ebert
|
r6082 | ' convert email glog') | ||
Christian Ebert
|
r6024 | |||
Christian Ebert
|
r5961 | # hg commands that trigger expansion only when writing to working dir, | ||
# not when reading filelog, and unexpand when reading from working dir | ||||
Christian Ebert
|
r6933 | restricted = 'merge record resolve qfold qimport qnew qpush qrefresh qrecord' | ||
Christian Ebert
|
r5961 | |||
Christian Ebert
|
r5815 | def utcdate(date): | ||
'''Returns hgdate in cvs-like UTC format.''' | ||||
return time.strftime('%Y/%m/%d %H:%M:%S', time.gmtime(date[0])) | ||||
Christian Ebert
|
r6115 | # make keyword tools accessible | ||
Christian Ebert
|
r6502 | kwtools = {'templater': None, 'hgcmd': '', 'inc': [], 'exc': ['.hg*']} | ||
Christian Ebert
|
r6114 | |||
Christian Ebert
|
r5815 | |||
class kwtemplater(object): | ||||
''' | ||||
Sets up keyword templates, corresponding keyword regex, and | ||||
provides keyword substitution functions. | ||||
''' | ||||
templates = { | ||||
'Revision': '{node|short}', | ||||
'Author': '{author|user}', | ||||
'Date': '{date|utcdate}', | ||||
'RCSFile': '{file|basename},v', | ||||
'Source': '{root}/{file},v', | ||||
'Id': '{file|basename},v {node|short} {date|utcdate} {author|user}', | ||||
'Header': '{root}/{file},v {node|short} {date|utcdate} {author|user}', | ||||
} | ||||
Christian Ebert
|
r6502 | def __init__(self, ui, repo): | ||
Christian Ebert
|
r5815 | self.ui = ui | ||
self.repo = repo | ||||
Christian Ebert
|
r6502 | self.matcher = util.matcher(repo.root, | ||
inc=kwtools['inc'], exc=kwtools['exc'])[1] | ||||
Christian Ebert
|
r6115 | self.restrict = kwtools['hgcmd'] in restricted.split() | ||
Christian Ebert
|
r5815 | |||
kwmaps = self.ui.configitems('keywordmaps') | ||||
if kwmaps: # override default templates | ||||
Christian Ebert
|
r6504 | kwmaps = [(k, templater.parsestring(v, False)) | ||
Christian Ebert
|
r5815 | for (k, v) in kwmaps] | ||
self.templates = dict(kwmaps) | ||||
escaped = map(re.escape, self.templates.keys()) | ||||
kwpat = r'\$(%s)(: [^$\n\r]*? )??\$' % '|'.join(escaped) | ||||
self.re_kw = re.compile(kwpat) | ||||
Matt Mackall
|
r5976 | templatefilters.filters['utcdate'] = utcdate | ||
Christian Ebert
|
r5815 | self.ct = cmdutil.changeset_templater(self.ui, self.repo, | ||
False, '', False) | ||||
Christian Ebert
|
r6114 | def getnode(self, path, fnode): | ||
'''Derives changenode from file path and filenode.''' | ||||
# used by kwfilelog.read and kwexpand | ||||
Matt Mackall
|
r6738 | c = self.repo.filectx(path, fileid=fnode) | ||
Christian Ebert
|
r6114 | return c.node() | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6114 | def substitute(self, data, path, node, subfunc): | ||
'''Replaces keywords in data with expanded template.''' | ||||
Christian Ebert
|
r5815 | def kwsub(mobj): | ||
kw = mobj.group(1) | ||||
self.ct.use_template(self.templates[kw]) | ||||
self.ui.pushbuffer() | ||||
Christian Ebert
|
r6114 | self.ct.show(changenode=node, root=self.repo.root, file=path) | ||
Christian Ebert
|
r6023 | ekw = templatefilters.firstline(self.ui.popbuffer()) | ||
return '$%s: %s $' % (kw, ekw) | ||||
Christian Ebert
|
r5815 | return subfunc(kwsub, data) | ||
Christian Ebert
|
r6114 | def expand(self, path, node, data): | ||
Christian Ebert
|
r5815 | '''Returns data with keywords expanded.''' | ||
Bryan O'Sullivan
|
r6508 | if not self.restrict and self.matcher(path) and not util.binary(data): | ||
Christian Ebert
|
r6114 | changenode = self.getnode(path, node) | ||
return self.substitute(data, path, changenode, self.re_kw.sub) | ||||
return data | ||||
Matt Mackall
|
r6749 | def iskwfile(self, path, flagfunc): | ||
Christian Ebert
|
r6114 | '''Returns true if path matches [keyword] pattern | ||
and is not a symbolic link. | ||||
Caveat: localrepository._link fails on Windows.''' | ||||
Matt Mackall
|
r6749 | return self.matcher(path) and not 'l' in flagfunc(path) | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6505 | def overwrite(self, node, expand, files): | ||
Christian Ebert
|
r6114 | '''Overwrites selected files expanding/shrinking keywords.''' | ||
if node is not None: # commit | ||||
Matt Mackall
|
r6747 | ctx = self.repo[node] | ||
Matt Mackall
|
r6739 | mf = ctx.manifest() | ||
Christian Ebert
|
r6114 | files = [f for f in ctx.files() if f in mf] | ||
notify = self.ui.debug | ||||
else: # kwexpand/kwshrink | ||||
Matt Mackall
|
r6747 | ctx = self.repo['.'] | ||
Matt Mackall
|
r6750 | mf = ctx.manifest() | ||
Christian Ebert
|
r6114 | notify = self.ui.note | ||
Matt Mackall
|
r6749 | candidates = [f for f in files if self.iskwfile(f, ctx.flags)] | ||
Christian Ebert
|
r6114 | if candidates: | ||
self.restrict = True # do not expand when reading | ||||
action = expand and 'expanding' or 'shrinking' | ||||
for f in candidates: | ||||
fp = self.repo.file(f) | ||||
data = fp.read(mf[f]) | ||||
Bryan O'Sullivan
|
r6508 | if util.binary(data): | ||
Christian Ebert
|
r6114 | continue | ||
if expand: | ||||
changenode = node or self.getnode(f, mf[f]) | ||||
data, found = self.substitute(data, f, changenode, | ||||
self.re_kw.subn) | ||||
else: | ||||
found = self.re_kw.search(data) | ||||
if found: | ||||
notify(_('overwriting %s %s keywords\n') % (f, action)) | ||||
self.repo.wwrite(f, data, mf.flags(f)) | ||||
self.repo.dirstate.normal(f) | ||||
self.restrict = False | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6114 | def shrinktext(self, text): | ||
'''Unconditionally removes all keyword substitutions from text.''' | ||||
return self.re_kw.sub(r'$\1$', text) | ||||
def shrink(self, fname, text): | ||||
Christian Ebert
|
r5815 | '''Returns text with all keyword substitutions removed.''' | ||
Bryan O'Sullivan
|
r6508 | if self.matcher(fname) and not util.binary(text): | ||
Christian Ebert
|
r6114 | return self.shrinktext(text) | ||
return text | ||||
def shrinklines(self, fname, lines): | ||||
'''Returns lines with keyword substitutions removed.''' | ||||
if self.matcher(fname): | ||||
text = ''.join(lines) | ||||
Bryan O'Sullivan
|
r6508 | if not util.binary(text): | ||
Christian Ebert
|
r6114 | return self.shrinktext(text).splitlines(True) | ||
return lines | ||||
def wread(self, fname, data): | ||||
'''If in restricted mode returns data read from wdir with | ||||
keyword substitutions removed.''' | ||||
return self.restrict and self.shrink(fname, data) or data | ||||
Christian Ebert
|
r5815 | |||
class kwfilelog(filelog.filelog): | ||||
''' | ||||
Subclass of filelog to hook into its read, add, cmp methods. | ||||
Keywords are "stored" unexpanded, and processed on reading. | ||||
''' | ||||
Christian Ebert
|
r6503 | def __init__(self, opener, kwt, path): | ||
Christian Ebert
|
r5815 | super(kwfilelog, self).__init__(opener, path) | ||
Christian Ebert
|
r6503 | self.kwt = kwt | ||
Christian Ebert
|
r6114 | self.path = path | ||
Christian Ebert
|
r5815 | |||
def read(self, node): | ||||
'''Expands keywords when reading filelog.''' | ||||
data = super(kwfilelog, self).read(node) | ||||
Christian Ebert
|
r6115 | return self.kwt.expand(self.path, node, data) | ||
Christian Ebert
|
r5815 | |||
def add(self, text, meta, tr, link, p1=None, p2=None): | ||||
'''Removes keyword substitutions when adding to filelog.''' | ||||
Christian Ebert
|
r6115 | text = self.kwt.shrink(self.path, text) | ||
Christian Ebert
|
r6504 | return super(kwfilelog, self).add(text, meta, tr, link, p1, p2) | ||
Christian Ebert
|
r5815 | |||
def cmp(self, node, text): | ||||
'''Removes keyword substitutions for comparison.''' | ||||
Christian Ebert
|
r6115 | text = self.kwt.shrink(self.path, text) | ||
Christian Ebert
|
r5815 | if self.renamed(node): | ||
t2 = super(kwfilelog, self).read(node) | ||||
return t2 != text | ||||
return revlog.revlog.cmp(self, node, text) | ||||
Matt Mackall
|
r6760 | def _status(ui, repo, kwt, unknown, *pats, **opts): | ||
Christian Ebert
|
r5815 | '''Bails out if [keyword] configuration is not active. | ||
Returns status of working directory.''' | ||||
Christian Ebert
|
r6115 | if kwt: | ||
Matt Mackall
|
r6582 | matcher = cmdutil.match(repo, pats, opts) | ||
Matt Mackall
|
r6760 | return repo.status(match=matcher, unknown=unknown, clean=True) | ||
Christian Ebert
|
r5815 | if ui.configitems('keyword'): | ||
raise util.Abort(_('[keyword] patterns cannot match')) | ||||
raise util.Abort(_('no [keyword] patterns configured')) | ||||
def _kwfwrite(ui, repo, expand, *pats, **opts): | ||||
Christian Ebert
|
r6114 | '''Selects files and passes them to kwtemplater.overwrite.''' | ||
Christian Ebert
|
r6672 | if repo.dirstate.parents()[1] != nullid: | ||
raise util.Abort(_('outstanding uncommitted merge')) | ||||
Christian Ebert
|
r6115 | kwt = kwtools['templater'] | ||
Matt Mackall
|
r6760 | status = _status(ui, repo, kwt, False, *pats, **opts) | ||
modified, added, removed, deleted = status[:4] | ||||
Christian Ebert
|
r5815 | if modified or added or removed or deleted: | ||
Christian Ebert
|
r6672 | raise util.Abort(_('outstanding uncommitted changes')) | ||
Christian Ebert
|
r5815 | wlock = lock = None | ||
try: | ||||
wlock = repo.wlock() | ||||
lock = repo.lock() | ||||
Matt Mackall
|
r6760 | kwt.overwrite(None, expand, status[6]) | ||
Christian Ebert
|
r5815 | finally: | ||
del wlock, lock | ||||
def demo(ui, repo, *args, **opts): | ||||
'''print [keywordmaps] configuration and an expansion example | ||||
Show current, custom, or default keyword template maps | ||||
and their expansion. | ||||
Extend current configuration by specifying maps as arguments | ||||
and optionally by reading from an additional hgrc file. | ||||
Override current keyword template maps with "default" option. | ||||
''' | ||||
def demostatus(stat): | ||||
ui.status(_('\n\t%s\n') % stat) | ||||
def demoitems(section, items): | ||||
ui.write('[%s]\n' % section) | ||||
for k, v in items: | ||||
ui.write('%s = %s\n' % (k, v)) | ||||
msg = 'hg keyword config and expansion example' | ||||
kwstatus = 'current' | ||||
fn = 'demo.txt' | ||||
branchname = 'demobranch' | ||||
tmpdir = tempfile.mkdtemp('', 'kwdemo.') | ||||
ui.note(_('creating temporary repo at %s\n') % tmpdir) | ||||
Christian Ebert
|
r6504 | repo = localrepo.localrepository(ui, tmpdir, True) | ||
Christian Ebert
|
r5815 | ui.setconfig('keyword', fn, '') | ||
if args or opts.get('rcfile'): | ||||
kwstatus = 'custom' | ||||
if opts.get('rcfile'): | ||||
ui.readconfig(opts.get('rcfile')) | ||||
if opts.get('default'): | ||||
kwstatus = 'default' | ||||
kwmaps = kwtemplater.templates | ||||
if ui.configitems('keywordmaps'): | ||||
# override maps from optional rcfile | ||||
Christian Ebert
|
r5946 | for k, v in kwmaps.iteritems(): | ||
Christian Ebert
|
r5815 | ui.setconfig('keywordmaps', k, v) | ||
elif args: | ||||
# simulate hgrc parsing | ||||
rcmaps = ['[keywordmaps]\n'] + [a + '\n' for a in args] | ||||
fp = repo.opener('hgrc', 'w') | ||||
fp.writelines(rcmaps) | ||||
fp.close() | ||||
ui.readconfig(repo.join('hgrc')) | ||||
if not opts.get('default'): | ||||
kwmaps = dict(ui.configitems('keywordmaps')) or kwtemplater.templates | ||||
Christian Ebert
|
r6502 | uisetup(ui) | ||
Christian Ebert
|
r5815 | reposetup(ui, repo) | ||
for k, v in ui.configitems('extensions'): | ||||
if k.endswith('keyword'): | ||||
extension = '%s = %s' % (k, v) | ||||
break | ||||
demostatus('config using %s keyword template maps' % kwstatus) | ||||
ui.write('[extensions]\n%s\n' % extension) | ||||
demoitems('keyword', ui.configitems('keyword')) | ||||
Christian Ebert
|
r5946 | demoitems('keywordmaps', kwmaps.iteritems()) | ||
Christian Ebert
|
r5815 | keywords = '$' + '$\n$'.join(kwmaps.keys()) + '$\n' | ||
repo.wopener(fn, 'w').write(keywords) | ||||
repo.add([fn]) | ||||
path = repo.wjoin(fn) | ||||
ui.note(_('\n%s keywords written to %s:\n') % (kwstatus, path)) | ||||
ui.note(keywords) | ||||
ui.note('\nhg -R "%s" branch "%s"\n' % (tmpdir, branchname)) | ||||
# silence branch command if not verbose | ||||
quiet = ui.quiet | ||||
Christian Ebert
|
r5825 | ui.quiet = not ui.verbose | ||
Christian Ebert
|
r5815 | commands.branch(ui, repo, branchname) | ||
ui.quiet = quiet | ||||
for name, cmd in ui.configitems('hooks'): | ||||
if name.split('.', 1)[0].find('commit') > -1: | ||||
repo.ui.setconfig('hooks', name, '') | ||||
ui.note(_('unhooked all commit hooks\n')) | ||||
ui.note('hg -R "%s" ci -m "%s"\n' % (tmpdir, msg)) | ||||
repo.commit(text=msg) | ||||
format = ui.verbose and ' in %s' % path or '' | ||||
demostatus('%s keywords expanded%s' % (kwstatus, format)) | ||||
ui.write(repo.wread(fn)) | ||||
ui.debug(_('\nremoving temporary repo %s\n') % tmpdir) | ||||
shutil.rmtree(tmpdir, ignore_errors=True) | ||||
def expand(ui, repo, *pats, **opts): | ||||
'''expand keywords in working directory | ||||
Run after (re)enabling keyword expansion. | ||||
kwexpand refuses to run if given files contain local changes. | ||||
''' | ||||
# 3rd argument sets expansion to True | ||||
_kwfwrite(ui, repo, True, *pats, **opts) | ||||
def files(ui, repo, *pats, **opts): | ||||
'''print files currently configured for keyword expansion | ||||
Crosscheck which files in working directory are potential targets for | ||||
keyword expansion. | ||||
That is, files matched by [keyword] config patterns but not symlinks. | ||||
''' | ||||
Christian Ebert
|
r6115 | kwt = kwtools['templater'] | ||
Matt Mackall
|
r6760 | status = _status(ui, repo, kwt, opts.get('untracked'), *pats, **opts) | ||
Christian Ebert
|
r5815 | modified, added, removed, deleted, unknown, ignored, clean = status | ||
Matt Mackall
|
r6762 | files = util.sort(modified + added + clean + unknown) | ||
Matt Mackall
|
r6747 | wctx = repo[None] | ||
Matt Mackall
|
r6749 | kwfiles = [f for f in files if kwt.iskwfile(f, wctx.flags)] | ||
Christian Ebert
|
r5815 | cwd = pats and repo.getcwd() or '' | ||
kwfstats = not opts.get('ignore') and (('K', kwfiles),) or () | ||||
if opts.get('all') or opts.get('ignore'): | ||||
kwfstats += (('I', [f for f in files if f not in kwfiles]),) | ||||
for char, filenames in kwfstats: | ||||
format = (opts.get('all') or ui.verbose) and '%s %%s\n' % char or '%s\n' | ||||
for f in filenames: | ||||
ui.write(format % repo.pathto(f, cwd)) | ||||
def shrink(ui, repo, *pats, **opts): | ||||
'''revert expanded keywords in working directory | ||||
Run before changing/disabling active keywords | ||||
or if you experience problems with "hg import" or "hg merge". | ||||
kwshrink refuses to run if given files contain local changes. | ||||
''' | ||||
# 3rd argument sets expansion to False | ||||
_kwfwrite(ui, repo, False, *pats, **opts) | ||||
Christian Ebert
|
r6502 | def uisetup(ui): | ||
'''Collects [keyword] config in kwtools. | ||||
Monkeypatches dispatch._parse if needed.''' | ||||
for pat, opt in ui.configitems('keyword'): | ||||
if opt != 'ignore': | ||||
kwtools['inc'].append(pat) | ||||
else: | ||||
kwtools['exc'].append(pat) | ||||
if kwtools['inc']: | ||||
Matt Mackall
|
r7216 | def kwdispatch_parse(orig, ui, args): | ||
Christian Ebert
|
r6502 | '''Monkeypatch dispatch._parse to obtain running hg command.''' | ||
Matt Mackall
|
r7216 | cmd, func, args, options, cmdoptions = orig(ui, args) | ||
Christian Ebert
|
r6502 | kwtools['hgcmd'] = cmd | ||
return cmd, func, args, options, cmdoptions | ||||
Matt Mackall
|
r7216 | extensions.wrapfunction(dispatch, '_parse', kwdispatch_parse) | ||
Christian Ebert
|
r6502 | |||
Christian Ebert
|
r5815 | def reposetup(ui, repo): | ||
'''Sets up repo as kwrepo for keyword substitution. | ||||
Overrides file method to return kwfilelog instead of filelog | ||||
if file matches user configuration. | ||||
Wraps commit to overwrite configured files with updated | ||||
keyword substitutions. | ||||
Christian Ebert
|
r6503 | Monkeypatches patch and webcommands.''' | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6051 | try: | ||
Christian Ebert
|
r6502 | if (not repo.local() or not kwtools['inc'] | ||
or kwtools['hgcmd'] in nokwcommands.split() | ||||
Christian Ebert
|
r6069 | or '.hg' in util.splitpath(repo.root) | ||
Christian Ebert
|
r6051 | or repo._url.startswith('bundle:')): | ||
return | ||||
except AttributeError: | ||||
pass | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6502 | kwtools['templater'] = kwt = kwtemplater(ui, repo) | ||
Christian Ebert
|
r5815 | |||
class kwrepo(repo.__class__): | ||||
Christian Ebert
|
r6114 | def file(self, f): | ||
Christian Ebert
|
r5815 | if f[0] == '/': | ||
f = f[1:] | ||||
Christian Ebert
|
r6503 | return kwfilelog(self.sopener, kwt, f) | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r5884 | def wread(self, filename): | ||
data = super(kwrepo, self).wread(filename) | ||||
Christian Ebert
|
r6115 | return kwt.wread(filename, data) | ||
Christian Ebert
|
r5884 | |||
Christian Ebert
|
r5815 | def commit(self, files=None, text='', user=None, date=None, | ||
Matt Mackall
|
r6603 | match=None, force=False, force_editor=False, | ||
Christian Ebert
|
r6022 | p1=None, p2=None, extra={}, empty_ok=False): | ||
Christian Ebert
|
r5815 | wlock = lock = None | ||
_p1 = _p2 = None | ||||
try: | ||||
wlock = self.wlock() | ||||
lock = self.lock() | ||||
# store and postpone commit hooks | ||||
Christian Ebert
|
r5946 | commithooks = {} | ||
Christian Ebert
|
r5815 | for name, cmd in ui.configitems('hooks'): | ||
if name.split('.', 1)[0] == 'commit': | ||||
Christian Ebert
|
r5946 | commithooks[name] = cmd | ||
Christian Ebert
|
r5815 | ui.setconfig('hooks', name, None) | ||
if commithooks: | ||||
# store parents for commit hook environment | ||||
if p1 is None: | ||||
_p1, _p2 = repo.dirstate.parents() | ||||
else: | ||||
_p1, _p2 = p1, p2 or nullid | ||||
_p1 = hex(_p1) | ||||
if _p2 == nullid: | ||||
_p2 = '' | ||||
else: | ||||
_p2 = hex(_p2) | ||||
Christian Ebert
|
r6504 | n = super(kwrepo, self).commit(files, text, user, date, match, | ||
force, force_editor, p1, p2, | ||||
extra, empty_ok) | ||||
Christian Ebert
|
r5815 | |||
# restore commit hooks | ||||
Christian Ebert
|
r5946 | for name, cmd in commithooks.iteritems(): | ||
Christian Ebert
|
r5815 | ui.setconfig('hooks', name, cmd) | ||
Christian Ebert
|
r6504 | if n is not None: | ||
Christian Ebert
|
r6505 | kwt.overwrite(n, True, None) | ||
Christian Ebert
|
r6504 | repo.hook('commit', node=n, parent1=_p1, parent2=_p2) | ||
return n | ||||
Christian Ebert
|
r5815 | finally: | ||
del wlock, lock | ||||
Christian Ebert
|
r6503 | # monkeypatches | ||
Matt Mackall
|
r7216 | def kwpatchfile_init(orig, self, ui, fname, missing=False): | ||
Christian Ebert
|
r6503 | '''Monkeypatch/wrap patch.patchfile.__init__ to avoid | ||
rejects or conflicts due to expanded keywords in working dir.''' | ||||
Matt Mackall
|
r7216 | orig(self, ui, fname, missing) | ||
Christian Ebert
|
r6503 | # shrink keywords read from working dir | ||
self.lines = kwt.shrinklines(self.fname, self.lines) | ||||
Matt Mackall
|
r7216 | def kw_diff(orig, repo, node1=None, node2=None, match=None, | ||
Christian Ebert
|
r6503 | fp=None, changes=None, opts=None): | ||
'''Monkeypatch patch.diff to avoid expansion except when | ||||
comparing against working dir.''' | ||||
if node2 is not None: | ||||
kwt.matcher = util.never | ||||
Matt Mackall
|
r6747 | elif node1 is not None and node1 != repo['.'].node(): | ||
Christian Ebert
|
r6503 | kwt.restrict = True | ||
Matt Mackall
|
r7216 | orig(repo, node1, node2, match, fp, changes, opts) | ||
Christian Ebert
|
r6667 | |||
Matt Mackall
|
r7216 | def kwweb_skip(orig, web, req, tmpl): | ||
'''Wraps webcommands.x turning off keyword expansion.''' | ||||
Christian Ebert
|
r6503 | kwt.matcher = util.never | ||
Matt Mackall
|
r7216 | return orig(web, req, tmpl) | ||
Christian Ebert
|
r6503 | |||
Christian Ebert
|
r5815 | repo.__class__ = kwrepo | ||
Christian Ebert
|
r6503 | |||
Matt Mackall
|
r7216 | extensions.wrapfunction(patch.patchfile, '__init__', kwpatchfile_init) | ||
extensions.wrapfunction(patch, 'diff', kw_diff) | ||||
for c in 'annotate changeset rev filediff diff'.split(): | ||||
extensions.wrapfunction(webcommands, c, kwweb_skip) | ||||
Christian Ebert
|
r5815 | |||
cmdtable = { | ||||
'kwdemo': | ||||
(demo, | ||||
[('d', 'default', None, _('show default keyword template maps')), | ||||
('f', 'rcfile', [], _('read maps from rcfile'))], | ||||
_('hg kwdemo [-d] [-f RCFILE] [TEMPLATEMAP]...')), | ||||
'kwexpand': (expand, commands.walkopts, | ||||
_('hg kwexpand [OPTION]... [FILE]...')), | ||||
'kwfiles': | ||||
(files, | ||||
[('a', 'all', None, _('show keyword status flags of all files')), | ||||
('i', 'ignore', None, _('show files excluded from expansion')), | ||||
('u', 'untracked', None, _('additionally show untracked files')), | ||||
] + commands.walkopts, | ||||
_('hg kwfiles [OPTION]... [FILE]...')), | ||||
'kwshrink': (shrink, commands.walkopts, | ||||
_('hg kwshrink [OPTION]... [FILE]...')), | ||||
} | ||||