redis 2.1.1 → 2.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (79) hide show
  1. data/.gitignore +8 -0
  2. data/CHANGELOG.md +34 -0
  3. data/README.md +190 -0
  4. data/Rakefile +194 -79
  5. data/benchmarking/logging.rb +62 -0
  6. data/benchmarking/pipeline.rb +51 -0
  7. data/benchmarking/speed.rb +21 -0
  8. data/benchmarking/suite.rb +24 -0
  9. data/benchmarking/thread_safety.rb +38 -0
  10. data/benchmarking/worker.rb +71 -0
  11. data/examples/basic.rb +15 -0
  12. data/examples/dist_redis.rb +43 -0
  13. data/examples/incr-decr.rb +17 -0
  14. data/examples/list.rb +26 -0
  15. data/examples/pubsub.rb +31 -0
  16. data/examples/sets.rb +36 -0
  17. data/examples/unicorn/config.ru +3 -0
  18. data/examples/unicorn/unicorn.rb +20 -0
  19. data/lib/redis.rb +612 -156
  20. data/lib/redis/client.rb +98 -57
  21. data/lib/redis/connection.rb +9 -134
  22. data/lib/redis/connection/command_helper.rb +45 -0
  23. data/lib/redis/connection/hiredis.rb +49 -0
  24. data/lib/redis/connection/registry.rb +12 -0
  25. data/lib/redis/connection/ruby.rb +131 -0
  26. data/lib/redis/connection/synchrony.rb +125 -0
  27. data/lib/redis/distributed.rb +161 -5
  28. data/lib/redis/pipeline.rb +6 -0
  29. data/lib/redis/version.rb +3 -0
  30. data/redis.gemspec +24 -0
  31. data/test/commands_on_hashes_test.rb +32 -0
  32. data/test/commands_on_lists_test.rb +60 -0
  33. data/test/commands_on_sets_test.rb +78 -0
  34. data/test/commands_on_sorted_sets_test.rb +109 -0
  35. data/test/commands_on_strings_test.rb +80 -0
  36. data/test/commands_on_value_types_test.rb +88 -0
  37. data/test/connection_handling_test.rb +87 -0
  38. data/test/db/.gitignore +1 -0
  39. data/test/distributed_blocking_commands_test.rb +53 -0
  40. data/test/distributed_commands_on_hashes_test.rb +12 -0
  41. data/test/distributed_commands_on_lists_test.rb +24 -0
  42. data/test/distributed_commands_on_sets_test.rb +85 -0
  43. data/test/distributed_commands_on_strings_test.rb +50 -0
  44. data/test/distributed_commands_on_value_types_test.rb +73 -0
  45. data/test/distributed_commands_requiring_clustering_test.rb +148 -0
  46. data/test/distributed_connection_handling_test.rb +25 -0
  47. data/test/distributed_internals_test.rb +18 -0
  48. data/test/distributed_key_tags_test.rb +53 -0
  49. data/test/distributed_persistence_control_commands_test.rb +24 -0
  50. data/test/distributed_publish_subscribe_test.rb +101 -0
  51. data/test/distributed_remote_server_control_commands_test.rb +31 -0
  52. data/test/distributed_sorting_test.rb +21 -0
  53. data/test/distributed_test.rb +60 -0
  54. data/test/distributed_transactions_test.rb +34 -0
  55. data/test/encoding_test.rb +16 -0
  56. data/test/error_replies_test.rb +53 -0
  57. data/test/helper.rb +145 -0
  58. data/test/internals_test.rb +157 -0
  59. data/test/lint/hashes.rb +114 -0
  60. data/test/lint/internals.rb +41 -0
  61. data/test/lint/lists.rb +93 -0
  62. data/test/lint/sets.rb +66 -0
  63. data/test/lint/sorted_sets.rb +167 -0
  64. data/test/lint/strings.rb +137 -0
  65. data/test/lint/value_types.rb +84 -0
  66. data/test/persistence_control_commands_test.rb +22 -0
  67. data/test/pipelining_commands_test.rb +123 -0
  68. data/test/publish_subscribe_test.rb +158 -0
  69. data/test/redis_mock.rb +80 -0
  70. data/test/remote_server_control_commands_test.rb +63 -0
  71. data/test/sorting_test.rb +44 -0
  72. data/test/synchrony_driver.rb +57 -0
  73. data/test/test.conf +8 -0
  74. data/test/thread_safety_test.rb +30 -0
  75. data/test/transactions_test.rb +100 -0
  76. data/test/unknown_commands_test.rb +14 -0
  77. data/test/url_param_test.rb +60 -0
  78. metadata +128 -19
  79. data/README.markdown +0 -129
