Skip to content

Commit

Permalink
Add support for CSharpHelper for Dictionary literals
Browse files Browse the repository at this point in the history
  • Loading branch information
yinzara committed Jun 27, 2022
1 parent f72888f commit 7f4c173
Show file tree
Hide file tree
Showing 2 changed files with 119 additions and 5 deletions.
88 changes: 86 additions & 2 deletions src/EFCore.Design/Design/Internal/CSharpHelper.cs
Original file line number Diff line number Diff line change
Expand Up @@ -810,6 +810,85 @@ private string List(Type type, IEnumerable values, bool vertical = false)
return builder.ToString();
}

/// <summary>
/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to
/// the same compatibility standards as public APIs. It may be changed or removed without notice in
/// any release. You should only use it directly in your code with extreme caution and knowing that
/// doing so can result in application failures when updating to a new Entity Framework Core release.
/// </summary>
public virtual string Literal<TKey, TValue>(Dictionary<TKey, TValue> dict, bool vertical = false)
where TKey : notnull
=> Dictionary(typeof(TKey), typeof(TValue), dict, vertical);

private string Dictionary(Type keyType, Type valueType, IDictionary dict, bool vertical = false)
{
var builder = new IndentedStringBuilder();

builder.Append("new Dictionary<")
.Append(Reference(keyType))
.Append(",")
.Append(Reference(valueType))
.Append(">");

var first = true;
foreach (var key in dict.Keys)
{
if (first)
{
builder.Append(" {");
if (vertical)
{
builder.AppendLine();
builder.IncrementIndent();
}
else
{
builder.Append(" ");
}
first = false;
}
else
{
builder.Append(",");

if (vertical)
{
builder.AppendLine();
}
else
{
builder.Append(" ");
}
}

builder.Append("[");
builder.Append(UnknownLiteral(key));
builder.Append("] = ");
builder.Append(UnknownLiteral(dict[key]));
}

if (first)
{
builder.Append("()");
}
else
{
if (vertical)
{
builder.AppendLine();
builder.DecrementIndent();
}
else
{
builder.Append(" ");
}

builder.Append("}");
}

return builder.ToString();
}

/// <summary>
/// This is an internal API that supports the Entity Framework Core infrastructure and not subject to
/// the same compatibility standards as public APIs. It may be changed or removed without notice in
Expand Down Expand Up @@ -917,9 +996,14 @@ public virtual string UnknownLiteral(object? value)
return Array(literalType.GetElementType()!, array);
}

if (value is IList list && value.GetType().IsGenericType && value.GetType().GetGenericTypeDefinition() == typeof(List<>))
var valueType = value.GetType();
if (value is IList list && valueType.IsGenericType && valueType.GetGenericTypeDefinition() == typeof(List<>))
{
return List(valueType.GetGenericArguments()[0], list);
}
if (value is IDictionary dict && valueType.IsGenericType && valueType.GetGenericTypeDefinition() == typeof(Dictionary<,>))
{
return List(value.GetType().GetGenericArguments()[0], list);
return Dictionary(valueType.GenericTypeArguments[0], valueType.GenericTypeArguments[1], dict);
}

var mapping = _typeMappingSource.FindMapping(literalType);
Expand Down
36 changes: 33 additions & 3 deletions test/EFCore.Design.Tests/Design/Internal/CSharpHelperTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -140,10 +140,40 @@ public void Literal_works_when_list_of_mixed_objects()
@"new List<object> { 1, ""two"" }");

[ConditionalFact]
public void Literal_works_when_list_with_ctor_arguments()
public void Literal_works_when_list_vertical()
=> Assert.Equal(
@"new List<object> {
1,
""two""
}",new CSharpHelper(TypeMappingSource).Literal(
new List<object> { 1, "two" }, true));

[ConditionalFact]
public void Literal_works_when_empty_dictionary()
=> Literal_works(
new Dictionary<string,int>(),
@"new Dictionary<string,int>()");

[ConditionalFact]
public void Literal_works_when_dictionary_with_single_element()
=> Literal_works(
new List<string>(new [] { "one" }) { "two", "three" },
@"new List<string> { ""one"", ""two"", ""three"" }");
new Dictionary<string,string> { ["one"] = "value" },
@"new Dictionary<string,string> { [""one""] = ""value"" }");

[ConditionalFact]
public void Literal_works_when_dictionary_of_mixed_objects()
=> Literal_works(
new Dictionary<string,object> { ["one"] = 1, ["two"] = "Two" },
@"new Dictionary<string,object> { [""one""] = 1, [""two""] = ""Two"" }");

[ConditionalFact]
public void Literal_works_when_dictionary_vertical()
=> Assert.Equal(
@"new Dictionary<int,object> {
[1] = 1,
[2] = ""Two""
}",new CSharpHelper(TypeMappingSource).Literal(
new Dictionary<int,object> { [1] = 1, [2] = "Two" }, true));

[ConditionalFact]
public void Literal_works_when_multiline_string()
Expand Down

0 comments on commit 7f4c173

Please sign in to comment.