-- Copyright 2006-2011 Mitchell mitchellcaladbolg.net. See LICENSE. -- CoffeeScript LPeg lexer. local l = lexer local token, word_match = l.token, l.word_match local P, S = l.lpeg.P, l.lpeg.S module(...) -- Whitespace. local ws = token(l.WHITESPACE, l.space^1) -- Comments. local block_comment = '###' * (l.any - '###')^0 * P('###')^-1 local line_comment = '#' * l.nonnewline_esc^0 local comment = token(l.COMMENT, block_comment + line_comment) -- Strings. local sq_str = l.delimited_range("'", '\\', true) local dq_str = l.delimited_range('"', '\\', true) local string = token(l.STRING, sq_str + dq_str) + P(function(input, index) if index == 1 then return index end local i = index while input:sub(i - 1, i - 1):match('[ \t\r\n\f]') do i = i - 1 end return input:sub(i - 1, i - 1):match('[+%-*%%^!=&|?:;,()%[%]{}]') and index end) * token(l.REGEX, l.delimited_range('/', '\\', nil, nil, '\n') * S('igm')^0) -- Numbers. local number = token(l.NUMBER, l.float + l.integer) -- Keywords. local keyword = token(l.KEYWORD, word_match { 'all', 'and', 'bind', 'break', 'by', 'case', 'catch', 'class', 'const', 'continue', 'default', 'delete', 'do', 'each', 'else', 'enum', 'export', 'extends', 'false', 'for', 'finally', 'function', 'if', 'import', 'in', 'instanceof', 'is', 'isnt', 'let', 'loop', 'native', 'new', 'no', 'not', 'of', 'off', 'on', 'or', 'return', 'super', 'switch', 'then', 'this', 'throw', 'true', 'try', 'typeof', 'unless', 'until', 'var', 'void', 'with', 'when', 'while', 'yes' }) -- Fields: object properties and methods. local field = token(l.FUNCTION, '.' * (S('_$') + l.alpha) * (S('_$') + l.alnum)^0) -- Identifiers. local identifier = token(l.IDENTIFIER, l.word) -- Operators. local operator = token(l.OPERATOR, S('+-/*%<>!=^&|?~:;.()[]{}')) _rules = { { 'whitespace', ws }, { 'keyword', keyword }, { 'field', field }, { 'identifier', identifier }, { 'comment', comment }, { 'number', number }, { 'string', string }, { 'operator', operator }, { 'any_char', l.any_char }, }