class Extlib::Pooling::Pool

def delete(instance)

def delete(instance)
  lock.synchronize do
    instance.instance_variable_set(:@__pool, nil)
    @used.delete(instance.object_id)
    wait.signal
  end
  nil
end

def dispose

def dispose
  flush!
  @resource.__pools.delete(@args)
  !Extlib::Pooling.pools.delete?(self).nil?
end

def expired?

def expired?
  @available.each do |instance|
    if Extlib.exiting || instance.instance_variable_get(:@__allocated_in_pool) + Extlib::Pooling.scavenger_interval <= (Time.now + 0.02)
      instance.dispose
      @available.delete(instance)
    end
  end
  size == 0
end

def flush!

def flush!
  @available.pop.dispose until @available.empty?
end

def initialize(max_size, resource, args)

def initialize(max_size, resource, args)
  raise ArgumentError.new("+max_size+ should be a Fixnum but was #{max_size.inspect}") unless Fixnum === max_size
  raise ArgumentError.new("+resource+ should be a Class but was #{resource.inspect}") unless Class === resource
  @max_size = max_size
  @resource = resource
  @args = args
  @available = []
  @used      = {}
  Extlib::Pooling.append_pool(self)
end

def inspect

def inspect
  "#<Extlib::Pooling::Pool<#{@resource.name}> available=#{@available.size} used=#{@used.size} size=#{@max_size}>"
end

def lock

def lock
  @resource.__pool_lock
end

def new

def new
  instance = nil
  begin
    lock.synchronize do
      if @available.size > 0
        instance = @available.pop
        @used[instance.object_id] = instance
      elsif @used.size < @max_size
        instance = @resource.__new(*@args)
        raise InvalidResourceError.new("#{@resource} constructor created a nil object") if instance.nil?
        raise InvalidResourceError.new("#{instance} is already part of the pool") if @used.include? instance
        instance.instance_variable_set(:@__pool, self)
        instance.instance_variable_set(:@__allocated_in_pool, Time.now)
        @used[instance.object_id] = instance
      else
        # Wait for another thread to release an instance.
        # If we exhaust the pool and don't release the active instance,
        # we'll wait here forever, so it's *very* important to always
        # release your services and *never* exhaust the pool within
        # a single thread.
        wait.wait(lock)
      end
    end
  end until instance
  instance
end

def release(instance)

def release(instance)
  lock.synchronize do
    instance.instance_variable_set(:@__allocated_in_pool, Time.now)
    @used.delete(instance.object_id)
    @available.push(instance)
    wait.signal
  end
  nil
end

def scavenge_interval

def scavenge_interval
  @resource.scavenge_interval
end

def size

def size
  @used.size + @available.size
end

def wait

def wait
  @resource.__pool_wait
end