_
__ _____| |_ _ __ ___ __ _ _ __ ___
\ \/ / __| __| '__/ _ \/ _` | '_ ` _ \
> <\__ \ |_| | | __/ (_| | | | | | |
/_/\_\___/\__|_| \___|\__,_|_| |_| |_|
- Only 26 core operators and factories
- Written in TypeScript
- Approximately 30 kB in size, without minification or gzip
- On average, faster than RxJS 4, Kefir, Bacon.js, as fast as RxJS 5, and slower than most.js
- Tailored for Cycle.js, or applications with limited use of
subscribe
import xs from 'xstream'
// Tick every second incremental numbers,
// only pass even numbers, then map them to their square,
// and stop after 5 seconds has passed
var stream = xs.periodic(1000)
.filter(i => i % 2 === 0)
.map(i => i * i)
.endWhen(xs.periodic(5000).take(1))
// So far, the stream is idle.
// As soon as it gets its first listener, it starts executing.
stream.addListener({
next: i => console.log(i),
error: err => console.error(err),
complete: () => console.log('completed'),
})
npm install xstream
import xs from 'xstream'
var xs = require('xstream').default
create
createWithMemory
createMimic
never
empty
throw
of
fromArray
fromPromise
periodic
merge
combine
addListener
removeListener
map
mapTo
filter
take
drop
last
startWith
endWhen
fold
replaceError
flatten
compose
remember
debug
shamefullySendNext
shamefullySendError
shamefullySendComplete
imitate
XStream has four fundamental types: Stream, Listener, Producer, and MemoryStream.
A Stream is an event emitter with multiple Listeners. When an event happens on the Stream, it is broadcast to all its Listeners at the same time.
Streams have methods attached to them called operators, such as map
, filter
, fold
, take
, etc. When called, an operator creates and returns another Stream. The returned Stream is actually a Listener of the source Stream (I forgot to tell you that Streams may be Listeners, too). So once the source Stream broadcasts an event, the event will pass through the operator logic and the returned Stream may perhaps broadcast its own event based on the source one.
You can also trigger an event to happen on a Stream with the shamefullySend*
methods. But you don't want to do that. Really, avoid doing that because it's not the reactive way and you'll be missing the point of this library. Ok?
A Listener is an object with three functions attached to it: next
, error
, and complete
. There is one function for each type of event a Stream may emit.
next
events are the typical type, they deliver a value.error
events abort (stop) the execution of the Stream, and happen when something goes wrong in the Stream (or upstream somewhere in the chain of operators)complete
events signal the peaceful stop of the execution of the Stream.
This is an example of a typical listener:
var listener = {
next: (value) => {
console.log('The Stream gave me a value: ', value);
},
error: (err) => {
console.error('The Stream gave me an error: ', err);
},
complete: () => {
console.log('The Stream told me it is done.');
},
}
And this is how you would attach that Listener to a Stream:
stream.addListener(listener)
And when you think the Listener is done, you can remove it from the Stream:
stream.removeListener(listener)
A Producer is like a machine that produces events to be broadcast on a Stream.
Events from a Stream must come from somewhere, right? That's why we need Producers. They are objects with two functions attached: start(listener)
and stop()
. Once you call start
with a listener
, the Producer will start generating events and it will send those to the listener. When you call stop()
, the Producer should quit doing its own thing.
Because Streams are Listeners, if you give a Stream as the Listener in start(stream)
, essentially the Producer is now generating events that will be broadcast on the Stream. Nice, huh? Now a bunch of listeners can be attached to the Stream and they can all get those events originally coming from the Producer. That's why xs.create(producer)
receives a Producer to be the heart of a new Stream. Check this out:
var producer = {
start: function (listener) {
this.id = setInterval(() => listener.next('yo'), 1000)
},
stop: function () {
clearInterval(this.id)
},
id: 0,
}
// This fellow delivers a 'yo' next event every 1 second
var stream = xs.create(producer)
But remember, a Producer has only one listener, but a Stream may have many listeners.
You may wonder "when is start
and stop
called", and that's actually a fairly tricky topic, so let's get back to that soon. First let me tell you about MemoryStreams.
A MemoryStream is just like a Stream: it has operators, it can have listeners attached, you can shamefully send events to it, blabla. But it has one special property: it has memory. It remembers the most recent (but just one) next
event that it sent to its listeners.
Why is that useful? If a new Listener is added after that next
event was sent, the MemoryStream will get its value stored in memory and will send it to the newly attached Listener.
This is important so MemoryStreams can represent values or pieces of state which are relevant even after some time has passed. You don't want to lose those, you want to keep them and send them to Listeners that arrive late, after the event was originally created.
A Stream controls its Producer according to its number of Listeners, using reference counting with a synchronous start
and a cancelable asynchronous stop
. That's how a Stream starts and stops, basically. Usually this part of XStream is not so relevant to remember when building applications, but if you want to understand it for debugging or curiosity, it's explained in plain English below.
When you create a Stream with xs.create(producer)
, the start()
function of the Producer is not yet called. The Stream is still "idle". It has the Producer, but the Producer was not turned on.
Once the first Listener is added to the Stream, the number of Listeners attached suddenly changed from 0
to 1
. That's when the Stream calls start
, because after all there is at least one Listener interested in this Stream.
More Listeners may be added in the future, but they don't affect whether the Producer will continue working or stop. Just the first Listener dictates when the Stream starts.
What matters for stopping the Producer is stream.removeListener
. When the last Listener leaves (or in other words, when the number of Listeners suddenly changes from 1
to 0
), the Stream schedules producer.stop()
to happen on the next event loop. That is, asynchronously. If, however, a new Listener is added (number goes from 0
to 1
) before that scheduled moment, the producer.stop()
will be cancelled, and the Producer will continue generating events for its Stream normally.
The reason the Producer is not suddenly (synchronously) stopped, is that it is often necessary to swap the single listener of a Stream, but still keep its ongoing execution. For instance:
var listenerA = {/* ... */}
var listenerB = {/* ... */}
// number goes from 0 to 1, so the Stream's Producer starts
stream.addListener(listenerA)
// ...
// number goes from 1 to 0, but then immediately goes back
// to 1, because listenerB was added
stream.removeListener(listenerA)
stream.addListener(listenerB)
// Stream's Producer does not stop, everything continues as before
It's still useful to eventually (asynchronously) stop a Stream's internal Producer, because you don't want useless computation lying around producing gibberish. At least I don't.
Factories are functions that create Streams, such as xs.create()
, xs.periodic()
, etc.
Creates a new Stream given a Producer.
producer: Producer
An optional Producer that dictates how to start, generate events, and stop the Stream.
Creates a new MemoryStream given a Producer.
producer: Producer
An optional Producer that dictates how to start, generate events, and stop the Stream.
Creates a new MimicStream, which can imitate
another Stream. Only a
MimicStream has the imitate()
method.
Creates a Stream that does nothing when started. It never emits any event.
Marble diagram:
never
-----------------------
Creates a Stream that immediately emits the "complete" notification when started, and that's it.
Marble diagram:
empty
-|
Creates a Stream that immediately emits an "error" notification with the
value you passed as the error
argument when the stream starts, and that's
it.
Marble diagram:
throw(X)
-X
error
The error event to emit on the created stream.
Creates a Stream that immediately emits the arguments that you give to of, then completes.
Marble diagram:
of(1,2,3)
123|
a
The first value you want to emit as an event on the stream.b
The second value you want to emit as an event on the stream. One or more of these values may be given as arguments.
Converts an array to a stream. The returned stream will emit synchronously all the items in the array, and then complete.
Marble diagram:
fromArray([1,2,3])
123|
array: Array
The array to be converted as a stream.
Converts a promise to a stream. The returned stream will emit the resolved value of the promise, and then complete. However, if the promise is rejected, the stream will emit the corresponding error.
Marble diagram:
fromPromise( ----42 )
-----------------42|
promise: Promise
The promise to be converted as a stream.
Creates a stream that periodically emits incremental numbers, every
period
milliseconds.
Marble diagram:
periodic(1000)
---0---1---2---3---4---...
period: number
The interval in milliseconds to use as a rate of emission.
Blends multiple streams together, emitting events from all of them concurrently.
merge takes multiple streams as arguments, and creates a stream that imitates each of the argument streams, in parallel.
Marble diagram:
--1----2-----3--------4---
----a-----b----c---d------
merge
--1-a--2--b--3-c---d--4---
stream1: Stream
A stream to merge together with other streams.stream2: Stream
A stream to merge together with other streams. Two or more streams may be given as arguments.
Combines multiple streams together to return a stream whose events are calculated from the latest events of each of the input streams.
combine remembers the most recent event from each of the input streams.
When any of the input streams emits an event, that event together with all
the other saved events are combined in the project
function which should
return a value. That value will be emitted on the output stream. It's
essentially a way of mixing the events from multiple streams according to a
formula.
Marble diagram:
--1----2-----3--------4---
----a-----b-----c--d------
combine((x,y) => x+y)
----1a-2a-2b-3b-3c-3d-4d--
project: Function
A function of type(x: T1, y: T2) => R
or similar that takes the most recent eventsx
andy
from the input streams and returns a value. The output stream will emit that value. The number of arguments for this function should match the number of input streams.stream1: Stream
A stream to combine together with other streams.stream2: Stream
A stream to combine together with other streams. Two or more streams may be given as arguments.
Methods are functions attached to a Stream instance, like stream.addListener()
. Operators are also methods, but return a new Stream, leaving the existing Stream unmodified, except for the fact that it has a child Stream attached as Listener.
Adds a Listener to the Stream.
listener: Listener\<T>
Removes a Listener from the Stream, assuming the Listener was added to it.
listener: Listener\<T>
Transforms each event from the input Stream through a project
function,
to get a Stream that emits those transformed events.
Marble diagram:
--1---3--5-----7------
map(i => i * 10)
--10--30-50----70-----
project: Function
A function of type(t: T) => U
that takes eventt
of typeT
from the input Stream and produces an event of typeU
, to be emitted on the output Stream.
It's like map
, but transforms each input event to always the same
constant value on the output Stream.
Marble diagram:
--1---3--5-----7-----
mapTo(10)
--10--10-10----10----
projectedValue
A value to emit on the output Stream whenever the input Stream emits any value.
Only allows events that pass the test given by the passes
argument.
Each event from the input stream is given to the passes
function. If the
function returns true
, the event is forwarded to the output stream,
otherwise it is ignored and not forwarded.
Marble diagram:
--1---2--3-----4-----5---6--7-8--
filter(i => i % 2 === 0)
------2--------4---------6----8--
passes: Function
A function of type(t: T) +> boolean
that takes an event from the input stream and checks if it passes, by returning a boolean.
Lets the first amount
many events from the input stream pass to the
output stream, then makes the output stream complete.
Marble diagram:
--a---b--c----d---e--
take(3)
--a---b--c|
amount: number
How many events to allow from the input stream before completing the output stream.
Ignores the first amount
many events from the input stream, and then
after that starts forwarding events from the input stream to the output
stream.
Marble diagram:
--a---b--c----d---e--
drop(3)
--------------d---e--
amount: number
How many events to ignore from the input stream before forwarding all events from the input stream to the output stream.
When the input stream completes, the output stream will emit the last event emitted by the input stream, and then will also complete.
Marble diagram:
--a---b--c--d----|
last()
-----------------d|
Prepends the given initial
value to the sequence of events emitted by the
input stream. The returned stream is a MemoryStream, which means it is
already remember()
'd.
Marble diagram:
---1---2-----3---
startWith(0)
0--1---2-----3---
initial
The value or event to prepend.
Uses another stream to determine when to complete the current stream.
When the given other
stream emits an event or completes, the output
stream will complete. Before that happens, the output stream will imitate
whatever happens on the input stream.
Marble diagram:
---1---2-----3--4----5----6---
endWhen( --------a--b--| )
---1---2-----3--4--|
other
Some other stream that is used to know when should the output stream of this operator complete.
"Folds" the stream onto itself.
Combines events from the past throughout
the entire execution of the input stream, allowing you to accumulate them
together. It's essentially like Array.prototype.reduce
. The returned
stream is a MemoryStream, which means it is already remember()
'd.
The output stream starts by emitting the seed
which you give as argument.
Then, when an event happens on the input stream, it is combined with that
seed value through the accumulate
function, and the output value is
emitted on the output stream. fold
remembers that output value as acc
("accumulator"), and then when a new input event t
happens, acc
will be
combined with that to produce the new acc
and so forth.
Marble diagram:
------1-----1--2----1----1------
fold((acc, x) => acc + x, 3)
3-----4-----5--7----8----9------
accumulate: Function
A function of type(acc: R, t: T) => R
that takes the previous accumulated valueacc
and the incoming event from the input stream and produces the new accumulated value.seed
The initial accumulated value, of typeR
.
Replaces an error with another stream.
When (and if) an error happens on the input stream, instead of forwarding
that error to the output stream, replaceError will call the replace
function which returns the stream that the output stream will imitate. And,
in case that new stream also emits an error, replace
will be called again
to get another stream to start imitating.
Marble diagram:
--1---2-----3--4-----X
replaceError( () => --10--| )
--1---2-----3--4--------10--|
replace: Function
A function of type(err) => Stream
that takes the error that occurred on the input stream or on the previous replacement stream and returns a new stream. The output stream will imitate the stream that this function returns.
Flattens a "stream of streams", handling only one nested stream at a time (no concurrency).
If the input stream is a stream that emits streams, then this operator will return an output stream which is a flat stream: emits regular events. The flattening happens without concurrency. It works like this: when the input stream emits a nested stream, flatten will start imitating that nested one. However, as soon as the next nested stream is emitted on the input stream, flatten will forget the previous nested one it was imitating, and will start imitating the new nested one.
Marble diagram:
--+--------+---------------
\ \
\ ----1----2---3--
--a--b----c----d--------
flatten
-----a--b------1----2---3--
Passes the input stream to a custom operator, to produce an output stream.
compose is a handy way of using an existing function in a chained style.
Instead of writing outStream = f(inStream)
you can write
outStream = inStream.compose(f)
.
operator: function
A function that takes a stream as input and returns a stream as well.
Returns an output stream that imitates the input stream, but also remembers the most recent event that happens on the input stream, so that a newly added listener will immediately receive that memorised event.
Returns an output stream that identically imitates the input stream, but
also runs a spy
function fo each event, to help you debug your app.
debug takes a spy
function as argument, and runs that for each event
happening on the input stream. If you don't provide the spy
argument,
then debug will just console.log
each event. This helps you to
understand the flow of events through some operator chain.
Please note that if the output stream has no listeners, then it will not
start, which means spy
will never run because no actual event happens in
that case.
Marble diagram:
--1----2-----3-----4--
debug
--1----2-----3-----4--
labelOrSpy: function
A string to use as the label when printing debug information on the console, or a 'spy' function that takes an event as argument, and does not need to return anything.
Forces the Stream to emit the given value to its listeners.
As the name indicates, if you use this, you are most likely doing something The Wrong Way. Please try to understand the reactive way before using this method. Use it only when you know what you are doing.
value
The "next" value you want to broadcast to all listeners of this Stream.
Forces the Stream to emit the given error to its listeners.
As the name indicates, if you use this, you are most likely doing something The Wrong Way. Please try to understand the reactive way before using this method. Use it only when you know what you are doing.
error: any
The error you want to broadcast to all the listeners of this Stream.
Forces the Stream to emit the "completed" event to its listeners.
As the name indicates, if you use this, you are most likely doing something The Wrong Way. Please try to understand the reactive way before using this method. Use it only when you know what you are doing.
This method exists only on a MimicStream, which is created through
xs.createMimic()
. imitate
changes this current MimicStream to imitate
the other
given stream.
The imitate method returns nothing. Instead, it changes the behavior of
the current stream, making it re-emit whatever events are emitted by the
given other
stream.
other: Stream
The stream to imitate on the current one. Must not be a MemoryStream.
The operators and factories listed above are the core functions. xstream
has plenty of extra operators, documented here.
Q: Why does imitate()
support a Stream but not a MemoryStream?
A: MemoryStreams are meant for representing "values over time" (your age), while Streams represent simply events (your birthdays). MemoryStreams are usually initialized with a value, and imitate()
is meant for creating circular dependencies of streams. If we would attempt to imitate a MemoryStream in a circular dependency, we would either get a race condition (where the symptom would be "nothing happens") or an infinite cyclic emission of values.
If you find yourself wanting to use imitate()
with a MemoryStream, you should rework your code around imitate()
to use a Stream instead. Look for the stream in the circular dependency that represents an event stream, and that would be a candidate for creating a MimicStream which then imitates the real event stream.
Q: What's the difference between xstream and RxJS?
A: Read this blog post on the topic.
Q: What is the equivalent of withLatestFrom
in xstream?
A.withLatestFrom(B, (a, b) => a + b)
can be achieved in xstream with
B.map(b => A.map(a => a + b)).flatten()
And can be interpreted/read as "when a B
event happens, remember it and map it to all the subsequent events of A
mapped to a + b
".
Acknowledgements: xstream is built by staltz and TylorS.
License: MIT
4.0.3 (2016-06-08)
- remember: fix remember() on producer-less streams (cbe806d)
4.0.2 (2016-06-08)
- Stream: fix small issue with private Stream members (61b5c12)
4.0.1 (2016-06-03)
- compose: improve compose type signature (38b1064)
4.0.0 (2016-06-03)
- core: remove instance combine() and merge() (00fc72c)
- core: improve signature of operators regarding types (#43) (116e9f2)
- core: Instance operators stream.combine() and stream.merge() removed. Use xs.combine() and xs.merge() instead.
- core: debug() now returns a MemoryStream if the input was also a MemoryStream. endWhen() now returns a MemoryStream if the input was also a MemoryStream. fold() now returns always a MemoryStream, not Stream. imitate() only works on conventional Stream, will throw error on MemoryStream. map() now returns a MemoryStream if the input was also a MemoryStream. mapTo() now returns a MemoryStream if the input was also a MemoryStream. replaceError() now returns a MemoryStream if the input was also a MemoryStream. startWith() now returns always a MemoryStream, not Stream. take() now returns a MemoryStream if the input was also a MemoryStream.
3.0.0 (2016-06-02)
- extra: change flattenSequentially and pairwise signatures (71df158)
- extra: move flattenConcurrently from core to extra (7d0fc01)
- imitate: fix imitate, should not add listener immediately (a6e39d2), closes #5 #5
- extra: Usage of flattenSequentially have changed, from compose(flattenSequentially()) to compose(flattenSequentially) and from compose(pairwise()) and compose(pairwise).
- extra: flattenConcurrently must be separately imported as an extra operator and used with .compose()
- imitate: imitate() method on Stream removed. New type introduced: MimicStream, which can be created through xs.createMimic(). A MimicStream has the method imitate(), which has the same API as before, but imitate does not trigger any Stream/Producer to start.
2.6.2 (2016-05-25)
2.6.1 (2016-05-23)
2.6.0 (2016-05-21)
- debug: add support for label argument to debug() (9231851)
2.5.0 (2016-05-21)
- extra: add new extra factory fromDiagram (d6c4ae5)
2.4.3 (2016-05-16)
- extra: add safety check against nulls for next() etc (cf82a8b)
- debounce: improve debounce speed/rate (8bf7903)
2.4.2 (2016-05-13)
- flatten: fix map+flatten fusion to respect filter+map fusion (6520550)
2.4.1 (2016-05-13)
- operators: add safety check against nulls for next() etc (5d433c3)
- operators: improve type metadata for operators with fusion (fb1e81c)
2.4.0 (2016-05-12)
- flatten: add ins field as metadata to flatten (cbc1f8b)
- extra: implement new extra operator: dropUntil (e06d502)
- extra: implement new extra operator: split (84742e8)
2.3.0 (2016-05-09)
- combine: fix combine() to export its Producer class (700a129)
- operators: add type metadata string to all operators/producers (a734fd4)
2.2.1 (2016-05-03)
- combine: apply some perf optimizations to combine (ee4ec4c)
2.2.0 (2016-05-02)
- combine: support zero streams args to combine() (1b3ca90)
2.1.4 (2016-05-02)
- combine: guard CombineListener against invalid out stream (74c6061)
- flatten: avoid cut() method in flattening (28afee9)
2.1.3 (2016-04-30)
- remember: return MemoryStream, not Stream (4f50922)
2.1.2 (2016-04-30)
- combine: fix CombineFactorySignature (c65bd0b)
2.1.1 (2016-04-30)
- remember: build safety against map+map fusion (079602c)
2.1.0 (2016-04-30)
- flatten: fix TypeScript output type (26f2241), closes #4
- flattenConcurrently: fix TypeScript output type (b5445a5), closes #4
- create: Throw an error if for incomplete producer (39c7c80)
2.0.2 (2016-04-28)
- filter: fix filter fusion logic. (8c417f9)
- Stream: improve way of fixing ils array concurrency (accd2d0)
2.0.1 (2016-04-28)
- take: fix take() behavior when stopping (438fc0f)
2.0.0 (2016-04-27)
- package: put extra operators in xstream/extra (2735a74)
- package: Import extra operators from xstream/extra/the-operator-you-want not from xstream/lib/extra/the-operator-you-want
1.1.1 (2016-04-27)
- addListener: throw an error if next, error or complete functions are missing (b6e9df3)
1.1.0 (2016-04-26)
- core: export all operator classes (10ef8f3)
- package: fix TS dependency on es6-promise, and bump (4c8adb8)
- package.json: add typings field, bump to 1.0.4 (bffd84b)
- typings: fix usage of ambient es6-promise (6b4ae8e)
- typings: make es6-promise an ambient dep, and bump (49edd74)
- extra: implement new flattenSequentially() extra operator (4a6e63e)
1.0.1 (2016-04-22)
- compose2: fix type signature errors (5c77ff9)
- core: fix map type signature (133c400)
- dropRepeats: move dropRepeats from core to extra (78851c8)
- filterMap: properly catch errors that could be thrown (8ff48a5)
- flattenConcurrently: fix inner management when optimization is off (da1f379)
- fromArray: rename from() producer to fromArray() (05f519a)
- fromEvent: rename static domEvent() to fromEvent() as extra (c481cc8)
- MemoryStream: fix how MemoryStream handles late sync emissions (00de09d)
- operator: add more tear down logic in _stop() in operators (2483107)
- operator: fix all operators redirection of error/complete (2caa2ca)
- package.json: no postinstall npm script anymore (4011aa1)
- periodic: rename interval() factory to periodic() (6a2adc5)
- src: make index be an import facade for core.ts (180f7c4)
- Stream: fix unsubscription semantics w.r.t. restarting (9a0f3af)
- Stream: stop the producer syncly after stream completes (faba7bf)
- Stream: stop the producer syncly after the Stream errors (6c803ac)
- Stream: use underscore for pseudo-private fields in Stream (95f2ebb)
- take: fix take() operator, and also combine and merge (c5fdfc0)
- concat: implement extra concat() operator (7652011)
- core: flatten and flattenConcurrently should optimize for FilterMapOperator (e1bebff)
- core: implement filter + map fusion (b0507e6)
- core: use filterMap fusion for map() + filter (a723fa4)
- createWithMemory: rename xs.MemoryStream to xs.createWithMemory (c88d6c2)
- debounce: implement debounce operator (7dfb709)
- debounce: make debounce an extra operator (34fd6c1)
- delay: implement extra operator delay() and compose() (48c5abc)
- domEvent: implement domEvent stream constructor (ad40a08)
- drop: rename skip() to drop() (cab26a9)
- dropRepeats: implement core instance operator dropRepeats() (b7dccf9)
- emptyObserver: makes emptyObserver noop functions instead of null (e1d2537)
- endWhen: implement operator endWhen(), add tests (23099ef)
- factory: add factory from() with FromMachine (e76acef)
- factory: implement merge() with MergeProducer (42b6f12)
- filterMap: implement all combinations of filter and map fusion (5eb5822)
- flatten: implement flatten operator, a.k.a. switch() (6255e53)
- flattenConcurrently: rename flatten to flattenConcurrently (b3a87ee)
- fromPromise: implement factory fromPromise() (ad0ccfd)
- imitate: implement imitate() operator for circular dependencies (6545670)
- index: export new domEvent constructor (870fdc6)
- mapTo: implement mapTo (f73bc8e)
- MapTo: adjust to more private variables (a5ed5ab)
- Observer: rename complete() callback to end() (d282684)
- operator: implement combine(), both static and instance (f65a6a3)
- operator: implement debug() operator with DebugMachine (e2a0342)
- operator: implement filter operator with FilterMachine (a74f160)
- operator: implement flatten() operator (4800873)
- operator: implement fold operator with FoldMachine (57453f2)
- operator: implement last() operator with LastMachine (747e255)
- operator: implement map operator with MapMachine (76df500)
- operator: implement skip operator with SkipMachine (32dd8ac)
- operator: implement take operator with TakeMachine (6e1d0db)
- pairwise: implement extra operator pairwise() (5b1ec51)
- remember: implement RememeberProducer (7279ad8)
- RememberOperator: adjust to work with MemoryStream (0898404)
- replaceError: implement replaceError(), wrap code with try-catch (ffa5976)
- shamefullySendNext: introduce shamefullySendNext and hide _next callback (552caff)
- startWith: implement startWith operator (3489ce3)
- Stream: add a concept of current value (cc5650f)
- Stream: add debounce to Stream prototype (f44b819)
- Stream: add mapTo to Stream prototype (58c83f9)
- Stream: add never() and empty() stream factories (04f59b0)
- Stream: implement really simply Stream and interval() factory (a3a08e7)
- Stream: implement Stream (86d68ff)
- Stream: implement xs.of() (f86fd49)
- takeUntil: implement and test takeUntil() (304bed1)
- throw: implement new static factory throw() (76879a5)
- core: have FilterMapOperator extend MapOperator (e0c153a)
- debug: improve performance of debug() operator, using Proxy class (9f766af)
- filter-map-reduce: add preliminary perf micro benchmarks (8b1f2d3)
- filter-map-reduce: improve filter-map-reduce test to actually do reduce() too (7ff9fd0)
- fold: improve performance by using shorter names (8a25fe7)
- from: improve from factory perf by renaming a var (a814c8a)
- fromArray: rename/fix from() to fromArray() in perf benchmarks (a433dd5)
- merge: add merge performance benchmark (de9f002)
- operator: fix all operators to refer this.proxy initially to emptyObserver (ad210fc)
- operator: replace operator proxies with class, improves perf (2e6ec27)
- perf: fix xstream perf benchmark for merge() (4758a1d)
- scan: add performance benchmark for fold (5d5ef94)
- skip: improve skip perf by using Proxy Observer class (5233f43)
- Stream: improve performance of Observer methods in Stream (465f22d)
- Stream: remove this.num in Stream to improve perf (53bcaad)
- Stream: squeeze kB size in map and filter fusion (23ac9d0)
- Stream: tiny saving of lookups and source code size (6527129)
- take: improve take() perf by using Proxy Observer class (6eae1a9)
- takeUntil: revert takeUntil implementation (6f62fc1)