Huge News!Announcing our $40M Series B led by Abstract Ventures.Learn More
Socket
Sign inDemoInstall
Socket

postcss-font-grabber

Package Overview
Dependencies
Maintainers
1
Versions
21
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

postcss-font-grabber - npm Package Compare versions

Comparing version 1.0.5 to 1.0.6

34

dist/lib/font-grabber.js

@@ -36,2 +36,4 @@ 'use strict';

function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }

@@ -94,3 +96,3 @@

return result === null ? null : _url2.default.parse(result[2]);
return result === null ? null : _url2.default.parse(result[1]);
}

@@ -166,3 +168,3 @@

return false;
} else if (regexes.isFontFaceSrcContainsRemoteUrlRegex.test(decl.value) === false) {
} else if (regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test(decl.value) === false) {
return false;

@@ -175,2 +177,15 @@ }

/**
*
* @param urlSrcSources
* @param srcSource
* @returns {Array}
*/
}, {
key: 'keepUrlSrcSourceOnly',
value: function keepUrlSrcSourceOnly(urlSrcSources, srcSource) {
return regexes.isRemoteFontUrlRegex.test(srcSource) ? [].concat(_toConsumableArray(urlSrcSources), [srcSource]) : urlSrcSources;
}
/**
* Download font file and update output CSS rule correspondingly.

@@ -193,13 +208,14 @@ *

//
// One src could have multiple `url()`, they are separated with `,`.
// One src could have multiple `source`, they are separated with `,`,
// so break it down and filter out those which isn't an `url` source.
//
var srcUrls = decl.value.split(',').map(function (value) {
return value.replace(regexes.trimRegex, '');
});
var urlSrcSources = decl.value.split(',').map(function trim(srcSources) {
return srcSources.replace(regexes.trimRegex, '');
}).reduce(FontGrabber.keepUrlSrcSourceOnly, []);
//
// Use `srcUrls` to generate Url objects for download.
// Use `urlSrcSources` to generate Url objects for download.
// This will check the validation of font url, and only keep which is
// unique.
var fontFileUrlObjects = srcUrls.map(FontGrabber.generateUrlObjectFromSrc).reduce(FontGrabber.keepUniqueAndValidFontFileUrlObject, []);
var fontFileUrlObjects = urlSrcSources.map(FontGrabber.generateUrlObjectFromSrc).reduce(FontGrabber.keepUniqueAndValidFontFileUrlObject, []);

@@ -322,2 +338,2 @@ //

module.exports = exports['default'];
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/lib/font-grabber.js"],"names":[],"mappings":"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","file":"font-grabber.js","sourcesContent":["/**\n *\n */\nimport path from 'path';\nimport url from 'url';\nimport Downloader from './downloader';\nimport * as regexes from './regexes';\nimport includes from 'lodash/fp/includes';\n\n\n/**\n * The Font Grabber.\n */\nclass FontGrabber {\n  /**\n   *\n   */\n  static downloader = new Downloader();\n\n  /**\n   *\n   * @param postcssOpts\n   */\n  static validatePostcssOptions (postcssOpts) {\n    if (! postcssOpts.from) {\n      throw new Error('postcss-font-grabber requires postcss \"from\" option.');\n    }\n    if (! postcssOpts.to) {\n      throw new Error('postcss-font-grabber requires postcss \"to\" option.');\n    }\n  }\n\n  /**\n   *\n   * @param iterator\n   * @returns {function()}\n   */\n  static iterateCSSRuleWith (iterator) {\n    return (rule) => {\n      rule.each(iterator);\n    };\n  }\n\n  /**\n   *\n   * @param src\n   * @returns {*}\n   */\n  static generateUrlObjectFromSrc (src) {\n    const result = regexes.extractUrlFromFontFaceSrcRegex.exec(src);\n\n    return (\n      result === null ?\n      null :\n      url.parse(result[2])\n    );\n  }\n\n  /**\n   *\n   * @param kept\n   * @param value\n   * @returns {*}\n   */\n  static keepUniqueAndValidFontFileUrlObject (kept, value) {\n    if (\n      value &&\n      regexes.validFontExtensionRegex.test(value.pathname) &&\n      (! includes(kept, value))\n    ) {\n      kept.push(value);\n    }\n\n    return kept;\n  }\n\n  /**\n   *\n   * @param downloadDir\n   * @returns {function()}\n   */\n  static makeFontDownloadJobDispatcher (downloadDir) {\n    return (fontUrlObj) => {\n      const filename = fontUrlObj.pathname.split('/').pop();\n\n      return {\n        url     : fontUrlObj.href,\n        filename: filename,\n        promise : FontGrabber.downloader.download(\n          fontUrlObj,\n          path.join(downloadDir, filename)\n        ),\n      };\n    };\n  }\n\n  /**\n   *\n   * @param opts\n   * @param postcssOpts\n   */\n  static reviewOptions (opts, postcssOpts) {\n    if (! opts.dirPath) {\n      opts.dirPath = path.dirname(postcssOpts.to);\n    }\n  }\n\n  /**\n   * Skip Font-Face Postcss object that is:\n   *   not a Declaration\n   *   or doesn't contain `src` property\n   *   or doesn't contain remote font file\n   *\n   * @param decl\n   * @returns {boolean}\n   */\n  static shouldProcessThisFontFaceDeclaration (decl) {\n    if (decl.type !== 'decl') {\n      return false;\n    } else if (decl.prop !== 'src') {\n      return false;\n    } else if (regexes.isFontFaceSrcContainsRemoteUrlRegex.test(decl.value) === false) {\n      return false;\n    }\n\n    return true;\n  }\n\n  /**\n   * Download font file and update output CSS rule correspondingly.\n   *\n   * @param decl Postcss Declaration object.\n   * @param saveDirPath\n   * @param cssFilePath\n   * @returns {Promise}\n   */\n  static downloadFontAndUpdateDeclaration (decl, saveDirPath, cssFilePath) {\n    //\n    // This will be used to calculate relative path.\n    //\n    const cssFileDirPath = path.dirname(cssFilePath);\n\n    //\n    // One src could have multiple `url()`, they are separated with `,`.\n    //\n    const srcUrls = decl.value\n      .split(',')\n      .map(value => value.replace(regexes.trimRegex, ''));\n\n    //\n    // Use `srcUrls` to generate Url objects for download.\n    // This will check the validation of font url, and only keep which is\n    // unique.\n    const fontFileUrlObjects = srcUrls\n      .map(FontGrabber.generateUrlObjectFromSrc)\n      .reduce(FontGrabber.keepUniqueAndValidFontFileUrlObject, []);\n\n    //\n    // If there is no font file needs to be download, end this function\n    // Must return a promise.\n    //\n    if (fontFileUrlObjects.length === 0) {\n      return Promise.resolve();\n    }\n\n    //\n    // Download font to `saveDirPath` using Url objects **concurrently**\n    // and return `job` objects that contain:\n    //\n    //   url: the full url needs to be replaced\n    //   filename: the name of the saved file\n    //   promise: a promise will be fulfilled when download completed\n    //\n    const jobs = fontFileUrlObjects.map(\n      FontGrabber.makeFontDownloadJobDispatcher(saveDirPath)\n    );\n\n    return Promise.all(jobs.map(job => job.promise))\n      .then(() => {\n        //\n        // The font file might be saved in a different directory to the CSS\n        // file, before replace the CSS rule, we have to derive the relative\n        // path between them.\n        //\n        const relativePath = path.relative(\n          cssFileDirPath,\n          saveDirPath\n        );\n\n        //\n        // Replace CSS rule with every font that downloaded.\n        //\n        jobs.map(job => {\n          decl.value = decl.value.replace(\n            job.url,\n\n            //\n            // Replace `\\\\` to `/` for Windows compatibility.\n            //\n            path.join(relativePath, job.filename).replace(/\\\\/g, '/')\n          );\n        });\n      });\n  }\n\n  /**\n   *\n   * @param downloader\n   */\n  static setDownloader (downloader) {\n    FontGrabber.downloader = downloader;\n  }\n\n  /**\n   *\n   * @returns {Downloader}\n   */\n  static getDownloader () {\n    return FontGrabber.downloader;\n  }\n\n  /**\n   * Make handle function for plugin to call with.\n   *\n   * @param opts\n   * @returns {Function}\n   */\n  static makePluginHandler (opts = {}) {\n    return function (css, result) {\n      //\n      // Get the options from Postcss for later use.\n      //\n      let postcssOpts = result.opts;\n\n      //\n      // If something is missing in the Postcss options, throw an Error.\n      //\n      FontGrabber.validatePostcssOptions(postcssOpts);\n\n      //\n      // Review options for Font Grabber (This may modify them).\n      //\n      FontGrabber.reviewOptions(opts, postcssOpts);\n\n      //\n      // Process every Declaration that matchs rule `font-face` concurrently.\n      //\n      let processPromises = [];\n\n      const declarationProcessor = (decl) => {\n        if (FontGrabber.shouldProcessThisFontFaceDeclaration(decl)) {\n          processPromises.push(\n            FontGrabber.downloadFontAndUpdateDeclaration(decl, opts.dirPath, postcssOpts.to)\n          );\n        }\n      };\n\n      css.walkAtRules(/font-face/, FontGrabber.iterateCSSRuleWith(declarationProcessor));\n\n      return (\n        processPromises.length === 0 ?\n        Promise.resolve() :\n        Promise.all(processPromises)\n      );\n    };\n  }\n}\n\nexport default FontGrabber;"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/lib/font-grabber.js"],"names":[],"mappings":"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","file":"font-grabber.js","sourcesContent":["/**\n *\n */\nimport path from 'path';\nimport url from 'url';\nimport Downloader from './downloader';\nimport * as regexes from './regexes';\nimport includes from 'lodash/fp/includes';\n\n\n/**\n * The Font Grabber.\n */\nclass FontGrabber {\n  /**\n   *\n   */\n  static downloader = new Downloader();\n  \n  /**\n   *\n   * @param postcssOpts\n   */\n  static validatePostcssOptions (postcssOpts) {\n    if (! postcssOpts.from) {\n      throw new Error('postcss-font-grabber requires postcss \"from\" option.');\n    }\n    if (! postcssOpts.to) {\n      throw new Error('postcss-font-grabber requires postcss \"to\" option.');\n    }\n  }\n  \n  /**\n   *\n   * @param iterator\n   * @returns {function()}\n   */\n  static iterateCSSRuleWith (iterator) {\n    return (rule) => {\n      rule.each(iterator);\n    };\n  }\n  \n  /**\n   *\n   * @param src\n   * @returns {*}\n   */\n  static generateUrlObjectFromSrc (src) {\n    const result = regexes.extractUrlFromFontFaceSrcRegex.exec(src);\n    \n    return (\n      result === null ?\n        null :\n        url.parse(result[1])\n    );\n  }\n  \n  /**\n   *\n   * @param kept\n   * @param value\n   * @returns {*}\n   */\n  static keepUniqueAndValidFontFileUrlObject (kept, value) {\n    if (\n      value &&\n      regexes.validFontExtensionRegex.test(value.pathname) &&\n      (! includes(kept, value))\n    ) {\n      kept.push(value);\n    }\n    \n    return kept;\n  }\n  \n  /**\n   *\n   * @param downloadDir\n   * @returns {function()}\n   */\n  static makeFontDownloadJobDispatcher (downloadDir) {\n    return (fontUrlObj) => {\n      const filename = fontUrlObj.pathname.split('/').pop();\n      \n      return {\n        url     : fontUrlObj.href,\n        filename: filename,\n        promise : FontGrabber.downloader.download(\n          fontUrlObj,\n          path.join(downloadDir, filename)\n        ),\n      };\n    };\n  }\n  \n  /**\n   *\n   * @param opts\n   * @param postcssOpts\n   */\n  static reviewOptions (opts, postcssOpts) {\n    if (! opts.dirPath) {\n      opts.dirPath = path.dirname(postcssOpts.to);\n    }\n  }\n  \n  /**\n   * Skip Font-Face Postcss object that is:\n   *   not a Declaration\n   *   or doesn't contain `src` property\n   *   or doesn't contain remote font file\n   *\n   * @param decl\n   * @returns {boolean}\n   */\n  static shouldProcessThisFontFaceDeclaration (decl) {\n    if (decl.type !== 'decl') {\n      return false;\n    } else if (decl.prop !== 'src') {\n      return false;\n    } else if (regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test(decl.value) === false) {\n      return false;\n    }\n    \n    return true;\n  }\n  \n  /**\n   *\n   * @param urlSrcSources\n   * @param srcSource\n   * @returns {Array}\n   */\n  static keepUrlSrcSourceOnly (urlSrcSources, srcSource) {\n    return regexes.isRemoteFontUrlRegex.test(srcSource) ?\n      [...urlSrcSources, srcSource] :\n      urlSrcSources;\n  }\n  \n  /**\n   * Download font file and update output CSS rule correspondingly.\n   *\n   * @param decl Postcss Declaration object.\n   * @param saveDirPath\n   * @param cssFilePath\n   * @returns {Promise}\n   */\n  static downloadFontAndUpdateDeclaration (decl, saveDirPath, cssFilePath) {\n    //\n    // This will be used to calculate relative path.\n    //\n    const cssFileDirPath = path.dirname(cssFilePath);\n\n    //\n    // One src could have multiple `source`, they are separated with `,`,\n    // so break it down and filter out those which isn't an `url` source.\n    //\n    const urlSrcSources = decl.value\n      .split(',')\n      .map(function trim (srcSources) {\n        return srcSources.replace(regexes.trimRegex, '');\n      })\n      .reduce(FontGrabber.keepUrlSrcSourceOnly, []);\n\n    //\n    // Use `urlSrcSources` to generate Url objects for download.\n    // This will check the validation of font url, and only keep which is\n    // unique.\n    const fontFileUrlObjects = urlSrcSources\n      .map(FontGrabber.generateUrlObjectFromSrc)\n      .reduce(FontGrabber.keepUniqueAndValidFontFileUrlObject, []);\n    \n    //\n    // If there is no font file needs to be download, end this function\n    // Must return a promise.\n    //\n    if (fontFileUrlObjects.length === 0) {\n      return Promise.resolve();\n    }\n    \n    //\n    // Download font to `saveDirPath` using Url objects **concurrently**\n    // and return `job` objects that contain:\n    //\n    //   url: the full url needs to be replaced\n    //   filename: the name of the saved file\n    //   promise: a promise will be fulfilled when download completed\n    //\n    const jobs = fontFileUrlObjects.map(\n      FontGrabber.makeFontDownloadJobDispatcher(saveDirPath)\n    );\n    \n    return Promise.all(jobs.map(job => job.promise))\n      .then(() => {\n        //\n        // The font file might be saved in a different directory to the CSS\n        // file, before replace the CSS rule, we have to derive the relative\n        // path between them.\n        //\n        const relativePath = path.relative(\n          cssFileDirPath,\n          saveDirPath\n        );\n        \n        //\n        // Replace CSS rule with every font that downloaded.\n        //\n        jobs.map(job => {\n          decl.value = decl.value.replace(\n            job.url,\n            \n            //\n            // Replace `\\\\` to `/` for Windows compatibility.\n            //\n            path.join(relativePath, job.filename).replace(/\\\\/g, '/')\n          );\n        });\n      });\n  }\n  \n  /**\n   *\n   * @param downloader\n   */\n  static setDownloader (downloader) {\n    FontGrabber.downloader = downloader;\n  }\n  \n  /**\n   *\n   * @returns {Downloader}\n   */\n  static getDownloader () {\n    return FontGrabber.downloader;\n  }\n  \n  /**\n   * Make handle function for plugin to call with.\n   *\n   * @param opts\n   * @returns {Function}\n   */\n  static makePluginHandler (opts = {}) {\n    return function (css, result) {\n      //\n      // Get the options from Postcss for later use.\n      //\n      let postcssOpts = result.opts;\n      \n      //\n      // If something is missing in the Postcss options, throw an Error.\n      //\n      FontGrabber.validatePostcssOptions(postcssOpts);\n      \n      //\n      // Review options for Font Grabber (This may modify them).\n      //\n      FontGrabber.reviewOptions(opts, postcssOpts);\n      \n      //\n      // Process every Declaration that matchs rule `font-face` concurrently.\n      //\n      let processPromises = [];\n      \n      const declarationProcessor = (decl) => {\n        if (FontGrabber.shouldProcessThisFontFaceDeclaration(decl)) {\n          processPromises.push(\n            FontGrabber.downloadFontAndUpdateDeclaration(decl, opts.dirPath, postcssOpts.to)\n          );\n        }\n      };\n      \n      css.walkAtRules(/font-face/, FontGrabber.iterateCSSRuleWith(declarationProcessor));\n      \n      return (\n        processPromises.length === 0 ?\n          Promise.resolve() :\n          Promise.all(processPromises)\n      );\n    };\n  }\n}\n\nexport default FontGrabber;"]}

@@ -10,6 +10,7 @@ "use strict";

var extractUrlFromFontFaceSrcRegex = /^url\s*\(\s*(\"|\')(https?:[^\)]*?)(\"|\')\s*\)/;
var extractUrlFromFontFaceSrcRegex = /^url\s*\(\s*[\'\"]?(https?:[^\)]*?)[\'\"]?\s*\)/;
var validFontExtensionRegex = /\.(ttf|otf|woff|eot|svg)$/;
var trimRegex = /(^\s+|\s+$)/g;
var isFontFaceSrcContainsRemoteUrlRegex = /^\s*url\s*\(\s*(\'|\"https?:)/;
var isFontFaceSrcContainsRemoteFontUrlRegex = /(\s*|,|^)url\s*\(\s*[\'\"]?https?:/;
var isRemoteFontUrlRegex = /^\s*url\s*\(\s*[\'\"]?https?:/;

@@ -19,3 +20,4 @@ exports.extractUrlFromFontFaceSrcRegex = extractUrlFromFontFaceSrcRegex;

exports.trimRegex = trimRegex;
exports.isFontFaceSrcContainsRemoteUrlRegex = isFontFaceSrcContainsRemoteUrlRegex;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9saWIvcmVnZXhlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFJQSxJQUFNLGlDQUFzQyxpREFBNUM7QUFDQSxJQUFNLDBCQUFzQywyQkFBNUM7QUFDQSxJQUFNLFlBQXNDLGNBQTVDO0FBQ0EsSUFBTSxzQ0FBc0MsK0JBQTVDOztRQUdFLDhCLEdBQUEsOEI7UUFDQSx1QixHQUFBLHVCO1FBQ0EsUyxHQUFBLFM7UUFDQSxtQyxHQUFBLG1DIiwiZmlsZSI6InJlZ2V4ZXMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIERlZmluZSBhbGwgcmVndWxhciBleHByZXNzaW9ucyB3aWxsIGJlIHVzZWQuXG4gKi9cblxuY29uc3QgZXh0cmFjdFVybEZyb21Gb250RmFjZVNyY1JlZ2V4ICAgICAgPSAvXnVybFxccypcXChcXHMqKFxcXCJ8XFwnKShodHRwcz86W15cXCldKj8pKFxcXCJ8XFwnKVxccypcXCkvO1xuY29uc3QgdmFsaWRGb250RXh0ZW5zaW9uUmVnZXggICAgICAgICAgICAgPSAvXFwuKHR0ZnxvdGZ8d29mZnxlb3R8c3ZnKSQvO1xuY29uc3QgdHJpbVJlZ2V4ICAgICAgICAgICAgICAgICAgICAgICAgICAgPSAvKF5cXHMrfFxccyskKS9nO1xuY29uc3QgaXNGb250RmFjZVNyY0NvbnRhaW5zUmVtb3RlVXJsUmVnZXggPSAvXlxccyp1cmxcXHMqXFwoXFxzKihcXCd8XFxcImh0dHBzPzopLztcblxuZXhwb3J0IHtcbiAgZXh0cmFjdFVybEZyb21Gb250RmFjZVNyY1JlZ2V4LFxuICB2YWxpZEZvbnRFeHRlbnNpb25SZWdleCxcbiAgdHJpbVJlZ2V4LFxuICBpc0ZvbnRGYWNlU3JjQ29udGFpbnNSZW1vdGVVcmxSZWdleCxcbn07XG4iXX0=
exports.isFontFaceSrcContainsRemoteFontUrlRegex = isFontFaceSrcContainsRemoteFontUrlRegex;
exports.isRemoteFontUrlRegex = isRemoteFontUrlRegex;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9saWIvcmVnZXhlcy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7QUFJQSxJQUFNLGlDQUEwQyxpREFBaEQ7QUFDQSxJQUFNLDBCQUEwQywyQkFBaEQ7QUFDQSxJQUFNLFlBQTBDLGNBQWhEO0FBQ0EsSUFBTSwwQ0FBMEMsb0NBQWhEO0FBQ0EsSUFBTSx1QkFBMEMsK0JBQWhEOztRQUdFLDhCLEdBQUEsOEI7UUFDQSx1QixHQUFBLHVCO1FBQ0EsUyxHQUFBLFM7UUFDQSx1QyxHQUFBLHVDO1FBQ0Esb0IsR0FBQSxvQiIsImZpbGUiOiJyZWdleGVzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBEZWZpbmUgYWxsIHJlZ3VsYXIgZXhwcmVzc2lvbnMgd2lsbCBiZSB1c2VkLlxuICovXG5cbmNvbnN0IGV4dHJhY3RVcmxGcm9tRm9udEZhY2VTcmNSZWdleCAgICAgICAgICA9IC9edXJsXFxzKlxcKFxccypbXFwnXFxcIl0/KGh0dHBzPzpbXlxcKV0qPylbXFwnXFxcIl0/XFxzKlxcKS87XG5jb25zdCB2YWxpZEZvbnRFeHRlbnNpb25SZWdleCAgICAgICAgICAgICAgICAgPSAvXFwuKHR0ZnxvdGZ8d29mZnxlb3R8c3ZnKSQvO1xuY29uc3QgdHJpbVJlZ2V4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID0gLyheXFxzK3xcXHMrJCkvZztcbmNvbnN0IGlzRm9udEZhY2VTcmNDb250YWluc1JlbW90ZUZvbnRVcmxSZWdleCA9IC8oXFxzKnwsfF4pdXJsXFxzKlxcKFxccypbXFwnXFxcIl0/aHR0cHM/Oi87XG5jb25zdCBpc1JlbW90ZUZvbnRVcmxSZWdleCAgICAgICAgICAgICAgICAgICAgPSAvXlxccyp1cmxcXHMqXFwoXFxzKltcXCdcXFwiXT9odHRwcz86LztcblxuZXhwb3J0IHtcbiAgZXh0cmFjdFVybEZyb21Gb250RmFjZVNyY1JlZ2V4LFxuICB2YWxpZEZvbnRFeHRlbnNpb25SZWdleCxcbiAgdHJpbVJlZ2V4LFxuICBpc0ZvbnRGYWNlU3JjQ29udGFpbnNSZW1vdGVGb250VXJsUmVnZXgsXG4gIGlzUmVtb3RlRm9udFVybFJlZ2V4LFxufTtcbiJdfQ==
{
"name": "postcss-font-grabber",
"version": "1.0.5",
"version": "1.0.6",
"description": "Grab remote font in @font-face, download it and update your CSS, end of story.",

@@ -5,0 +5,0 @@ "author": "AaronJan <aaronjan@qq.com>",

@@ -19,3 +19,3 @@ /**

static downloader = new Downloader();
/**

@@ -33,3 +33,3 @@ *

}
/**

@@ -45,3 +45,3 @@ *

}
/**

@@ -54,10 +54,10 @@ *

const result = regexes.extractUrlFromFontFaceSrcRegex.exec(src);
return (
result === null ?
null :
url.parse(result[2])
null :
url.parse(result[1])
);
}
/**

@@ -77,6 +77,6 @@ *

}
return kept;
}
/**

@@ -90,3 +90,3 @@ *

const filename = fontUrlObj.pathname.split('/').pop();
return {

@@ -102,3 +102,3 @@ url : fontUrlObj.href,

}
/**

@@ -114,3 +114,3 @@ *

}
/**

@@ -130,10 +130,22 @@ * Skip Font-Face Postcss object that is:

return false;
} else if (regexes.isFontFaceSrcContainsRemoteUrlRegex.test(decl.value) === false) {
} else if (regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test(decl.value) === false) {
return false;
}
return true;
}
/**
*
* @param urlSrcSources
* @param srcSource
* @returns {Array}
*/
static keepUrlSrcSourceOnly (urlSrcSources, srcSource) {
return regexes.isRemoteFontUrlRegex.test(srcSource) ?
[...urlSrcSources, srcSource] :
urlSrcSources;
}
/**
* Download font file and update output CSS rule correspondingly.

@@ -153,16 +165,20 @@ *

//
// One src could have multiple `url()`, they are separated with `,`.
// One src could have multiple `source`, they are separated with `,`,
// so break it down and filter out those which isn't an `url` source.
//
const srcUrls = decl.value
const urlSrcSources = decl.value
.split(',')
.map(value => value.replace(regexes.trimRegex, ''));
.map(function trim (srcSources) {
return srcSources.replace(regexes.trimRegex, '');
})
.reduce(FontGrabber.keepUrlSrcSourceOnly, []);
//
// Use `srcUrls` to generate Url objects for download.
// Use `urlSrcSources` to generate Url objects for download.
// This will check the validation of font url, and only keep which is
// unique.
const fontFileUrlObjects = srcUrls
const fontFileUrlObjects = urlSrcSources
.map(FontGrabber.generateUrlObjectFromSrc)
.reduce(FontGrabber.keepUniqueAndValidFontFileUrlObject, []);
//

@@ -175,3 +191,3 @@ // If there is no font file needs to be download, end this function

}
//

@@ -188,3 +204,3 @@ // Download font to `saveDirPath` using Url objects **concurrently**

);
return Promise.all(jobs.map(job => job.promise))

@@ -201,3 +217,3 @@ .then(() => {

);
//

@@ -209,3 +225,3 @@ // Replace CSS rule with every font that downloaded.

job.url,
//

@@ -219,3 +235,3 @@ // Replace `\\` to `/` for Windows compatibility.

}
/**

@@ -228,3 +244,3 @@ *

}
/**

@@ -237,3 +253,3 @@ *

}
/**

@@ -251,3 +267,3 @@ * Make handle function for plugin to call with.

let postcssOpts = result.opts;
//

@@ -257,3 +273,3 @@ // If something is missing in the Postcss options, throw an Error.

FontGrabber.validatePostcssOptions(postcssOpts);
//

@@ -263,3 +279,3 @@ // Review options for Font Grabber (This may modify them).

FontGrabber.reviewOptions(opts, postcssOpts);
//

@@ -269,3 +285,3 @@ // Process every Declaration that matchs rule `font-face` concurrently.

let processPromises = [];
const declarationProcessor = (decl) => {

@@ -278,9 +294,9 @@ if (FontGrabber.shouldProcessThisFontFaceDeclaration(decl)) {

};
css.walkAtRules(/font-face/, FontGrabber.iterateCSSRuleWith(declarationProcessor));
return (
processPromises.length === 0 ?
Promise.resolve() :
Promise.all(processPromises)
Promise.resolve() :
Promise.all(processPromises)
);

@@ -287,0 +303,0 @@ };

@@ -5,6 +5,7 @@ /**

const extractUrlFromFontFaceSrcRegex = /^url\s*\(\s*(\"|\')(https?:[^\)]*?)(\"|\')\s*\)/;
const validFontExtensionRegex = /\.(ttf|otf|woff|eot|svg)$/;
const trimRegex = /(^\s+|\s+$)/g;
const isFontFaceSrcContainsRemoteUrlRegex = /^\s*url\s*\(\s*(\'|\"https?:)/;
const extractUrlFromFontFaceSrcRegex = /^url\s*\(\s*[\'\"]?(https?:[^\)]*?)[\'\"]?\s*\)/;
const validFontExtensionRegex = /\.(ttf|otf|woff|eot|svg)$/;
const trimRegex = /(^\s+|\s+$)/g;
const isFontFaceSrcContainsRemoteFontUrlRegex = /(\s*|,|^)url\s*\(\s*[\'\"]?https?:/;
const isRemoteFontUrlRegex = /^\s*url\s*\(\s*[\'\"]?https?:/;

@@ -15,3 +16,4 @@ export {

trimRegex,
isFontFaceSrcContainsRemoteUrlRegex,
isFontFaceSrcContainsRemoteFontUrlRegex,
isRemoteFontUrlRegex,
};

@@ -10,15 +10,15 @@ /**

'http://dummy.com',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url('http://dummy.com')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url('http://dummy.com')")
);
t.is(
'http://dummy.com',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url ('http://dummy.com')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url ('http://dummy.com')")
);
t.is(
'http://dummy.com',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url (\"http://dummy.com\")")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url (\"http://dummy.com\")")
);
t.is(
'https://dummy.com',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url ('https://dummy.com')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url ('https://dummy.com')")
);

@@ -28,16 +28,20 @@

'http://dummy.com/dummy.html',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url('http://dummy.com/dummy.html')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url('http://dummy.com/dummy.html')")
);
t.is(
'http://dummy.com/dummy.html',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url ('http://dummy.com/dummy.html')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url ('http://dummy.com/dummy.html')")
);
t.is(
'http://dummy.com/dummy.html',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url (\"http://dummy.com/dummy.html\")")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url (\"http://dummy.com/dummy.html\")")
);
t.is(
'https://dummy.com/dummy.html',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 2, "url ('https://dummy.com/dummy.html')")
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, "url ('https://dummy.com/dummy.html')")
);
t.is(
'https://fonts.gstatic.com/s/sourcesanspro/v9/toadOcfmlt9b38dHJxOBGLsbIrGiHa6JIepkyt5c0A0.ttf',
execRegex(regexes.extractUrlFromFontFaceSrcRegex, 1, 'url(https://fonts.gstatic.com/s/sourcesanspro/v9/toadOcfmlt9b38dHJxOBGLsbIrGiHa6JIepkyt5c0A0.ttf) format(\'truetype\')')
);
});

@@ -61,8 +65,9 @@

test('isFontFaceSrcContainsRemoteUrlRegex', t => {
t.true(regexes.isFontFaceSrcContainsRemoteUrlRegex.test("url('http://dummy.com')"));
t.true(regexes.isFontFaceSrcContainsRemoteUrlRegex.test("url('https://dummy.com')"));
t.true(regexes.isFontFaceSrcContainsRemoteUrlRegex.test('url("https://dummy.com")'));
t.true(regexes.isFontFaceSrcContainsRemoteUrlRegex.test(' url ("https://dummy.com")'));
t.true(regexes.isFontFaceSrcContainsRemoteUrlRegex.test(' url ( "https://dummy.com" )'));
test('isFontFaceSrcContainsRemoteFontUrlRegex', t => {
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test("url('http://dummy.com')"));
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test("url('https://dummy.com')"));
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test('url("https://dummy.com")'));
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test(' url ("https://dummy.com")'));
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test(' url ( "https://dummy.com" )'));
t.true(regexes.isFontFaceSrcContainsRemoteFontUrlRegex.test('local(\'Source Sans Pro Bold\'), local(\'SourceSansPro-Bold\'), url(https://fonts.gstatic.com/s/sourcesanspro/v9/toadOcfmlt9b38dHJxOBGLsbIrGiHa6JIepkyt5c0A0.ttf) format(\'truetype\')'));
});

@@ -69,0 +74,0 @@

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc