CsToml.Extensions
1.1.0
See the version list below for details.
dotnet add package CsToml.Extensions --version 1.1.0
NuGet\Install-Package CsToml.Extensions -Version 1.1.0
<PackageReference Include="CsToml.Extensions" Version="1.1.0" />
paket add CsToml.Extensions --version 1.1.0
#r "nuget: CsToml.Extensions, 1.1.0"
// Install CsToml.Extensions as a Cake Addin #addin nuget:?package=CsToml.Extensions&version=1.1.0 // Install CsToml.Extensions as a Cake Tool #tool nuget:?package=CsToml.Extensions&version=1.1.0
CsToml - TOML Parser/Serializer for .NET
CsToml is TOML Parser/Serializer for .NET.
For more information about TOML, visit the official website at https://toml.io/en/
[!NOTE] The officially released versions are CsToml Ver. 1.1.0, CsToml.Extensions Ver. 1.1.0, and CsToml.Generator Ver. 1.1.0.
using CsToml;
using System.Text;
var tomlText = @"
key = ""value""
number = 123
"u8;
var document = CsTomlSerializer.Deserialize<TomlDocument>(tomlText);
if (document!.RootNode["key"u8].TryGetString(out var value))
{
Console.WriteLine($"key = {value}"); // key = value
}
Console.WriteLine($"key = {document!.RootNode["key"u8].GetString()}"); // key = value
using var serializedTomlText = CsTomlSerializer.Serialize(document);
Console.WriteLine(Encoding.UTF8.GetString(serializedTomlText.ByteSpan));
// key = "value"
// number = 123
Table of Contents
- Feature
- Installation
- Built-in support type
- Serialize and deserialize TomlDocument
- Find values from TomlDocument
- Serialize and deserialize custom classes (CsToml.Generator)
- Extensions (CsToml.Extensions)
- UnitTest
- License
Feature
- TOML v1.0.0 supported
- Implemented in .NET 8 and C# 12.(supports .NET 8 or later. )
- Supports I/O APIs (
IBufferWriter<byte>
,ReadOnlySpan<byte>
,ReadOnlySequence<byte>
) - By parsing directly in UTF-8 (byte array) instead of UTF-16 (strings), low allocation and high speed are achieved.
- All standard TOML v1.0.0 test cases are passed..
- The serialization interface and implementation is influenced by MemoryPack and VYaml.
Installation
This library is distributed via NuGet.
PM> Install-Package CsToml
Additional features are available by installing optional documents.(learn more in our extensions section)
PM> Install-Package CsToml.Extensions
PM> Install-Package CsToml.Generator
Built-in support type
These types can be serialized/deserialize by default.
- .NET Built-in types(
bool
,long
,double
,string
etc) DateTime
,DateTimeOffset
,DateOnly
,TimeOnly
,TimeSpan
Enum
,Half
,Int128
,UInt128
,BigInteger
Uri
,Version
,Guid
,Nullable
T[]
,Memory<>
,ReadOnlyMemory<>
List<>
,Stack<>
,HashSet<>
,SortedSet<>
,Queue<>
,LinkedList<>
,ReadOnlyCollection<>
ConcurrentQueue<>
,ConcurrentStack<>
,ConcurrentBag<>
IEnumerable<>
,ICollection<>
,IReadOnlyCollection<>
,IList<>
,IReadOnlyList<>
,ISet<>
,IReadOnlySet<>
Dictionary<string, object?>
,IDictionary<string, object?>
KeyValuePair<>
,Tuple<,...>
,ValueTuple<,...>
Serialize and deserialize TomlDocument
By specifying TomlDocument
, serialization and deserialization can be performed while preserving the TOML data structure.
Call CsTomlSerializer.Deserialize<TomlDocument>(tomlText)
to deserialize a UTF-8 string (ReadOnlySpan<byte>
or ReadOnlySequence<byte>
) in TOML format.
The second argument is CsTomlSerializerOptions
, which does not need to be specified explicitly at this time.
It may be used to add optional features in the future.
Call CsTomlSerializer.Serialize
to Serialize TomlDocument
.
You can return a ByteMemoryResult or get a utf8 byte array via IBufferWriter<byte>.
var tomlText = @"
key = ""value""
number = 123
"u8;
var document = CsTomlSerializer.Deserialize<TomlDocument>(tomlText);
// The first is obtained by ByteMemoryResult.
using var result = CsTomlSerializer.Serialize(document);
Console.WriteLine(Encoding.UTF8.GetString(result.ByteSpan));
// The second is obtained via IBufferWriter<byte>.
var bufferWriter = new ArrayBufferWriter<byte>();
CsTomlSerializer.Serialize(ref bufferWriter, document);
If a syntax error is found during deserialization, an CsTomlSerializeException
is thrown after deserialization.
The contents of the thrown exception can be viewed at CsTomlException.InnerException
.
var tomlText = @"
key = ""value""
number = ""Error
"u8;
try
{
// throw CsTomlException
var error = CsTomlSerializer.Deserialize<TomlDocument>(tomlText);
}
catch(CsTomlSerializeException ctse)
{
foreach (var cte in ctse.Exceptions)
{
// A syntax error (CsTomlException) occurred while parsing line 3 of the TOML file. Check InnerException for details.
var e = cte.InnerException; // InnerException: 10 is a character that cannot be converted to a number.
}
}
Find values from TomlDocument
It can be obtained via indexers([ReadOnlySpan<char>]
,[ReadOnlySpan<byte>]
,[int index]
) from TomlDocument.RootNode
property.
var tomlText = @"
key = 123
dotted.keys = ""value""
array = [1, ""2"", 3]
inlineTable = { key = ""value2"", number = 123 }
configurations = [1, {key = [ { key2 = [""VALUE""]}]}]
[table]
key = ""value3""
[[arrayOfTables]]
key = ""value4""
[[arrayOfTables]]
number = 123
"u8;
var document = CsTomlSerializer.Deserialize<TomlDocument>(tomlText);
var key = document!.RootNode["key"u8].GetInt64(); // 123
var dottedKeys = document!.RootNode["dotted"u8]["keys"u8].GetString(); // "value"
var array = document!.RootNode["array"u8].GetArray(); // [1, 2, 3]
var item1 = array[0].GetInt64(); // 1
var item2 = array[1].GetString(); // "2"
var item3 = array[2].GetInt64(); // 3
// Same as "array[0].GetInt64()"
var item1_2 = document!.RootNode["array"u8].GetArrayValue(0).GetInt64();
var inlineTable = document!.RootNode["inlineTable"u8]["key"u8].GetString(); // "value2"
var configurations = document!.RootNode["configurations"u8][1]["key"u8][0]["key2"u8][0].GetString(); // "VALUE"
var table = document!.RootNode["table"u8]["key"u8].GetString(); // "value3"
var arrayOfTables = document!.RootNode["arrayOfTables"u8][0]["key"u8].GetString(); // "value4"
var arrayOfTables2 = document!.RootNode["arrayOfTables"u8][1]["number"u8].GetString(); // 123
var tuple = document!.RootNode["array"u8].GetValue<Tuple<long, string, long>>(); // Tuple<long, string, long>(1, "2", 3)
TomlValue
and TomlDocumentNode
have APIs for accessing and casting values.
APIs with the Get
prefix throw an CsTomlException
on failure and return a value on success.
APIs with the TryGet
prefix return false on failure and set the value to the argument of the out parameter on success.
CanGetValue
can be used to see which Toml value types can be converted.
GetValue<T>
and TryGetValue<T>
can be used to obtain a value converted from a Toml value type to a specified type.
public bool CanGetValue(TomlValueFeature feature)
public ReadOnlyCollection<TomlValue> GetArray()
public TomlValue GetArrayValue(int index)
public string GetString()
public long GetInt64()
public double GetDouble()
public bool GetBool()
public DateTime GetDateTime()
public DateTimeOffset GetDateTimeOffset()
public DateOnly GetDateOnly()
public TimeOnly GetTimeOnly()
public object GetObject()
public T GetNumber<T>() where T : struct, INumberBase<T>
public T GetValue<T>()
public bool TryGetArray(out IReadOnlyList<TomlValue> value)
public bool TryGetArrayValue(int index, out TomlValue value)
public bool TryGetString(out string value)
public bool TryGetInt64(out long value)
public bool TryGetDouble(out double value)
public bool TryGetBool(out bool value)
public bool TryGetDateTime(out DateTime value)
public bool TryGetDateTimeOffset(out DateTimeOffset value)
public bool TryGetDateOnly(out DateOnly value)
public bool TryGetTimeOnly(out TimeOnly value)
public bool TryGetObject(out object value)
public bool TryGetNumber<T>(out T value) where T : struct, INumberBase<T>
public bool TryGetValue<T>(out T value)
Serialize and deserialize custom classes (CsToml.Generator
)
Define the class to be serialized and assign the [TomlSerializedObject]
and [TomlValueOnSerialized]
attribute and the partial keyword.
[TomlValueOnSerialized]
can only be given to read-write (they have both a get and a set accessor) properties.
[TomlSerializedObject]
public partial class CsTomlClass
{
[TomlValueOnSerialized]
public string Key { get; set; }
[TomlValueOnSerialized]
public int? Number { get; set; }
[TomlValueOnSerialized]
public int[] Array { get; set; }
[TomlValueOnSerialized(aliasName: "alias")]
public string Value { get; set; }
[TomlValueOnSerialized]
public TableClass Table { get; set; } = new TableClass();
}
Adding the above attributes will generate code for serialization/deserialization by Source Generators.
Property names with [TomlValueOnSerialized]
are used as keys in the TOML document.
The key name can also be changed with [TomlValueOnSerialized(aliasName)]
.
<details><summary>Generated Code</summary>
#nullable enable
#pragma warning disable CS0219 // The variable 'variable' is assigned but its value is never used
#pragma warning disable CS8600 // Converting null literal or possible null value to non-nullable type.
#pragma warning disable CS8601 // Possible null reference assignment.
#pragma warning disable CS8602 // Dereference of a possibly null reference.
#pragma warning disable CS8603 // Possible null reference return.
#pragma warning disable CS8604 // Possible null reference argument for parameter.
#pragma warning disable CS8619 // Possible null reference assignment fix
using CsToml;
using CsToml.Formatter;
using CsToml.Formatter.Resolver;
namespace ConsoleApp;
partial class CsTomlClass : ITomlSerializedObject<CsTomlClass>
{
static CsTomlClass ITomlSerializedObject<CsTomlClass>.Deserialize(ref TomlDocumentNode rootNode, CsTomlSerializerOptions options)
{
var target = new CsTomlClass();
var __Key__RootNode = rootNode["Key"u8];
target.Key = options.Resolver.GetFormatter<string>()!.Deserialize(ref __Key__RootNode, options);
var __Value__RootNode = rootNode["alias"u8];
target.Value = options.Resolver.GetFormatter<string>()!.Deserialize(ref __Value__RootNode, options);
var __Array__RootNode = rootNode["Array"u8];
target.Array = options.Resolver.GetFormatter<int[]>()!.Deserialize(ref __Array__RootNode, options);
var __Number__RootNode = rootNode["Number"u8];
target.Number = options.Resolver.GetFormatter<int?>()!.Deserialize(ref __Number__RootNode, options);
var __Table__RootNode = rootNode["Table"u8];
target.Table = options.Resolver.GetFormatter<global::ConsoleApp.TableClass>()!.Deserialize(ref __Table__RootNode, options);
return target;
}
static void ITomlSerializedObject<CsTomlClass>.Serialize<TBufferWriter>(ref Utf8TomlDocumentWriter<TBufferWriter> writer, CsTomlClass target, CsTomlSerializerOptions options)
{
writer.WriteKey("Key"u8);
writer.WriteEqual();
options.Resolver.GetFormatter<String>()!.Serialize(ref writer, target.Key, options);
writer.EndKeyValue();
writer.WriteKey("alias"u8);
writer.WriteEqual();
options.Resolver.GetFormatter<String>()!.Serialize(ref writer, target.Value, options);
writer.EndKeyValue();
writer.WriteKey("Array"u8);
writer.WriteEqual();
options.Resolver.GetFormatter<int[]>()!.Serialize(ref writer, target.Array, options);
writer.EndKeyValue();
writer.WriteKey("Number"u8);
writer.WriteEqual();
options.Resolver.GetFormatter<int?>()!.Serialize(ref writer, target.Number, options);
writer.EndKeyValue();
writer.PushKey("Table"u8);
options.Resolver.GetFormatter<TableClass>()!.Serialize(ref writer, target.Table, options);
writer.PopKey();
}
static void ITomlSerializedObjectRegister.Register()
{
TomlSerializedObjectFormatterResolver.Register(new TomlSerializedObjectFormatter<CsTomlClass>());
}
}
</details>
As a result, it can be serialized and deserialized as follows. Custom class serialization does not preserve the layout of the original TOML text.
var tomlText = @"
Key = ""value""
Number = 123
Array = [1, 2, 3]
alias = ""alias""
[Table]
Key = ""value""
Number = 123
"u8;
var value = CsTomlSerializer.Deserialize<CsTomlClass>(tomlText);
using var serializedText = CsTomlSerializer.Serialize(value);
// Key = "value"
// alias = "alias"
// Array = [ 1, 2, 3 ]
// Number = 123
// Table.Key = "value"
// Table.Number = 123
var serializedTomlText = Encoding.UTF8.GetString(serializedText.ByteSpan);
Extensions (CsToml.Extensions
)
CsToml.Extensions
provides APIs to serialize and deserialize Toml files on disk.
// deserialize from TOML File
var document = CsTomlFileSerializer.Deserialize<TomlDocument>("test.toml");
var document2 = await CsTomlFileSerializer.DeserializeAsync<TomlDocument>("test.toml");
// serialize To TOML File
CsTomlFileSerializer.Serialize("test.toml", document);
await CsTomlFileSerializer.SerializeAsync("test.toml", document);
CsTomlFileSerializer.Deserialize
and CsTomlFileSerializer.DeserializeAsync
deserialize UTF8 strings in TOML files into TomlDocument
.
CsTomlFileSerializer.Serialize
and CsTomlFileSerializer.SerializeAsync
serialize the UTF8 string of TomlDocument
to the TOML file.
CsToml.Extensions
uses Cysharp/NativeMemoryArray as a third party library.
UnitTest
Please note that we are using the TOML files located in the ‘tests/’ directory of the ‘toml-test repository (MIT License)’ for some of our unit tests.
License
MIT License. Some code is implemented based on dotnet/runtime, Please check the original license.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net8.0
- CsToml (>= 1.1.0)
- NativeMemoryArray (>= 1.2.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.