diff options
Diffstat (limited to 'lib/arel/relations')
-rw-r--r-- | lib/arel/relations/alias.rb | 19 | ||||
-rw-r--r-- | lib/arel/relations/compound.rb | 16 | ||||
-rw-r--r-- | lib/arel/relations/deletion.rb | 25 | ||||
-rw-r--r-- | lib/arel/relations/grouping.rb | 19 | ||||
-rw-r--r-- | lib/arel/relations/insertion.rb | 28 | ||||
-rw-r--r-- | lib/arel/relations/join.rb | 92 | ||||
-rw-r--r-- | lib/arel/relations/nil.rb | 12 | ||||
-rw-r--r-- | lib/arel/relations/order.rb | 21 | ||||
-rw-r--r-- | lib/arel/relations/projection.rb | 23 | ||||
-rw-r--r-- | lib/arel/relations/relation.rb | 158 | ||||
-rw-r--r-- | lib/arel/relations/selection.rb | 21 | ||||
-rw-r--r-- | lib/arel/relations/skip.rb | 15 | ||||
-rw-r--r-- | lib/arel/relations/table.rb | 48 | ||||
-rw-r--r-- | lib/arel/relations/take.rb | 15 | ||||
-rw-r--r-- | lib/arel/relations/update.rb | 30 | ||||
-rw-r--r-- | lib/arel/relations/writing.rb | 4 |
16 files changed, 546 insertions, 0 deletions
diff --git a/lib/arel/relations/alias.rb b/lib/arel/relations/alias.rb new file mode 100644 index 0000000000..2dab52515f --- /dev/null +++ b/lib/arel/relations/alias.rb @@ -0,0 +1,19 @@ +module Arel + class Alias < Compound + attr_reader :alias + + def initialize(relation, aliaz) + @relation, @alias = relation, aliaz + end + + def alias? + true + end + + def ==(other) + self.class == other.class and + relation == other.relation and + @alias == other.alias + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/compound.rb b/lib/arel/relations/compound.rb new file mode 100644 index 0000000000..4ffac6d1c3 --- /dev/null +++ b/lib/arel/relations/compound.rb @@ -0,0 +1,16 @@ +module Arel + class Compound < Relation + attr_reader :relation + + hash_on :relation + + delegate :joins, :selects, :orders, :groupings, :table_sql, :inserts, :taken, + :skipped, :name, :alias, :aggregation?, :alias?, :prefix_for, :column_for, + :engine, + :to => :relation + + def attributes + relation.attributes.collect { |a| a.bind(self) } + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/deletion.rb b/lib/arel/relations/deletion.rb new file mode 100644 index 0000000000..6c802ba905 --- /dev/null +++ b/lib/arel/relations/deletion.rb @@ -0,0 +1,25 @@ +module Arel + class Deletion < Writing + def initialize(relation) + @relation = relation + end + + def to_sql(formatter = nil) + [ + "DELETE", + "FROM #{table_sql}", + ("WHERE #{selects.collect(&:to_sql).join('\n\tAND ')}" unless selects.blank? ), + ("LIMIT #{taken}" unless taken.blank? ), + ].compact.join("\n") + end + + def call(connection = engine.connection) + connection.delete(to_sql) + end + + def ==(other) + self.class == other.class and + relation == other.relation + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/grouping.rb b/lib/arel/relations/grouping.rb new file mode 100644 index 0000000000..ccca600360 --- /dev/null +++ b/lib/arel/relations/grouping.rb @@ -0,0 +1,19 @@ +module Arel + class Grouping < Compound + attr_reader :expressions, :groupings + + def initialize(relation, *groupings) + @relation, @groupings = relation, groupings.collect { |g| g.bind(relation) } + end + + def ==(other) + self.class == other.class and + relation == other.relation and + groupings == other.groupings + end + + def aggregation? + true + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/insertion.rb b/lib/arel/relations/insertion.rb new file mode 100644 index 0000000000..37e7be8757 --- /dev/null +++ b/lib/arel/relations/insertion.rb @@ -0,0 +1,28 @@ +module Arel + class Insertion < Writing + attr_reader :record + + def initialize(relation, record) + @relation, @record = relation, record.bind(relation) + end + + def to_sql(formatter = nil) + [ + "INSERT", + "INTO #{table_sql}", + "(#{record.keys.collect(&:to_sql).join(', ')})", + "VALUES (#{record.collect { |key, value| key.format(value) }.join(', ')})" + ].join("\n") + end + + def call(connection = engine.connection) + connection.insert(to_sql) + end + + def ==(other) + self.class == other.class and + relation == other.relation and + record == other.record + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/join.rb b/lib/arel/relations/join.rb new file mode 100644 index 0000000000..fb51ea0260 --- /dev/null +++ b/lib/arel/relations/join.rb @@ -0,0 +1,92 @@ +module Arel + class Join < Relation + attr_reader :join_sql, :relation1, :relation2, :predicates + + delegate :engine, :to => :relation1 + + hash_on :relation1 + + def initialize(join_sql, relation1, relation2 = Nil.new, *predicates) + @join_sql, @relation1, @relation2, @predicates = join_sql, relation1, relation2, predicates + end + + def ==(other) + self.class == other.class and + predicates == other.predicates and ( + (relation1 == other.relation1 and relation2 == other.relation2) or + (relation2 == other.relation1 and relation1 == other.relation2) + ) + end + + def attributes + (externalize(relation1).attributes + + externalize(relation2).attributes).collect { |a| a.bind(self) } + end + + def prefix_for(attribute) + if relation1[attribute] && !relation2[attribute] + externalize(relation1).prefix_for(attribute) + elsif relation2[attribute] && !relation1[attribute] + externalize(relation2).prefix_for(attribute) + else + if (attribute % relation1[attribute]).size < (attribute % relation2[attribute]).size + externalize(relation1).prefix_for(attribute) + else + externalize(relation2).prefix_for(attribute) + end + end + end + + def joins + this_join = [ + join_sql, + externalize(relation2).table_sql, + ("ON" unless predicates.blank?), + predicates.collect { |p| p.bind(self).to_sql }.join(' AND ') + ].compact.join(" ") + [relation1.joins, relation2.joins, this_join].compact.join(" ") + end + + def selects + externalize(relation1).selects + externalize(relation2).selects + end + + def table_sql + externalize(relation1).table_sql + end + + private + def externalize(relation) + Externalizer.new(relation) + end + + Externalizer = Struct.new(:relation) do + delegate :engine, :to => :relation + + def table_sql + case + when relation.aggregation? + relation.to_sql(Sql::TableReference.new(engine)) + when relation.alias? + relation.table_sql + ' AS ' + engine.quote_table_name(relation.alias.to_s) + else + relation.table_sql + end + end + + def selects + relation.aggregation?? [] : relation.selects + end + + def attributes + relation.aggregation?? relation.attributes.collect(&:to_attribute) : relation.attributes + end + + def prefix_for(attribute) + if relation[attribute] + relation.alias?? relation.alias : relation.prefix_for(attribute) + end + end + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/nil.rb b/lib/arel/relations/nil.rb new file mode 100644 index 0000000000..3c1d413953 --- /dev/null +++ b/lib/arel/relations/nil.rb @@ -0,0 +1,12 @@ +module Arel + class Nil < Relation + def table_sql; '' end + + def to_s; '' end + + def ==(other) + self.class == other.class + end + end + +end
\ No newline at end of file diff --git a/lib/arel/relations/order.rb b/lib/arel/relations/order.rb new file mode 100644 index 0000000000..91526da02c --- /dev/null +++ b/lib/arel/relations/order.rb @@ -0,0 +1,21 @@ +module Arel + class Order < Compound + attr_reader :ordering + + def initialize(relation, *orders) + ordering = orders.pop + @relation = orders.empty?? relation : Order.new(relation, *orders) + @ordering = ordering.bind(@relation) + end + + def ==(other) + self.class == other.class and + relation == other.relation and + ordering == other.ordering + end + + def orders + relation.orders + [ordering] + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/projection.rb b/lib/arel/relations/projection.rb new file mode 100644 index 0000000000..f09d4f894b --- /dev/null +++ b/lib/arel/relations/projection.rb @@ -0,0 +1,23 @@ +module Arel + class Projection < Compound + attr_reader :projections + + def initialize(relation, *projections) + @relation, @projections = relation, projections + end + + def attributes + projections.collect { |p| p.bind(self) } + end + + def ==(other) + self.class == other.class and + relation == other.relation and + projections == other.projections + end + + def aggregation? + attributes.any?(&:aggregation?) + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/relation.rb b/lib/arel/relations/relation.rb new file mode 100644 index 0000000000..60fb7bd00a --- /dev/null +++ b/lib/arel/relations/relation.rb @@ -0,0 +1,158 @@ +module Arel + class Relation + def session + Session.new + end + + module Enumerable + include ::Enumerable + + def each(&block) + session.read(self).each(&block) + end + + def first + session.read(self).first + end + end + include Enumerable + + module Operations + def join(other = nil) + case other + when String + Join.new(other, self) + when Relation + JoinOperation.new("INNER JOIN", self, other) + else + self + end + end + + def outer_join(other) + JoinOperation.new("LEFT OUTER JOIN", self, other) + end + + def [](index) + case index + when Symbol, String + attribute_for_name(index) + when Attribute, Expression + attribute_for_attribute(index) + end + end + + def select(*predicates) + predicates.all?(&:blank?) ? self : Selection.new(self, *predicates) + end + + def project(*attributes) + attributes.all?(&:blank?) ? self : Projection.new(self, *attributes) + end + + def as(aliaz = nil) + aliaz.blank?? self : Alias.new(self, aliaz) + end + + def order(*attributes) + attributes.all?(&:blank?) ? self : Order.new(self, *attributes) + end + + def take(taken = nil) + taken.blank?? self : Take.new(self, taken) + end + + def skip(skipped = nil) + skipped.blank?? self : Skip.new(self, skipped) + end + + def group(*groupings) + groupings.all?(&:blank?) ? self : Grouping.new(self, *groupings) + end + + module Writes + def insert(record) + session.create Insertion.new(self, record); self + end + + def update(assignments) + session.update Update.new(self, assignments); self + end + + def delete + session.delete Deletion.new(self); self + end + end + include Writes + + JoinOperation = Struct.new(:join_sql, :relation1, :relation2) do + def on(*predicates) + Join.new(join_sql, relation1, relation2, *predicates) + end + end + end + include Operations + + module Externalizable + def aggregation? + false + end + + def alias? + false + end + end + include Externalizable + + def to_sql(formatter = Sql::SelectStatement.new(engine)) + formatter.select [ + "SELECT #{attributes.collect { |a| a.to_sql(Sql::SelectClause.new(engine)) }.join(', ')}", + "FROM #{table_sql}", + (joins unless joins.blank? ), + ("WHERE #{selects.collect { |s| s.to_sql(Sql::WhereClause.new(engine)) }.join("\n\tAND ")}" unless selects.blank? ), + ("ORDER BY #{orders.collect { |o| o.to_sql(Sql::OrderClause.new(engine)) }.join(', ')}" unless orders.blank? ), + ("GROUP BY #{groupings.collect(&:to_sql)}" unless groupings.blank? ), + ("LIMIT #{taken}" unless taken.blank? ), + ("OFFSET #{skipped}" unless skipped.blank? ) + ].compact.join("\n"), self.alias + end + alias_method :to_s, :to_sql + + def inclusion_predicate_sql + "IN" + end + + def call(connection = engine.connection) + connection.select_all(to_sql) + end + + module AttributeAccessors + def attribute_for_name(name) + attributes.detect { |a| a.alias_or_name.to_s == name.to_s } + end + + def attribute_for_attribute(attribute) + attributes.detect { |a| a =~ attribute } + end + end + include AttributeAccessors + + def bind(relation) + self + end + + def format(object) + object.to_sql(Sql::WhereCondition.new(engine)) + end + + def attributes; [] end + def selects; [] end + def orders; [] end + def inserts; [] end + def groupings; [] end + def joins; nil end + def taken; nil end + def skipped; nil end + def alias; nil end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/selection.rb b/lib/arel/relations/selection.rb new file mode 100644 index 0000000000..38a40e1b76 --- /dev/null +++ b/lib/arel/relations/selection.rb @@ -0,0 +1,21 @@ +module Arel + class Selection < Compound + attr_reader :predicate + + def initialize(relation, *predicates) + predicate = predicates.shift + @relation = predicates.empty?? relation : Selection.new(relation, *predicates) + @predicate = predicate.bind(@relation) + end + + def ==(other) + self.class == other.class and + relation == other.relation and + predicate == other.predicate + end + + def selects + relation.selects + [predicate] + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/skip.rb b/lib/arel/relations/skip.rb new file mode 100644 index 0000000000..f17e439ebf --- /dev/null +++ b/lib/arel/relations/skip.rb @@ -0,0 +1,15 @@ +module Arel + class Skip < Compound + attr_reader :skipped + + def initialize(relation, skipped) + @relation, @skipped = relation, skipped + end + + def ==(other) + self.class == other.class and + relation == other.relation and + skipped == other.skipped + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/table.rb b/lib/arel/relations/table.rb new file mode 100644 index 0000000000..cdc03df623 --- /dev/null +++ b/lib/arel/relations/table.rb @@ -0,0 +1,48 @@ +module Arel + class Table < Relation + cattr_accessor :engine + attr_reader :name, :engine + + hash_on :name + + def initialize(name, engine = Table.engine) + @name, @engine = name.to_s, engine + end + + def attributes + @attributes ||= columns.collect do |column| + Attribute.new(self, column.name.to_sym) + end + end + + def prefix_for(attribute) + self[attribute] and name + end + + def column_for(attribute) + self[attribute] and columns.detect { |c| c.name == attribute.name.to_s } + end + + def ==(other) + self.class == other.class and + name == other.name + end + + def columns + @columns ||= engine.columns(name, "#{name} Columns") + end + + def reset + @attributes = @columns = nil + end + + def table_sql + engine.quote_table_name(name) + end + + private + def qualifications + attributes.zip(attributes.collect(&:qualified_name)).to_hash + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/take.rb b/lib/arel/relations/take.rb new file mode 100644 index 0000000000..d2743d7a6e --- /dev/null +++ b/lib/arel/relations/take.rb @@ -0,0 +1,15 @@ +module Arel + class Take < Compound + attr_reader :taken + + def initialize(relation, taken) + @relation, @taken = relation, taken + end + + def ==(other) + self.class == other.class and + relation == other.relation and + taken == other.taken + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/update.rb b/lib/arel/relations/update.rb new file mode 100644 index 0000000000..f1f6776f15 --- /dev/null +++ b/lib/arel/relations/update.rb @@ -0,0 +1,30 @@ +module Arel + class Update < Writing + attr_reader :assignments + + def initialize(relation, assignments) + @relation, @assignments = relation, assignments.bind(relation) + end + + def to_sql(formatter = nil) + [ + "UPDATE #{table_sql} SET", + assignments.collect do |attribute, value| + "#{value.format(attribute)} = #{attribute.format(value)}" + end.join(",\n"), + ("WHERE #{selects.collect(&:to_sql).join('\n\tAND ')}" unless selects.blank? ), + ("LIMIT #{taken}" unless taken.blank? ) + ].join("\n") + end + + def call(connection = engine.connection) + connection.update(to_sql) + end + + def ==(other) + self.class == other.class and + relation == other.relation and + assignments == other.assignments + end + end +end
\ No newline at end of file diff --git a/lib/arel/relations/writing.rb b/lib/arel/relations/writing.rb new file mode 100644 index 0000000000..b871e5a520 --- /dev/null +++ b/lib/arel/relations/writing.rb @@ -0,0 +1,4 @@ +module Arel + class Writing < Compound + end +end
\ No newline at end of file |