Class: Fortnox::API::Model::Base

Inherits:
Types::Model show all
Defined in:
lib/fortnox/api/models/base.rb

Direct Known Subclasses

Article, Customer, Document, Label, Project, TermsOfPayment, Unit

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Methods inherited from Types::Model

#initialize, is?

Constructor Details

This class inherits a constructor from Fortnox::API::Types::Model

Instance Attribute Details

#parentObject



84
85
86
# File 'lib/fortnox/api/models/base.rb', line 84

def parent
  @parent || self.class.new(self.class::STUB.dup)
end

#unsavedObject

TODO(jonas): Restructure this class a bit, it is not very readable.



12
13
14
# File 'lib/fortnox/api/models/base.rb', line 12

def unsaved
  @unsaved
end

Class Method Details

.attribute(name, *args) ⇒ Object



15
16
17
18
19
20
21
# File 'lib/fortnox/api/models/base.rb', line 15

def self.attribute(name, *args)
  define_method("#{name}?") do
    !send(name).nil?
  end

  super
end

.new(hash = {}) ⇒ Object



23
24
25
26
27
28
29
30
31
32
33
# File 'lib/fortnox/api/models/base.rb', line 23

def self.new(hash = {})
  begin
    obj = preserve_meta_properties(hash) do
      super(hash)
    end
  rescue Dry::Struct::Error => e
    raise Fortnox::API::AttributeError, e
  end

  IceNine.deep_freeze(obj)
end

.preserve_meta_properties(hash) ⇒ Object

dry-types filter anything that isn’t specified as an attribute on the class that is being instantiated. This wrapper preserves the meta properties we need to track object state during that initilisation and sets them on the object after dry-types is done with it.



102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
# File 'lib/fortnox/api/models/base.rb', line 102

def self.preserve_meta_properties(hash)
  is_unsaved = hash.delete(:unsaved) { true }
  is_new = hash.delete(:new) { true }
  parent = hash.delete(:parent) { nil }

  obj = yield

  # TODO: remove new, unsaved, saved
  obj.instance_variable_set(:@unsaved, is_unsaved)
  obj.instance_variable_set(:@saved, !is_unsaved)
  obj.instance_variable_set(:@new, is_new)
  obj.instance_variable_set(:@parent, parent)

  obj
end

.stubObject



35
36
37
# File 'lib/fortnox/api/models/base.rb', line 35

def self.stub
  new(self::STUB.dup)
end

Instance Method Details

#==(other) ⇒ Object

Generic comparison, by value, use .eql? or .equal? for object identity.



67
68
69
70
# File 'lib/fortnox/api/models/base.rb', line 67

def ==(other)
  return false unless other.is_a? self.class
  to_hash == other.to_hash
end

#attributes(*options) ⇒ Object

This filtering logic could be improved since it is currently O(N*M).



40
41
42
43
44
45
46
47
48
# File 'lib/fortnox/api/models/base.rb', line 40

def attributes(*options)
  return self.class.schema if options.nil?

  options = Array(options)

  self.class.schema.find_all do |_name, attribute|
    options.all? { |option| attribute.is?(option) }
  end
end

#new?Boolean

Returns:

  • (Boolean)


72
73
74
# File 'lib/fortnox/api/models/base.rb', line 72

def new?
  @new
end

#parent?Boolean

Returns:

  • (Boolean)


80
81
82
# File 'lib/fortnox/api/models/base.rb', line 80

def parent?
  !@parent.nil?
end

#saved?Boolean

Returns:

  • (Boolean)


76
77
78
# File 'lib/fortnox/api/models/base.rb', line 76

def saved?
  @saved
end

#to_hash(recursive = false) ⇒ Object



88
89
90
91
92
93
94
# File 'lib/fortnox/api/models/base.rb', line 88

def to_hash(recursive = false)
  return super() if recursive

  self.class.schema.keys.each_with_object({}) do |key, result|
    result[key] = self[key]
  end
end

#unique_idObject



50
51
52
# File 'lib/fortnox/api/models/base.rb', line 50

def unique_id
  send(self.class::UNIQUE_ID)
end

#update(hash) ⇒ Object



54
55
56
57
58
59
60
61
62
63
64
# File 'lib/fortnox/api/models/base.rb', line 54

def update(hash)
  old_attributes = to_hash
  new_attributes = old_attributes.merge(hash)

  return self if new_attributes == old_attributes

  new_hash = new_attributes.delete_if { |_, value| value.nil? }
  new_hash[:new] = @new
  new_hash[:parent] = self
  self.class.new(new_hash)
end