]>
git.saurik.com Git - redis.git/blob - client-libraries/ruby/spec/redis_spec.rb
1 require File
.dirname(__FILE__
) +
'/spec_helper'
17 # use database 15 for testing so we dont accidentally step on you real data
18 @r = Redis
.new
:db => 15
26 @r.keys('*').each
{|k
| @r.del k
}
33 it
"should be able connect without a timeout" do
34 lambda
{ Redis
.new
:timeout => 0 }.should_not raise_error
37 it
"should be able to provide a logger" do
39 r
= Redis
.new
:db => 15, :logger => Logger
.new(log
)
41 log
.string
.should
include("ping")
44 it
"should be able to PING" do
45 @r.ping
.should
== 'PONG'
48 it
"should be able to GET a key" do
49 @r['foo'].should
== 'bar'
52 it
"should be able to SET a key" do
54 @r['foo'].should
== 'nik'
57 it
"should properly handle trailing newline characters" do
59 @r['foo'].should
== "bar\n"
62 it
"should store and retrieve all possible characters at the beginning and the end of a string" do
63 (0..255).each
do |char_idx
|
64 string
= "#{char_idx.chr}---#{char_idx.chr}"
66 @r['foo'].should
== string
70 it
"should be able to SET a key with an expiry" do
71 @r.set('foo', 'bar', 1)
72 @r['foo'].should
== 'bar'
74 @r['foo'].should
== nil
77 it
"should be able to return a TTL for a key" do
78 @r.set('foo', 'bar', 1)
79 @r.ttl('foo').should
== 1
82 it
"should be able to SETNX" do
84 @r['foo'].should
== 'nik'
86 @r['foo'].should
== 'nik'
89 it
"should be able to GETSET" do
90 @r.getset('foo', 'baz').should
== 'bar'
91 @r['foo'].should
== 'baz'
94 it
"should be able to INCR a key" do
96 @r.incr('counter').should
== 1
97 @r.incr('counter').should
== 2
98 @r.incr('counter').should
== 3
101 it
"should be able to INCRBY a key" do
103 @r.incrby('counter', 1).should
== 1
104 @r.incrby('counter', 2).should
== 3
105 @r.incrby('counter', 3).should
== 6
108 it
"should be able to DECR a key" do
110 @r.incr('counter').should
== 1
111 @r.incr('counter').should
== 2
112 @r.incr('counter').should
== 3
113 @r.decr('counter').should
== 2
114 @r.decr('counter', 2).should
== 0
117 it
"should be able to RANDKEY" do
118 @r.randkey
.should_not be_nil
121 it
"should be able to RENAME a key" do
125 @r.rename
'foo', 'bar'
126 @r['bar'].should
== 'hi'
129 it
"should be able to RENAMENX a key" do
134 @r.renamenx
'foo', 'bar'
135 @r['bar'].should
== 'ohai'
138 it
"should be able to get DBSIZE of the database" do
140 dbsize_without_foo
= @r.dbsize
142 dbsize_with_foo
= @r.dbsize
144 dbsize_with_foo
.should
== dbsize_without_foo +
1
147 it
"should be able to EXPIRE a key" do
150 @r['foo'].should
== "bar"
152 @r['foo'].should
== nil
155 it
"should be able to EXISTS" do
157 @r.exists('foo').should be_true
159 @r.exists('foo').should be_false
162 it
"should be able to KEYS" do
163 @r.keys("f*").each
{ |key
| @r.del key
}
167 @r.keys("f*").sort
.should
== ['f','fo', 'foo'].sort
170 it
"should be able to return a random key (RANDOMKEY)" do
171 3.times
{ @r.exists(@r.randomkey
).should be_true
}
174 it
"should be able to check the TYPE of a key" do
176 @r.type('foo').should
== "string"
178 @r.type('foo').should
== "none"
181 it
"should be able to push to the head of a list (LPUSH)" do
182 @r.lpush
"list", 'hello'
184 @r.type('list').should
== "list"
185 @r.llen('list').should
== 2
186 @r.lpop('list').should
== '42'
189 it
"should be able to push to the tail of a list (RPUSH)" do
190 @r.rpush
"list", 'hello'
191 @r.type('list').should
== "list"
192 @r.llen('list').should
== 1
195 it
"should be able to pop the tail of a list (RPOP)" do
196 @r.rpush
"list", 'hello'
197 @r.rpush
"list", 'goodbye'
198 @r.type('list').should
== "list"
199 @r.llen('list').should
== 2
200 @r.rpop('list').should
== 'goodbye'
203 it
"should be able to pop the head of a list (LPOP)" do
204 @r.rpush
"list", 'hello'
205 @r.rpush
"list", 'goodbye'
206 @r.type('list').should
== "list"
207 @r.llen('list').should
== 2
208 @r.lpop('list').should
== 'hello'
211 it
"should be able to get the length of a list (LLEN)" do
212 @r.rpush
"list", 'hello'
213 @r.rpush
"list", 'goodbye'
214 @r.type('list').should
== "list"
215 @r.llen('list').should
== 2
218 it
"should be able to get a range of values from a list (LRANGE)" do
219 @r.rpush
"list", 'hello'
220 @r.rpush
"list", 'goodbye'
224 @r.type('list').should
== "list"
225 @r.llen('list').should
== 5
226 @r.lrange('list', 2, -1).should
== ['1', '2', '3']
229 it
"should be able to trim a list (LTRIM)" do
230 @r.rpush
"list", 'hello'
231 @r.rpush
"list", 'goodbye'
235 @r.type('list').should
== "list"
236 @r.llen('list').should
== 5
237 @r.ltrim
'list', 0, 1
238 @r.llen('list').should
== 2
239 @r.lrange('list', 0, -1).should
== ['hello', 'goodbye']
242 it
"should be able to get a value by indexing into a list (LINDEX)" do
243 @r.rpush
"list", 'hello'
244 @r.rpush
"list", 'goodbye'
245 @r.type('list').should
== "list"
246 @r.llen('list').should
== 2
247 @r.lindex('list', 1).should
== 'goodbye'
250 it
"should be able to set a value by indexing into a list (LSET)" do
251 @r.rpush
"list", 'hello'
252 @r.rpush
"list", 'hello'
253 @r.type('list').should
== "list"
254 @r.llen('list').should
== 2
255 @r.lset('list', 1, 'goodbye').should
== 'OK'
256 @r.lindex('list', 1).should
== 'goodbye'
259 it
"should be able to remove values from a list (LREM)" do
260 @r.rpush
"list", 'hello'
261 @r.rpush
"list", 'goodbye'
262 @r.type('list').should
== "list"
263 @r.llen('list').should
== 2
264 @r.lrem('list', 1, 'hello').should
== 1
265 @r.lrange('list', 0, -1).should
== ['goodbye']
268 it
"should be able add members to a set (SADD)" do
269 @r.sadd
"set", 'key1'
270 @r.sadd
"set", 'key2'
271 @r.type('set').should
== "set"
272 @r.scard('set').should
== 2
273 @r.smembers('set').sort
.should
== ['key1', 'key2'].sort
276 it
"should be able delete members to a set (SREM)" do
277 @r.sadd
"set", 'key1'
278 @r.sadd
"set", 'key2'
279 @r.type('set').should
== "set"
280 @r.scard('set').should
== 2
281 @r.smembers('set').sort
.should
== ['key1', 'key2'].sort
282 @r.srem('set', 'key1')
283 @r.scard('set').should
== 1
284 @r.smembers('set').should
== ['key2']
287 it
"should be able count the members of a set (SCARD)" do
288 @r.sadd
"set", 'key1'
289 @r.sadd
"set", 'key2'
290 @r.type('set').should
== "set"
291 @r.scard('set').should
== 2
294 it
"should be able test for set membership (SISMEMBER)" do
295 @r.sadd
"set", 'key1'
296 @r.sadd
"set", 'key2'
297 @r.type('set').should
== "set"
298 @r.scard('set').should
== 2
299 @r.sismember('set', 'key1').should be_true
300 @r.sismember('set', 'key2').should be_true
301 @r.sismember('set', 'notthere').should be_false
304 it
"should be able to do set intersection (SINTER)" do
305 @r.sadd
"set", 'key1'
306 @r.sadd
"set", 'key2'
307 @r.sadd
"set2", 'key2'
308 @r.sinter('set', 'set2').should
== ['key2']
311 it
"should be able to do set intersection and store the results in a key (SINTERSTORE)" do
312 @r.sadd
"set", 'key1'
313 @r.sadd
"set", 'key2'
314 @r.sadd
"set2", 'key2'
315 @r.sinterstore('newone', 'set', 'set2').should
== 1
316 @r.smembers('newone').should
== ['key2']
319 it
"should be able to do set union (SUNION)" do
320 @r.sadd
"set", 'key1'
321 @r.sadd
"set", 'key2'
322 @r.sadd
"set2", 'key2'
323 @r.sadd
"set2", 'key3'
324 @r.sunion('set', 'set2').sort
.should
== ['key1','key2','key3'].sort
327 it
"should be able to do set union and store the results in a key (SUNIONSTORE)" do
328 @r.sadd
"set", 'key1'
329 @r.sadd
"set", 'key2'
330 @r.sadd
"set2", 'key2'
331 @r.sadd
"set2", 'key3'
332 @r.sunionstore('newone', 'set', 'set2').should
== 3
333 @r.smembers('newone').sort
.should
== ['key1','key2','key3'].sort
336 it
"should be able to do set difference (SDIFF)" do
341 @r.sdiff('set', 'set2').should
== ['a']
344 it
"should be able to do set difference and store the results in a key (SDIFFSTORE)" do
349 @r.sdiffstore('newone', 'set', 'set2')
350 @r.smembers('newone').should
== ['a']
353 it
"should be able move elements from one set to another (SMOVE)" do
357 @r.smove('set1', 'set2', 'a').should be_true
358 @r.sismember('set2', 'a').should be_true
362 it
"should be able to do crazy SORT queries" do
363 # The 'Dogs' is capitialized on purpose
364 @r['dog_1'] = 'louie'
372 @r.sort('Dogs', :get => 'dog_*', :limit => [0,1]).should
== ['louie']
373 @r.sort('Dogs', :get => 'dog_*', :limit => [0,1], :order => 'desc alpha').should
== ['taj']
376 it
"should be able to handle array of :get using SORT" do
377 @r['dog:1:name'] = 'louie'
378 @r['dog:1:breed'] = 'mutt'
380 @r['dog:2:name'] = 'lucy'
381 @r['dog:2:breed'] = 'poodle'
383 @r['dog:3:name'] = 'max'
384 @r['dog:3:breed'] = 'hound'
386 @r['dog:4:name'] = 'taj'
387 @r['dog:4:breed'] = 'terrier'
389 @r.sort('dogs', :get => ['dog:*:name', 'dog:*:breed'], :limit => [0,1]).should
== ['louie', 'mutt']
390 @r.sort('dogs', :get => ['dog:*:name', 'dog:*:breed'], :limit => [0,1], :order => 'desc alpha').should
== ['taj', 'terrier']
393 it
"should provide info (INFO)" do
394 [:last_save_time, :redis_version, :total_connections_received, :connected_clients, :total_commands_processed, :connected_slaves, :uptime_in_seconds, :used_memory, :uptime_in_days, :changes_since_last_save].each
do |x
|
395 @r.info
.keys
.should
include(x
)
399 it
"should be able to flush the database (FLUSHDB)" do
400 @r['key1'] = 'keyone'
401 @r['key2'] = 'keytwo'
402 @r.keys('*').sort
.should
== ['foo', 'key1', 'key2'].sort
#foo from before
404 @r.keys('*').should
== []
407 it
"should raise exception when manually try to change the database" do
408 lambda
{ @r.select(0) }.should raise_error
411 it
"should be able to provide the last save time (LASTSAVE)" do
412 savetime
= @r.lastsave
413 Time
.at(savetime
).class.should
== Time
414 Time
.at(savetime
).should
<= Time
.now
417 it
"should be able to MGET keys" do
420 @r.mget('foo', 'bar').should
== ['1000', '2000']
421 @r.mget('foo', 'bar', 'baz').should
== ['1000', '2000', nil]
424 it
"should be able to mapped MGET keys" do
427 @r.mapped_mget('foo', 'bar').should
== { 'foo' => '1000', 'bar' => '2000'}
428 @r.mapped_mget('foo', 'baz', 'bar').should
== { 'foo' => '1000', 'bar' => '2000'}
431 it
"should bgsave" do
432 @r.bgsave
.should
== 'OK'
435 it
"should be able to ECHO" do
436 @r.echo("message in a bottle\n").should
== "message in a bottle\n"
439 it
"should raise error when invoke MONITOR" do
440 lambda
{ @r.monitor
}.should raise_error
443 it
"should raise error when invoke SYNC" do
444 lambda
{ @r.sync
}.should raise_error
447 it
"should handle multiple servers" do
449 @r = DistRedis
.new(:hosts=> ['localhost:6379', '127.0.0.1:6379'], :db => 15)
452 @r[idx
] = "foo#{idx}"
456 @r[idx
].should
== "foo#{idx}"
460 it
"should be able to pipeline writes" do
461 @r.pipelined
do |pipeline
|
462 pipeline
.lpush
'list', "hello"
463 pipeline
.lpush
'list', 42
466 @r.type('list').should
== "list"
467 @r.llen('list').should
== 2
468 @r.lpop('list').should
== '42'
471 it
"should AUTH when connecting with a password" do
472 r
= Redis
.new(:password => 'secret')
474 r
.should_receive(:call_command).with(['auth', 'secret'])