* Rewrite stream * Rename HALT to SKIP * Rename HALT to SKIP * Remove valueOf and toString * Update docs for HALT to SKIP * Rename halt to skip in test * Add test for combining nested streams atomically * Update change-log.md * Test basic SKIP * Add deprecated HALT * Combine continues with ended streams * Fix fantasy-land/of to match spec * Don't use arrow function * Improve scan description * Fix merge artifact
648 lines
16 KiB
JavaScript
648 lines
16 KiB
JavaScript
"use strict"
|
|
|
|
var o = require("../../ospec/ospec")
|
|
var Stream = require("../stream")
|
|
|
|
o.spec("stream", function() {
|
|
o.spec("stream", function() {
|
|
o("works as getter/setter", function() {
|
|
var stream = Stream(1)
|
|
var initialValue = stream()
|
|
stream(2)
|
|
var newValue = stream()
|
|
|
|
o(initialValue).equals(1)
|
|
o(newValue).equals(2)
|
|
})
|
|
o("has undefined value by default", function() {
|
|
var stream = Stream()
|
|
|
|
o(stream()).equals(undefined)
|
|
})
|
|
o("can update to undefined", function() {
|
|
var stream = Stream(1)
|
|
stream(undefined)
|
|
|
|
o(stream()).equals(undefined)
|
|
})
|
|
o("can be stream of streams", function() {
|
|
var stream = Stream(Stream(1))
|
|
|
|
o(stream()()).equals(1)
|
|
})
|
|
o("can SKIP", function() {
|
|
var a = Stream(2)
|
|
var b = a.map(function(value) {
|
|
return value === 5
|
|
? Stream.SKIP
|
|
: value
|
|
})
|
|
|
|
a(5)
|
|
|
|
o(b()).equals(2)
|
|
})
|
|
o("can HALT", function() {
|
|
var a = Stream(2)
|
|
var b = a.map(function(value) {
|
|
return value === 5
|
|
? Stream.HALT
|
|
: value
|
|
})
|
|
|
|
a(5)
|
|
|
|
o(b()).equals(2)
|
|
})
|
|
o("warns HALT deprecated", function() {
|
|
var log = console.log
|
|
var warning = ""
|
|
console.log = function(a) {
|
|
warning = a
|
|
}
|
|
|
|
Stream.HALT
|
|
|
|
console.log = log
|
|
|
|
o(warning).equals("HALT is deprecated and has been renamed to SKIP")
|
|
})
|
|
})
|
|
o.spec("combine", function() {
|
|
o("transforms value", function() {
|
|
var stream = Stream()
|
|
var doubled = Stream.combine(function(s) {return s() * 2}, [stream])
|
|
|
|
stream(2)
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("transforms default value", function() {
|
|
var stream = Stream(2)
|
|
var doubled = Stream.combine(function(s) {return s() * 2}, [stream])
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("transforms multiple values", function() {
|
|
var s1 = Stream()
|
|
var s2 = Stream()
|
|
var added = Stream.combine(function(s1, s2) {return s1() + s2()}, [s1, s2])
|
|
|
|
s1(2)
|
|
s2(3)
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("transforms multiple default values", function() {
|
|
var s1 = Stream(2)
|
|
var s2 = Stream(3)
|
|
var added = Stream.combine(function(s1, s2) {return s1() + s2()}, [s1, s2])
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("transforms mixed default and late-bound values", function() {
|
|
var s1 = Stream(2)
|
|
var s2 = Stream()
|
|
var added = Stream.combine(function(s1, s2) {return s1() + s2()}, [s1, s2])
|
|
|
|
s2(3)
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("combines atomically", function() {
|
|
var count = 0
|
|
var a = Stream()
|
|
var b = Stream.combine(function(a) {return a() * 2}, [a])
|
|
var c = Stream.combine(function(a) {return a() * a()}, [a])
|
|
var d = Stream.combine(function(b, c) {
|
|
count++
|
|
return b() + c()
|
|
}, [b, c])
|
|
|
|
a(3)
|
|
|
|
o(d()).equals(15)
|
|
o(count).equals(1)
|
|
})
|
|
|
|
o("combines default value atomically", function() {
|
|
var count = 0
|
|
var a = Stream(3)
|
|
var b = Stream.combine(function(a) {return a() * 2}, [a])
|
|
var c = Stream.combine(function(a) {return a() * a()}, [a])
|
|
var d = Stream.combine(function(b, c) {
|
|
count++
|
|
return b() + c()
|
|
}, [b, c])
|
|
|
|
o(d()).equals(15)
|
|
o(count).equals(1)
|
|
})
|
|
o("combines and maps nested streams atomically", function() {
|
|
var count = 0
|
|
var a = Stream(3)
|
|
var b = Stream.combine(function(a) {return a() * 2}, [a])
|
|
var c = Stream.combine(function(a) {return a() * a()}, [a])
|
|
var d = c.map(function(x){return x})
|
|
var e = Stream.combine(function(x) {return x()}, [d])
|
|
var f = Stream.combine(function(b, e) {
|
|
count++
|
|
return b() + e()
|
|
}, [b, e])
|
|
|
|
o(f()).equals(15)
|
|
o(count).equals(1)
|
|
})
|
|
o("combine lists only changed upstreams in last arg", function() {
|
|
var streams = []
|
|
var a = Stream()
|
|
var b = Stream()
|
|
Stream.combine(function(a, b, changed) {
|
|
streams = changed
|
|
}, [a, b])
|
|
|
|
a(3)
|
|
b(5)
|
|
|
|
o(streams.length).equals(2)
|
|
o(streams[0]).equals(a)
|
|
o(streams[1]).equals(b)
|
|
})
|
|
o("combine continues with ended streams", function() {
|
|
var a = Stream()
|
|
var b = Stream()
|
|
var combined = Stream.combine(function(a, b) {
|
|
return a() + b()
|
|
}, [a, b])
|
|
|
|
a(3)
|
|
a.end(true)
|
|
b(5)
|
|
|
|
o(combined()).equals(8)
|
|
})
|
|
o("combine lists only changed upstreams in last arg with default value", function() {
|
|
var streams = []
|
|
var a = Stream(3)
|
|
var b = Stream(5)
|
|
Stream.combine(function(a, b, changed) {
|
|
streams = changed
|
|
}, [a, b])
|
|
|
|
a(7)
|
|
|
|
o(streams.length).equals(1)
|
|
o(streams[0]).equals(a)
|
|
})
|
|
o("combine can return undefined", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.combine(function() {
|
|
return undefined
|
|
}, [a])
|
|
|
|
o(b()).equals(undefined)
|
|
})
|
|
o("combine can return stream", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.combine(function() {
|
|
return Stream(2)
|
|
}, [a])
|
|
|
|
o(b()()).equals(2)
|
|
})
|
|
o("combine can return pending stream", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.combine(function() {
|
|
return Stream()
|
|
}, [a])
|
|
|
|
o(b()()).equals(undefined)
|
|
})
|
|
o("combine can skip", function() {
|
|
var count = 0
|
|
var a = Stream(1)
|
|
var b = Stream.combine(function() {
|
|
return Stream.SKIP
|
|
}, [a])["fantasy-land/map"](function() {
|
|
count++
|
|
return 1
|
|
})
|
|
|
|
o(b()).equals(undefined)
|
|
o(count).equals(0)
|
|
})
|
|
o("combine can conditionaly skip", function() {
|
|
var count = 0
|
|
var skip = false
|
|
var a = Stream(1)
|
|
var b = Stream.combine(function(a) {
|
|
if (skip) {
|
|
return Stream.SKIP
|
|
}
|
|
return a()
|
|
}, [a])["fantasy-land/map"](function(a) {
|
|
count++
|
|
return a
|
|
})
|
|
o(b()).equals(1)
|
|
o(count).equals(1)
|
|
skip = true
|
|
count = 0
|
|
a(2)
|
|
o(b()).equals(1)
|
|
o(count).equals(0)
|
|
})
|
|
o("combine will throw with a helpful error if given non-stream values", function () {
|
|
var spy = o.spy()
|
|
var a = Stream(1)
|
|
var thrown = null;
|
|
try {
|
|
Stream.combine(spy, [a, ""])
|
|
} catch (e) {
|
|
thrown = e
|
|
}
|
|
|
|
o(thrown).notEquals(null)
|
|
o(thrown.constructor === TypeError).equals(false)
|
|
o(spy.callCount).equals(0)
|
|
})
|
|
})
|
|
o.spec("lift", function() {
|
|
o("transforms value", function() {
|
|
var stream = Stream()
|
|
var doubled = Stream.lift(function(s) {return s * 2}, stream)
|
|
|
|
stream(2)
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("transforms default value", function() {
|
|
var stream = Stream(2)
|
|
var doubled = Stream.lift(function(s) {return s * 2}, stream)
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("transforms multiple values", function() {
|
|
var s1 = Stream()
|
|
var s2 = Stream()
|
|
var added = Stream.lift(function(s1, s2) {return s1 + s2}, s1, s2)
|
|
|
|
s1(2)
|
|
s2(3)
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("transforms multiple default values", function() {
|
|
var s1 = Stream(2)
|
|
var s2 = Stream(3)
|
|
var added = Stream.lift(function(s1, s2) {return s1 + s2}, s1, s2)
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("transforms mixed default and late-bound values", function() {
|
|
var s1 = Stream(2)
|
|
var s2 = Stream()
|
|
var added = Stream.lift(function(s1, s2) {return s1 + s2}, s1, s2)
|
|
|
|
s2(3)
|
|
|
|
o(added()).equals(5)
|
|
})
|
|
o("lifts atomically", function() {
|
|
var count = 0
|
|
var a = Stream()
|
|
var b = Stream.lift(function(a) {return a * 2}, a)
|
|
var c = Stream.lift(function(a) {return a * a}, a)
|
|
var d = Stream.lift(function(b, c) {
|
|
count++
|
|
return b + c
|
|
}, b, c)
|
|
|
|
a(3)
|
|
|
|
o(d()).equals(15)
|
|
o(count).equals(1)
|
|
})
|
|
o("lifts default value atomically", function() {
|
|
var count = 0
|
|
var a = Stream(3)
|
|
var b = Stream.lift(function(a) {return a * 2}, a)
|
|
var c = Stream.lift(function(a) {return a * a}, a)
|
|
var d = Stream.lift(function(b, c) {
|
|
count++
|
|
return b + c
|
|
}, b, c)
|
|
|
|
o(d()).equals(15)
|
|
o(count).equals(1)
|
|
})
|
|
o("lift can return undefined", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.lift(function() {
|
|
return undefined
|
|
}, a)
|
|
|
|
o(b()).equals(undefined)
|
|
})
|
|
o("lift can return stream", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.lift(function() {
|
|
return Stream(2)
|
|
}, a)
|
|
|
|
o(b()()).equals(2)
|
|
})
|
|
o("lift can return pending stream", function() {
|
|
var a = Stream(1)
|
|
var b = Stream.lift(function() {
|
|
return Stream()
|
|
}, a)
|
|
|
|
o(b()()).equals(undefined)
|
|
})
|
|
o("lift can halt", function() {
|
|
var count = 0
|
|
var a = Stream(1)
|
|
var b = Stream.lift(function() {
|
|
return Stream.HALT
|
|
}, a)["fantasy-land/map"](function() {
|
|
count++
|
|
return 1
|
|
})
|
|
|
|
o(b()).equals(undefined)
|
|
o(count).equals(0)
|
|
})
|
|
o("lift will throw with a helpful error if given non-stream values", function () {
|
|
var spy = o.spy()
|
|
var a = Stream(1)
|
|
var thrown = null;
|
|
try {
|
|
Stream.lift(spy, a, "")
|
|
} catch (e) {
|
|
thrown = e
|
|
}
|
|
|
|
o(thrown).notEquals(null)
|
|
o(thrown.constructor === TypeError).equals(false)
|
|
o(spy.callCount).equals(0)
|
|
})
|
|
})
|
|
o.spec("merge", function() {
|
|
o("transforms an array of streams to an array of values", function() {
|
|
var all = Stream.merge([
|
|
Stream(10),
|
|
Stream("20"),
|
|
Stream({value: 30}),
|
|
])
|
|
|
|
o(all()).deepEquals([10, "20", {value: 30}])
|
|
})
|
|
o("remains pending until all streams are active", function() {
|
|
var straggler = Stream()
|
|
|
|
var all = Stream.merge([
|
|
Stream(10),
|
|
Stream("20"),
|
|
straggler,
|
|
])
|
|
|
|
o(all()).equals(undefined)
|
|
|
|
straggler(30)
|
|
o(all()).deepEquals([10, "20", 30])
|
|
})
|
|
o("calls run callback after all parents are active", function() {
|
|
var value = 0
|
|
var id = function(value) {return value}
|
|
var a = Stream()
|
|
var b = Stream()
|
|
|
|
Stream.merge([a.map(id), b.map(id)]).map(function(data) {
|
|
value = data[0] + data[1]
|
|
return undefined
|
|
})
|
|
|
|
a(1)
|
|
b(2)
|
|
o(value).equals(3)
|
|
|
|
a(3)
|
|
b(4)
|
|
o(value).equals(7)
|
|
})
|
|
})
|
|
o.spec("end", function() {
|
|
o("end stream works", function() {
|
|
var stream = Stream()
|
|
var doubled = Stream.combine(function(stream) {return stream() * 2}, [stream])
|
|
|
|
stream.end(true)
|
|
|
|
stream(3)
|
|
|
|
o(doubled()).equals(undefined)
|
|
})
|
|
o("end stream works with default value", function() {
|
|
var stream = Stream(2)
|
|
var doubled = Stream.combine(function(stream) {return stream() * 2}, [stream])
|
|
|
|
stream.end(true)
|
|
|
|
stream(3)
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("cannot add downstream to ended stream", function() {
|
|
var stream = Stream(2)
|
|
stream.end(true)
|
|
|
|
var doubled = Stream.combine(function(stream) {return stream() * 2}, [stream])
|
|
stream(3)
|
|
|
|
o(doubled()).equals(undefined)
|
|
})
|
|
o("upstream does not affect ended stream", function() {
|
|
var stream = Stream(2)
|
|
var doubled = Stream.combine(function(stream) {return stream() * 2}, [stream])
|
|
|
|
doubled.end(true)
|
|
|
|
stream(4)
|
|
|
|
o(doubled()).equals(4)
|
|
})
|
|
o("end stream can be mapped to", function() {
|
|
var stream = Stream()
|
|
var spy = o.spy()
|
|
|
|
stream.end.map(spy)
|
|
|
|
o(spy.callCount).equals(0)
|
|
|
|
stream.end(true)
|
|
|
|
o(spy.callCount).equals(1)
|
|
})
|
|
})
|
|
o.spec("toJSON", function() {
|
|
o("works", function() {
|
|
o(Stream(1).toJSON()).equals(1)
|
|
o(Stream("a").toJSON()).equals("a")
|
|
o(Stream(true).toJSON()).equals(true)
|
|
o(Stream(null).toJSON()).equals(null)
|
|
o(Stream(undefined).toJSON()).equals(undefined)
|
|
o(Stream({a: 1}).toJSON()).deepEquals({a: 1})
|
|
o(Stream([1, 2, 3]).toJSON()).deepEquals([1, 2, 3])
|
|
o(Stream().toJSON()).equals(undefined)
|
|
o(Stream(new Date(0)).toJSON()).equals(new Date(0).toJSON())
|
|
})
|
|
o("works w/ JSON.stringify", function() {
|
|
o(JSON.stringify(Stream(1))).equals(JSON.stringify(1))
|
|
o(JSON.stringify(Stream("a"))).equals(JSON.stringify("a"))
|
|
o(JSON.stringify(Stream(true))).equals(JSON.stringify(true))
|
|
o(JSON.stringify(Stream(null))).equals(JSON.stringify(null))
|
|
o(JSON.stringify(Stream(undefined))).equals(JSON.stringify(undefined))
|
|
o(JSON.stringify(Stream({a: 1}))).deepEquals(JSON.stringify({a: 1}))
|
|
o(JSON.stringify(Stream([1, 2, 3]))).deepEquals(JSON.stringify([1, 2, 3]))
|
|
o(JSON.stringify(Stream())).equals(JSON.stringify(undefined))
|
|
o(JSON.stringify(Stream(new Date(0)))).equals(JSON.stringify(new Date(0)))
|
|
})
|
|
})
|
|
o.spec("map", function() {
|
|
o("works", function() {
|
|
var stream = Stream()
|
|
var doubled = stream["fantasy-land/map"](function(value) {return value * 2})
|
|
|
|
stream(3)
|
|
|
|
o(doubled()).equals(6)
|
|
})
|
|
o("works with default value", function() {
|
|
var stream = Stream(3)
|
|
var doubled = stream["fantasy-land/map"](function(value) {return value * 2})
|
|
|
|
o(doubled()).equals(6)
|
|
})
|
|
o("works with undefined value", function() {
|
|
var stream = Stream()
|
|
var mapped = stream["fantasy-land/map"](function(value) {return String(value)})
|
|
|
|
stream(undefined)
|
|
|
|
o(mapped()).equals("undefined")
|
|
})
|
|
o("works with default undefined value", function() {
|
|
var stream = Stream(undefined)
|
|
var mapped = stream["fantasy-land/map"](function(value) {return String(value)})
|
|
|
|
o(mapped()).equals("undefined")
|
|
})
|
|
o("works with pending stream", function() {
|
|
var stream = Stream(undefined)
|
|
var mapped = stream["fantasy-land/map"](function() {return Stream()})
|
|
|
|
o(mapped()()).equals(undefined)
|
|
})
|
|
o("has alias", function() {
|
|
var stream = Stream(undefined)
|
|
|
|
o(stream["fantasy-land/map"]).equals(stream.map)
|
|
})
|
|
})
|
|
o.spec("ap", function() {
|
|
o("works", function() {
|
|
var apply = Stream(function(value) {return value * 2})
|
|
var stream = Stream(3)
|
|
var applied = stream["fantasy-land/ap"](apply)
|
|
|
|
o(applied()).equals(6)
|
|
|
|
apply(function(value) {return value / 3})
|
|
|
|
o(applied()).equals(1)
|
|
|
|
stream(9)
|
|
|
|
o(applied()).equals(3)
|
|
})
|
|
o("works with undefined value", function() {
|
|
var apply = Stream(function(value) {return String(value)})
|
|
var stream = Stream(undefined)
|
|
var applied = stream["fantasy-land/ap"](apply)
|
|
|
|
o(applied()).equals("undefined")
|
|
|
|
apply(function(value) {return String(value) + "a"})
|
|
|
|
o(applied()).equals("undefineda")
|
|
})
|
|
})
|
|
o.spec("fantasy-land", function() {
|
|
o.spec("functor", function() {
|
|
o("identity", function() {
|
|
var stream = Stream(3)
|
|
var mapped = stream["fantasy-land/map"](function(value) {return value})
|
|
|
|
o(stream()).equals(mapped())
|
|
})
|
|
o("composition", function() {
|
|
function f(x) {return x * 2}
|
|
function g(x) {return x * x}
|
|
|
|
var stream = Stream(3)
|
|
|
|
var mapped = stream["fantasy-land/map"](function(value) {return f(g(value))})
|
|
var composed = stream["fantasy-land/map"](g)["fantasy-land/map"](f)
|
|
|
|
o(mapped()).equals(18)
|
|
o(mapped()).equals(composed())
|
|
})
|
|
})
|
|
o.spec("apply", function() {
|
|
o("composition", function() {
|
|
var a = Stream(function(value) {return value * 2})
|
|
var u = Stream(function(value) {return value * 3})
|
|
var v = Stream(5)
|
|
|
|
var mapped = v["fantasy-land/ap"](u["fantasy-land/ap"](a["fantasy-land/map"](function(f) {
|
|
return function(g) {
|
|
return function(x) {
|
|
return f(g(x))
|
|
}
|
|
}
|
|
})))
|
|
|
|
var composed = v["fantasy-land/ap"](u)["fantasy-land/ap"](a)
|
|
|
|
o(mapped()).equals(30)
|
|
o(mapped()).equals(composed())
|
|
})
|
|
})
|
|
o.spec("applicative", function() {
|
|
o("identity", function() {
|
|
var a = Stream["fantasy-land/of"](function(value) {return value})
|
|
var v = Stream(5)
|
|
|
|
o(v["fantasy-land/ap"](a)()).equals(5)
|
|
o(v["fantasy-land/ap"](a)()).equals(v())
|
|
})
|
|
o("homomorphism", function() {
|
|
var a = Stream(0)
|
|
var f = function(value) {return value * 2}
|
|
var x = 3
|
|
|
|
o(a.constructor["fantasy-land/of"](x)["fantasy-land/ap"](a.constructor["fantasy-land/of"](f))()).equals(6)
|
|
o(a.constructor["fantasy-land/of"](x)["fantasy-land/ap"](a.constructor["fantasy-land/of"](f))()).equals(a.constructor["fantasy-land/of"](f(x))())
|
|
})
|
|
o("interchange", function() {
|
|
var u = Stream(function(value) {return value * 2})
|
|
var a = Stream()
|
|
var y = 3
|
|
|
|
o(a.constructor["fantasy-land/of"](y)["fantasy-land/ap"](u)()).equals(6)
|
|
o(a.constructor["fantasy-land/of"](y)["fantasy-land/ap"](u)()).equals(u["fantasy-land/ap"](a.constructor["fantasy-land/of"](function(f) {return f(y)}))())
|
|
})
|
|
})
|
|
})
|
|
})
|