"use strict" var o = require("../../ospec/ospec") var callAsync = require("../../test-utils/callAsync") var Stream = require("../../util/stream") o.spec("stream", function() { o.spec("stream", function() { o("works as getter/setter", function() { var stream = 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.stream() o(stream()).equals(undefined) }) o("can update to undefined", function() { var stream = Stream.stream(1) stream(undefined) o(stream()).equals(undefined) }) }) o.spec("combine", function() { o("transforms value", function() { var stream = 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.stream(2) var doubled = Stream.combine(function(s) {return s() * 2}, [stream]) o(doubled()).equals(4) }) o("transforms multiple values", function() { var s1 = Stream.stream() var s2 = Stream.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.stream(2) var s2 = Stream.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.stream(2) var s2 = Stream.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.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.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("combine lists only changed upstreams in last arg", function() { var streams = [] var a = Stream.stream() var b = Stream.stream() var c = Stream.combine(function(a, b, changed) { streams = changed }, [a, b]) a(3) b(5) o(streams.length).equals(1) o(streams[0]).equals(b) }) o("combine lists only changed upstreams in last arg with default value", function() { var streams = [] var a = Stream.stream(3) var b = Stream.stream(5) var c = 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.stream(1) var b = Stream.combine(function(a) { return undefined }, [a]) o(b()).equals(undefined) }) o("combine absorbs streams", function() { var a = Stream.stream(1) var b = Stream.combine(function(a) { return Stream.stream(2) }, [a]) o(b()).equals(2) }) o("combine absorbs errored streams", function() { var a = Stream.stream(1) var b = Stream.combine(function(a) { return Stream.reject(new Error("error")) }, [a]) o(b()).equals(undefined) o(b.error().message).equals("error") }) o("combine absorbs ready streams", function() { var count = 0 var a = Stream.stream(1) var b = Stream.combine(function(a) { return Stream.stream() }, [a]) var c = Stream.combine(function(b) { count++ return 2 }, [b]) o(c()).equals(undefined) o(count).equals(0) }) }) o.spec("end", function() { o("end stream works", function() { var stream = 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.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.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.stream(2) var doubled = Stream.combine(function(stream) {return stream() * 2}, [stream]) doubled.end(true) stream(4) o(doubled()).equals(4) }) }) o.spec("error", function() { o("error() works", function() { var stream = Stream.stream() var errored = Stream.combine(function(stream) {throw new Error("error")}, [stream]) stream(3) o(errored()).equals(undefined) o(errored.error().message).equals("error") }) o("error() works with default value", function() { var stream = Stream.stream(3) var errored = Stream.combine(function(stream) {throw new Error("error")}, [stream]) o(errored()).equals(undefined) o(errored.error().message).equals("error") }) o("error() removes error on valid value", function() { var stream = Stream.stream("a") var doubled = Stream.combine(function(stream) { if (typeof stream() !== "number") throw new Error("error") else return stream() * 2 }, [stream]) stream(3) o(doubled()).equals(6) o(doubled.error()).equals(undefined) }) o("error() triggers catch", function() { var count = 0 var stream = Stream.stream(1) var handled = stream.catch(function() { count++ return 2 }) stream.error(new Error("error")) o(handled()).equals(2) o(handled.error()).equals(undefined) o(count).equals(1) }) o("thrown error propagates downstream", function() { var count = 0 var stream = Stream.stream(1) .map(function() {throw new Error("error")}) .map(function(value) { count++ return value * 2 }) .map(function(value) { count++ return value * 3 }) o(stream()).equals(undefined) o(stream.error().message).equals("error") o(count).equals(0) }) o("set error propagates downstream", function() { var count = 0 var stream = Stream.stream() var mapped = stream.map(function(value) { count++ return value * 2 }) .map(function(value) { count++ return value * 3 }) stream.error(new Error("error")) o(mapped()).equals(undefined) o(mapped.error().message).equals("error") o(count).equals(0) }) o("error.map works", function() { var stream = Stream.stream(1) var mappedFromError = stream.error.map(function(value) { return "from" + value.message }) o(mappedFromError()).equals(undefined) stream.error(new Error("error")) o(mappedFromError()).equals("fromerror") }) o("error from error.map propagates", function() { var stream = Stream.stream(1) var mappedFromError = stream.error.map(function(value) { return "from" + value.message }).map(function(value) { return "a" + value }) o(mappedFromError()).equals(undefined) stream.error(new Error("error")) o(mappedFromError()).equals("afromerror") }) o("error thrown from error.map propagates downstream", function() { var count = 0 var stream = Stream.stream(1) var mappedFromError = stream.error.map(function(value) { throw new Error("b") }) var downstream = mappedFromError.map(function() { count++ }) o(mappedFromError()).equals(undefined) stream.error(new Error("a")) o(mappedFromError()).equals(undefined) o(mappedFromError.error().message).equals("b") o(downstream()).equals(undefined) o(downstream.error().message).equals("b") o(count).equals(0) }) o("error.map absorbs streams", function() { var stream = Stream.reject(new Error("error")) var error = stream.error.map(function(value) { return Stream.stream(1) }) o(error()).equals(1) }) }) o.spec("reject", function() { o("reject works", function() { var stream = Stream.reject(new Error("error")) o(stream()).equals(undefined) o(stream.error().message).equals("error") }) o("rejected propagates downstream", function() { var count = 0 var stream = Stream.reject(new Error("error")) .map(function(value) { count++ return value * 2 }) .map(function(value) { count++ return value * 3 }) o(stream()).equals(undefined) o(stream.error().message).equals("error") }) o("rejected removes error on value", function() { var stream = Stream.reject(new Error("error")) var doubled = stream.map(function(value) { return value * 2 }) stream(1) o(doubled()).equals(2) o(stream.error()).equals(undefined) }) o("combined rejected yields first error", function() { var count = 0 var a = Stream.reject(new Error("a")) var b = Stream.reject(new Error("b")) var combined = Stream.combine(function(a, b) { count++ return a() + b() }, [a, b]) o(combined()).equals(undefined) o(combined.error().message).equals("a") o(count).equals(0) }) }) o.spec("catch", function() { o("catch works from reject", function() { var count = 0 var stream = Stream.reject(new Error("error")).catch(function(e) { count++ return "no" + e.message }).map(function(value) { return value + "mapped" }) o(count).equals(1) o(stream()).equals("noerrormapped") o(stream.error()).equals(undefined) }) o("catch works from combine", function() { var count = 0 var stream = Stream.combine(function() {throw new Error("error")}, [Stream.stream(1)]).catch(function(e) { count++ return "no" + e.message }).map(function(value) { return value + "mapped" }) o(count).equals(1) o(stream()).equals("noerrormapped") o(stream.error()).equals(undefined) }) o("catch is not called if no error", function() { var count = 0 var stream = Stream.stream() var handled = stream.map(function(value) {return value + value}).catch(function(e) { count++ return "no" + e.message }).map(function(value) { return value + "mapped" }) stream("a") o(count).equals(0) o(handled()).equals("aamapped") o(handled.error()).equals(undefined) }) o("catch is not called if no error with default value", function() { var count = 0 var stream = Stream.stream("a").map(function(value) {return value + value}).catch(function(e) { count++ return "no" + e.message }).map(function(value) { return value + "mapped" }) o(count).equals(0) o(stream()).equals("aamapped") o(stream.error()).equals(undefined) }) o("throwing from catch rejects", function() { var stream = Stream.reject(new Error("a")).catch(function(e) { throw new Error("b") }) var mapped = stream.map(function(value) {return value + "ok"}) o(stream()).equals(undefined) o(stream.error().message).equals("b") o(mapped()).equals(undefined) o(mapped.error().message).equals("b") }) o("catch can return undefined", function() { var stream = Stream.reject(new Error("b")).catch(function(e) {}).map(function(value) {return String(value)}) o(stream()).equals("undefined") o(stream.error()).equals(undefined) }) o("catch does not prevent sibling error propagation", function() { var a = Stream.reject(new Error("a")) var b = a.map(function(value) {return value + "b"}).catch(function(e) {}) var c = a.map(function(value) {return value + "c"}) var d = Stream.combine(function(b, c) {return b() + c()}, [b, c]) o(d()).equals(undefined) o(d.error().message).equals("a") }) o("catches absorbed rejected stream", function() { var caught var stream = Stream.stream(1).map(function() { return Stream.reject(new Error("error")) }).catch(function(value) { caught = value return "no" + value.message }).map(function(value) { return value + "mapped" }) o(stream()).equals("noerrormapped") }) }) o.spec("map", function() { o("works", function() { var stream = Stream.stream() var doubled = stream.map(function(value) {return value * 2}) stream(3) o(doubled()).equals(6) }) o("works with default value", function() { var stream = Stream.stream(3) var doubled = stream.map(function(value) {return value * 2}) o(doubled()).equals(6) }) o("works with undefined value", function() { var stream = Stream.stream() var mapped = stream.map(function(value) {return String(value)}) stream(undefined) o(mapped()).equals("undefined") }) o("works with default undefined value", function() { var stream = Stream.stream(undefined) var mapped = stream.map(function(value) {return String(value)}) o(mapped()).equals("undefined") }) }) o.spec("ap", function() { o("works", function() { var apply = Stream.stream(function(value) {return value * 2}) var stream = Stream.stream(3) var applied = apply.ap(stream) 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.stream(function(value) {return String(value)}) var stream = Stream.stream(undefined) var applied = apply.ap(stream) o(applied()).equals("undefined") apply(function(value) {return String(value) + "a"}) o(applied()).equals("undefineda") }) }) o.spec("absorbing semantics", function() { o("absorbs stream", function() { var stream = Stream.stream(Stream.stream(1)) var doubled = stream.map(function(value) {return value * 2}) o(stream()).equals(1) o(doubled()).equals(2) }) o("absorbs stream on update", function() { var stream = Stream.stream() var doubled = stream.map(function(value) {return value * 2}) stream(Stream.stream(1)) o(stream()).equals(1) o(doubled()).equals(2) }) o("absorbs errored stream", function() { var stream = Stream.stream(Stream.reject(1)) var doubled = stream.error.map(function(value) {return value * 2}) o(stream()).equals(undefined) o(stream.error()).equals(1) o(doubled()).equals(2) }) o("absorbs pending stream", function() { var count = 0 var stream = Stream.stream(Stream.stream()) var mapped = stream.map(function() { count++ }) o(stream()).equals(undefined) o(count).equals(0) }) o("absorbs ended stream", function() { var count = 0 var ended = Stream.stream() ended.end(true) var stream = Stream.stream(ended) var mapped = stream.map(function() { count++ }) stream(1) o(count).equals(0) }) o("combine absorbs stream", function() { var stream = Stream.stream(1) var combined = Stream.combine(function(stream) { return Stream.stream(2) }, [stream]) var doubled = combined.map(function(value) {return value * 2}) o(combined()).equals(2) o(doubled()).equals(4) }) o("combine absorbs errored stream", function() { var stream = Stream.stream(1) var combined = Stream.combine(function(stream) { return Stream.reject(2) }, [stream]) var doubled = combined.error.map(function(value) {return value * 2}) o(combined()).equals(undefined) o(combined.error()).equals(2) o(doubled()).equals(4) }) o("combine absorbs pending stream", function() { var count = 0 var stream = Stream.stream(1) var combined = Stream.combine(function(stream) { return Stream.stream() }, [stream]) var mapped = combined.map(function() { count++ }) o(combined()).equals(undefined) o(count).equals(0) }) o("combine absorbs ended stream", function() { var count = 0 var stream = Stream.stream(1) var combined = Stream.combine(function(stream) { var ended = Stream.stream(2) ended.end(true) return ended }, [stream]) var mapped = combined.map(function() { count++ }) o(combined()).equals(2) o(count).equals(0) }) o("reject absorbs stream", function() { var stream = Stream.reject(Stream.stream(1)) var doubled = stream.error.map(function(value) {return value * 2}) o(stream.error()).equals(1) o(doubled()).equals(2) }) o("reject absorbs errored stream", function() { var count = 0 var stream = Stream.reject(Stream.reject(new Error("error"))) var mapped = stream.error.map(function() { count++ }) o(stream.error().message).equals("error") o(count).equals(1) }) o("reject absorbs pending stream", function() { var count = 0 var stream = Stream.reject(Stream.stream()) var mapped = stream.error.map(function() { count++ }) o(stream.error()).equals(undefined) o(count).equals(0) }) o("reject absorbs ended stream", function() { var count = 0 var ended = Stream.stream(1) ended.end(true) var stream = Stream.reject(ended) var mapped = stream.error.map(function() { count++ }) o(count).equals(0) }) o("ended stream absorbs stream", function() { var stream = Stream.stream() stream.end(true) var doubled = stream.map(function(value) {return value * 2}) stream(Stream.stream(1)) o(stream()).equals(1) o(doubled()).equals(undefined) }) o("ended stream absorbs pending stream", function() { var stream = Stream.stream() stream.end(true) var doubled = stream.map(function(value) {return value * 2}) stream(Stream.stream()) o(stream()).equals(undefined) o(doubled()).equals(undefined) }) o("ended stream absorbs errored stream", function() { var stream = Stream.stream() stream.end(true) var doubled = stream.map(function(value) {return value * 2}) stream(Stream.reject(new Error("error"))) o(stream.error().message).equals("error") o(doubled()).equals(undefined) }) o("ended stream absorbs ended stream", function() { var stream = Stream.stream(1) stream.end(true) var ended = Stream.stream(2) ended.end(true) var doubled = stream.map(function(value) {return value * 2}) stream(ended) o(stream()).equals(2) o(doubled()).equals(undefined) }) }) o.spec("fantasy-land", function() { o.spec("functor", function() { o("identity", function() { var stream = Stream.stream(3) var mapped = stream.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.stream(3) var mapped = stream.map(function(value) {return f(g(value))}) var composed = stream.map(g).map(f) o(mapped()).equals(18) o(mapped()).equals(composed()) }) }) o.spec("apply", function() { o("composition", function() { var a = Stream.stream(function(value) {return value * 2}) var u = Stream.stream(function(value) {return value * 3}) var v = Stream.stream(5) var mapped = a.map(function(f) { return function(g) { return function(x) { return f(g(x)) } } }).ap(u).ap(v) var composed = a.ap(u.ap(v)) o(mapped()).equals(30) o(mapped()).equals(composed()) }) }) o.spec("applicative", function() { o("identity", function() { var a = Stream.stream().of(function(value) {return value}) var v = Stream.stream(5) o(a.ap(v)()).equals(5) o(a.ap(v)()).equals(v()) }) o("homomorphism", function() { var a = Stream.stream(0) var f = function(value) {return value * 2} var x = 3 o(a.of(f).ap(a.of(x))()).equals(6) o(a.of(f).ap(a.of(x))()).equals(a.of(f(x))()) }) o("interchange", function() { var u = Stream.stream(function(value) {return value * 2}) var a = Stream.stream() var y = 3 o(u.ap(a.of(y))()).equals(6) o(u.ap(a.of(y))()).equals(a.of(function(f) {return f(y)}).ap(u)()) }) }) }) })