rubyredis.rb 3.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
# RubyRedis is an alternative implementatin of Ruby client library written
# by Salvatore Sanfilippo.
#
# The aim of this library is to create an alternative client library that is
# much simpler and does not implement every command explicitly but uses
# method_missing instead.

require 'socket'

class RedisClient
    BulkCommands = {
        "set"=>true, "setnx"=>true, "rpush"=>true, "lpush"=>true, "lset"=>true,
        "lrem"=>true, "sadd"=>true, "srem"=>true, "sismember"=>true,
        "echo"=>true, "getset"=>true, "smove"=>true
    }

17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
    ConvertToBool = lambda{|r| r == 0 ? false : r}

    ReplyProcessor = {
        "exists" => ConvertToBool,
        "sismember"=> ConvertToBool,
        "sadd"=> ConvertToBool,
        "srem"=> ConvertToBool,
        "smove"=> ConvertToBool,
        "move"=> ConvertToBool,
        "setnx"=> ConvertToBool,
        "del"=> ConvertToBool,
        "renamenx"=> ConvertToBool,
        "expire"=> ConvertToBool,
        "keys" => lambda{|r| r.split(" ")},
        "info" => lambda{|r| 
            info = {}
            r.each_line {|kv|
                k,v = kv.split(':', 2)
                k,v = k.chomp, v = v.chomp
                info[k.to_sym] = v
            }
            info
        }
    }

    def convert_to_bool(r)
        r == 0 ? false : r
    end

46 47 48 49 50
    def initialize(opts={})
        opts = {:host => 'localhost', :port => '6379', :db => 0}.merge(opts)
        @host = opts[:host]
        @port = opts[:port]
        @db = opts[:db]
51
        connect_to_server
52 53 54 55 56 57 58
    end

    def to_s
        "Redis Client connected to #{@host}:#{@port} against DB #{@db}"
    end

    def connect_to_server
59 60
        @sock = TCPSocket.new(@host, @port, 0)
        call_command(["select",@db]) if @db != 0
61 62 63 64 65 66 67
    end

    def method_missing(*argv)
        call_command(argv)
    end

    def call_command(argv)
68 69 70 71 72 73 74 75 76 77 78 79 80
        # this wrapper to raw_call_command handle reconnection on socket
        # error. We try to reconnect just one time, otherwise let the error
        # araise.
        begin
            raw_call_command(argv)
        rescue Errno::ECONNRESET
            @sock.close
            connect_to_server
            raw_call_command(argv)
        end
    end

    def raw_call_command(argv)
81 82 83
        bulk = nil
        argv[0] = argv[0].to_s.downcase
        if BulkCommands[argv[0]]
84
            bulk = argv[-1].to_s
85 86 87 88
            argv[-1] = bulk.length
        end
        @sock.write(argv.join(" ")+"\r\n")
        @sock.write(bulk+"\r\n") if bulk
89 90 91 92

        # Post process the reply if needed
        processor = ReplyProcessor[argv[0]]
        processor ? processor.call(read_reply) : read_reply
93 94
    end

95 96 97 98
    def select(*args)
        raise "SELECT not allowed, use the :db option when creating the object"
    end

99 100 101 102 103 104 105 106
    def [](key)
        get(key)
    end

    def []=(key,value)
        set(key,value)
    end

107 108
    def read_reply
        line = @sock.gets
109
        raise Errno::ECONNRESET,"Connection lost" if !line
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133
        case line[0..0]
        when "-"
            raise line.strip
        when "+"
            line[1..-1].strip
        when ":"
            line[1..-1].to_i
        when "$"
            bulklen = line[1..-1].to_i
            return nil if bulklen == -1
            data = @sock.read(bulklen)
            @sock.read(2) # CRLF
            data
        when "*"
            objects = line[1..-1].to_i
            return nil if bulklen == -1
            res = []
            objects.times {
                res << read_reply
            }
            res
        end
    end
end