namespace Immutable { using System; public interface Option : System.Collections.Generic.IEnumerable { U Match_(Func Some, Func None); bool IsSome { get; } bool IsNone { get; } } public static class Option { public static Option Some(T value) => new Types.Some(value); public static Option None() => new Types.None(); private static class Types { public class Some : Option, System.Collections.IEnumerable { public readonly T value; public Some(T value) { this.value = value; } public U Match_(Func Some, Func None) => Some(value); public bool IsSome { get => true; } public bool IsNone { get => false; } public System.Collections.Generic.IEnumerator GetEnumerator() => value.Singleton().GetEnumerator(); System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() => this.GetEnumerator(); } public class None : Option, System.Collections.IEnumerable { public None() { } public U Match_(Func Some, Func None) => None(); public bool IsSome { get => false; } public bool IsNone { get => true; } public System.Collections.Generic.IEnumerator GetEnumerator() => System.Linq.Enumerable.Empty().GetEnumerator(); System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() => this.GetEnumerator(); } } } public static class OptionExtensionMethods { public static Option Some(this T value) => Option.Some(value); public static U Match(this Option o, Func Some, Func None) => o.Match_(Some: Some, None: None); public static U Match(this Option o, Func Some, U None) => o.Match_(Some: Some, None: () => None); public static Option Map(this Option o, Func some) => o.Match_(Some: value => some(value).Some(), None: () => Option.None()); public static Option IfSome(this Option o, Func some) => o.Map(some); public static Option Bind(this Option o, Func> f) => o.Match_(Some: some => f(some), None: () => Option.None()); public static T Else(this Option o, Func none) => o.Match_(Some: some => some, None: none); public static T Else(this Option o, T none) => o.Match_(Some: some => some, None: () => none); public static Option Else(this Option o, Func> none) => o.Match_(Some: value => value.Some(), None: none); public static T ElseThrow(this Option o, Func none) => o.Match_(Some: value => value, None: () => throw none()); } }