ng-http-caching 15.2.4 → 16.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (28) hide show
  1. package/esm2022/lib/ng-http-caching-interceptor.service.mjs +72 -0
  2. package/esm2022/lib/ng-http-caching.module.mjs +43 -0
  3. package/esm2022/lib/ng-http-caching.service.mjs +330 -0
  4. package/{esm2020 → esm2022}/lib/storage/ng-http-caching-browser-storage.mjs +122 -122
  5. package/{esm2020 → esm2022}/lib/storage/ng-http-caching-local-storage.mjs +6 -6
  6. package/{esm2020 → esm2022}/lib/storage/ng-http-caching-memory-storage.mjs +26 -26
  7. package/{esm2020 → esm2022}/lib/storage/ng-http-caching-session-storage.mjs +6 -6
  8. package/{esm2020 → esm2022}/lib/storage/ng-http-caching-storage.interface.mjs +1 -1
  9. package/{esm2020 → esm2022}/ng-http-caching.mjs +4 -4
  10. package/{esm2020 → esm2022}/public-api.mjs +11 -11
  11. package/{fesm2020 → fesm2022}/ng-http-caching.mjs +578 -578
  12. package/{fesm2020 → fesm2022}/ng-http-caching.mjs.map +1 -1
  13. package/index.d.ts +5 -5
  14. package/lib/ng-http-caching-interceptor.service.d.ts +15 -15
  15. package/lib/ng-http-caching.module.d.ts +9 -9
  16. package/lib/ng-http-caching.service.d.ts +213 -213
  17. package/lib/storage/ng-http-caching-browser-storage.d.ts +18 -18
  18. package/lib/storage/ng-http-caching-local-storage.d.ts +4 -4
  19. package/lib/storage/ng-http-caching-memory-storage.d.ts +12 -12
  20. package/lib/storage/ng-http-caching-session-storage.d.ts +4 -4
  21. package/lib/storage/ng-http-caching-storage.interface.d.ts +31 -31
  22. package/package.json +7 -13
  23. package/public-api.d.ts +8 -8
  24. package/esm2020/lib/ng-http-caching-interceptor.service.mjs +0 -71
  25. package/esm2020/lib/ng-http-caching.module.mjs +0 -42
  26. package/esm2020/lib/ng-http-caching.service.mjs +0 -329
  27. package/fesm2015/ng-http-caching.mjs +0 -606
  28. package/fesm2015/ng-http-caching.mjs.map +0 -1
