Zooper.Gorilla.Generators 1.3.0

dotnet add package Zooper.Gorilla.Generators --version 1.3.0
                    
NuGet\Install-Package Zooper.Gorilla.Generators -Version 1.3.0
                    
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="Zooper.Gorilla.Generators" Version="1.3.0" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="Zooper.Gorilla.Generators" Version="1.3.0" />
                    
Directory.Packages.props
<PackageReference Include="Zooper.Gorilla.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 Zooper.Gorilla.Generators --version 1.3.0
                    
#r "nuget: Zooper.Gorilla.Generators, 1.3.0"
                    
#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 Zooper.Gorilla.Generators@1.3.0
                    
#: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=Zooper.Gorilla.Generators&version=1.3.0
                    
Install as a Cake Addin
#tool nuget:?package=Zooper.Gorilla.Generators&version=1.3.0
                    
Install as a Cake Tool

Gorilla

<img src="icon.png" alt="drawing" width="256"/>

A .NET source generator for creating discriminated unions in C# with OneOf. Inspired by the flutter package freezed and the F# discriminated union pattern.

๐Ÿš€ Overview

Gorilla helps you create type-safe, exhaustive discriminated unions in C#. It automatically generates matching methods, type checks, and conversions to make working with union types intuitive and safe. Gorilla uses OneOf under the hood to implement the discriminated union pattern but adds a layer of compile-time convenience and safety through source generation.

Gorilla supports:

  • top-level flat unions
  • unions nested inside classes, interfaces, and other containing types
  • abstract hierarchical unions where nested sub-unions participate in outer Match(...) dispatch
  • generated JSON converters for both flat and hierarchical unions

๐Ÿ“ฆ Installation

# Install both packages
dotnet add package Zooper.Gorilla.Attributes
dotnet add package Zooper.Gorilla.Generators

๐Ÿ”ง Usage

1. Define your union type

Add the [DiscriminatedUnion] attribute to a partial type and declare one [Variant] factory per case:

using Zooper.Gorilla.Attributes;

[DiscriminatedUnion]
public partial class SignInError
{
	[Variant]
	public static partial SignInError ServiceUnavailable();

	[Variant]
	public static partial SignInError InvalidCredentials();

	[Variant]
	public static partial SignInError InternalError();
}

2. Use your union type

return SignInError.ServiceUnavailable();

๐Ÿ—ƒ๏ธ JSON Serialization

Gorilla can generate JSON converters automatically when your project references System.Text.Json or Newtonsoft.Json.

  • System.Text.Json support emits a generated JsonConverter<T> and applies [System.Text.Json.Serialization.JsonConverter(...)]
  • Newtonsoft.Json support emits a generated JsonConverter<T> and applies [Newtonsoft.Json.JsonConverterAttribute(...)]
  • the discriminator field defaults to $type
  • converter generation can be overridden per union with GenerateJsonConverter, GenerateNewtonsoftJsonConverter, and DiscriminatorFieldName
using System.Text.Json;

var value = CreateProfileDto.Person("John", "Doe");

var json = JsonSerializer.Serialize(value);
var roundTripped = JsonSerializer.Deserialize<CreateProfileDto>(json)!;

JSON support works for both flat unions and hierarchical unions.

๐Ÿงฉ Nested Unions

Nested unions can stay attached to the owning contract or payload type instead of being flattened into top-level declarations.

using Zooper.Gorilla.Attributes;

public partial interface IEntityCreatedContract
{
	public sealed record ContractVersion1(
		EntityId EntityId,
		EntityName Name,
		IEntityPayload Info,
		EntityOrder Order) : IEntityCreatedContract;

	public partial interface IEntityPayload
	{
		[DiscriminatedUnion]
		public sealed partial class V1 : IEntityPayload
		{
			[Variant]
			public static partial V1 Created();

			[Variant]
			public static partial V1 Standard(string category, bool isVisible);
		}
	}
}

That generated union can still be matched normally:

var payload = IEntityCreatedContract.IEntityPayload.V1.Standard("alpha", true);

var description = payload.Match(
	created => "created",
	standard => $"standard:{standard.Category}:{standard.IsVisible}");

See the full sample in Zooper.Gorilla.Sample/NestedContractSamples.cs.

๐ŸŒฒ Hierarchical Unions

Abstract unions can contain nested sub-unions that also flow through the outer union API.

using Zooper.Gorilla.Attributes;

[DiscriminatedUnion]
public abstract partial record ContractOutcome
{
	[Variant]
	public static partial ContractOutcome Success(string contractId);

	[DiscriminatedUnion]
	public abstract partial record Rejected : ContractOutcome
	{
		[Variant]
		public static partial Rejected Validation(string field);

		[DiscriminatedUnion]
		public abstract partial record Security : Rejected
		{
			[Variant]
			public static partial Security Unauthorized(string reason);
		}
	}
}

Each level gets its own Match(...), while nested values still participate in outer matching:

ContractOutcome outcome = ContractOutcome.Rejected.Security.Unauthorized("expired-session");

var description = outcome.Match(
	success => $"success:{success.ContractId}",
	rejected => rejected.Match(
		validation => $"validation:{validation.Field}",
		security => security.Match(
			unauthorized => $"unauthorized:{unauthorized.Reason}")));

See the sample in Zooper.Gorilla.Sample/ContractOutcome.cs.

๐Ÿงช Sample Project

The sample project includes:

๐Ÿค Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

๐Ÿ“„ License

This project is licensed under the MIT License - see the LICENSE file for details.

๐Ÿ™ Acknowledgements

Made with โค๏ธ by the Zooper team

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

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.3.0 93 5/22/2026
1.2.0 139 4/29/2026
1.1.0 103 4/17/2026
1.0.5 168 3/17/2026
1.0.4 113 3/17/2026
1.0.3 102 3/17/2026
1.0.2 677 4/16/2025
1.0.2-preview.7 64 3/17/2026