Source: lib/net/data_uri_plugin.js

  1. /**
  2. * @license
  3. * Copyright 2016 Google Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. goog.provide('shaka.net.DataUriPlugin');
  18. goog.require('shaka.log');
  19. goog.require('shaka.net.NetworkingEngine');
  20. goog.require('shaka.util.AbortableOperation');
  21. goog.require('shaka.util.Error');
  22. goog.require('shaka.util.StringUtils');
  23. goog.require('shaka.util.Uint8ArrayUtils');
  24. /**
  25. * @namespace
  26. * @summary A networking plugin to handle data URIs.
  27. * @see https://developer.mozilla.org/en-US/docs/Web/HTTP/data_URIs
  28. * @param {string} uri
  29. * @param {shaka.extern.Request} request
  30. * @param {shaka.net.NetworkingEngine.RequestType} requestType
  31. * @param {shaka.extern.ProgressUpdated} progressUpdated Called when a
  32. * progress event happened.
  33. * @return {!shaka.extern.IAbortableOperation.<shaka.extern.Response>}
  34. * @export
  35. */
  36. shaka.net.DataUriPlugin = function(uri, request, requestType, progressUpdated) {
  37. try {
  38. let parsed = shaka.net.DataUriPlugin.parse(uri);
  39. /** @type {shaka.extern.Response} */
  40. let response = {
  41. uri: uri,
  42. originalUri: uri,
  43. data: parsed.data,
  44. headers: {
  45. 'content-type': parsed.contentType,
  46. },
  47. };
  48. return shaka.util.AbortableOperation.completed(response);
  49. } catch (error) {
  50. return shaka.util.AbortableOperation.failed(error);
  51. }
  52. };
  53. /**
  54. * @param {string} uri
  55. * @return {{data: ArrayBuffer, contentType: string}}
  56. */
  57. shaka.net.DataUriPlugin.parse = function(uri) {
  58. // Extract the scheme.
  59. let parts = uri.split(':');
  60. if (parts.length < 2 || parts[0] != 'data') {
  61. shaka.log.error('Bad data URI, failed to parse scheme');
  62. throw new shaka.util.Error(
  63. shaka.util.Error.Severity.CRITICAL,
  64. shaka.util.Error.Category.NETWORK,
  65. shaka.util.Error.Code.MALFORMED_DATA_URI,
  66. uri);
  67. }
  68. let path = parts.slice(1).join(':');
  69. // Extract the encoding and MIME type (required but can be empty).
  70. let infoAndData = path.split(',');
  71. if (infoAndData.length < 2) {
  72. shaka.log.error('Bad data URI, failed to extract encoding and MIME type');
  73. throw new shaka.util.Error(
  74. shaka.util.Error.Severity.CRITICAL,
  75. shaka.util.Error.Category.NETWORK,
  76. shaka.util.Error.Code.MALFORMED_DATA_URI,
  77. uri);
  78. }
  79. let info = infoAndData[0];
  80. let dataStr = window.decodeURIComponent(infoAndData.slice(1).join(','));
  81. // The MIME type is always the first thing in the semicolon-separated list
  82. // of type parameters. It may be blank.
  83. const typeInfoList = info.split(';');
  84. const contentType = typeInfoList[0];
  85. // Check for base64 encoding, which is always the last in the
  86. // semicolon-separated list if present.
  87. let base64Encoded = false;
  88. if (typeInfoList.length > 1 &&
  89. typeInfoList[typeInfoList.length - 1] == 'base64') {
  90. base64Encoded = true;
  91. typeInfoList.pop();
  92. }
  93. // Convert the data.
  94. /** @type {ArrayBuffer} */
  95. let data;
  96. if (base64Encoded) {
  97. data = shaka.util.Uint8ArrayUtils.fromBase64(dataStr).buffer;
  98. } else {
  99. data = shaka.util.StringUtils.toUTF8(dataStr);
  100. }
  101. return {data, contentType};
  102. };
  103. shaka.net.NetworkingEngine.registerScheme('data', shaka.net.DataUriPlugin);