CorsPlugin.js 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. module.exports = class CorsPlugin {
  2. constructor ({ publicPath, crossorigin, integrity }) {
  3. this.crossorigin = crossorigin
  4. this.integrity = integrity
  5. this.publicPath = publicPath
  6. }
  7. apply (compiler) {
  8. const ID = `vue-cli-cors-plugin`
  9. compiler.hooks.compilation.tap(ID, compilation => {
  10. const ssri = require('ssri')
  11. const computeHash = url => {
  12. const filename = url.replace(this.publicPath, '')
  13. const asset = compilation.assets[filename]
  14. if (asset) {
  15. const src = asset.source()
  16. const integrity = ssri.fromData(src, {
  17. algorithms: ['sha384']
  18. })
  19. return integrity.toString()
  20. }
  21. }
  22. compilation.hooks.htmlWebpackPluginAlterAssetTags.tap(ID, data => {
  23. const tags = [...data.head, ...data.body]
  24. if (this.crossorigin != null) {
  25. tags.forEach(tag => {
  26. if (tag.tagName === 'script' || tag.tagName === 'link') {
  27. tag.attributes.crossorigin = this.crossorigin
  28. }
  29. })
  30. }
  31. if (this.integrity) {
  32. tags.forEach(tag => {
  33. if (tag.tagName === 'script') {
  34. const hash = computeHash(tag.attributes.src)
  35. if (hash) {
  36. tag.attributes.integrity = hash
  37. }
  38. } else if (tag.tagName === 'link' && tag.attributes.rel === 'stylesheet') {
  39. const hash = computeHash(tag.attributes.href)
  40. if (hash) {
  41. tag.attributes.integrity = hash
  42. }
  43. }
  44. })
  45. // when using SRI, Chrome somehow cannot reuse
  46. // the preloaded resource, and causes the files to be downloaded twice.
  47. // this is a Chrome bug (https://bugs.chromium.org/p/chromium/issues/detail?id=677022)
  48. // for now we disable preload if SRI is used.
  49. data.head = data.head.filter(tag => {
  50. return !(
  51. tag.tagName === 'link' &&
  52. tag.attributes.rel === 'preload'
  53. )
  54. })
  55. }
  56. })
  57. compilation.hooks.htmlWebpackPluginAfterHtmlProcessing.tap(ID, data => {
  58. data.html = data.html.replace(/\scrossorigin=""/g, ' crossorigin')
  59. })
  60. })
  61. }
  62. }