keyword.py
749 lines
| 27.9 KiB
| text/x-python
|
PythonLexer
/ hgext / keyword.py
Christian Ebert
|
r5815 | # keyword.py - $Keyword$ expansion for Mercurial | ||
# | ||||
Christian Ebert
|
r23723 | # Copyright 2007-2015 Christian Ebert <blacktrash@gmx.net> | ||
Christian Ebert
|
r5815 | # | ||
Martin Geisler
|
r8225 | # This software may be used and distributed according to the terms of the | ||
Matt Mackall
|
r10263 | # GNU General Public License version 2 or any later version. | ||
Christian Ebert
|
r5815 | # | ||
# $Id$ | ||||
# | ||||
Mads Kiilerich
|
r17424 | # Keyword expansion hack against the grain of a Distributed SCM | ||
Christian Ebert
|
r5815 | # | ||
# 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 | ||||
Martin Geisler
|
r7993 | # files (like LaTeX packages), that are mostly addressed to an | ||
# audience not running a version control system. | ||||
Christian Ebert
|
r5815 | # | ||
# For in-depth discussion refer to | ||||
Dirkjan Ochtman
|
r8936 | # <http://mercurial.selenic.com/wiki/KeywordPlan>. | ||
Christian Ebert
|
r5815 | # | ||
# Keyword expansion is based on Mercurial's changeset template mappings. | ||||
# | ||||
# Binary files are not touched. | ||||
# | ||||
# 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. | ||||
Cédric Duval
|
r8894 | '''expand keywords in tracked files | ||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | This extension expands RCS/CVS-like or self-customized $Keywords$ in | ||
tracked text files selected by your configuration. | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | 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. | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r12203 | Keywords expand to the changeset data pertaining to the latest change | ||
relative to the working directory parent of each file. | ||||
Christian Ebert
|
r11214 | Configuration is done in the [keyword], [keywordset] and [keywordmaps] | ||
sections of hgrc files. | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9157 | Example:: | ||
Christian Ebert
|
r5815 | |||
[keyword] | ||||
# expand keywords in every python file except those matching "x*" | ||||
**.py = | ||||
x* = ignore | ||||
Christian Ebert
|
r11214 | [keywordset] | ||
# prefer svn- over cvs-like default keywordmaps | ||||
svn = True | ||||
Christian Ebert
|
r12390 | .. note:: | ||
Simon Heimberg
|
r19997 | |||
Christian Ebert
|
r12390 | The more specific you are in your filename patterns the less you | ||
lose speed in huge repositories. | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | For [keywordmaps] template mapping and expansion demonstration and | ||
Martin Geisler
|
r10973 | control run :hg:`kwdemo`. See :hg:`help templates` for a list of | ||
Christian Ebert
|
r9307 | available templates and filters. | ||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r13885 | Three additional date template filters are provided: | ||
Christian Ebert
|
r11213 | |||
Martin Geisler
|
r13885 | :``utcdate``: "2006/09/18 15:13:13" | ||
:``svnutcdate``: "2006-09-18 15:13:13Z" | ||||
:``svnisodate``: "2006-09-18 08:13:13 -700 (Mon, 18 Sep 2006)" | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r10973 | The default template mappings (view with :hg:`kwdemo -d`) can be | ||
replaced with customized keywords and templates. Again, run | ||||
Christian Ebert
|
r13025 | :hg:`kwdemo` to control the results of your configuration changes. | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r13270 | Before changing/disabling active keywords, you must run :hg:`kwshrink` | ||
to avoid storing expanded keywords in the change history. | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | To force expansion after enabling it, or a configuration change, run | ||
Martin Geisler
|
r10973 | :hg:`kwexpand`. | ||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | 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. | ||||
Christian Ebert
|
r5815 | ''' | ||
Nicolas Dumazet
|
r12709 | from mercurial import commands, context, cmdutil, dispatch, filelog, extensions | ||
Christian Ebert
|
r12628 | from mercurial import localrepo, match, patch, templatefilters, templater, util | ||
Augie Fackler
|
r20033 | from mercurial import scmutil, pathutil | ||
Christian Ebert
|
r6072 | from mercurial.hgweb import webcommands | ||
Christian Ebert
|
r5815 | from mercurial.i18n import _ | ||
Christian Ebert
|
r21981 | import os, re, tempfile | ||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r14300 | cmdtable = {} | ||
command = cmdutil.command(cmdtable) | ||||
Augie Fackler
|
r16743 | testedwith = 'internal' | ||
Martin Geisler
|
r14300 | |||
Christian Ebert
|
r6024 | # hg commands that do not act on keywords | ||
Christian Ebert
|
r12626 | nokwcommands = ('add addremove annotate bundle export grep incoming init log' | ||
' outgoing push tip verify 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 | ||||
FUJIWARA Katsunori
|
r21703 | restricted = ('merge kwexpand kwshrink record qrecord resolve transplant' | ||
FUJIWARA Katsunori
|
r21708 | ' unshelve rebase graft backout histedit fetch') | ||
Christian Ebert
|
r5961 | |||
Christian Ebert
|
r11168 | # names of extensions using dorecord | ||
recordextensions = 'record' | ||||
Christian Ebert
|
r11045 | |||
Christian Ebert
|
r13078 | colortable = { | ||
'kwfiles.enabled': 'green bold', | ||||
Christian Ebert
|
r13079 | 'kwfiles.deleted': 'cyan bold underline', | ||
Christian Ebert
|
r13078 | 'kwfiles.enabledunknown': 'green', | ||
'kwfiles.ignored': 'bold', | ||||
'kwfiles.ignoredunknown': 'none' | ||||
} | ||||
Christian Ebert
|
r11213 | # date like in cvs' $Date | ||
Patrick Mezard
|
r13592 | def utcdate(text): | ||
Christian Ebert
|
r13633 | ''':utcdate: Date. Returns a UTC-date in this format: "2009/08/18 11:00:13". | ||
''' | ||||
Christian Ebert
|
r17755 | return util.datestr((util.parsedate(text)[0], 0), '%Y/%m/%d %H:%M:%S') | ||
Christian Ebert
|
r11213 | # date like in svn's $Date | ||
Patrick Mezard
|
r13592 | def svnisodate(text): | ||
Christian Ebert
|
r13633 | ''':svnisodate: Date. Returns a date in this format: "2009-08-18 13:00:13 | ||
+0200 (Tue, 18 Aug 2009)". | ||||
''' | ||||
Patrick Mezard
|
r13592 | return util.datestr(text, '%Y-%m-%d %H:%M:%S %1%2 (%a, %d %b %Y)') | ||
Christian Ebert
|
r11213 | # date like in svn's $Id | ||
Patrick Mezard
|
r13592 | def svnutcdate(text): | ||
Christian Ebert
|
r13633 | ''':svnutcdate: Date. Returns a UTC-date in this format: "2009-08-18 | ||
11:00:13Z". | ||||
''' | ||||
Christian Ebert
|
r17755 | return util.datestr((util.parsedate(text)[0], 0), '%Y-%m-%d %H:%M:%SZ') | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r13634 | templatefilters.filters.update({'utcdate': utcdate, | ||
'svnisodate': svnisodate, | ||||
'svnutcdate': svnutcdate}) | ||||
Christian Ebert
|
r6115 | # make keyword tools accessible | ||
Christian Ebert
|
r11678 | kwtools = {'templater': None, 'hgcmd': ''} | ||
Christian Ebert
|
r6114 | |||
Christian Ebert
|
r11214 | def _defaultkwmaps(ui): | ||
'''Returns default keywordmaps according to keywordset configuration.''' | ||||
Christian Ebert
|
r5815 | templates = { | ||
'Revision': '{node|short}', | ||||
'Author': '{author|user}', | ||||
Christian Ebert
|
r11214 | } | ||
kwsets = ({ | ||||
Christian Ebert
|
r5815 | 'Date': '{date|utcdate}', | ||
Christian Ebert
|
r9943 | 'RCSfile': '{file|basename},v', | ||
timeless
|
r9950 | 'RCSFile': '{file|basename},v', # kept for backwards compatibility | ||
# with hg-keyword | ||||
Christian Ebert
|
r5815 | '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
|
r11214 | }, { | ||
'Date': '{date|svnisodate}', | ||||
'Id': '{file|basename},v {node|short} {date|svnutcdate} {author|user}', | ||||
'LastChangedRevision': '{node|short}', | ||||
'LastChangedBy': '{author|user}', | ||||
'LastChangedDate': '{date|svnisodate}', | ||||
}) | ||||
templates.update(kwsets[ui.configbool('keywordset', 'svn')]) | ||||
return templates | ||||
Christian Ebert
|
r12625 | def _shrinktext(text, subfunc): | ||
'''Helper for keyword expansion removal in text. | ||||
Depending on subfunc also returns number of substitutions.''' | ||||
return subfunc(r'$\1$', text) | ||||
Christian Ebert
|
r12723 | def _preselect(wstatus, changed): | ||
Mads Kiilerich
|
r17424 | '''Retrieves modified and added files from a working directory state | ||
Christian Ebert
|
r12723 | and returns the subset of each contained in given changed files | ||
retrieved from a change context.''' | ||||
Martin von Zweigbergk
|
r22918 | modified = [f for f in wstatus.modified if f in changed] | ||
added = [f for f in wstatus.added if f in changed] | ||||
Christian Ebert
|
r12723 | return modified, added | ||
Christian Ebert
|
r12625 | |||
Christian Ebert
|
r11214 | class kwtemplater(object): | ||
''' | ||||
Sets up keyword templates, corresponding keyword regex, and | ||||
provides keyword substitution functions. | ||||
''' | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r11678 | def __init__(self, ui, repo, inc, exc): | ||
Christian Ebert
|
r5815 | self.ui = ui | ||
self.repo = repo | ||||
Christian Ebert
|
r11678 | self.match = match.match(repo.root, '', [], inc, exc) | ||
Christian Ebert
|
r6115 | self.restrict = kwtools['hgcmd'] in restricted.split() | ||
Christian Ebert
|
r16809 | self.postcommit = False | ||
Christian Ebert
|
r5815 | |||
kwmaps = self.ui.configitems('keywordmaps') | ||||
if kwmaps: # override default templates | ||||
Christian Ebert
|
r9081 | self.templates = dict((k, templater.parsestring(v, False)) | ||
for k, v in kwmaps) | ||||
Christian Ebert
|
r11214 | else: | ||
self.templates = _defaultkwmaps(self.ui) | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r12926 | @util.propertycache | ||
def escape(self): | ||||
'''Returns bar-separated and escaped keywords.''' | ||||
return '|'.join(map(re.escape, self.templates.keys())) | ||||
@util.propertycache | ||||
def rekw(self): | ||||
'''Returns regex for unexpanded keywords.''' | ||||
return re.compile(r'\$(%s)\$' % self.escape) | ||||
@util.propertycache | ||||
def rekwexp(self): | ||||
'''Returns regex for expanded keywords.''' | ||||
return re.compile(r'\$(%s): [^$\n\r]*? \$' % self.escape) | ||||
Dirkjan Ochtman
|
r7375 | def substitute(self, data, path, ctx, subfunc): | ||
Christian Ebert
|
r6114 | '''Replaces keywords in data with expanded template.''' | ||
Christian Ebert
|
r5815 | def kwsub(mobj): | ||
kw = mobj.group(1) | ||||
Matt Mackall
|
r20668 | ct = cmdutil.changeset_templater(self.ui, self.repo, False, None, | ||
Matt Mackall
|
r20667 | self.templates[kw], '', False) | ||
Christian Ebert
|
r5815 | self.ui.pushbuffer() | ||
Christian Ebert
|
r10894 | ct.show(ctx, 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
|
r12920 | def linkctx(self, path, fileid): | ||
'''Similar to filelog.linkrev, but returns a changectx.''' | ||||
return self.repo.filectx(path, fileid=fileid).changectx() | ||||
Christian Ebert
|
r6114 | def expand(self, path, node, data): | ||
Christian Ebert
|
r5815 | '''Returns data with keywords expanded.''' | ||
Christian Ebert
|
r8638 | if not self.restrict and self.match(path) and not util.binary(data): | ||
Christian Ebert
|
r12920 | ctx = self.linkctx(path, node) | ||
Christian Ebert
|
r12926 | return self.substitute(data, path, ctx, self.rekw.sub) | ||
Christian Ebert
|
r6114 | return data | ||
Christian Ebert
|
r12627 | def iskwfile(self, cand, ctx): | ||
'''Returns subset of candidates which are configured for keyword | ||||
Christian Ebert
|
r15324 | expansion but are not symbolic links.''' | ||
Brodie Rao
|
r16686 | return [f for f in cand if self.match(f) and 'l' not in ctx.flags(f)] | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r12685 | def overwrite(self, ctx, candidates, lookup, expand, rekw=False): | ||
Christian Ebert
|
r6114 | '''Overwrites selected files expanding/shrinking keywords.''' | ||
Christian Ebert
|
r16809 | if self.restrict or lookup or self.postcommit: # exclude kw_copy | ||
Christian Ebert
|
r12627 | candidates = self.iskwfile(candidates, ctx) | ||
Christian Ebert
|
r12625 | if not candidates: | ||
return | ||||
Christian Ebert
|
r12844 | kwcmd = self.restrict and lookup # kwexpand/kwshrink | ||
Christian Ebert
|
r12625 | if self.restrict or expand and lookup: | ||
Christian Ebert
|
r11350 | mf = ctx.manifest() | ||
Christian Ebert
|
r15030 | if self.restrict or rekw: | ||
re_kw = self.rekw | ||||
else: | ||||
re_kw = self.rekwexp | ||||
if expand: | ||||
msg = _('overwriting %s expanding keywords\n') | ||||
else: | ||||
msg = _('overwriting %s shrinking keywords\n') | ||||
Christian Ebert
|
r12625 | for f in candidates: | ||
if self.restrict: | ||||
data = self.repo.file(f).read(mf[f]) | ||||
else: | ||||
data = self.repo.wread(f) | ||||
if util.binary(data): | ||||
continue | ||||
if expand: | ||||
Christian Ebert
|
r23622 | parents = ctx.parents() | ||
Christian Ebert
|
r12625 | if lookup: | ||
Christian Ebert
|
r15083 | ctx = self.linkctx(f, mf[f]) | ||
Christian Ebert
|
r23622 | elif self.restrict and len(parents) > 1: | ||
# merge commit | ||||
# in case of conflict f is in modified state during | ||||
# merge, even if f does not differ from f in parent | ||||
for p in parents: | ||||
if f in p and not p[f].cmp(ctx[f]): | ||||
ctx = p[f].changectx() | ||||
break | ||||
Christian Ebert
|
r15083 | data, found = self.substitute(data, f, ctx, re_kw.subn) | ||
Christian Ebert
|
r12625 | elif self.restrict: | ||
Christian Ebert
|
r12926 | found = re_kw.search(data) | ||
Christian Ebert
|
r12625 | else: | ||
Christian Ebert
|
r12926 | data, found = _shrinktext(data, re_kw.subn) | ||
Christian Ebert
|
r12625 | if found: | ||
self.ui.note(msg % f) | ||||
Christian Ebert
|
r15083 | fp = self.repo.wopener(f, "wb", atomictemp=True) | ||
fp.write(data) | ||||
fp.close() | ||||
Christian Ebert
|
r12844 | if kwcmd: | ||
Christian Ebert
|
r12625 | self.repo.dirstate.normal(f) | ||
Christian Ebert
|
r16809 | elif self.postcommit: | ||
Christian Ebert
|
r12625 | self.repo.dirstate.normallookup(f) | ||
Christian Ebert
|
r6114 | |||
def shrink(self, fname, text): | ||||
Christian Ebert
|
r5815 | '''Returns text with all keyword substitutions removed.''' | ||
Christian Ebert
|
r8638 | if self.match(fname) and not util.binary(text): | ||
Christian Ebert
|
r12926 | return _shrinktext(text, self.rekwexp.sub) | ||
Christian Ebert
|
r6114 | return text | ||
def shrinklines(self, fname, lines): | ||||
'''Returns lines with keyword substitutions removed.''' | ||||
Christian Ebert
|
r8638 | if self.match(fname): | ||
Christian Ebert
|
r6114 | text = ''.join(lines) | ||
Bryan O'Sullivan
|
r6508 | if not util.binary(text): | ||
Christian Ebert
|
r12926 | return _shrinktext(text, self.rekwexp.sub).splitlines(True) | ||
Christian Ebert
|
r6114 | return lines | ||
def wread(self, fname, data): | ||||
'''If in restricted mode returns data read from wdir with | ||||
keyword substitutions removed.''' | ||||
Christian Ebert
|
r15030 | if self.restrict: | ||
return self.shrink(fname, data) | ||||
return 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
|
r12628 | if self.renamed(node): | ||
return data | ||||
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
|
r12628 | return super(kwfilelog, self).cmp(node, text) | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r14835 | def _status(ui, repo, wctx, kwt, *pats, **opts): | ||
Christian Ebert
|
r5815 | '''Bails out if [keyword] configuration is not active. | ||
Returns status of working directory.''' | ||||
Christian Ebert
|
r6115 | if kwt: | ||
Christian Ebert
|
r14835 | return repo.status(match=scmutil.match(wctx, pats, opts), clean=True, | ||
Christian Ebert
|
r10652 | unknown=opts.get('unknown') or opts.get('all')) | ||
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
|
r11320 | wctx = repo[None] | ||
if len(wctx.parents()) > 1: | ||||
Christian Ebert
|
r6672 | raise util.Abort(_('outstanding uncommitted merge')) | ||
Christian Ebert
|
r6115 | kwt = kwtools['templater'] | ||
Christian Ebert
|
r10604 | wlock = repo.wlock() | ||
Christian Ebert
|
r5815 | try: | ||
Christian Ebert
|
r14835 | status = _status(ui, repo, wctx, kwt, *pats, **opts) | ||
Martin von Zweigbergk
|
r22918 | if status.modified or status.added or status.removed or status.deleted: | ||
Christian Ebert
|
r10604 | raise util.Abort(_('outstanding uncommitted changes')) | ||
Martin von Zweigbergk
|
r22918 | kwt.overwrite(wctx, status.clean, True, expand) | ||
Christian Ebert
|
r5815 | finally: | ||
Christian Ebert
|
r10604 | wlock.release() | ||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r14300 | @command('kwdemo', | ||
[('d', 'default', None, _('show default keyword template maps')), | ||||
('f', 'rcfile', '', | ||||
_('read maps from rcfile'), _('FILE'))], | ||||
Gregory Szorc
|
r21776 | _('hg kwdemo [-d] [-f RCFILE] [TEMPLATEMAP]...'), | ||
optionalrepo=True) | ||||
Christian Ebert
|
r5815 | def demo(ui, repo, *args, **opts): | ||
'''print [keywordmaps] configuration and an expansion example | ||||
Martin Geisler
|
r7993 | Show current, custom, or default keyword template maps and their | ||
timeless
|
r8763 | expansions. | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r9281 | Extend the current configuration by specifying maps as arguments | ||
and using -f/--rcfile to source an external hgrc file. | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r9281 | Use -d/--default to disable current configuration. | ||
Christian Ebert
|
r9307 | |||
Martin Geisler
|
r11193 | See :hg:`help templates` for information on templates and filters. | ||
Christian Ebert
|
r5815 | ''' | ||
def demoitems(section, items): | ||||
ui.write('[%s]\n' % section) | ||||
Martin Geisler
|
r9942 | for k, v in sorted(items): | ||
Christian Ebert
|
r5815 | ui.write('%s = %s\n' % (k, v)) | ||
fn = 'demo.txt' | ||||
tmpdir = tempfile.mkdtemp('', 'kwdemo.') | ||||
Martin Geisler
|
r8027 | ui.note(_('creating temporary repository at %s\n') % tmpdir) | ||
Simon Heimberg
|
r18825 | repo = localrepo.localrepository(repo.baseui, tmpdir, True) | ||
Mads Kiilerich
|
r20790 | ui.setconfig('keyword', fn, '', 'keyword') | ||
Christian Ebert
|
r13298 | svn = ui.configbool('keywordset', 'svn') | ||
# explicitly set keywordset for demo output | ||||
Mads Kiilerich
|
r20790 | ui.setconfig('keywordset', 'svn', svn, 'keyword') | ||
Christian Ebert
|
r9281 | |||
uikwmaps = ui.configitems('keywordmaps') | ||||
Christian Ebert
|
r5815 | if args or opts.get('rcfile'): | ||
Christian Ebert
|
r9281 | ui.status(_('\n\tconfiguration using custom keyword template maps\n')) | ||
if uikwmaps: | ||||
ui.status(_('\textending current template maps\n')) | ||||
if opts.get('default') or not uikwmaps: | ||||
Christian Ebert
|
r13298 | if svn: | ||
ui.status(_('\toverriding default svn keywordset\n')) | ||||
else: | ||||
ui.status(_('\toverriding default cvs keywordset\n')) | ||||
Christian Ebert
|
r9281 | if opts.get('rcfile'): | ||
ui.readconfig(opts.get('rcfile')) | ||||
if 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')) | ||||
kwmaps = dict(ui.configitems('keywordmaps')) | ||||
elif opts.get('default'): | ||||
Christian Ebert
|
r13298 | if svn: | ||
ui.status(_('\n\tconfiguration using default svn keywordset\n')) | ||||
else: | ||||
ui.status(_('\n\tconfiguration using default cvs keywordset\n')) | ||||
Christian Ebert
|
r11214 | kwmaps = _defaultkwmaps(ui) | ||
Christian Ebert
|
r9281 | if uikwmaps: | ||
ui.status(_('\tdisabling current template maps\n')) | ||||
Christian Ebert
|
r5946 | for k, v in kwmaps.iteritems(): | ||
Mads Kiilerich
|
r20790 | ui.setconfig('keywordmaps', k, v, 'keyword') | ||
Christian Ebert
|
r9281 | else: | ||
ui.status(_('\n\tconfiguration using current keyword template maps\n')) | ||||
Christian Ebert
|
r15030 | if uikwmaps: | ||
kwmaps = dict(uikwmaps) | ||||
else: | ||||
kwmaps = _defaultkwmaps(ui) | ||||
Christian Ebert
|
r9281 | |||
Christian Ebert
|
r6502 | uisetup(ui) | ||
Christian Ebert
|
r5815 | reposetup(ui, repo) | ||
Christian Ebert
|
r10714 | ui.write('[extensions]\nkeyword =\n') | ||
Christian Ebert
|
r5815 | demoitems('keyword', ui.configitems('keyword')) | ||
Christian Ebert
|
r13298 | demoitems('keywordset', ui.configitems('keywordset')) | ||
Christian Ebert
|
r5946 | demoitems('keywordmaps', kwmaps.iteritems()) | ||
Martin Geisler
|
r9942 | keywords = '$' + '$\n$'.join(sorted(kwmaps.keys())) + '$\n' | ||
Dan Villiom Podlaski Christiansen
|
r14168 | repo.wopener.write(fn, keywords) | ||
Dirkjan Ochtman
|
r11303 | repo[None].add([fn]) | ||
Christian Ebert
|
r10713 | ui.note(_('\nkeywords written to %s:\n') % fn) | ||
Christian Ebert
|
r5815 | ui.note(keywords) | ||
Christian Ebert
|
r20113 | wlock = repo.wlock() | ||
try: | ||||
repo.dirstate.setbranch('demobranch') | ||||
finally: | ||||
wlock.release() | ||||
Christian Ebert
|
r5815 | for name, cmd in ui.configitems('hooks'): | ||
if name.split('.', 1)[0].find('commit') > -1: | ||||
Mads Kiilerich
|
r20790 | repo.ui.setconfig('hooks', name, '', 'keyword') | ||
Christian Ebert
|
r10499 | msg = _('hg keyword configuration and expansion example') | ||
Simon Heimberg
|
r20240 | ui.note(("hg ci -m '%s'\n" % msg)) | ||
Christian Ebert
|
r5815 | repo.commit(text=msg) | ||
Christian Ebert
|
r9281 | ui.status(_('\n\tkeywords expanded\n')) | ||
Christian Ebert
|
r5815 | ui.write(repo.wread(fn)) | ||
Christian Ebert
|
r23722 | for root, dirs, files in os.walk(tmpdir): | ||
Christian Ebert
|
r21981 | for f in files: | ||
Christian Ebert
|
r23722 | util.unlinkpath(repo.vfs.reljoin(root, f)) | ||
Christian Ebert
|
r5815 | |||
Gregory Szorc
|
r21784 | @command('kwexpand', | ||
commands.walkopts, | ||||
_('hg kwexpand [OPTION]... [FILE]...'), | ||||
inferrepo=True) | ||||
Christian Ebert
|
r5815 | def expand(ui, repo, *pats, **opts): | ||
timeless
|
r8763 | '''expand keywords in the working directory | ||
Christian Ebert
|
r5815 | |||
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) | ||||
Martin Geisler
|
r14300 | @command('kwfiles', | ||
[('A', 'all', None, _('show keyword status flags of all files')), | ||||
('i', 'ignore', None, _('show files excluded from expansion')), | ||||
('u', 'unknown', None, _('only show unknown (not tracked) files')), | ||||
] + commands.walkopts, | ||||
Gregory Szorc
|
r21784 | _('hg kwfiles [OPTION]... [FILE]...'), | ||
inferrepo=True) | ||||
Christian Ebert
|
r5815 | def files(ui, repo, *pats, **opts): | ||
Christian Ebert
|
r8957 | '''show files configured for keyword expansion | ||
Christian Ebert
|
r8950 | |||
Martin Geisler
|
r9264 | List which files in the working directory are matched by the | ||
[keyword] configuration patterns. | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r9264 | Useful to prevent inadvertent keyword expansion and to speed up | ||
execution by including only files that are actual candidates for | ||||
expansion. | ||||
Christian Ebert
|
r8950 | |||
Martin Geisler
|
r10973 | See :hg:`help keyword` on how to construct patterns both for | ||
Martin Geisler
|
r9264 | inclusion and exclusion of files. | ||
Christian Ebert
|
r8957 | |||
Christian Ebert
|
r9494 | With -A/--all and -v/--verbose the codes used to show the status | ||
Martin Geisler
|
r9264 | of files are:: | ||
Christian Ebert
|
r9195 | |||
K = keyword expansion candidate | ||||
Christian Ebert
|
r9491 | k = keyword expansion candidate (not tracked) | ||
Christian Ebert
|
r9195 | I = ignored | ||
Christian Ebert
|
r9491 | i = ignored (not tracked) | ||
Christian Ebert
|
r5815 | ''' | ||
Christian Ebert
|
r6115 | kwt = kwtools['templater'] | ||
Christian Ebert
|
r14835 | wctx = repo[None] | ||
status = _status(ui, repo, wctx, kwt, *pats, **opts) | ||||
Christian Ebert
|
r9493 | cwd = pats and repo.getcwd() or '' | ||
files = [] | ||||
Christian Ebert
|
r10652 | if not opts.get('unknown') or opts.get('all'): | ||
Martin von Zweigbergk
|
r22918 | files = sorted(status.modified + status.added + status.clean) | ||
Christian Ebert
|
r12627 | kwfiles = kwt.iskwfile(files, wctx) | ||
Martin von Zweigbergk
|
r22918 | kwdeleted = kwt.iskwfile(status.deleted, wctx) | ||
kwunknown = kwt.iskwfile(status.unknown, wctx) | ||||
Christian Ebert
|
r9493 | if not opts.get('ignore') or opts.get('all'): | ||
Christian Ebert
|
r13079 | showfiles = kwfiles, kwdeleted, kwunknown | ||
Christian Ebert
|
r9493 | else: | ||
Christian Ebert
|
r13079 | showfiles = [], [], [] | ||
Christian Ebert
|
r5815 | if opts.get('all') or opts.get('ignore'): | ||
Christian Ebert
|
r9493 | showfiles += ([f for f in files if f not in kwfiles], | ||
Martin von Zweigbergk
|
r22918 | [f for f in status.unknown if f not in kwunknown]) | ||
Christian Ebert
|
r13079 | kwlabels = 'enabled deleted enabledunknown ignored ignoredunknown'.split() | ||
Christian Ebert
|
r17057 | kwstates = zip(kwlabels, 'K!kIi', showfiles) | ||
fm = ui.formatter('kwfiles', opts) | ||||
fmt = '%.0s%s\n' | ||||
if opts.get('all') or ui.verbose: | ||||
fmt = '%s %s\n' | ||||
for kwstate, char, filenames in kwstates: | ||||
label = 'kwfiles.' + kwstate | ||||
Christian Ebert
|
r5815 | for f in filenames: | ||
Christian Ebert
|
r17057 | fm.startitem() | ||
fm.write('kwstatus path', fmt, char, | ||||
repo.pathto(f, cwd), label=label) | ||||
fm.end() | ||||
Christian Ebert
|
r5815 | |||
Gregory Szorc
|
r21784 | @command('kwshrink', | ||
commands.walkopts, | ||||
_('hg kwshrink [OPTION]... [FILE]...'), | ||||
inferrepo=True) | ||||
Christian Ebert
|
r5815 | def shrink(ui, repo, *pats, **opts): | ||
timeless
|
r8763 | '''revert expanded keywords in the working directory | ||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r13270 | Must be run before changing/disabling active keywords. | ||
Christian Ebert
|
r5815 | |||
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): | ||
Christian Ebert
|
r11678 | ''' Monkeypatches dispatch._parse to retrieve user command.''' | ||
Christian Ebert
|
r6502 | |||
Christian Ebert
|
r11678 | def kwdispatch_parse(orig, ui, args): | ||
'''Monkeypatch dispatch._parse to obtain running hg command.''' | ||||
cmd, func, args, options, cmdoptions = orig(ui, args) | ||||
kwtools['hgcmd'] = cmd | ||||
return cmd, func, args, options, cmdoptions | ||||
Christian Ebert
|
r6502 | |||
Christian Ebert
|
r11678 | 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 | |||
Matt Mackall
|
r7853 | try: | ||
Christian Ebert
|
r11678 | if (not repo.local() or kwtools['hgcmd'] in nokwcommands.split() | ||
Matt Mackall
|
r7853 | or '.hg' in util.splitpath(repo.root) | ||
or repo._url.startswith('bundle:')): | ||||
return | ||||
except AttributeError: | ||||
pass | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r11678 | inc, exc = [], ['.hg*'] | ||
for pat, opt in ui.configitems('keyword'): | ||||
if opt != 'ignore': | ||||
inc.append(pat) | ||||
else: | ||||
exc.append(pat) | ||||
if not inc: | ||||
return | ||||
kwtools['templater'] = kwt = kwtemplater(ui, repo, inc, exc) | ||||
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
|
r9096 | def commit(self, *args, **opts): | ||
Christian Ebert
|
r8996 | # use custom commitctx for user commands | ||
# other extensions can still wrap repo.commitctx directly | ||||
Christian Ebert
|
r9096 | self.commitctx = self.kwcommitctx | ||
try: | ||||
Christian Ebert
|
r10495 | return super(kwrepo, self).commit(*args, **opts) | ||
Christian Ebert
|
r9096 | finally: | ||
del self.commitctx | ||||
Christian Ebert
|
r8996 | |||
def kwcommitctx(self, ctx, error=False): | ||||
Christian Ebert
|
r10604 | n = super(kwrepo, self).commitctx(ctx, error) | ||
# no lock needed, only called from repo.commit() which already locks | ||||
Christian Ebert
|
r16809 | if not kwt.postcommit: | ||
Christian Ebert
|
r12625 | restrict = kwt.restrict | ||
kwt.restrict = True | ||||
Christian Ebert
|
r11320 | kwt.overwrite(self[n], sorted(ctx.added() + ctx.modified()), | ||
Christian Ebert
|
r12625 | False, True) | ||
kwt.restrict = restrict | ||||
Christian Ebert
|
r10604 | return n | ||
Christian Ebert
|
r5815 | |||
Greg Ward
|
r15183 | def rollback(self, dryrun=False, force=False): | ||
Christian Ebert
|
r12723 | wlock = self.wlock() | ||
Christian Ebert
|
r12498 | try: | ||
if not dryrun: | ||||
Christian Ebert
|
r12604 | changed = self['.'].files() | ||
Greg Ward
|
r15183 | ret = super(kwrepo, self).rollback(dryrun, force) | ||
Christian Ebert
|
r12498 | if not dryrun: | ||
ctx = self['.'] | ||||
Martin von Zweigbergk
|
r23079 | modified, added = _preselect(ctx.status(), changed) | ||
Christian Ebert
|
r12723 | kwt.overwrite(ctx, modified, True, True) | ||
Christian Ebert
|
r12625 | kwt.overwrite(ctx, added, True, False) | ||
Christian Ebert
|
r12498 | return ret | ||
finally: | ||||
wlock.release() | ||||
Christian Ebert
|
r6503 | # monkeypatches | ||
Patrick Mezard
|
r14566 | def kwpatchfile_init(orig, self, ui, gp, backend, store, eolmode=None): | ||
Christian Ebert
|
r6503 | '''Monkeypatch/wrap patch.patchfile.__init__ to avoid | ||
rejects or conflicts due to expanded keywords in working dir.''' | ||||
Patrick Mezard
|
r14566 | orig(self, ui, gp, backend, store, eolmode) | ||
Christian Ebert
|
r6503 | # shrink keywords read from working dir | ||
self.lines = kwt.shrinklines(self.fname, self.lines) | ||||
Dirkjan Ochtman
|
r7308 | def kw_diff(orig, repo, node1=None, node2=None, match=None, changes=None, | ||
Martin Geisler
|
r12167 | opts=None, prefix=''): | ||
Christian Ebert
|
r12497 | '''Monkeypatch patch.diff to avoid expansion.''' | ||
kwt.restrict = True | ||||
Martin Geisler
|
r12167 | return orig(repo, node1, node2, match, changes, opts, prefix) | ||
Christian Ebert
|
r6667 | |||
Matt Mackall
|
r7216 | def kwweb_skip(orig, web, req, tmpl): | ||
'''Wraps webcommands.x turning off keyword expansion.''' | ||||
Christian Ebert
|
r8638 | kwt.match = util.never | ||
Matt Mackall
|
r7216 | return orig(web, req, tmpl) | ||
Christian Ebert
|
r6503 | |||
Christian Ebert
|
r16810 | def kw_amend(orig, ui, repo, commitfunc, old, extra, pats, opts): | ||
'''Wraps cmdutil.amend expanding keywords after amend.''' | ||||
wlock = repo.wlock() | ||||
try: | ||||
kwt.postcommit = True | ||||
newid = orig(ui, repo, commitfunc, old, extra, pats, opts) | ||||
if newid != old.node(): | ||||
ctx = repo[newid] | ||||
kwt.restrict = True | ||||
kwt.overwrite(ctx, ctx.files(), False, True) | ||||
kwt.restrict = False | ||||
return newid | ||||
finally: | ||||
wlock.release() | ||||
Christian Ebert
|
r12626 | def kw_copy(orig, ui, repo, pats, opts, rename=False): | ||
'''Wraps cmdutil.copy so that copy/rename destinations do not | ||||
contain expanded keywords. | ||||
Christian Ebert
|
r13069 | Note that the source of a regular file destination may also be a | ||
symlink: | ||||
Christian Ebert
|
r12626 | hg cp sym x -> x is symlink | ||
cp sym x; hg cp -A sym x -> x is file (maybe expanded keywords) | ||||
Christian Ebert
|
r13069 | For the latter we have to follow the symlink to find out whether its | ||
target is configured for expansion and we therefore must unexpand the | ||||
keywords in the destination.''' | ||||
Christian Ebert
|
r16811 | wlock = repo.wlock() | ||
try: | ||||
orig(ui, repo, pats, opts, rename) | ||||
if opts.get('dry_run'): | ||||
return | ||||
wctx = repo[None] | ||||
cwd = repo.getcwd() | ||||
Christian Ebert
|
r13069 | |||
Christian Ebert
|
r16811 | def haskwsource(dest): | ||
'''Returns true if dest is a regular file and configured for | ||||
expansion or a symlink which points to a file configured for | ||||
expansion. ''' | ||||
source = repo.dirstate.copied(dest) | ||||
if 'l' in wctx.flags(source): | ||||
Augie Fackler
|
r20033 | source = pathutil.canonpath(repo.root, cwd, | ||
Christian Ebert
|
r16811 | os.path.realpath(source)) | ||
return kwt.match(source) | ||||
Christian Ebert
|
r13069 | |||
Christian Ebert
|
r16811 | candidates = [f for f in repo.dirstate.copies() if | ||
'l' not in wctx.flags(f) and haskwsource(f)] | ||||
kwt.overwrite(wctx, candidates, False, False) | ||||
finally: | ||||
wlock.release() | ||||
Christian Ebert
|
r12626 | |||
Christian Ebert
|
r11045 | def kw_dorecord(orig, ui, repo, commitfunc, *pats, **opts): | ||
'''Wraps record.dorecord expanding keywords after recording.''' | ||||
wlock = repo.wlock() | ||||
try: | ||||
# record returns 0 even when nothing has changed | ||||
# therefore compare nodes before and after | ||||
Christian Ebert
|
r16809 | kwt.postcommit = True | ||
Christian Ebert
|
r11045 | ctx = repo['.'] | ||
Martin von Zweigbergk
|
r23079 | wstatus = ctx.status() | ||
Christian Ebert
|
r11045 | ret = orig(ui, repo, commitfunc, *pats, **opts) | ||
Christian Ebert
|
r12630 | recctx = repo['.'] | ||
if ctx != recctx: | ||||
Christian Ebert
|
r12723 | modified, added = _preselect(wstatus, recctx.files()) | ||
Christian Ebert
|
r12625 | kwt.restrict = False | ||
Christian Ebert
|
r12685 | kwt.overwrite(recctx, modified, False, True) | ||
kwt.overwrite(recctx, added, False, True, True) | ||||
Christian Ebert
|
r12625 | kwt.restrict = True | ||
Christian Ebert
|
r11045 | return ret | ||
finally: | ||||
wlock.release() | ||||
Nicolas Dumazet
|
r12709 | def kwfilectx_cmp(orig, self, fctx): | ||
# keyword affects data size, comparing wdir and filelog size does | ||||
# not make sense | ||||
Christian Ebert
|
r12732 | if (fctx._filerev is None and | ||
(self._repo._encodefilterpats or | ||||
Brodie Rao
|
r16686 | kwt.match(fctx.path()) and 'l' not in fctx.flags() or | ||
Christian Ebert
|
r15871 | self.size() - 4 == fctx.size()) or | ||
Christian Ebert
|
r12732 | self.size() == fctx.size()): | ||
return self._filelog.cmp(self._filenode, fctx.data()) | ||||
return True | ||||
Nicolas Dumazet
|
r12709 | extensions.wrapfunction(context.filectx, 'cmp', kwfilectx_cmp) | ||
Matt Mackall
|
r7216 | extensions.wrapfunction(patch.patchfile, '__init__', kwpatchfile_init) | ||
Christian Ebert
|
r12497 | extensions.wrapfunction(patch, 'diff', kw_diff) | ||
Christian Ebert
|
r16810 | extensions.wrapfunction(cmdutil, 'amend', kw_amend) | ||
Christian Ebert
|
r12626 | extensions.wrapfunction(cmdutil, 'copy', kw_copy) | ||
Matt Mackall
|
r7216 | for c in 'annotate changeset rev filediff diff'.split(): | ||
extensions.wrapfunction(webcommands, c, kwweb_skip) | ||||
Christian Ebert
|
r11168 | for name in recordextensions.split(): | ||
try: | ||||
record = extensions.find(name) | ||||
extensions.wrapfunction(record, 'dorecord', kw_dorecord) | ||||
except KeyError: | ||||
pass | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r13299 | repo.__class__ = kwrepo | ||