UNPKG

rxjs

Version:

Reactive Extensions for modern JavaScript

91 lines (86 loc) 3.81 kB
import { Observable } from '../Observable'; import { innerFrom } from './innerFrom'; import { Subscription } from '../Subscription'; import { ObservableInput, ObservableInputTuple } from '../types'; import { argsOrArgArray } from '../util/argsOrArgArray'; import { OperatorSubscriber } from '../operators/OperatorSubscriber'; import { Subscriber } from '../Subscriber'; export function race<T extends readonly unknown[]>(inputs: [...ObservableInputTuple<T>]): Observable<T[number]>; export function race<T extends readonly unknown[]>(...inputs: [...ObservableInputTuple<T>]): Observable<T[number]>; /** * Returns an observable that mirrors the first source observable to emit an item. * * ![](race.png) * * `race` returns an observable, that when subscribed to, subscribes to all source observables immediately. * As soon as one of the source observables emits a value, the result unsubscribes from the other sources. * The resulting observable will forward all notifications, including error and completion, from the "winning" * source observable. * * If one of the used source observable throws an errors before a first notification * the race operator will also throw an error, no matter if another source observable * could potentially win the race. * * `race` can be useful for selecting the response from the fastest network connection for * HTTP or WebSockets. `race` can also be useful for switching observable context based on user * input. * * ## Example * ### Subscribes to the observable that was the first to start emitting. * * ```ts * import { race, interval } from 'rxjs'; * import { mapTo } from 'rxjs/operators'; * * const obs1 = interval(1000).pipe(mapTo('fast one')); * const obs2 = interval(3000).pipe(mapTo('medium one')); * const obs3 = interval(5000).pipe(mapTo('slow one')); * * race(obs3, obs1, obs2) * .subscribe( * winner => console.log(winner) * ); * * // Outputs * // a series of 'fast one' * ``` * * @param {...Observables} ...observables sources used to race for which Observable emits first. * @return {Observable} an Observable that mirrors the output of the first Observable to emit an item. */ export function race<T>(...sources: (ObservableInput<T> | ObservableInput<T>[])[]): Observable<any> { sources = argsOrArgArray(sources); // If only one source was passed, just return it. Otherwise return the race. return sources.length === 1 ? innerFrom(sources[0] as ObservableInput<T>) : new Observable<T>(raceInit(sources as ObservableInput<T>[])); } /** * An observable initializer function for both the static version and the * operator version of race. * @param sources The sources to race */ export function raceInit<T>(sources: ObservableInput<T>[]) { return (subscriber: Subscriber<T>) => { let subscriptions: Subscription[] = []; // Subscribe to all of the sources. Note that we are checking `subscriptions` here // Is is an array of all actively "racing" subscriptions, and it is `null` after the // race has been won. So, if we have racer that synchronously "wins", this loop will // stop before it subscribes to any more. for (let i = 0; subscriptions && !subscriber.closed && i < sources.length; i++) { subscriptions.push( innerFrom(sources[i] as ObservableInput<T>).subscribe( new OperatorSubscriber(subscriber, (value) => { if (subscriptions) { // We're still racing, but we won! So unsubscribe // all other subscriptions that we have, except this one. for (let s = 0; s < subscriptions.length; s++) { s !== i && subscriptions[s].unsubscribe(); } subscriptions = null!; } subscriber.next(value); }) ) ); } }; }