W4k.Either.CodeGeneration
1.1.0
dotnet add package W4k.Either.CodeGeneration --version 1.1.0
NuGet\Install-Package W4k.Either.CodeGeneration -Version 1.1.0
<PackageReference Include="W4k.Either.CodeGeneration" Version="1.1.0" />
paket add W4k.Either.CodeGeneration --version 1.1.0
#r "nuget: W4k.Either.CodeGeneration, 1.1.0"
// Install W4k.Either.CodeGeneration as a Cake Addin #addin nuget:?package=W4k.Either.CodeGeneration&version=1.1.0 // Install W4k.Either.CodeGeneration as a Cake Tool #tool nuget:?package=W4k.Either.CodeGeneration&version=1.1.0
W4k.Either.CodeGeneration
⚠️ Code generator - despite targeting netstandard2.0
- requires project to use at least net6.0
(due to features used in generated code).
All types distributed with W4k.Either
package are generated using this source generator.
To see how to use those types and what is being generated, please refer to W4k.Either
package
or find generated code snapshots in tests.
The W4k.Either.CodeGeneration
package enables you to generate your own types with custom logic. It generates types in two ways:
Generic type
Whenever you need another type with different properties representing several different things, you can declare generic
type using EitherAttribute
. Compiler will generate rest of type for you:
[Either]
public readonly partial struct Gelf<TCamille, TCat, TCrichton>
{
}
Predefined types
In case you know what types you want to use, you can specify them using EitherAttribute
following way:
// use `EitherAttribute` and provide types in its constructor
[Either(typeof(string), typeof(int))]
public readonly partial struct StringOrInt
{
}
// use generic `EitherAttribute<,>`
[Either<string, int>]
public readonly partial struct StringOrInt
{
}
Installation
Reference W4k.Either.CodeGeneration
package in your project and use generated types:
<ItemGroup>
<PackageReference Include="W4k.Either.CodeGeneration" Version="1.1.0" PrivateAssets="All" />
</ItemGroup>
Usage
Decorating your types
Declare your type as partial
and decorate it with EitherAttribute
. Remember to mark the containing type as partial
as well if the type is nested.
Please adhere to these rules:
- The type itself MUST be
partial
. - If the type is nested, the containing type MUST be
partial
as well. - There must be type defined either as generic type or as predefined type using attribute.
Other properties and behaviors include:
- When declaring a generic type, it is up to you to define the type parameter constraint.
- When implementing
IEquatable<T>
, theEquals
andGetHashCode
methods are automatically generated for you, unless you have implemented them yourself. - When implementing
ISerializable
, the serializable constructor and theGetObjectData
method are automatically generated for you, unless you have implemented them yourself. - You are allowed to declare constructors matching those which would be generated. In that case, the constructors are not generated to avoid causing conflicts.
[Either]
[Serializable] // <- make type serializable
public readonly partial struct Polymorph<T1, T2> : IEquatable<Polymorph<T1, T2>>, ISerializable // <- `IEquatable<>` and `ISerializable` are implemented by code generator for you
where T1 : struct // <- generator reflects constraints in generated code
where T2 : notnull, ICrewMember
{
// you can declare constructor with same signature as generator would normaly produce (generator will skip it)
public Polymorph(T2 value)
{
// NB! you are responsible for checking input - if you care about it ¯\_(ツ)_/¯
ArgumentNullException.ThrowIfNull(value);
// your custom logic
if (value is Lister)
{
throw new ArgumentException("...");
}
// NB! you are responsible for proper initialization of the instance;
_idx = 2;
_v1 = default;
_v2 = value;
}
// custom property
public bool IsRimmer => _idx == 2 && _v2 is Rimmer;
}
Notice that value fields starts at 1
, as well as state index value:
- State is stored in field
_idx
:0
is reserved for invalid state (as this is default value)1
,2
, ...,n
are used as state index (up to255
)
- Values are indexed from
1
as well corresponding to state index (_idx = 1
means that value is kept in field_v1
)
You can declare any constructor you want, but keep in mind you need to initialize the monad properly - always set _idx
and _v#
fields.
Skip generating (some) members
When decorating your type by [Either]
attribute, it's possible to let generator know that you want to skip (some) members
normally generated. This is done by specifying Skip
property of the attribute:
[Either(Skip = new[] { "Case", "Switch*" })]
public partial readonly struct ThisOrThat<TLeft, TRight>
{
}
Code above generates type while skipping generating Case
property and all Switch
methods.
Members you can omit from being generated:
Member name / alias | Skip generating... |
---|---|
"Case" |
Case property |
"TryPick" |
TryPick method |
"Bind*" |
all Bind methods |
"Bind" |
Bind method |
"Bind<TState>" |
Bind method override with provided state |
"Map*" |
all Map methods |
"Map" |
Map method |
"Map<TState>" |
Map method override with provided state |
"Match*" |
all Match , MatchAsync , Switch and SwitchAsync methods |
"Match" |
Match and Switch method |
"Match<TState>" |
Match and Switch method override with provided state |
"MatchAsync" |
MatchAsync and SwitchAsync method |
"MatchAsync<TState>" |
MatchAsync and SwitchAsync method overrides with provided state |
"Switch*" |
all Switch and SwitchAsync methods |
"Switch" |
Switch method |
"Switch<TState>" |
Switch method override with provided state |
"SwitchAsync" |
SwitchAsync method |
"SwitchAsync<TState>" |
SwitchAsync method override with provided state |
Switch
method implementation relies on Match
, thus skipping generation of Match
will also skip Switch
method.
Shapes and symbols icons created by Freepik - Flaticon
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net6.0 is compatible. 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. |
-
.NETStandard 2.0
- No dependencies.
-
net6.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.1.0 | 197 | 7/6/2023 |
1.0.0 | 138 | 7/1/2023 |
0.2.0-preview | 124 | 6/16/2023 |
0.1.0-preview | 124 | 6/10/2023 |