Cleaned up metaprogramming utility and lenses for records
This commit is contained in:
parent
b3d41496d5
commit
36ed190a5c
54
Lexer.cs
54
Lexer.cs
|
@ -14,45 +14,47 @@ public static partial class Lexer {
|
||||||
public static class Rules {
|
public static class Rules {
|
||||||
private static Rule Rule(S oldState, UnicodeCategory cat, S throughState, S newState = null)
|
private static Rule Rule(S oldState, UnicodeCategory cat, S throughState, S newState = null)
|
||||||
=> new Rule(
|
=> new Rule(
|
||||||
oldState,
|
oldState: oldState,
|
||||||
cat.ToString(),
|
description: cat.ToString(),
|
||||||
c => c.codePoints
|
test: c => c.codePoints
|
||||||
.First()
|
.First()
|
||||||
.Match(some: (x => x.UnicodeCategory(0) == cat),
|
.Match(some: (x => x.UnicodeCategory(0) == cat),
|
||||||
none: false),
|
none: false),
|
||||||
throughState,
|
throughState: throughState,
|
||||||
newState ?? throughState);
|
newState: newState ?? throughState);
|
||||||
|
|
||||||
private static Rule Rule(S oldState, EOF eof, S throughState, S newState = null)
|
private static Rule Rule(S oldState, EOF eof, S throughState, S newState = null)
|
||||||
=> new Rule(
|
=> new Rule(
|
||||||
oldState,
|
oldState: oldState,
|
||||||
"End of file",
|
description: "End of file",
|
||||||
c => c.endOfFile,
|
test: c => c.endOfFile,
|
||||||
throughState,
|
throughState: throughState,
|
||||||
newState ?? throughState);
|
newState: newState ?? throughState);
|
||||||
|
|
||||||
private static string CharDescription(char c)
|
private static string CharDescription(char c)
|
||||||
=> (c == '"') ? "'\"'" : $"\"{c.ToString()}\"";
|
=> (c == '"') ? "'\"'" : $"\"{c.ToString()}\"";
|
||||||
|
|
||||||
private static Rule Rule(S oldState, char c, S throughState, S newState = null)
|
private static Rule Rule(S oldState, char c, S throughState, S newState = null)
|
||||||
=> new Rule(
|
=> new Rule(
|
||||||
oldState,
|
oldState: oldState,
|
||||||
CharDescription(c),
|
description: CharDescription(c),
|
||||||
x => x.codePoints
|
test: x => x.codePoints
|
||||||
.Single()
|
.Single()
|
||||||
.Match(some: xx => xx == c.ToString(),
|
.Match(some: xx => xx == c.ToString(),
|
||||||
none: false),
|
none: false),
|
||||||
throughState,
|
throughState: throughState,
|
||||||
newState ?? throughState);
|
newState: newState ?? throughState);
|
||||||
|
|
||||||
private static Rule Rule(S oldState, char[] cs, S throughState, S newState = null) {
|
private static Rule Rule(S oldState, char[] cs, S throughState, S newState = null) {
|
||||||
var csl = cs.Select(x => x.ToString()).ToImmutableList();
|
var csl = cs.Select(x => x.ToString()).ToImmutableList();
|
||||||
return new Rule(
|
return new Rule(
|
||||||
oldState,
|
oldState: oldState,
|
||||||
", ".Join(cs.Select(CharDescription)),
|
description: ", ".Join(cs.Select(CharDescription)),
|
||||||
x => x.codePoints.Single().Match(some: csl.Contains, none: false),
|
test: x => x.codePoints
|
||||||
throughState,
|
.Single()
|
||||||
newState ?? throughState);
|
.Match(some: csl.Contains, none: false),
|
||||||
|
throughState: throughState,
|
||||||
|
newState: newState ?? throughState);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static EOF EOF = new EOF();
|
public static EOF EOF = new EOF();
|
||||||
|
|
24
Makefile
24
Makefile
|
@ -1,22 +1,26 @@
|
||||||
CS := $(shell find . -not \( -path ./.git \) -not \( -name '*Generator.cs' \) -name '*.cs')
|
CS := $(shell find . -not \( -path ./.git \) -not \( -name '*Generator.cs' \) -not \( -name '*Generated.cs' \) -name '*.cs')
|
||||||
GENERATORS := $(shell find . -not \( -path ./.git \) -not \( -path ./T4/Generator.cs \) -name '*Generator.cs')
|
META := $(shell find ./T4/ -name '*.cs')
|
||||||
|
GENERATORS := $(shell find . -not \( -path ./.git \) -not \( -path './T4/*' \) -name '*Generator.cs')
|
||||||
GENERATED := $(patsubst %Generator.cs,%Generated.cs,$(GENERATORS))
|
GENERATED := $(patsubst %Generator.cs,%Generated.cs,$(GENERATORS))
|
||||||
|
|
||||||
.PHONY: run
|
.PHONY: run
|
||||||
run: main.exe
|
run: main.exe Makefile
|
||||||
MONO_PATH=/usr/lib/mono/4.5/:/usr/lib/mono/4.5/Facades/ mono main.exe
|
MONO_PATH=/usr/lib/mono/4.5/:/usr/lib/mono/4.5/Facades/ mono $<
|
||||||
|
|
||||||
main.exe: $(sort $(CS) $(GENERATED))
|
main.exe: $(CS) $(GENERATED) Makefile
|
||||||
mcs -out:$@ \
|
mcs -out:$@ \
|
||||||
/reference:/usr/lib/mono/fsharp/FSharp.Core.dll \
|
/reference:/usr/lib/mono/fsharp/FSharp.Core.dll \
|
||||||
/reference:/usr/lib/mono/4.5/System.Collections.Immutable.dll \
|
/reference:/usr/lib/mono/4.5/System.Collections.Immutable.dll \
|
||||||
/reference:/usr/lib/mono/4.5/Facades/netstandard.dll \
|
/reference:/usr/lib/mono/4.5/Facades/netstandard.dll \
|
||||||
$^
|
$(filter-out Makefile, $^)
|
||||||
|
|
||||||
%Generated.cs: .%Generator.exe
|
%Generated.cs: .%Generator.exe Makefile
|
||||||
mono $<
|
MONO_PATH=/usr/lib/mono/4.5/:/usr/lib/mono/4.5/Facades/ mono $(filter-out Makefile, $<)
|
||||||
|
|
||||||
.%Generator.exe: %Generator.cs T4/Generator.cs
|
.%Generator.exe: %Generator.cs $(META) Makefile
|
||||||
mcs -out:$@ $^
|
mcs -out:$@ \
|
||||||
|
/reference:/usr/lib/mono/4.5/System.Collections.Immutable.dll \
|
||||||
|
/reference:/usr/lib/mono/4.5/Facades/netstandard.dll \
|
||||||
|
$(filter-out Makefile, $^)
|
||||||
|
|
||||||
|
|
||||||
|
|
20
Parser.cs
20
Parser.cs
|
@ -13,20 +13,20 @@ public static partial class Parser {
|
||||||
public static PrecedenceDAG DefaultPrecedenceDAG = new PrecedenceDAG();
|
public static PrecedenceDAG DefaultPrecedenceDAG = new PrecedenceDAG();
|
||||||
|
|
||||||
public static DAGNode With(DAGNode node, Operator op) {
|
public static DAGNode With(DAGNode node, Operator op) {
|
||||||
/* var newOp = op.fixity.Match(
|
var newOp = op.fixity.Match(
|
||||||
Closed: () => node.WithClosed(op),
|
Closed: () => node.lens.closed.Cons(op),
|
||||||
InfixLeftAssociative: () => node.WithInfixLeftAssociative(op),
|
InfixLeftAssociative: () => node.lens.infixLeftAssociative.Cons(op),
|
||||||
InfixRightAssociative: () => node.WithInfixRightAssociative(op),
|
InfixRightAssociative: () => node.lens.infixRightAssociative.Cons(op),
|
||||||
InfixNonAssociative: () => node.WithInfixNonAssociative(op),
|
InfixNonAssociative: () => node.lens.infixNonAssociative.Cons(op),
|
||||||
Prefix: () => node.WithPrefix(op),
|
Prefix: () => node.lens.prefix.Cons(op),
|
||||||
Postfix: () => node.WithPostFix(op),
|
Postfix: () => node.lens.postfix.Cons(op),
|
||||||
Terminal: () => node.WithTerminal(op)
|
Terminal: () => node.lens.terminal.Cons(op)
|
||||||
);*/
|
);
|
||||||
// op.fixity, parts, holes
|
// op.fixity, parts, holes
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static PrecedenceDAG With(PrecedenceDAG precedenceDAG, Operator @operator) {
|
public static PrecedenceDAG With(PrecedenceDAG precedenceDAG, Operator @operator) {
|
||||||
/*precedenceDAG.update(
|
/*precedenceDAG.update(
|
||||||
dagNode => dagNode.Add(@operator)
|
dagNode => dagNode.Add(@operator)
|
||||||
);*/
|
);*/
|
||||||
|
|
197
T4/Generator.cs
197
T4/Generator.cs
|
@ -1,210 +1,59 @@
|
||||||
// Code quality of this file: low.
|
// Code quality of this file: medium.
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.Collections.Immutable;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
|
||||||
public enum Kind {
|
public enum Kind {
|
||||||
Record,
|
Record,
|
||||||
Variant,
|
Variant,
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class Generator {
|
public static class Generator {
|
||||||
public static void WriteVariant(this System.IO.StreamWriter o, string header, string footer, string qualifier, string name, Dictionary<string, string> variant) {
|
public static void Generate(string outputFile, string singleHeader, string header, string footer, string qualifier, ImmutableDictionary<string, Tuple<Kind, ImmutableDictionary<string, string>>> types) {
|
||||||
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} : IEquatable<{name}> {{");
|
|
||||||
|
|
||||||
o.WriteLine($" public class Visitor<T> {{");
|
|
||||||
foreach (var @case in variant) {
|
|
||||||
var C = @case.Key;
|
|
||||||
var Ty = @case.Value;
|
|
||||||
|
|
||||||
o.WriteLine($" public Func<{Ty == null ? "" : $"{Ty}, "}T> {C} {{ get; set; }} ");
|
|
||||||
}
|
|
||||||
o.WriteLine($" }}");
|
|
||||||
|
|
||||||
o.WriteLine($" public abstract T Match_<T>(Visitor<T> c);");
|
|
||||||
foreach (var @case in variant) {
|
|
||||||
var C = @case.Key;
|
|
||||||
var Ty = @case.Value;
|
|
||||||
o.WriteLine($" public static {name} {C}{Ty == null
|
|
||||||
? $" = new Constructors.{C}()"
|
|
||||||
: $"({Ty} value) => new Constructors.{C}(value)"};");
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach (var @case in variant) {
|
|
||||||
var C = @case.Key;
|
|
||||||
var Ty = @case.Value;
|
|
||||||
o.WriteLine($" public virtual Immutable.Option<Constructors.{C}> As{C}() => Immutable.Option.None<Constructors.{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($" }}");
|
|
||||||
o.WriteLine($"");
|
|
||||||
o.WriteLine($" public override abstract bool Equals(Object other);");
|
|
||||||
o.WriteLine($" public abstract bool Equals({name} other);");
|
|
||||||
o.WriteLine($"");
|
|
||||||
o.WriteLine($" public override abstract int GetHashCode();");
|
|
||||||
o.WriteLine($" public static class Constructors {{");
|
|
||||||
|
|
||||||
foreach (var @case in variant) {
|
|
||||||
var C = @case.Key;
|
|
||||||
var Ty = @case.Value;
|
|
||||||
|
|
||||||
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_<T>(Visitor<T> c) => c.{C}({Ty == null ? "" : "value"});");
|
|
||||||
o.WriteLine($" public override Immutable.Option<{C}> As{C}() => Immutable.Option.Some<{C}>(this);");
|
|
||||||
o.WriteLine($" public static bool operator ==({C} a, {C} b)");
|
|
||||||
o.WriteLine($" => Equality.Operator(a, b);");
|
|
||||||
o.WriteLine($" public static bool operator !=({C} a, {C} b)");
|
|
||||||
o.WriteLine($" => !(a == b);");
|
|
||||||
o.WriteLine($" public override bool Equals(object other)");
|
|
||||||
if (Ty == null) {
|
|
||||||
o.WriteLine($" => Equality.Untyped<{C}>(this, other, x => x as {C});");
|
|
||||||
} else {
|
|
||||||
o.WriteLine($" => Equality.Untyped<{C}>(this, other, x => x as {C}, x => x.value);");
|
|
||||||
}
|
|
||||||
o.WriteLine($" public override bool Equals({name} other)");
|
|
||||||
o.WriteLine($" => Equality.Equatable<{name}>(this, other);");
|
|
||||||
o.WriteLine($" public override int GetHashCode()");
|
|
||||||
if (Ty == null) {
|
|
||||||
o.WriteLine($" => HashCode.Combine(\"{C}\");");
|
|
||||||
} else {
|
|
||||||
o.WriteLine($" => HashCode.Combine(\"{C}\", this.value);");
|
|
||||||
}
|
|
||||||
o.WriteLine("");
|
|
||||||
o.WriteLine($" public override string ToString() => \"{C}\";");
|
|
||||||
o.WriteLine($" }}");
|
|
||||||
o.WriteLine("");
|
|
||||||
}
|
|
||||||
|
|
||||||
o.WriteLine($" }}");
|
|
||||||
o.WriteLine($" }}");
|
|
||||||
o.WriteLine("");
|
|
||||||
o.WriteLine($"}}");
|
|
||||||
|
|
||||||
o.WriteLine($"public static class {name}ExtensionMethods {{");
|
|
||||||
o.WriteLine($" public static T Match<T>(");
|
|
||||||
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}{name}.Visitor<T> {{");
|
|
||||||
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<string, string> record) {
|
|
||||||
o.WriteLine($"{header}");
|
|
||||||
o.WriteLine("");
|
|
||||||
o.WriteLine($" public sealed class {name} : IEquatable<{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($" public static bool operator ==({name} a, {name} b)");
|
|
||||||
o.WriteLine($" => Equality.Operator(a, b);");
|
|
||||||
o.WriteLine($" public static bool operator !=({name} a, {name} b)");
|
|
||||||
o.WriteLine($" => !(a == b);");
|
|
||||||
o.WriteLine($" public override bool Equals(object other)");
|
|
||||||
o.WriteLine($" => Equality.Untyped<{name}>(this, other, x => x as {name},");
|
|
||||||
o.WriteLine(String.Join(",\n", record.Select(@field =>
|
|
||||||
$" x => x.{@field.Key}")));
|
|
||||||
o.WriteLine($" );");
|
|
||||||
o.WriteLine($" public bool Equals({name} other)");
|
|
||||||
o.WriteLine($" => Equality.Equatable<{name}>(this, other);");
|
|
||||||
o.WriteLine($" public override int GetHashCode()");
|
|
||||||
o.WriteLine($" => Equality.HashCode(\"{name}\",");
|
|
||||||
o.WriteLine(String.Join(",\n", record.Select(@field =>
|
|
||||||
$" this.{@field.Key}")));
|
|
||||||
o.WriteLine($" );");
|
|
||||||
foreach (var @field in record) {
|
|
||||||
var F = @field.Key;
|
|
||||||
var noAtF = F.StartsWith("@") ? F.Substring(1) : F;
|
|
||||||
var caseF = Char.ToUpper(noAtF[0]) + noAtF.Substring(1);
|
|
||||||
var Ty = @field.Value;
|
|
||||||
o.Write($" public {name} With{caseF}({Ty} {F}) => new {name}(");
|
|
||||||
o.Write(String.Join(", ", record.Select(@f => $"{f.Key}: {f.Key}")));
|
|
||||||
o.WriteLine(");");
|
|
||||||
}
|
|
||||||
|
|
||||||
o.WriteLine($" }}");
|
|
||||||
o.WriteLine($"{footer}");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void Generate(string outputFile, string singleHeader, string header, string footer, string qualifier, Dictionary<string, Tuple<Kind, Dictionary<string, string>>> types) {
|
|
||||||
using (var o = new System.IO.StreamWriter(outputFile)) {
|
using (var o = new System.IO.StreamWriter(outputFile)) {
|
||||||
o.WriteLine("// This file was generated by Generator.cs");
|
Action<string> w = o.WriteLine;
|
||||||
o.WriteLine("");
|
w("// This file was generated by Generator.cs");
|
||||||
|
w("");
|
||||||
|
|
||||||
o.WriteLine("using System;");
|
w("using System;");
|
||||||
o.WriteLine($"{singleHeader}");
|
w($"{singleHeader}");
|
||||||
o.WriteLine("");
|
w("");
|
||||||
foreach (var type in types) {
|
foreach (var type in types) {
|
||||||
var name = type.Key;
|
var name = type.Key;
|
||||||
var kind = type.Value.Item1;
|
var kind = type.Value.Item1;
|
||||||
var components = type.Value.Item2;
|
var components = type.Value.Item2;
|
||||||
switch (kind) {
|
switch (kind) {
|
||||||
case Kind.Record:
|
case Kind.Record:
|
||||||
o.WriteRecord(header, footer, qualifier, name, @components);
|
w.Record(header, footer, qualifier, name, @components);
|
||||||
break;
|
break;
|
||||||
case Kind.Variant:
|
case Kind.Variant:
|
||||||
o.WriteVariant(header, footer, qualifier, name, @components);
|
w.Variant(header, footer, qualifier, name, @components);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
o.WriteLine("");
|
w("");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Below are shorthands for making the last argument to Generate().
|
// Below are shorthands for making the last argument to Generate().
|
||||||
public static Dictionary<string, Tuple<Kind, Dictionary<string, string>>> Types(params Tuple<string, Tuple<Kind, Dictionary<string, string>>>[] types)
|
public static ImmutableDictionary<string, Tuple<Kind, ImmutableDictionary<string, string>>> Types(params Tuple<string, Tuple<Kind, ImmutableDictionary<string, string>>>[] types)
|
||||||
=> types.ToDictionary(t => t.Item1, t => t.Item2);
|
=> types.ToImmutableDictionary(t => t.Item1, t => t.Item2);
|
||||||
|
|
||||||
public static Tuple<string, Tuple<Kind, Dictionary<string, string>>> Record(string name, params Tuple<string, string>[] fields)
|
public static Tuple<string, Tuple<Kind, ImmutableDictionary<string, string>>> Record(string name, params Tuple<string, string>[] fields)
|
||||||
=> new Tuple<string, Tuple<Kind, Dictionary<string, string>>>(
|
=> new Tuple<string, Tuple<Kind, ImmutableDictionary<string, string>>>(
|
||||||
name,
|
name,
|
||||||
new Tuple<Kind, Dictionary<string, string>>(
|
new Tuple<Kind, ImmutableDictionary<string, string>>(
|
||||||
Kind.Record,
|
Kind.Record,
|
||||||
fields.ToDictionary(t => t.Item1, t => t.Item2)));
|
fields.ToImmutableDictionary(t => t.Item1, t => t.Item2)));
|
||||||
|
|
||||||
public static Tuple<string, Tuple<Kind, Dictionary<string, string>>> Variant(string name, params Tuple<string, string>[] cases)
|
public static Tuple<string, Tuple<Kind, ImmutableDictionary<string, string>>> Variant(string name, params Tuple<string, string>[] cases)
|
||||||
=> new Tuple<string, Tuple<Kind, Dictionary<string, string>>>(
|
=> new Tuple<string, Tuple<Kind, ImmutableDictionary<string, string>>>(
|
||||||
name,
|
name,
|
||||||
new Tuple<Kind, Dictionary<string, string>>(
|
new Tuple<Kind, ImmutableDictionary<string, string>>(
|
||||||
Kind.Variant,
|
Kind.Variant,
|
||||||
cases.ToDictionary(t => t.Item1, t => t.Item2)));
|
cases.ToImmutableDictionary(t => t.Item1, t => t.Item2)));
|
||||||
|
|
||||||
public static Tuple<string, string> Field(string type, string name)
|
public static Tuple<string, string> Field(string type, string name)
|
||||||
=> new Tuple<string, string>(name, type);
|
=> new Tuple<string, string>(name, type);
|
||||||
|
|
115
T4/RecordGenerator.cs
Normal file
115
T4/RecordGenerator.cs
Normal file
|
@ -0,0 +1,115 @@
|
||||||
|
// Code quality of this file: medium.
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using Record = System.Collections.Immutable.ImmutableDictionary<string, string>;
|
||||||
|
|
||||||
|
public static class RecordGenerator {
|
||||||
|
private static void Fields(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
foreach (var @field in record) {
|
||||||
|
var F = @field.Key;
|
||||||
|
var Ty = @field.Value;
|
||||||
|
w($" public readonly {Ty} {F};");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Constructor(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public {name}(");
|
||||||
|
w(String.Join(",\n", record.Select(@field =>
|
||||||
|
$" {@field.Value} {@field.Key}")));
|
||||||
|
w($" ) {{");
|
||||||
|
foreach (var @field in record) {
|
||||||
|
var F = @field.Key;
|
||||||
|
var Ty = @field.Value;
|
||||||
|
w($" this.{F} = {F};");
|
||||||
|
}
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Equality(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public static bool operator ==({name} a, {name} b)");
|
||||||
|
w($" => Equality.Operator(a, b);");
|
||||||
|
w($" public static bool operator !=({name} a, {name} b)");
|
||||||
|
w($" => !(a == b);");
|
||||||
|
w($" public override bool Equals(object other)");
|
||||||
|
w($" => Equality.Untyped<{name}>(this, other, x => x as {name},");
|
||||||
|
w(String.Join(",\n", record.Select(@field =>
|
||||||
|
$" x => x.{@field.Key}")));
|
||||||
|
w($" );");
|
||||||
|
w($" public bool Equals({name} other)");
|
||||||
|
w($" => Equality.Equatable<{name}>(this, other);");
|
||||||
|
w($" public override int GetHashCode()");
|
||||||
|
w($" => Equality.HashCode(\"{name}\",");
|
||||||
|
w(String.Join(",\n", record.Select(@field =>
|
||||||
|
$" this.{@field.Key}")));
|
||||||
|
w($" );");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void With(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
foreach (var @field in record) {
|
||||||
|
var F = @field.Key;
|
||||||
|
var noAtF = F.StartsWith("@") ? F.Substring(1) : F;
|
||||||
|
var caseF = Char.ToUpper(noAtF[0]) + noAtF.Substring(1);
|
||||||
|
var Ty = @field.Value;
|
||||||
|
w($" public {name} With{caseF}({Ty} {F}) => new {name}("
|
||||||
|
+ String.Join(", ", record.Select(@f => $"{f.Key}: {f.Key}"))
|
||||||
|
+ ");");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Lens(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public Lens<{name}> lens {{ get => ChainLens(x => x); }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void ChainLens(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public Lens<Whole> ChainLens<Whole>(System.Func<{name}, Whole> wrap) => new Lens<Whole>(wrap: wrap, oldHole: this);");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Lenses(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public sealed class Lens<Whole> : ILens<{name}, Whole> {{");
|
||||||
|
w($" public readonly System.Func<{name}, Whole> wrap;");
|
||||||
|
w($" public readonly {name} oldHole;");
|
||||||
|
w($"");
|
||||||
|
w($" public Lens(System.Func<{name}, Whole> wrap, {name} oldHole) {{");
|
||||||
|
w($" this.wrap = wrap;");
|
||||||
|
w($" this.oldHole = oldHole;");
|
||||||
|
w($" }}");
|
||||||
|
foreach (var @field in record) {
|
||||||
|
var F = @field.Key;
|
||||||
|
var noAtF = F.StartsWith("@") ? F.Substring(1) : F;
|
||||||
|
var caseF = Char.ToUpper(noAtF[0]) + noAtF.Substring(1);
|
||||||
|
var Ty = @field.Value;
|
||||||
|
w($" public ILens<{Ty},Whole> {F}");
|
||||||
|
w($" => oldHole.{F}.ChainLens(");
|
||||||
|
w($" value => wrap(oldHole.With{caseF}(value)));");
|
||||||
|
}
|
||||||
|
w($" public Whole Update(Func<{name}, {name}> update) => wrap(update(oldHole));");
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void RecordClass(this Action<string> w, string qualifier, string name, Record record) {
|
||||||
|
w($" public sealed class {name} : IEquatable<{name}> {{");
|
||||||
|
w.Fields(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.Constructor(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.Equality(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.With(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.Lens(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.ChainLens(qualifier, name, record);
|
||||||
|
w($"");
|
||||||
|
w.Lenses(qualifier, name, record);
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void Record(this Action<string> w, string header, string footer, string qualifier, string name, Record record) {
|
||||||
|
w($"{header}");
|
||||||
|
w("");
|
||||||
|
w.RecordClass(qualifier, name, record);
|
||||||
|
w($"{footer}");
|
||||||
|
}
|
||||||
|
}
|
192
T4/VariantGenerator.cs
Normal file
192
T4/VariantGenerator.cs
Normal file
|
@ -0,0 +1,192 @@
|
||||||
|
// Code quality of this file: medium.
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using Variant = System.Collections.Immutable.ImmutableDictionary<string, string>;
|
||||||
|
|
||||||
|
public static class VariantGenerator {
|
||||||
|
private static void MatchExampleComment(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" /* To match against an instance of {name}, write:");
|
||||||
|
w($" x.Match(");
|
||||||
|
w(String.Join(",\n", variant.Select(@case =>
|
||||||
|
$" {@case.Key}: {@case.Value == null ? "()" : "value"} => throw new NotImplementedException()")));
|
||||||
|
w($" )");
|
||||||
|
w($" */");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void PrivateConstructor(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" private {name}() {{}}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Visitor(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public class Visitor<T> {{");
|
||||||
|
foreach (var @case in variant) {
|
||||||
|
var C = @case.Key;
|
||||||
|
var Ty = @case.Value;
|
||||||
|
|
||||||
|
w($" public Func<{Ty == null ? "" : $"{Ty}, "}T> {C} {{ get; set; }} ");
|
||||||
|
}
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Match_(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public abstract T Match_<T>(Visitor<T> c);");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseShorthands(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
foreach (var @case in variant) {
|
||||||
|
var C = @case.Key;
|
||||||
|
var Ty = @case.Value;
|
||||||
|
w($" public static {name} {C}{Ty == null
|
||||||
|
? $" = new Cases.{C}()"
|
||||||
|
: $"({Ty} value) => new Cases.{C}(value)"};");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void As(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
foreach (var @case in variant) {
|
||||||
|
var C = @case.Key;
|
||||||
|
var Ty = @case.Value;
|
||||||
|
w($" public virtual Immutable.Option<{Ty == null ? "Immutable.Unit" : Ty}> As{C}() => Immutable.Option.None<{Ty == null ? "Immutable.Unit" : Ty}>();");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Lens(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public LeafLens<{name}> Lens {{ get => ChainLens(x => x); }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void ChainLens(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public LeafLens<{name}, Whole> ChainLens<Whole>(System.Func<{name}, Whole> wrap) => new LeafLens<{name}, Whole>(wrap: wrap, oldHole: this);");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void GetTag(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" private string GetTag() {{");
|
||||||
|
w($" return this.Match(");
|
||||||
|
w(String.Join(",\n", variant.Select(@case =>
|
||||||
|
$" {@case.Key}: {@case.Value == null ? "()" : "value"} => \"{@case.Key}\"")));
|
||||||
|
w($" );");
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Equality(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public static bool operator ==({name} a, {name} b)");
|
||||||
|
w($" => Equality.Operator(a, b);");
|
||||||
|
w($" public static bool operator !=({name} a, {name} b)");
|
||||||
|
w($" => !(a == b);");
|
||||||
|
w($" public override abstract bool Equals(Object other);");
|
||||||
|
w($" public abstract bool Equals({name} other);");
|
||||||
|
w($"");
|
||||||
|
w($" public override abstract int GetHashCode();");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseValue(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
if (Ty != null) {
|
||||||
|
w($" public readonly {Ty} value;");
|
||||||
|
w($"");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseConstructor(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
w($" public {C}({Ty == null ? "" : $"{Ty} value"}) {{ {Ty == null ? "" : $"this.value = value; "}}}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseMatch_(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
w($" public override T Match_<T>(Visitor<T> c) => c.{C}({Ty == null ? "" : "value"});");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseAs(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
w($" public override Immutable.Option<{Ty == null ? "Immutable.Unit" : Ty}> As{C}() => Immutable.Option.Some<{Ty == null ? "Immutable.Unit" : Ty}>({Ty == null ? "Immutable.Unit.unit" : "this.value"});");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseEquality(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
w($" public static bool operator ==({C} a, {C} b)");
|
||||||
|
w($" => Equality.Operator(a, b);");
|
||||||
|
w($" public static bool operator !=({C} a, {C} b)");
|
||||||
|
w($" => !(a == b);");
|
||||||
|
w($" public override bool Equals(object other)");
|
||||||
|
if (Ty == null) {
|
||||||
|
w($" => Equality.Untyped<{C}>(this, other, x => x as {C});");
|
||||||
|
} else {
|
||||||
|
w($" => Equality.Untyped<{C}>(this, other, x => x as {C}, x => x.value);");
|
||||||
|
}
|
||||||
|
w($" public override bool Equals({name} other)");
|
||||||
|
w($" => Equality.Equatable<{name}>(this, other);");
|
||||||
|
w($" public override int GetHashCode()");
|
||||||
|
if (Ty == null) {
|
||||||
|
w($" => HashCode.Combine(\"{C}\");");
|
||||||
|
} else {
|
||||||
|
w($" => HashCode.Combine(\"{C}\", this.value);");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void CaseToString(this Action<string> w, string qualifier, string name, string C, string Ty) {
|
||||||
|
w($" public override string ToString() => \"{C}\";");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void Cases(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
foreach (var @case in variant) {
|
||||||
|
var C = @case.Key;
|
||||||
|
var Ty = @case.Value;
|
||||||
|
|
||||||
|
w($" public sealed class {C} : {name} {{");
|
||||||
|
w.CaseValue(qualifier, name, C, Ty);
|
||||||
|
w.CaseConstructor(qualifier, name, C, Ty);
|
||||||
|
w($"");
|
||||||
|
w.CaseMatch_(qualifier, name, C, Ty);
|
||||||
|
w($"");
|
||||||
|
w.CaseAs(qualifier, name, C, Ty);
|
||||||
|
w($"");
|
||||||
|
w.CaseEquality(qualifier, name, C, Ty);
|
||||||
|
w($"");
|
||||||
|
w.CaseToString(qualifier, name, C, Ty);
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void VariantClass(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($" public abstract class {name} : IEquatable<{name}> {{");
|
||||||
|
w.PrivateConstructor(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.Visitor(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.Match_(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.CaseShorthands(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.As(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.ChainLens(qualifier, name, variant);
|
||||||
|
w($"");
|
||||||
|
w.Equality(qualifier, name, variant);
|
||||||
|
w($" public static class Cases {{");
|
||||||
|
w.Cases(qualifier, name, variant);
|
||||||
|
w($" }}");
|
||||||
|
w($" }}");
|
||||||
|
w("");
|
||||||
|
w($"}}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void ExtensionMethods(this Action<string> w, string qualifier, string name, Variant variant) {
|
||||||
|
w($"public static class {name}ExtensionMethods {{");
|
||||||
|
w($" public static T Match<T>(");
|
||||||
|
w($" this {qualifier}{name} e,");
|
||||||
|
w(String.Join(",\n", variant.Select(c =>
|
||||||
|
$" Func<{c.Value == null ? "" : $"{c.Value}, "}T> {c.Key}")));
|
||||||
|
w($" ) {{");
|
||||||
|
w($" return e.Match_(new {qualifier}{name}.Visitor<T> {{");
|
||||||
|
w(String.Join(",\n", variant.Select(c =>
|
||||||
|
$" {c.Key} = {c.Key}")));
|
||||||
|
w($" }});");
|
||||||
|
w($" }}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void Variant(this Action<string> w, string header, string footer, string qualifier, string name, Variant variant) {
|
||||||
|
w($"{header}");
|
||||||
|
w($"");
|
||||||
|
w.MatchExampleComment(qualifier, name, variant);
|
||||||
|
w.VariantClass(qualifier, name, variant);
|
||||||
|
w.ExtensionMethods(qualifier, name, variant);
|
||||||
|
w($"{footer}");
|
||||||
|
}
|
||||||
|
}
|
|
@ -11,6 +11,8 @@ public static class Global {
|
||||||
|
|
||||||
public static void Log (string str) => Console.WriteLine(str);
|
public static void Log (string str) => Console.WriteLine(str);
|
||||||
|
|
||||||
|
public static Unit unit() => Unit.unit;
|
||||||
|
|
||||||
public static Option<T> None<T>() => Option.None<T>();
|
public static Option<T> None<T>() => Option.None<T>();
|
||||||
|
|
||||||
public static ImmutableList<T> ImmutableList<T>(params T[] xs)
|
public static ImmutableList<T> ImmutableList<T>(params T[] xs)
|
||||||
|
|
19
Utils/Immutable/Unit.cs
Normal file
19
Utils/Immutable/Unit.cs
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
using System;
|
||||||
|
|
||||||
|
namespace Immutable {
|
||||||
|
public sealed class Unit : IEquatable<Unit> {
|
||||||
|
public static readonly Unit unit = new Unit();
|
||||||
|
private Unit() {}
|
||||||
|
public static bool operator ==(Unit a, Unit b)
|
||||||
|
=> Equality.Operator(a, b);
|
||||||
|
public static bool operator !=(Unit a, Unit b)
|
||||||
|
=> !(a == b);
|
||||||
|
public override bool Equals(object other)
|
||||||
|
=> Equality.Untyped<Unit>(this, other, x => x as Unit);
|
||||||
|
public bool Equals(Unit other)
|
||||||
|
=> Equality.Equatable<Unit>(this, other);
|
||||||
|
public override int GetHashCode()
|
||||||
|
=> HashCode.Combine("Unit");
|
||||||
|
public override string ToString() => "Unit";
|
||||||
|
}
|
||||||
|
}
|
64
Utils/Lens.cs
Normal file
64
Utils/Lens.cs
Normal file
|
@ -0,0 +1,64 @@
|
||||||
|
// Code quality of this file: low.
|
||||||
|
|
||||||
|
using System;
|
||||||
|
using System.Collections.Immutable;
|
||||||
|
|
||||||
|
public interface ILens<Hole, Whole> {
|
||||||
|
Whole Update(Func<Hole, Hole> update);
|
||||||
|
}
|
||||||
|
|
||||||
|
public sealed class ImmutableListLens<T, Whole> : ILens<ImmutableList<T>, Whole> {
|
||||||
|
public readonly System.Func<ImmutableList<T>, Whole> wrap;
|
||||||
|
public readonly ImmutableList<T> oldHole;
|
||||||
|
|
||||||
|
public ImmutableListLens(System.Func<ImmutableList<T>, Whole> wrap, ImmutableList<T> oldHole) {
|
||||||
|
this.wrap = wrap;
|
||||||
|
this.oldHole = oldHole;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Put methods with the following signature here to focus on sub-parts of the list as needed.
|
||||||
|
// public ILens<ImmutableList<T>,Whole> sub-part => oldHole.sub-part.ChainLens(value => oldHole.with-sub-part(value));
|
||||||
|
|
||||||
|
public Whole Update(Func<ImmutableList<T>, ImmutableList<T>> update) => wrap(update(oldHole));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lenses for primitive types and other types that are not
|
||||||
|
// interesting to further focus.
|
||||||
|
public sealed class LeafLens<T, Whole> : ILens<T, Whole> {
|
||||||
|
public readonly System.Func<T, Whole> wrap;
|
||||||
|
public readonly T oldHole;
|
||||||
|
|
||||||
|
public LeafLens(System.Func<T, Whole> wrap, T oldHole) {
|
||||||
|
this.wrap = wrap;
|
||||||
|
this.oldHole = oldHole;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Whole Update(Func<T, T> update) => wrap(update(oldHole));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class LensExtensionMethods {
|
||||||
|
public static Whole Update<Hole, Whole>(this ILens<Hole, Whole> lens, Hole newHole)
|
||||||
|
=> lens.Update(oldHole => newHole);
|
||||||
|
|
||||||
|
public static Whole Cons<T, Whole>(this ILens<ImmutableList<T>, Whole> lens, T value)
|
||||||
|
=> lens.Update(oldHole => oldHole.Cons(value));
|
||||||
|
|
||||||
|
public static ImmutableListLens<T, Whole>
|
||||||
|
ChainLens<T, Whole>(
|
||||||
|
this ImmutableList<T> hole,
|
||||||
|
System.Func<ImmutableList<T>, Whole> wrap)
|
||||||
|
=> new ImmutableListLens<T, Whole>(wrap: wrap, oldHole: hole);
|
||||||
|
|
||||||
|
public static ILens<string, Whole> ChainLens<Whole>(this string hole, System.Func<string, Whole> wrap) => new LeafLens<string, Whole>(wrap: wrap, oldHole: hole);
|
||||||
|
|
||||||
|
public static ILens<Func<GraphemeCluster,bool>, Whole> ChainLens<Whole>(this Func<GraphemeCluster,bool> hole, System.Func<Func<GraphemeCluster,bool>, Whole> wrap) => new LeafLens<Func<GraphemeCluster,bool>, Whole>(wrap: wrap, oldHole: hole);
|
||||||
|
|
||||||
|
public class FocusableLeaf<T> {
|
||||||
|
private readonly T value;
|
||||||
|
public FocusableLeaf(T value) { this.value = value; }
|
||||||
|
public LeafLens<T, Whole> ChainLens<Whole>(Func<T, Whole> wrap)
|
||||||
|
=> new LeafLens<T, Whole>(wrap: wrap, oldHole: value);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ILens<ImmutableList<string>, Whole> ChainLens<Whole>(this ImmutableList<string> hole, System.Func<ImmutableList<string>, Whole> wrap) => new ImmutableListLens<string, Whole>(wrap: wrap, oldHole: hole);
|
||||||
|
}
|
|
@ -13,6 +13,9 @@ public struct GraphemeCluster {
|
||||||
this.codePoints = codePoints;
|
this.codePoints = codePoints;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public LeafLens<GraphemeCluster, Whole> ChainLens<Whole>(Func<GraphemeCluster, Whole> wrap)
|
||||||
|
=> new LeafLens<GraphemeCluster, Whole>(wrap: wrap, oldHole: this);
|
||||||
|
|
||||||
public static implicit operator GraphemeCluster(char c)
|
public static implicit operator GraphemeCluster(char c)
|
||||||
=> new GraphemeCluster(false, c.ToString(), c.ToString().Singleton());
|
=> new GraphemeCluster(false, c.ToString(), c.ToString().Singleton());
|
||||||
}
|
}
|
||||||
|
|
1
Utils/Variant.cs
Normal file
1
Utils/Variant.cs
Normal file
|
@ -0,0 +1 @@
|
||||||
|
public interface IVariant {}
|
Loading…
Reference in New Issue
Block a user