module Braintree
class Configuration
API_VERSION = "4" # :nodoc:
DEFAULT_ENDPOINT = "api" # :nodoc:
READABLE_ATTRIBUTES = [
:merchant_id,
:public_key,
:private_key,
:client_id,
:client_secret,
:access_token,
]
WRITABLE_ATTRIBUTES = [
:custom_user_agent,
:endpoint,
:http_open_timeout,
:http_read_timeout,
:logger,
:merchant_id,
:public_key,
:private_key,
:environment,
]
class << self
attr_writer *WRITABLE_ATTRIBUTES
end
attr_reader *READABLE_ATTRIBUTES
def self.expectant_reader(*attributes) # :nodoc:
attributes.each do |attribute|
(class << self; self; end).send(:define_method, attribute) do
attribute_value = instance_variable_get("@#{attribute}")
raise ConfigurationError.new("Braintree::Configuration.#{attribute.to_s} needs to be set") unless attribute_value
attribute_value
end
end
end
expectant_reader *([:environment] + READABLE_ATTRIBUTES)
# Sets the Braintree environment to use. Valid values are :sandbox and :production
def self.environment=(env)
unless [:development, :qa, :sandbox, :production].include?(env)
raise ArgumentError, "#{env.inspect} is not a valid environment"
end
@environment = env
end
def self.gateway # :nodoc:
Braintree::Gateway.new(instantiate)
end
def self.instantiate # :nodoc:
config = new(
:custom_user_agent => @custom_user_agent,
:endpoint => @endpoint,
:environment => environment,
:http_open_timeout => http_open_timeout,
:http_read_timeout => http_read_timeout,
:logger => logger,
:merchant_id => merchant_id,
:private_key => private_key,
:public_key => public_key
)
end
def self.http_open_timeout
@http_open_timeout ||= 60
end
def self.http_read_timeout
@http_read_timeout ||= 60
end
def self.logger
@logger ||= _default_logger
end
def self.signature_service
instantiate.signature_service
end
def self.sha256_signature_service
instantiate.sha256_signature_service
end
def initialize(options = {})
WRITABLE_ATTRIBUTES.each do |attr|
instance_variable_set "@#{attr}", options[attr]
end
_check_for_mixed_credentials(options)
parser = Braintree::CredentialsParser.new
if options[:client_id] || options[:client_secret]
parser.parse_client_credentials(options[:client_id], options[:client_secret])
@client_id = parser.client_id
@client_secret = parser.client_secret
@environment = parser.environment
elsif options[:access_token]
parser.parse_access_token(options[:access_token])
@access_token = parser.access_token
@environment = parser.environment
@merchant_id = parser.merchant_id
else
@merchant_id = options[:merchant_id] || options[:partner_id]
end
end
def _check_for_mixed_credentials(options)
if (options[:client_id] || options[:client_secret]) && (options[:public_key] || options[:private_key])
raise ConfigurationError.new("Braintree::Gateway cannot be initialized with mixed credential types: client_id and client_secret mixed with public_key and private_key.")
end
if (options[:client_id] || options[:client_secret]) && (options[:access_token])
raise ConfigurationError.new("Braintree::Gateway cannot be initialized with mixed credential types: client_id and client_secret mixed with access_token.")
end
if (options[:public_key] || options[:private_key]) && (options[:access_token])
raise ConfigurationError.new("Braintree::Gateway cannot be initialized with mixed credential types: public_key and private_key mixed with access_token.")
end
end
def api_version # :nodoc:
API_VERSION
end
def base_merchant_path # :nodoc:
"/merchants/#{merchant_id}"
end
def base_url
"#{protocol}://#{server}:#{port}"
end
def base_merchant_url # :nodoc:
"#{base_url}#{base_merchant_path}"
end
def ca_file # :nodoc:
File.expand_path(File.join(File.dirname(__FILE__), "..", "ssl", "api_braintreegateway_com.ca.crt"))
end
def endpoint
@endpoint || DEFAULT_ENDPOINT
end
def http # :nodoc:
Http.new(self)
end
def logger
@logger ||= self.class._default_logger
end
def port # :nodoc:
case @environment
when :development, :integration
ENV['GATEWAY_PORT'] || 3000
when :production, :qa, :sandbox
443
end
end
def protocol # :nodoc:
ssl? ? "https" : "http"
end
def http_open_timeout
@http_open_timeout
end
def http_read_timeout
@http_read_timeout
end
def server # :nodoc:
case @environment
when :development, :integration
"localhost"
when :production
"#{endpoint}.braintreegateway.com"
when :qa
"gateway.qa.braintreepayments.com"
when :sandbox
"api.sandbox.braintreegateway.com"
end
end
def auth_url
case @environment
when :development, :integration
"http://auth.venmo.dev:9292"
when :production
"https://auth.venmo.com"
when :qa
"https://auth.venmo.qa2.braintreegateway.com"
when :sandbox
"https://auth.venmo.sandbox.braintreegateway.com"
end
end
def ssl? # :nodoc:
case @environment
when :development, :integration
false
when :production, :qa, :sandbox
true
end
end
def user_agent # :nodoc:
base_user_agent = "Braintree Ruby Gem #{Braintree::Version::String}"
@custom_user_agent ? "#{base_user_agent} (#{@custom_user_agent})" : base_user_agent
end
def self._default_logger # :nodoc:
logger = Logger.new(STDOUT)
logger.level = Logger::INFO
logger
end
def inspect
super.gsub(/@private_key=\".*\"/, '@private_key="[FILTERED]"')
end
def client_credentials?
!client_id.nil?
end
def assert_has_client_credentials
if client_id.nil? || client_secret.nil?
raise ConfigurationError.new("Braintree::Gateway client_id and client_secret are required.")
end
end
def assert_has_access_token_or_keys
if (public_key.nil? || private_key.nil?) && access_token.nil?
raise ConfigurationError.new("Braintree::Gateway public_key and private_key are required.")
end
end
def signature_service
@signature_service ||= SignatureService.new(@private_key)
end
def sha256_signature_service
@sha256_signature_service ||= SignatureService.new(@private_key, SHA256Digest)
end
end
end