class Liquid::Parser
def argument
def argument str = +"" # might be a keyword argument (identifier: expression) if look(:id) && look(:colon, 1) str << consume << consume << ' ' end str << expression str end
def consume(type = nil)
def consume(type = nil) token = @tokens[@p] if type && token[0] != type raise SyntaxError, "Expected #{type} but found #{@tokens[@p].first}" end @p += 1 token[1] end
def consume?(type)
Returns the token's contents if it was consumed
Only consumes the token if it matches the type
def consume?(type) token = @tokens[@p] return false unless token && token[0] == type @p += 1 token[1] end
def expression
def expression token = @tokens[@p] case token[0] when :id str = consume str << variable_lookups when :open_square str = consume str << expression str << consume(:close_square) str << variable_lookups when :string, :number consume when :open_round consume first = expression consume(:dotdot) last = expression consume(:close_round) "(#{first}..#{last})" else raise SyntaxError, "#{token} is not a valid expression" end end
def id?(str)
def id?(str) token = @tokens[@p] return false unless token && token[0] == :id return false unless token[1] == str @p += 1 token[1] end
def initialize(input)
def initialize(input) l = Lexer.new(input) @tokens = l.tokenize @p = 0 # pointer to current location end
def jump(point)
def jump(point) @p = point end
def look(type, ahead = 0)
def look(type, ahead = 0) tok = @tokens[@p + ahead] return false unless tok tok[0] == type end
def variable_lookups
def variable_lookups str = +"" loop do if look(:open_square) str << consume str << expression str << consume(:close_square) elsif look(:dot) str << consume str << consume(:id) else break end end str end