UNPKG

mithril

Version:

A framework for building brilliant applications

667 lines (569 loc) 16.7 kB
"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) }) // NOTE: this *must* be the *only* uses of `Stream.HALT` in the entire // test suite. o("HALT is a deprecated alias of SKIP and warns once", function() { var log = console.log var warnings = [] console.log = function(a) { warnings.push(a) } try { o(Stream.HALT).equals(Stream.SKIP) o(warnings).deepEquals(["HALT is deprecated and has been renamed to SKIP"]) o(Stream.HALT).equals(Stream.SKIP) o(warnings).deepEquals(["HALT is deprecated and has been renamed to SKIP"]) o(Stream.HALT).equals(Stream.SKIP) o(warnings).deepEquals(["HALT is deprecated and has been renamed to SKIP"]) } finally { console.log = log } }) }) 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("combine callback not called when child stream was ended", function () { var spy = o.spy() var a = Stream(1) var b = Stream(2) var mapped = Stream.combine(spy, [a, b]) mapped.end(true) a(11) o(spy.callCount).equals(1) }) }) 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.SKIP }, 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("ended stream works like a container", function() { var stream = Stream(1) stream.end(true) stream(2) o(stream()).equals(2) }) }) 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("mapping function is not invoked after ending", function () { var stream = Stream(undefined) var fn = o.spy() var mapped = stream.map(fn) mapped.end(true) stream(undefined) o(fn.callCount).equals(1) }) }) 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)}))()) }) }) }) })