omy.Utils.IO.Serialization.Generators 1.2.1

dotnet add package omy.Utils.IO.Serialization.Generators --version 1.2.1
                    
NuGet\Install-Package omy.Utils.IO.Serialization.Generators -Version 1.2.1
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="omy.Utils.IO.Serialization.Generators" Version="1.2.1" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="omy.Utils.IO.Serialization.Generators" Version="1.2.1" />
                    
Directory.Packages.props
<PackageReference Include="omy.Utils.IO.Serialization.Generators" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add omy.Utils.IO.Serialization.Generators --version 1.2.1
                    
#r "nuget: omy.Utils.IO.Serialization.Generators, 1.2.1"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package omy.Utils.IO.Serialization.Generators@1.2.1
                    
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=omy.Utils.IO.Serialization.Generators&version=1.2.1
                    
Install as a Cake Addin
#tool nuget:?package=omy.Utils.IO.Serialization.Generators&version=1.2.1
                    
Install as a Cake Tool

Utils.IO.Serialization.Generators

Utils.IO.Serialization.Generators ships a Roslyn source generator that produces strongly typed reader and writer extension methods for DTOs annotated with GenerateReaderWriterAttribute. It targets .NET 9 and extends the Utils.IO.Serialization primitives by removing repetitive serialization boilerplate.

Features

  • Discovers classes, structs, and records decorated with [GenerateReaderWriter].
  • Inspects members tagged with [Field(order)] to preserve wire ordering.
  • Emits Read{Type} and Write{Type} extension methods for IReader and IWriter.
  • Reuses custom serializers automatically when a member type already exposes matching extensions.
  • Generates partial, editor-friendly code with XML documentation.

Preparing a model

Annotate each serializable field or property with the FieldAttribute specifying the binary order. The generator supports nested models and nullable members.

using Utils.IO.Serialization;

[GenerateReaderWriter]
public partial class InventoryEntry
{
    [Field(0)]
    public required int Id { get; set; }

    [Field(1)]
    public required string Name { get; set; }

    [Field(2)]
    public PriceTag Price = new();
}

[GenerateReaderWriter]
public partial class PriceTag
{
    [Field(0)]
    public decimal Amount { get; set; }

    [Field(1)]
    public string Currency { get; set; } = "EUR";
}

Usage example

using System.IO;
using Utils.IO.Serialization;
using Utils.IO.Serialization.Generators;

var buffer = new MemoryStream();
IWriter writer = new BinaryWriterAdapter(new BinaryWriter(buffer));
IReader reader = new BinaryReaderAdapter(new BinaryReader(buffer));

var entry = new InventoryEntry { Id = 7, Name = "Sprocket", Price = new PriceTag { Amount = 19.95m, Currency = "USD" } };
writer.WriteInventoryEntry(entry);

buffer.Position = 0;
InventoryEntry roundTrip = reader.ReadInventoryEntry();

The generated extensions serialize the members in declaration order, nesting calls to generated serializers for PriceTag. When a member already exposes custom Read{Member} or Write{Member} extensions, the generator invokes them instead of emitting generic calls.

Additional scenarios

Nested collections and dictionaries

Complex object graphs composed of collections are supported out of the box.

[GenerateReaderWriter]
public partial class Warehouse
{
    [Field(0)]
    public List<InventoryEntry> Items { get; set; } = new();

    [Field(1)]
    public Dictionary<string, PriceTag> RegionalPrices { get; set; } = new();
}

var warehouse = new Warehouse
{
    Items =
    [
        new InventoryEntry { Id = 1, Name = "Widget", Price = new PriceTag { Amount = 9.99m } },
        new InventoryEntry { Id = 2, Name = "Gadget", Price = new PriceTag { Amount = 14.5m } },
    ],
};
warehouse.RegionalPrices["EU"] = new PriceTag { Amount = 8.99m, Currency = "EUR" };

writer.WriteWarehouse(warehouse);

Enumerables are serialized using their existing element serializers, so nested graphs stay consistent with individual entity serialization logic.

Version tolerant models

When new fields are appended with higher Field numbers the generator keeps backward compatibility by making them optional.

[GenerateReaderWriter]
public partial class InventoryEntry
{
    [Field(0)]
    public required int Id { get; set; }

    [Field(1)]
    public required string Name { get; set; }

    [Field(2)]
    public PriceTag Price = new();

    [Field(3)]
    public string? Description { get; set; }
}

Older readers that only know the first three fields still deserialize correctly because the generated code checks the buffer length before attempting to access new fields.

Working with spans and pooled buffers

The generated extensions are regular methods that accept any IWriter/IReader implementation so they integrate with high-performance pipelines.

using System.Buffers;
using Utils.IO.Serialization;

ArrayPool<byte> pool = ArrayPool<byte>.Shared;
byte[] rented = pool.Rent(4096);

try
{
    var writer = new SpanWriterAdapter(rented);
    writer.WriteInventoryEntry(entry);

    var reader = new SpanReaderAdapter(writer.WrittenSpan);
    InventoryEntry clone = reader.ReadInventoryEntry();
}
finally
{
    pool.Return(rented);
}

Adapters such as SpanWriterAdapter and SpanReaderAdapter allow serialization without heap allocations which is ideal for messaging systems.

Advanced scenario: reusing manual serializers

You can mix generated serializers with hand-written ones to handle special data formats.

using Utils.IO.Serialization;

public static class MoneySerializer
{
    public static void WritePriceTag(this IWriter writer, PriceTag value)
    {
        writer.WriteDecimal(value.Amount);
        writer.WriteString(value.Currency);
    }

    public static PriceTag ReadPriceTag(this IReader reader)
    {
        return new PriceTag
        {
            Amount = reader.ReadDecimal(),
            Currency = reader.ReadString(),
        };
    }
}

[GenerateReaderWriter]
public partial class Invoice
{
    [Field(0)]
    public required int Number { get; set; }

    [Field(1)]
    public required PriceTag Total { get; set; }
}

Because MoneySerializer already exposes WritePriceTag and ReadPriceTag, the generator defers to them when producing the WriteInvoice and ReadInvoice implementations, ensuring bespoke formatting is preserved.

Streaming large collections

Generated extensions work nicely with streaming scenarios by combining them with buffered adapters.

using System.IO;
using Utils.IO.Serialization;

await using var stream = File.OpenWrite("report.bin");
await using var writer = new BinaryWriterAdapter(new BinaryWriter(stream));

foreach (InventoryEntry item in inventory)
{
    writer.WriteInventoryEntry(item);
}

Since the generator only emits member-level serialization logic, you can control buffering and chunk sizes directly on the Stream or adapter used during serialization.

Hybrid streaming with asynchronous writers

The same generated extensions can be paired with asynchronous PipeWriter implementations for network streaming.

using System.IO.Pipelines;
using Utils.IO.Serialization;

Pipe pipe = new();
PipeWriter pipeWriter = pipe.Writer;

await using var writer = new PipeWriterAdapter(pipeWriter);
foreach (InventoryEntry item in inventory)
{
    writer.WriteInventoryEntry(item);
    await writer.FlushAsync();
}

Because flushing is under the caller's control you can batch messages, append framing headers, or interleave generated payloads with custom metadata before sending them over the network.

Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed.  net10.0 was computed.  net10.0-android was computed.  net10.0-browser was computed.  net10.0-ios was computed.  net10.0-maccatalyst was computed.  net10.0-macos was computed.  net10.0-tvos was computed.  net10.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.
  • .NETStandard 2.0

    • No dependencies.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last Updated
1.2.1 213 11/3/2025