mysql2 0.3.18-x64-mingw32
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/README.md +539 -0
- data/examples/eventmachine.rb +21 -0
- data/examples/threaded.rb +20 -0
- data/ext/mysql2/client.c +1416 -0
- data/ext/mysql2/client.h +56 -0
- data/ext/mysql2/extconf.rb +177 -0
- data/ext/mysql2/infile.c +122 -0
- data/ext/mysql2/infile.h +1 -0
- data/ext/mysql2/mysql2_ext.c +12 -0
- data/ext/mysql2/mysql2_ext.h +45 -0
- data/ext/mysql2/mysql_enc_name_to_ruby.h +168 -0
- data/ext/mysql2/mysql_enc_to_ruby.h +246 -0
- data/ext/mysql2/result.c +684 -0
- data/ext/mysql2/result.h +23 -0
- data/ext/mysql2/wait_for_single_fd.h +36 -0
- data/lib/mysql2.rb +64 -0
- data/lib/mysql2/2.0/mysql2.so +0 -0
- data/lib/mysql2/2.1/mysql2.so +0 -0
- data/lib/mysql2/client.rb +90 -0
- data/lib/mysql2/console.rb +5 -0
- data/lib/mysql2/em.rb +56 -0
- data/lib/mysql2/error.rb +80 -0
- data/lib/mysql2/mysql2.rb +2 -0
- data/lib/mysql2/result.rb +5 -0
- data/lib/mysql2/version.rb +3 -0
- data/spec/configuration.yml.example +17 -0
- data/spec/em/em_spec.rb +135 -0
- data/spec/my.cnf.example +9 -0
- data/spec/mysql2/client_spec.rb +897 -0
- data/spec/mysql2/error_spec.rb +83 -0
- data/spec/mysql2/result_spec.rb +505 -0
- data/spec/rcov.opts +3 -0
- data/spec/spec_helper.rb +87 -0
- data/spec/test_data +1 -0
- data/support/libmysql.def +219 -0
- data/support/mysql_enc_to_ruby.rb +82 -0
- data/support/ruby_enc_to_mysql.rb +61 -0
- data/vendor/README +654 -0
- data/vendor/libmysql.dll +0 -0
- metadata +149 -0
data/spec/my.cnf.example
ADDED
@@ -0,0 +1,897 @@
|
|
1
|
+
# encoding: UTF-8
|
2
|
+
require 'spec_helper'
|
3
|
+
|
4
|
+
describe Mysql2::Client do
|
5
|
+
context "using defaults file" do
|
6
|
+
let(:cnf_file) { File.expand_path('../../my.cnf', __FILE__) }
|
7
|
+
|
8
|
+
it "should not raise an exception for valid defaults group" do
|
9
|
+
lambda {
|
10
|
+
opts = DatabaseCredentials['root'].merge(:default_file => cnf_file, :default_group => "test")
|
11
|
+
@client = Mysql2::Client.new(opts)
|
12
|
+
}.should_not raise_error(Mysql2::Error)
|
13
|
+
end
|
14
|
+
|
15
|
+
it "should not raise an exception without default group" do
|
16
|
+
lambda {
|
17
|
+
@client = Mysql2::Client.new(DatabaseCredentials['root'].merge(:default_file => cnf_file))
|
18
|
+
}.should_not raise_error(Mysql2::Error)
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
it "should raise an exception upon connection failure" do
|
23
|
+
lambda {
|
24
|
+
# The odd local host IP address forces the mysql client library to
|
25
|
+
# use a TCP socket rather than a domain socket.
|
26
|
+
Mysql2::Client.new DatabaseCredentials['root'].merge('host' => '127.0.0.2', 'port' => 999999)
|
27
|
+
}.should raise_error(Mysql2::Error)
|
28
|
+
end
|
29
|
+
|
30
|
+
if defined? Encoding
|
31
|
+
it "should raise an exception on create for invalid encodings" do
|
32
|
+
lambda {
|
33
|
+
Mysql2::Client.new(DatabaseCredentials['root'].merge(:encoding => "fake"))
|
34
|
+
}.should raise_error(Mysql2::Error)
|
35
|
+
end
|
36
|
+
|
37
|
+
it "should not raise an exception on create for a valid encoding" do
|
38
|
+
lambda {
|
39
|
+
Mysql2::Client.new(DatabaseCredentials['root'].merge(:encoding => "utf8"))
|
40
|
+
}.should_not raise_error(Mysql2::Error)
|
41
|
+
|
42
|
+
lambda {
|
43
|
+
Mysql2::Client.new(DatabaseCredentials['root'].merge(:encoding => "big5"))
|
44
|
+
}.should_not raise_error(Mysql2::Error)
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
it "should accept connect flags and pass them to #connect" do
|
49
|
+
klient = Class.new(Mysql2::Client) do
|
50
|
+
attr_reader :connect_args
|
51
|
+
def connect *args
|
52
|
+
@connect_args ||= []
|
53
|
+
@connect_args << args
|
54
|
+
end
|
55
|
+
end
|
56
|
+
client = klient.new :flags => Mysql2::Client::FOUND_ROWS
|
57
|
+
(client.connect_args.last[6] & Mysql2::Client::FOUND_ROWS).should be_true
|
58
|
+
end
|
59
|
+
|
60
|
+
it "should default flags to (REMEMBER_OPTIONS, LONG_PASSWORD, LONG_FLAG, TRANSACTIONS, PROTOCOL_41, SECURE_CONNECTION)" do
|
61
|
+
klient = Class.new(Mysql2::Client) do
|
62
|
+
attr_reader :connect_args
|
63
|
+
def connect *args
|
64
|
+
@connect_args ||= []
|
65
|
+
@connect_args << args
|
66
|
+
end
|
67
|
+
end
|
68
|
+
client = klient.new
|
69
|
+
(client.connect_args.last[6] & (Mysql2::Client::REMEMBER_OPTIONS |
|
70
|
+
Mysql2::Client::LONG_PASSWORD |
|
71
|
+
Mysql2::Client::LONG_FLAG |
|
72
|
+
Mysql2::Client::TRANSACTIONS |
|
73
|
+
Mysql2::Client::PROTOCOL_41 |
|
74
|
+
Mysql2::Client::SECURE_CONNECTION)).should be_true
|
75
|
+
end
|
76
|
+
|
77
|
+
it "should execute init command" do
|
78
|
+
options = DatabaseCredentials['root'].dup
|
79
|
+
options[:init_command] = "SET @something = 'setting_value';"
|
80
|
+
client = Mysql2::Client.new(options)
|
81
|
+
result = client.query("SELECT @something;")
|
82
|
+
result.first['@something'].should eq('setting_value')
|
83
|
+
end
|
84
|
+
|
85
|
+
it "should send init_command after reconnect" do
|
86
|
+
options = DatabaseCredentials['root'].dup
|
87
|
+
options[:init_command] = "SET @something = 'setting_value';"
|
88
|
+
options[:reconnect] = true
|
89
|
+
client = Mysql2::Client.new(options)
|
90
|
+
|
91
|
+
result = client.query("SELECT @something;")
|
92
|
+
result.first['@something'].should eq('setting_value')
|
93
|
+
|
94
|
+
# get the current connection id
|
95
|
+
result = client.query("SELECT CONNECTION_ID()")
|
96
|
+
first_conn_id = result.first['CONNECTION_ID()']
|
97
|
+
|
98
|
+
# break the current connection
|
99
|
+
begin
|
100
|
+
client.query("KILL #{first_conn_id}")
|
101
|
+
rescue Mysql2::Error
|
102
|
+
end
|
103
|
+
|
104
|
+
client.ping # reconnect now
|
105
|
+
|
106
|
+
# get the new connection id
|
107
|
+
result = client.query("SELECT CONNECTION_ID()")
|
108
|
+
second_conn_id = result.first['CONNECTION_ID()']
|
109
|
+
|
110
|
+
# confirm reconnect by checking the new connection id
|
111
|
+
first_conn_id.should_not == second_conn_id
|
112
|
+
|
113
|
+
# At last, check that the init command executed
|
114
|
+
result = client.query("SELECT @something;")
|
115
|
+
result.first['@something'].should eq('setting_value')
|
116
|
+
end
|
117
|
+
|
118
|
+
it "should have a global default_query_options hash" do
|
119
|
+
Mysql2::Client.should respond_to(:default_query_options)
|
120
|
+
end
|
121
|
+
|
122
|
+
it "should be able to connect via SSL options" do
|
123
|
+
ssl = @client.query "SHOW VARIABLES LIKE 'have_ssl'"
|
124
|
+
ssl_uncompiled = ssl.any? {|x| x['Value'] == 'OFF'}
|
125
|
+
pending("DON'T WORRY, THIS TEST PASSES - but SSL is not compiled into your MySQL daemon.") if ssl_uncompiled
|
126
|
+
ssl_disabled = ssl.any? {|x| x['Value'] == 'DISABLED'}
|
127
|
+
pending("DON'T WORRY, THIS TEST PASSES - but SSL is not enabled in your MySQL daemon.") if ssl_disabled
|
128
|
+
|
129
|
+
# You may need to adjust the lines below to match your SSL certificate paths
|
130
|
+
ssl_client = nil
|
131
|
+
lambda {
|
132
|
+
ssl_client = Mysql2::Client.new(
|
133
|
+
:sslkey => '/etc/mysql/client-key.pem',
|
134
|
+
:sslcert => '/etc/mysql/client-cert.pem',
|
135
|
+
:sslca => '/etc/mysql/ca-cert.pem',
|
136
|
+
:sslcapath => '/etc/mysql/',
|
137
|
+
:sslcipher => 'DHE-RSA-AES256-SHA'
|
138
|
+
)
|
139
|
+
}.should_not raise_error(Mysql2::Error)
|
140
|
+
|
141
|
+
results = ssl_client.query("SHOW STATUS WHERE Variable_name = \"Ssl_version\" OR Variable_name = \"Ssl_cipher\"").to_a
|
142
|
+
results[0]['Variable_name'].should eql('Ssl_cipher')
|
143
|
+
results[0]['Value'].should_not be_nil
|
144
|
+
results[0]['Value'].should be_kind_of(String)
|
145
|
+
results[0]['Value'].should_not be_empty
|
146
|
+
|
147
|
+
results[1]['Variable_name'].should eql('Ssl_version')
|
148
|
+
results[1]['Value'].should_not be_nil
|
149
|
+
results[1]['Value'].should be_kind_of(String)
|
150
|
+
results[1]['Value'].should_not be_empty
|
151
|
+
|
152
|
+
ssl_client.close
|
153
|
+
end
|
154
|
+
|
155
|
+
it "should not leave dangling connections after garbage collection" do
|
156
|
+
GC.start
|
157
|
+
sleep 0.300 # Let GC do its work
|
158
|
+
client = Mysql2::Client.new(DatabaseCredentials['root'])
|
159
|
+
before_count = client.query("SHOW STATUS LIKE 'Threads_connected'").first['Value'].to_i
|
160
|
+
|
161
|
+
10.times do
|
162
|
+
Mysql2::Client.new(DatabaseCredentials['root']).query('SELECT 1')
|
163
|
+
end
|
164
|
+
after_count = client.query("SHOW STATUS LIKE 'Threads_connected'").first['Value'].to_i
|
165
|
+
after_count.should == before_count + 10
|
166
|
+
|
167
|
+
GC.start
|
168
|
+
sleep 0.300 # Let GC do its work
|
169
|
+
final_count = client.query("SHOW STATUS LIKE 'Threads_connected'").first['Value'].to_i
|
170
|
+
final_count.should == before_count
|
171
|
+
end
|
172
|
+
|
173
|
+
if Process.respond_to?(:fork)
|
174
|
+
it "should not close connections when running in a child process" do
|
175
|
+
GC.start
|
176
|
+
sleep 1 if defined? Rubinius # Let the rbx GC thread do its work
|
177
|
+
client = Mysql2::Client.new(DatabaseCredentials['root'])
|
178
|
+
|
179
|
+
fork do
|
180
|
+
client.query('SELECT 1')
|
181
|
+
client = nil
|
182
|
+
GC.start
|
183
|
+
sleep 1 if defined? Rubinius # Let the rbx GC thread do its work
|
184
|
+
end
|
185
|
+
|
186
|
+
Process.wait
|
187
|
+
|
188
|
+
# this will throw an error if the underlying socket was shutdown by the
|
189
|
+
# child's GC
|
190
|
+
expect { client.query('SELECT 1') }.to_not raise_exception
|
191
|
+
end
|
192
|
+
end
|
193
|
+
|
194
|
+
it "should be able to connect to database with numeric-only name" do
|
195
|
+
lambda {
|
196
|
+
creds = DatabaseCredentials['numericuser']
|
197
|
+
@client.query "CREATE DATABASE IF NOT EXISTS `#{creds['database']}`"
|
198
|
+
@client.query "GRANT ALL ON `#{creds['database']}`.* TO #{creds['username']}@`#{creds['host']}`"
|
199
|
+
client = Mysql2::Client.new creds
|
200
|
+
@client.query "DROP DATABASE IF EXISTS `#{creds['database']}`"
|
201
|
+
}.should_not raise_error
|
202
|
+
end
|
203
|
+
|
204
|
+
it "should respond to #close" do
|
205
|
+
@client.should respond_to(:close)
|
206
|
+
end
|
207
|
+
|
208
|
+
it "should be able to close properly" do
|
209
|
+
@client.close.should be_nil
|
210
|
+
lambda {
|
211
|
+
@client.query "SELECT 1"
|
212
|
+
}.should raise_error(Mysql2::Error)
|
213
|
+
end
|
214
|
+
|
215
|
+
it "should respond to #query" do
|
216
|
+
@client.should respond_to(:query)
|
217
|
+
end
|
218
|
+
|
219
|
+
it "should respond to #warning_count" do
|
220
|
+
@client.should respond_to(:warning_count)
|
221
|
+
end
|
222
|
+
|
223
|
+
context "#warning_count" do
|
224
|
+
context "when no warnings" do
|
225
|
+
it "should 0" do
|
226
|
+
@client.query('select 1')
|
227
|
+
@client.warning_count.should == 0
|
228
|
+
end
|
229
|
+
end
|
230
|
+
context "when has a warnings" do
|
231
|
+
it "should > 0" do
|
232
|
+
# "the statement produces extra information that can be viewed by issuing a SHOW WARNINGS"
|
233
|
+
# http://dev.mysql.com/doc/refman/5.0/en/explain-extended.html
|
234
|
+
@client.query("explain extended select 1")
|
235
|
+
@client.warning_count.should > 0
|
236
|
+
end
|
237
|
+
end
|
238
|
+
end
|
239
|
+
|
240
|
+
it "should respond to #query_info" do
|
241
|
+
@client.should respond_to(:query_info)
|
242
|
+
end
|
243
|
+
|
244
|
+
context "#query_info" do
|
245
|
+
context "when no info present" do
|
246
|
+
it "should 0" do
|
247
|
+
@client.query('select 1')
|
248
|
+
@client.query_info.should be_empty
|
249
|
+
@client.query_info_string.should be_nil
|
250
|
+
end
|
251
|
+
end
|
252
|
+
context "when has some info" do
|
253
|
+
it "should retrieve it" do
|
254
|
+
@client.query "USE test"
|
255
|
+
@client.query "CREATE TABLE IF NOT EXISTS infoTest (`id` int(11) NOT NULL AUTO_INCREMENT, blah INT(11), PRIMARY KEY (`id`))"
|
256
|
+
|
257
|
+
# http://dev.mysql.com/doc/refman/5.0/en/mysql-info.html says
|
258
|
+
# # Note that mysql_info() returns a non-NULL value for INSERT ... VALUES only for the multiple-row form of the statement (that is, only if multiple value lists are specified).
|
259
|
+
@client.query("INSERT INTO infoTest (blah) VALUES (1234),(4535)")
|
260
|
+
|
261
|
+
@client.query_info.should eql({:records => 2, :duplicates => 0, :warnings => 0})
|
262
|
+
@client.query_info_string.should eq('Records: 2 Duplicates: 0 Warnings: 0')
|
263
|
+
|
264
|
+
@client.query "DROP TABLE infoTest"
|
265
|
+
end
|
266
|
+
end
|
267
|
+
end
|
268
|
+
|
269
|
+
context ":local_infile" do
|
270
|
+
before(:all) do
|
271
|
+
@client_i = Mysql2::Client.new DatabaseCredentials['root'].merge(:local_infile => true)
|
272
|
+
local = @client_i.query "SHOW VARIABLES LIKE 'local_infile'"
|
273
|
+
local_enabled = local.any? {|x| x['Value'] == 'ON'}
|
274
|
+
pending("DON'T WORRY, THIS TEST PASSES - but LOCAL INFILE is not enabled in your MySQL daemon.") unless local_enabled
|
275
|
+
|
276
|
+
@client_i.query %[
|
277
|
+
CREATE TABLE IF NOT EXISTS infileTest (
|
278
|
+
id MEDIUMINT NOT NULL AUTO_INCREMENT PRIMARY KEY,
|
279
|
+
foo VARCHAR(10),
|
280
|
+
bar MEDIUMTEXT
|
281
|
+
)
|
282
|
+
]
|
283
|
+
end
|
284
|
+
|
285
|
+
after(:all) do
|
286
|
+
@client_i.query "DROP TABLE infileTest"
|
287
|
+
end
|
288
|
+
|
289
|
+
it "should raise an error when local_infile is disabled" do
|
290
|
+
client = Mysql2::Client.new DatabaseCredentials['root'].merge(:local_infile => false)
|
291
|
+
lambda {
|
292
|
+
client.query "LOAD DATA LOCAL INFILE 'spec/test_data' INTO TABLE infileTest"
|
293
|
+
}.should raise_error(Mysql2::Error, %r{command is not allowed})
|
294
|
+
end
|
295
|
+
|
296
|
+
it "should raise an error when a non-existent file is loaded" do
|
297
|
+
lambda {
|
298
|
+
@client_i.query "LOAD DATA LOCAL INFILE 'this/file/is/not/here' INTO TABLE infileTest"
|
299
|
+
}.should_not raise_error(Mysql2::Error, %r{file not found: this/file/is/not/here})
|
300
|
+
end
|
301
|
+
|
302
|
+
it "should LOAD DATA LOCAL INFILE" do
|
303
|
+
@client_i.query "LOAD DATA LOCAL INFILE 'spec/test_data' INTO TABLE infileTest"
|
304
|
+
info = @client_i.query_info
|
305
|
+
info.should eql({:records => 1, :deleted => 0, :skipped => 0, :warnings => 0})
|
306
|
+
|
307
|
+
result = @client_i.query "SELECT * FROM infileTest"
|
308
|
+
result.first.should eql({'id' => 1, 'foo' => 'Hello', 'bar' => 'World'})
|
309
|
+
end
|
310
|
+
end
|
311
|
+
|
312
|
+
it "should expect connect_timeout to be a positive integer" do
|
313
|
+
lambda {
|
314
|
+
Mysql2::Client.new(:connect_timeout => -1)
|
315
|
+
}.should raise_error(Mysql2::Error)
|
316
|
+
end
|
317
|
+
|
318
|
+
it "should expect read_timeout to be a positive integer" do
|
319
|
+
lambda {
|
320
|
+
Mysql2::Client.new(:read_timeout => -1)
|
321
|
+
}.should raise_error(Mysql2::Error)
|
322
|
+
end
|
323
|
+
|
324
|
+
it "should expect write_timeout to be a positive integer" do
|
325
|
+
lambda {
|
326
|
+
Mysql2::Client.new(:write_timeout => -1)
|
327
|
+
}.should raise_error(Mysql2::Error)
|
328
|
+
end
|
329
|
+
|
330
|
+
context "#query" do
|
331
|
+
it "should let you query again if iterating is finished when streaming" do
|
332
|
+
@client.query("SELECT 1 UNION SELECT 2", :stream => true, :cache_rows => false).each.to_a
|
333
|
+
|
334
|
+
expect {
|
335
|
+
@client.query("SELECT 1 UNION SELECT 2", :stream => true, :cache_rows => false)
|
336
|
+
}.to_not raise_exception(Mysql2::Error)
|
337
|
+
end
|
338
|
+
|
339
|
+
it "should not let you query again if iterating is not finished when streaming" do
|
340
|
+
@client.query("SELECT 1 UNION SELECT 2", :stream => true, :cache_rows => false).first
|
341
|
+
|
342
|
+
expect {
|
343
|
+
@client.query("SELECT 1 UNION SELECT 2", :stream => true, :cache_rows => false)
|
344
|
+
}.to raise_exception(Mysql2::Error)
|
345
|
+
end
|
346
|
+
|
347
|
+
it "should only accept strings as the query parameter" do
|
348
|
+
lambda {
|
349
|
+
@client.query ["SELECT 'not right'"]
|
350
|
+
}.should raise_error(TypeError)
|
351
|
+
end
|
352
|
+
|
353
|
+
it "should not retain query options set on a query for subsequent queries, but should retain it in the result" do
|
354
|
+
result = @client.query "SELECT 1", :something => :else
|
355
|
+
@client.query_options[:something].should be_nil
|
356
|
+
result.instance_variable_get('@query_options').should eql(@client.query_options.merge(:something => :else))
|
357
|
+
@client.instance_variable_get('@current_query_options').should eql(@client.query_options.merge(:something => :else))
|
358
|
+
|
359
|
+
result = @client.query "SELECT 1"
|
360
|
+
result.instance_variable_get('@query_options').should eql(@client.query_options)
|
361
|
+
@client.instance_variable_get('@current_query_options').should eql(@client.query_options)
|
362
|
+
end
|
363
|
+
|
364
|
+
it "should allow changing query options for subsequent queries" do
|
365
|
+
@client.query_options.merge!(:something => :else)
|
366
|
+
result = @client.query "SELECT 1"
|
367
|
+
@client.query_options[:something].should eql(:else)
|
368
|
+
result.instance_variable_get('@query_options')[:something].should eql(:else)
|
369
|
+
|
370
|
+
# Clean up after this test
|
371
|
+
@client.query_options.delete(:something)
|
372
|
+
@client.query_options[:something].should be_nil
|
373
|
+
end
|
374
|
+
|
375
|
+
it "should return results as a hash by default" do
|
376
|
+
@client.query("SELECT 1").first.class.should eql(Hash)
|
377
|
+
end
|
378
|
+
|
379
|
+
it "should be able to return results as an array" do
|
380
|
+
@client.query("SELECT 1", :as => :array).first.class.should eql(Array)
|
381
|
+
@client.query("SELECT 1").each(:as => :array)
|
382
|
+
end
|
383
|
+
|
384
|
+
it "should be able to return results with symbolized keys" do
|
385
|
+
@client.query("SELECT 1", :symbolize_keys => true).first.keys[0].class.should eql(Symbol)
|
386
|
+
end
|
387
|
+
|
388
|
+
it "should require an open connection" do
|
389
|
+
@client.close
|
390
|
+
lambda {
|
391
|
+
@client.query "SELECT 1"
|
392
|
+
}.should raise_error(Mysql2::Error)
|
393
|
+
end
|
394
|
+
|
395
|
+
if RUBY_PLATFORM !~ /mingw|mswin/
|
396
|
+
it "should not allow another query to be sent without fetching a result first" do
|
397
|
+
@client.query("SELECT 1", :async => true)
|
398
|
+
lambda {
|
399
|
+
@client.query("SELECT 1")
|
400
|
+
}.should raise_error(Mysql2::Error)
|
401
|
+
end
|
402
|
+
|
403
|
+
it "should describe the thread holding the active query" do
|
404
|
+
thr = Thread.new { @client.query("SELECT 1", :async => true) }
|
405
|
+
|
406
|
+
thr.join
|
407
|
+
begin
|
408
|
+
@client.query("SELECT 1")
|
409
|
+
rescue Mysql2::Error => e
|
410
|
+
message = e.message
|
411
|
+
end
|
412
|
+
re = Regexp.escape(thr.inspect)
|
413
|
+
message.should match(Regexp.new(re))
|
414
|
+
end
|
415
|
+
|
416
|
+
it "should timeout if we wait longer than :read_timeout" do
|
417
|
+
client = Mysql2::Client.new(DatabaseCredentials['root'].merge(:read_timeout => 1))
|
418
|
+
lambda {
|
419
|
+
client.query("SELECT sleep(2)")
|
420
|
+
}.should raise_error(Mysql2::Error)
|
421
|
+
end
|
422
|
+
|
423
|
+
if !defined? Rubinius
|
424
|
+
# XXX this test is not deterministic (because Unix signal handling is not)
|
425
|
+
# and may fail on a loaded system
|
426
|
+
it "should run signal handlers while waiting for a response" do
|
427
|
+
mark = {}
|
428
|
+
trap(:USR1) { mark[:USR1] = Time.now }
|
429
|
+
begin
|
430
|
+
mark[:START] = Time.now
|
431
|
+
pid = fork do
|
432
|
+
sleep 1 # wait for client "SELECT sleep(2)" query to start
|
433
|
+
Process.kill(:USR1, Process.ppid)
|
434
|
+
sleep # wait for explicit kill to prevent GC disconnect
|
435
|
+
end
|
436
|
+
@client.query("SELECT sleep(2)")
|
437
|
+
mark[:END] = Time.now
|
438
|
+
mark.include?(:USR1).should be_true
|
439
|
+
(mark[:USR1] - mark[:START]).should >= 1
|
440
|
+
(mark[:USR1] - mark[:START]).should < 1.3
|
441
|
+
(mark[:END] - mark[:USR1]).should > 0.9
|
442
|
+
(mark[:END] - mark[:START]).should >= 2
|
443
|
+
(mark[:END] - mark[:START]).should < 2.3
|
444
|
+
Process.kill(:TERM, pid)
|
445
|
+
Process.waitpid2(pid)
|
446
|
+
ensure
|
447
|
+
trap(:USR1, 'DEFAULT')
|
448
|
+
end
|
449
|
+
end
|
450
|
+
end
|
451
|
+
|
452
|
+
it "#socket should return a Fixnum (file descriptor from C)" do
|
453
|
+
@client.socket.class.should eql(Fixnum)
|
454
|
+
@client.socket.should_not eql(0)
|
455
|
+
end
|
456
|
+
|
457
|
+
it "#socket should require an open connection" do
|
458
|
+
@client.close
|
459
|
+
lambda {
|
460
|
+
@client.socket
|
461
|
+
}.should raise_error(Mysql2::Error)
|
462
|
+
end
|
463
|
+
|
464
|
+
it "should close the connection when an exception is raised" do
|
465
|
+
begin
|
466
|
+
Timeout.timeout(1, Timeout::Error) do
|
467
|
+
@client.query("SELECT sleep(2)")
|
468
|
+
end
|
469
|
+
rescue Timeout::Error
|
470
|
+
end
|
471
|
+
|
472
|
+
lambda {
|
473
|
+
@client.query("SELECT 1")
|
474
|
+
}.should raise_error(Mysql2::Error, 'closed MySQL connection')
|
475
|
+
end
|
476
|
+
|
477
|
+
it "should handle Timeouts without leaving the connection hanging if reconnect is true" do
|
478
|
+
client = Mysql2::Client.new(DatabaseCredentials['root'].merge(:reconnect => true))
|
479
|
+
begin
|
480
|
+
Timeout.timeout(1, Timeout::Error) do
|
481
|
+
client.query("SELECT sleep(2)")
|
482
|
+
end
|
483
|
+
rescue Timeout::Error
|
484
|
+
end
|
485
|
+
|
486
|
+
lambda {
|
487
|
+
client.query("SELECT 1")
|
488
|
+
}.should_not raise_error(Mysql2::Error)
|
489
|
+
end
|
490
|
+
|
491
|
+
it "should handle Timeouts without leaving the connection hanging if reconnect is set to true after construction true" do
|
492
|
+
client = Mysql2::Client.new(DatabaseCredentials['root'])
|
493
|
+
begin
|
494
|
+
Timeout.timeout(1, Timeout::Error) do
|
495
|
+
client.query("SELECT sleep(2)")
|
496
|
+
end
|
497
|
+
rescue Timeout::Error
|
498
|
+
end
|
499
|
+
|
500
|
+
lambda {
|
501
|
+
client.query("SELECT 1")
|
502
|
+
}.should raise_error(Mysql2::Error)
|
503
|
+
|
504
|
+
client.reconnect = true
|
505
|
+
|
506
|
+
begin
|
507
|
+
Timeout.timeout(1, Timeout::Error) do
|
508
|
+
client.query("SELECT sleep(2)")
|
509
|
+
end
|
510
|
+
rescue Timeout::Error
|
511
|
+
end
|
512
|
+
|
513
|
+
lambda {
|
514
|
+
client.query("SELECT 1")
|
515
|
+
}.should_not raise_error(Mysql2::Error)
|
516
|
+
|
517
|
+
end
|
518
|
+
|
519
|
+
it "threaded queries should be supported" do
|
520
|
+
threads, results = [], {}
|
521
|
+
lock = Mutex.new
|
522
|
+
connect = lambda{
|
523
|
+
Mysql2::Client.new(DatabaseCredentials['root'])
|
524
|
+
}
|
525
|
+
Timeout.timeout(0.7) do
|
526
|
+
5.times {
|
527
|
+
threads << Thread.new do
|
528
|
+
result = connect.call.query("SELECT sleep(0.5) as result")
|
529
|
+
lock.synchronize do
|
530
|
+
results[Thread.current.object_id] = result
|
531
|
+
end
|
532
|
+
end
|
533
|
+
}
|
534
|
+
end
|
535
|
+
threads.each{|t| t.join }
|
536
|
+
results.keys.sort.should eql(threads.map{|t| t.object_id }.sort)
|
537
|
+
end
|
538
|
+
|
539
|
+
it "evented async queries should be supported" do
|
540
|
+
# should immediately return nil
|
541
|
+
@client.query("SELECT sleep(0.1)", :async => true).should eql(nil)
|
542
|
+
|
543
|
+
io_wrapper = IO.for_fd(@client.socket)
|
544
|
+
loops = 0
|
545
|
+
loop do
|
546
|
+
if IO.select([io_wrapper], nil, nil, 0.05)
|
547
|
+
break
|
548
|
+
else
|
549
|
+
loops += 1
|
550
|
+
end
|
551
|
+
end
|
552
|
+
|
553
|
+
# make sure we waited some period of time
|
554
|
+
(loops >= 1).should be_true
|
555
|
+
|
556
|
+
result = @client.async_result
|
557
|
+
result.class.should eql(Mysql2::Result)
|
558
|
+
end
|
559
|
+
end
|
560
|
+
|
561
|
+
context "Multiple results sets" do
|
562
|
+
before(:each) do
|
563
|
+
@multi_client = Mysql2::Client.new(DatabaseCredentials['root'].merge(:flags => Mysql2::Client::MULTI_STATEMENTS))
|
564
|
+
end
|
565
|
+
|
566
|
+
it "should raise an exception when one of multiple statements fails" do
|
567
|
+
result = @multi_client.query("SELECT 1 AS 'set_1'; SELECT * FROM invalid_table_name; SELECT 2 AS 'set_2';")
|
568
|
+
result.first['set_1'].should be(1)
|
569
|
+
lambda {
|
570
|
+
@multi_client.next_result
|
571
|
+
}.should raise_error(Mysql2::Error)
|
572
|
+
@multi_client.next_result.should be_false
|
573
|
+
end
|
574
|
+
|
575
|
+
it "returns multiple result sets" do
|
576
|
+
@multi_client.query("SELECT 1 AS 'set_1'; SELECT 2 AS 'set_2'").first.should eql({ 'set_1' => 1 })
|
577
|
+
|
578
|
+
@multi_client.next_result.should be_true
|
579
|
+
@multi_client.store_result.first.should eql({ 'set_2' => 2 })
|
580
|
+
|
581
|
+
@multi_client.next_result.should be_false
|
582
|
+
end
|
583
|
+
|
584
|
+
it "does not interfere with other statements" do
|
585
|
+
@multi_client.query("SELECT 1 AS 'set_1'; SELECT 2 AS 'set_2'")
|
586
|
+
while( @multi_client.next_result )
|
587
|
+
@multi_client.store_result
|
588
|
+
end
|
589
|
+
|
590
|
+
@multi_client.query("SELECT 3 AS 'next'").first.should == { 'next' => 3 }
|
591
|
+
end
|
592
|
+
|
593
|
+
it "will raise on query if there are outstanding results to read" do
|
594
|
+
@multi_client.query("SELECT 1; SELECT 2; SELECT 3")
|
595
|
+
lambda {
|
596
|
+
@multi_client.query("SELECT 4")
|
597
|
+
}.should raise_error(Mysql2::Error)
|
598
|
+
end
|
599
|
+
|
600
|
+
it "#abandon_results! should work" do
|
601
|
+
@multi_client.query("SELECT 1; SELECT 2; SELECT 3")
|
602
|
+
@multi_client.abandon_results!
|
603
|
+
lambda {
|
604
|
+
@multi_client.query("SELECT 4")
|
605
|
+
}.should_not raise_error(Mysql2::Error)
|
606
|
+
end
|
607
|
+
|
608
|
+
it "#more_results? should work" do
|
609
|
+
@multi_client.query("SELECT 1 AS 'set_1'; SELECT 2 AS 'set_2'")
|
610
|
+
@multi_client.more_results?.should be_true
|
611
|
+
|
612
|
+
@multi_client.next_result
|
613
|
+
@multi_client.store_result
|
614
|
+
|
615
|
+
@multi_client.more_results?.should be_false
|
616
|
+
end
|
617
|
+
end
|
618
|
+
end
|
619
|
+
|
620
|
+
it "should respond to #socket" do
|
621
|
+
@client.should respond_to(:socket)
|
622
|
+
end
|
623
|
+
|
624
|
+
if RUBY_PLATFORM =~ /mingw|mswin/
|
625
|
+
it "#socket should raise as it's not supported" do
|
626
|
+
lambda {
|
627
|
+
@client.socket
|
628
|
+
}.should raise_error(Mysql2::Error)
|
629
|
+
end
|
630
|
+
end
|
631
|
+
|
632
|
+
it "should respond to escape" do
|
633
|
+
Mysql2::Client.should respond_to(:escape)
|
634
|
+
end
|
635
|
+
|
636
|
+
context "escape" do
|
637
|
+
it "should return a new SQL-escape version of the passed string" do
|
638
|
+
Mysql2::Client.escape("abc'def\"ghi\0jkl%mno").should eql("abc\\'def\\\"ghi\\0jkl%mno")
|
639
|
+
end
|
640
|
+
|
641
|
+
it "should return the passed string if nothing was escaped" do
|
642
|
+
str = "plain"
|
643
|
+
Mysql2::Client.escape(str).object_id.should eql(str.object_id)
|
644
|
+
end
|
645
|
+
|
646
|
+
it "should not overflow the thread stack" do
|
647
|
+
lambda {
|
648
|
+
Thread.new { Mysql2::Client.escape("'" * 256 * 1024) }.join
|
649
|
+
}.should_not raise_error(SystemStackError)
|
650
|
+
end
|
651
|
+
|
652
|
+
it "should not overflow the process stack" do
|
653
|
+
lambda {
|
654
|
+
Thread.new { Mysql2::Client.escape("'" * 1024 * 1024 * 4) }.join
|
655
|
+
}.should_not raise_error(SystemStackError)
|
656
|
+
end
|
657
|
+
|
658
|
+
unless RUBY_VERSION =~ /1.8/
|
659
|
+
it "should carry over the original string's encoding" do
|
660
|
+
str = "abc'def\"ghi\0jkl%mno"
|
661
|
+
escaped = Mysql2::Client.escape(str)
|
662
|
+
escaped.encoding.should eql(str.encoding)
|
663
|
+
|
664
|
+
str.encode!('us-ascii')
|
665
|
+
escaped = Mysql2::Client.escape(str)
|
666
|
+
escaped.encoding.should eql(str.encoding)
|
667
|
+
end
|
668
|
+
end
|
669
|
+
end
|
670
|
+
|
671
|
+
it "should respond to #escape" do
|
672
|
+
@client.should respond_to(:escape)
|
673
|
+
end
|
674
|
+
|
675
|
+
context "#escape" do
|
676
|
+
it "should return a new SQL-escape version of the passed string" do
|
677
|
+
@client.escape("abc'def\"ghi\0jkl%mno").should eql("abc\\'def\\\"ghi\\0jkl%mno")
|
678
|
+
end
|
679
|
+
|
680
|
+
it "should return the passed string if nothing was escaped" do
|
681
|
+
str = "plain"
|
682
|
+
@client.escape(str).object_id.should eql(str.object_id)
|
683
|
+
end
|
684
|
+
|
685
|
+
it "should not overflow the thread stack" do
|
686
|
+
lambda {
|
687
|
+
Thread.new { @client.escape("'" * 256 * 1024) }.join
|
688
|
+
}.should_not raise_error(SystemStackError)
|
689
|
+
end
|
690
|
+
|
691
|
+
it "should not overflow the process stack" do
|
692
|
+
lambda {
|
693
|
+
Thread.new { @client.escape("'" * 1024 * 1024 * 4) }.join
|
694
|
+
}.should_not raise_error(SystemStackError)
|
695
|
+
end
|
696
|
+
|
697
|
+
it "should require an open connection" do
|
698
|
+
@client.close
|
699
|
+
lambda {
|
700
|
+
@client.escape ""
|
701
|
+
}.should raise_error(Mysql2::Error)
|
702
|
+
end
|
703
|
+
end
|
704
|
+
|
705
|
+
it "should respond to #info" do
|
706
|
+
@client.should respond_to(:info)
|
707
|
+
end
|
708
|
+
|
709
|
+
it "#info should return a hash containing the client version ID and String" do
|
710
|
+
info = @client.info
|
711
|
+
info.class.should eql(Hash)
|
712
|
+
info.should have_key(:id)
|
713
|
+
info[:id].class.should eql(Fixnum)
|
714
|
+
info.should have_key(:version)
|
715
|
+
info[:version].class.should eql(String)
|
716
|
+
end
|
717
|
+
|
718
|
+
if defined? Encoding
|
719
|
+
context "strings returned by #info" do
|
720
|
+
it "should default to the connection's encoding if Encoding.default_internal is nil" do
|
721
|
+
with_internal_encoding nil do
|
722
|
+
@client.info[:version].encoding.should eql(Encoding.find('utf-8'))
|
723
|
+
|
724
|
+
client2 = Mysql2::Client.new(DatabaseCredentials['root'].merge(:encoding => 'ascii'))
|
725
|
+
client2.info[:version].encoding.should eql(Encoding.find('us-ascii'))
|
726
|
+
end
|
727
|
+
end
|
728
|
+
|
729
|
+
it "should use Encoding.default_internal" do
|
730
|
+
with_internal_encoding 'utf-8' do
|
731
|
+
@client.info[:version].encoding.should eql(Encoding.default_internal)
|
732
|
+
end
|
733
|
+
|
734
|
+
with_internal_encoding 'us-ascii' do
|
735
|
+
@client.info[:version].encoding.should eql(Encoding.default_internal)
|
736
|
+
end
|
737
|
+
end
|
738
|
+
end
|
739
|
+
end
|
740
|
+
|
741
|
+
it "should respond to #server_info" do
|
742
|
+
@client.should respond_to(:server_info)
|
743
|
+
end
|
744
|
+
|
745
|
+
it "#server_info should return a hash containing the client version ID and String" do
|
746
|
+
server_info = @client.server_info
|
747
|
+
server_info.class.should eql(Hash)
|
748
|
+
server_info.should have_key(:id)
|
749
|
+
server_info[:id].class.should eql(Fixnum)
|
750
|
+
server_info.should have_key(:version)
|
751
|
+
server_info[:version].class.should eql(String)
|
752
|
+
end
|
753
|
+
|
754
|
+
it "#server_info should require an open connection" do
|
755
|
+
@client.close
|
756
|
+
lambda {
|
757
|
+
@client.server_info
|
758
|
+
}.should raise_error(Mysql2::Error)
|
759
|
+
end
|
760
|
+
|
761
|
+
if defined? Encoding
|
762
|
+
context "strings returned by #server_info" do
|
763
|
+
it "should default to the connection's encoding if Encoding.default_internal is nil" do
|
764
|
+
with_internal_encoding nil do
|
765
|
+
@client.server_info[:version].encoding.should eql(Encoding.find('utf-8'))
|
766
|
+
|
767
|
+
client2 = Mysql2::Client.new(DatabaseCredentials['root'].merge(:encoding => 'ascii'))
|
768
|
+
client2.server_info[:version].encoding.should eql(Encoding.find('us-ascii'))
|
769
|
+
end
|
770
|
+
end
|
771
|
+
|
772
|
+
it "should use Encoding.default_internal" do
|
773
|
+
with_internal_encoding 'utf-8' do
|
774
|
+
@client.server_info[:version].encoding.should eql(Encoding.default_internal)
|
775
|
+
end
|
776
|
+
|
777
|
+
with_internal_encoding 'us-ascii' do
|
778
|
+
@client.server_info[:version].encoding.should eql(Encoding.default_internal)
|
779
|
+
end
|
780
|
+
end
|
781
|
+
end
|
782
|
+
end
|
783
|
+
|
784
|
+
it "should raise a Mysql2::Error exception upon connection failure" do
|
785
|
+
lambda {
|
786
|
+
Mysql2::Client.new :host => "localhost", :username => 'asdfasdf8d2h', :password => 'asdfasdfw42'
|
787
|
+
}.should raise_error(Mysql2::Error)
|
788
|
+
|
789
|
+
lambda {
|
790
|
+
Mysql2::Client.new DatabaseCredentials['root']
|
791
|
+
}.should_not raise_error(Mysql2::Error)
|
792
|
+
end
|
793
|
+
|
794
|
+
context 'write operations api' do
|
795
|
+
before(:each) do
|
796
|
+
@client.query "USE test"
|
797
|
+
@client.query "CREATE TABLE IF NOT EXISTS lastIdTest (`id` BIGINT NOT NULL AUTO_INCREMENT, blah INT(11), PRIMARY KEY (`id`))"
|
798
|
+
end
|
799
|
+
|
800
|
+
after(:each) do
|
801
|
+
@client.query "DROP TABLE lastIdTest"
|
802
|
+
end
|
803
|
+
|
804
|
+
it "should respond to #last_id" do
|
805
|
+
@client.should respond_to(:last_id)
|
806
|
+
end
|
807
|
+
|
808
|
+
it "#last_id should return a Fixnum, the from the last INSERT/UPDATE" do
|
809
|
+
@client.last_id.should eql(0)
|
810
|
+
@client.query "INSERT INTO lastIdTest (blah) VALUES (1234)"
|
811
|
+
@client.last_id.should eql(1)
|
812
|
+
end
|
813
|
+
|
814
|
+
it "should respond to #last_id" do
|
815
|
+
@client.should respond_to(:last_id)
|
816
|
+
end
|
817
|
+
|
818
|
+
it "#last_id should return a Fixnum, the from the last INSERT/UPDATE" do
|
819
|
+
@client.query "INSERT INTO lastIdTest (blah) VALUES (1234)"
|
820
|
+
@client.affected_rows.should eql(1)
|
821
|
+
@client.query "UPDATE lastIdTest SET blah=4321 WHERE id=1"
|
822
|
+
@client.affected_rows.should eql(1)
|
823
|
+
end
|
824
|
+
|
825
|
+
it "#last_id should handle BIGINT auto-increment ids above 32 bits" do
|
826
|
+
# The id column type must be BIGINT. Surprise: INT(x) is limited to 32-bits for all values of x.
|
827
|
+
# Insert a row with a given ID, this should raise the auto-increment state
|
828
|
+
@client.query "INSERT INTO lastIdTest (id, blah) VALUES (5000000000, 5000)"
|
829
|
+
@client.last_id.should eql(5000000000)
|
830
|
+
@client.query "INSERT INTO lastIdTest (blah) VALUES (5001)"
|
831
|
+
@client.last_id.should eql(5000000001)
|
832
|
+
end
|
833
|
+
end
|
834
|
+
|
835
|
+
it "should respond to #thread_id" do
|
836
|
+
@client.should respond_to(:thread_id)
|
837
|
+
end
|
838
|
+
|
839
|
+
it "#thread_id should be a Fixnum" do
|
840
|
+
@client.thread_id.class.should eql(Fixnum)
|
841
|
+
end
|
842
|
+
|
843
|
+
it "should respond to #ping" do
|
844
|
+
@client.should respond_to(:ping)
|
845
|
+
end
|
846
|
+
|
847
|
+
context "select_db" do
|
848
|
+
before(:each) do
|
849
|
+
2.times do |i|
|
850
|
+
@client.query("CREATE DATABASE test_selectdb_#{i}")
|
851
|
+
@client.query("USE test_selectdb_#{i}")
|
852
|
+
@client.query("CREATE TABLE test#{i} (`id` int NOT NULL PRIMARY KEY)")
|
853
|
+
end
|
854
|
+
end
|
855
|
+
|
856
|
+
after(:each) do
|
857
|
+
2.times do |i|
|
858
|
+
@client.query("DROP DATABASE test_selectdb_#{i}")
|
859
|
+
end
|
860
|
+
end
|
861
|
+
|
862
|
+
it "should respond to #select_db" do
|
863
|
+
@client.should respond_to(:select_db)
|
864
|
+
end
|
865
|
+
|
866
|
+
it "should switch databases" do
|
867
|
+
@client.select_db("test_selectdb_0")
|
868
|
+
@client.query("SHOW TABLES").first.values.first.should eql("test0")
|
869
|
+
@client.select_db("test_selectdb_1")
|
870
|
+
@client.query("SHOW TABLES").first.values.first.should eql("test1")
|
871
|
+
@client.select_db("test_selectdb_0")
|
872
|
+
@client.query("SHOW TABLES").first.values.first.should eql("test0")
|
873
|
+
end
|
874
|
+
|
875
|
+
it "should raise a Mysql2::Error when the database doesn't exist" do
|
876
|
+
lambda {
|
877
|
+
@client.select_db("nopenothere")
|
878
|
+
}.should raise_error(Mysql2::Error)
|
879
|
+
end
|
880
|
+
|
881
|
+
it "should return the database switched to" do
|
882
|
+
@client.select_db("test_selectdb_1").should eq("test_selectdb_1")
|
883
|
+
end
|
884
|
+
end
|
885
|
+
|
886
|
+
it "#thread_id should return a boolean" do
|
887
|
+
@client.ping.should eql(true)
|
888
|
+
@client.close
|
889
|
+
@client.ping.should eql(false)
|
890
|
+
end
|
891
|
+
|
892
|
+
unless RUBY_VERSION =~ /1.8/
|
893
|
+
it "should respond to #encoding" do
|
894
|
+
@client.should respond_to(:encoding)
|
895
|
+
end
|
896
|
+
end
|
897
|
+
end
|