using System; using System.Linq; using System.Collections.Generic; using System.Collections.Immutable; using Immutable; public static class Collection { public static void ForEach(this IEnumerable x, Action f) => x.ToImmutableList().ForEach(f); public static ImmutableList Cons(this ImmutableList l, T x) => l.Add(x); public static ImmutableList> Cons(this ImmutableList> l, T x, U y) => l.Cons(Tuple.Create(x,y)); public static ImmutableList> Cons(this ImmutableList> l, T x, U y, V z) => l.Cons(Tuple.Create(x,y,z)); public static void Deconstruct(this Tuple t, out A a, out B b) { a = t.Item1; b = t.Item2; } public struct Item { public readonly T item; public readonly long index; public readonly bool first; public readonly bool last; public Item(T item, long index, bool first, bool last) { this.item = item; this.index = index; this.first = first; this.last = last; } } public static IEnumerable> Indexed(this IEnumerable e) { long i = 0L; bool first = true; // These default(…) are written below before being read T prevX = default(T); long prevI = default(long); bool prevFirst = default(bool); foreach (var x in e) { if (!first) { yield return new Item(prevX, prevI, prevFirst, false); } prevX = x; prevI = i; prevFirst = first; first = false; i++; } if (!first) { yield return new Item(prevX, prevI, prevFirst, true); } } public struct Peekable : IEnumerator, System.Collections.IEnumerator { private IEnumerator e; private bool peeked; private T previous; public T Current { get => peeked ? previous : e.Current; } object System.Collections.IEnumerator.Current { get => this.Current; } public bool MoveNext() { this.peeked = false; this.previous = default(T); // guarded by peeked return this.e.MoveNext(); } public bool Peek() { if (this.peeked) { throw new Exception("Already peeked once"); } else { this.previous = e.Current; this.peeked = true; return this.e.MoveNext(); } } public void Dispose() { e.Dispose(); } public void Reset() { e.Reset(); } public Peekable(IEnumerable e) { this.e = e.GetEnumerator(); this.peeked = false; this.previous = default(T); // guarded by peeked } } public static Peekable Peekable(this IEnumerable e) { return new Peekable(e); } public static IEnumerable SingleUseEnumerable(this IEnumerator e) { while (e.MoveNext()) { yield return e.Current; } } public static IEnumerable TakeUntil(this IEnumerable e, Func f) => e.TakeWhile(x => !f(x)); public static IEnumerable Singleton(this T x) { yield return x; } public static string Join(this string separator, IEnumerable strings) => String.Join(separator, strings); public static Option First(this IEnumerable ie) { var e = ie.GetEnumerator(); if (e.MoveNext()) { return e.Current.Some(); } else { return Option.None(); } } public static Option Last(this IEnumerable ie) { var e = ie.GetEnumerator(); T element = default(T); bool found = false; while (e.MoveNext()) { element = e.Current; found = true; } if (found) { return element.Some(); } else { return Option.None(); } } public static Option First(this IEnumerable ie, Func predicate) { var e = ie.GetEnumerator(); while (e.MoveNext()) { if (predicate(e.Current)) { return e.Current.Some(); } } return Option.None(); } public static Option First(this IEnumerable ie, Func> selector) { var e = ie.GetEnumerator(); while (e.MoveNext()) { var found = selector(e.Current); if (found.IsSome) { return found; } } return Option.None(); } public static Option Single(this IEnumerable ie) { var e = ie.GetEnumerator(); if (e.MoveNext()) { var value = e.Current; if (e.MoveNext()) { return Option.None(); } else { return value.Some(); } } else { return Option.None(); } } public static Option GetValue(this ImmutableDictionary d, K key) { V result = default(V); if (d.TryGetValue(key, out result)) { return result.Some(); } else { return Option.None(); } } public static V GetOrDefault(this ImmutableDictionary d, K key, V defaultValue) { V result = default(V); if (d.TryGetValue(key, out result)) { return result; } else { return defaultValue; } } public static Option Aggregate(this IEnumerable ie, Func f) { var e = ie.GetEnumerator(); if (e.MoveNext()) { var accumulator = e.Current; while (e.MoveNext()) { accumulator = f(accumulator, e.Current); } return accumulator.Some(); } return Option.None(); } public static string JoinWith(this IEnumerable strings, string joiner) // TODO: use StringBuilder, there is no complexity info in the docs. => String.Join(joiner, strings); public static string JoinToStringWith(this IEnumerable objects, string joiner) // TODO: use StringBuilder, there is no complexity info in the docs. => String.Join(joiner, objects.Select(o => o.ToString())); public static bool SetEquals(this ImmutableHashSet a, ImmutableHashSet b) => a.All(x => b.Contains(x)) && b.All(x => a.Contains(x)); public static (Option firstElement, ImmutableQueue rest) Dequeue(this ImmutableQueue q) { if (q.IsEmpty) { return (Option.None(), q); } else { T firstElement; var rest = q.Dequeue(out firstElement); return (firstElement.Some(), rest); } } public static IEnumerable SkipLastWhile(this IEnumerable e, Func predicate) { var pending = ImmutableQueue.Empty; foreach (var x in e) { if (predicate(x)) { pending = pending.Enqueue(x); } else { while (!pending.IsEmpty) { yield return pending.Peek(); pending = pending.Dequeue(); } yield return x; } } } public static Option BindFold(this IEnumerable e, A init, Func> f) { var acc = init; foreach (var x in e) { var newAcc = f(acc, x); if (newAcc.IsNone) { break; } else { acc = newAcc.ElseThrow(new Exception("impossible")); } } return acc.Some(); } public static Option>> BindFoldMap(this IEnumerable e, A init, Func>> f) => e.BindFold( (init, ImmutableStack.Empty), (accL, x) => f(accL.Item1, x).IfSome((newAcc, result) => (newAcc, accL.Item2.Push(result))) ).IfSome((acc, l) => (acc, l.Reverse())); public static A FoldWhileSome(this A init, Func> f) { var lastGood = init; while (true) { var @new = f(lastGood); if (@new.IsNone) { return lastGood; } else { lastGood = @new.ElseThrow(() => new Exception("impossible")); } } } public static Option> FoldWhileSome(this Option> init, Func>> f) => init.IfSome(ab1 => ab1.FoldWhileSome(ab => f(ab.Item1, ab.Item2))); public static Option> FoldWhileSome(this Option> init, Func>> f) => init.IfSome(ab1 => ab1.FoldWhileSome(ab => f(ab.Item1, ab.Item2))); public static ValueTuple> FoldMapWhileSome(this A init, Func>> f) => FoldWhileSome( (init, ImmutableStack.Empty), accL => f(accL.Item1).IfSome((newAcc, result) => (newAcc, accL.Item2.Push(result))) ).Pipe((acc, l) => (acc, l.Reverse())); }