|
|
Windows needs ';' as a file separator in an environment variable, and MSYS
|
|
|
doesn't automatically convert it in every case.
|
|
|
|
|
|
#if windows
|
|
|
$ path_list_var() {
|
|
|
> echo $1 | sed 's/:/;/'
|
|
|
> }
|
|
|
#else
|
|
|
$ path_list_var() {
|
|
|
> echo $1
|
|
|
> }
|
|
|
#endif
|
|
|
|
|
|
|
|
|
hide outer repo
|
|
|
$ hg init
|
|
|
|
|
|
Invalid syntax: no value
|
|
|
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> novaluekey
|
|
|
> EOF
|
|
|
$ hg showconfig
|
|
|
config error at $TESTTMP/.hg/hgrc:1: novaluekey
|
|
|
[30]
|
|
|
|
|
|
Invalid syntax: no key
|
|
|
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> =nokeyvalue
|
|
|
> EOF
|
|
|
$ hg showconfig
|
|
|
config error at $TESTTMP/.hg/hgrc:1: =nokeyvalue
|
|
|
[30]
|
|
|
|
|
|
Test hint about invalid syntax from leading white space
|
|
|
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> key=value
|
|
|
> EOF
|
|
|
$ hg showconfig
|
|
|
config error at $TESTTMP/.hg/hgrc:1: unexpected leading whitespace: key=value
|
|
|
[30]
|
|
|
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> [section]
|
|
|
> key=value
|
|
|
> EOF
|
|
|
$ hg showconfig
|
|
|
config error at $TESTTMP/.hg/hgrc:1: unexpected leading whitespace: [section]
|
|
|
[30]
|
|
|
|
|
|
Reset hgrc
|
|
|
|
|
|
$ echo > .hg/hgrc
|
|
|
|
|
|
Test case sensitive configuration
|
|
|
|
|
|
$ cat <<EOF >> $HGRCPATH
|
|
|
> [Section]
|
|
|
> KeY = Case Sensitive
|
|
|
> key = lower case
|
|
|
> EOF
|
|
|
|
|
|
$ hg showconfig Section
|
|
|
Section.KeY=Case Sensitive
|
|
|
Section.key=lower case
|
|
|
|
|
|
$ hg showconfig Section -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "Section.KeY",
|
|
|
"source": "*.hgrc:*", (glob)
|
|
|
"value": "Case Sensitive"
|
|
|
},
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "Section.key",
|
|
|
"source": "*.hgrc:*", (glob)
|
|
|
"value": "lower case"
|
|
|
}
|
|
|
]
|
|
|
$ hg showconfig Section.KeY -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "Section.KeY",
|
|
|
"source": "*.hgrc:*", (glob)
|
|
|
"value": "Case Sensitive"
|
|
|
}
|
|
|
]
|
|
|
$ hg showconfig -Tjson | tail -7
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "*", (glob)
|
|
|
"source": "*", (glob)
|
|
|
"value": "*" (glob)
|
|
|
}
|
|
|
]
|
|
|
|
|
|
Test config default of various types:
|
|
|
|
|
|
{"defaultvalue": ""} for -T'json(defaultvalue)' looks weird, but that's
|
|
|
how the templater works. Unknown keywords are evaluated to "".
|
|
|
|
|
|
dynamicdefault
|
|
|
|
|
|
$ hg config --config alias.foo= alias -Tjson
|
|
|
[
|
|
|
{
|
|
|
"name": "alias.foo",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config alias.foo= alias -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": ""}
|
|
|
]
|
|
|
$ hg config --config alias.foo= alias -T'{defaultvalue}\n'
|
|
|
|
|
|
|
|
|
null
|
|
|
|
|
|
$ hg config --config auth.cookiefile= auth -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "auth.cookiefile",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config auth.cookiefile= auth -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": null}
|
|
|
]
|
|
|
$ hg config --config auth.cookiefile= auth -T'{defaultvalue}\n'
|
|
|
|
|
|
|
|
|
false
|
|
|
|
|
|
$ hg config --config commands.commit.post-status= commands -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": false,
|
|
|
"name": "commands.commit.post-status",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config commands.commit.post-status= commands -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": false}
|
|
|
]
|
|
|
$ hg config --config commands.commit.post-status= commands -T'{defaultvalue}\n'
|
|
|
False
|
|
|
|
|
|
true
|
|
|
|
|
|
$ hg config --config format.dotencode= format.dotencode -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": true,
|
|
|
"name": "format.dotencode",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config format.dotencode= format.dotencode -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": true}
|
|
|
]
|
|
|
$ hg config --config format.dotencode= format.dotencode -T'{defaultvalue}\n'
|
|
|
True
|
|
|
|
|
|
bytes
|
|
|
|
|
|
$ hg config --config commands.resolve.mark-check= commands -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": "none",
|
|
|
"name": "commands.resolve.mark-check",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config commands.resolve.mark-check= commands -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": "none"}
|
|
|
]
|
|
|
$ hg config --config commands.resolve.mark-check= commands -T'{defaultvalue}\n'
|
|
|
none
|
|
|
|
|
|
empty list
|
|
|
|
|
|
$ hg config --config commands.show.aliasprefix= commands -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": [],
|
|
|
"name": "commands.show.aliasprefix",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config commands.show.aliasprefix= commands -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": []}
|
|
|
]
|
|
|
$ hg config --config commands.show.aliasprefix= commands -T'{defaultvalue}\n'
|
|
|
|
|
|
|
|
|
nonempty list
|
|
|
|
|
|
$ hg config --config progress.format= progress -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": ["topic", "bar", "number", "estimate"],
|
|
|
"name": "progress.format",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config progress.format= progress -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": ["topic", "bar", "number", "estimate"]}
|
|
|
]
|
|
|
$ hg config --config progress.format= progress -T'{defaultvalue}\n'
|
|
|
topic bar number estimate
|
|
|
|
|
|
int
|
|
|
|
|
|
$ hg config --config profiling.freq= profiling -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": 1000,
|
|
|
"name": "profiling.freq",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config profiling.freq= profiling -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": 1000}
|
|
|
]
|
|
|
$ hg config --config profiling.freq= profiling -T'{defaultvalue}\n'
|
|
|
1000
|
|
|
|
|
|
float
|
|
|
|
|
|
$ hg config --config profiling.showmax= profiling -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": 0.999,
|
|
|
"name": "profiling.showmax",
|
|
|
"source": "--config",
|
|
|
"value": ""
|
|
|
}
|
|
|
]
|
|
|
$ hg config --config profiling.showmax= profiling -T'json(defaultvalue)'
|
|
|
[
|
|
|
{"defaultvalue": 0.999}
|
|
|
]
|
|
|
$ hg config --config profiling.showmax= profiling -T'{defaultvalue}\n'
|
|
|
0.999
|
|
|
|
|
|
Test empty config source:
|
|
|
|
|
|
$ cat <<EOF > emptysource.py
|
|
|
> def reposetup(ui, repo):
|
|
|
> ui.setconfig(b'empty', b'source', b'value')
|
|
|
> EOF
|
|
|
$ cp .hg/hgrc .hg/hgrc.orig
|
|
|
$ cat <<EOF >> .hg/hgrc
|
|
|
> [extensions]
|
|
|
> emptysource = `pwd`/emptysource.py
|
|
|
> EOF
|
|
|
|
|
|
$ hg config --source empty.source
|
|
|
none: value
|
|
|
$ hg config empty.source -Tjson
|
|
|
[
|
|
|
{
|
|
|
"defaultvalue": null,
|
|
|
"name": "empty.source",
|
|
|
"source": "",
|
|
|
"value": "value"
|
|
|
}
|
|
|
]
|
|
|
|
|
|
$ cp .hg/hgrc.orig .hg/hgrc
|
|
|
|
|
|
Test "%unset"
|
|
|
|
|
|
$ cat >> $HGRCPATH <<EOF
|
|
|
> [unsettest]
|
|
|
> local-hgrcpath = should be unset (HGRCPATH)
|
|
|
> %unset local-hgrcpath
|
|
|
>
|
|
|
> global = should be unset (HGRCPATH)
|
|
|
>
|
|
|
> both = should be unset (HGRCPATH)
|
|
|
>
|
|
|
> set-after-unset = should be unset (HGRCPATH)
|
|
|
> EOF
|
|
|
|
|
|
$ cat >> .hg/hgrc <<EOF
|
|
|
> [unsettest]
|
|
|
> local-hgrc = should be unset (.hg/hgrc)
|
|
|
> %unset local-hgrc
|
|
|
>
|
|
|
> %unset global
|
|
|
>
|
|
|
> both = should be unset (.hg/hgrc)
|
|
|
> %unset both
|
|
|
>
|
|
|
> set-after-unset = should be unset (.hg/hgrc)
|
|
|
> %unset set-after-unset
|
|
|
> set-after-unset = should be set (.hg/hgrc)
|
|
|
> EOF
|
|
|
|
|
|
$ hg showconfig unsettest
|
|
|
unsettest.set-after-unset=should be set (.hg/hgrc)
|
|
|
|
|
|
Test exit code when no config matches
|
|
|
|
|
|
$ hg config Section.idontexist
|
|
|
[1]
|
|
|
|
|
|
sub-options in [paths] aren't expanded
|
|
|
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> [paths]
|
|
|
> foo = ~/foo
|
|
|
> foo:suboption = ~/foo
|
|
|
> EOF
|
|
|
|
|
|
$ hg showconfig paths
|
|
|
paths.foo=~/foo
|
|
|
paths.foo:suboption=~/foo
|
|
|
|
|
|
note: The path expansion no longer happens at the config level, but the path is
|
|
|
still expanded:
|
|
|
|
|
|
$ hg path | grep foo
|
|
|
foo = $TESTTMP/foo
|
|
|
|
|
|
edit failure
|
|
|
|
|
|
$ HGEDITOR=false hg config --edit
|
|
|
abort: edit failed: false exited with status 1
|
|
|
[10]
|
|
|
|
|
|
config affected by environment variables
|
|
|
|
|
|
$ EDITOR=e1 VISUAL=e2 hg config --source | grep 'ui\.editor'
|
|
|
$VISUAL: ui.editor=e2
|
|
|
|
|
|
$ VISUAL=e2 hg config --source --config ui.editor=e3 | grep 'ui\.editor'
|
|
|
--config: ui.editor=e3
|
|
|
|
|
|
$ PAGER=p1 hg config --source | grep 'pager\.pager'
|
|
|
$PAGER: pager.pager=p1
|
|
|
|
|
|
$ PAGER=p1 hg config --source --config pager.pager=p2 | grep 'pager\.pager'
|
|
|
--config: pager.pager=p2
|
|
|
|
|
|
verify that aliases are evaluated as well
|
|
|
|
|
|
$ hg init aliastest
|
|
|
$ cd aliastest
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> [ui]
|
|
|
> user = repo user
|
|
|
> EOF
|
|
|
$ touch index
|
|
|
$ unset HGUSER
|
|
|
$ hg ci -Am test
|
|
|
adding index
|
|
|
$ hg log --template '{author}\n'
|
|
|
repo user
|
|
|
$ cd ..
|
|
|
|
|
|
alias has lower priority
|
|
|
|
|
|
$ hg init aliaspriority
|
|
|
$ cd aliaspriority
|
|
|
$ cat > .hg/hgrc << EOF
|
|
|
> [ui]
|
|
|
> user = alias user
|
|
|
> username = repo user
|
|
|
> EOF
|
|
|
$ touch index
|
|
|
$ unset HGUSER
|
|
|
$ hg ci -Am test
|
|
|
adding index
|
|
|
$ hg log --template '{author}\n'
|
|
|
repo user
|
|
|
$ cd ..
|
|
|
|
|
|
configs should be read in lexicographical order
|
|
|
|
|
|
$ mkdir configs
|
|
|
$ for i in `$TESTDIR/seq.py 10 99`; do
|
|
|
> printf "[section]\nkey=$i" > configs/$i.rc
|
|
|
> done
|
|
|
$ HGRCPATH=configs hg config section.key
|
|
|
99
|
|
|
|
|
|
Configuration priority
|
|
|
======================
|
|
|
|
|
|
setup necessary file
|
|
|
|
|
|
$ cat > file-A.rc << EOF
|
|
|
> [config-test]
|
|
|
> basic = value-A
|
|
|
> pre-include= value-A
|
|
|
> %include ./included.rc
|
|
|
> post-include= value-A
|
|
|
> [command-templates]
|
|
|
> log = "value-A\n"
|
|
|
> EOF
|
|
|
|
|
|
$ cat > file-B.rc << EOF
|
|
|
> [config-test]
|
|
|
> basic = value-B
|
|
|
> [ui]
|
|
|
> logtemplate = "value-B\n"
|
|
|
> EOF
|
|
|
|
|
|
|
|
|
$ cat > included.rc << EOF
|
|
|
> [config-test]
|
|
|
> pre-include= value-included
|
|
|
> post-include= value-included
|
|
|
> EOF
|
|
|
|
|
|
$ cat > file-C.rc << EOF
|
|
|
> %include ./included-alias-C.rc
|
|
|
> [ui]
|
|
|
> logtemplate = "value-C\n"
|
|
|
> EOF
|
|
|
|
|
|
$ cat > included-alias-C.rc << EOF
|
|
|
> [command-templates]
|
|
|
> log = "value-included\n"
|
|
|
> EOF
|
|
|
|
|
|
|
|
|
$ cat > file-D.rc << EOF
|
|
|
> [command-templates]
|
|
|
> log = "value-D\n"
|
|
|
> %include ./included-alias-D.rc
|
|
|
> EOF
|
|
|
|
|
|
$ cat > included-alias-D.rc << EOF
|
|
|
> [ui]
|
|
|
> logtemplate = "value-included\n"
|
|
|
> EOF
|
|
|
|
|
|
Simple order checking
|
|
|
---------------------
|
|
|
|
|
|
If file B is read after file A, value from B overwrite value from A.
|
|
|
|
|
|
$ HGRCPATH=`path_list_var "file-A.rc:file-B.rc"` hg config config-test.basic
|
|
|
value-B
|
|
|
|
|
|
Ordering from include
|
|
|
---------------------
|
|
|
|
|
|
value from an include overwrite value defined before the include, but not the one defined after the include
|
|
|
|
|
|
$ HGRCPATH="file-A.rc" hg config config-test.pre-include
|
|
|
value-included
|
|
|
$ HGRCPATH="file-A.rc" hg config config-test.post-include
|
|
|
value-A
|
|
|
|
|
|
command line override
|
|
|
---------------------
|
|
|
|
|
|
$ HGRCPATH=`path_list_var "file-A.rc:file-B.rc"` hg config config-test.basic --config config-test.basic=value-CLI
|
|
|
value-CLI
|
|
|
|
|
|
Alias ordering
|
|
|
--------------
|
|
|
|
|
|
The official config is now `command-templates.log`, the historical
|
|
|
`ui.logtemplate` is a valid alternative for it.
|
|
|
|
|
|
When both are defined, The config value read the last "win", this should keep
|
|
|
being true if the config have other alias. In other word, the config value read
|
|
|
earlier will be considered "lower level" and the config read later would be
|
|
|
considered "higher level". And higher level values wins.
|
|
|
|
|
|
$ HGRCPATH="file-A.rc" hg log -r .
|
|
|
value-A
|
|
|
$ HGRCPATH="file-B.rc" hg log -r .
|
|
|
value-B
|
|
|
$ HGRCPATH=`path_list_var "file-A.rc:file-B.rc"` hg log -r .
|
|
|
value-B
|
|
|
|
|
|
Alias and include
|
|
|
-----------------
|
|
|
|
|
|
The pre/post include priority should also apply when tie-breaking alternatives.
|
|
|
See the case above for details about the two config options used.
|
|
|
|
|
|
$ HGRCPATH="file-C.rc" hg log -r .
|
|
|
value-C
|
|
|
$ HGRCPATH="file-D.rc" hg log -r .
|
|
|
value-included
|
|
|
|
|
|
command line override
|
|
|
---------------------
|
|
|
|
|
|
$ HGRCPATH=`path_list_var "file-A.rc:file-B.rc"` hg log -r . --config ui.logtemplate="value-CLI\n"
|
|
|
value-CLI
|
|
|
|