public enum Diff: Comparable, CustomDebugStringConvertible, CustomDocConvertible { case Empty case Patch(Term, Term) indirect case Copy(Syntax) public static func Insert(term: Term) -> Diff { return .Patch(.Empty, term) } public static func Delete(term: Term) -> Diff { return .Patch(term, .Empty) } public init(_ term: Term) { switch term { case .Empty: self = .Empty case let .Roll(s): self = .Copy(s.map(Diff.init)) } } public var doc: Doc { switch self { case .Empty: return .Empty case let .Patch(a, b): return .Horizontal([ .Wrap(.Text("{-"), Doc(a), .Text("-}")), .Wrap(.Text("{+"), Doc(b), .Text("+}")) ]) case let .Copy(a): return a.doc } } public var debugDescription: String { switch self { case .Empty: return ".Empty" case let .Patch(a, b): return ".Patch(\(String(reflecting: a)), \(String(reflecting: b)))" case let .Copy(a): return ".Copy(\(String(reflecting: a)))" } } public var magnitude: Int { switch self { case .Empty: return 0 case .Patch: return 1 case let .Copy(s): return s.map { $0.magnitude }.reduce(0, combine: +) } } public init(_ a: Term, _ b: Term) { switch (a, b) { case (.Empty, .Empty): self = .Empty case let (.Roll(a), .Roll(b)): switch (a, b) { case let (.Apply(a, aa), .Apply(b, bb)): // fixme: SES self = .Copy(.Apply(Diff(a, b), Diff.diff(aa, bb))) case let (.Abstract(p1, b1), .Abstract(p2, b2)): self = .Copy(.Abstract(Diff.diff(p1, p2), Diff(b1, b2))) case let (.Assign(n1, v1), .Assign(n2, v2)) where n1 == n2: self = .Copy(.Assign(n2, Diff(v1, v2))) case let (.Variable(n1), .Variable(n2)) where n1 == n2: self = .Copy(.Variable(n2)) case let (.Literal(v1), .Literal(v2)) where v1 == v2: self = .Copy(.Literal(v2)) case let (.Group(n1, v1), .Group(n2, v2)): self = .Copy(.Group(Diff(n1, n2), Diff.diff(v1, v2))) default: self = .Patch(Term(a), Term(b)) } default: self = Patch(a, b) } } public static func diff(a: C1, _ b: C2) -> [Diff] { func magnitude(diffs: Stream) -> Int { return diffs.map { $0.magnitude }.reduce(0, combine: +) } func min(a: A, _ rest: A..., _ isLessThan: (A, A) -> Bool) -> A { return rest.reduce(a, combine: { isLessThan($0, $1) ? $0 : $1 }) } func diff(a: Stream, _ b: Stream) -> Stream { switch (a, b) { case (.Nil, .Nil): return .Nil case (.Nil, .Cons): return b.map(Diff.Insert) case (.Cons, .Nil): return a.map(Diff.Delete) case let (.Cons(x, xs), .Cons(y, ys)): let here = Stream.Cons(Diff(x, y), Memo { diff(xs.value, ys.value) }) let insert = Stream.Cons(Diff.Insert(y), Memo { diff(a, ys.value) }) let delete = Stream.Cons(Diff.Delete(x), Memo { diff(xs.value, b) }) return min(here, insert, delete) { magnitude($0) < magnitude($1) } } } return Array(diff(Stream(sequence: a), Stream(sequence: b))) } }