upsert 2.9.10-java
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.
- checksums.yaml +7 -0
- data/.gitignore +19 -0
- data/.ruby-version +1 -0
- data/.standard.yml +1 -0
- data/.travis.yml +63 -0
- data/.yardopts +2 -0
- data/CHANGELOG +265 -0
- data/Gemfile +20 -0
- data/LICENSE +24 -0
- data/README.md +411 -0
- data/Rakefile +54 -0
- data/lib/upsert.rb +284 -0
- data/lib/upsert/active_record_upsert.rb +12 -0
- data/lib/upsert/binary.rb +8 -0
- data/lib/upsert/column_definition.rb +79 -0
- data/lib/upsert/column_definition/mysql.rb +24 -0
- data/lib/upsert/column_definition/postgresql.rb +66 -0
- data/lib/upsert/column_definition/sqlite3.rb +34 -0
- data/lib/upsert/connection.rb +37 -0
- data/lib/upsert/connection/Java_ComMysqlJdbc_JDBC4Connection.rb +31 -0
- data/lib/upsert/connection/Java_OrgPostgresqlJdbc_PgConnection.rb +33 -0
- data/lib/upsert/connection/Java_OrgSqlite_Conn.rb +17 -0
- data/lib/upsert/connection/Mysql2_Client.rb +76 -0
- data/lib/upsert/connection/PG_Connection.rb +35 -0
- data/lib/upsert/connection/SQLite3_Database.rb +28 -0
- data/lib/upsert/connection/jdbc.rb +105 -0
- data/lib/upsert/connection/postgresql.rb +24 -0
- data/lib/upsert/connection/sqlite3.rb +19 -0
- data/lib/upsert/merge_function.rb +73 -0
- data/lib/upsert/merge_function/Java_ComMysqlJdbc_JDBC4Connection.rb +42 -0
- data/lib/upsert/merge_function/Java_OrgPostgresqlJdbc_PgConnection.rb +27 -0
- data/lib/upsert/merge_function/Java_OrgSqlite_Conn.rb +10 -0
- data/lib/upsert/merge_function/Mysql2_Client.rb +36 -0
- data/lib/upsert/merge_function/PG_Connection.rb +26 -0
- data/lib/upsert/merge_function/SQLite3_Database.rb +10 -0
- data/lib/upsert/merge_function/mysql.rb +66 -0
- data/lib/upsert/merge_function/postgresql.rb +365 -0
- data/lib/upsert/merge_function/sqlite3.rb +43 -0
- data/lib/upsert/row.rb +59 -0
- data/lib/upsert/version.rb +3 -0
- data/spec/active_record_upsert_spec.rb +26 -0
- data/spec/binary_spec.rb +21 -0
- data/spec/correctness_spec.rb +190 -0
- data/spec/database_functions_spec.rb +106 -0
- data/spec/database_spec.rb +121 -0
- data/spec/hstore_spec.rb +249 -0
- data/spec/jruby_spec.rb +9 -0
- data/spec/logger_spec.rb +52 -0
- data/spec/misc/get_postgres_reserved_words.rb +12 -0
- data/spec/misc/mysql_reserved.txt +226 -0
- data/spec/misc/pg_reserved.txt +742 -0
- data/spec/multibyte_spec.rb +27 -0
- data/spec/postgresql_spec.rb +94 -0
- data/spec/precision_spec.rb +11 -0
- data/spec/reserved_words_spec.rb +50 -0
- data/spec/sequel_spec.rb +57 -0
- data/spec/spec_helper.rb +417 -0
- data/spec/speed_spec.rb +44 -0
- data/spec/threaded_spec.rb +57 -0
- data/spec/timezones_spec.rb +58 -0
- data/spec/type_safety_spec.rb +12 -0
- data/travis/install_postgres.sh +18 -0
- data/travis/run_docker_db.sh +20 -0
- data/travis/tune_mysql.sh +7 -0
- data/upsert-java.gemspec +14 -0
- data/upsert.gemspec +13 -0
- data/upsert.gemspec.common +106 -0
- metadata +373 -0
data/Rakefile
ADDED
@@ -0,0 +1,54 @@
|
|
1
|
+
#!/usr/bin/env rake
|
2
|
+
require "bundler/gem_helper"
|
3
|
+
case RUBY_PLATFORM
|
4
|
+
when "java"
|
5
|
+
Bundler::GemHelper.install_tasks name: "upsert-java"
|
6
|
+
else
|
7
|
+
Bundler::GemHelper.install_tasks name: "upsert"
|
8
|
+
end
|
9
|
+
require "rspec/core/rake_task"
|
10
|
+
|
11
|
+
RSpec::Core::RakeTask.new(:spec) do |t|
|
12
|
+
t.rspec_opts = "--format documentation"
|
13
|
+
end
|
14
|
+
|
15
|
+
task :default => :spec
|
16
|
+
|
17
|
+
task :rspec_all_databases do
|
18
|
+
results = {}
|
19
|
+
|
20
|
+
dbs = %w{ postgresql mysql sqlite3 }
|
21
|
+
if ENV['DB']
|
22
|
+
dbs = ENV['DB'].split(',')
|
23
|
+
end
|
24
|
+
|
25
|
+
dbs.each do |db|
|
26
|
+
puts
|
27
|
+
puts '#'*50
|
28
|
+
puts "# Running specs against #{db}"
|
29
|
+
puts '#'*50
|
30
|
+
puts
|
31
|
+
|
32
|
+
if RUBY_VERSION >= '1.9'
|
33
|
+
pid = spawn({'DB' => db}, 'rspec', '--format', 'documentation', File.expand_path('../spec', __FILE__))
|
34
|
+
Process.waitpid pid
|
35
|
+
results[db] = $?.success?
|
36
|
+
else
|
37
|
+
exec({'DB' => db}, 'rspec', '--format', 'documentation', File.expand_path('../spec', __FILE__))
|
38
|
+
end
|
39
|
+
|
40
|
+
end
|
41
|
+
puts results.inspect
|
42
|
+
end
|
43
|
+
|
44
|
+
task :n, :from, :to do |t, args|
|
45
|
+
Dir[File.expand_path("../lib/upsert/**/#{args.from}.*", __FILE__)].each do |path|
|
46
|
+
dir = File.dirname(path)
|
47
|
+
File.open("#{dir}/#{args.to}.rb", 'w') do |f|
|
48
|
+
f.write File.read(path).gsub(args.from, args.to)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
52
|
+
|
53
|
+
require 'yard'
|
54
|
+
YARD::Rake::YardocTask.new
|
data/lib/upsert.rb
ADDED
@@ -0,0 +1,284 @@
|
|
1
|
+
require 'bigdecimal'
|
2
|
+
require 'thread'
|
3
|
+
require 'logger'
|
4
|
+
|
5
|
+
require 'upsert/version'
|
6
|
+
require 'upsert/binary'
|
7
|
+
require 'upsert/connection'
|
8
|
+
require 'upsert/merge_function'
|
9
|
+
require 'upsert/column_definition'
|
10
|
+
require 'upsert/row'
|
11
|
+
|
12
|
+
class Upsert
|
13
|
+
class << self
|
14
|
+
# What logger to use.
|
15
|
+
# @return [#info,#warn,#debug]
|
16
|
+
attr_writer :logger
|
17
|
+
MUTEX_FOR_PERFORM = Mutex.new
|
18
|
+
|
19
|
+
# The current logger
|
20
|
+
# @return [#info,#warn,#debug]
|
21
|
+
def logger
|
22
|
+
@logger || MUTEX_FOR_PERFORM.synchronize do
|
23
|
+
@logger ||= if defined?(::Rails) and (rails_logger = ::Rails.logger)
|
24
|
+
rails_logger
|
25
|
+
elsif defined?(::ActiveRecord) and ::ActiveRecord.const_defined?(:Base) and (ar_logger = ::ActiveRecord::Base.logger)
|
26
|
+
ar_logger
|
27
|
+
else
|
28
|
+
my_logger = Logger.new $stderr
|
29
|
+
case ENV['UPSERT_DEBUG']
|
30
|
+
when 'true'
|
31
|
+
my_logger.level = Logger::DEBUG
|
32
|
+
when 'false'
|
33
|
+
my_logger.level = Logger::INFO
|
34
|
+
end
|
35
|
+
my_logger
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
def mutex_for_row(upsert, row)
|
41
|
+
retrieve_mutex(upsert.table_name, row.selector.keys)
|
42
|
+
end
|
43
|
+
|
44
|
+
def mutex_for_function(upsert, row)
|
45
|
+
retrieve_mutex(upsert.table_name, row.selector.keys, row.setter.keys)
|
46
|
+
end
|
47
|
+
|
48
|
+
# TODO: Rewrite this to use the thread_safe gem, perhaps?
|
49
|
+
def retrieve_mutex(*args)
|
50
|
+
# ||= isn't an atomic operation
|
51
|
+
MUTEX_FOR_PERFORM.synchronize do
|
52
|
+
@mutex_cache ||= {}
|
53
|
+
end
|
54
|
+
|
55
|
+
@mutex_cache.fetch(args.flatten.join('::')) do |k|
|
56
|
+
MUTEX_FOR_PERFORM.synchronize do
|
57
|
+
# We still need the ||= because this block could have
|
58
|
+
# theoretically been entered simultaneously by two threads
|
59
|
+
# but the actual assignment is protected by the mutex
|
60
|
+
@mutex_cache[k] ||= Mutex.new
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
# @param [Mysql2::Client,Sqlite3::Database,PG::Connection,#metal] connection A supported database connection.
|
66
|
+
#
|
67
|
+
# Clear any database functions that may have been created.
|
68
|
+
#
|
69
|
+
# Currently only applies to PostgreSQL.
|
70
|
+
def clear_database_functions(connection)
|
71
|
+
dummy = new(connection, :dummy)
|
72
|
+
dummy.clear_database_functions
|
73
|
+
end
|
74
|
+
|
75
|
+
# @param [String] v A string containing binary data that should be inserted/escaped as such.
|
76
|
+
#
|
77
|
+
# @return [Upsert::Binary]
|
78
|
+
def binary(v)
|
79
|
+
Binary.new v
|
80
|
+
end
|
81
|
+
|
82
|
+
# More efficient way of upserting multiple rows at once.
|
83
|
+
#
|
84
|
+
# @param [Mysql2::Client,Sqlite3::Database,PG::Connection,#metal] connection A supported database connection.
|
85
|
+
# @param [String,Symbol] table_name The name of the table into which you will be upserting.
|
86
|
+
#
|
87
|
+
# @yield [Upsert] An +Upsert+ object in batch mode. You can call #row on it multiple times and it will try to optimize on speed.
|
88
|
+
#
|
89
|
+
# @return [nil]
|
90
|
+
#
|
91
|
+
# @example Many at once
|
92
|
+
# Upsert.batch(Pet.connection, Pet.table_name) do |upsert|
|
93
|
+
# upsert.row({:name => 'Jerry'}, :breed => 'beagle')
|
94
|
+
# upsert.row({:name => 'Pierre'}, :breed => 'tabby')
|
95
|
+
# end
|
96
|
+
def batch(connection, table_name, options = {})
|
97
|
+
upsert = new connection, table_name, options
|
98
|
+
yield upsert
|
99
|
+
end
|
100
|
+
|
101
|
+
# @deprecated Use .batch instead.
|
102
|
+
alias :stream :batch
|
103
|
+
|
104
|
+
# @private
|
105
|
+
def class_name(metal)
|
106
|
+
if RUBY_PLATFORM == 'java'
|
107
|
+
metal.class.name || metal.get_class.name
|
108
|
+
else
|
109
|
+
metal.class.name
|
110
|
+
end
|
111
|
+
end
|
112
|
+
|
113
|
+
# @private
|
114
|
+
def flavor(metal)
|
115
|
+
case class_name(metal)
|
116
|
+
when /sqlite/i
|
117
|
+
'Sqlite3'
|
118
|
+
when /mysql/i
|
119
|
+
'Mysql'
|
120
|
+
when /pg/i, /postgres/i
|
121
|
+
'Postgresql'
|
122
|
+
else
|
123
|
+
raise "[upsert] #{metal} not supported"
|
124
|
+
end
|
125
|
+
end
|
126
|
+
|
127
|
+
# @private
|
128
|
+
def adapter(metal)
|
129
|
+
metal_class_name = class_name metal
|
130
|
+
METAL_CLASS_ALIAS.fetch(metal_class_name, metal_class_name).gsub /\W+/, '_'
|
131
|
+
end
|
132
|
+
|
133
|
+
# @private
|
134
|
+
def metal(connection)
|
135
|
+
metal = connection.respond_to?(:raw_connection) ? connection.raw_connection : connection
|
136
|
+
if metal.class.name.to_s.start_with?('ActiveRecord::ConnectionAdapters')
|
137
|
+
metal = metal.connection
|
138
|
+
end
|
139
|
+
metal
|
140
|
+
end
|
141
|
+
|
142
|
+
# @private
|
143
|
+
def utc(time)
|
144
|
+
if time.is_a? DateTime
|
145
|
+
usec = time.sec_fraction * SEC_FRACTION
|
146
|
+
if time.offset != 0
|
147
|
+
time = time.new_offset(0)
|
148
|
+
end
|
149
|
+
Time.utc time.year, time.month, time.day, time.hour, time.min, time.sec, usec
|
150
|
+
elsif time.utc?
|
151
|
+
time
|
152
|
+
else
|
153
|
+
time.utc
|
154
|
+
end
|
155
|
+
end
|
156
|
+
|
157
|
+
# @private
|
158
|
+
def utc_iso8601(time, tz = true)
|
159
|
+
t = utc time
|
160
|
+
s = t.strftime(ISO8601_DATETIME) + '.' + (USEC_SPRINTF % t.usec)
|
161
|
+
tz ? (s + UTC_TZ) : s
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
SINGLE_QUOTE = %{'}
|
166
|
+
DOUBLE_QUOTE = %{"}
|
167
|
+
BACKTICK = %{`}
|
168
|
+
X_AND_SINGLE_QUOTE = %{x'}
|
169
|
+
USEC_SPRINTF = '%06d'
|
170
|
+
if RUBY_VERSION >= '1.9.0'
|
171
|
+
SEC_FRACTION = 1e6
|
172
|
+
NANO_FRACTION = 1e9
|
173
|
+
else
|
174
|
+
SEC_FRACTION = 8.64e10
|
175
|
+
NANO_FRACTION = 8.64e13
|
176
|
+
end
|
177
|
+
ISO8601_DATETIME = '%Y-%m-%d %H:%M:%S'
|
178
|
+
ISO8601_DATE = '%F'
|
179
|
+
UTC_TZ = '+00:00'
|
180
|
+
NULL_WORD = 'NULL'
|
181
|
+
METAL_CLASS_ALIAS = {
|
182
|
+
'PGConn' => 'PG::Connection',
|
183
|
+
'org.sqlite.Conn' => 'Java::OrgSqlite::Conn', # for some reason, org.sqlite.Conn doesn't have a ruby class name
|
184
|
+
'Sequel::Postgres::Adapter' => 'PG::Connection', # Only the Postgres adapter needs an alias
|
185
|
+
}
|
186
|
+
CREATED_COL_REGEX = /\Acreated_(at|on)\z/
|
187
|
+
|
188
|
+
# @return [Upsert::Connection]
|
189
|
+
attr_reader :connection
|
190
|
+
|
191
|
+
# @return [String]
|
192
|
+
attr_reader :table_name
|
193
|
+
|
194
|
+
# @private
|
195
|
+
attr_reader :merge_function_class
|
196
|
+
|
197
|
+
# @private
|
198
|
+
attr_reader :flavor
|
199
|
+
|
200
|
+
# @private
|
201
|
+
attr_reader :adapter
|
202
|
+
|
203
|
+
# @private
|
204
|
+
def assume_function_exists?
|
205
|
+
@assume_function_exists
|
206
|
+
end
|
207
|
+
|
208
|
+
# @param [Mysql2::Client,Sqlite3::Database,PG::Connection,#metal] connection A supported database connection.
|
209
|
+
# @param [String,Symbol] table_name The name of the table into which you will be upserting.
|
210
|
+
# @param [Hash] options
|
211
|
+
# @option options [TrueClass,FalseClass] :assume_function_exists (true) Assume the function has already been defined correctly by another process.
|
212
|
+
def initialize(connection, table_name, options = {})
|
213
|
+
@table_name = self.class.normalize_table_name(table_name)
|
214
|
+
metal = Upsert.metal connection
|
215
|
+
@flavor = Upsert.flavor metal
|
216
|
+
@adapter = Upsert.adapter metal
|
217
|
+
# todo memoize
|
218
|
+
Dir[File.expand_path("../upsert/**/{#{flavor.downcase},#{adapter}}.rb", __FILE__)].each do |path|
|
219
|
+
require path
|
220
|
+
end
|
221
|
+
@connection = Connection.const_get(adapter).new self, metal
|
222
|
+
@merge_function_class = MergeFunction.const_get adapter
|
223
|
+
@merge_function_cache = {}
|
224
|
+
@assume_function_exists = options.fetch :assume_function_exists, @flavor != "Postgresql"
|
225
|
+
|
226
|
+
@merge_function_mutex = Mutex.new
|
227
|
+
@row_mutex = Mutex.new
|
228
|
+
end
|
229
|
+
|
230
|
+
# Upsert a row given a selector and a setter.
|
231
|
+
#
|
232
|
+
# The selector values are used as setters if it's a new row. So if your selector is `name=Jerry` and your setter is `age=4`, and there is no Jerry yet, then a new row will be created with name Jerry and age 4.
|
233
|
+
#
|
234
|
+
# @see http://api.mongodb.org/ruby/1.6.4/Mongo/Collection.html#update-instance_method Loosely based on the upsert functionality of the mongo-ruby-driver #update method
|
235
|
+
#
|
236
|
+
# @param [Hash] selector Key-value pairs that will be used to find or create a row.
|
237
|
+
# @param [Hash] setter Key-value pairs that will be set on the row, whether it previously existed or not.
|
238
|
+
#
|
239
|
+
# @return [nil]
|
240
|
+
#
|
241
|
+
# @example One at a time
|
242
|
+
# upsert = Upsert.new Pet.connection, Pet.table_name
|
243
|
+
# upsert.row({:name => 'Jerry'}, :breed => 'beagle')
|
244
|
+
# upsert.row({:name => 'Pierre'}, :breed => 'tabby')
|
245
|
+
def row(selector, setter = {}, options = nil)
|
246
|
+
row_object = Row.new(selector, setter, options)
|
247
|
+
self.class.mutex_for_row(self, row_object).synchronize do
|
248
|
+
merge_function(row_object).execute(row_object)
|
249
|
+
nil
|
250
|
+
end
|
251
|
+
end
|
252
|
+
|
253
|
+
# @private
|
254
|
+
def clear_database_functions
|
255
|
+
merge_function_class.clear connection
|
256
|
+
end
|
257
|
+
|
258
|
+
def merge_function(row)
|
259
|
+
cache_key = [row.selector.keys, row.setter.keys]
|
260
|
+
self.class.mutex_for_function(self, row).synchronize do
|
261
|
+
@merge_function_cache[cache_key] ||=
|
262
|
+
merge_function_class.new(self, row.selector.keys, row.setter.keys, assume_function_exists?)
|
263
|
+
end
|
264
|
+
end
|
265
|
+
|
266
|
+
# @private
|
267
|
+
def quoted_table_name
|
268
|
+
@quoted_table_name ||= table_name.map { |t| connection.quote_ident(t) }.join(".")
|
269
|
+
end
|
270
|
+
|
271
|
+
# @private
|
272
|
+
def column_definitions
|
273
|
+
@column_definitions ||= ColumnDefinition.const_get(flavor).all connection, quoted_table_name
|
274
|
+
end
|
275
|
+
|
276
|
+
# @private
|
277
|
+
def self.normalize_table_name(table_name)
|
278
|
+
if defined?(Sequel) && table_name.is_a?(::Sequel::SQL::QualifiedIdentifier)
|
279
|
+
[table_name.table, table_name.column]
|
280
|
+
else
|
281
|
+
[*table_name].map(&:to_s)
|
282
|
+
end
|
283
|
+
end
|
284
|
+
end
|
@@ -0,0 +1,12 @@
|
|
1
|
+
class Upsert
|
2
|
+
module ActiveRecordUpsert
|
3
|
+
def upsert(selector, setter = {})
|
4
|
+
ActiveRecord::Base.connection_pool.with_connection do |c|
|
5
|
+
upsert = Upsert.new c, table_name
|
6
|
+
upsert.row selector, setter
|
7
|
+
end
|
8
|
+
end
|
9
|
+
end
|
10
|
+
end
|
11
|
+
|
12
|
+
ActiveRecord::Base.extend Upsert::ActiveRecordUpsert
|
@@ -0,0 +1,79 @@
|
|
1
|
+
class Upsert
|
2
|
+
# @private
|
3
|
+
class ColumnDefinition
|
4
|
+
class << self
|
5
|
+
# activerecord-3.2.X/lib/active_record/connection_adapters/XXXXXXXXX_adapter.rb#column_definitions
|
6
|
+
def all(connection, table_name)
|
7
|
+
raise "not impl"
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
TIME_DETECTOR = /date|time/i
|
12
|
+
|
13
|
+
attr_reader :name
|
14
|
+
attr_reader :sql_type
|
15
|
+
attr_reader :default
|
16
|
+
attr_reader :quoted_name
|
17
|
+
attr_reader :quoted_selector_name
|
18
|
+
attr_reader :quoted_setter_name
|
19
|
+
|
20
|
+
def initialize(connection, name, sql_type, default)
|
21
|
+
@name = name
|
22
|
+
@sql_type = sql_type
|
23
|
+
@temporal_query = !!(sql_type =~ TIME_DETECTOR)
|
24
|
+
@default = default
|
25
|
+
@quoted_name = connection.quote_ident name
|
26
|
+
@quoted_selector_name = connection.quote_ident "#{name}_sel"
|
27
|
+
@quoted_setter_name = connection.quote_ident "#{name}_set"
|
28
|
+
end
|
29
|
+
|
30
|
+
def to_selector_arg
|
31
|
+
"#{quoted_selector_name} #{arg_type}"
|
32
|
+
end
|
33
|
+
|
34
|
+
def to_setter_arg
|
35
|
+
"#{quoted_setter_name} #{arg_type}"
|
36
|
+
end
|
37
|
+
|
38
|
+
def to_setter
|
39
|
+
"#{quoted_name} = #{to_setter_value}"
|
40
|
+
end
|
41
|
+
|
42
|
+
def to_selector
|
43
|
+
equality(quoted_name, to_selector_value)
|
44
|
+
end
|
45
|
+
|
46
|
+
def temporal?
|
47
|
+
@temporal_query
|
48
|
+
end
|
49
|
+
|
50
|
+
def equality(left, right)
|
51
|
+
"(#{left} = #{right} OR (#{left} IS NULL AND #{right} IS NULL))"
|
52
|
+
end
|
53
|
+
|
54
|
+
def arg_type
|
55
|
+
if temporal?
|
56
|
+
'character varying(255)'
|
57
|
+
else
|
58
|
+
sql_type
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
def to_setter_value
|
63
|
+
if temporal?
|
64
|
+
"CAST(#{quoted_setter_name} AS #{sql_type})"
|
65
|
+
else
|
66
|
+
quoted_setter_name
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def to_selector_value
|
71
|
+
if temporal?
|
72
|
+
"CAST(#{quoted_selector_name} AS #{sql_type})"
|
73
|
+
else
|
74
|
+
quoted_selector_name
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
end
|
79
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
class Upsert
|
2
|
+
class ColumnDefinition
|
3
|
+
# @private
|
4
|
+
class Mysql < ColumnDefinition
|
5
|
+
class << self
|
6
|
+
def all(connection, quoted_table_name)
|
7
|
+
connection.execute("SHOW COLUMNS FROM #{quoted_table_name}").map do |row|
|
8
|
+
# {"Field"=>"name", "Type"=>"varchar(255)", "Null"=>"NO", "Key"=>"PRI", "Default"=>nil, "Extra"=>""}
|
9
|
+
name = row['Field'] || row['COLUMN_NAME'] || row[:Field] || row[:COLUMN_NAME]
|
10
|
+
type = row['Type'] || row['COLUMN_TYPE'] || row[:Type] || row[:COLUMN_TYPE]
|
11
|
+
default = row['Default'] || row['COLUMN_DEFAULT'] || row[:Default] || row[:COLUMN_DEFAULT]
|
12
|
+
new connection, name, type, default
|
13
|
+
end.sort_by do |cd|
|
14
|
+
cd.name
|
15
|
+
end
|
16
|
+
end
|
17
|
+
end
|
18
|
+
|
19
|
+
def equality(left, right)
|
20
|
+
"#{left} <=> #{right}"
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|