require 'cassandra' module Moneta module Adapters # Cassandra backend # @api public # @author Potapov Sergey (aka Blake) class Cassandra include Defaults include ExpiresSupport attr_reader :backend # @param [Hash] options # @option options [String] :keyspace ('moneta') Cassandra keyspace # @option options [String] :column_family ('moneta') Cassandra column family # @option options [String] :host ('127.0.0.1') Server host name # @option options [Integer] :port (9160) Server port # @option options [Integer] :expires Default expiration time # @option options [::Cassandra] :backend Use existing backend instance # @option options Other options passed to `Cassandra#new` def initialize(options = {}) self.default_expires = options.delete(:expires) @cf = (options.delete(:column_family) || 'moneta').to_sym if options[:backend] @backend = options[:backend] else keyspace = options.delete(:keyspace) || 'moneta' options[:retries] ||= 3 options[:connect_timeout] ||= 10 options[:timeout] ||= 10 @backend = ::Cassandra.new('system', "#{options[:host] || '127.0.0.1'}:#{options[:port] || 9160}", options) unless @backend.keyspaces.include?(keyspace) cf_def = ::Cassandra::ColumnFamily.new(keyspace: keyspace, name: @cf.to_s) ks_def = ::Cassandra::Keyspace.new(name: keyspace, strategy_class: 'SimpleStrategy', strategy_options: { 'replication_factor' => '1' }, replication_factor: 1, cf_defs: [cf_def]) # Wait for keyspace to be created (issue #24) 10.times do begin @backend.add_keyspace(ks_def) rescue Exception => ex warn "Moneta::Adapters::Cassandra - #{ex.message}" end break if @backend.keyspaces.include?(keyspace) sleep 0.1 end end @backend.keyspace = keyspace end end # (see Proxy#key?) def key?(key, options = {}) if @backend.exists?(@cf, key) load(key, options) if options.include?(:expires) true else false end end # (see Proxy#load) def load(key, options = {}) if value = @backend.get(@cf, key) expires = expires_value(options, nil) @backend.insert(@cf, key, {'value' => value['value'] }, ttl: expires || nil) if expires != nil value['value'] end end # (see Proxy#store) def store(key, value, options = {}) @backend.insert(@cf, key, {'value' => value}, ttl: expires_value(options) || nil) value end # (see Proxy#delete) def delete(key, options = {}) if value = load(key, options) @backend.remove(@cf, key) value end end # (see Proxy#clear) def clear(options = {}) @backend.clear_column_family!(@cf) self end # (see Proxy#close) def close @backend.disconnect! nil end end end end