// Code quality of this file: low. using System; using System.Collections.Generic; using System.Linq; public enum Kind { Record, Variant, } public static class Generator { public static void WriteVariant(this System.IO.StreamWriter o, string header, string footer, string qualifier, string name, Dictionary variant) { o.WriteLine($"{header}"); o.WriteLine(""); o.WriteLine($" /* To match against an instance of {name}, write:"); o.WriteLine($" x.Match("); o.WriteLine(String.Join(",\n", variant.Select(@case => $" {@case.Key}: {@case.Value == null ? "()" : "value"} => throw new NotImplementedYetException(),"))); o.WriteLine($" )"); o.WriteLine($" */"); o.WriteLine($" public abstract class {name} {{"); o.WriteLine($" public abstract T Match_(Visitor c);"); foreach (var @case in variant) { var C = @case.Key; var Ty = @case.Value; o.WriteLine($" public static {name} {C}{Ty == null ? $" = new {C}()" : $"({Ty} value) => new {C}(value)"};"); } foreach (var @case in variant) { var C = @case.Key; var Ty = @case.Value; o.WriteLine($" public virtual Immutable.Option<{C}> As{C}() => Immutable.Option.None<{C}>();"); } o.WriteLine($" private string GetTag() {{"); o.WriteLine($" return this.Match("); o.WriteLine(String.Join(",\n", variant.Select(@case => $" {@case.Key}: {@case.Value == null ? "()" : "value"} => \"{@case.Key}\""))); o.WriteLine($" );"); o.WriteLine($" }}"); /* public abstract override bool Equals(object other) { if (object.ReferenceEquals(other, null) || !this.GetType().Equals(other.GetType())) { return false; } else { var cast = (S)other; return String.Equal(this.GetTag(), other.GetTag) && } */ o.WriteLine($" }}"); o.WriteLine(""); foreach (var @case in variant) { var C = @case.Key; var Ty = @case.Value; o.WriteLine( $" public partial class Visitor {{" + $" public Func<{Ty == null ? "" : $"{Ty}, "}T> {C} {{ get; set; }} " + @"}"); o.WriteLine(""); o.WriteLine($" public sealed class {C} : {name} {{"); if (Ty != null) { o.WriteLine($" public readonly {Ty} value;"); } o.WriteLine($" public {C}({Ty == null ? "" : $"{Ty} value"}) {{ {Ty == null ? "" : $"this.value = value; "}}}"); o.WriteLine($" public override T Match_(Visitor c) => c.{C}({Ty == null ? "" : "value"});"); o.WriteLine($" public override Immutable.Option<{C}> As{C}() => Immutable.Option.Some<{C}>(this);"); o.WriteLine($" public override bool Equals(object other) {{"); if (Ty == null) { o.WriteLine($" return (other is {C});"); } else { o.WriteLine($" var cast = other as {C};"); o.WriteLine($" if (Object.ReferenceEquals(cast, null)) {{"); o.WriteLine($" return false;"); o.WriteLine($" }} else {{"); o.WriteLine($" return Equality.Field<{C}, {Ty}>(this, cast, x => x.value, (x, y) => ((Object)x).Equals(y));"); o.WriteLine($" }}"); } o.WriteLine($" }}"); o.WriteLine($" public override int GetHashCode() {{"); if (Ty == null) { o.WriteLine($" return \"C\".GetHashCode();"); } else { o.WriteLine($" return HashCode.Combine(\"{C}\", this.value);"); } o.WriteLine($" }}"); o.WriteLine(""); o.WriteLine($" public override string ToString() => \"{C}\";"); o.WriteLine($" }}"); o.WriteLine(""); } o.WriteLine($"}}"); o.WriteLine($"public static class {name}ExtensionMethods {{"); o.WriteLine($" public static T Match("); o.WriteLine($" this {qualifier}{name} e,"); o.WriteLine(String.Join(",\n", variant.Select(c => $" Func<{c.Value == null ? "" : $"{c.Value}, "}T> {c.Key}"))); o.WriteLine($" ) {{"); o.WriteLine($" return e.Match_(new {qualifier}Visitor {{"); o.WriteLine(String.Join(",\n", variant.Select(c => $" {c.Key} = {c.Key}"))); o.WriteLine($" }});"); o.WriteLine($" }}"); o.WriteLine($"{footer}"); } public static void WriteRecord(this System.IO.StreamWriter o, string header, string footer, string qualifier, string name, Dictionary record) { o.WriteLine($"{header}"); o.WriteLine(""); o.WriteLine($" public class {name} {{"); foreach (var @field in record) { var F = @field.Key; var Ty = @field.Value; o.WriteLine($" public readonly {Ty} {F};"); } o.WriteLine($" public {name}("); o.WriteLine(String.Join(",\n", record.Select(@field => $" {@field.Value} {@field.Key}"))); o.WriteLine($" ) {{"); foreach (var @field in record) { var F = @field.Key; var Ty = @field.Value; o.WriteLine($" this.{F} = {F};"); } o.WriteLine($" }}"); o.WriteLine($" }}"); o.WriteLine($"{footer}"); } public static void Generate(string outputFile, string header, string footer, string qualifier, Dictionary>> types) { using (var o = new System.IO.StreamWriter(outputFile)) { o.WriteLine("// This file was generated by Generator.cs"); o.WriteLine(""); o.WriteLine("using System;"); o.WriteLine(""); foreach (var type in types) { var name = type.Key; var kind = type.Value.Item1; var components = type.Value.Item2; switch (kind) { case Kind.Record: o.WriteRecord(header, footer, qualifier, name, @components); break; case Kind.Variant: o.WriteVariant(header, footer, qualifier, name, @components); break; } } } } // Below are shorthands for making the last argument to Generate(). public static Dictionary>> Types(params Tuple>>[] types) => types.ToDictionary(t => t.Item1, t => t.Item2); public static Tuple>> Record(string name, params Tuple[] fields) => new Tuple>>( name, new Tuple>( Kind.Record, fields.ToDictionary(t => t.Item1, t => t.Item2))); public static Tuple>> Variant(string name, params Tuple[] cases) => new Tuple>>( name, new Tuple>( Kind.Variant, cases.ToDictionary(t => t.Item1, t => t.Item2))); public static Tuple Field(string name, string type) => new Tuple(name, type); public static Tuple Case(string name, string type) => new Tuple(name, type); public static Tuple Case(string name) => new Tuple(name, null); }