Current File : //proc/self/root/kunden/usr/share/gems/gems/memcache-client-1.8.5/lib/memcache/event_machine.rb
# Extensions for using memcache-client with EventMachine

raise "memcache/event_machine requires Ruby 1.9" if RUBY_VERSION < '1.9'

require 'memcache'
require 'eventmachine'
require 'fiber'

class MemCache
  
  # Since we are working in a single Thread, multiple Fiber environment,
  # disable the multithread Mutex as it will not work.
#  DEFAULT_OPTIONS[:multithread] = false

  module EventedServer

    def fiber_key
      @fiber_key ||= "memcached-#{@host}-#{@port}"
    end
    
    def socket
      sock = Thread.current[fiber_key]
      return sock if sock and not sock.closed?

      Thread.current[fiber_key] = nil

      # If the host was dead, don't retry for a while.
      return if @retry and @retry > Time.now
    
      Thread.current[fiber_key] ||= begin
        sock = EM::SocketConnection.connect(@host, @port, @timeout)
        yielding = true
        fiber = Fiber.current
        sock.callback do
          @status = 'CONNECTED'
          @retry  = nil
          yielding = false
          fiber.resume if Fiber.current != fiber
        end
        sock.errback do
          sock = nil
          yielding = false
          fiber.resume if Fiber.current != fiber
        end
        Fiber.yield if yielding
        sock
      end
    end

    def close
      sock = Thread.current[fiber_key]
      if sock
        sock.close if !sock.closed?
        Thread.current[fiber_key] = nil
      end
      @retry  = nil
      @status = "NOT CONNECTED"
    end

  end
end

module EM
  module SocketConnection
    include EM::Deferrable

    def self.connect(host, port, timeout)
      EM.connect(host, port, self) do |conn|
        conn.pending_connect_timeout = timeout
      end
    end

    def initialize
      @connected = false
      @index = 0
      @buf = ''
    end

    def closed?
      !@connected
    end

    def close
      @connected = false
      close_connection(true)
    end

    def write(buf)
      send_data(buf)
    end

    def read(size)
      if can_read?(size)
        yank(size)
      else
        fiber = Fiber.current
        @size = size
        @callback = proc { |data|
          fiber.resume(data)
        }
        # TODO Can leak fiber if the connection dies while
        # this fiber is yielded, waiting for data
        Fiber.yield
      end
    end
    
    SEP = "\r\n"

    def gets
      while true
        # Read to ensure we have some data in the buffer
        line = read(2)
        # Reset the buffer index to zero
        @buf = @buf.slice(@index..-1)
        @index = 0
        if eol = @buf.index(SEP)
          line << yank(eol + SEP.size)
          break
        else
          # EOL not in the current buffer
          line << yank(@buf.size)
        end
      end
      line
    end

    def can_read?(size)
      @buf.size >= @index + size
    end

    # EM callbacks

    def receive_data(data)
      @buf << data

      if @callback and can_read?(@size)
        callback = @callback
        data = yank(@size)
        @callback = @size = nil
        callback.call(data)
      end
    end

    def post_init
      @connected = true
      succeed
    end

    def unbind
      if @connected
        @connected = false
      else
        fail
      end
    end
    
    private
    
    BUFFER_SIZE = 4096

    def yank(len)      
      data = @buf.slice(@index, len)
      @index += len
      @index = @buf.size if @index > @buf.size
      if @index >= BUFFER_SIZE
        @buf = @buf.slice(@index..-1)
        @index = 0
      end
      data
    end
    
  end
end