@@ -11,6 +11,14 @@ public class VariableConfigurationProvider : IConfigurationProvider
11
11
{
12
12
private readonly IConfiguration _config ;
13
13
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
+
14
22
public VariableConfigurationProvider ( VariableConfigurationSource source )
15
23
{
16
24
source = source ?? throw new ArgumentNullException ( nameof ( source ) ) ;
@@ -35,23 +43,32 @@ public void Load() { }
35
43
public bool TryGet ( string key , out string value )
36
44
{
37
45
value = _config [ key ] ;
38
- bool getResult = true ;
39
46
if ( string . IsNullOrEmpty ( value ) )
40
47
{
41
48
return false ;
42
49
}
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 =>
44
58
{
45
- if ( ! match . Groups [ "variableName" ] . Success )
59
+ var variableGroup = match . Groups [ "variableName" ] ;
60
+ if ( ! variableGroup . Success )
46
61
{
47
62
return match . Groups [ 0 ] . Value ;
48
63
}
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 ) ;
51
68
} ) ;
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 ;
55
72
}
56
73
}
57
74
}
0 commit comments