Skip to content

Commit 6683041

Browse files
authored
Merge pull request #2 from ellenfieldn/add-comments
Added comments.
2 parents 726dae4 + bf6aead commit 6683041

File tree

4 files changed

+33
-8
lines changed

4 files changed

+33
-8
lines changed

Readme.md

+1
Original file line numberDiff line numberDiff line change
@@ -79,3 +79,4 @@ Assuming that there is a variable named 'Z' that contains the value 'zzzz':
7979
| asd\\$zzzzasd | asd$${Z}asd |
8080
| asd\\zzzzasd | asd\\\\${Z}asd |
8181
| asd\\\\zzzzasd | asd\\\\\\\\${Z}asd |
82+
| \\\\server\\path | \\\\server\\path |

src/VariableConfig/VariableConfigurationProvider.cs

+25-8
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,14 @@ public class VariableConfigurationProvider : IConfigurationProvider
1111
{
1212
private readonly IConfiguration _config;
1313

14+
/*
15+
* If we have an odd number of backslashes directly before ${...}, then we know the $ is escaped
16+
* '(?<=[^\\]|^)' Find a character to anchor to -- either the start of the string or a non-backslash
17+
* ((?:\\\\)*) Grab any number of '\\' that precede a '$'. This will only match if we have an even number.
18+
* \$\{(?<variableName>.*?)\} Anything of the form ${...} is our variable name.
19+
*/
20+
private static readonly string _regexThatMatchesVariables = @"(?<=[^\\]|^)((?:\\\\)*)\$\{(?<variableName>.*?)\}";
21+
1422
public VariableConfigurationProvider(VariableConfigurationSource source)
1523
{
1624
source = source ?? throw new ArgumentNullException(nameof(source));
@@ -35,23 +43,32 @@ public void Load() { }
3543
public bool TryGet(string key, out string value)
3644
{
3745
value = _config[key];
38-
bool getResult = true;
3946
if (string.IsNullOrEmpty(value))
4047
{
4148
return false;
4249
}
43-
value = Regex.Replace(value, @"([^\\]|^)\\(?>(\\\\)*)\$|\${(?<variableName>.*?)}", match =>
50+
var getResult = ReplaceVariable(ref value);
51+
return !string.IsNullOrEmpty(value) && getResult;
52+
}
53+
54+
private bool ReplaceVariable(ref string after)
55+
{
56+
var getResult = true;
57+
after = Regex.Replace(after, _regexThatMatchesVariables, match =>
4458
{
45-
if(!match.Groups["variableName"].Success)
59+
var variableGroup = match.Groups["variableName"];
60+
if (!variableGroup.Success)
4661
{
4762
return match.Groups[0].Value;
4863
}
49-
getResult &= TryGet(match.Groups["variableName"].Value, out string innerMatchResult);
50-
return innerMatchResult;
64+
getResult = TryGet(variableGroup.Value, out string innerMatchResult);
65+
return match.Groups.Where(g => g.Name != "0" && !string.IsNullOrEmpty(g.Value))
66+
.Select(g => (g.Name == "variableName") ? innerMatchResult : Regex.Replace(g.Value, @"\\\\", @"\")) //Unescape '\\' as '\'
67+
.Aggregate((first, second) => first + second);
5168
});
52-
value = Regex.Replace(value, @"\\\\", @"\");
53-
value = Regex.Replace(value, @"\\\${", @"${");
54-
return !string.IsNullOrEmpty(value) && getResult;
69+
after = Regex.Replace(after, @"\\\${", @"${"); //Unescape '\${' as '${'
70+
after = Regex.Replace(after, @"\\\\\$", @"\$"); //Unescape leftover double backslashes that precede a $
71+
return getResult;
5572
}
5673
}
5774
}

test/VariableConfig.Test/Escaping.config

+1
Original file line numberDiff line numberDiff line change
@@ -14,4 +14,5 @@
1414
<test11>${Z}</test11>
1515
<test12>\${Z}</test12>
1616
<test13>\\${Z}</test13>
17+
<test14>\\server\path</test14>
1718
</AppConfiguration>

test/VariableConfig.Test/Escaping.cs

+6
Original file line numberDiff line numberDiff line change
@@ -95,5 +95,11 @@ public void Test13()
9595
{
9696
Assert.AreEqual(@"\zzzz", Configuration["Test13"]);
9797
}
98+
99+
[TestMethod]
100+
public void Test14()
101+
{
102+
Assert.AreEqual(@"\\server\path", Configuration["Test14"]);
103+
}
98104
}
99105
}

0 commit comments

Comments
 (0)