Empact-ar-extensions 0.9.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/ChangeLog +145 -0
- data/README +167 -0
- data/Rakefile +61 -0
- data/config/database.yml +7 -0
- data/config/database.yml.template +7 -0
- data/config/mysql.schema +72 -0
- data/config/postgresql.schema +39 -0
- data/db/migrate/generic_schema.rb +97 -0
- data/db/migrate/mysql_schema.rb +32 -0
- data/db/migrate/oracle_schema.rb +5 -0
- data/db/migrate/version.rb +4 -0
- data/init.rb +31 -0
- data/lib/ar-extensions.rb +5 -0
- data/lib/ar-extensions/adapters/abstract_adapter.rb +146 -0
- data/lib/ar-extensions/adapters/mysql.rb +10 -0
- data/lib/ar-extensions/adapters/oracle.rb +14 -0
- data/lib/ar-extensions/adapters/postgresql.rb +9 -0
- data/lib/ar-extensions/adapters/sqlite.rb +7 -0
- data/lib/ar-extensions/create_and_update.rb +508 -0
- data/lib/ar-extensions/create_and_update/mysql.rb +7 -0
- data/lib/ar-extensions/csv.rb +309 -0
- data/lib/ar-extensions/delete.rb +143 -0
- data/lib/ar-extensions/delete/mysql.rb +3 -0
- data/lib/ar-extensions/extensions.rb +509 -0
- data/lib/ar-extensions/finder_options.rb +275 -0
- data/lib/ar-extensions/finder_options/mysql.rb +6 -0
- data/lib/ar-extensions/finders.rb +96 -0
- data/lib/ar-extensions/foreign_keys.rb +70 -0
- data/lib/ar-extensions/fulltext.rb +62 -0
- data/lib/ar-extensions/fulltext/mysql.rb +44 -0
- data/lib/ar-extensions/import.rb +354 -0
- data/lib/ar-extensions/import/mysql.rb +50 -0
- data/lib/ar-extensions/import/postgresql.rb +0 -0
- data/lib/ar-extensions/import/sqlite.rb +22 -0
- data/lib/ar-extensions/insert_select.rb +178 -0
- data/lib/ar-extensions/insert_select/mysql.rb +7 -0
- data/lib/ar-extensions/synchronize.rb +30 -0
- data/lib/ar-extensions/temporary_table.rb +131 -0
- data/lib/ar-extensions/temporary_table/mysql.rb +3 -0
- data/lib/ar-extensions/union.rb +204 -0
- data/lib/ar-extensions/union/mysql.rb +6 -0
- data/lib/ar-extensions/util/sql_generation.rb +27 -0
- data/lib/ar-extensions/util/support_methods.rb +32 -0
- data/lib/ar-extensions/version.rb +9 -0
- metadata +128 -0
@@ -0,0 +1,50 @@
|
|
1
|
+
module ActiveRecord::Extensions::ConnectionAdapters::MysqlAdapter # :nodoc:
|
2
|
+
|
3
|
+
include ActiveRecord::Extensions::Import::ImportSupport
|
4
|
+
include ActiveRecord::Extensions::Import::OnDuplicateKeyUpdateSupport
|
5
|
+
|
6
|
+
# Returns a generated ON DUPLICATE KEY UPDATE statement given the passed
|
7
|
+
# in +args+.
|
8
|
+
def sql_for_on_duplicate_key_update( table_name, *args ) # :nodoc:
|
9
|
+
sql = ' ON DUPLICATE KEY UPDATE '
|
10
|
+
arg = args.first
|
11
|
+
if arg.is_a?( Array )
|
12
|
+
sql << sql_for_on_duplicate_key_update_as_array( table_name, arg )
|
13
|
+
elsif arg.is_a?( Hash )
|
14
|
+
sql << sql_for_on_duplicate_key_update_as_hash( table_name, arg )
|
15
|
+
elsif arg.is_a?( String )
|
16
|
+
sql << arg
|
17
|
+
else
|
18
|
+
raise ArgumentError.new( "Expected Array or Hash" )
|
19
|
+
end
|
20
|
+
sql
|
21
|
+
end
|
22
|
+
|
23
|
+
def sql_for_on_duplicate_key_update_as_array( table_name, arr ) # :nodoc:
|
24
|
+
results = arr.map do |column|
|
25
|
+
qc = quote_column_name( column )
|
26
|
+
"#{table_name}.#{qc}=VALUES(#{qc})"
|
27
|
+
end
|
28
|
+
results.join( ',' )
|
29
|
+
end
|
30
|
+
|
31
|
+
def sql_for_on_duplicate_key_update_as_hash( table_name, hsh ) # :nodoc:
|
32
|
+
sql = ' ON DUPLICATE KEY UPDATE '
|
33
|
+
results = hsh.map do |column1, column2|
|
34
|
+
qc1 = quote_column_name( column1 )
|
35
|
+
qc2 = quote_column_name( column2 )
|
36
|
+
"#{table_name}.#{qc1}=VALUES( #{qc2} )"
|
37
|
+
end
|
38
|
+
results.join( ',')
|
39
|
+
end
|
40
|
+
|
41
|
+
#return true if the statement is a duplicate key record error
|
42
|
+
def duplicate_key_update_error?(exception)# :nodoc:
|
43
|
+
exception.is_a?(ActiveRecord::StatementInvalid) && exception.to_s.include?('Duplicate entry')
|
44
|
+
end
|
45
|
+
|
46
|
+
end
|
47
|
+
|
48
|
+
ActiveRecord::ConnectionAdapters::MysqlAdapter.class_eval do
|
49
|
+
include ActiveRecord::Extensions::ConnectionAdapters::MysqlAdapter
|
50
|
+
end
|
File without changes
|
@@ -0,0 +1,22 @@
|
|
1
|
+
module ActiveRecord::Extensions::ConnectionAdapters::SQLiteAdapter # :nodoc:
|
2
|
+
include ActiveRecord::Extensions::Import::ImportSupport
|
3
|
+
|
4
|
+
def post_sql_statements( table_name, options )
|
5
|
+
[]
|
6
|
+
end
|
7
|
+
|
8
|
+
def insert_many( sql, values, *args ) # :nodoc:
|
9
|
+
sql2insert = []
|
10
|
+
values.each do |value|
|
11
|
+
sql2insert << "#{sql} #{value};"
|
12
|
+
end
|
13
|
+
|
14
|
+
raw_connection.transaction { |db| db.execute_batch(sql2insert.join("\n")) }
|
15
|
+
number_of_rows_inserted = sql2insert.size
|
16
|
+
end
|
17
|
+
|
18
|
+
end
|
19
|
+
|
20
|
+
ActiveRecord::ConnectionAdapters::SQLiteAdapter.class_eval do
|
21
|
+
include ActiveRecord::Extensions::ConnectionAdapters::SQLiteAdapter
|
22
|
+
end
|
@@ -0,0 +1,178 @@
|
|
1
|
+
# Insert records in bulk with a select statement
|
2
|
+
#
|
3
|
+
# == Parameters
|
4
|
+
# * +options+ - the options used for the finder sql (select)
|
5
|
+
#
|
6
|
+
# === Options
|
7
|
+
# Any valid finder options (options for <tt>ActiveRecord::Base.find(:all)</tt> )such as <tt>:joins</tt>, <tt>:conditions</tt>, <tt>:include</tt>, etc including:
|
8
|
+
# * <tt>:from</tt> - the symbol, class name or class used for the finder SQL (select)
|
9
|
+
# * <tt>:on_duplicate_key_update</tt> - an array of fields to update, or a custom string
|
10
|
+
# * <tt>:select</tt> - An array of fields to select or custom string. The SQL will be sanitized and ? replaced with values as with <tt>:conditions</tt>.
|
11
|
+
# * <tt>:ignore => true </tt> - will ignore any duplicates
|
12
|
+
# * <tt>:into</tt> - Specifies the columns for which data will be inserted. An array of fields to select or custom string.
|
13
|
+
#
|
14
|
+
# == Examples
|
15
|
+
# Create cart items for all books for shopping cart <tt>@cart+
|
16
|
+
# setting the +copies+ field to 1, the +updated_at+ field to Time.now and the +created_at+ field to the database function now()
|
17
|
+
# CartItem.insert_select(:from => :book,
|
18
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
19
|
+
# :into => [:book_id, :shopping_cart_id, :copies, :updated_at, :created_at]})
|
20
|
+
#
|
21
|
+
# GENERATED SQL example (MySQL):
|
22
|
+
# INSERT INTO `cart_items` ( `book_id`, `shopping_cart_id`, `copies`, `updated_at`, `created_at` )
|
23
|
+
# SELECT books.id, '134', 1, '2009-03-02 18:28:25', now() FROM `books`
|
24
|
+
#
|
25
|
+
# A similar example that
|
26
|
+
# * uses the class +Book+ instead of symbol <tt>:book</tt>
|
27
|
+
# * a custom string (instead of an Array) for the <tt>:select</tt> of the +insert_options+
|
28
|
+
# * Updates the +updated_at+ field of all existing cart item. This assumes there is a unique composite index on the +book_id+ and +shopping_cart_id+ fields
|
29
|
+
#
|
30
|
+
# CartItem.insert_select(:from => Book,
|
31
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
32
|
+
# :into => 'cart_items.book_id, shopping_cart_id, copies, updated_at, created_at',
|
33
|
+
# :on_duplicate_key_update => [:updated_at])
|
34
|
+
# GENERATED SQL example (MySQL):
|
35
|
+
# INSERT INTO `cart_items` ( cart_items.book_id, shopping_cart_id, copies, updated_at, created_at )
|
36
|
+
# SELECT books.id, '138', 1, '2009-03-02 18:32:34', now() FROM `books`
|
37
|
+
# ON DUPLICATE KEY UPDATE `cart_items`.`updated_at`=VALUES(`updated_at`)
|
38
|
+
#
|
39
|
+
#
|
40
|
+
# Similar example ignoring duplicates
|
41
|
+
# CartItem.insert_select(:from => :book,
|
42
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
43
|
+
# :into => [:book_id, :shopping_cart_id, :copies, :updated_at, :created_at],
|
44
|
+
# :ignore => true)
|
45
|
+
#
|
46
|
+
# == Developers
|
47
|
+
# * Blythe Dunham http://blythedunham.com
|
48
|
+
#
|
49
|
+
# == Homepage
|
50
|
+
# * Project Site: http://www.continuousthinking.com/tags/arext
|
51
|
+
# * Rubyforge Project: http://rubyforge.org/projects/arext
|
52
|
+
# * Anonymous SVN: svn checkout svn://rubyforge.org/var/svn/arext
|
53
|
+
#
|
54
|
+
|
55
|
+
module ActiveRecord::Extensions::ConnectionAdapters; end
|
56
|
+
|
57
|
+
module ActiveRecord::Extensions::InsertSelectSupport #:nodoc:
|
58
|
+
def supports_insert_select? #:nodoc:
|
59
|
+
true
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
class ActiveRecord::Base
|
64
|
+
|
65
|
+
include ActiveRecord::Extensions::SqlGeneration
|
66
|
+
|
67
|
+
class << self
|
68
|
+
# Insert records in bulk with a select statement
|
69
|
+
#
|
70
|
+
# == Parameters
|
71
|
+
# * +options+ - the options used for the finder sql (select)
|
72
|
+
#
|
73
|
+
# === Options
|
74
|
+
# Any valid finder options (options for <tt>ActiveRecord::Base.find(:all)</tt> )such as <tt>:joins</tt>, <tt>:conditions</tt>, <tt>:include</tt>, etc including:
|
75
|
+
# * <tt>:from</tt> - the symbol, class name or class used for the finder SQL (select)
|
76
|
+
# * <tt>:on_duplicate_key_update</tt> - an array of fields to update, or a custom string
|
77
|
+
# * <tt>:select</tt> - An array of fields to select or custom string. The SQL will be sanitized and ? replaced with values as with <tt>:conditions</tt>.
|
78
|
+
# * <tt>:ignore => true </tt> - will ignore any duplicates
|
79
|
+
# * <tt>:into</tt> - Specifies the columns for which data will be inserted. An array of fields to select or custom string.
|
80
|
+
#
|
81
|
+
# == Examples
|
82
|
+
# Create cart items for all books for shopping cart <tt>@cart+
|
83
|
+
# setting the +copies+ field to 1, the +updated_at+ field to Time.now and the +created_at+ field to the database function now()
|
84
|
+
# CartItem.insert_select(:from => :book,
|
85
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
86
|
+
# :into => [:book_id, :shopping_cart_id, :copies, :updated_at, :created_at]})
|
87
|
+
#
|
88
|
+
# GENERATED SQL example (MySQL):
|
89
|
+
# INSERT INTO `cart_items` ( `book_id`, `shopping_cart_id`, `copies`, `updated_at`, `created_at` )
|
90
|
+
# SELECT books.id, '134', 1, '2009-03-02 18:28:25', now() FROM `books`
|
91
|
+
#
|
92
|
+
# A similar example that
|
93
|
+
# * uses the class +Book+ instead of symbol <tt>:book</tt>
|
94
|
+
# * a custom string (instead of an Array) for the <tt>:select</tt> of the +insert_options+
|
95
|
+
# * Updates the +updated_at+ field of all existing cart item. This assumes there is a unique composite index on the +book_id+ and +shopping_cart_id+ fields
|
96
|
+
#
|
97
|
+
# CartItem.insert_select(:from => Book,
|
98
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
99
|
+
# :into => 'cart_items.book_id, shopping_cart_id, copies, updated_at, created_at',
|
100
|
+
# :on_duplicate_key_update => [:updated_at])
|
101
|
+
# GENERATED SQL example (MySQL):
|
102
|
+
# INSERT INTO `cart_items` ( cart_items.book_id, shopping_cart_id, copies, updated_at, created_at )
|
103
|
+
# SELECT books.id, '138', 1, '2009-03-02 18:32:34', now() FROM `books`
|
104
|
+
# ON DUPLICATE KEY UPDATE `cart_items`.`updated_at`=VALUES(`updated_at`)
|
105
|
+
#
|
106
|
+
#
|
107
|
+
# Similar example ignoring duplicates
|
108
|
+
# CartItem.insert_select(:from => :book,
|
109
|
+
# :select => ['books.id, ?, ?, ?, now()', @cart.to_param, 1, Time.now],
|
110
|
+
# :into => [:book_id, :shopping_cart_id, :copies, :updated_at, :created_at],
|
111
|
+
# :ignore => true)
|
112
|
+
def insert_select(options={})
|
113
|
+
select_obj = options.delete(:from).to_s.classify.constantize
|
114
|
+
#TODO: add batch support for high volume inserts
|
115
|
+
#return insert_select_batch(select_obj, select_options, insert_options) if insert_options[:batch]
|
116
|
+
sql = construct_insert_select_sql(select_obj, options)
|
117
|
+
connection.insert(sql, "#{name} Insert Select #{select_obj}")
|
118
|
+
end
|
119
|
+
|
120
|
+
protected
|
121
|
+
|
122
|
+
def construct_insert_select_sql(select_obj, options)#:nodoc:
|
123
|
+
construct_ar_extension_sql(gather_insert_options(options), valid_insert_select_options) do |sql, into_op|
|
124
|
+
sql << " INTO #{quoted_table_name} "
|
125
|
+
sql << "( #{into_column_sql(options.delete(:into))} ) "
|
126
|
+
|
127
|
+
#sanitize the select sql based on the select object
|
128
|
+
sql << select_obj.send(:finder_sql_to_string, sanitize_select_options(options))
|
129
|
+
sql
|
130
|
+
end
|
131
|
+
end
|
132
|
+
|
133
|
+
# return a list of the column names quoted accordingly
|
134
|
+
# nil => All columns except primary key (auto update)
|
135
|
+
# String => Exact String
|
136
|
+
# Array
|
137
|
+
# needs sanitation ["?, ?", 5, 'test'] => "5, 'test'" or [":date", {:date => Date.today}] => "12-30-2006"]
|
138
|
+
# list of strings or symbols returns quoted values [:start, :name] => `start`, `name` or ['abc'] => `start`
|
139
|
+
def select_column_sql(field_list=nil)#:nodoc:
|
140
|
+
if field_list.kind_of?(String)
|
141
|
+
field_list.dup
|
142
|
+
elsif ((field_list.kind_of?(Array) && field_list.first.is_a?(String)) &&
|
143
|
+
(field_list.last.is_a?(Hash) || field_list.first.include?('?')))
|
144
|
+
sanitize_sql(field_list)
|
145
|
+
else
|
146
|
+
field_list = field_list.blank? ? self.column_names - [self.primary_key] : [field_list].flatten
|
147
|
+
field_list.collect{|field| self.connection.quote_column_name(field.to_s) }.join(", ")
|
148
|
+
end
|
149
|
+
end
|
150
|
+
|
151
|
+
alias_method :into_column_sql, :select_column_sql
|
152
|
+
|
153
|
+
#sanitize the select options for insert select
|
154
|
+
def sanitize_select_options(options)#:nodoc:
|
155
|
+
o = options.dup
|
156
|
+
select = o.delete :select
|
157
|
+
o[:override_select] = select ? select_column_sql(select) : ' * '
|
158
|
+
o
|
159
|
+
end
|
160
|
+
|
161
|
+
|
162
|
+
def valid_insert_select_options#:nodoc:
|
163
|
+
@@valid_insert_select_options ||= [:command, :into_pre, :into_post,
|
164
|
+
:into_keywords, :ignore,
|
165
|
+
:on_duplicate_key_update]
|
166
|
+
end
|
167
|
+
|
168
|
+
#move all the insert options to a seperate map
|
169
|
+
def gather_insert_options(options)#:nodoc:
|
170
|
+
into_options = valid_insert_select_options.inject(:command => 'INSERT') do |map, o|
|
171
|
+
v = options.delete(o)
|
172
|
+
map[o] = v if v
|
173
|
+
map
|
174
|
+
end
|
175
|
+
end
|
176
|
+
|
177
|
+
end
|
178
|
+
end
|
@@ -0,0 +1,7 @@
|
|
1
|
+
#insert select functionality is dependent on finder options and import
|
2
|
+
require 'ar-extensions/finder_options/mysql'
|
3
|
+
require 'ar-extensions/import/mysql'
|
4
|
+
|
5
|
+
ActiveRecord::ConnectionAdapters::MysqlAdapter.class_eval do
|
6
|
+
include ActiveRecord::Extensions::InsertSelectSupport
|
7
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
module ActiveRecord # :nodoc:
|
2
|
+
class Base # :nodoc:
|
3
|
+
|
4
|
+
# Synchronizes the passed in ActiveRecord instances with data
|
5
|
+
# from the database. This is like calling reload
|
6
|
+
# on an individual ActiveRecord instance but it is intended for use on
|
7
|
+
# multiple instances.
|
8
|
+
#
|
9
|
+
# This uses one query for all instance updates and then updates existing
|
10
|
+
# instances rather sending one query for each instance
|
11
|
+
def self.synchronize(instances, key=self.primary_key)
|
12
|
+
return if instances.empty?
|
13
|
+
|
14
|
+
keys = instances.map(&"#{key}".to_sym)
|
15
|
+
klass = instances.first.class
|
16
|
+
fresh_instances = klass.find( :all, :conditions=>{ key=>keys }, :order=>"#{key} ASC" )
|
17
|
+
|
18
|
+
instances.each_with_index do |instance, index|
|
19
|
+
instance.clear_aggregation_cache
|
20
|
+
instance.clear_association_cache
|
21
|
+
instance.instance_variable_set '@attributes', fresh_instances[index].attributes
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
# See ActiveRecord::ConnectionAdapters::AbstractAdapter.synchronize
|
26
|
+
def synchronize(instances, key=ActiveRecord::Base.primary_key)
|
27
|
+
self.class.synchronize(instances, key)
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,131 @@
|
|
1
|
+
module ActiveRecord::Extensions::TemporaryTableSupport # :nodoc:
|
2
|
+
def supports_temporary_tables? #:nodoc:
|
3
|
+
true
|
4
|
+
end
|
5
|
+
end
|
6
|
+
|
7
|
+
class ActiveRecord::Base
|
8
|
+
# Returns true if the underlying database connection supports temporary tables
|
9
|
+
def self.supports_temporary_tables?
|
10
|
+
connection.supports_temporary_tables?
|
11
|
+
rescue NoMethodError
|
12
|
+
false
|
13
|
+
end
|
14
|
+
|
15
|
+
######################################################################
|
16
|
+
# Creates a temporary table given the passed in options hash. The
|
17
|
+
# temporary table is created based off from another table the
|
18
|
+
# current model class. This method returns the constant for the new
|
19
|
+
# new model. This can also be used with block form (see below).
|
20
|
+
#
|
21
|
+
# == Parameters
|
22
|
+
# * options - the options hash used to define the temporary table.
|
23
|
+
#
|
24
|
+
# ==== Options
|
25
|
+
# <tt>:table_name</tt>::the desired name of the temporary table. If not supplied
|
26
|
+
# then a name of "temp_" + the current table_name of the current model
|
27
|
+
# will be used.
|
28
|
+
# <tt>:like</tt>:: the table model you want to base the temporary tables
|
29
|
+
# structure off from. If this is not supplied then the table_name of the
|
30
|
+
# current model will be used.
|
31
|
+
# <tt>:model_name</tt>:: the name of the model you want to use for the temporary
|
32
|
+
# table. This must be compliant with Ruby's naming conventions for
|
33
|
+
# constants. If this is not supplied a rails-generated table name will
|
34
|
+
# be created which is based off from the table_name of the temporary table.
|
35
|
+
# IE: Account.create_temporary_table creates the TempAccount model class
|
36
|
+
#
|
37
|
+
# ==== Example 1, using defaults
|
38
|
+
#
|
39
|
+
# class Project < ActiveRecord::Base
|
40
|
+
# end
|
41
|
+
#
|
42
|
+
# > t = Project.create_temporary_table
|
43
|
+
# > t.class
|
44
|
+
# => "TempProject"
|
45
|
+
# > t.superclass
|
46
|
+
# => Project
|
47
|
+
#
|
48
|
+
# This creates a temporary table named 'temp_projects' and creates a constant
|
49
|
+
# name TempProject. The table structure is copied from the 'projects' table.
|
50
|
+
# TempProject is a subclass of Project as you would expect.
|
51
|
+
#
|
52
|
+
# ==== Example 2, using <tt>:table_name</tt> and <tt>:model options</tt>
|
53
|
+
#
|
54
|
+
# Project.create_temporary_table :table_name => 'my_projects', :model => 'MyProject'
|
55
|
+
#
|
56
|
+
# This creates a temporary table named 'my_projects' and creates a constant named
|
57
|
+
# MyProject. The table structure is copied from the 'projects' table.
|
58
|
+
#
|
59
|
+
# ==== Example 3, using <tt>:like</tt>
|
60
|
+
#
|
61
|
+
# ActiveRecord::Base.create_temporary_table :like => Project
|
62
|
+
#
|
63
|
+
# This is the same as calling Project.create_temporary_table.
|
64
|
+
#
|
65
|
+
# ==== Example 4, using block form
|
66
|
+
#
|
67
|
+
# Project.create_temporary_table do |t|
|
68
|
+
# # ...
|
69
|
+
# end
|
70
|
+
#
|
71
|
+
# Using the block form will automatically drop the temporary table
|
72
|
+
# when the block exits. +t+ which is passed into the block is the temporary
|
73
|
+
# table class. In the above example +t+ equals TempProject. The block form
|
74
|
+
# can be used with all of the available options.
|
75
|
+
#
|
76
|
+
# === See
|
77
|
+
#
|
78
|
+
# * +drop+
|
79
|
+
#
|
80
|
+
######################################################################
|
81
|
+
def self.create_temporary_table(opts={})
|
82
|
+
opts[:temporary] ||= !opts[:permanent]
|
83
|
+
opts[:like] ||= self
|
84
|
+
opts[:table_name] ||= "temp_#{self.table_name}"
|
85
|
+
opts[:model_name] ||= ActiveSupport::Inflector.classify(opts[:table_name])
|
86
|
+
|
87
|
+
if Object.const_defined?(opts[:model_name])
|
88
|
+
raise Exception, "Model #{opts[:model_name]} already exists!"
|
89
|
+
end
|
90
|
+
|
91
|
+
like_table_name = opts[:like].table_name || self.table_name
|
92
|
+
|
93
|
+
connection.execute <<-SQL
|
94
|
+
CREATE #{opts[:temporary] ? 'TEMPORARY' : ''} TABLE #{opts[:table_name]}
|
95
|
+
LIKE #{like_table_name}
|
96
|
+
SQL
|
97
|
+
|
98
|
+
# Sample evaluation:
|
99
|
+
#
|
100
|
+
# class ::TempFood < Food
|
101
|
+
# set_table_name :temp_food
|
102
|
+
#
|
103
|
+
# def self.drop
|
104
|
+
# connection.execute "DROP TABLE temp_foo"
|
105
|
+
# Object.send(:remove_const, self.name.to_sym)
|
106
|
+
# end
|
107
|
+
# end
|
108
|
+
class_eval(<<-RUBY, __FILE__, __LINE__)
|
109
|
+
class ::#{opts[:model_name]} < #{self.name}
|
110
|
+
set_table_name :#{opts[:table_name]}
|
111
|
+
|
112
|
+
def self.drop
|
113
|
+
connection.execute "DROP TABLE #{opts[:table_name]};"
|
114
|
+
Object.send(:remove_const, self.name.to_sym)
|
115
|
+
end
|
116
|
+
end
|
117
|
+
RUBY
|
118
|
+
|
119
|
+
model = Object.const_get(opts[:model_name])
|
120
|
+
|
121
|
+
if block_given?
|
122
|
+
begin
|
123
|
+
yield(model)
|
124
|
+
ensure
|
125
|
+
model.drop
|
126
|
+
end
|
127
|
+
else
|
128
|
+
return model
|
129
|
+
end
|
130
|
+
end
|
131
|
+
end
|
@@ -0,0 +1,204 @@
|
|
1
|
+
module ActiveRecord::Extensions::Union#:nodoc:
|
2
|
+
module UnionSupport #:nodoc:
|
3
|
+
def supports_union? #:nodoc:
|
4
|
+
true
|
5
|
+
end
|
6
|
+
end
|
7
|
+
end
|
8
|
+
|
9
|
+
class ActiveRecord::Base
|
10
|
+
supports_extension :union
|
11
|
+
|
12
|
+
extend ActiveRecord::Extensions::SqlGeneration
|
13
|
+
class << self
|
14
|
+
# Find a union of two or more queries
|
15
|
+
# === Args
|
16
|
+
# Each argument is a hash map of options sent to <tt>:find :all</tt>
|
17
|
+
# including <tt>:conditions</tt>, <tt>:join</tt>, <tt>:group</tt>,
|
18
|
+
# <tt>:having</tt>, and <tt>:limit</tt>
|
19
|
+
#
|
20
|
+
# In addition the following options are accepted
|
21
|
+
# * <tt>:pre_sql</tt> inserts SQL before the SELECT statement of this protion of the +union+
|
22
|
+
# * <tt>:post_sql</tt> appends additional SQL to the end of the statement
|
23
|
+
# * <tt>:override_select</tt> is used to override the <tt>SELECT</tt> clause of eager loaded associations
|
24
|
+
#
|
25
|
+
# == Examples
|
26
|
+
# Find the union of a San Fran zipcode with a Seattle zipcode
|
27
|
+
# union_args1 = {:conditions => ['zip_id = ?', 94010], :select => :phone_number_id}
|
28
|
+
# union_args2 = {:conditions => ['zip_id = ?', 98102], :select => :phone_number_id}
|
29
|
+
# Contact.find_union(union_args1, union_args2, ...)
|
30
|
+
#
|
31
|
+
# SQL> (SELECT phone_number_id FROM contacts WHERE zip_id = 94010) UNION
|
32
|
+
# (SELECT phone_number_id FROM contacts WHERE zip_id = 98102) UNION ...
|
33
|
+
#
|
34
|
+
# == Global Options
|
35
|
+
# To specify global options that apply to the entire union, specify a hash as the
|
36
|
+
# first parameter with a key <tt>:union_options</tt>. Valid options include
|
37
|
+
# <tt>:group</tt>, <tt>:having</tt>, <tt>:order</tt>, and <tt>:limit</tt>
|
38
|
+
#
|
39
|
+
#
|
40
|
+
# Example:
|
41
|
+
# Contact.find_union(:union_options => {:limit => 10, :order => 'created_on'},
|
42
|
+
# union_args1, union_args2, ...)
|
43
|
+
#
|
44
|
+
# SQL> ((select phone_number_id from contacts ...) UNION (select phone_number_id from contacts ...)) order by created_on limit 10
|
45
|
+
#
|
46
|
+
def find_union(*args)
|
47
|
+
supports_union!
|
48
|
+
find_by_sql(find_union_sql(*args))
|
49
|
+
end
|
50
|
+
|
51
|
+
# Count across a union of two or more queries
|
52
|
+
# === Args
|
53
|
+
# * +column_name+ - The column to count. Defaults to all ('*')
|
54
|
+
# * <tt>*args</tt> - Each additional argument is a hash map of options used by <tt>:find :all</tt>
|
55
|
+
# including <tt>:conditions</tt>, <tt>:join</tt>, <tt>:group</tt>,
|
56
|
+
# <tt>:having</tt>, and <tt>:limit</tt>
|
57
|
+
#
|
58
|
+
# In addition the following options are accepted
|
59
|
+
# * <tt>:pre_sql</tt> inserts SQL before the SELECT statement of this protion of the +union+
|
60
|
+
# * <tt>:post_sql</tt> appends additional SQL to the end of the statement
|
61
|
+
# * <tt>:override_select</tt> is used to override the <tt>SELECT</tt> clause of eager loaded associations
|
62
|
+
#
|
63
|
+
# Note that distinct is implied so a record that matches more than one
|
64
|
+
# portion of the union is counted only once.
|
65
|
+
#
|
66
|
+
# == Global Options
|
67
|
+
# To specify global options that apply to the entire union, specify a hash as the
|
68
|
+
# first parameter with a key <tt>:union_options</tt>. Valid options include
|
69
|
+
# <tt>:group</tt>, <tt>:having</tt>, <tt>:order</tt>, and <tt>:limit</tt>
|
70
|
+
#
|
71
|
+
# == Examples
|
72
|
+
# Count the number of people who live in Seattle and San Francisco
|
73
|
+
# Contact.count_union(:phone_number_id,
|
74
|
+
# {:conditions => ['zip_id = ?, 94010]'},
|
75
|
+
# {:conditions => ['zip_id = ?', 98102]})
|
76
|
+
# SQL> select count(*) from ((select phone_number_id from contacts ...) UNION (select phone_number_id from contacts ...)) as counter_tbl;
|
77
|
+
def count_union(column_name, *args)
|
78
|
+
supports_union!
|
79
|
+
count_val = calculate_union(:count, column_name, *args)
|
80
|
+
(args.length == 1 && args.first[:limit] && args.first[:limit].to_i < count_val) ? args.first[:limit].to_i : count_val
|
81
|
+
end
|
82
|
+
|
83
|
+
protected
|
84
|
+
|
85
|
+
#do a union of specified calculation. Only for simple calculations
|
86
|
+
def calculate_union(operation, column_name, *args)#:nodoc:
|
87
|
+
union_options = remove_union_options(args)
|
88
|
+
|
89
|
+
|
90
|
+
if args.length == 1
|
91
|
+
column_name = '*' if column_name == :all
|
92
|
+
calculate(operation, column_name, args.first.update(union_options))
|
93
|
+
|
94
|
+
# For more than one map of options, count off the subquery of all the column_name fields unioned together
|
95
|
+
# For example, if column_name is phone_number_id the generated query is
|
96
|
+
# Contact.calculate_union(:count, :phone_number_id, args)
|
97
|
+
# SQL> select count(*) from
|
98
|
+
# ((select phone_number_id from contacts ...)
|
99
|
+
# UNION
|
100
|
+
# (select phone_number_id from contacts ...)) as counter_tbl
|
101
|
+
else
|
102
|
+
column_name = primary_key if column_name == :all
|
103
|
+
column = column_for column_name
|
104
|
+
column_name = "#{table_name}.#{column_name}" unless column_name.to_s.include?('.')
|
105
|
+
|
106
|
+
group_by = union_options.delete(:group)
|
107
|
+
having = union_options.delete(:having)
|
108
|
+
query_alias = union_options.delete(:query_alias)||"#{operation}_giraffe"
|
109
|
+
|
110
|
+
|
111
|
+
#aggregate_alias should be table_name_id
|
112
|
+
aggregate_alias = column_alias_for('', column_name)
|
113
|
+
#main alias is operation_table_name_id
|
114
|
+
main_aggregate_alias = column_alias_for(operation, column_name)
|
115
|
+
|
116
|
+
sql = "SELECT "
|
117
|
+
sql << (group_by ? "#{group_by}, #{operation}(#{aggregate_alias})" : "#{operation}(*)")
|
118
|
+
sql << " AS #{main_aggregate_alias}"
|
119
|
+
sql << " FROM ("
|
120
|
+
|
121
|
+
#by nature of the union the results will always be distinct, so remove distinct column here
|
122
|
+
sql << args.inject([]){|l, a|
|
123
|
+
calc = "(#{construct_calculation_sql_with_extension('', column_name, a)})"
|
124
|
+
#for group by we need to select the group by column also
|
125
|
+
calc.gsub!(" AS #{aggregate_alias}", " AS #{aggregate_alias}, #{group_by} ") if group_by
|
126
|
+
l << calc
|
127
|
+
}.join(" UNION ")
|
128
|
+
|
129
|
+
add_union_options!(sql, union_options)
|
130
|
+
|
131
|
+
sql << ") as #{query_alias}"
|
132
|
+
|
133
|
+
if group_by
|
134
|
+
#add groupings
|
135
|
+
sql << " GROUP BY #{group_by}"
|
136
|
+
sql << " HAVING #{having}" if having
|
137
|
+
|
138
|
+
calculated_data = connection.select_all(sql)
|
139
|
+
|
140
|
+
calculated_data.inject(ActiveSupport::OrderedHash.new) do |all, row|
|
141
|
+
key = type_cast_calculated_value(row[group_by], column_for(group_by.to_s))
|
142
|
+
value = row[main_aggregate_alias]
|
143
|
+
all << [key, type_cast_calculated_value(value, column_for(column), operation)]
|
144
|
+
end
|
145
|
+
|
146
|
+
else
|
147
|
+
count_by_sql(sql)
|
148
|
+
end
|
149
|
+
end
|
150
|
+
end
|
151
|
+
|
152
|
+
|
153
|
+
#Add Global Union options
|
154
|
+
def add_union_options!(sql, options)#:nodoc:
|
155
|
+
sql << " GROUP BY #{options[:group]} " if options[:group]
|
156
|
+
|
157
|
+
if options[:order] || options[:limit]
|
158
|
+
scope = scope(:find)
|
159
|
+
add_order!(sql, options[:order], scope)
|
160
|
+
add_limit!(sql, options, scope)
|
161
|
+
end
|
162
|
+
sql
|
163
|
+
end
|
164
|
+
|
165
|
+
#Remove the global union options
|
166
|
+
def remove_union_options(args)#:nodoc:
|
167
|
+
args.first.is_a?(Hash) && args.first.has_key?(:union_options) ? (args.shift)[:union_options] : {}
|
168
|
+
end
|
169
|
+
|
170
|
+
def construct_calculation_sql_with_extension(operation, column_name, options)
|
171
|
+
construct_ar_extension_sql(options.merge(:command => '', :keywords => nil, :distinct => nil)) {|sql, o|
|
172
|
+
calc_sql = construct_calculation_sql(operation, column_name, options)
|
173
|
+
|
174
|
+
#this is really gross but prevents us from rewriting construct_calculation_sql
|
175
|
+
calc_sql.gsub!(/^SELECT\s/, "SELECT #{options[:keywords]} ") if options[:keywords]
|
176
|
+
|
177
|
+
sql << calc_sql
|
178
|
+
}
|
179
|
+
end
|
180
|
+
|
181
|
+
# Return the sql for union of the query options specified on the command line
|
182
|
+
# If the first parameter is a map containing :union_options, use these
|
183
|
+
def find_union_sql(*args)#:nodoc:
|
184
|
+
options = remove_union_options(args)
|
185
|
+
|
186
|
+
if args.length == 1
|
187
|
+
return finder_sql_to_string(args.first.update(options))
|
188
|
+
end
|
189
|
+
|
190
|
+
sql = args.inject([]) do |sql_list, union_args|
|
191
|
+
part = union_args.merge(:force_eager_load => true,
|
192
|
+
:override_select => union_args[:select]||"#{quoted_table_name}.*",
|
193
|
+
:select => nil)
|
194
|
+
sql_list << "(#{finder_sql_to_string(part)})"
|
195
|
+
sql_list
|
196
|
+
end.join(" UNION ")
|
197
|
+
|
198
|
+
|
199
|
+
add_union_options!(sql, options)
|
200
|
+
sql
|
201
|
+
end
|
202
|
+
end
|
203
|
+
end
|
204
|
+
|