@@ -8,6 +8,12 @@ class Redis
8
8
 
9
9
  def call(*args)
10
10
  @commands << args
11
+ nil
12
+ end
13
+
14
+ def call_pipelined(commands, options = {})
15
+ @commands.concat commands
16
+ nil
11
17
  end
12
18
  end
13
19
  end
@@ -0,0 +1,3 @@
1
+ class Redis
2
+ VERSION = "2.2.0"
3
+ end
@@ -0,0 +1,24 @@
1
+ # -*- encoding: utf-8 -*-
2
+
3
+ $:.unshift File.expand_path("../lib", __FILE__)
4
+ require "redis/version"
5
+
6
+ Gem::Specification.new do |s|
7
+ s.name = %q{redis}
8
+ s.version = Redis::VERSION
9
+
10
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
11
+ s.authors = ["Ezra Zygmuntowicz", "Taylor Weibley", "Matthew Clark", "Brian McKinney", "Salvatore Sanfilippo", "Luca Guidi", "Michel Martens", "Damian Janowski", "Pieter Noordhuis"]
12
+ s.autorequire = %q{redis}
13
+ s.description = %q{Ruby client library for Redis, the key value storage server}
14
+ s.summary = %q{Ruby client library for Redis, the key value storage server}
15
+ s.email = %q{ez@engineyard.com}
16
+ s.homepage = %q{http://github.com/ezmobius/redis-rb}
17
+ s.rubyforge_project = "redis-rb"
18
+
19
+ s.files = `git ls-files`.split("\n")
20
+ s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
21
+ s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
22
+ s.require_paths = ["lib"]
23
+ s.rubygems_version = %q{1.5.0}
24
+ end
@@ -0,0 +1,32 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+
5
+ setup do
6
+ init Redis.new(OPTIONS)
7
+ end
8
+
9
+ load './test/lint/hashes.rb'
10
+
11
+ test "HSETNX" do |r|
12
+ r.hset("foo", "f1", "s1")
13
+ r.hsetnx("foo", "f1", "s2")
14
+
15
+ assert "s1" == r.hget("foo", "f1")
16
+
17
+ r.del("foo")
18
+ r.hsetnx("foo", "f1", "s2")
19
+
20
+ assert "s2" == r.hget("foo", "f1")
21
+ end
22
+
23
+ test "Mapped HMGET in a pipeline returns plain array" do |r|
24
+ r.hset("foo", "f1", "s1")
25
+ r.hset("foo", "f2", "s2")
26
+
27
+ result = r.pipelined do
28
+ assert nil == r.mapped_hmget("foo", "f1", "f2")
29
+ end
30
+
31
+ assert result[0] == ["s1", "s2"]
32
+ end
@@ -0,0 +1,60 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+
5
+ setup do
6
+ init Redis.new(OPTIONS)
7
+ end
8
+
9
+ load './test/lint/lists.rb'
10
+
11
+ test "RPUSHX" do |r|
12
+ r.rpushx "foo", "s1"
13
+ r.rpush "foo", "s2"
14
+ r.rpushx "foo", "s3"
15
+
16
+ assert 2 == r.llen("foo")
17
+ assert ["s2", "s3"] == r.lrange("foo", 0, -1)
18
+ end
19
+
20
+ test "LPUSHX" do |r|
21
+ r.lpushx "foo", "s1"
22
+ r.lpush "foo", "s2"
23
+ r.lpushx "foo", "s3"
24
+
25
+ assert 2 == r.llen("foo")
26
+ assert ["s3", "s2"] == r.lrange("foo", 0, -1)
27
+ end
28
+
29
+ test "LINSERT" do |r|
30
+ r.rpush "foo", "s1"
31
+ r.rpush "foo", "s3"
32
+ r.linsert "foo", :before, "s3", "s2"
33
+
34
+ assert ["s1", "s2", "s3"] == r.lrange("foo", 0, -1)
35
+
36
+ assert_raise(RuntimeError) do
37
+ r.linsert "foo", :anywhere, "s3", "s2"
38
+ end
39
+ end
40
+
41
+ test "RPOPLPUSH" do |r|
42
+ r.rpush "foo", "s1"
43
+ r.rpush "foo", "s2"
44
+
45
+ assert "s2" == r.rpoplpush("foo", "bar")
46
+ assert ["s2"] == r.lrange("bar", 0, -1)
47
+ assert "s1" == r.rpoplpush("foo", "bar")
48
+ assert ["s1", "s2"] == r.lrange("bar", 0, -1)
49
+ end
50
+
51
+ test "BRPOPLPUSH" do |r|
52
+ r.rpush "foo", "s1"
53
+ r.rpush "foo", "s2"
54
+
55
+ assert_equal "s2", r.brpoplpush("foo", "bar", 1)
56
+
57
+ assert_equal nil, r.brpoplpush("baz", "qux", 1)
58
+
59
+ assert_equal ["s2"], r.lrange("bar", 0, -1)
60
+ end
@@ -0,0 +1,78 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+
5
+ setup do
6
+ init Redis.new(OPTIONS)
7
+ end
8
+
9
+ load './test/lint/sets.rb'
10
+
11
+ test "SMOVE" do |r|
12
+ r.sadd "foo", "s1"
13
+ r.sadd "bar", "s2"
14
+
15
+ assert r.smove("foo", "bar", "s1")
16
+ assert r.sismember("bar", "s1")
17
+ end
18
+
19
+ test "SINTER" do |r|
20
+ r.sadd "foo", "s1"
21
+ r.sadd "foo", "s2"
22
+ r.sadd "bar", "s2"
23
+
24
+ assert ["s2"] == r.sinter("foo", "bar")
25
+ end
26
+
27
+ test "SINTERSTORE" do |r|
28
+ r.sadd "foo", "s1"
29
+ r.sadd "foo", "s2"
30
+ r.sadd "bar", "s2"
31
+
32
+ r.sinterstore("baz", "foo", "bar")
33
+
34
+ assert ["s2"] == r.smembers("baz")
35
+ end
36
+
37
+ test "SUNION" do |r|
38
+ r.sadd "foo", "s1"
39
+ r.sadd "foo", "s2"
40
+ r.sadd "bar", "s2"
41
+ r.sadd "bar", "s3"
42
+
43
+ assert ["s1", "s2", "s3"] == r.sunion("foo", "bar").sort
44
+ end
45
+
46
+ test "SUNIONSTORE" do |r|
47
+ r.sadd "foo", "s1"
48
+ r.sadd "foo", "s2"
49
+ r.sadd "bar", "s2"
50
+ r.sadd "bar", "s3"
51
+
52
+ r.sunionstore("baz", "foo", "bar")
53
+
54
+ assert ["s1", "s2", "s3"] == r.smembers("baz").sort
55
+ end
56
+
57
+ test "SDIFF" do |r|
58
+ r.sadd "foo", "s1"
59
+ r.sadd "foo", "s2"
60
+ r.sadd "bar", "s2"
61
+ r.sadd "bar", "s3"
62
+
63
+ assert ["s1"] == r.sdiff("foo", "bar")
64
+ assert ["s3"] == r.sdiff("bar", "foo")
65
+ end
66
+
67
+ test "SDIFFSTORE" do |r|
68
+ r.sadd "foo", "s1"
69
+ r.sadd "foo", "s2"
70
+ r.sadd "bar", "s2"
71
+ r.sadd "bar", "s3"
72
+
73
+ r.sdiffstore("baz", "foo", "bar")
74
+
75
+ assert ["s1"] == r.smembers("baz")
76
+ end
77
+
78
+
@@ -0,0 +1,109 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+
5
+ setup do
6
+ init Redis.new(OPTIONS)
7
+ end
8
+
9
+ load './test/lint/sorted_sets.rb'
10
+
11
+ test "ZCOUNT" do |r|
12
+ r.zadd "foo", 1, "s1"
13
+ r.zadd "foo", 2, "s2"
14
+ r.zadd "foo", 3, "s3"
15
+
16
+ assert 2 == r.zcount("foo", 2, 3)
17
+ end
18
+
19
+ test "ZUNIONSTORE" do |r|
20
+ r.zadd "foo", 1, "s1"
21
+ r.zadd "bar", 2, "s2"
22
+ r.zadd "foo", 3, "s3"
23
+ r.zadd "bar", 4, "s4"
24
+
25
+ assert 4 == r.zunionstore("foobar", ["foo", "bar"])
26
+ assert ["s1", "s2", "s3", "s4"] == r.zrange("foobar", 0, -1)
27
+ end
28
+
29
+ test "ZUNIONSTORE with WEIGHTS" do |r|
30
+ r.zadd "foo", 1, "s1"
31
+ r.zadd "foo", 3, "s3"
32
+ r.zadd "bar", 20, "s2"
33
+ r.zadd "bar", 40, "s4"
34
+
35
+ assert 4 == r.zunionstore("foobar", ["foo", "bar"])
36
+ assert ["s1", "s3", "s2", "s4"] == r.zrange("foobar", 0, -1)
37
+
38
+ assert 4 == r.zunionstore("foobar", ["foo", "bar"], :weights => [10, 1])
39
+ assert ["s1", "s2", "s3", "s4"] == r.zrange("foobar", 0, -1)
40
+ end
41
+
42
+ test "ZUNIONSTORE with AGGREGATE" do |r|
43
+ r.zadd "foo", 1, "s1"
44
+ r.zadd "foo", 2, "s2"
45
+ r.zadd "bar", 4, "s2"
46
+ r.zadd "bar", 3, "s3"
47
+
48
+ assert 3 == r.zunionstore("foobar", ["foo", "bar"])
49
+ assert ["s1", "s3", "s2"] == r.zrange("foobar", 0, -1)
50
+
51
+ assert 3 == r.zunionstore("foobar", ["foo", "bar"], :aggregate => :min)
52
+ assert ["s1", "s2", "s3"] == r.zrange("foobar", 0, -1)
53
+
54
+ assert 3 == r.zunionstore("foobar", ["foo", "bar"], :aggregate => :max)
55
+ assert ["s1", "s3", "s2"] == r.zrange("foobar", 0, -1)
56
+ end
57
+
58
+ test "ZINTERSTORE" do |r|
59
+ r.zadd "foo", 1, "s1"
60
+ r.zadd "bar", 2, "s1"
61
+ r.zadd "foo", 3, "s3"
62
+ r.zadd "bar", 4, "s4"
63
+
64
+ assert 1 == r.zinterstore("foobar", ["foo", "bar"])
65
+ assert ["s1"] == r.zrange("foobar", 0, -1)
66
+ end
67
+
68
+ test "ZINTERSTORE with WEIGHTS" do |r|
69
+ r.zadd "foo", 1, "s1"
70
+ r.zadd "foo", 2, "s2"
71
+ r.zadd "foo", 3, "s3"
72
+ r.zadd "bar", 20, "s2"
73
+ r.zadd "bar", 30, "s3"
74
+ r.zadd "bar", 40, "s4"
75
+
76
+ assert 2 == r.zinterstore("foobar", ["foo", "bar"])
77
+ assert ["s2", "s3"] == r.zrange("foobar", 0, -1)
78
+
79
+ assert 2 == r.zinterstore("foobar", ["foo", "bar"], :weights => [10, 1])
80
+ assert ["s2", "s3"] == r.zrange("foobar", 0, -1)
81
+
82
+ assert "40" == r.zscore("foobar", "s2")
83
+ assert "60" == r.zscore("foobar", "s3")
84
+ end
85
+
86
+ test "ZINTERSTORE with AGGREGATE" do |r|
87
+ r.zadd "foo", 1, "s1"
88
+ r.zadd "foo", 2, "s2"
89
+ r.zadd "foo", 3, "s3"
90
+ r.zadd "bar", 20, "s2"
91
+ r.zadd "bar", 30, "s3"
92
+ r.zadd "bar", 40, "s4"
93
+
94
+ assert 2 == r.zinterstore("foobar", ["foo", "bar"])
95
+ assert ["s2", "s3"] == r.zrange("foobar", 0, -1)
96
+ assert "22" == r.zscore("foobar", "s2")
97
+ assert "33" == r.zscore("foobar", "s3")
98
+
99
+ assert 2 == r.zinterstore("foobar", ["foo", "bar"], :aggregate => :min)
100
+ assert ["s2", "s3"] == r.zrange("foobar", 0, -1)
101
+ assert "2" == r.zscore("foobar", "s2")
102
+ assert "3" == r.zscore("foobar", "s3")
103
+
104
+ assert 2 == r.zinterstore("foobar", ["foo", "bar"], :aggregate => :max)
105
+ assert ["s2", "s3"] == r.zrange("foobar", 0, -1)
106
+ assert "20" == r.zscore("foobar", "s2")
107
+ assert "30" == r.zscore("foobar", "s3")
108
+ end
109
+
@@ -0,0 +1,80 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+
5
+ setup do
6
+ init Redis.new(OPTIONS)
7
+ end
8
+
9
+ load './test/lint/strings.rb'
10
+
11
+ test "MGET" do |r|
12
+ r.set("foo", "s1")
13
+ r.set("bar", "s2")
14
+
15
+ assert ["s1", "s2"] == r.mget("foo", "bar")
16
+ assert ["s1", "s2", nil] == r.mget("foo", "bar", "baz")
17
+ end
18
+
19
+ test "MGET mapped" do |r|
20
+ r.set("foo", "s1")
21
+ r.set("bar", "s2")
22
+
23
+ response = r.mapped_mget("foo", "bar")
24
+
25
+ assert "s1" == response["foo"]
26
+ assert "s2" == response["bar"]
27
+
28
+ response = r.mapped_mget("foo", "bar", "baz")
29
+
30
+ assert "s1" == response["foo"]
31
+ assert "s2" == response["bar"]
32
+ assert nil == response["baz"]
33
+ end
34
+
35
+ test "Mapped MGET in a pipeline returns plain array" do |r|
36
+ r.set("foo", "s1")
37
+ r.set("bar", "s2")
38
+
39
+ result = r.pipelined do
40
+ assert nil == r.mapped_mget("foo", "bar")
41
+ end
42
+
43
+ assert result[0] == ["s1", "s2"]
44
+ end
45
+
46
+ test "MSET" do |r|
47
+ r.mset(:foo, "s1", :bar, "s2")
48
+
49
+ assert "s1" == r.get("foo")
50
+ assert "s2" == r.get("bar")
51
+ end
52
+
53
+ test "MSET mapped" do |r|
54
+ r.mapped_mset(:foo => "s1", :bar => "s2")
55
+
56
+ assert "s1" == r.get("foo")
57
+ assert "s2" == r.get("bar")
58
+ end
59
+
60
+ test "MSETNX" do |r|
61
+ r.set("foo", "s1")
62
+ r.msetnx(:foo, "s2", :bar, "s3")
63
+
64
+ assert "s1" == r.get("foo")
65
+ assert nil == r.get("bar")
66
+ end
67
+
68
+ test "MSETNX mapped" do |r|
69
+ r.set("foo", "s1")
70
+ r.mapped_msetnx(:foo => "s2", :bar => "s3")
71
+
72
+ assert "s1" == r.get("foo")
73
+ assert nil == r.get("bar")
74
+ end
75
+
76
+ test "STRLEN" do |r|
77
+ r.set "foo", "lorem"
78
+
79
+ assert 5 == r.strlen("foo")
80
+ end
@@ -0,0 +1,88 @@
1
+ # encoding: UTF-8
2
+
3
+ require File.expand_path("./helper", File.dirname(__FILE__))
4
+ require File.expand_path("./redis_mock", File.dirname(__FILE__))
5
+
6
+ include RedisMock::Helper
7
+
8
+ setup do
9
+ init Redis.new(OPTIONS)
10
+ end
11
+
12
+ load "./test/lint/value_types.rb"
13
+
14
+ test "DEL" do |r|
15
+ r.set "foo", "s1"
16
+ r.set "bar", "s2"
17
+ r.set "baz", "s3"
18
+
19
+ assert ["bar", "baz", "foo"] == r.keys("*").sort
20
+
21
+ assert 1 == r.del("foo")
22
+
23
+ assert ["bar", "baz"] == r.keys("*").sort
24
+
25
+ assert 2 == r.del("bar", "baz")
26
+
27
+ assert [] == r.keys("*").sort
28
+ end
29
+
30
+ test "RANDOMKEY" do |r|
31
+ assert r.randomkey.to_s.empty?
32
+
33
+ r.set("foo", "s1")
34
+
35
+ assert "foo" == r.randomkey
36
+
37
+ r.set("bar", "s2")
38
+
39
+ 4.times do
40
+ assert ["foo", "bar"].include?(r.randomkey)
41
+ end
42
+ end
43
+
44
+ test "RENAME" do |r|
45
+ r.set("foo", "s1")
46
+ r.rename "foo", "bar"
47
+
48
+ assert "s1" == r.get("bar")
49
+ assert nil == r.get("foo")
50
+ end
51
+
52
+ test "RENAMENX" do |r|
53
+ r.set("foo", "s1")
54
+ r.set("bar", "s2")
55
+
56
+ assert false == r.renamenx("foo", "bar")
57
+
58
+ assert "s1" == r.get("foo")
59
+ assert "s2" == r.get("bar")
60
+ end
61
+
62
+ test "DBSIZE" do |r|
63
+ assert 0 == r.dbsize
64
+
65
+ r.set("foo", "s1")
66
+
67
+ assert 1 == r.dbsize
68
+ end
69
+
70
+ test "FLUSHDB" do |r|
71
+ r.set("foo", "s1")
72
+ r.set("bar", "s2")
73
+
74
+ assert 2 == r.dbsize
75
+
76
+ r.flushdb
77
+
78
+ assert 0 == r.dbsize
79
+ end
80
+
81
+ test "FLUSHALL" do
82
+ redis_mock(:flushall => lambda { "+FLUSHALL" }) do
83
+ redis = Redis.new(OPTIONS.merge(:port => 6380))
84
+
85
+ assert "FLUSHALL" == redis.flushall
86
+ end
87
+ end
88
+