#!/usr/bin/env node

var path = require('path'),
    fs = require('../lib/less-node/fs'),
    os = require('os'),
    utils = require('../lib/less/utils'),
    Constants = require('../lib/less/constants'),
    errno,
    mkdirp;

try {
    errno = require('errno');
} catch (err) {
    errno = null;
}

var less = require('../lib/less-node'),
    pluginManager = new less.PluginManager(less),
    fileManager = new less.FileManager(),
    plugins = [],
    queuePlugins = [],
    args = process.argv.slice(1),
    silent = false,
    verbose = false,
    options = less.options;

options.plugins = plugins;
options.reUsePluginManager = true;

var sourceMapOptions = {};
var continueProcessing = true;

// Calling process.exit does not flush stdout always. Instead of exiting the process, we set the process' exitCode,
// close all handles and wait for the event loop to exit the process.
// @see https://github.com/nodejs/node/issues/6409
// Unfortunately, node 0.10.x does not support setting process.exitCode, so we need to call reallyExit() explicitly.
// @see https://nodejs.org/api/process.html#process_process_exitcode
// Additionally we also need to make sure that uncaughtExceptions are never swallowed.
// @see https://github.com/less/less.js/issues/2881
// This code can safely be removed if node 0.10.x is not supported anymore.
process.on('exit', function() { process.reallyExit(process.exitCode); });
process.on('uncaughtException', function(err) {
    console.error(err);
    process.exitCode = 1;
});
// This code will still be required because otherwise rejected promises would not be reported to the user
process.on('unhandledRejection', function(err) {
    console.error(err);
    process.exitCode = 1;
});

var checkArgFunc = function(arg, option) {
    if (!option) {
        console.error(arg + ' option requires a parameter');
        continueProcessing = false;
        process.exitCode = 1;
        return false;
    }
    return true;
};

var checkBooleanArg = function(arg) {
    var onOff = /^((on|t|true|y|yes)|(off|f|false|n|no))$/i.exec(arg);
    if (!onOff) {
        console.error(' unable to parse ' + 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(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.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 = 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),
                mapDir = path.dirname(mapFilename),
                outputDir = path.dirname(output);
            // find the path from the map to the output file
            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.sourceMapBasepath === undefined) {
        sourceMapOptions.sourceMapBasepath = input ? path.dirname(input) : process.cwd();
    }

    if (sourceMapOptions.sourceMapRootpath === undefined) {
        var pathToMap = path.dirname((sourceMapFileInline ? output : sourceMapOptions.sourceMapFullFilename) || '.'),
            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 (filepath) {
        var dir = path.dirname(filepath),
            cmd,
            existsSync = fs.existsSync || path.existsSync;
        if (!existsSync(dir)) {
            if (mkdirp === undefined) {
                try {mkdirp = require('mkdirp');}
                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(outputbase + ': ');
    }

    if (!sourceMapFileInline) {
        var writeSourceMap = function(output, onDone) {
            output = output || '';
            var filename = sourceMapOptions.sourceMapFullFilename;
            ensureDirectory(filename);
            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 += err.code + ' ' + err.message;
                    }
                    console.error('lessc: failed to create file ' + filename);
                    console.error(description);
                    process.exitCode = 1;
                } else {
                    less.logger.info('lessc: wrote ' + filename);
                }
                onDone();
            });
        };
    }

    var writeSourceMapIfNeeded = function(output, onDone) {
        if (options.sourceMap && !sourceMapFileInline) {
            writeSourceMap(output, onDone);
        } else {
            onDone();
        }
    };

    var writeOutput = function(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 += err.code + ' ' + err.message;
                    }
                    console.error('lessc: failed to create file ' + output);
                    console.error(description);
                    process.exitCode = 1;
                } else {
                    less.logger.info('lessc: wrote ' + output);
                    onSuccess();
                }
            });
        } else if (!options.depends) {
            process.stdout.write(result.css);
            onSuccess();
        }
    };

    var logDependencies = function(options, result) {
        if (options.depends) {
            var depends = '';
            for (var i = 0; i < result.imports.length; i++) {
                depends += result.imports[i] + ' ';
            }
            console.log(depends);
        }
    };

    var parseLessFile = function (e, data) {
        if (e) {
            console.error('lessc: ' + 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(msg) {
                if (verbose) {
                    console.log(msg);
                }
            },
            warn: function(msg) {
                // do not show warning if the silent option is used
                if (!silent) {
                    console.warn(msg);
                }
            },
            error: function(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 ' + 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 ' + 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 'ie-compat':
                options.ieCompat = true;
                break;
            case 'js':
                options.javascriptEnabled = true;
                break;
            case 'no-js':
                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 'rp':
            case 'rootpath':
                if (checkArgFunc(arg, match[2])) {
                    options.rootpath = match[2].replace(/\\/g, '/');
                }
                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 ' + 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.STRICT_LEGACY;
                    }
                }
                break;
            case 'm':
            case 'math':
                if (checkArgFunc(arg, match[2])) {
                    options.math = match[2];
                }
                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(/^([^=]+)(=(.*))?/),
                    name = splitupArg[1],
                    pluginOptions = splitupArg[3];
                queuePlugins.push({ name: name, options: pluginOptions });
                break;
            default:
                queuePlugins.push({ name: arg, options: match[2], default: true });
                break;
        }
    });

    if (queuePlugins.length > 0) {
        processPluginQueue();
    }
    else {
        render();
    }

})();