Reactive Caching of HTTP Service

前端 未结 3 809
后悔当初
后悔当初 2021-02-12 13:01

I am using RsJS 5 (5.0.1) to cache in Angular 2. It works well.

The meat of the caching function is:

const observable = Observable.defer(
    () => a         


        
3条回答
  •  小鲜肉
    小鲜肉 (楼主)
    2021-02-12 13:42

    See this demo: https://jsbin.com/todude/10/edit?js,console

    Notice that I'm trying to get cached results at 1200ms when the case is invalidated and then at 1300ms when the previous request is still pending (it takes 200ms). Both results are received as they should.

    This happens because when you subscribe and the publishReplay() doesn't contain any valid value it won't emit anything and won't complete immediately (thanks to take(1)) so it needs to subscribe to its source which makes the HTTP requests (this in fact happens in refCount()).

    Then the second subscriber won't receive anything as well and will be added to the array of observers in publishReplay(). It won't make another subscription because it's already subscribed to its source (refCount()) and is waiting for response.

    So the situation you're describing shouldn't happen I think. Eventually make a demo that demonstrates your problem.

    EDIT:

    Emitting both invalidated item and fresh items

    The following example shows a little different functionality than the linked example. If the cached response is invalidated it'll be emitted anyway and then it receives also the new value. This means the subscriber receives one or two values:

    • 1 value: The cached value
    • 2 values: The invalidated cached value and then new a fresh value that'll be cached from now on.

    The code could look like the following:

    let counter = 1;
    const RECACHE_INTERVAL = 1000;
    
    function mockDataFetch() {
      return Observable.of(counter++)
        .delay(200);
    }
    
    let source = Observable.defer(() => {
      const now = (new Date()).getTime();
    
      return mockDataFetch()
        .map(response => {
          return {
            'timestamp': now,
            'response': response,
          };
        });
    });
    
    let updateRequest = source
      .publishReplay(1)
      .refCount()
      .concatMap(value => {
        if (value.timestamp + RECACHE_INTERVAL > (new Date()).getTime()) {
          return Observable.from([value.response, null]);
        } else {
          return Observable.of(value.response);
        }
      })
      .takeWhile(value => value);
    
    
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 0:", val)), 0);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 50:", val)), 50);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 200:", val)), 200);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1200:", val)), 1200);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1300:", val)), 1300);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1500:", val)), 1500);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 3500:", val)), 3500);
    

    See live demo: https://jsbin.com/ketemi/2/edit?js,console

    This prints to console the following output:

    Response 0: 1
    Response 50: 1
    Response 200: 1
    Response 1200: 1
    Response 1300: 1
    Response 1200: 2
    Response 1300: 2
    Response 1500: 2
    Response 3500: 2
    Response 3500: 3
    

    Notice 1200 and 1300 received first the old cached value 1 immediately and then another value with the fresh 2 value.
    On the other hand 1500 received only the new value because 2 is already cached and is valid.

    The most confusing thing is probably why am I using concatMap().takeWhile(). This is because I need to make sure that the fresh response (not the invalidated) is the last value before sending complete notification and there's probably no operator for that (neither first() nor takeWhile() are applicable for this use-case).

    Emitting only the current item without waiting for refresh

    Yet another use-case could be when we want to emit only the cached value while not waiting for fresh response from the HTTP request.

    let counter = 1;
    const RECACHE_INTERVAL = 1000;
    
    function mockDataFetch() {
      return Observable.of(counter++)
        .delay(200);
    }
    
    let source = Observable.defer(() => {
      const now = (new Date()).getTime();
    
      return mockDataFetch()
        .map(response => {
          return {
            'timestamp': now,
            'response': response,
          };
        });
    });
    
    let updateRequest = source
      .publishReplay(1)
      .refCount()
      .concatMap((value, i) => {
        if (i === 0) {
          if (value.timestamp + RECACHE_INTERVAL > (new Date()).getTime()) { // is cached item valid?
            return Observable.from([value.response, null]);
          } else {
            return Observable.of(value.response);
          }
        }
        return Observable.of(null);
      })
      .takeWhile(value => value);
    
    
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 0:", val)), 0);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 50:", val)), 50);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 200:", val)), 200);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1200:", val)), 1200);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1300:", val)), 1300);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 1500:", val)), 1500);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 3500:", val)), 3500);
    setTimeout(() => updateRequest.subscribe(val => console.log("Response 3800:", val)), 3800);
    

    See live demo: https://jsbin.com/kebapu/2/edit?js,console

    This example prints to console:

    Response 0: 1
    Response 50: 1
    Response 200: 1
    Response 1200: 1
    Response 1300: 1
    Response 1500: 2
    Response 3500: 2
    Response 3800: 3
    

    Notice that both 1200 and 1300 receive value 1 because that's the cached value even though it's invalid now. The first call at 1200 just spawns a new HTTP request without waiting for its response and emits only the cached value. Then at 1500 the fresh value is cached so it's just reemitted. The same applies at 3500 and 3800.

    Note, that the subscriber at 1200 will receive the next notification immediately but the complete notification will be sent only after the HTTP request has finished. We need to wait because if we sent complete right after next it'd make the chain to dispose its disposables which should also cancel the HTTP request (which is what we definitely don't want to do).

提交回复
热议问题