module SQLTree::Node
# Auto-loades files for Node subclasses that reside in the
# node directory, based on the classname.
def self.const_missing(const)
SQLTree.load_default_class_file(SQLTree::Node, const)
end
# The SQLTree::Node::Base class is the superclass for all node
# types that are used to represent SQL queries.
#
# This class implements some helper methods, and enables the
# SQLTree::Node::NodeType['SQL fragment'] construct to parse SQL
# queries.
class Base
def initialize(attributes = {}) # :nodoc:
attributes.each { |key, value| send(:"#{key}=", value) }
end
# Pretty prints this instance for inspection
def inspect
"#{self.class.name}[#{self.to_sql}]"
end
# Quotes a variable name so that it can be safely used within
# SQL queries.
# name:: The name of the variable to quote.
def quote_var(name)
"#{SQLTree.identifier_quote_char}#{name}#{SQLTree.identifier_quote_char}" # TODO: MySQL style variable quoting
end
# Quotes a string so that it can be used safey within an SQL query.
# str:: The string to quote.
def quote_str(str)
"'#{str.gsub("'", "''")}'"
end
# Registers the children and leafs attributes in the subclass
def self.inherited(subclass)
class << subclass; attr_accessor :children, :leafs; end
subclass.children = []
subclass.leafs = []
end
# Adds another leaf to this node class.
def self.leaf(name)
self.leafs << name
self.send(:attr_accessor, name)
end
# Adds another child to this node class.
def self.child(name)
self.children << name
self.send(:attr_accessor, name)
end
# Compares this node with another node, returns true if the nodes are equal.
def ==(other)
other.kind_of?(self.class) && equal_children?(other) && equal_leafs?(other)
end
# Returns true if all children of the current object and the other object are equal.
def equal_children?(other)
self.class.children.all? { |child| send(child) == other.send(child) }
end
# Returns true if all leaf values of the current object and the other object are equal.
def equal_leafs?(other)
self.class.leafs.all? { |leaf| send(leaf) == other.send(leaf) }
end
# Parses an SQL fragment tree from a stream of tokens.
#
# This method should be implemented by each subclass.
# This method should not be called directly, but the
# SQLTree::Node::Subclass#[] should be called to
# parse an SQL fragment provided as a string.
#
# tokens:: the token stream to use for parsing.
def self.parse(tokens)
raise 'Only implemented in subclasses!'
end
def self.parse_list(tokens, item_class = SQLTree::Node::Expression)
items = [item_class.parse(tokens)]
while SQLTree::Token::COMMA === tokens.peek
tokens.consume(SQLTree::Token::COMMA)
items << item_class.parse(tokens)
end
return items
end
# Parses a string, expecting it to be parsable to an instance of
# the current class.
#
# This method will construct a new parser that will tokenize the
# string, and will then present the stream of tokens to the
# self.parse method of the current class.
#
# sql:: The SQL string to parse
# options:: A Hash of options to send to the parser.
def self.[](sql, options = {})
parser = SQLTree::Parser.new(sql, options)
self.parse(parser)
end
end
end