| var SqlParser = Editor.Parser = (function () {
    function wordRegexp(words) {
        return new RegExp("^(?:" + words.join("|") + ")$", "i");
    }
    var functions = wordRegexp([
        "abs", "acos", "adddate", "aes_encrypt", "aes_decrypt", "ascii",
        "asin", "atan", "atan2", "avg", "benchmark", "bin", "bit_and",
        "bit_count", "bit_length", "bit_or", "cast", "ceil", "ceiling",
        "char_length", "character_length", "coalesce", "concat", "concat_ws",
        "connection_id", "conv", "convert", "cos", "cot", "count", "curdate",
        "current_date", "current_time", "current_timestamp", "current_user",
        "curtime", "database", "date_add", "date_format", "date_sub",
        "dayname", "dayofmonth", "dayofweek", "dayofyear", "decode", "degrees",
        "des_encrypt", "des_decrypt", "elt", "encode", "encrypt", "exp",
        "export_set", "extract", "field", "find_in_set", "floor", "format",
        "found_rows", "from_days", "from_unixtime", "get_lock", "greatest",
        "group_unique_users", "hex", "ifnull", "inet_aton", "inet_ntoa", "instr",
        "interval", "is_free_lock", "isnull", "last_insert_id", "lcase", "least",
        "left", "length", "ln", "load_file", "locate", "log", "log2", "log10",
        "lower", "lpad", "ltrim", "make_set", "master_pos_wait", "max", "md5",
        "mid", "min", "mod", "monthname", "now", "nullif", "oct", "octet_length",
        "ord", "password", "period_add", "period_diff", "pi", "position",
        "pow", "power", "quarter", "quote", "radians", "rand", "release_lock",
        "repeat", "reverse", "right", "round", "rpad", "rtrim", "sec_to_time",
        "session_user", "sha", "sha1", "sign", "sin", "soundex", "space", "sqrt",
        "std", "stddev", "strcmp", "subdate", "substring", "substring_index",
        "sum", "sysdate", "system_user", "tan", "time_format", "time_to_sec",
        "to_days", "trim", "ucase", "unique_users", "unix_timestamp", "upper",
        "user", "version", "week", "weekday", "yearweek"
    ]);
    var keywords = wordRegexp([
        "alter", "grant", "revoke", "primary", "key", "table", "start",
        "transaction", "select", "update", "insert", "delete", "create", "describe",
        "from", "into", "values", "where", "join", "inner", "left", "natural", "and",
        "or", "in", "not", "xor", "like", "using", "on", "order", "group", "by",
        "asc", "desc", "limit", "offset", "union", "all", "as", "distinct", "set",
        "commit", "rollback", "replace", "view", "database", "separator", "if",
        "exists", "null", "truncate", "status", "show", "lock", "unique"
    ]);
    var types = wordRegexp([
        "bigint", "binary", "bit", "blob", "bool", "char", "character", "date",
        "datetime", "dec", "decimal", "double", "enum", "float", "float4", "float8",
        "int", "int1", "int2", "int3", "int4", "int8", "integer", "long", "longblob",
        "longtext", "mediumblob", "mediumint", "mediumtext", "middleint", "nchar",
        "numeric", "real", "set", "smallint", "text", "time", "timestamp", "tinyblob",
        "tinyint", "tinytext", "varbinary", "varchar", "year"
    ]);
    var operators = wordRegexp([
        ":=", "<", "<=", "==", "<>", ">", ">=", "like", "rlike", "in", "xor", "between"
    ]);
    var operatorChars = /[*+\-<>=&|:\/]/;
    var tokenizeSql = (function () {
        function normal(source, setState) {
            var ch = source.next();
            if (ch == "@" || ch == "$") {
                source.nextWhileMatches(/[\w\d]/);
                return "sql-var";
            }
            else if (ch == "\"" || ch == "'" || ch == "`") {
                setState(inLiteral(ch));
                return null;
            }
            else if (ch == "," || ch == ";") {
                return "sql-separator"
            }
            else if (ch == '-') {
                if (source.peek() == "-") {
                    while (!source.endOfLine()) source.next();
                    return "sql-comment";
                }
                else if (/\d/.test(source.peek())) {
                    source.nextWhileMatches(/\d/);
                    if (source.peek() == '.') {
                        source.next();
                        source.nextWhileMatches(/\d/);
                    }
                    return "sql-number";
                }
                else
                    return "sql-operator";
            }
            else if (operatorChars.test(ch)) {
                source.nextWhileMatches(operatorChars);
                return "sql-operator";
            }
            else if (/\d/.test(ch)) {
                source.nextWhileMatches(/\d/);
                if (source.peek() == '.') {
                    source.next();
                    source.nextWhileMatches(/\d/);
                }
                return "sql-number";
            }
            else if (/[()]/.test(ch)) {
                return "sql-punctuation";
            }
            else {
                source.nextWhileMatches(/[_\w\d]/);
                var word = source.get(), type;
                if (operators.test(word))
                    type = "sql-operator";
                else if (keywords.test(word))
                    type = "sql-keyword";
                else if (functions.test(word))
                    type = "sql-function";
                else if (types.test(word))
                    type = "sql-type";
                else
                    type = "sql-word";
                return {style: type, content: word};
            }
        }
        function inLiteral(quote) {
            return function (source, setState) {
                var escaped = false;
                while (!source.endOfLine()) {
                    var ch = source.next();
                    if (ch == quote && !escaped) {
                        setState(normal);
                        break;
                    }
                    escaped = !escaped && ch == "\\";
                }
                return quote == "`" ? "sql-word" : "sql-literal";
            };
        }
        return function (source, startState) {
            return tokenizer(source, startState || normal);
        };
    })();
    function indentSql(context) {
        return function (nextChars) {
            var firstChar = nextChars && nextChars.charAt(0);
            var closing = context && firstChar == context.type;
            if (!context)
                return 0;
            else if (context.align)
                return context.col - (closing ? context.width : 0);
            else
                return context.indent + (closing ? 0 : indentUnit);
        }
    }
    function parseSql(source) {
        var tokens = tokenizeSql(source);
        var context = null, indent = 0, col = 0;
        function pushContext(type, width, align) {
            context = {prev: context, indent: indent, col: col, type: type, width: width, align: align};
        }
        function popContext() {
            context = context.prev;
        }
        var iter = {
            next: function () {
                var token = tokens.next();
                var type = token.style, content = token.content, width = token.value.length;
                if (content == "\n") {
                    token.indentation = indentSql(context);
                    indent = col = 0;
                    if (context && context.align == null) context.align = false;
                }
                else if (type == "whitespace" && col == 0) {
                    indent = width;
                }
                else if (!context && type != "sql-comment") {
                    pushContext(";", 0, false);
                }
                if (content != "\n") col += width;
                if (type == "sql-punctuation") {
                    if (content == "(")
                        pushContext(")", width);
                    else if (content == ")")
                        popContext();
                }
                else if (type == "sql-separator" && content == ";" && context && !context.prev) {
                    popContext();
                }
                return token;
            },
            copy: function () {
                var _context = context, _indent = indent, _col = col, _tokenState = tokens.state;
                return function (source) {
                    tokens = tokenizeSql(source, _tokenState);
                    context = _context;
                    indent = _indent;
                    col = _col;
                    return iter;
                };
            }
        };
        return iter;
    }
    return {make: parseSql, electricChars: ")"};
})();
 |