NavigationRoute.js 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110
  1. /*
  2. Copyright 2018 Google LLC
  3. Use of this source code is governed by an MIT-style
  4. license that can be found in the LICENSE file or at
  5. https://opensource.org/licenses/MIT.
  6. */
  7. import { assert } from 'workbox-core/_private/assert.js';
  8. import { logger } from 'workbox-core/_private/logger.js';
  9. import { Route } from './Route.js';
  10. import './_version.js';
  11. /**
  12. * NavigationRoute makes it easy to create a
  13. * {@link workbox-routing.Route} that matches for browser
  14. * [navigation requests]{@link https://developers.google.com/web/fundamentals/primers/service-workers/high-performance-loading#first_what_are_navigation_requests}.
  15. *
  16. * It will only match incoming Requests whose
  17. * {@link https://fetch.spec.whatwg.org/#concept-request-mode|mode}
  18. * is set to `navigate`.
  19. *
  20. * You can optionally only apply this route to a subset of navigation requests
  21. * by using one or both of the `denylist` and `allowlist` parameters.
  22. *
  23. * @memberof workbox-routing
  24. * @extends workbox-routing.Route
  25. */
  26. class NavigationRoute extends Route {
  27. /**
  28. * If both `denylist` and `allowlist` are provided, the `denylist` will
  29. * take precedence and the request will not match this route.
  30. *
  31. * The regular expressions in `allowlist` and `denylist`
  32. * are matched against the concatenated
  33. * [`pathname`]{@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLHyperlinkElementUtils/pathname}
  34. * and [`search`]{@link https://developer.mozilla.org/en-US/docs/Web/API/HTMLHyperlinkElementUtils/search}
  35. * portions of the requested URL.
  36. *
  37. * *Note*: These RegExps may be evaluated against every destination URL during
  38. * a navigation. Avoid using
  39. * [complex RegExps](https://github.com/GoogleChrome/workbox/issues/3077),
  40. * or else your users may see delays when navigating your site.
  41. *
  42. * @param {workbox-routing~handlerCallback} handler A callback
  43. * function that returns a Promise resulting in a Response.
  44. * @param {Object} options
  45. * @param {Array<RegExp>} [options.denylist] If any of these patterns match,
  46. * the route will not handle the request (even if a allowlist RegExp matches).
  47. * @param {Array<RegExp>} [options.allowlist=[/./]] If any of these patterns
  48. * match the URL's pathname and search parameter, the route will handle the
  49. * request (assuming the denylist doesn't match).
  50. */
  51. constructor(handler, { allowlist = [/./], denylist = [] } = {}) {
  52. if (process.env.NODE_ENV !== 'production') {
  53. assert.isArrayOfClass(allowlist, RegExp, {
  54. moduleName: 'workbox-routing',
  55. className: 'NavigationRoute',
  56. funcName: 'constructor',
  57. paramName: 'options.allowlist',
  58. });
  59. assert.isArrayOfClass(denylist, RegExp, {
  60. moduleName: 'workbox-routing',
  61. className: 'NavigationRoute',
  62. funcName: 'constructor',
  63. paramName: 'options.denylist',
  64. });
  65. }
  66. super((options) => this._match(options), handler);
  67. this._allowlist = allowlist;
  68. this._denylist = denylist;
  69. }
  70. /**
  71. * Routes match handler.
  72. *
  73. * @param {Object} options
  74. * @param {URL} options.url
  75. * @param {Request} options.request
  76. * @return {boolean}
  77. *
  78. * @private
  79. */
  80. _match({ url, request }) {
  81. if (request && request.mode !== 'navigate') {
  82. return false;
  83. }
  84. const pathnameAndSearch = url.pathname + url.search;
  85. for (const regExp of this._denylist) {
  86. if (regExp.test(pathnameAndSearch)) {
  87. if (process.env.NODE_ENV !== 'production') {
  88. logger.log(`The navigation route ${pathnameAndSearch} is not ` +
  89. `being used, since the URL matches this denylist pattern: ` +
  90. `${regExp.toString()}`);
  91. }
  92. return false;
  93. }
  94. }
  95. if (this._allowlist.some((regExp) => regExp.test(pathnameAndSearch))) {
  96. if (process.env.NODE_ENV !== 'production') {
  97. logger.debug(`The navigation route ${pathnameAndSearch} ` + `is being used.`);
  98. }
  99. return true;
  100. }
  101. if (process.env.NODE_ENV !== 'production') {
  102. logger.log(`The navigation route ${pathnameAndSearch} is not ` +
  103. `being used, since the URL being navigated to doesn't ` +
  104. `match the allowlist.`);
  105. }
  106. return false;
  107. }
  108. }
  109. export { NavigationRoute };