keyword.py
798 lines
| 28.5 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 | ||||
Matt Mackall
|
r26421 | # <https://mercurial-scm.org/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. | ||||
# | ||||
timeless
|
r29969 | # Run 'hg help keyword' and 'hg kwdemo' to get info on configuration. | ||
Christian Ebert
|
r5815 | |||
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 | ''' | ||
Christian Ebert
|
r28321 | |||
from __future__ import absolute_import | ||||
import os | ||||
import re | ||||
import tempfile | ||||
FUJIWARA Katsunori
|
r33067 | import weakref | ||
Christian Ebert
|
r28321 | |||
Yuya Nishihara
|
r29205 | from mercurial.i18n import _ | ||
Christian Ebert
|
r6072 | from mercurial.hgweb import webcommands | ||
Christian Ebert
|
r28321 | |||
from mercurial import ( | ||||
cmdutil, | ||||
context, | ||||
dispatch, | ||||
error, | ||||
extensions, | ||||
filelog, | ||||
localrepo, | ||||
match, | ||||
patch, | ||||
pathutil, | ||||
FUJIWARA Katsunori
|
r28694 | registrar, | ||
Christian Ebert
|
r28321 | scmutil, | ||
templatefilters, | ||||
util, | ||||
) | ||||
Christian Ebert
|
r5815 | |||
Martin Geisler
|
r14300 | cmdtable = {} | ||
Yuya Nishihara
|
r32337 | command = registrar.command(cmdtable) | ||
Augie Fackler
|
r29841 | # Note for extension authors: ONLY specify testedwith = 'ships-with-hg-core' for | ||
Augie Fackler
|
r25186 | # extensions which SHIP WITH MERCURIAL. Non-mainline extensions should | ||
# be specifying the version(s) of Mercurial they are tested with, or | ||||
# leave the attribute unspecified. | ||||
Augie Fackler
|
r29841 | testedwith = 'ships-with-hg-core' | ||
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 | |||
FUJIWARA Katsunori
|
r33065 | # webcommands that do not act on keywords | ||
nokwwebcommands = ('annotate changeset rev filediff diff comparison') | ||||
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' | ||||
} | ||||
FUJIWARA Katsunori
|
r28694 | templatefilter = registrar.templatefilter() | ||
Christian Ebert
|
r11213 | # date like in cvs' $Date | ||
FUJIWARA Katsunori
|
r28694 | @templatefilter('utcdate') | ||
Patrick Mezard
|
r13592 | def utcdate(text): | ||
FUJIWARA Katsunori
|
r28694 | '''Date. Returns a UTC-date in this format: "2009/08/18 11:00:13". | ||
Christian Ebert
|
r13633 | ''' | ||
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 | ||
FUJIWARA Katsunori
|
r28694 | @templatefilter('svnisodate') | ||
Patrick Mezard
|
r13592 | def svnisodate(text): | ||
FUJIWARA Katsunori
|
r28694 | '''Date. Returns a date in this format: "2009-08-18 13:00:13 | ||
Christian Ebert
|
r13633 | +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 | ||
FUJIWARA Katsunori
|
r28694 | @templatefilter('svnutcdate') | ||
Patrick Mezard
|
r13592 | def svnutcdate(text): | ||
FUJIWARA Katsunori
|
r28694 | '''Date. Returns a UTC-date in this format: "2009-08-18 | ||
Christian Ebert
|
r13633 | 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
|
r6115 | # make keyword tools accessible | ||
FUJIWARA Katsunori
|
r33070 | kwtools = {'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 | ||
FUJIWARA Katsunori
|
r33067 | self._repo = weakref.ref(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 | ||||
Yuya Nishihara
|
r24987 | self.templates = dict(kwmaps) | ||
Christian Ebert
|
r11214 | else: | ||
self.templates = _defaultkwmaps(self.ui) | ||||
Christian Ebert
|
r5815 | |||
FUJIWARA Katsunori
|
r33067 | @property | ||
def repo(self): | ||||
return self._repo() | ||||
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) | ||||
Yuya Nishihara
|
r32837 | ct = cmdutil.makelogtemplater(self.ui, self.repo, | ||
self.templates[kw]) | ||||
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) | ||||
Angel Ezquerra
|
r23879 | fp = self.repo.wvfs(f, "wb", atomictemp=True) | ||
Christian Ebert
|
r15083 | 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'): | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('[keyword] patterns cannot match')) | ||
raise error.Abort(_('no [keyword] patterns configured')) | ||||
Christian Ebert
|
r5815 | |||
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: | ||||
Pierre-Yves David
|
r26587 | raise error.Abort(_('outstanding uncommitted merge')) | ||
FUJIWARA Katsunori
|
r33070 | kwt = getattr(repo, '_keywordkwt', None) | ||
Bryan O'Sullivan
|
r27815 | with repo.wlock(): | ||
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: | ||
Pierre-Yves David
|
r26587 | raise error.Abort(_('outstanding uncommitted changes')) | ||
Martin von Zweigbergk
|
r22918 | kwt.overwrite(wctx, status.clean, True, expand) | ||
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) | ||
Christian Ebert
|
r29634 | if repo is None: | ||
baseui = ui | ||||
else: | ||||
baseui = repo.baseui | ||||
repo = localrepo.localrepository(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 | ||||
Christian Ebert
|
r28458 | rcmaps = '[keywordmaps]\n%s\n' % '\n'.join(args) | ||
repo.vfs.write('hgrc', rcmaps) | ||||
Pierre-Yves David
|
r31331 | ui.readconfig(repo.vfs.join('hgrc')) | ||
Christian Ebert
|
r9281 | 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) | ||
FUJIWARA Katsunori
|
r29397 | 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' | ||
Angel Ezquerra
|
r23879 | repo.wvfs.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) | ||
Bryan O'Sullivan
|
r27816 | with repo.wlock(): | ||
Christian Ebert
|
r20113 | repo.dirstate.setbranch('demobranch') | ||
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
|
r24905 | repo.wvfs.rmtree(repo.root) | ||
Christian Ebert
|
r5815 | |||
Gregory Szorc
|
r21784 | @command('kwexpand', | ||
Yuya Nishihara
|
r32375 | cmdutil.walkopts, | ||
Gregory Szorc
|
r21784 | _('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')), | ||||
Yuya Nishihara
|
r32375 | ] + cmdutil.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 | ''' | ||
FUJIWARA Katsunori
|
r33070 | kwt = getattr(repo, '_keywordkwt', None) | ||
Christian Ebert
|
r14835 | wctx = repo[None] | ||
status = _status(ui, repo, wctx, kwt, *pats, **opts) | ||||
Jordi Gutiérrez Hermoso
|
r24306 | if pats: | ||
cwd = repo.getcwd() | ||||
else: | ||||
cwd = '' | ||||
Christian Ebert
|
r9493 | 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', | ||
Yuya Nishihara
|
r32375 | cmdutil.walkopts, | ||
Gregory Szorc
|
r21784 | _('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) | ||||
FUJIWARA Katsunori
|
r33071 | # monkeypatches | ||
def kwpatchfile_init(orig, self, ui, gp, backend, store, eolmode=None): | ||||
'''Monkeypatch/wrap patch.patchfile.__init__ to avoid | ||||
rejects or conflicts due to expanded keywords in working dir.''' | ||||
orig(self, ui, gp, backend, store, eolmode) | ||||
kwt = getattr(getattr(backend, 'repo', None), '_keywordkwt', None) | ||||
if kwt: | ||||
# shrink keywords read from working dir | ||||
self.lines = kwt.shrinklines(self.fname, self.lines) | ||||
def kwdiff(orig, repo, *args, **kwargs): | ||||
'''Monkeypatch patch.diff to avoid expansion.''' | ||||
kwt = getattr(repo, '_keywordkwt', None) | ||||
if kwt: | ||||
restrict = kwt.restrict | ||||
kwt.restrict = True | ||||
try: | ||||
for chunk in orig(repo, *args, **kwargs): | ||||
yield chunk | ||||
finally: | ||||
if kwt: | ||||
kwt.restrict = restrict | ||||
def kwweb_skip(orig, web, req, tmpl): | ||||
'''Wraps webcommands.x turning off keyword expansion.''' | ||||
kwt = getattr(web.repo, '_keywordkwt', None) | ||||
if kwt: | ||||
origmatch = kwt.match | ||||
kwt.match = util.never | ||||
try: | ||||
for chunk in orig(web, req, tmpl): | ||||
yield chunk | ||||
finally: | ||||
if kwt: | ||||
kwt.match = origmatch | ||||
def kw_amend(orig, ui, repo, commitfunc, old, extra, pats, opts): | ||||
'''Wraps cmdutil.amend expanding keywords after amend.''' | ||||
kwt = getattr(repo, '_keywordkwt', None) | ||||
if kwt is None: | ||||
return orig(ui, repo, commitfunc, old, extra, pats, opts) | ||||
with repo.wlock(): | ||||
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 | ||||
def kw_copy(orig, ui, repo, pats, opts, rename=False): | ||||
'''Wraps cmdutil.copy so that copy/rename destinations do not | ||||
contain expanded keywords. | ||||
Note that the source of a regular file destination may also be a | ||||
symlink: | ||||
hg cp sym x -> x is symlink | ||||
cp sym x; hg cp -A sym x -> x is file (maybe expanded keywords) | ||||
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.''' | ||||
kwt = getattr(repo, '_keywordkwt', None) | ||||
if kwt is None: | ||||
return orig(ui, repo, pats, opts, rename) | ||||
with repo.wlock(): | ||||
orig(ui, repo, pats, opts, rename) | ||||
if opts.get('dry_run'): | ||||
return | ||||
wctx = repo[None] | ||||
cwd = repo.getcwd() | ||||
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): | ||||
source = pathutil.canonpath(repo.root, cwd, | ||||
os.path.realpath(source)) | ||||
return kwt.match(source) | ||||
candidates = [f for f in repo.dirstate.copies() if | ||||
'l' not in wctx.flags(f) and haskwsource(f)] | ||||
kwt.overwrite(wctx, candidates, False, False) | ||||
def kw_dorecord(orig, ui, repo, commitfunc, *pats, **opts): | ||||
'''Wraps record.dorecord expanding keywords after recording.''' | ||||
kwt = getattr(repo, '_keywordkwt', None) | ||||
if kwt is None: | ||||
return orig(ui, repo, commitfunc, *pats, **opts) | ||||
with repo.wlock(): | ||||
# record returns 0 even when nothing has changed | ||||
# therefore compare nodes before and after | ||||
kwt.postcommit = True | ||||
ctx = repo['.'] | ||||
wstatus = ctx.status() | ||||
ret = orig(ui, repo, commitfunc, *pats, **opts) | ||||
recctx = repo['.'] | ||||
if ctx != recctx: | ||||
modified, added = _preselect(wstatus, recctx.files()) | ||||
kwt.restrict = False | ||||
kwt.overwrite(recctx, modified, False, True) | ||||
kwt.overwrite(recctx, added, False, True, True) | ||||
kwt.restrict = True | ||||
return ret | ||||
def kwfilectx_cmp(orig, self, fctx): | ||||
if fctx._customcmp: | ||||
return fctx.cmp(self) | ||||
kwt = getattr(self._repo, '_keywordkwt', None) | ||||
if kwt is None: | ||||
return orig(self, fctx) | ||||
# keyword affects data size, comparing wdir and filelog size does | ||||
# not make sense | ||||
if (fctx._filenode is None and | ||||
(self._repo._encodefilterpats or | ||||
kwt.match(fctx.path()) and 'l' not in fctx.flags() or | ||||
self.size() - 4 == fctx.size()) or | ||||
self.size() == fctx.size()): | ||||
return self._filelog.cmp(self._filenode, fctx.data()) | ||||
return True | ||||
Christian Ebert
|
r5815 | |||
Christian Ebert
|
r6502 | def uisetup(ui): | ||
FUJIWARA Katsunori
|
r33071 | ''' Monkeypatches dispatch._parse to retrieve user command. | ||
Overrides file method to return kwfilelog instead of filelog | ||||
if file matches user configuration. | ||||
Wraps commit to overwrite configured files with updated | ||||
keyword substitutions. | ||||
Monkeypatches patch and webcommands.''' | ||||
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 | |||
FUJIWARA Katsunori
|
r33071 | extensions.wrapfunction(context.filectx, 'cmp', kwfilectx_cmp) | ||
extensions.wrapfunction(patch.patchfile, '__init__', kwpatchfile_init) | ||||
extensions.wrapfunction(patch, 'diff', kwdiff) | ||||
extensions.wrapfunction(cmdutil, 'amend', kw_amend) | ||||
extensions.wrapfunction(cmdutil, 'copy', kw_copy) | ||||
extensions.wrapfunction(cmdutil, 'dorecord', kw_dorecord) | ||||
for c in nokwwebcommands.split(): | ||||
extensions.wrapfunction(webcommands, c, kwweb_skip) | ||||
Christian Ebert
|
r5815 | def reposetup(ui, repo): | ||
FUJIWARA Katsunori
|
r33071 | '''Sets up repo as kwrepo for keyword substitution.''' | ||
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 | ||||
FUJIWARA Katsunori
|
r33070 | 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:] | ||||
Angel Ezquerra
|
r23878 | return kwfilelog(self.svfs, 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
|
r32935 | with self.wlock(): | ||
origrestrict = kwt.restrict | ||||
try: | ||||
if not dryrun: | ||||
changed = self['.'].files() | ||||
ret = super(kwrepo, self).rollback(dryrun, force) | ||||
if not dryrun: | ||||
ctx = self['.'] | ||||
modified, added = _preselect(ctx.status(), changed) | ||||
kwt.restrict = False | ||||
kwt.overwrite(ctx, modified, True, True) | ||||
kwt.overwrite(ctx, added, True, False) | ||||
return ret | ||||
finally: | ||||
kwt.restrict = origrestrict | ||||
Christian Ebert
|
r12498 | |||
FUJIWARA Katsunori
|
r33067 | repo.__class__ = kwrepo | ||
repo._keywordkwt = kwt | ||||