1
1
mirror of https://github.com/github/semantic.git synced 2025-01-04 21:47:07 +03:00
semantic/prototype/doubt-json/main.swift

109 lines
2.6 KiB
Swift
Raw Normal View History

import Cocoa
2015-10-14 21:17:56 +03:00
import Doubt
import Prelude
2015-10-08 02:20:40 +03:00
2015-10-14 21:21:12 +03:00
typealias Term = Cofree<JSONLeaf, Int>
2015-10-14 21:00:46 +03:00
typealias Diff = Free<JSONLeaf, Patch<Term>>
2015-10-14 04:45:48 +03:00
enum JSONLeaf: CustomJSONConvertible, CustomStringConvertible, Equatable {
2015-10-08 02:21:26 +03:00
case Number(Double)
case Boolean(Bool)
case String(Swift.String)
case Null
2015-10-08 03:08:34 +03:00
2015-10-09 17:14:07 +03:00
// MARK: CustomJSONConvertible
var JSON: Doubt.JSON {
switch self {
case let .Number(n):
return .Number(n)
case let .Boolean(b):
return .Boolean(b)
case let .String(s):
return .String(s)
case .Null:
return .Null
}
}
// MARK: CustomStringConvertible
2015-10-08 03:08:34 +03:00
var description: Swift.String {
switch self {
case let .Number(n):
return Swift.String(n)
case let .Boolean(b):
return Swift.String(b)
case let .String(s):
return Swift.String(reflecting: s)
case .Null:
return "null"
}
}
2015-10-08 02:21:26 +03:00
}
2015-10-08 02:43:00 +03:00
func == (left: JSONLeaf, right: JSONLeaf) -> Bool {
switch (left, right) {
case let (.Number(a), .Number(b)):
return a == b
case let (.Boolean(a), .Boolean(b)):
return a == b
case let (.String(a), .String(b)):
return a == b
case (.Null, .Null):
return true
default:
return false
}
}
2015-10-08 02:36:58 +03:00
extension JSON {
init?(path: Swift.String) {
guard let data = try? NSData(contentsOfFile: path, options: []) else { return nil }
guard let object = try? NSJSONSerialization.JSONObjectWithData(data, options: .AllowFragments) else { return nil }
self.init(object: object)
}
2015-10-08 02:36:43 +03:00
var term: Term {
2015-10-14 21:17:56 +03:00
func annotate(json: Syntax<Term, JSONLeaf>) -> Term {
2015-10-14 21:40:21 +03:00
return Cofree(size(json), json)
2015-10-14 21:17:56 +03:00
}
2015-10-14 21:32:26 +03:00
func size(syntax: Syntax<Term, JSONLeaf>) -> Int {
switch syntax {
case .Leaf:
return 1
case let .Indexed(i):
return 1 + i.map { size($0.unwrap) }.reduce(0, combine: +)
case let .Keyed(i):
return 1 + i.values.map { size($0.unwrap) }.reduce(0, combine: +)
}
}
switch self {
case let .Array(a):
2015-10-14 21:17:56 +03:00
return annotate(.Indexed(a.map { $0.term }))
case let .Dictionary(d):
2015-10-14 21:17:56 +03:00
return annotate(.Keyed(Swift.Dictionary(elements: d.map { ($0, $1.term) })))
case let .Number(n):
2015-10-14 21:17:56 +03:00
return annotate(.Leaf(.Number(n)))
case let .Boolean(b):
2015-10-14 21:17:56 +03:00
return annotate(.Leaf(.Boolean(b)))
case let .String(s):
2015-10-14 21:17:56 +03:00
return annotate(.Leaf(.String(s)))
case .Null:
2015-10-14 21:17:56 +03:00
return annotate(.Leaf(.Null))
}
}
}
let arguments = BoundsCheckedArray(array: Process.arguments)
if let a = arguments[1].flatMap(JSON.init), b = arguments[2].flatMap(JSON.init) {
2015-10-14 21:17:56 +03:00
let diff = Algorithm<Term, Diff>(a.term, b.term).evaluate(Cofree.equals(annotation: const(true), leaf: ==))
2015-10-14 21:22:19 +03:00
if let JSON = NSString(data: diff.JSON(ifPure: { $0.JSON { $0.JSON } }, ifLeaf: { $0.JSON }).serialize(), encoding: NSUTF8StringEncoding) {
2015-10-09 17:20:46 +03:00
print(JSON)
}
}