@@ -0,0 +1,72 @@
1
+ import { HttpResponse } from '@angular/common/http';
2
+ import { Injectable } from '@angular/core';
3
+ import { asapScheduler, scheduled } from 'rxjs';
4
+ import { tap, finalize, shareReplay } from 'rxjs/operators';
5
+ import { NgHttpCachingHeadersList } from './ng-http-caching.service';
6
+ import * as i0 from "@angular/core";
7
+ import * as i1 from "./ng-http-caching.service";
8
+ /**
9
+ * Fix for https://github.com/ReactiveX/rxjs/issues/7241
10
+ */
11
+ function* _of(value) {
12
+ yield value;
13
+ }
14
+ class NgHttpCachingInterceptorService {
15
+ constructor(cacheService) {
16
+ this.cacheService = cacheService;
17
+ }
18
+ intercept(req, next) {
19
+ // run garbage collector
20
+ this.cacheService.runGc();
21
+ // Don't cache if it's not cacheable
22
+ if (!this.cacheService.isCacheable(req)) {
23
+ return this.sendRequest(req, next);
24
+ }
25
+ // Checked if there is pending response for this request
26
+ const cachedObservable = this.cacheService.getFromQueue(req);
27
+ if (cachedObservable) {
28
+ // console.log('cachedObservable',cachedObservable);
29
+ return cachedObservable;
30
+ }
31
+ // Checked if there is cached response for this request
32
+ const cachedResponse = this.cacheService.getFromCache(req);
33
+ if (cachedResponse) {
34
+ // console.log('cachedResponse');
35
+ return scheduled(_of(cachedResponse.clone()), asapScheduler);
36
+ }
37
+ // If the request of going through for first time
38
+ // then let the request proceed and cache the response
39
+ // console.log('sendRequest', req);
40
+ const shared = this.sendRequest(req, next).pipe(tap(event => {
41
+ if (event instanceof HttpResponse) {
42
+ this.cacheService.addToCache(req, event.clone());
43
+ }
44
+ }), finalize(() => {
45
+ // delete pending request
46
+ this.cacheService.deleteFromQueue(req);
47
+ }), shareReplay());
48
+ // add pending request to queue for cache parallell request
49
+ this.cacheService.addToQueue(req, shared);
50
+ return shared;
51
+ }
52
+ /**
53
+ * Send http request (next handler)
54
+ */
55
+ sendRequest(req, next) {
56
+ let cloned = req.clone();
57
+ // trim custom headers before send request
58
+ NgHttpCachingHeadersList.forEach(ngHttpCachingHeaders => {
59
+ if (cloned.headers.has(ngHttpCachingHeaders)) {
60
+ cloned = cloned.clone({ headers: cloned.headers.delete(ngHttpCachingHeaders) });
61
+ }
62
+ });
63
+ return next.handle(cloned);
64
+ }
65
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingInterceptorService, deps: [{ token: i1.NgHttpCachingService }], target: i0.ɵɵFactoryTarget.Injectable }); }
66
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingInterceptorService }); }
67
+ }
68
+ export { NgHttpCachingInterceptorService };
69
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingInterceptorService, decorators: [{
70
+ type: Injectable
71
+ }], ctorParameters: function () { return [{ type: i1.NgHttpCachingService }]; } });
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmctaHR0cC1jYWNoaW5nLWludGVyY2VwdG9yLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9uZy1odHRwLWNhY2hpbmcvc3JjL2xpYi9uZy1odHRwLWNhY2hpbmctaW50ZXJjZXB0b3Iuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQXdELFlBQVksRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQzFHLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDM0MsT0FBTyxFQUFFLGFBQWEsRUFBYyxTQUFTLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDNUQsT0FBTyxFQUFFLEdBQUcsRUFBRSxRQUFRLEVBQUUsV0FBVyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDNUQsT0FBTyxFQUF3Qix3QkFBd0IsRUFBRSxNQUFNLDJCQUEyQixDQUFDOzs7QUFFM0Y7O0dBRUc7QUFDSCxRQUFRLENBQUMsQ0FBQyxHQUFHLENBQUksS0FBUTtJQUN2QixNQUFNLEtBQUssQ0FBQztBQUNkLENBQUM7QUFFRCxNQUNhLCtCQUErQjtJQUUxQyxZQUE2QixZQUFrQztRQUFsQyxpQkFBWSxHQUFaLFlBQVksQ0FBc0I7SUFBSSxDQUFDO0lBRXBFLFNBQVMsQ0FBQyxHQUFxQixFQUFFLElBQWlCO1FBQ2hELHdCQUF3QjtRQUN4QixJQUFJLENBQUMsWUFBWSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBRTFCLG9DQUFvQztRQUNwQyxJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFDLEVBQUU7WUFDdkMsT0FBTyxJQUFJLENBQUMsV0FBVyxDQUFDLEdBQUcsRUFBRSxJQUFJLENBQUMsQ0FBQztTQUNwQztRQUVELHdEQUF3RDtRQUN4RCxNQUFNLGdCQUFnQixHQUEyQyxJQUFJLENBQUMsWUFBWSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUNyRyxJQUFJLGdCQUFnQixFQUFFO1lBQ3BCLG9EQUFvRDtZQUNwRCxPQUFPLGdCQUFnQixDQUFDO1NBQ3pCO1FBRUQsdURBQXVEO1FBQ3ZELE1BQU0sY0FBYyxHQUFrQyxJQUFJLENBQUMsWUFBWSxDQUFDLFlBQVksQ0FBQyxHQUFHLENBQUMsQ0FBQztRQUMxRixJQUFJLGNBQWMsRUFBRTtZQUNsQixpQ0FBaUM7WUFDakMsT0FBTyxTQUFTLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxFQUFFLGFBQWEsQ0FBQyxDQUFDO1NBQzlEO1FBRUQsaURBQWlEO1FBQ2pELHNEQUFzRDtRQUN0RCxtQ0FBbUM7UUFDbkMsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLENBQUMsSUFBSSxDQUM3QyxHQUFHLENBQUMsS0FBSyxDQUFDLEVBQUU7WUFDVixJQUFJLEtBQUssWUFBWSxZQUFZLEVBQUU7Z0JBQ2pDLElBQUksQ0FBQyxZQUFZLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUMsQ0FBQzthQUNsRDtRQUNILENBQUMsQ0FBQyxFQUNGLFFBQVEsQ0FBQyxHQUFHLEVBQUU7WUFDWix5QkFBeUI7WUFDekIsSUFBSSxDQUFDLFlBQVksQ0FBQyxlQUFlLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDekMsQ0FBQyxDQUFDLEVBQ0YsV0FBVyxFQUFFLENBQ2QsQ0FBQztRQUVGLDJEQUEyRDtRQUMzRCxJQUFJLENBQUMsWUFBWSxDQUFDLFVBQVUsQ0FBQyxHQUFHLEVBQUUsTUFBTSxDQUFDLENBQUM7UUFFMUMsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsV0FBVyxDQUFDLEdBQXFCLEVBQUUsSUFBaUI7UUFDbEQsSUFBSSxNQUFNLEdBQXFCLEdBQUcsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUMzQywwQ0FBMEM7UUFDMUMsd0JBQXdCLENBQUMsT0FBTyxDQUFDLG9CQUFvQixDQUFDLEVBQUU7WUFDdEQsSUFBSSxNQUFNLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxvQkFBb0IsQ0FBQyxFQUFFO2dCQUM1QyxNQUFNLEdBQUcsTUFBTSxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxvQkFBb0IsQ0FBQyxFQUFFLENBQUMsQ0FBQzthQUNqRjtRQUNILENBQUMsQ0FBQyxDQUFDO1FBQ0gsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzdCLENBQUM7OEdBN0RVLCtCQUErQjtrSEFBL0IsK0JBQStCOztTQUEvQiwrQkFBK0I7MkZBQS9CLCtCQUErQjtrQkFEM0MsVUFBVSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEh0dHBFdmVudCwgSHR0cEhhbmRsZXIsIEh0dHBJbnRlcmNlcHRvciwgSHR0cFJlcXVlc3QsIEh0dHBSZXNwb25zZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbi9odHRwJztcclxuaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBhc2FwU2NoZWR1bGVyLCBPYnNlcnZhYmxlLCBzY2hlZHVsZWQgfSBmcm9tICdyeGpzJztcclxuaW1wb3J0IHsgdGFwLCBmaW5hbGl6ZSwgc2hhcmVSZXBsYXkgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XHJcbmltcG9ydCB7IE5nSHR0cENhY2hpbmdTZXJ2aWNlLCBOZ0h0dHBDYWNoaW5nSGVhZGVyc0xpc3QgfSBmcm9tICcuL25nLWh0dHAtY2FjaGluZy5zZXJ2aWNlJztcclxuXHJcbi8qKlxyXG4gKiBGaXggZm9yIGh0dHBzOi8vZ2l0aHViLmNvbS9SZWFjdGl2ZVgvcnhqcy9pc3N1ZXMvNzI0MVxyXG4gKi9cclxuZnVuY3Rpb24qIF9vZjxUPih2YWx1ZTogVCk6IEdlbmVyYXRvcjxUPiB7XHJcbiAgeWllbGQgdmFsdWU7XHJcbn1cclxuXHJcbkBJbmplY3RhYmxlKClcclxuZXhwb3J0IGNsYXNzIE5nSHR0cENhY2hpbmdJbnRlcmNlcHRvclNlcnZpY2UgaW1wbGVtZW50cyBIdHRwSW50ZXJjZXB0b3Ige1xyXG5cclxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IGNhY2hlU2VydmljZTogTmdIdHRwQ2FjaGluZ1NlcnZpY2UpIHsgfVxyXG5cclxuICBpbnRlcmNlcHQocmVxOiBIdHRwUmVxdWVzdDxhbnk+LCBuZXh0OiBIdHRwSGFuZGxlcik6IE9ic2VydmFibGU8SHR0cEV2ZW50PGFueT4+IHtcclxuICAgIC8vIHJ1biBnYXJiYWdlIGNvbGxlY3RvclxyXG4gICAgdGhpcy5jYWNoZVNlcnZpY2UucnVuR2MoKTtcclxuXHJcbiAgICAvLyBEb24ndCBjYWNoZSBpZiBpdCdzIG5vdCBjYWNoZWFibGVcclxuICAgIGlmICghdGhpcy5jYWNoZVNlcnZpY2UuaXNDYWNoZWFibGUocmVxKSkge1xyXG4gICAgICByZXR1cm4gdGhpcy5zZW5kUmVxdWVzdChyZXEsIG5leHQpO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIENoZWNrZWQgaWYgdGhlcmUgaXMgcGVuZGluZyByZXNwb25zZSBmb3IgdGhpcyByZXF1ZXN0XHJcbiAgICBjb25zdCBjYWNoZWRPYnNlcnZhYmxlOiBPYnNlcnZhYmxlPEh0dHBFdmVudDxhbnk+PiB8IHVuZGVmaW5lZCA9IHRoaXMuY2FjaGVTZXJ2aWNlLmdldEZyb21RdWV1ZShyZXEpO1xyXG4gICAgaWYgKGNhY2hlZE9ic2VydmFibGUpIHtcclxuICAgICAgLy8gY29uc29sZS5sb2coJ2NhY2hlZE9ic2VydmFibGUnLGNhY2hlZE9ic2VydmFibGUpO1xyXG4gICAgICByZXR1cm4gY2FjaGVkT2JzZXJ2YWJsZTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBDaGVja2VkIGlmIHRoZXJlIGlzIGNhY2hlZCByZXNwb25zZSBmb3IgdGhpcyByZXF1ZXN0XHJcbiAgICBjb25zdCBjYWNoZWRSZXNwb25zZTogSHR0cFJlc3BvbnNlPGFueT4gfCB1bmRlZmluZWQgPSB0aGlzLmNhY2hlU2VydmljZS5nZXRGcm9tQ2FjaGUocmVxKTtcclxuICAgIGlmIChjYWNoZWRSZXNwb25zZSkge1xyXG4gICAgICAvLyBjb25zb2xlLmxvZygnY2FjaGVkUmVzcG9uc2UnKTtcclxuICAgICAgcmV0dXJuIHNjaGVkdWxlZChfb2YoY2FjaGVkUmVzcG9uc2UuY2xvbmUoKSksIGFzYXBTY2hlZHVsZXIpO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIElmIHRoZSByZXF1ZXN0IG9mIGdvaW5nIHRocm91Z2ggZm9yIGZpcnN0IHRpbWVcclxuICAgIC8vIHRoZW4gbGV0IHRoZSByZXF1ZXN0IHByb2NlZWQgYW5kIGNhY2hlIHRoZSByZXNwb25zZVxyXG4gICAgLy8gY29uc29sZS5sb2coJ3NlbmRSZXF1ZXN0JywgcmVxKTtcclxuICAgIGNvbnN0IHNoYXJlZCA9IHRoaXMuc2VuZFJlcXVlc3QocmVxLCBuZXh0KS5waXBlKFxyXG4gICAgICB0YXAoZXZlbnQgPT4ge1xyXG4gICAgICAgIGlmIChldmVudCBpbnN0YW5jZW9mIEh0dHBSZXNwb25zZSkge1xyXG4gICAgICAgICAgdGhpcy5jYWNoZVNlcnZpY2UuYWRkVG9DYWNoZShyZXEsIGV2ZW50LmNsb25lKCkpO1xyXG4gICAgICAgIH1cclxuICAgICAgfSksXHJcbiAgICAgIGZpbmFsaXplKCgpID0+IHtcclxuICAgICAgICAvLyBkZWxldGUgcGVuZGluZyByZXF1ZXN0XHJcbiAgICAgICAgdGhpcy5jYWNoZVNlcnZpY2UuZGVsZXRlRnJvbVF1ZXVlKHJlcSk7XHJcbiAgICAgIH0pLFxyXG4gICAgICBzaGFyZVJlcGxheSgpXHJcbiAgICApO1xyXG5cclxuICAgIC8vIGFkZCBwZW5kaW5nIHJlcXVlc3QgdG8gcXVldWUgZm9yIGNhY2hlIHBhcmFsbGVsbCByZXF1ZXN0XHJcbiAgICB0aGlzLmNhY2hlU2VydmljZS5hZGRUb1F1ZXVlKHJlcSwgc2hhcmVkKTtcclxuXHJcbiAgICByZXR1cm4gc2hhcmVkO1xyXG4gIH1cclxuXHJcbiAgLyoqXHJcbiAgICogU2VuZCBodHRwIHJlcXVlc3QgKG5leHQgaGFuZGxlcilcclxuICAgKi9cclxuICBzZW5kUmVxdWVzdChyZXE6IEh0dHBSZXF1ZXN0PGFueT4sIG5leHQ6IEh0dHBIYW5kbGVyKTogT2JzZXJ2YWJsZTxIdHRwRXZlbnQ8YW55Pj4ge1xyXG4gICAgbGV0IGNsb25lZDogSHR0cFJlcXVlc3Q8YW55PiA9IHJlcS5jbG9uZSgpO1xyXG4gICAgLy8gdHJpbSBjdXN0b20gaGVhZGVycyBiZWZvcmUgc2VuZCByZXF1ZXN0XHJcbiAgICBOZ0h0dHBDYWNoaW5nSGVhZGVyc0xpc3QuZm9yRWFjaChuZ0h0dHBDYWNoaW5nSGVhZGVycyA9PiB7XHJcbiAgICAgIGlmIChjbG9uZWQuaGVhZGVycy5oYXMobmdIdHRwQ2FjaGluZ0hlYWRlcnMpKSB7XHJcbiAgICAgICAgY2xvbmVkID0gY2xvbmVkLmNsb25lKHsgaGVhZGVyczogY2xvbmVkLmhlYWRlcnMuZGVsZXRlKG5nSHR0cENhY2hpbmdIZWFkZXJzKSB9KTtcclxuICAgICAgfVxyXG4gICAgfSk7XHJcbiAgICByZXR1cm4gbmV4dC5oYW5kbGUoY2xvbmVkKTtcclxuICB9XHJcbn1cclxuIl19
@@ -0,0 +1,43 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { HTTP_INTERCEPTORS } from '@angular/common/http';
3
+ import { NG_HTTP_CACHING_CONFIG, NgHttpCachingService, } from './ng-http-caching.service';
4
+ import { NgHttpCachingInterceptorService } from './ng-http-caching-interceptor.service';
5
+ import * as i0 from "@angular/core";
6
+ class NgHttpCachingModule {
7
+ static forRoot(ngHttpCachingConfig) {
8
+ return {
9
+ ngModule: NgHttpCachingModule,
10
+ providers: [
11
+ {
12
+ provide: NG_HTTP_CACHING_CONFIG,
13
+ useValue: ngHttpCachingConfig,
14
+ },
15
+ ],
16
+ };
17
+ }
18
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
19
+ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingModule }); }
20
+ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingModule, providers: [
21
+ NgHttpCachingService,
22
+ {
23
+ provide: HTTP_INTERCEPTORS,
24
+ useClass: NgHttpCachingInterceptorService,
25
+ multi: true,
26
+ },
27
+ ] }); }
28
+ }
29
+ export { NgHttpCachingModule };
30
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingModule, decorators: [{
31
+ type: NgModule,
32
+ args: [{
33
+ providers: [
34
+ NgHttpCachingService,
35
+ {
36
+ provide: HTTP_INTERCEPTORS,
37
+ useClass: NgHttpCachingInterceptorService,
38
+ multi: true,
39
+ },
40
+ ]
41
+ }]
42
+ }] });
43
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmctaHR0cC1jYWNoaW5nLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL25nLWh0dHAtY2FjaGluZy9zcmMvbGliL25nLWh0dHAtY2FjaGluZy5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFFBQVEsRUFBdUIsTUFBTSxlQUFlLENBQUM7QUFDOUQsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFDekQsT0FBTyxFQUNMLHNCQUFzQixFQUV0QixvQkFBb0IsR0FDckIsTUFBTSwyQkFBMkIsQ0FBQztBQUNuQyxPQUFPLEVBQUUsK0JBQStCLEVBQUUsTUFBTSx1Q0FBdUMsQ0FBQzs7QUFFeEYsTUFVYSxtQkFBbUI7SUFDOUIsTUFBTSxDQUFDLE9BQU8sQ0FDWixtQkFBeUM7UUFFekMsT0FBTztZQUNMLFFBQVEsRUFBRSxtQkFBbUI7WUFDN0IsU0FBUyxFQUFFO2dCQUNUO29CQUNFLE9BQU8sRUFBRSxzQkFBc0I7b0JBQy9CLFFBQVEsRUFBRSxtQkFBbUI7aUJBQzlCO2FBQ0Y7U0FDRixDQUFDO0lBQ0osQ0FBQzs4R0FiVSxtQkFBbUI7K0dBQW5CLG1CQUFtQjsrR0FBbkIsbUJBQW1CLGFBVG5CO1lBQ1Qsb0JBQW9CO1lBQ3BCO2dCQUNFLE9BQU8sRUFBRSxpQkFBaUI7Z0JBQzFCLFFBQVEsRUFBRSwrQkFBK0I7Z0JBQ3pDLEtBQUssRUFBRSxJQUFJO2FBQ1o7U0FDRjs7U0FFVSxtQkFBbUI7MkZBQW5CLG1CQUFtQjtrQkFWL0IsUUFBUTttQkFBQztvQkFDUixTQUFTLEVBQUU7d0JBQ1Qsb0JBQW9CO3dCQUNwQjs0QkFDRSxPQUFPLEVBQUUsaUJBQWlCOzRCQUMxQixRQUFRLEVBQUUsK0JBQStCOzRCQUN6QyxLQUFLLEVBQUUsSUFBSTt5QkFDWjtxQkFDRjtpQkFDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5nTW9kdWxlLCBNb2R1bGVXaXRoUHJvdmlkZXJzIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7IEhUVFBfSU5URVJDRVBUT1JTIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG5pbXBvcnQge1xyXG4gIE5HX0hUVFBfQ0FDSElOR19DT05GSUcsXHJcbiAgTmdIdHRwQ2FjaGluZ0NvbmZpZyxcclxuICBOZ0h0dHBDYWNoaW5nU2VydmljZSxcclxufSBmcm9tICcuL25nLWh0dHAtY2FjaGluZy5zZXJ2aWNlJztcclxuaW1wb3J0IHsgTmdIdHRwQ2FjaGluZ0ludGVyY2VwdG9yU2VydmljZSB9IGZyb20gJy4vbmctaHR0cC1jYWNoaW5nLWludGVyY2VwdG9yLnNlcnZpY2UnO1xyXG5cclxuQE5nTW9kdWxlKHtcclxuICBwcm92aWRlcnM6IFtcclxuICAgIE5nSHR0cENhY2hpbmdTZXJ2aWNlLFxyXG4gICAge1xyXG4gICAgICBwcm92aWRlOiBIVFRQX0lOVEVSQ0VQVE9SUyxcclxuICAgICAgdXNlQ2xhc3M6IE5nSHR0cENhY2hpbmdJbnRlcmNlcHRvclNlcnZpY2UsXHJcbiAgICAgIG11bHRpOiB0cnVlLFxyXG4gICAgfSxcclxuICBdXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBOZ0h0dHBDYWNoaW5nTW9kdWxlIHtcclxuICBzdGF0aWMgZm9yUm9vdChcclxuICAgIG5nSHR0cENhY2hpbmdDb25maWc/OiBOZ0h0dHBDYWNoaW5nQ29uZmlnXHJcbiAgKTogTW9kdWxlV2l0aFByb3ZpZGVyczxOZ0h0dHBDYWNoaW5nTW9kdWxlPiB7XHJcbiAgICByZXR1cm4ge1xyXG4gICAgICBuZ01vZHVsZTogTmdIdHRwQ2FjaGluZ01vZHVsZSxcclxuICAgICAgcHJvdmlkZXJzOiBbXHJcbiAgICAgICAge1xyXG4gICAgICAgICAgcHJvdmlkZTogTkdfSFRUUF9DQUNISU5HX0NPTkZJRyxcclxuICAgICAgICAgIHVzZVZhbHVlOiBuZ0h0dHBDYWNoaW5nQ29uZmlnLFxyXG4gICAgICAgIH0sXHJcbiAgICAgIF0sXHJcbiAgICB9O1xyXG4gIH1cclxufVxyXG4iXX0=
@@ -0,0 +1,330 @@
1
+ import { Injectable, InjectionToken, Inject, Optional, VERSION, isDevMode } from '@angular/core';
2
+ import { NgHttpCachingMemoryStorage } from './storage/ng-http-caching-memory-storage';
3
+ import * as i0 from "@angular/core";
4
+ export const NG_HTTP_CACHING_CONFIG = new InjectionToken('ng-http-caching.config');
5
+ export var NgHttpCachingStrategy;
6
+ (function (NgHttpCachingStrategy) {
7
+ /**
8
+ * All request are cacheable if HTTP method is into `allowedMethod`
9
+ */
10
+ NgHttpCachingStrategy["ALLOW_ALL"] = "ALLOW_ALL";
11
+ /**
12
+ * Only the request with `X-NG-HTTP-CACHING-ALLOW-CACHE` header are cacheable if HTTP method is into `allowedMethod`
13
+ */
14
+ NgHttpCachingStrategy["DISALLOW_ALL"] = "DISALLOW_ALL";
15
+ })(NgHttpCachingStrategy || (NgHttpCachingStrategy = {}));
16
+ export var NgHttpCachingHeaders;
17
+ (function (NgHttpCachingHeaders) {
18
+ /**
19
+ * Request is cacheable if HTTP method is into `allowedMethod`
20
+ */
21
+ NgHttpCachingHeaders["ALLOW_CACHE"] = "X-NG-HTTP-CACHING-ALLOW-CACHE";
22
+ /**
23
+ * Request isn't cacheable
24
+ */
25
+ NgHttpCachingHeaders["DISALLOW_CACHE"] = "X-NG-HTTP-CACHING-DISALLOW-CACHE";
26
+ /**
27
+ * Specific cache lifetime for the request
28
+ */
29
+ NgHttpCachingHeaders["LIFETIME"] = "X-NG-HTTP-CACHING-LIFETIME";
30
+ /**
31
+ * You can tag multiple request by adding this header with the same tag and
32
+ * using `NgHttpCachingService.clearCacheByTag(tag: string)` for delete all the tagged request
33
+ */
34
+ NgHttpCachingHeaders["TAG"] = "X-NG-HTTP-CACHING-TAG";
35
+ })(NgHttpCachingHeaders || (NgHttpCachingHeaders = {}));
36
+ export const NgHttpCachingHeadersList = Object.values(NgHttpCachingHeaders);
37
+ export const NG_HTTP_CACHING_SECOND_IN_MS = 1000;
38
+ export const NG_HTTP_CACHING_MINUTE_IN_MS = NG_HTTP_CACHING_SECOND_IN_MS * 60;
39
+ export const NG_HTTP_CACHING_HOUR_IN_MS = NG_HTTP_CACHING_MINUTE_IN_MS * 60;
40
+ export const NG_HTTP_CACHING_DAY_IN_MS = NG_HTTP_CACHING_HOUR_IN_MS * 24;
41
+ export const NG_HTTP_CACHING_WEEK_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 7;
42
+ export const NG_HTTP_CACHING_MONTH_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 30;
43
+ export const NG_HTTP_CACHING_YEAR_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 365;
44
+ export const NgHttpCachingConfigDefault = {
45
+ store: new NgHttpCachingMemoryStorage(),
46
+ lifetime: NG_HTTP_CACHING_HOUR_IN_MS,
47
+ version: VERSION.major,
48
+ allowedMethod: ['GET', 'HEAD'],
49
+ cacheStrategy: NgHttpCachingStrategy.ALLOW_ALL,
50
+ };
51
+ class NgHttpCachingService {
52
+ constructor(config) {
53
+ this.queue = new Map();
54
+ this.gcLock = false;
55
+ this.devMode = isDevMode();
56
+ if (config) {
57
+ this.config = { ...NgHttpCachingConfigDefault, ...config };
58
+ }
59
+ else {
60
+ this.config = { ...NgHttpCachingConfigDefault };
61
+ }
62
+ // start cache clean
63
+ this.runGc();
64
+ }
65
+ /**
66
+ * Return the config
67
+ */
68
+ getConfig() {
69
+ return this.config;
70
+ }
71
+ /**
72
+ * Return the queue map
73
+ */
74
+ getQueue() {
75
+ return this.queue;
76
+ }
77
+ /**
78
+ * Return the cache store
79
+ */
80
+ getStore() {
81
+ return this.config.store;
82
+ }
83
+ /**
84
+ * Return response from cache
85
+ */
86
+ getFromCache(req) {
87
+ const key = this.getKey(req);
88
+ const cached = this.config.store.get(key);
89
+ if (!cached) {
90
+ return undefined;
91
+ }
92
+ if (this.isExpired(cached)) {
93
+ this.clearCacheByKey(key);
94
+ return undefined;
95
+ }
96
+ return this.deepFreeze(cached.response);
97
+ }
98
+ /**
99
+ * Add response to cache
100
+ */
101
+ addToCache(req, res) {
102
+ const entry = {
103
+ url: req.urlWithParams,
104
+ response: res,
105
+ request: req,
106
+ addedTime: Date.now(),
107
+ version: this.config.version,
108
+ };
109
+ if (this.isValid(entry)) {
110
+ const key = this.getKey(req);
111
+ this.config.store.set(key, entry);
112
+ return true;
113
+ }
114
+ return false;
115
+ }
116
+ /**
117
+ * Delete response from cache
118
+ */
119
+ deleteFromCache(req) {
120
+ const key = this.getKey(req);
121
+ return this.clearCacheByKey(key);
122
+ }
123
+ /**
124
+ * Clear the cache
125
+ */
126
+ clearCache() {
127
+ this.config.store.clear();
128
+ }
129
+ /**
130
+ * Clear the cache by key
131
+ */
132
+ clearCacheByKey(key) {
133
+ return this.config.store.delete(key);
134
+ }
135
+ /**
136
+ * Clear the cache by regex
137
+ */
138
+ clearCacheByRegex(regex) {
139
+ this.config.store.forEach((_, key) => {
140
+ if (regex.test(key)) {
141
+ this.clearCacheByKey(key);
142
+ }
143
+ });
144
+ }
145
+ /**
146
+ * Clear the cache by TAG
147
+ */
148
+ clearCacheByTag(tag) {
149
+ this.config.store.forEach((entry, key) => {
150
+ const tagHeader = entry.request.headers.get(NgHttpCachingHeaders.TAG);
151
+ if (tagHeader && tagHeader.split(',').includes(tag)) {
152
+ this.clearCacheByKey(key);
153
+ }
154
+ });
155
+ }
156
+ /**
157
+ * Run garbage collector (delete expired cache entry)
158
+ */
159
+ runGc() {
160
+ if (this.gcLock) {
161
+ return false;
162
+ }
163
+ this.gcLock = true;
164
+ this.config.store.forEach((entry, key) => {
165
+ if (this.isExpired(entry)) {
166
+ this.clearCacheByKey(key);
167
+ }
168
+ });
169
+ this.gcLock = false;
170
+ return true;
171
+ }
172
+ /**
173
+ * Return true if cache entry is expired
174
+ */
175
+ isExpired(entry) {
176
+ // if user provide custom method, use it
177
+ if (typeof this.config.isExpired === 'function') {
178
+ const result = this.config.isExpired(entry);
179
+ // if result is undefined, normal behaviour is provided
180
+ if (result !== undefined) {
181
+ return result;
182
+ }
183
+ }
184
+ // config/default lifetime
185
+ let lifetime = this.config.lifetime;
186
+ // request has own lifetime
187
+ const headerLifetime = entry.request.headers.get(NgHttpCachingHeaders.LIFETIME);
188
+ if (headerLifetime) {
189
+ lifetime = +headerLifetime;
190
+ }
191
+ // never expire if 0
192
+ if (lifetime === 0) {
193
+ return false;
194
+ }
195
+ // wrong lifetime
196
+ if (lifetime < 0 || isNaN(lifetime)) {
197
+ throw new Error('lifetime must be greater than or equal 0');
198
+ }
199
+ return entry.addedTime + lifetime < Date.now();
200
+ }
201
+ /**
202
+ * Return true if cache entry is valid for store in the cache
203
+ */
204
+ isValid(entry) {
205
+ // if user provide custom method, use it
206
+ if (typeof this.config.isValid === 'function') {
207
+ const result = this.config.isValid(entry);
208
+ // if result is undefined, normal behaviour is provided
209
+ if (result !== undefined) {
210
+ return result;
211
+ }
212
+ }
213
+ // different version
214
+ if (this.config.version !== entry.version) {
215
+ return false;
216
+ }
217
+ return true;
218
+ }
219
+ /**
220
+ * Return true if the request is cacheable
221
+ */
222
+ isCacheable(req) {
223
+ // if user provide custom method, use it
224
+ if (typeof this.config.isCacheable === 'function') {
225
+ const result = this.config.isCacheable(req);
226
+ // if result is undefined, normal behaviour is provided
227
+ if (result !== undefined) {
228
+ return result;
229
+ }
230
+ }
231
+ // request has disallow cache header
232
+ if (req.headers.has(NgHttpCachingHeaders.DISALLOW_CACHE)) {
233
+ return false;
234
+ }
235
+ // strategy is disallow all...
236
+ if (this.config.cacheStrategy === NgHttpCachingStrategy.DISALLOW_ALL) {
237
+ // request isn't allowed if come without allow header
238
+ if (!req.headers.has(NgHttpCachingHeaders.ALLOW_CACHE)) {
239
+ return false;
240
+ }
241
+ }
242
+ // if allowed method is only ALL, allow all http methos
243
+ if (this.config.allowedMethod.length === 1) {
244
+ if (this.config.allowedMethod[0] === 'ALL') {
245
+ return true;
246
+ }
247
+ }
248
+ // request is allowed if method is in allowedMethod
249
+ return this.config.allowedMethod.indexOf(req.method) !== -1;
250
+ }
251
+ /**
252
+ * Return the cache key.
253
+ * Default key is http method plus url with query parameters, eg.:
254
+ * `GET@https://github.com/nigrosimone/ng-http-caching`
255
+ */
256
+ getKey(req) {
257
+ // if user provide custom method, use it
258
+ if (typeof this.config.getKey === 'function') {
259
+ const result = this.config.getKey(req);
260
+ // if result is undefined, normal behaviour is provided
261
+ if (result !== undefined) {
262
+ return result;
263
+ }
264
+ }
265
+ // default key is req.method plus url with query parameters
266
+ return req.method + '@' + req.urlWithParams;
267
+ }
268
+ /**
269
+ * Return observable from cache
270
+ */
271
+ getFromQueue(req) {
272
+ const key = this.getKey(req);
273
+ const cached = this.queue.get(key);
274
+ if (!cached) {
275
+ return undefined;
276
+ }
277
+ return cached;
278
+ }
279
+ /**
280
+ * Add observable to cache
281
+ */
282
+ addToQueue(req, obs) {
283
+ const key = this.getKey(req);
284
+ this.queue.set(key, obs);
285
+ }
286
+ /**
287
+ * Delete observable from cache
288
+ */
289
+ deleteFromQueue(req) {
290
+ const key = this.getKey(req);
291
+ return this.queue.delete(key);
292
+ }
293
+ /**
294
+ * Recursively Object.freeze simple Javascript structures consisting of plain objects, arrays, and primitives.
295
+ * Make the data immutable.
296
+ * @returns immutable object
297
+ */
298
+ deepFreeze(object) {
299
+ // No freezing in production (for better performance).
300
+ if (!this.devMode || !object || typeof object !== 'object') {
301
+ return object;
302
+ }
303
+ // When already frozen, we assume its children are frozen (for better performance).
304
+ // This should be true if you always use `deepFreeze` to freeze objects.
305
+ //
306
+ // Note that Object.isFrozen will also return `true` for primitives (numbers,
307
+ // strings, booleans, undefined, null), so there is no need to check for
308
+ // those explicitly.
309
+ if (Object.isFrozen(object)) {
310
+ return object;
311
+ }
312
+ // At this point we know that we're dealing with either an array or plain object, so
313
+ // just freeze it and recurse on its values.
314
+ Object.freeze(object);
315
+ Object.keys(object).forEach(key => this.deepFreeze(object[key]));
316
+ return object;
317
+ }
318
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingService, deps: [{ token: NG_HTTP_CACHING_CONFIG, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); }
319
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingService }); }
320
+ }
321
+ export { NgHttpCachingService };
322
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.0", ngImport: i0, type: NgHttpCachingService, decorators: [{
323
+ type: Injectable
324
+ }], ctorParameters: function () { return [{ type: undefined, decorators: [{
325
+ type: Inject,
326
+ args: [NG_HTTP_CACHING_CONFIG]
327
+ }, {
328
+ type: Optional
329
+ }] }]; } });
330
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-http-caching.service.js","sourceRoot":"","sources":["../../../../projects/ng-http-caching/src/lib/ng-http-caching.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, InjectionToken, Inject, Optional, VERSION, isDevMode } from '@angular/core';\r\nimport { HttpRequest, HttpResponse, HttpEvent } from '@angular/common/http';\r\nimport { Observable } from 'rxjs/internal/Observable';\r\nimport { NgHttpCachingStorageInterface } from './storage/ng-http-caching-storage.interface';\r\nimport { NgHttpCachingMemoryStorage } from './storage/ng-http-caching-memory-storage';\r\n\r\nexport interface NgHttpCachingEntry<K = any, T = any> {\r\n  /**\r\n   * URL\r\n   */\r\n  url: string;\r\n  /**\r\n   * HttpResponse\r\n   */\r\n  response: HttpResponse<T>;\r\n  /**\r\n   * HttpRequest\r\n   */\r\n  request: HttpRequest<K>;\r\n  /**\r\n   * Timestam of add to cache time\r\n   */\r\n  addedTime: number;\r\n  /**\r\n   * Cache version\r\n   */\r\n  version: string;\r\n}\r\n\r\nexport const NG_HTTP_CACHING_CONFIG = new InjectionToken<NgHttpCachingConfig>(\r\n  'ng-http-caching.config'\r\n);\r\n\r\nexport enum NgHttpCachingStrategy {\r\n  /**\r\n   * All request are cacheable if HTTP method is into `allowedMethod`\r\n   */\r\n  ALLOW_ALL = 'ALLOW_ALL',\r\n  /**\r\n   * Only the request with `X-NG-HTTP-CACHING-ALLOW-CACHE` header are cacheable if HTTP method is into `allowedMethod`\r\n   */\r\n  DISALLOW_ALL = 'DISALLOW_ALL'\r\n}\r\n\r\nexport enum NgHttpCachingHeaders {\r\n  /**\r\n   * Request is cacheable if HTTP method is into `allowedMethod`\r\n   */\r\n  ALLOW_CACHE = 'X-NG-HTTP-CACHING-ALLOW-CACHE',\r\n  /**\r\n   * Request isn't cacheable\r\n   */\r\n  DISALLOW_CACHE = 'X-NG-HTTP-CACHING-DISALLOW-CACHE',\r\n  /**\r\n   * Specific cache lifetime for the request\r\n   */\r\n  LIFETIME = 'X-NG-HTTP-CACHING-LIFETIME',\r\n  /**\r\n   * You can tag multiple request by adding this header with the same tag and \r\n   * using `NgHttpCachingService.clearCacheByTag(tag: string)` for delete all the tagged request\r\n   */\r\n  TAG = 'X-NG-HTTP-CACHING-TAG'\r\n}\r\n\r\nexport const NgHttpCachingHeadersList = Object.values(NgHttpCachingHeaders);\r\n\r\nexport const NG_HTTP_CACHING_SECOND_IN_MS = 1000;\r\nexport const NG_HTTP_CACHING_MINUTE_IN_MS = NG_HTTP_CACHING_SECOND_IN_MS * 60;\r\nexport const NG_HTTP_CACHING_HOUR_IN_MS = NG_HTTP_CACHING_MINUTE_IN_MS * 60;\r\nexport const NG_HTTP_CACHING_DAY_IN_MS = NG_HTTP_CACHING_HOUR_IN_MS * 24;\r\nexport const NG_HTTP_CACHING_WEEK_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 7;\r\nexport const NG_HTTP_CACHING_MONTH_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 30;\r\nexport const NG_HTTP_CACHING_YEAR_IN_MS = NG_HTTP_CACHING_DAY_IN_MS * 365;\r\n\r\nexport interface NgHttpCachingConfig {\r\n  /**\r\n   * Set the cache store. You can implement your custom store by implement the `NgHttpCachingStorageInterface` interface, eg.:\r\n   */\r\n  store?: NgHttpCachingStorageInterface;\r\n  /**\r\n   * Number of millisecond that a response is stored in the cache. \r\n   * You can set specific \"lifetime\" for each request by add the header `X-NG-HTTP-CACHING-LIFETIME` (see example below).\r\n   */\r\n  lifetime?: number;\r\n  /**\r\n   * Array of allowed HTTP methods to cache. \r\n   * You can allow multiple methods, eg.: `['GET', 'POST', 'PUT', 'DELETE', 'HEAD']` or \r\n   * allow all methods by: `['ALL']`. If `allowedMethod` is an empty array (`[]`), no response are cached.\r\n   * *Warning!* `NgHttpCaching` use the full url (url with query parameters) as unique key for the cached response,\r\n   * this is correct for the `GET` request but is _potentially_ wrong for other type of request (eg. `POST`, `PUT`). \r\n   * You can set a different \"key\" by customizing the `getKey` config method (see `getKey` section).\r\n   */\r\n  allowedMethod?: string[];\r\n  /**\r\n   * Set the cache strategy, possible strategies are:\r\n   * - `NgHttpCachingStrategy.ALLOW_ALL`: All request are cacheable if HTTP method is into `allowedMethod`;\r\n   * - `NgHttpCachingStrategy.DISALLOW_ALL`: Only the request with `X-NG-HTTP-CACHING-ALLOW-CACHE` header are cacheable if HTTP method is into `allowedMethod`;\r\n   */\r\n  cacheStrategy?: NgHttpCachingStrategy;\r\n  /**\r\n   * Cache version. When you have a breaking change, change the version, and it'll delete the current cache automatically.\r\n   * The default value is Angular major version (eg. 13), in this way, the cache is invalitaded on every Angular upgrade.\r\n   */\r\n  version?: string;\r\n  /**\r\n   * If this function return `true` the request is expired and a new request is send to backend, if return `false` isn't expired. \r\n   * If the result is `undefined`, the normal behaviour is provided.\r\n   */\r\n  isExpired?: <K, T>(entry: NgHttpCachingEntry<K, T>) => boolean | undefined;\r\n  /**\r\n   * If this function return `true` the request is cacheable, if return `false` isn't cacheable. \r\n   * If the result is `undefined`, the normal behaviour is provided.\r\n   */\r\n  isCacheable?: <K>(req: HttpRequest<K>) => boolean | undefined;\r\n  /**\r\n   * This function return the unique key (`string`) for store the response into the cache. \r\n   * If the result is `undefined`, the normal behaviour is provided.\r\n   */\r\n  getKey?: <K>(req: HttpRequest<K>) => string | undefined;\r\n  /**\r\n   * If this function return `true` the cache entry is valid and can be strored, if return `false` isn't valid. \r\n   * If the result is `undefined`, the normal behaviour is provided.\r\n   */\r\n  isValid?: <K, T>(entry: NgHttpCachingEntry<K, T>) => boolean | undefined;\r\n}\r\n\r\nexport interface NgHttpCachingDefaultConfig extends NgHttpCachingConfig {\r\n  store: NgHttpCachingStorageInterface;\r\n  lifetime: number;\r\n  allowedMethod: string[];\r\n  cacheStrategy: NgHttpCachingStrategy;\r\n  version: string;\r\n}\r\n\r\nexport const NgHttpCachingConfigDefault: Readonly<NgHttpCachingDefaultConfig> = {\r\n  store: new NgHttpCachingMemoryStorage(),\r\n  lifetime: NG_HTTP_CACHING_HOUR_IN_MS,\r\n  version: VERSION.major,\r\n  allowedMethod: ['GET', 'HEAD'],\r\n  cacheStrategy: NgHttpCachingStrategy.ALLOW_ALL,\r\n};\r\n\r\n@Injectable()\r\nexport class NgHttpCachingService {\r\n\r\n  private readonly queue = new Map<string, Observable<HttpEvent<any>>>();\r\n\r\n  private readonly config: NgHttpCachingDefaultConfig;\r\n\r\n  private gcLock = false;\r\n\r\n  private devMode: boolean = isDevMode();\r\n\r\n  constructor(\r\n    @Inject(NG_HTTP_CACHING_CONFIG) @Optional() config: Readonly<NgHttpCachingConfig>\r\n  ) {\r\n    if (config) {\r\n      this.config = { ...NgHttpCachingConfigDefault, ...config };\r\n    } else {\r\n      this.config = { ...NgHttpCachingConfigDefault };\r\n    }\r\n    // start cache clean\r\n    this.runGc();\r\n  }\r\n\r\n  /**\r\n   * Return the config\r\n   */\r\n  getConfig(): Readonly<NgHttpCachingConfig> {\r\n    return this.config;\r\n  }\r\n\r\n  /**\r\n   * Return the queue map\r\n   */\r\n  getQueue(): Readonly<Map<string, Observable<HttpEvent<any>>>> {\r\n    return this.queue;\r\n  }\r\n\r\n  /**\r\n   * Return the cache store\r\n   */\r\n  getStore(): Readonly<NgHttpCachingStorageInterface> {\r\n    return this.config.store;\r\n  }\r\n\r\n  /**\r\n   * Return response from cache\r\n   */\r\n  getFromCache<K, T>(req: HttpRequest<K>): Readonly<HttpResponse<T>> | undefined {\r\n    const key: string = this.getKey(req);\r\n    const cached: NgHttpCachingEntry<K, T> | undefined = this.config.store.get<K, T>(key);\r\n\r\n    if (!cached) {\r\n      return undefined;\r\n    }\r\n\r\n    if (this.isExpired(cached)) {\r\n      this.clearCacheByKey(key);\r\n      return undefined;\r\n    }\r\n\r\n    return this.deepFreeze(cached.response);\r\n  }\r\n\r\n  /**\r\n   * Add response to cache\r\n   */\r\n  addToCache<K, T>(req: HttpRequest<K>, res: HttpResponse<T>): boolean {\r\n    const entry: NgHttpCachingEntry<K, T> = {\r\n      url: req.urlWithParams,\r\n      response: res,\r\n      request: req,\r\n      addedTime: Date.now(),\r\n      version: this.config.version,\r\n    };\r\n    if (this.isValid(entry)) {\r\n      const key: string = this.getKey(req);\r\n      this.config.store.set(key, entry);\r\n      return true;\r\n    }\r\n    return false;\r\n  }\r\n\r\n  /**\r\n   * Delete response from cache\r\n   */\r\n  deleteFromCache<K>(req: HttpRequest<K>): boolean {\r\n    const key: string = this.getKey(req);\r\n    return this.clearCacheByKey(key);\r\n  }\r\n\r\n  /**\r\n   * Clear the cache\r\n   */\r\n  clearCache(): void {\r\n    this.config.store.clear();\r\n  }\r\n\r\n  /**\r\n   * Clear the cache by key\r\n   */\r\n  clearCacheByKey(key: string): boolean {\r\n    return this.config.store.delete(key);\r\n  }\r\n\r\n  /**\r\n   * Clear the cache by regex\r\n   */\r\n  clearCacheByRegex<K, T>(regex: RegExp): void {\r\n    this.config.store.forEach<K, T>((_: NgHttpCachingEntry<K, T>, key: string) => {\r\n      if (regex.test(key)) {\r\n        this.clearCacheByKey(key);\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Clear the cache by TAG\r\n   */\r\n  clearCacheByTag<K, T>(tag: string): void {\r\n    this.config.store.forEach<K, T>((entry: NgHttpCachingEntry<K, T>, key: string) => {\r\n      const tagHeader = entry.request.headers.get(NgHttpCachingHeaders.TAG);\r\n      if (tagHeader && tagHeader.split(',').includes(tag)) {\r\n        this.clearCacheByKey(key);\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Run garbage collector (delete expired cache entry)\r\n   */\r\n  runGc<K, T>(): boolean {\r\n    if (this.gcLock) {\r\n      return false;\r\n    }\r\n    this.gcLock = true;\r\n    this.config.store.forEach<K, T>((entry: NgHttpCachingEntry<K, T>, key: string) => {\r\n      if (this.isExpired(entry)) {\r\n        this.clearCacheByKey(key);\r\n      }\r\n    });\r\n    this.gcLock = false;\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Return true if cache entry is expired\r\n   */\r\n  isExpired<K, T>(entry: NgHttpCachingEntry<K, T>): boolean {\r\n    // if user provide custom method, use it\r\n    if (typeof this.config.isExpired === 'function') {\r\n      const result = this.config.isExpired(entry);\r\n      // if result is undefined, normal behaviour is provided\r\n      if (result !== undefined) {\r\n        return result;\r\n      }\r\n    }\r\n    // config/default lifetime\r\n    let lifetime: number = this.config.lifetime;\r\n    // request has own lifetime\r\n    const headerLifetime = entry.request.headers.get(NgHttpCachingHeaders.LIFETIME);\r\n    if (headerLifetime) {\r\n      lifetime = +headerLifetime;\r\n    }\r\n    // never expire if 0\r\n    if (lifetime === 0) {\r\n      return false;\r\n    }\r\n    // wrong lifetime\r\n    if (lifetime < 0 || isNaN(lifetime)) {\r\n      throw new Error('lifetime must be greater than or equal 0');\r\n    }\r\n    return entry.addedTime + lifetime < Date.now();\r\n  }\r\n\r\n  /**\r\n   * Return true if cache entry is valid for store in the cache\r\n   */\r\n  isValid<K, T>(entry: NgHttpCachingEntry<K, T>): boolean {\r\n    // if user provide custom method, use it\r\n    if (typeof this.config.isValid === 'function') {\r\n      const result = this.config.isValid(entry);\r\n      // if result is undefined, normal behaviour is provided\r\n      if (result !== undefined) {\r\n        return result;\r\n      }\r\n    }\r\n    // different version\r\n    if (this.config.version !== entry.version) {\r\n      return false;\r\n    }\r\n    return true;\r\n  }\r\n\r\n  /**\r\n   * Return true if the request is cacheable\r\n   */\r\n  isCacheable<K>(req: HttpRequest<K>): boolean {\r\n    // if user provide custom method, use it\r\n    if (typeof this.config.isCacheable === 'function') {\r\n      const result = this.config.isCacheable(req);\r\n      // if result is undefined, normal behaviour is provided\r\n      if (result !== undefined) {\r\n        return result;\r\n      }\r\n    }\r\n    // request has disallow cache header\r\n    if (req.headers.has(NgHttpCachingHeaders.DISALLOW_CACHE)) {\r\n      return false;\r\n    }\r\n    // strategy is disallow all...\r\n    if (this.config.cacheStrategy === NgHttpCachingStrategy.DISALLOW_ALL) {\r\n      // request isn't allowed if come without allow header\r\n      if (!req.headers.has(NgHttpCachingHeaders.ALLOW_CACHE)) {\r\n        return false;\r\n      }\r\n    }\r\n    // if allowed method is only ALL, allow all http methos\r\n    if (this.config.allowedMethod.length === 1) {\r\n      if (this.config.allowedMethod[0] === 'ALL') {\r\n        return true;\r\n      }\r\n    }\r\n    // request is allowed if method is in allowedMethod\r\n    return this.config.allowedMethod.indexOf(req.method) !== -1;\r\n  }\r\n\r\n  /**\r\n   * Return the cache key.\r\n   * Default key is http method plus url with query parameters, eg.:\r\n   * `GET@https://github.com/nigrosimone/ng-http-caching`\r\n   */\r\n  getKey<K>(req: HttpRequest<K>): string {\r\n    // if user provide custom method, use it\r\n    if (typeof this.config.getKey === 'function') {\r\n      const result = this.config.getKey(req);\r\n      // if result is undefined, normal behaviour is provided\r\n      if (result !== undefined) {\r\n        return result;\r\n      }\r\n    }\r\n    // default key is req.method plus url with query parameters\r\n    return req.method + '@' + req.urlWithParams;\r\n  }\r\n\r\n  /**\r\n   * Return observable from cache\r\n   */\r\n  getFromQueue<K, T>(req: HttpRequest<K>): Observable<HttpEvent<T>> | undefined {\r\n    const key: string = this.getKey(req);\r\n    const cached: Observable<HttpEvent<T>> | undefined = this.queue.get(key);\r\n\r\n    if (!cached) {\r\n      return undefined;\r\n    }\r\n\r\n    return cached;\r\n  }\r\n\r\n  /**\r\n   * Add observable to cache\r\n   */\r\n  addToQueue<K, T>(req: HttpRequest<K>, obs: Observable<HttpEvent<T>>): void {\r\n    const key: string = this.getKey(req);\r\n    this.queue.set(key, obs);\r\n  }\r\n\r\n  /**\r\n   * Delete observable from cache\r\n   */\r\n  deleteFromQueue<K>(req: HttpRequest<K>): boolean {\r\n    const key: string = this.getKey(req);\r\n    return this.queue.delete(key);\r\n  }\r\n\r\n  /**\r\n   * Recursively Object.freeze simple Javascript structures consisting of plain objects, arrays, and primitives.\r\n   * Make the data immutable.\r\n   * @returns immutable object\r\n   */\r\n  private deepFreeze<S>(object: S): Readonly<S> {\r\n    // No freezing in production (for better performance).\r\n    if (!this.devMode || !object || typeof object !== 'object') {\r\n      return object as Readonly<S>;\r\n    }\r\n\r\n    // When already frozen, we assume its children are frozen (for better performance).\r\n    // This should be true if you always use `deepFreeze` to freeze objects.\r\n    //\r\n    // Note that Object.isFrozen will also return `true` for primitives (numbers,\r\n    // strings, booleans, undefined, null), so there is no need to check for\r\n    // those explicitly.\r\n    if (Object.isFrozen(object)) {\r\n      return object as Readonly<S>;\r\n    }\r\n\r\n    // At this point we know that we're dealing with either an array or plain object, so\r\n    // just freeze it and recurse on its values.\r\n    Object.freeze(object);\r\n    Object.keys(object).forEach(key => this.deepFreeze((object as any)[key]));\r\n\r\n    return object as Readonly<S>;\r\n  }\r\n}\r\n"]}