mirror of
https://github.com/stripe/stripe-ruby.git
synced 2025-05-31 00:00:37 -04:00
151 lines
3.7 KiB
Ruby
151 lines
3.7 KiB
Ruby
module Stripe
|
|
class StripeObject
|
|
include Enumerable
|
|
|
|
attr_accessor :api_key
|
|
@@permanent_attributes = Set.new([:api_key])
|
|
|
|
# The default :id method is deprecated and isn't useful to us
|
|
if method_defined?(:id)
|
|
undef :id
|
|
end
|
|
|
|
def initialize(id=nil, api_key=nil)
|
|
@api_key = api_key
|
|
@values = {}
|
|
# This really belongs in APIResource, but not putting it there allows us
|
|
# to have a unified inspect method
|
|
@unsaved_values = Set.new
|
|
@transient_values = Set.new
|
|
self.id = id if id
|
|
end
|
|
|
|
def self.construct_from(values, api_key=nil)
|
|
obj = self.new(values[:id], api_key)
|
|
obj.refresh_from(values, api_key)
|
|
obj
|
|
end
|
|
|
|
def to_s(*args)
|
|
Stripe::JSON.dump(@values, :pretty => true)
|
|
end
|
|
|
|
def inspect()
|
|
id_string = (self.respond_to?(:id) && !self.id.nil?) ? " id=#{self.id}" : ""
|
|
"#<#{self.class}:0x#{self.object_id.to_s(16)}#{id_string}> JSON: " + Stripe::JSON.dump(@values, :pretty => true)
|
|
end
|
|
|
|
def refresh_from(values, api_key, partial=false)
|
|
@api_key = api_key
|
|
|
|
removed = partial ? Set.new : Set.new(@values.keys - values.keys)
|
|
added = Set.new(values.keys - @values.keys)
|
|
# Wipe old state before setting new. This is useful for e.g. updating a
|
|
# customer, where there is no persistent card parameter. Mark those values
|
|
# which don't persist as transient
|
|
|
|
instance_eval do
|
|
remove_accessors(removed)
|
|
add_accessors(added)
|
|
end
|
|
removed.each do |k|
|
|
@values.delete(k)
|
|
@transient_values.add(k)
|
|
@unsaved_values.delete(k)
|
|
end
|
|
values.each do |k, v|
|
|
@values[k] = Util.convert_to_stripe_object(v, api_key)
|
|
@transient_values.delete(k)
|
|
@unsaved_values.delete(k)
|
|
end
|
|
end
|
|
|
|
def [](k)
|
|
k = k.to_sym if k.kind_of?(String)
|
|
@values[k]
|
|
end
|
|
|
|
def []=(k, v)
|
|
send(:"#{k}=", v)
|
|
end
|
|
|
|
def keys
|
|
@values.keys
|
|
end
|
|
|
|
def values
|
|
@values.values
|
|
end
|
|
|
|
def to_json(*a)
|
|
Stripe::JSON.dump(@values)
|
|
end
|
|
|
|
def as_json(*a)
|
|
@values.as_json(*a)
|
|
end
|
|
|
|
def to_hash
|
|
@values
|
|
end
|
|
|
|
def each(&blk)
|
|
@values.each(&blk)
|
|
end
|
|
|
|
protected
|
|
|
|
def metaclass
|
|
class << self; self; end
|
|
end
|
|
|
|
def remove_accessors(keys)
|
|
metaclass.instance_eval do
|
|
keys.each do |k|
|
|
next if @@permanent_attributes.include?(k)
|
|
k_eq = :"#{k}="
|
|
remove_method(k) if method_defined?(k)
|
|
remove_method(k_eq) if method_defined?(k_eq)
|
|
end
|
|
end
|
|
end
|
|
|
|
def add_accessors(keys)
|
|
metaclass.instance_eval do
|
|
keys.each do |k|
|
|
next if @@permanent_attributes.include?(k)
|
|
k_eq = :"#{k}="
|
|
define_method(k) { @values[k] }
|
|
define_method(k_eq) do |v|
|
|
@values[k] = v
|
|
@unsaved_values.add(k)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
def method_missing(name, *args)
|
|
# TODO: only allow setting in updateable classes.
|
|
if name.to_s.end_with?('=')
|
|
attr = name.to_s[0...-1].to_sym
|
|
@values[attr] = args[0]
|
|
@unsaved_values.add(attr)
|
|
add_accessors([attr])
|
|
return
|
|
else
|
|
return @values[name] if @values.has_key?(name)
|
|
end
|
|
|
|
begin
|
|
super
|
|
rescue NoMethodError => e
|
|
if @transient_values.include?(name)
|
|
raise NoMethodError.new(e.message + ". HINT: The '#{name}' attribute was set in the past, however. It was then wiped when refreshing the object with the result returned by Stripe's API, probably as a result of a save(). The attributes currently available on this object are: #{@values.keys.join(', ')}")
|
|
else
|
|
raise
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|