| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662 | #!/usr/bin/env node/* eslint indent: [2, 2, {"SwitchCase": 1}] */"use strict";var path = require('path');var fs = require('../lib/less-node/fs').default;var os = require('os');var utils = require('../lib/less/utils');var Constants = require('../lib/less/constants');var less = require('../lib/less-node').default;var errno;var mkdirp;try {  errno = require('errno');} catch (err) {  errno = null;}var pluginManager = new less.PluginManager(less);var fileManager = new less.FileManager();var plugins = [];var queuePlugins = [];var args = process.argv.slice(1);var silent = false;var verbose = false;var options = less.options;options.plugins = plugins;options.reUsePluginManager = true;var sourceMapOptions = {};var continueProcessing = true;var checkArgFunc = function checkArgFunc(arg, option) {  if (!option) {    console.error("".concat(arg, " option requires a parameter"));    continueProcessing = false;    process.exitCode = 1;    return false;  }  return true;};var checkBooleanArg = function checkBooleanArg(arg) {  var onOff = /^((on|t|true|y|yes)|(off|f|false|n|no))$/i.exec(arg);  if (!onOff) {    console.error(" unable to parse ".concat(arg, " as a boolean. use one of on/t/true/y/yes/off/f/false/n/no"));    continueProcessing = false;    process.exitCode = 1;    return false;  }  return Boolean(onOff[2]);};var parseVariableOption = function parseVariableOption(option, variables) {  var parts = option.split('=', 2);  variables[parts[0]] = parts[1];};var sourceMapFileInline = false;function printUsage() {  less.lesscHelper.printUsage();  pluginManager.Loader.printUsage(plugins);  continueProcessing = false;}function render() {  if (!continueProcessing) {    return;  }  var input = args[1];  if (input && input != '-') {    input = path.resolve(process.cwd(), input);  }  var output = args[2];  var outputbase = args[2];  if (output) {    output = path.resolve(process.cwd(), output);  }  if (options.disablePluginRule && queuePlugins.length > 0) {    console.error('--plugin and --disable-plugin-rule may not be used at the same time');    process.exitCode = 1;    return;  }  if (options.sourceMap) {    sourceMapOptions.sourceMapInputFilename = input;    if (!sourceMapOptions.sourceMapFullFilename) {      if (!output && !sourceMapFileInline) {        console.error('the sourcemap option only has an optional filename if the css filename is given');        console.error('consider adding --source-map-map-inline which embeds the sourcemap into the css');        process.exitCode = 1;        return;      } // its in the same directory, so always just the basename      if (output) {        sourceMapOptions.sourceMapOutputFilename = path.basename(output);        sourceMapOptions.sourceMapFullFilename = "".concat(output, ".map");      } // its in the same directory, so always just the basename      if ('sourceMapFullFilename' in sourceMapOptions) {        sourceMapOptions.sourceMapFilename = path.basename(sourceMapOptions.sourceMapFullFilename);      }    } else if (options.sourceMap && !sourceMapFileInline) {      var mapFilename = path.resolve(process.cwd(), sourceMapOptions.sourceMapFullFilename);      var mapDir = path.dirname(mapFilename);      var outputDir = path.dirname(output); // find the path from the map to the output file      // eslint-disable-next-line max-len      sourceMapOptions.sourceMapOutputFilename = path.join(path.relative(mapDir, outputDir), path.basename(output)); // make the sourcemap filename point to the sourcemap relative to the css file output directory      sourceMapOptions.sourceMapFilename = path.join(path.relative(outputDir, mapDir), path.basename(sourceMapOptions.sourceMapFullFilename));    }    if (sourceMapOptions.sourceMapURL && sourceMapOptions.disableSourcemapAnnotation) {      console.error('You cannot provide flag --source-map-url with --source-map-no-annotation.');      console.error('Please remove one of those flags.');      process.exitcode = 1;      return;    }  }  if (sourceMapOptions.sourceMapBasepath === undefined) {    sourceMapOptions.sourceMapBasepath = input ? path.dirname(input) : process.cwd();  }  if (sourceMapOptions.sourceMapRootpath === undefined) {    var pathToMap = path.dirname((sourceMapFileInline ? output : sourceMapOptions.sourceMapFullFilename) || '.');    var pathToInput = path.dirname(sourceMapOptions.sourceMapInputFilename || '.');    sourceMapOptions.sourceMapRootpath = path.relative(pathToMap, pathToInput);  }  if (!input) {    console.error('lessc: no input files');    console.error('');    printUsage();    process.exitCode = 1;    return;  }  var ensureDirectory = function ensureDirectory(filepath) {    var dir = path.dirname(filepath);    var cmd;    var existsSync = fs.existsSync || path.existsSync;    if (!existsSync(dir)) {      if (mkdirp === undefined) {        try {          mkdirp = require('make-dir');        } catch (e) {          mkdirp = null;        }      }      cmd = mkdirp && mkdirp.sync || fs.mkdirSync;      cmd(dir);    }  };  if (options.depends) {    if (!outputbase) {      console.error('option --depends requires an output path to be specified');      process.exitCode = 1;      return;    }    process.stdout.write("".concat(outputbase, ": "));  }  if (!sourceMapFileInline) {    var writeSourceMap = function writeSourceMap() {      var output = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : '';      var onDone = arguments.length > 1 ? arguments[1] : undefined;      var filename = sourceMapOptions.sourceMapFullFilename;      ensureDirectory(filename);      // To fix https://github.com/less/less.js/issues/3646      output = output.toString();            fs.writeFile(filename, output, 'utf8', function (err) {        if (err) {          var description = 'Error: ';          if (errno && errno.errno[err.errno]) {            description += errno.errno[err.errno].description;          } else {            description += "".concat(err.code, " ").concat(err.message);          }          console.error("lessc: failed to create file ".concat(filename));          console.error(description);          process.exitCode = 1;        } else {          less.logger.info("lessc: wrote ".concat(filename));        }        onDone();      });    };  }  var writeSourceMapIfNeeded = function writeSourceMapIfNeeded(output, onDone) {    if (options.sourceMap && !sourceMapFileInline) {      writeSourceMap(output, onDone);    } else {      onDone();    }  };  var writeOutput = function writeOutput(output, result, onSuccess) {    if (options.depends) {      onSuccess();    } else if (output) {      ensureDirectory(output);      fs.writeFile(output, result.css, {        encoding: 'utf8'      }, function (err) {        if (err) {          var description = 'Error: ';          if (errno && errno.errno[err.errno]) {            description += errno.errno[err.errno].description;          } else {            description += "".concat(err.code, " ").concat(err.message);          }          console.error("lessc: failed to create file ".concat(output));          console.error(description);          process.exitCode = 1;        } else {          less.logger.info("lessc: wrote ".concat(output));          onSuccess();        }      });    } else if (!options.depends) {      process.stdout.write(result.css);      onSuccess();    }  };  var logDependencies = function logDependencies(options, result) {    if (options.depends) {      var depends = '';      for (var i = 0; i < result.imports.length; i++) {        depends += "".concat(result.imports[i], " ");      }      console.log(depends);    }  };  var parseLessFile = function parseLessFile(e, data) {    if (e) {      console.error("lessc: ".concat(e.message));      process.exitCode = 1;      return;    }    data = data.replace(/^\uFEFF/, '');    options.paths = [path.dirname(input)].concat(options.paths);    options.filename = input;    if (options.lint) {      options.sourceMap = false;    }    sourceMapOptions.sourceMapFileInline = sourceMapFileInline;    if (options.sourceMap) {      options.sourceMap = sourceMapOptions;    }    less.logger.addListener({      info: function info(msg) {        if (verbose) {          console.log(msg);        }      },      warn: function warn(msg) {        // do not show warning if the silent option is used        if (!silent) {          console.warn(msg);        }      },      error: function error(msg) {        console.error(msg);      }    });    less.render(data, options).then(function (result) {      if (!options.lint) {        writeOutput(output, result, function () {          writeSourceMapIfNeeded(result.map, function () {            logDependencies(options, result);          });        });      }    }, function (err) {      if (!options.silent) {        console.error(err.toString({          stylize: options.color && less.lesscHelper.stylize        }));      }      process.exitCode = 1;    });  };  if (input != '-') {    fs.readFile(input, 'utf8', parseLessFile);  } else {    process.stdin.resume();    process.stdin.setEncoding('utf8');    var buffer = '';    process.stdin.on('data', function (data) {      buffer += data;    });    process.stdin.on('end', function () {      parseLessFile(false, buffer);    });  }}function processPluginQueue() {  var x = 0;  function pluginError(name) {    console.error("Unable to load plugin ".concat(name, " please make sure that it is installed under or at the same level as less"));    process.exitCode = 1;  }  function pluginFinished(plugin) {    x++;    plugins.push(plugin);    if (x === queuePlugins.length) {      render();    }  }  queuePlugins.forEach(function (queue) {    var context = utils.clone(options);    pluginManager.Loader.loadPlugin(queue.name, process.cwd(), context, less.environment, fileManager).then(function (data) {      pluginFinished({        fileContent: data.contents,        filename: data.filename,        options: queue.options      });    }).catch(function () {      pluginError(queue.name);    });  });} // self executing function so we can return(function () {  args = args.filter(function (arg) {    var match;    match = arg.match(/^-I(.+)$/);    if (match) {      options.paths.push(match[1]);      return false;    }    match = arg.match(/^--?([a-z][0-9a-z-]*)(?:=(.*))?$/i);    if (match) {      arg = match[1];    } else {      return arg;    }    switch (arg) {      case 'v':      case 'version':        console.log("lessc ".concat(less.version.join('.'), " (Less Compiler) [JavaScript]"));        continueProcessing = false;        break;      case 'verbose':        verbose = true;        break;      case 's':      case 'silent':        silent = true;        break;      case 'l':      case 'lint':        options.lint = true;        break;      case 'strict-imports':        options.strictImports = true;        break;      case 'h':      case 'help':        printUsage();        break;      case 'x':      case 'compress':        options.compress = true;        break;      case 'insecure':        options.insecure = true;        break;      case 'M':      case 'depends':        options.depends = true;        break;      case 'max-line-len':        if (checkArgFunc(arg, match[2])) {          options.maxLineLen = parseInt(match[2], 10);          if (options.maxLineLen <= 0) {            options.maxLineLen = -1;          }        }        break;      case 'no-color':        options.color = false;        break;      case 'js':        options.javascriptEnabled = true;        break;      case 'no-js':        // eslint-disable-next-line max-len        console.error('The "--no-js" argument is deprecated, as inline JavaScript is disabled by default. Use "--js" to enable inline JavaScript (not recommended).');        break;      case 'include-path':        if (checkArgFunc(arg, match[2])) {          // ; supported on windows.          // : supported on windows and linux, excluding a drive letter like C:\ so C:\file:D:\file parses to 2          options.paths = match[2].split(os.type().match(/Windows/) ? /:(?!\\)|;/ : ':').map(function (p) {            if (p) {              return path.resolve(process.cwd(), p);            }          });        }        break;      case 'line-numbers':        if (checkArgFunc(arg, match[2])) {          options.dumpLineNumbers = match[2];        }        break;      case 'source-map':        options.sourceMap = true;        if (match[2]) {          sourceMapOptions.sourceMapFullFilename = match[2];        }        break;      case 'source-map-rootpath':        if (checkArgFunc(arg, match[2])) {          sourceMapOptions.sourceMapRootpath = match[2];        }        break;      case 'source-map-basepath':        if (checkArgFunc(arg, match[2])) {          sourceMapOptions.sourceMapBasepath = match[2];        }        break;      case 'source-map-inline':      case 'source-map-map-inline':        sourceMapFileInline = true;        options.sourceMap = true;        break;      case 'source-map-include-source':      case 'source-map-less-inline':        sourceMapOptions.outputSourceFiles = true;        break;      case 'source-map-url':        if (checkArgFunc(arg, match[2])) {          sourceMapOptions.sourceMapURL = match[2];        }        break;      case 'source-map-no-annotation':        sourceMapOptions.disableSourcemapAnnotation = true;        break;      case 'rp':      case 'rootpath':        if (checkArgFunc(arg, match[2])) {          options.rootpath = match[2].replace(/\\/g, '/');        }        break;            case 'ie-compat':        console.warn('The --ie-compat option is deprecated, as it has no effect on compilation.');        break;      case 'relative-urls':        console.warn('The --relative-urls option has been deprecated. Use --rewrite-urls=all.');        options.rewriteUrls = Constants.RewriteUrls.ALL;        break;      case 'ru':      case 'rewrite-urls':        var m = match[2];        if (m) {          if (m === 'local') {            options.rewriteUrls = Constants.RewriteUrls.LOCAL;          } else if (m === 'off') {            options.rewriteUrls = Constants.RewriteUrls.OFF;          } else if (m === 'all') {            options.rewriteUrls = Constants.RewriteUrls.ALL;          } else {            console.error("Unknown rewrite-urls argument ".concat(m));            continueProcessing = false;            process.exitCode = 1;          }        } else {          options.rewriteUrls = Constants.RewriteUrls.ALL;        }        break;      case 'sm':      case 'strict-math':        console.warn('The --strict-math option has been deprecated. Use --math=strict.');        if (checkArgFunc(arg, match[2])) {          if (checkBooleanArg(match[2])) {            options.math = Constants.Math.PARENS;          }        }        break;      case 'm':      case 'math':        var m = match[2];        if (checkArgFunc(arg, m)) {          if (m === 'always') {            console.warn('--math=always is deprecated and will be removed in the future.');            options.math = Constants.Math.ALWAYS;          } else if (m === 'parens-division') {            options.math = Constants.Math.PARENS_DIVISION;          } else if (m === 'parens' || m === 'strict') {            options.math = Constants.Math.PARENS;          } else if (m === 'strict-legacy') {            console.warn('--math=strict-legacy has been removed. Defaulting to --math=strict');            options.math = Constants.Math.PARENS;          }        }        break;      case 'su':      case 'strict-units':        if (checkArgFunc(arg, match[2])) {          options.strictUnits = checkBooleanArg(match[2]);        }        break;      case 'global-var':        if (checkArgFunc(arg, match[2])) {          if (!options.globalVars) {            options.globalVars = {};          }          parseVariableOption(match[2], options.globalVars);        }        break;      case 'modify-var':        if (checkArgFunc(arg, match[2])) {          if (!options.modifyVars) {            options.modifyVars = {};          }          parseVariableOption(match[2], options.modifyVars);        }        break;      case 'url-args':        if (checkArgFunc(arg, match[2])) {          options.urlArgs = match[2];        }        break;      case 'plugin':        var splitupArg = match[2].match(/^([^=]+)(=(.*))?/);        var name = splitupArg[1];        var pluginOptions = splitupArg[3];        queuePlugins.push({          name: name,          options: pluginOptions        });        break;      case 'disable-plugin-rule':        options.disablePluginRule = true;        break;        default:        queuePlugins.push({          name: arg,          options: match[2],          default: true        });        break;    }  });  if (queuePlugins.length > 0) {    processPluginQueue();  } else {    render();  }})();
 |