Debugging the parser

This commit is contained in:
Suzanne Soy 2020-08-31 13:36:14 +00:00
parent 829bff6c2b
commit 74c3abcccc
10 changed files with 210 additions and 61 deletions

View File

@ -11,10 +11,10 @@ public static class DefaultGrammar {
.WithOperator("equality", NonAssociative, "int|terminal|additive|multiplicative", S.Eq, "int|terminal|additive|multiplicative") .WithOperator("equality", NonAssociative, "int|terminal|additive|multiplicative", S.Eq, "int|terminal|additive|multiplicative")
.WithOperator("int", NonAssociative, S.Int) .WithOperator("int", NonAssociative, S.Int)
.WithOperator("additive", LeftAssociative, "int|terminal|multiplicative", S.Plus, "int|terminal|multiplicative") .WithOperator("additive", LeftAssociative, "int|terminal|multiplicative", S.Plus, "int|terminal|multiplicative")
// .WithOperator("multiplicative", LeftAssociative, "int|terminal", S.Times, "int|terminal") .WithOperator("multiplicative", LeftAssociative, "int|terminal", S.Times, "int|terminal")
.WithOperator("terminal", NonAssociative, S.Ident) .WithOperator("terminal", NonAssociative, S.Ident)
// This is the root set of operators // This is the root set of operators
.WithOperator("program", NonAssociative, .WithOperator("program", NonAssociative,
// "bool" // TODO: this needs aliases // "bool" // TODO: this needs aliases
"equality|terminal", S.And, "equality|terminal"); "terminal", S.And, "terminal");// TODO: re-add equality|
} }

View File

@ -213,22 +213,36 @@ public static partial class Lexer {
[F] [F]
private partial class SkipInitialEmptyWhitespace { private partial class SkipInitialEmptyWhitespace {
public IImmutableEnumerator<Lexeme> F(
IImmutableEnumerator<Lexeme> lx
)
=> lx.FirstAndRest().Match(
Some: hdtl =>
// skip the initial empty whitespace
"".Equals(hdtl.Item1.lexeme)
? hdtl.Item2
: hdtl.Item1.ImSingleton().Concat(hdtl.Item2),
None: Empty<Lexeme>());
}
// TODO: move this to a .Filter() extension method.
[F]
private partial class DiscardWhitespace {
public IImmutableEnumerator<Lexeme> F( public IImmutableEnumerator<Lexeme> F(
IImmutableEnumerator<Lexeme> lx IImmutableEnumerator<Lexeme> lx
) )
=> lx.FirstAndRest().Match<Tuple<Lexeme, IImmutableEnumerator<Lexeme>>, IImmutableEnumerator<Lexeme>>( => lx.FirstAndRest().Match<Tuple<Lexeme, IImmutableEnumerator<Lexeme>>, IImmutableEnumerator<Lexeme>>(
Some: hdtl => Some: hdtl =>
// skip the initial empty whitespace // skip the initial empty whitespace
string.Equals( hdtl.Item1.state.Equals(S.Space)
"",
hdtl.Item1.lexeme)
? hdtl.Item2 ? hdtl.Item2
: hdtl.Item1.ImSingleton().Concat(hdtl.Item2), : hdtl.Item1.ImSingleton().Concat(hdtl.Item2.Lazy(DiscardWhitespace.Eq)),
None: Empty<Lexeme>()); None: Empty<Lexeme>());
} }
public static IImmutableEnumerator<Lexeme> Lex(string source) public static IImmutableEnumerator<Lexeme> Lex(string source)
=> Lex1(source) => Lex1(source)
.Flatten() .Flatten()
.Lazy(SkipInitialEmptyWhitespace.Eq); //.Lazy(SkipInitialEmptyWhitespace.Eq)
.Lazy(DiscardWhitespace.Eq);
} }

161
MixFix.cs
View File

@ -21,9 +21,20 @@ public static partial class MixFix {
public static Grammar1 Or(params Grammar1[] xs) public static Grammar1 Or(params Grammar1[] xs)
=> Or(xs.ToImmutableList()); => Or(xs.ToImmutableList());
public static Grammar1 Empty
= new Grammar1.Cases.Sequence(Enumerable.Empty<Grammar1>());
public static Grammar1 Impossible
= new Grammar1.Cases.Or(Enumerable.Empty<Grammar1>());
// TODO: inline the OR, detect impossible cases (Or of 0)
public static Grammar1 Sequence(IEnumerable<Grammar1> xs) { public static Grammar1 Sequence(IEnumerable<Grammar1> xs) {
var filteredXs = xs.Where(x => !x.IsEmpty); var filteredXs = xs.Where(x => !x.IsEmpty);
if (filteredXs.Count() == 1) { if (filteredXs.Any(x => x.IsImpossible)) {
return Impossible;
} else if (filteredXs.Count() == 0) {
return Empty;
} else if (filteredXs.Count() == 1) {
return filteredXs.Single().ElseThrow(() => new Exception("TODO: use an either to prove that this is safe.")); return filteredXs.Single().ElseThrow(() => new Exception("TODO: use an either to prove that this is safe."));
} else { } else {
return new Grammar1.Cases.Sequence(filteredXs); return new Grammar1.Cases.Sequence(filteredXs);
@ -31,8 +42,17 @@ public static partial class MixFix {
} }
public static Grammar1 Or(IEnumerable<Grammar1> xs) { public static Grammar1 Or(IEnumerable<Grammar1> xs) {
var filteredXs = xs.Where(x => !x.IsEmpty); var filteredXsNoEmpty =
if (filteredXs.Count() == 1) { xs.Where(x => !x.IsImpossible)
.Where(x => !x.IsEmpty);
var filteredXs =
( xs.Any(x => x.IsEmpty)
&& !filteredXsNoEmpty.Any(x => x.AllowsEmpty))
? Empty.Cons(filteredXsNoEmpty)
: filteredXsNoEmpty;
if (filteredXs.All(x => x.IsImpossible)) {
return new Grammar1.Cases.Or(Enumerable.Empty<Grammar1>());
} else if (filteredXs.Count() == 1) {
return filteredXs.Single().ElseThrow(() => new Exception("TODO: use an either to prove that this is safe.")); return filteredXs.Single().ElseThrow(() => new Exception("TODO: use an either to prove that this is safe."));
} else { } else {
return new Grammar1.Cases.Or(filteredXs); return new Grammar1.Cases.Or(filteredXs);
@ -42,20 +62,43 @@ public static partial class MixFix {
public static Grammar1 RepeatOnePlus(Grammar1 g) public static Grammar1 RepeatOnePlus(Grammar1 g)
=> g.IsEmpty => g.IsEmpty
? Grammar1.Empty ? Grammar1.Empty
: g.IsImpossible
? Grammar1.Impossible
: new Grammar1.Cases.RepeatOnePlus(g); : new Grammar1.Cases.RepeatOnePlus(g);
public static Grammar1 Empty = new Grammar1.Cases.Or(Enumerable.Empty<Grammar1>());
public bool IsEmpty { public bool IsEmpty {
get => this.Match( get => this.Match(
Or: l => l.Count() == 0, Or: l => l.All(g => g.IsEmpty),
Sequence: l => l.Count() == 0, Sequence: l => l.All(g => g.IsEmpty),
RepeatOnePlus: g => g.IsEmpty, RepeatOnePlus: g => g.IsEmpty,
Terminal: t => false, Terminal: t => false,
Rule: r => false Rule: r => false
); );
} }
// TODO: cache this!
public bool AllowsEmpty {
get => this.Match(
Or: l => l.Any(g => g.AllowsEmpty),
Sequence: l => l.All(g => g.IsEmpty),
// This one should not be true, if it is
// then the precedence graph may be ill-formed?
RepeatOnePlus: g => g.AllowsEmpty,
Terminal: t => false,
Rule: r => false
);
}
public bool IsImpossible {
get => this.Match(
Or: l => l.All(g => g.IsImpossible),
Sequence: l => l.Any(g => g.IsImpossible),
RepeatOnePlus: g => g.IsImpossible,
Terminal: t => false,
Rule: r => false
);
}
public static Grammar1 operator |(Grammar1 a, Grammar1 b) public static Grammar1 operator |(Grammar1 a, Grammar1 b)
=> Or(a, b); => Or(a, b);
@ -65,8 +108,8 @@ public static partial class MixFix {
public static implicit operator Grammar1((Grammar1 a, Grammar1 b, Grammar1 c) gs) public static implicit operator Grammar1((Grammar1 a, Grammar1 b, Grammar1 c) gs)
=> Sequence(gs.a, gs.b, gs.c); => Sequence(gs.a, gs.b, gs.c);
public static bool operator true(Grammar1 g) => !g.IsEmpty; public static bool operator true(Grammar1 g) => !g.IsImpossible;
public static bool operator false(Grammar1 g) => g.IsEmpty; public static bool operator false(Grammar1 g) => g.IsImpossible;
public Grammar1 this[string multiplicity] { public Grammar1 this[string multiplicity] {
get { get {
@ -85,20 +128,55 @@ public static partial class MixFix {
private string Paren(bool paren, string s) private string Paren(bool paren, string s)
=> paren ? $"({s})" : s; => paren ? $"({s})" : s;
string CustomToString() string CustomToString() =>
=> this.Match<string>( this.IsEmpty ? "Empty"
Or: l => l.Count() == 0 : this.IsImpossible ? "Impossible"
? "Or(Empty)" : this.Match<string>(
: Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(" | ")), Or: l =>
Sequence: l => l.Count() == 0 Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(" | ")),
? "Sequence(Empty)" Sequence: l =>
: Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(",")), Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(", ")),
RepeatOnePlus: g => $"({g.Str()})+", RepeatOnePlus: g => $"{g.Str()}+",
Terminal: t => t.Str(), Terminal: t => t.Str(),
Rule: r => r Rule: r => r
); );
} }
public partial class Grammar2 {
public bool IsEmpty {
get => this.Match(
Or: l => l.All(g => g.IsEmpty),
Sequence: l => l.All(g => g.IsEmpty),
RepeatOnePlus: g => g.IsEmpty,
Terminal: t => false
);
}
public bool IsImpossible {
get => this.Match(
Or: l => l.All(g => g.IsImpossible),
Sequence: l => l.Any(g => g.IsImpossible),
RepeatOnePlus: g => g.IsImpossible,
Terminal: t => false
);
}
private string Paren(bool paren, string s)
=> paren ? $"({s})" : s;
string CustomToString() =>
this.IsEmpty ? "Empty"
: this.IsImpossible ? "Impossible"
: this.Match<string>(
Or: l =>
Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(" | ")),
Sequence: l =>
Paren(l.Count() != 1, l.Select(x => x.Str()).JoinWith(", ")),
RepeatOnePlus: g => $"{g.Str()}+",
Terminal: t => t.Str()
);
}
public partial class Operator { public partial class Operator {
private string CustomToString() private string CustomToString()
=> $"Operator(\"{precedenceGroup}\", {fixity}, {parts.Select(x => x.Match(Hole: h => h.Select(g => g.ToString()).JoinWith("|"), Name: n => $"\"{n}\"")).JoinWith(", ")})"; => $"Operator(\"{precedenceGroup}\", {fixity}, {parts.Select(x => x.Match(Hole: h => h.Select(g => g.ToString()).JoinWith("|"), Name: n => $"\"{n}\"")).JoinWith(", ")})";
@ -222,7 +300,7 @@ public static partial class MixFix {
= new PrecedenceDAG(EmptyDAGNode); = new PrecedenceDAG(EmptyDAGNode);
public static Whole Add<Whole>(this ILens<DAGNode, Whole> node, Operator @operator) { public static Whole Add<Whole>(this ILens<DAGNode, Whole> node, Operator @operator) {
return @operator.fixity.Match( return @operator.Cons(@operator.fixity.Match(
Closed: Closed:
() => node.Closed(), () => node.Closed(),
Prefix: Prefix:
@ -235,7 +313,7 @@ public static partial class MixFix {
() => node.InfixRightAssociative(), () => node.InfixRightAssociative(),
InfixLeftAssociative: InfixLeftAssociative:
() => node.InfixLeftAssociative() () => node.InfixLeftAssociative()
).Cons(@operator); ));
} }
public static void CheckHole(PrecedenceDAG precedenceDAG, Operator @operator, string name, Option<Hole> existing, Option<Hole> @new) { public static void CheckHole(PrecedenceDAG precedenceDAG, Operator @operator, string name, Option<Hole> existing, Option<Hole> @new) {
@ -344,15 +422,27 @@ public static partial class MixFix {
var infixl = node.infixLeftAssociative.ToGrammar1(); var infixl = node.infixLeftAssociative.ToGrammar1();
var infixr = node.infixRightAssociative.ToGrammar1(); var infixr = node.infixRightAssociative.ToGrammar1();
//Log("closed.IsImpossible:"+closed.IsImpossible);
// TODO: BUG: only include these parts if there are // TODO: BUG: only include these parts if there are
// any operators with that fixity. // any operators with that fixity, with the code below
// they are empty.
return return
closed (closed ? closed : Grammar1.Impossible)
| (nonAssoc ? (lsucc, nonAssoc, rsucc) : Grammar1.Empty) | (nonAssoc ? (lsucc, nonAssoc, rsucc) : Grammar1.Impossible)
// TODO: post-processsing of the rightassoc list.
| ((prefix || infixr) ? ((prefix || (lsucc, infixr))["+"], rsucc) : Grammar1.Impossible)
// TODO: post-processsing of the leftassoc list. // TODO: post-processsing of the leftassoc list.
| ((prefix || infixr) ? ((prefix | (lsucc, infixr))["+"], rsucc) : Grammar1.Empty) | ((postfix || infixl) ? (lsucc, (postfix || (infixl, rsucc))["+"]) : Grammar1.Impossible);
// TODO: post-processsing of the leftassoc list.
| ((postfix || infixl) ? (lsucc, (postfix | (infixl, rsucc))["+"]) : Grammar1.Empty);
} }
public static EquatableDictionary<string, Grammar1> ToGrammar1(this PrecedenceDAG precedenceDAG) public static EquatableDictionary<string, Grammar1> ToGrammar1(this PrecedenceDAG precedenceDAG)
@ -363,15 +453,28 @@ public static partial class MixFix {
private static Grammar2 Recur(Func<Grammar1, EquatableDictionary<string, Grammar1>, Grammar2> recur, Grammar1 grammar1, EquatableDictionary<string, Grammar1> labeled) private static Grammar2 Recur(Func<Grammar1, EquatableDictionary<string, Grammar1>, Grammar2> recur, Grammar1 grammar1, EquatableDictionary<string, Grammar1> labeled)
=> grammar1.Match<Grammar2>( => grammar1.Match<Grammar2>(
// TODO: throw exception if lookup fails // TODO: throw exception if lookup fails
Rule: r => recur(labeled[r], labeled), Rule: r => {
Grammar1 lr = null;
try {
lr = labeled[r];
} catch (Exception e) {
throw new ParserExtensionException($"Internal error: could not find node {r} in labeled grammar. It only contains labels for: {labeled.Select(kvp => kvp.Key.ToString()).JoinWith(", ")}.");
}
return recur(labeled[r], labeled);
},
Terminal: t => Grammar2.Terminal(t), Terminal: t => Grammar2.Terminal(t),
Sequence: l => Grammar2.Sequence(l.Select(g => recur(g, labeled))), Sequence: l => Grammar2.Sequence(l.Select(g => recur(g, labeled))),
Or: l => Grammar2.Or(l.Select(g => recur(g, labeled))), Or: l => Grammar2.Or(l.Select(g => recur(g, labeled))),
RepeatOnePlus: g => Grammar2.RepeatOnePlus(recur(g, labeled)) RepeatOnePlus: g => Grammar2.RepeatOnePlus(recur(g, labeled))
); );
public static Grammar2 ToGrammar2(this EquatableDictionary<string, Grammar1> labeled) public static Grammar2 ToGrammar2(this EquatableDictionary<string, Grammar1> labeled) {
=> Func.YMemoize<Grammar1, EquatableDictionary<string, Grammar1>, Grammar2>(Recur)(Grammar1.Rule("program"), labeled); foreach (var kvp in labeled) {
Log($"{kvp.Key} -> {kvp.Value.ToString()}");
}
Log("");
return Func.YMemoize<Grammar1, EquatableDictionary<string, Grammar1>, Grammar2>(Recur)(Grammar1.Rule("program"), labeled);
}
public static Grammar2 ToGrammar2(this PrecedenceDAG precedenceDAG) public static Grammar2 ToGrammar2(this PrecedenceDAG precedenceDAG)
=> precedenceDAG.ToGrammar1().ToGrammar2(); => precedenceDAG.ToGrammar1().ToGrammar2();

View File

@ -25,26 +25,34 @@ public static partial class Parser {
.FirstAndRest() .FirstAndRest()
.Match( .Match(
None: () => None: () =>
throw new Exception("EOF, what to do?"), //throw new Exception("EOF, what to do?"),
None<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>>(),
Some: firstRest => { Some: firstRest => {
// Case("IImmutableEnumerable<AstNode>", "Operator"))
var first = firstRest.Item1; var first = firstRest.Item1;
var rest = firstRest.Item2; var rest = firstRest.Item2;
Log(first.lexeme);
Log(grammar.ToString());
Log(grammar.Match(
RepeatOnePlus: _ => "RepeatOnePlus",
Or: _ => "Or",
Sequence: _ => "Sequence",
Terminal: t => "Terminal:"+t.ToString()));
return grammar.Match( return grammar.Match(
RepeatOnePlus: g => RepeatOnePlus: g =>
rest.FoldMapWhileSome(restI => Parse3(restI, g)) rest.FoldMapWhileSome(restI => Parse3(restI, g))
.If<IImmutableEnumerator<Lexeme>, IEnumerable<AstNode>>((restN, nodes) => nodes.Count() > 1) .If<IImmutableEnumerator<Lexeme>, IEnumerable<AstNode>>((restN, nodes) => nodes.Count() > 1)
.IfSome((restN, nodes) => (restN, AstNode.Operator(nodes))), .IfSome((restN, nodes) => (restN, AstNode.Operator(nodes))),
//.IfSome(rest1 =>
// TODO: remove IfSome above (useless) && aggregate
// WhileSome(rest1, restI => Parse3(restI, g))),
// TODO: to check for ambiguous parses, we can use // TODO: to check for ambiguous parses, we can use
// .SingleArg(…) instead of .FirstArg(…). // .Single(…) instead of .First(…).
Or: l => Or: l =>
l.First(g => Parse3(rest, g)), l.First(g => Parse3(rest, g)),
Sequence: l => Sequence: l => {
l.BindFoldMap(rest, (restI, g) => Parse3(restI, g)) return l.BindFoldMap(rest, (restI, g) => Parse3(restI, g))
.IfSome((restN, nodes) => (restN, AstNode.Operator(nodes))), .IfSome((restN, nodes) => {
Log($"{nodes.Count()}/{l.Count()}");
return (restN, AstNode.Operator(nodes));
});
},
Terminal: t => Terminal: t =>
first.state.Equals(t) first.state.Equals(t)
? (rest, ? (rest,
@ -60,7 +68,7 @@ public static partial class Parser {
public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>> Parse2(string source) { public static Option<ValueTuple<IImmutableEnumerator<Lexeme>, AstNode>> Parse2(string source) {
Grammar2 grammar = Grammar2 grammar =
DefaultGrammar.DefaultPrecedenceDAG.ToGrammar2(); DefaultGrammar.DefaultPrecedenceDAG.ToGrammar2();
Log(grammar.Str()); //Log(grammar.Str());
var P = Func.YMemoize< var P = Func.YMemoize<
IImmutableEnumerator<Lexeme>, IImmutableEnumerator<Lexeme>,
@ -73,6 +81,11 @@ public static partial class Parser {
} }
public static Ast.Expr Parse(string source) { public static Ast.Expr Parse(string source) {
Log("");
Log("" + Parse2(source).ToString());
Log("");
Environment.Exit(0);
return Lexer.Lex(source) return Lexer.Lex(source)
.SelectMany(lexeme => .SelectMany(lexeme =>
lexeme.state.Match( lexeme.state.Match(

View File

@ -1 +1 @@
42 true && false

View File

@ -8,14 +8,20 @@ public static class Collection {
public static void ForEach<T>(this IEnumerable<T> x, Action<T> f) public static void ForEach<T>(this IEnumerable<T> x, Action<T> f)
=> x.ToImmutableList().ForEach(f); => x.ToImmutableList().ForEach(f);
public static ImmutableList<T> Cons<T>(this ImmutableList<T> l, T x) public static ImmutableList<T> Cons<T>(this T x, ImmutableList<T> l)
=> l.Add(x); => l.Add(x);
public static ImmutableList<Tuple<T,U>> Cons<T,U>(this ImmutableList<Tuple<T,U>> l, T x, U y) public static IEnumerable<T> Cons<T>(this T x, IEnumerable<T> l)
=> l.Cons(Tuple.Create(x,y)); => x.Singleton().Concat(l);
public static ImmutableList<Tuple<T,U,V>> Cons<T,U,V>(this ImmutableList<Tuple<T,U,V>> l, T x, U y, V z) public static IEnumerable<T> Concat<T>(this IEnumerable<T> l, T x)
=> l.Cons(Tuple.Create(x,y,z)); => l.Concat(x.Singleton());
public static ImmutableList<Tuple<T,U>> Add<T,U>(this ImmutableList<Tuple<T,U>> l, T x, U y)
=> l.Add(Tuple.Create(x,y));
public static ImmutableList<Tuple<T,U,V>> Add<T,U,V>(this ImmutableList<Tuple<T,U,V>> l, T x, U y, V z)
=> l.Add(Tuple.Create(x,y,z));
public static void Deconstruct<A, B>(this Tuple<A, B> t, out A a, out B b) { public static void Deconstruct<A, B>(this Tuple<A, B> t, out A a, out B b) {
a = t.Item1; a = t.Item1;
@ -241,7 +247,7 @@ public static class Collection {
foreach (var x in e) { foreach (var x in e) {
var newAcc = f(acc, x); var newAcc = f(acc, x);
if (newAcc.IsNone) { if (newAcc.IsNone) {
break; return Option.None<A>();
} else { } else {
acc = newAcc.ElseThrow(new Exception("impossible")); acc = newAcc.ElseThrow(new Exception("impossible"));
} }

View File

@ -27,6 +27,9 @@ namespace Immutable {
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
=> this.GetEnumerator(); => this.GetEnumerator();
public override string ToString()
=> $"Some({value.ToString()})";
} }
public class None<T> : Option<T>, System.Collections.IEnumerable { public class None<T> : Option<T>, System.Collections.IEnumerable {
@ -42,6 +45,9 @@ namespace Immutable {
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
=> this.GetEnumerator(); => this.GetEnumerator();
public override string ToString()
=> $"None";
} }
} }
} }

View File

@ -10,8 +10,8 @@ public static class LensExtensionMethods {
public static Whole Update<Hole, Whole>(this ILens<Hole, Whole> lens, Hole newHole) public static Whole Update<Hole, Whole>(this ILens<Hole, Whole> lens, Hole newHole)
=> lens.Update(oldHole => newHole); => lens.Update(oldHole => newHole);
public static Whole Cons<T, Whole>(this ILens<ImmutableList<T>, Whole> lens, T value) public static Whole Cons<T, Whole>(this T value, ILens<ImmutableList<T>, Whole> lens)
=> lens.Update(oldHole => oldHole.Cons(value)); => lens.Update(oldHole => value.Cons(oldHole));
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<string, Whole> ChainLens<Whole>(this string hole, System.Func<string, Whole> wrap) => new LeafLens<string, Whole>(wrap: wrap, oldHole: hole);

View File

@ -1,5 +1,6 @@
using System; using System;
using System.Linq; using System.Linq;
using System.Collections.Generic;
using System.Collections.Immutable; using System.Collections.Immutable;
public interface IString { public interface IString {
@ -25,6 +26,12 @@ public static class ToStringImplementations {
public static string Str<T>(this ImmutableHashSet<string> h) public static string Str<T>(this ImmutableHashSet<string> h)
=> $"ImmutableHashSet({h.Select(x => x.Str<string>()).JoinWith(", ")})"; => $"ImmutableHashSet({h.Select(x => x.Str<string>()).JoinWith(", ")})";
public static string Str<T>(this IEnumerable<MixFix.Grammar2> e)
=> $"IEnumerabl({e.Select(x => x.Str<MixFix.Grammar2>()).JoinWith(", ")})";
public static string Str<T>(this IEnumerable<Ast.AstNode> e)
=> $"IEnumerab({e.Select(x => x.Str<Ast.AstNode>()).JoinWith(", ")})";
public static string Str<Grammar>(this ImmutableDictionary<string,Grammar> h) public static string Str<Grammar>(this ImmutableDictionary<string,Grammar> h)
=> $"ImmutableDictionary(\n{h.Select(x => $" {x.Key.Str<string>()}:{x.Value.Str<Grammar>()}").JoinWith(",\n")}\n)"; => $"ImmutableDictionary(\n{h.Select(x => $" {x.Key.Str<string>()}:{x.Value.Str<Grammar>()}").JoinWith(",\n")}\n)";

View File

@ -47,7 +47,7 @@ public static class MainClass {
Console.WriteLine($"\x1b[1;33m{source}: expected {expectedStr} but got {actualStr}.\x1b[m\n"); Console.WriteLine($"\x1b[1;33m{source}: expected {expectedStr} but got {actualStr}.\x1b[m\n");
return false; return false;
} else { } else {
Console.Write("\x1b[1;32mOK\x1b[m\r"); Console.Write("\x1b[1;32mOK\x1b[m"); // \r at the end for quiet
return true; return true;
} }
} }
@ -58,13 +58,13 @@ public static class MainClass {
// first-class functions by using repeated .Add() // first-class functions by using repeated .Add()
// See https://repl.it/@suzannesoy/WarlikeWorstTraining#main.cs // See https://repl.it/@suzannesoy/WarlikeWorstTraining#main.cs
var compilers = ImmutableList<Tuple<string, Compiler, Exe>>.Empty var compilers = ImmutableList<Tuple<string, Compiler, Exe>>.Empty
.Cons(" js ", Compilers.JS.Compile, Exe("node")) .Add(" js ", Compilers.JS.Compile, Exe("node"))
.Cons("eval", Evaluator.Evaluate, Exe("cat")); .Add("eval", Evaluator.Evaluate, Exe("cat"));
var total = 0; var total = 0;
var passed = 0; var passed = 0;
var failed = 0; var failed = 0;
foreach (var t in Dir("Tests/").GetFiles("*.e", SearchOption.AllDirectories)) { foreach (var t in Dir("Tests/").GetFiles("*.e", SearchOption.AllDirectories).OrderBy(f => f.ToString())) {
foreach (var compiler in compilers) { foreach (var compiler in compilers) {
if (RunTest(compiler.Item1, compiler.Item2, compiler.Item3, t)) { if (RunTest(compiler.Item1, compiler.Item2, compiler.Item3, t)) {
passed++; passed++;