Skip to content

Commit

Permalink
fix(TomlTable): fixes key serialization by adding better key name che…
Browse files Browse the repository at this point in the history
…cks 729f57

Closes SamboyCoding#30
Closes SamboyCoding#32
  • Loading branch information
tcortega committed Jul 14, 2023
1 parent 729f57b commit fa17b89
Show file tree
Hide file tree
Showing 6 changed files with 97 additions and 44 deletions.
26 changes: 26 additions & 0 deletions Tomlet.Tests/DictionaryTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,4 +27,30 @@ public void DictionariesAsFieldsWork()
//Just make sure this doesn't throw
var serialized = TomletMain.TomlStringFrom(obj);
}

[Fact]
public void DictionaryKeysShouldBeProperlyEscaped()
{
var dictionary = new Dictionary<string, string>
{
{"normal-key", "normal-key"},
{"normal_key", "normal_key"},
{"normalkey", "normalkey"},
{"key with space", "\"key with spaces\""},
{"key!with{}(*%&)random[other+symbols", "\"key!with{}(*%&)random[other+symbols\""},
{"key/with/slashes", "\"key/with/slashes\""},
{"Nam\\e", "\"Nam\\\\e\""}
};

var obj = new ClassWithDictionary
{
GenericDictionary = dictionary
};

var serialized = TomletMain.TomlStringFrom(obj);
foreach(var (_, expectedValue) in dictionary)
{
Assert.Contains(expectedValue, serialized);
}
}
}
54 changes: 37 additions & 17 deletions Tomlet.Tests/TableTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,26 @@ public void TablesCanHaveQuotedKeyNames()
var tomlString = document.SerializedValue.Trim();
Assert.Equal(inputString, tomlString);
}

[Theory]
[InlineData("normal-key", "normal-key")]
[InlineData("normal_key", "normal_key")]
[InlineData("normalkey", "normalkey")]
[InlineData("\"key with spaces\"", "\"key with spaces\"")]
[InlineData("key!with{}(*%&)random[other+symbols", "\"key!with{}(*%&)random[other+symbols\"")]
[InlineData("key/with/slashes", "\"key/with/slashes\"")]
[InlineData("Nam\\e", "\"Nam\\\\e\"")]
public void KeysShouldBeSerializedCorrectly(string inputKey, string expectedKey)
{
var inputString = $"""
["Test Table"]
{inputKey} = "value"
""";

var document = GetDocument(inputString);
var serializedDocument = document.SerializedValue.Trim();
Assert.Contains(expectedKey, serializedDocument);
}

[Fact]
public void TablesCanHaveQuotedDottedNames()
Expand All @@ -58,23 +78,23 @@ public void TablesCanHaveQuotedDottedNames()
}

[Fact]
public void TableIteratorEqualsEntries()
{
var document = GetDocument(TestResources.TableIteratorInput);

Assert.Single(document.Entries);

Assert.NotNull(document.GetSubTable("table-1"));

var table1 = document.GetSubTable("table-1");

Assert.Equal(4, table1.Entries.Count);

foreach (var entry in table1)
{
Assert.True(table1.Entries.ContainsKey(entry.Key));
Assert.Equal(table1.Entries[entry.Key], entry.Value);
}
public void TableIteratorEqualsEntries()
{
var document = GetDocument(TestResources.TableIteratorInput);

Assert.Single(document.Entries);

Assert.NotNull(document.GetSubTable("table-1"));

var table1 = document.GetSubTable("table-1");

Assert.Equal(4, table1.Entries.Count);

foreach (var entry in table1)
{
Assert.True(table1.Entries.ContainsKey(entry.Key));
Assert.Equal(table1.Entries[entry.Key], entry.Value);
}
}
}
}
1 change: 1 addition & 0 deletions Tomlet.Tests/TestModelClasses/ClassWithDictionary.cs
Original file line number Diff line number Diff line change
Expand Up @@ -5,4 +5,5 @@ namespace Tomlet.Tests.TestModelClasses;
public class ClassWithDictionary
{
public Dictionary<string, Subname> name;
public Dictionary<string, string> GenericDictionary;
}
2 changes: 2 additions & 0 deletions Tomlet.Tests/Tomlet.Tests.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
<TargetFramework>net6.0</TargetFramework>

<IsPackable>false</IsPackable>

<LangVersion>latest</LangVersion>
</PropertyGroup>

<ItemGroup>
Expand Down
54 changes: 29 additions & 25 deletions Tomlet/Models/TomlTable.cs
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
using System;
using System.Collections;
using System.Collections;
using System.Collections.Generic;
using System.Diagnostics.CodeAnalysis;
using System.Linq;
Expand Down Expand Up @@ -36,7 +36,7 @@ public override string SerializedValue

var builder = new StringBuilder("{ ");

builder.Append(string.Join(", ", Entries.Select(o => o.Key + " = " + o.Value.SerializedValue).ToArray()));
builder.Append(string.Join(", ", Entries.Select(o => EscapeKeyIfNeeded(o.Key) + " = " + o.Value.SerializedValue).ToArray()));

builder.Append(" }");

Expand Down Expand Up @@ -136,10 +136,8 @@ private void WriteValueToStringBuilder(string? keyName, string subKey, StringBui
builder.Append('\n');
}

private string EscapeKeyIfNeeded(string key)
private static string EscapeKeyIfNeeded(string key)
{
var didQuote = false;

if (key.StartsWith("\"") && key.EndsWith("\"") && key.Count(c => c == '"') == 2)
//Already double quoted
return key;
Expand All @@ -148,18 +146,24 @@ private string EscapeKeyIfNeeded(string key)
//Already single quoted
return key;

if (key.Contains("\"") || key.Contains("'"))
if (IsValidKey(key))
return key;

key = TomlUtils.EscapeStringValue(key);
return TomlUtils.AddCorrectQuotes(key);
}

private static bool IsValidKey(string key)
{
foreach (var c in key)
{
key = TomlUtils.AddCorrectQuotes(key);
didQuote = true;
if (!char.IsLetterOrDigit(c) && c != '_' && c != '-')
{
return false;
}
}

var escaped = TomlUtils.EscapeStringValue(key);

if (escaped.Contains(" ") || escaped.Contains("\\") && !didQuote)
escaped = TomlUtils.AddCorrectQuotes(escaped);

return escaped;
return true;
}

internal void ParserPutValue(string key, TomlValue value, int lineNumber)
Expand Down Expand Up @@ -454,16 +458,16 @@ public TomlTable GetSubTable(string key)
throw new TomlTypeMismatchException(typeof(TomlTable), value.GetType(), typeof(TomlTable));

return tbl;
}

IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}

public IEnumerator<KeyValuePair<string, TomlValue>> GetEnumerator()
{
return Entries.GetEnumerator();
}
}

IEnumerator IEnumerable.GetEnumerator()
{
return GetEnumerator();
}

public IEnumerator<KeyValuePair<string, TomlValue>> GetEnumerator()
{
return Entries.GetEnumerator();
}
}
}
4 changes: 2 additions & 2 deletions Tomlet/TomlUtils.cs
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,10 @@ public static string EscapeStringValue(string key)

public static string AddCorrectQuotes(string key)
{
if (key.Contains("'") && key.Contains("\""))
if (key.Contains('\'') && key.Contains('"'))
throw new InvalidTomlKeyException(key);

if (key.Contains("\""))
if (key.Contains('"'))
return $"'{key}'";

return $"\"{key}\"";
Expand Down

0 comments on commit fa17b89

Please sign in to comment.