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.
Files changed (68) hide show
  1. checksums.yaml +7 -0
  2. data/.gitignore +19 -0
  3. data/.ruby-version +1 -0
  4. data/.standard.yml +1 -0
  5. data/.travis.yml +63 -0
  6. data/.yardopts +2 -0
  7. data/CHANGELOG +265 -0
  8. data/Gemfile +20 -0
  9. data/LICENSE +24 -0
  10. data/README.md +411 -0
  11. data/Rakefile +54 -0
  12. data/lib/upsert.rb +284 -0
  13. data/lib/upsert/active_record_upsert.rb +12 -0
  14. data/lib/upsert/binary.rb +8 -0
  15. data/lib/upsert/column_definition.rb +79 -0
  16. data/lib/upsert/column_definition/mysql.rb +24 -0
  17. data/lib/upsert/column_definition/postgresql.rb +66 -0
  18. data/lib/upsert/column_definition/sqlite3.rb +34 -0
  19. data/lib/upsert/connection.rb +37 -0
  20. data/lib/upsert/connection/Java_ComMysqlJdbc_JDBC4Connection.rb +31 -0
  21. data/lib/upsert/connection/Java_OrgPostgresqlJdbc_PgConnection.rb +33 -0
  22. data/lib/upsert/connection/Java_OrgSqlite_Conn.rb +17 -0
  23. data/lib/upsert/connection/Mysql2_Client.rb +76 -0
  24. data/lib/upsert/connection/PG_Connection.rb +35 -0
  25. data/lib/upsert/connection/SQLite3_Database.rb +28 -0
  26. data/lib/upsert/connection/jdbc.rb +105 -0
  27. data/lib/upsert/connection/postgresql.rb +24 -0
  28. data/lib/upsert/connection/sqlite3.rb +19 -0
  29. data/lib/upsert/merge_function.rb +73 -0
  30. data/lib/upsert/merge_function/Java_ComMysqlJdbc_JDBC4Connection.rb +42 -0
  31. data/lib/upsert/merge_function/Java_OrgPostgresqlJdbc_PgConnection.rb +27 -0
  32. data/lib/upsert/merge_function/Java_OrgSqlite_Conn.rb +10 -0
  33. data/lib/upsert/merge_function/Mysql2_Client.rb +36 -0
  34. data/lib/upsert/merge_function/PG_Connection.rb +26 -0
  35. data/lib/upsert/merge_function/SQLite3_Database.rb +10 -0
  36. data/lib/upsert/merge_function/mysql.rb +66 -0
  37. data/lib/upsert/merge_function/postgresql.rb +365 -0
  38. data/lib/upsert/merge_function/sqlite3.rb +43 -0
  39. data/lib/upsert/row.rb +59 -0
  40. data/lib/upsert/version.rb +3 -0
  41. data/spec/active_record_upsert_spec.rb +26 -0
  42. data/spec/binary_spec.rb +21 -0
  43. data/spec/correctness_spec.rb +190 -0
  44. data/spec/database_functions_spec.rb +106 -0
  45. data/spec/database_spec.rb +121 -0
  46. data/spec/hstore_spec.rb +249 -0
  47. data/spec/jruby_spec.rb +9 -0
  48. data/spec/logger_spec.rb +52 -0
  49. data/spec/misc/get_postgres_reserved_words.rb +12 -0
  50. data/spec/misc/mysql_reserved.txt +226 -0
  51. data/spec/misc/pg_reserved.txt +742 -0
  52. data/spec/multibyte_spec.rb +27 -0
  53. data/spec/postgresql_spec.rb +94 -0
  54. data/spec/precision_spec.rb +11 -0
  55. data/spec/reserved_words_spec.rb +50 -0
  56. data/spec/sequel_spec.rb +57 -0
  57. data/spec/spec_helper.rb +417 -0
  58. data/spec/speed_spec.rb +44 -0
  59. data/spec/threaded_spec.rb +57 -0
  60. data/spec/timezones_spec.rb +58 -0
  61. data/spec/type_safety_spec.rb +12 -0
  62. data/travis/install_postgres.sh +18 -0
  63. data/travis/run_docker_db.sh +20 -0
  64. data/travis/tune_mysql.sh +7 -0
  65. data/upsert-java.gemspec +14 -0
  66. data/upsert.gemspec +13 -0
  67. data/upsert.gemspec.common +106 -0
  68. metadata +373 -0
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 7eb840a86c1973e535fae818ad6212aeae072512
4
+ data.tar.gz: 617afaddc0e769d688a4e3dba95e9beeef436596
5
+ SHA512:
6
+ metadata.gz: 900bce378101ac8da7235eb365f9a1b568b5d95480c9a32193186aaa34293b7ea1efb2af6557c833604646b9bd1b052bc3bc80bc384497e627b24039666078f9
7
+ data.tar.gz: 5d48a452e7d60bab795739152ddae77f169fca3685d21b3b931bf0f73998ac1af37ce9332d68a885873e2d07cf198125369b721dcae54ac48c11dd4d6edfe9e2
@@ -0,0 +1,19 @@
1
+ .DS_Store
2
+ *.gem
3
+ *.rbc
4
+ .bundle
5
+ .config
6
+ .yardoc
7
+ Gemfile.lock
8
+ InstalledFiles
9
+ _yardoc
10
+ coverage
11
+ doc/
12
+ lib/bundler/man
13
+ pkg
14
+ rdoc
15
+ spec/reports
16
+ test/tmp
17
+ test/version_tmp
18
+ tmp
19
+ *.log
@@ -0,0 +1 @@
1
+ jruby-9.1.17.0
@@ -0,0 +1 @@
1
+ ruby_version: 2.2.4
@@ -0,0 +1,63 @@
1
+ dist: xenial
2
+ language: ruby
3
+ cache: bundler
4
+ services:
5
+ - docker
6
+ rvm:
7
+ - 2.6
8
+ - 2.5
9
+ - 2.4
10
+ - 2.3
11
+ - 2.2
12
+ - jruby-9.1.14.0
13
+ - jruby-9.1.17.0
14
+ - jruby-9.2.7.0
15
+ env:
16
+ global:
17
+ - USERNAME=travis
18
+ - PASSWORD=
19
+ - DB_USER=upsert_test
20
+ - DB_PASSWORD=upsert_test
21
+ - DB_NAME=upsert_test
22
+ matrix:
23
+ - DB=postgresql DB_VERSION=postgres:9.4
24
+ - DB=postgresql DB_VERSION=postgres:9.5
25
+ - DB=postgresql DB_VERSION=postgres:9.6
26
+ - DB=postgresql DB_VERSION=postgres:10
27
+ - DB=postgresql DB_VERSION=postgres:11
28
+ - DB=postgresql DB_VERSION=postgres:12
29
+ - DB=postgresql DB_VERSION=postgres:9.4 UNIQUE_CONSTRAINT=true
30
+ - DB=postgresql DB_VERSION=postgres:9.5 UNIQUE_CONSTRAINT=true
31
+ - DB=postgresql DB_VERSION=postgres:9.6 UNIQUE_CONSTRAINT=true
32
+ - DB=postgresql DB_VERSION=postgres:10 UNIQUE_CONSTRAINT=true
33
+ - DB=postgresql DB_VERSION=postgres:11 UNIQUE_CONSTRAINT=true
34
+ - DB=postgresql DB_VERSION=postgres:12 UNIQUE_CONSTRAINT=true
35
+ - DB=mysql DB_VERSION=mysql:5.6
36
+ - DB=mysql DB_VERSION=mysql:5.7
37
+ - DB=mysql DB_VERSION=mysql:8
38
+ matrix:
39
+ exclude:
40
+ - rvm: 2.6
41
+ env: DB=postgresql DB_VERSION=postgres:9.4
42
+ - rvm: 2.6
43
+ env: DB=postgresql DB_VERSION=postgres:9.5
44
+ - rvm: 2.6
45
+ env: DB=postgresql DB_VERSION=postgres:9.4 UNIQUE_CONSTRAINT=true
46
+ - rvm: 2.6
47
+ env: DB=postgresql DB_VERSION=postgres:9.5 UNIQUE_CONSTRAINT=true
48
+ - rvm: jruby-9.2.7
49
+ env: DB=postgresql DB_VERSION=postgres:9.4
50
+ - rvm: jruby-9.2.7
51
+ env: DB=postgresql DB_VERSION=postgres:9.5
52
+ - rvm: jruby-9.2.7
53
+ env: DB=postgresql DB_VERSION=postgres:9.4 UNIQUE_CONSTRAINT=true
54
+ - rvm: jruby-9.2.7
55
+ env: DB=postgresql DB_VERSION=postgres:9.5 UNIQUE_CONSTRAINT=true
56
+ allow_failures:
57
+ - env: DB=postgresql DB_VERSION=postgres:12
58
+ - env: DB=postgresql DB_VERSION=postgres:12 UNIQUE_CONSTRAINT=true
59
+ before_install:
60
+ - ./travis/run_docker_db.sh
61
+ - bundle --version
62
+ - gem --version
63
+ script: ./travis/run_specs.sh
@@ -0,0 +1,2 @@
1
+ --no-private
2
+ --readme README.md
@@ -0,0 +1,265 @@
1
+ -- no version -- / 2019-06-05
2
+
3
+ * Enhancements
4
+
5
+ * Bump development Ruby version to 2.5 since Ruby 2.2 is no longer supported.
6
+ This should not affect usage of the gem, only local development for people
7
+ working *on* the gem. Ruby 2.2 is also not dropped from Upsert compatibility
8
+ at this time but you should consider upgrading to newer Ruby versions anyway.
9
+
10
+ 2.2.1 / 2017-04-20
11
+
12
+ * Bug fixes
13
+
14
+ * Fix unique constraint detection on pg >9.5.5 (@pnomolos https://github.com/seamusabshere/upsert/pull/99)
15
+ * Fix Ruby 1.9 tests
16
+
17
+ 2.2.0 / 2017-04-14
18
+
19
+ * Enhancements
20
+
21
+ * Use native "upsert" on Postgres 9.5+! (thanks to @pnomolos https://github.com/seamusabshere/upsert/pull/79)
22
+ * More modern CI tests
23
+
24
+ 2.1.2 / 2016-02-25
25
+
26
+ * Enhancements
27
+
28
+ * Test on Ruby 2.3 - thanks @Ch4s3 https://github.com/seamusabshere/upsert/pull/70
29
+
30
+ * Bug fixes
31
+
32
+ * Stop using Thread.exclusive - thanks @hpetru https://github.com/seamusabshere/upsert/pull/67
33
+
34
+ 2.1.1 / 2016-02-12
35
+
36
+ * Enhancements
37
+
38
+ * Assume function exists to avoid huge amounts of recreation
39
+
40
+ 2.1.0 / 2015-03-13
41
+
42
+ * Bug fixes
43
+
44
+ * Thread safety with Sidekiq! thanks @evadne and @thbar ! https://github.com/seamusabshere/upsert/pull/47
45
+
46
+ * Known issues
47
+
48
+ * speed_spec fails against activerecord-import on mysql... need some advice on properly testing it
49
+
50
+ 2.0.4 / 2015-01-27
51
+
52
+ * Bug fixes
53
+
54
+ * Support mysql returning column info as symbols - thanks @pnomolos
55
+
56
+ * Enhancements
57
+
58
+ * Travis and misc test fixes - thanks @raviolicode
59
+ * Backwards compat with 1.8 - thanks @raviolicode
60
+
61
+ 2.0.3 / 2013-11-27
62
+
63
+ * Bug fixes
64
+
65
+ * Add parentheses to equality expressions
66
+
67
+ * Enhancements
68
+
69
+ * On Postgres, use (A = B OR (A IS NULL AND B IS NULL)) instead of A IS NOT DISTINCT FROM B because the latter does not use indexes
70
+ * pass :eager_nullify => false as the third argument to Upsert#row to disable clearing of null HStore keys
71
+
72
+ 2.0.2 / 2013-11-06
73
+
74
+ * Bug fixes
75
+
76
+ * Properly check for NULL equality when creating the UPSERT functions - thanks @pnomolos - https://github.com/seamusabshere/upsert/issues/25
77
+ * When using Mysql2 client (MRI), don't pass timezone to DateTime columns - thanks @kjeremy! - https://github.com/seamusabshere/upsert/issues/24
78
+
79
+ 2.0.1 / 2013-07-24
80
+
81
+ * Bug fixes
82
+
83
+ * Rookie mistake - gsub!'ed an input arg, sorry. Blew up on a frozen string, thank goodness.
84
+
85
+ 2.0.0 / 2013-07-24
86
+
87
+ * Breaking changes
88
+
89
+ * For Postgres Hstore, null keys are deliberately deleted - so there's no way to set "foo" => NULL - "foo" will just get deleted as a key
90
+ * Name merge functions "upsert1_2_0" instead of "upsert_"
91
+
92
+ 1.2.0 / 2013-04-04
93
+
94
+ * Breaking changes
95
+
96
+ * columns named "created_at" and "created_on" will only be set if it's a new row - thanks @derekharmel! https://github.com/seamusabshere/upsert/pull/15
97
+
98
+ * Enhancements
99
+
100
+ * Detect invalid columns passed in either selector or setter - inspired by @atandrau, thanks! https://github.com/seamusabshere/upsert/issues/18
101
+
102
+ * Bug fixes
103
+
104
+ * Always convert symbols to strings when used as bind vars - thanks @towerhe! - https://github.com/seamusabshere/upsert/pull/16
105
+
106
+ 1.1.7 / 2013-01-15
107
+
108
+ * Enhancements
109
+
110
+ * :assume_function_exists option to avoid creating same merge function over and over
111
+ * Don't die on first occurrence of "tuple concurrently updated"
112
+
113
+ 1.1.6 / 2012-12-20
114
+
115
+ * Bug fixes
116
+
117
+ * Require pg-hstore >=1.1.1 which doesn't escape single quotes and backslashes when output is going to be used in a bind variable (as it is here)
118
+
119
+ 1.1.5 / 2012-12-10
120
+
121
+ * Bug fixes
122
+
123
+ * Properly cast "timestamp without time zone" in postgres - thanks @markmarijnissen! https://github.com/seamusabshere/upsert/issues/11
124
+
125
+ 1.1.4 / 2012-12-06
126
+
127
+ * Enhancements
128
+
129
+ * Use latest pg-hstore gem with standardized namespace
130
+
131
+ 1.1.3 / 2012-12-06
132
+
133
+ * Bug fix
134
+
135
+ * Don't die/fail trying to update an HStore column that has reverted to NULL.
136
+
137
+ 1.1.2 / 2012-12-06
138
+
139
+ * Enhancements
140
+
141
+ * Support for PostgreSQL's HStore
142
+
143
+ 1.1.1 / 2012-12-03
144
+
145
+ * Bug fixes
146
+
147
+ * Removed inadvertent dependency on ActiveSupport - thanks @thbar! https://github.com/seamusabshere/upsert/issues/10
148
+
149
+ 1.1.0 / 2012-11-26
150
+
151
+ * Enhancements
152
+
153
+ * Works on JRuby using bare-metal JDBC!
154
+ * Simplified.
155
+
156
+ 1.0.2 / 2012-11-12
157
+
158
+ * Bug fixes
159
+
160
+ * Fix filenames - they were updated on an apparently case-insensitive setup. Thanks @ihough! (https://github.com/seamusabshere/upsert/pull/8)
161
+ * Deliberately drop MySQL procedures before creating them. Also thanks to @ihough!
162
+
163
+ 1.0.1 / 2012-11-07
164
+
165
+ * Bug fixes
166
+
167
+ * Fix incorrect gem description
168
+
169
+ 1.0.0 / 2012-11-07
170
+
171
+ * Breaking changes (well, not really)
172
+
173
+ * Not using INSERT ... ON DUPLICATE KEY UPDATE for MySQL!
174
+
175
+ * Enhancements
176
+
177
+ * Replaced ON DUPLICATE KEY with a true merge function (procedure)
178
+ * Simplified code - buffering is no longer used anywhere
179
+ * Clarified documentation
180
+
181
+ * Bug fixes
182
+
183
+ * MySQL upserts won't fail if you have a multi-key selector and no multi-column UNIQUE index to cover them (https://github.com/seamusabshere/upsert/issues/6)
184
+
185
+ 0.5.0 / 2012-09-21
186
+
187
+ * Breaking changes (well, not really)
188
+
189
+ * "document" (as in the second argument to #row) has been renamed to "setter"!
190
+
191
+ * Bug fixes
192
+
193
+ * If you say upsert({:name => 'Jerry', :color => 'red'}), make sure that it only affects rows really meeting those conditions
194
+ * Always sort selector and setter keys - i.e., column names - before doing anything with them
195
+ * Support PostgreSQL 9.1+
196
+ * Support MRI 1.8
197
+
198
+ * Enhancements
199
+
200
+ * Slightly faster benchmarks for SQlite3 and MySQL
201
+ * Slightly slower on PostgreSQL (probably because the merge function requires more arguments), but more accurate
202
+ * Slightly clearer code structure
203
+ * Use bind parameters instead of quoting for PostgreSQL and SQLite3.
204
+ * Provide Upsert.clear_database_functions(connection) (currently only for PostgreSQL)
205
+ * Don't subclass String for Upset::Binary... hopefully save some strcpy()s?
206
+
207
+ 0.4.0 / 2012-09-04
208
+
209
+ * Bug fixes
210
+
211
+ * Don't raise TooBig - rely on Mysql2 to complain about oversized packets
212
+
213
+ * Enhancements
214
+
215
+ * Re-use PostgreSQL merge functions across connections, even outside of batch mode. Huzzah!
216
+ * For MySQL, increase speed for one-off upserts by not checking packet size
217
+ * Allow configuring Upsert.logger. Defaults to Rails.logger or Logger.new($stderr). If you set env var UPSERT_DEBUG=true then it will set log level to debug.
218
+
219
+ 0.3.4 / 2012-07-03
220
+
221
+ * Bug fixes
222
+
223
+ * Allow upserting by auto-increment primary key (thanks @atandrau https://github.com/seamusabshere/upsert/issues/3)
224
+
225
+ * Enhancements
226
+
227
+ * Make setter an optional argument
228
+
229
+ 0.3.3 / 2012-06-26
230
+
231
+ * Bug fixes
232
+
233
+ * Properly quote table names - don't assume that everybody has ANSI_QUOTES turned on in MySQL :)
234
+
235
+ 0.3.2 / 2012-06-22
236
+
237
+ * Enhancements
238
+
239
+ * Make sure ::PGconn is recognized as ::PG::Connection (thanks @joevandyk https://github.com/seamusabshere/upsert/issues/2)
240
+
241
+ 0.3.1 / 2012-06-21
242
+
243
+ * Bug fixes
244
+
245
+ * On PostgreSQL, stop nullifying columns that weren't even involved in the upsert
246
+ * On SQLite, properly join WHERE conditions with ' AND ' instead of ','
247
+
248
+ 0.3.0 / 2012-06-21
249
+
250
+ * Enhancements
251
+
252
+ * Remove all the sampling - just keep a cumulative total of sql bytes as we build up an ON DUPLICATE KEY UPDATE query.
253
+ * Deprecate Upsert.stream in favor of Upsert.batch (but provide an alias for backwards compat)
254
+
255
+ 0.2.2 / 2012-06-21
256
+
257
+ * Bug fixes
258
+
259
+ * Correct and simplify how sql length is calculated when batching MySQL upserts.
260
+
261
+ 0.2.1 / 2012-06-21
262
+
263
+ * Enhancements
264
+
265
+ * Added support for Ruby 1.8.7
data/Gemfile ADDED
@@ -0,0 +1,20 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in upsert.gemspec
4
+
5
+ if Gem::Version.new(Bundler::VERSION) >= Gem::Version.new("2.0.0")
6
+ gemspec glob: RUBY_PLATFORM == "java" ? "upsert-java.gemspec" : "upsert.gemspec"
7
+ else
8
+ gemspec name: RUBY_PLATFORM == "java" ? "upsert-java" : "upsert"
9
+ end
10
+
11
+ case RUBY_PLATFORM
12
+ when "java"
13
+ gem "ffi", platforms: :jruby
14
+ else
15
+ gem "ffi"
16
+ end
17
+
18
+ group "test" do
19
+ gem "testmetrics_rspec"
20
+ end
data/LICENSE ADDED
@@ -0,0 +1,24 @@
1
+ Copyright (c) 2013-2019 Seamus Abshere
2
+ Copyright (c) 2017-2019 Philip Schalm
3
+ Portions Copyright (c) 2019 The JRuby Team
4
+
5
+ MIT License
6
+
7
+ Permission is hereby granted, free of charge, to any person obtaining
8
+ a copy of this software and associated documentation files (the
9
+ "Software"), to deal in the Software without restriction, including
10
+ without limitation the rights to use, copy, modify, merge, publish,
11
+ distribute, sublicense, and/or sell copies of the Software, and to
12
+ permit persons to whom the Software is furnished to do so, subject to
13
+ the following conditions:
14
+
15
+ The above copyright notice and this permission notice shall be
16
+ included in all copies or substantial portions of the Software.
17
+
18
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
19
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
21
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
22
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
23
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
24
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,411 @@
1
+ # Upsert
2
+
3
+ [![Build Status](https://travis-ci.org/seamusabshere/upsert.svg?branch=master)](https://travis-ci.org/seamusabshere/upsert)
4
+
5
+ Make it easy to upsert on traditional RDBMS like MySQL, PostgreSQL, and SQLite3—hey look NoSQL!. Transparently creates (and re-uses) stored procedures/functions when necessary.
6
+
7
+ You pass it a bare-metal connection to the database like `Mysql2::Client` (from `mysql2` gem on MRI) or `Java::OrgPostgresqlJdbc4::Jdbc4Connection` (from `jdbc-postgres` on Jruby).
8
+
9
+ As databases start to natively support SQL MERGE (which is basically upsert), this library will take advantage (but you won't have to change your code).
10
+
11
+ Does **not** depend on ActiveRecord.
12
+
13
+ Does **not** use `INSERT ON DUPLICATE KEY UPDATE` on MySQL as this only works if you are very careful about creating unique indexes.
14
+
15
+ 70–90%+ faster than emulating upsert with ActiveRecord.
16
+
17
+ Supports MRI and JRuby.
18
+
19
+ ## Usage
20
+
21
+ You pass a __selector__ that uniquely identifies a row, whether it exists or not. You also pass a __setter__, attributes that should be set on that row.
22
+
23
+ Syntax inspired by [mongo-ruby-driver's update method](http://api.mongodb.org/ruby/1.6.4/Mongo/Collection.html#update-instance_method).
24
+
25
+ ### Basic
26
+
27
+ ```ruby
28
+ connection = Mysql2::Client.new([...])
29
+ table_name = :pets
30
+ upsert = Upsert.new connection, table_name
31
+ # N times...
32
+ upsert.row({:name => 'Jerry'}, :breed => 'beagle', :created_at => Time.now)
33
+ ```
34
+
35
+ The `created_at` and `created_on` columns are used for inserts, but ignored on updates.
36
+
37
+ So just to reiterate you've got a __selector__ and a __setter__:
38
+
39
+ ```ruby
40
+ selector = { :name => 'Jerry' }
41
+ setter = { :breed => 'beagle' }
42
+ upsert.row(selector, setter)
43
+ ```
44
+
45
+ ### Batch mode
46
+
47
+ By organizing your upserts into a batch, we can do work behind the scenes to make them faster.
48
+
49
+ ```ruby
50
+ connection = Mysql2::Client.new([...])
51
+ Upsert.batch(connection, :pets) do |upsert|
52
+ # N times...
53
+ upsert.row({:name => 'Jerry'}, :breed => 'beagle')
54
+ upsert.row({:name => 'Pierre'}, :breed => 'tabby')
55
+ end
56
+ ```
57
+
58
+ Batch mode is tested to be about 80% faster on PostgreSQL, MySQL, and SQLite3 than other ways to emulate upsert (see the tests, which fail if they are not faster).
59
+
60
+ ### Native Postgres upsert
61
+
62
+ `INSERT ... ON CONFLICT DO UPDATE` is used when Postgres 9.5+ is detected and *unique constraint are in place.*
63
+
64
+ **Note: ** You must have a **unique constraint** on the column(s) you're using as a selector. A unique index won't work. See https://github.com/seamusabshere/upsert/issues/98#issuecomment-295341405 for more information and some ways to check.
65
+
66
+ If you don't have unique constraints, it will fall back to the classic Upsert gem user-defined function, which does not require a constraint.
67
+
68
+ ### ActiveRecord helper method
69
+
70
+ ```ruby
71
+ require 'upsert/active_record_upsert'
72
+ # N times...
73
+ Pet.upsert({:name => 'Jerry'}, :breed => 'beagle')
74
+ ```
75
+
76
+ ## Wishlist
77
+
78
+ Pull requests for any of these would be greatly appreciated:
79
+
80
+ 1. Cache JDBC PreparedStatement objects.
81
+ 1. Sanity check my three benchmarks (four if you include activerecord-import on MySQL). Do they accurately represent optimized alternatives?
82
+ 1. Provide `require 'upsert/debug'` that will make sure you are selecting on columns that have unique indexes
83
+ 1. Test that `Upsert` instances accept arbitrary columns, even within a batch, which is what people probably expect.
84
+ 1. [@antage](https://github.com/antage)'s idea for "true" upserting: (from https://github.com/seamusabshere/upsert/issues/17)
85
+
86
+ ```ruby
87
+ selector = { id: 15 }
88
+ update_setter = { count: Upsert.sql('count + 1') }
89
+ insert_setter = { count: 1 }
90
+ upsert.row_with_two_setter(update_setter, insert_setter, selector)
91
+ ```
92
+
93
+ ## Real-world usage
94
+
95
+ <p><a href="http://angel.co/faraday"><img src="https://s3.amazonaws.com/photos.angel.co/startups/i/175701-a63ebd1b56a401e905963c64958204d4-medium_jpg.jpg" alt="Faraday logo"/></a></p>
96
+
97
+ We use `upsert` for [big data at Faraday](http://angel.co/faraday). Originally written to speed up the [`data_miner`](https://github.com/seamusabshere/data_miner) data mining library.
98
+
99
+ ## Supported databases/drivers
100
+
101
+ <table>
102
+ <tr>
103
+ <th>*</th>
104
+ <th>MySQL</th>
105
+ <th>PostgreSQL</th>
106
+ <th>SQLite3</th>
107
+ </tr>
108
+ <tr>
109
+ <th>MRI</th>
110
+ <td><a href="https://rubygems.org/gems/mysql2">mysql2</a></td>
111
+ <td><a href="https://rubygems.org/gems/pg">pg</a></td>
112
+ <td><a href="https://rubygems.org/gems/sqlite3">sqlite3</a></td>
113
+ </tr>
114
+ <tr>
115
+ <th>JRuby</th>
116
+ <td><a href="https://rubygems.org/gems/jdbc-mysql">jdbc-mysql</a></td>
117
+ <td><a href="https://rubygems.org/gems/jdbc-postgres">jdbc-postgres</a></td>
118
+ <td><a href="https://rubygems.org/gems/jdbc-sqlite3">jdbc-sqlite3</a></td>
119
+ </tr>
120
+ </table>
121
+
122
+ See below for details about what SQL MERGE trick (emulation of upsert) is used, performance, code examples, etc.
123
+
124
+ ### Rails / ActiveRecord
125
+
126
+ (Assuming that one of the other three supported drivers is being used under the covers).
127
+
128
+ * add "upsert" to your Gemfile and
129
+ * run bundle install
130
+
131
+ ```ruby
132
+ Upsert.new Pet.connection, Pet.table_name
133
+ ```
134
+
135
+ #### Speed
136
+
137
+ Depends on the driver being used!
138
+
139
+ #### SQL MERGE trick
140
+
141
+ Depends on the driver being used!
142
+
143
+ ### MySQL
144
+
145
+ On MRI, use the [mysql2](https://rubygems.org/gems/mysql2) driver.
146
+
147
+ ```ruby
148
+ require 'mysql2'
149
+ connection = Mysql2::Connection.new(:username => 'root', :password => 'password', :database => 'upsert_test')
150
+ table_name = :pets
151
+ upsert = Upsert.new(connection, table_name)
152
+ ```
153
+
154
+ On JRuby, use the [jdbc-mysql](https://rubygems.org/gems/jdbc-mysql) driver.
155
+
156
+ ```ruby
157
+ require 'jdbc/mysql'
158
+ java.sql.DriverManager.register_driver com.mysql.jdbc.Driver.new
159
+ connection = java.sql.DriverManager.get_connection "jdbc:mysql://127.0.0.1/mydatabase?user=root&password=password"
160
+ ```
161
+
162
+ #### Speed
163
+
164
+ From the tests (updated 11/7/12):
165
+
166
+ Upsert was 82% faster than find + new/set/save
167
+ Upsert was 85% faster than find_or_create + update_attributes
168
+ Upsert was 90% faster than create + rescue/find/update
169
+ Upsert was 46% faster than faking upserts with activerecord-import (note: in question as of 3/13/15, need some expert advice)
170
+
171
+ #### SQL MERGE trick
172
+
173
+ Thanks to [Dennis Hennen's StackOverflow response!](http://stackoverflow.com/questions/11371479/how-to-translate-postgresql-merge-db-aka-upsert-function-into-mysql/)!
174
+
175
+ ```sql
176
+ CREATE PROCEDURE upsert_pets_SEL_name_A_tag_number_SET_name_A_tag_number(`name_sel` varchar(255), `tag_number_sel` int(11), `name_set` varchar(255), `tag_number_set` int(11))
177
+ BEGIN
178
+ DECLARE done BOOLEAN;
179
+ REPEAT
180
+ BEGIN
181
+ -- If there is a unique key constraint error then
182
+ -- someone made a concurrent insert. Reset the sentinel
183
+ -- and try again.
184
+ DECLARE ER_DUP_UNIQUE CONDITION FOR 23000;
185
+ DECLARE ER_INTEG CONDITION FOR 1062;
186
+ DECLARE CONTINUE HANDLER FOR ER_DUP_UNIQUE BEGIN
187
+ SET done = FALSE;
188
+ END;
189
+
190
+ DECLARE CONTINUE HANDLER FOR ER_INTEG BEGIN
191
+ SET done = TRUE;
192
+ END;
193
+
194
+ SET done = TRUE;
195
+ SELECT COUNT(*) INTO @count FROM `pets` WHERE `name` = `name_sel` AND `tag_number` = `tag_number_sel`;
196
+ -- Race condition here. If a concurrent INSERT is made after
197
+ -- the SELECT but before the INSERT below we'll get a duplicate
198
+ -- key error. But the handler above will take care of that.
199
+ IF @count > 0 THEN
200
+ -- UPDATE table_name SET b = b_SET WHERE a = a_SEL;
201
+ UPDATE `pets` SET `name` = `name_set`, `tag_number` = `tag_number_set` WHERE `name` = `name_sel` AND `tag_number` = `tag_number_sel`;
202
+ ELSE
203
+ -- INSERT INTO table_name (a, b) VALUES (k, data);
204
+ INSERT INTO `pets` (`name`, `tag_number`) VALUES (`name_set`, `tag_number_set`);
205
+ END IF;
206
+ END;
207
+ UNTIL done END REPEAT;
208
+ END
209
+ ```
210
+
211
+ ### PostgreSQL
212
+
213
+ On MRI, use the [pg](https://rubygems.org/gems/pg) driver.
214
+
215
+ ```ruby
216
+ require 'pg'
217
+ connection = PG.connect(:dbname => 'upsert_test')
218
+ table_name = :pets
219
+ upsert = Upsert.new(connection, table_name)
220
+ ```
221
+
222
+ On JRuby, use the [jdbc-postgres](https://rubygems.org/gems/jdbc-postgres) driver.
223
+
224
+ ```ruby
225
+ require 'jdbc/postgres'
226
+ java.sql.DriverManager.register_driver org.postgresql.Driver.new
227
+ connection = java.sql.DriverManager.get_connection "jdbc:postgresql://127.0.0.1/mydatabase?user=root&password=password"
228
+ ```
229
+
230
+ If you want to use HStore, make the `pg-hstore` gem available and pass a Hash in setters:
231
+
232
+ ```ruby
233
+ gem 'pg-hstore'
234
+ require 'pg_hstore'
235
+ upsert.row({:name => 'Bill'}, :mydata => {:a => 1, :b => 2})
236
+ ```
237
+
238
+ #### PostgreSQL notes
239
+
240
+ - Upsert doesn't do any type casting, so if you attempt to do something like the following:
241
+ `upsert.row({ :name => 'A Name' }, :tag_number => 'bob')`
242
+ you'll get an error which reads something like:
243
+ `invalid input syntax for integer: "bob"`
244
+
245
+
246
+
247
+ #### Speed
248
+
249
+ From the tests (updated 9/21/12):
250
+
251
+ Upsert was 72% faster than find + new/set/save
252
+ Upsert was 79% faster than find_or_create + update_attributes
253
+ Upsert was 83% faster than create + rescue/find/update
254
+ # (can't compare to activerecord-import because you can't fake it on pg)
255
+
256
+ #### SQL MERGE trick
257
+
258
+ Adapted from the [canonical PostgreSQL upsert example](http://www.postgresql.org/docs/current/interactive/plpgsql-control-structures.html#PLPGSQL-ERROR-TRAPPING):
259
+
260
+ ```sql
261
+ CREATE OR REPLACE FUNCTION upsert_pets_SEL_name_A_tag_number_SET_name_A_tag_number("name_sel" character varying(255), "tag_number_sel" integer, "name_set" character varying(255), "tag_number_set" integer) RETURNS VOID AS
262
+ $$
263
+ DECLARE
264
+ first_try INTEGER := 1;
265
+ BEGIN
266
+ LOOP
267
+ -- first try to update the key
268
+ UPDATE "pets" SET "name" = "name_set", "tag_number" = "tag_number_set"
269
+ WHERE "name" = "name_sel" AND "tag_number" = "tag_number_sel";
270
+ IF found THEN
271
+ RETURN;
272
+ END IF;
273
+ -- not there, so try to insert the key
274
+ -- if someone else inserts the same key concurrently,
275
+ -- we could get a unique-key failure
276
+ BEGIN
277
+ INSERT INTO "pets"("name", "tag_number") VALUES ("name_set", "tag_number_set");
278
+ RETURN;
279
+ EXCEPTION WHEN unique_violation THEN
280
+ -- seamusabshere 9/20/12 only retry once
281
+ IF (first_try = 1) THEN
282
+ first_try := 0;
283
+ ELSE
284
+ RETURN;
285
+ END IF;
286
+ -- Do nothing, and loop to try the UPDATE again.
287
+ END;
288
+ END LOOP;
289
+ END;
290
+ $$
291
+ LANGUAGE plpgsql;
292
+ ```
293
+
294
+ I slightly modified it so that it only retries once - don't want infinite loops.
295
+
296
+ ### Sqlite
297
+
298
+ On MRI, use the [sqlite3](https://rubygems.org/gems/sqlite3) driver.
299
+
300
+ ```ruby
301
+ require 'sqlite3'
302
+ connection = SQLite3::Database.open(':memory:')
303
+ table_name = :pets
304
+ upsert = Upsert.new(connection, table_name)
305
+ ```
306
+
307
+ On JRuby, use the [jdbc-sqlite3](https://rubygems.org/gems/jdbc-sqlite3) driver.
308
+
309
+ ```ruby
310
+ # TODO somebody please verify
311
+ require 'jdbc/sqlite3'
312
+ java.sql.DriverManager.register_driver org.sqlite.Driver.new
313
+ connection = java.sql.DriverManager.get_connection "jdbc:sqlite://127.0.0.1/mydatabase?user=root&password=password"
314
+ ```
315
+
316
+ #### Speed
317
+
318
+ From the tests (updated 9/21/12):
319
+
320
+ Upsert was 77% faster than find + new/set/save
321
+ Upsert was 80% faster than find_or_create + update_attributes
322
+ Upsert was 85% faster than create + rescue/find/update
323
+ # (can't compare to activerecord-import because you can't fake it on sqlite3)
324
+
325
+ #### SQL MERGE trick
326
+
327
+ Thanks to [@dan04's answer on StackOverflow](http://stackoverflow.com/questions/2717590/sqlite-upsert-on-duplicate-key-update):
328
+
329
+ **Please note! This will only work properly on Sqlite if one of the columns being used as the "selector" are a primary key or unique index**
330
+
331
+ ```sql
332
+ INSERT OR IGNORE INTO visits VALUES (127.0.0.1, 1);
333
+ UPDATE visits SET visits = 1 WHERE ip LIKE 127.0.0.1;
334
+ ```
335
+
336
+ ## Features
337
+
338
+ ### Tested to be fast and portable
339
+
340
+ In addition to correctness, the library's tests check that it is
341
+
342
+ 1. Faster than comparable upsert techniques
343
+ 2. Compatible with supported databases
344
+
345
+ ### Not dependent on ActiveRecord
346
+
347
+ As below, all you need is a raw database connection like a `Mysql2::Connection`, `PG::Connection` or a `SQLite3::Database`. These are equivalent:
348
+
349
+ ```ruby
350
+ # with activerecord
351
+ Upsert.new ActiveRecord::Base.connection, :pets
352
+ # with activerecord, prettier
353
+ Upsert.new Pet.connection, Pet.table_name
354
+ # without activerecord
355
+ Upsert.new Mysql2::Connection.new([...]), :pets
356
+ ```
357
+
358
+ ### For a specific use case, faster and more portable than `activerecord-import`
359
+
360
+ You could also use [activerecord-import](https://github.com/zdennis/activerecord-import) to upsert:
361
+
362
+ ```ruby
363
+ Pet.import columns, all_values, :timestamps => false, :on_duplicate_key_update => columns
364
+ ```
365
+
366
+ `activerecord-import`, however, only works on MySQL and requires ActiveRecord&mdash;and if all you are doing is upserts, `upsert` is tested to be 40% faster. And you don't have to put all of the rows to be upserted into a single huge array - you can batch them using `Upsert.batch`.
367
+
368
+ ## Gotchas
369
+
370
+ ### No automatic typecasting beyond what the adapter/driver provides
371
+
372
+ We don't have any logic to convert integers into strings, strings into integers, etc. in order to satisfy PostgreSQL/etc.'s strictness on this issue.
373
+
374
+ So if you try to upsert a blank string (`''`) into an integer field in PostgreSQL, you will get an error.
375
+
376
+ ### Dates and times are converted to UTC
377
+
378
+ Datetimes are immediately converted to UTC and sent to the database as ISO8601 strings.
379
+
380
+ If you're using MySQL, make sure server/connection timezone is UTC. If you're using Rails and/or ActiveRecord, you might want to check `ActiveRecord::Base.default_timezone`... it should probably be `:utc`.
381
+
382
+ In general, run some upserts and make sure datetimes get persisted like you expect.
383
+
384
+ ### Clearning all library-generated functions
385
+
386
+ Place the following in to a rake task (so you don't globally redefine the `NAME_PREFIX` constant)
387
+
388
+ ```ruby
389
+ Upsert::MergeFunction::NAME_PREFIX = "upsert"
390
+
391
+ # ActiveRecord
392
+ Upsert.clear_database_functions(ActiveRecord::Base.connection)
393
+
394
+ # Sequel
395
+ DB.synchronize do |conn|
396
+ Upsert.clear_database_functions(conn)
397
+ end
398
+ ```
399
+
400
+ ### Doesn't work with transactional fixtures
401
+
402
+ Per https://github.com/seamusabshere/upsert/issues/23 you might have issues if you try to use transactional fixtures and this library.
403
+
404
+ ##
405
+ Testmetrics - https://www.testmetrics.app/seamusabshere/upsert
406
+
407
+ ## Copyright
408
+
409
+ Copyright 2013-2019 Seamus Abshere
410
+ Copyright 2017-2019 Philip Schalm
411
+ Portions Copyright (c) 2019 The JRuby Team