form_obj 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 37cb560bca46560cd655e1741e7a818fb3c9217c
4
+ data.tar.gz: fd1faa6a1ffd99974e23cff0e99a2c02f91551be
5
+ SHA512:
6
+ metadata.gz: bab457aec9e9a0d01faa28bb1e7d4ac0e36cbfeb94a5de393d691f15bf6cad7d62aa014422aad97438f8ea29573e508e5899cea2e53cceaffc4ac21ff7fc0606
7
+ data.tar.gz: 37b7c80c65d6fd042cea4559dc826fa2caeb02a52046abd66441dc48ac52ac91a71594fd44a3e24c78274253ed787679e2acc810a718dd299b847c2fa07b0418
data/.gitignore ADDED
@@ -0,0 +1,11 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /_yardoc/
4
+ /coverage/
5
+ /doc/
6
+ /pkg/
7
+ /spec/reports/
8
+ /tmp/
9
+
10
+ # rspec failure tracking
11
+ .rspec_status
data/.rspec ADDED
@@ -0,0 +1,3 @@
1
+ --format documentation
2
+ --color
3
+ --require spec_helper
data/.ruby-version ADDED
@@ -0,0 +1 @@
1
+ 2.2.8
data/.travis.yml ADDED
@@ -0,0 +1,5 @@
1
+ sudo: false
2
+ language: ruby
3
+ rvm:
4
+ - 2.2.8
5
+ before_install: gem install bundler -v 1.16.1
data/Gemfile ADDED
@@ -0,0 +1,6 @@
1
+ source "https://rubygems.org"
2
+
3
+ git_source(:github) {|repo_name| "https://github.com/#{repo_name}" }
4
+
5
+ # Specify your gem's dependencies in form_obj.gemspec
6
+ gemspec
data/Gemfile.lock ADDED
@@ -0,0 +1,74 @@
1
+ PATH
2
+ remote: .
3
+ specs:
4
+ form_obj (0.1.0)
5
+ activemodel (>= 3.2)
6
+ typed_array (>= 1.0.0)
7
+
8
+ GEM
9
+ remote: https://rubygems.org/
10
+ specs:
11
+ actionpack (3.2.22.5)
12
+ activemodel (= 3.2.22.5)
13
+ activesupport (= 3.2.22.5)
14
+ builder (~> 3.0.0)
15
+ erubis (~> 2.7.0)
16
+ journey (~> 1.0.4)
17
+ rack (~> 1.4.5)
18
+ rack-cache (~> 1.2)
19
+ rack-test (~> 0.6.1)
20
+ sprockets (~> 2.2.1)
21
+ activemodel (3.2.22.5)
22
+ activesupport (= 3.2.22.5)
23
+ builder (~> 3.0.0)
24
+ activesupport (3.2.22.5)
25
+ i18n (~> 0.6, >= 0.6.4)
26
+ multi_json (~> 1.0)
27
+ builder (3.0.4)
28
+ concurrent-ruby (1.0.5)
29
+ diff-lcs (1.3)
30
+ erubis (2.7.0)
31
+ hike (1.2.3)
32
+ i18n (0.9.3)
33
+ concurrent-ruby (~> 1.0)
34
+ journey (1.0.4)
35
+ multi_json (1.13.1)
36
+ rack (1.4.7)
37
+ rack-cache (1.7.1)
38
+ rack (>= 0.4)
39
+ rack-test (0.6.3)
40
+ rack (>= 1.0)
41
+ rake (10.5.0)
42
+ rspec (3.7.0)
43
+ rspec-core (~> 3.7.0)
44
+ rspec-expectations (~> 3.7.0)
45
+ rspec-mocks (~> 3.7.0)
46
+ rspec-core (3.7.1)
47
+ rspec-support (~> 3.7.0)
48
+ rspec-expectations (3.7.0)
49
+ diff-lcs (>= 1.2.0, < 2.0)
50
+ rspec-support (~> 3.7.0)
51
+ rspec-mocks (3.7.0)
52
+ diff-lcs (>= 1.2.0, < 2.0)
53
+ rspec-support (~> 3.7.0)
54
+ rspec-support (3.7.1)
55
+ sprockets (2.2.3)
56
+ hike (~> 1.2)
57
+ multi_json (~> 1.0)
58
+ rack (~> 1.0)
59
+ tilt (~> 1.1, != 1.3.0)
60
+ tilt (1.4.1)
61
+ typed_array (1.0.1)
62
+
63
+ PLATFORMS
64
+ ruby
65
+
66
+ DEPENDENCIES
67
+ actionpack (>= 3.2)
68
+ bundler (~> 1.16)
69
+ form_obj!
70
+ rake (~> 10.0)
71
+ rspec (~> 3.0)
72
+
73
+ BUNDLED WITH
74
+ 1.16.1
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2018 Alexander Koltun
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,39 @@
1
+ # FormObj
2
+
3
+ Welcome to your new gem! In this directory, you'll find the files you need to be able to package up your Ruby library into a gem. Put your Ruby code in the file `lib/form_obj`. To experiment with that code, run `bin/console` for an interactive prompt.
4
+
5
+ TODO: Delete this and the text above, and describe your gem
6
+
7
+ ## Installation
8
+
9
+ Add this line to your application's Gemfile:
10
+
11
+ ```ruby
12
+ gem 'form_obj'
13
+ ```
14
+
15
+ And then execute:
16
+
17
+ $ bundle
18
+
19
+ Or install it yourself as:
20
+
21
+ $ gem install form_obj
22
+
23
+ ## Usage
24
+
25
+ TODO: Write usage instructions here
26
+
27
+ ## Development
28
+
29
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
30
+
31
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
32
+
33
+ ## Contributing
34
+
35
+ Bug reports and pull requests are welcome on GitHub at https://github.com/[USERNAME]/form_obj.
36
+
37
+ ## License
38
+
39
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,6 @@
1
+ require "bundler/gem_tasks"
2
+ require "rspec/core/rake_task"
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+
6
+ task :default => :spec
data/bin/console ADDED
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "bundler/setup"
4
+ require "form_obj"
5
+
6
+ # You can add fixtures and/or initialization code here to make experimenting
7
+ # with your gem easier. You can also use a different console, if you like.
8
+
9
+ # (If you use this, don't forget to add pry to your Gemfile!)
10
+ # require "pry"
11
+ # Pry.start
12
+
13
+ require "irb"
14
+ IRB.start(__FILE__)
data/bin/setup ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
data/form_obj.gemspec ADDED
@@ -0,0 +1,43 @@
1
+
2
+ lib = File.expand_path("../lib", __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require "form_obj/version"
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "form_obj"
8
+ spec.version = FormObj::VERSION
9
+ spec.authors = ["Alexander Koltun"]
10
+ spec.email = ["alexander.koltun@gmail.com"]
11
+
12
+ spec.summary = %q{Simple but powerful form object compatible with Rails form builders.}
13
+ spec.description = %q{Form Object with simple DSL which allows nested Form Objects and arrays of Form Objects. Form
14
+ Object is compatible with Rails form builders, can update its attributes from a hash and serialize them to a hash.
15
+ Form Object attributes could be mapped to models attributes and Form Object can be loaded from and saved to models as
16
+ well as serialized to a hash which reflects a model. ActiveModel::Errors could be copied from a model to Form Object.}
17
+ spec.homepage = "https://github.com/akoltun/form_obj"
18
+ spec.license = "MIT"
19
+
20
+ # Prevent pushing this gem to RubyGems.org. To allow pushes either set the 'allowed_push_host'
21
+ # to allow pushing to a single host or delete this section to allow pushing to any host.
22
+ if spec.respond_to?(:metadata)
23
+ spec.metadata["allowed_push_host"] = "https://rubygems.org"
24
+ else
25
+ raise "RubyGems 2.0 or newer is required to protect against " \
26
+ "public gem pushes."
27
+ end
28
+
29
+ spec.files = `git ls-files -z`.split("\x0").reject do |f|
30
+ f.match(%r{^(test|spec|features)/})
31
+ end
32
+ spec.bindir = "exe"
33
+ spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
34
+ spec.require_paths = ["lib"]
35
+
36
+ spec.add_runtime_dependency "typed_array", ">= 1.0.0"
37
+ spec.add_runtime_dependency "activemodel", ">= 3.2"
38
+
39
+ spec.add_development_dependency "bundler", "~> 1.16"
40
+ spec.add_development_dependency "rake", "~> 10.0"
41
+ spec.add_development_dependency "rspec", "~> 3.0"
42
+ spec.add_development_dependency "actionpack", ">= 3.2"
43
+ end
@@ -0,0 +1,96 @@
1
+ require 'typed_array'
2
+
3
+ class FormObj
4
+ class Array < TypedArray
5
+ def initialize(klass, hash: false, model_class: nil)
6
+ @item_hash = hash
7
+ @model_class = model_class
8
+ super(klass)
9
+ end
10
+
11
+ def item_hash?
12
+ @item_hash
13
+ end
14
+
15
+ def create
16
+ self << (item = item_class.new({}, hash: @item_hash))
17
+ item
18
+ end
19
+
20
+ def update_attributes(vals)
21
+ ids_exists = []
22
+ items_to_add = []
23
+
24
+ vals.map(&:with_indifferent_access).each do |val|
25
+ id = val[item_class.primary_key]
26
+ item = self.find { |i| i.primary_key == id }
27
+ if item
28
+ item.update_attributes(val)
29
+ ids_exists << id
30
+ else
31
+ items_to_add << val
32
+ end
33
+ end
34
+
35
+ ids_to_remove = self.map(&:primary_key) - ids_exists
36
+ self.delete_if { |item| ids_to_remove.include? item.primary_key }
37
+
38
+ items_to_add.each do |item|
39
+ self.create.update_attributes(item)
40
+ end
41
+
42
+ sort! { |a, b| vals.index { |val| val[item_class.primary_key] == a.primary_key } <=> vals.index { |val| val[item_class.primary_key] == b.primary_key } }
43
+ end
44
+
45
+ def save_to_models(models)
46
+ model_primary_key = self.item_class.model_primary_key
47
+ default_models = models[:default]
48
+ ids_exists = []
49
+ items_to_add = []
50
+
51
+ self.each do |item|
52
+ id = item.primary_key
53
+ model = if default_models.respond_to?("find_by_#{model_primary_key}")
54
+ default_models.send("find_by_#{model_primary_key}", id)
55
+ elsif @item_hash
56
+ default_models.find { |m| (m.key?(model_primary_key.to_sym) ? m[model_primary_key.to_sym] : m[model_primary_key.to_s]) == id }
57
+ else
58
+ default_models.find { |m| m.send(model_primary_key) == id }
59
+ end
60
+ if model
61
+ item.save_to_models(models.merge(default: model))
62
+ ids_exists << id
63
+ else
64
+ items_to_add << item
65
+ end
66
+ end
67
+
68
+ ids_to_remove = if @item_hash
69
+ default_models.map { |m| m.key?(model_primary_key.to_sym) ? m[model_primary_key.to_sym] : m[model_primary_key.to_s] }
70
+ else
71
+ default_models.map(&(model_primary_key.to_sym))
72
+ end - ids_exists
73
+ if default_models.respond_to?(:destroy_all)
74
+ default_models.destroy_all(model_primary_key => ids_to_remove)
75
+ elsif @item_hash
76
+ default_models.delete_if { |m| ids_to_remove.include? (m.key?(model_primary_key.to_sym) ? m[model_primary_key.to_sym] : m[model_primary_key.to_s]) }
77
+ else
78
+ default_models.delete_if { |m| ids_to_remove.include? m.send(model_primary_key) }
79
+ end
80
+
81
+ items_to_add.each do |item|
82
+ default_models << model = (@model_class.is_a?(String) ? @model_class.constantize : @model_class).new
83
+ item.save_to_models(models.merge(default: model))
84
+ end
85
+ end
86
+
87
+ def to_hash
88
+ self.map(&:to_hash)
89
+ end
90
+
91
+ def export_to_model_hash(models)
92
+ self.each { |item| models[:default] << item.export_to_model_hash(models.merge(default: {}))[:default] }
93
+ models
94
+ end
95
+ end
96
+ end
@@ -0,0 +1,25 @@
1
+ class FormObj
2
+ class Attribute
3
+ attr_reader :name, :subform, :model, :model_attributes, :model_class
4
+
5
+ def initialize(name, subform = false, model: :default, model_attribute: nil, model_class: nil, hash: false, array: false)
6
+ @subform = subform
7
+ @array = array
8
+ @hash = hash
9
+
10
+ @model_attributes = model_attribute === false ? [] : (model_attribute || name).to_s.split('.')
11
+ @name = name.to_s.start_with?(':') ? name.to_s[1..-1] : name.to_s
12
+
13
+ @model = model
14
+ @model_class = model_class.is_a?(Enumerable) ? model_class : [model_class || (hash ? Hash : name.to_s.camelize)]
15
+ end
16
+
17
+ def hash?
18
+ @hash
19
+ end
20
+
21
+ def array?
22
+ @array
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,3 @@
1
+ class FormObj
2
+ VERSION = "0.1.0"
3
+ end
data/lib/form_obj.rb ADDED
@@ -0,0 +1,433 @@
1
+ require "form_obj/version"
2
+ require 'form_obj/attribute'
3
+ require 'form_obj/array'
4
+ require 'result_obj'
5
+ require 'active_model'
6
+
7
+ class FormObj
8
+ class UnknownAttributeError < RuntimeError; end
9
+ class WrongHashAttributeValue < RuntimeError; end
10
+ class WrongArrayAttributeValue < RuntimeError; end
11
+
12
+ extend ::ActiveModel::Naming
13
+ extend ::ActiveModel::Translation
14
+
15
+ include ::ActiveModel::Conversion
16
+ include ::ActiveModel::Validations
17
+
18
+ private
19
+
20
+ class_attribute :_attributes, instance_predicate: false, instance_reader: false, instance_writer: false
21
+ self._attributes = []
22
+
23
+ public
24
+
25
+ attr_accessor :persisted
26
+ attr_reader :errors
27
+
28
+ class_attribute :primary_key, instance_predicate: false, instance_reader: false, instance_writer: false
29
+ self.primary_key = :id
30
+
31
+ def self.model_name
32
+ @model_name || super
33
+ end
34
+
35
+ def self.model_primary_key
36
+ primary_key_attributes = self._attributes.find { |attr| attr.name == self.primary_key.to_s }.model_attributes
37
+ raise 'Primary key could not be mapped to nested model' if primary_key_attributes.size > 1
38
+ primary_key_attributes.first
39
+ end
40
+
41
+ def self.attribute(name, opts = {}, &block)
42
+ primary_key = opts.delete(:primary_key)
43
+ hash = opts[:hash]
44
+ klass = opts.delete(:class)
45
+ if !klass && block_given?
46
+ klass = Class.new(FormObj, &block)
47
+ klass.instance_variable_set(:@model_name, ActiveModel::Name.new(klass, nil, name.to_s))
48
+ end
49
+ self._attributes += [attr = Attribute.new(name, klass, opts)]
50
+
51
+ if klass
52
+ klass.primary_key = primary_key if primary_key
53
+
54
+ if opts[:array]
55
+ define_method name do
56
+ instance_variable_get("@#{name}") || instance_variable_set("@#{name}", FormObj::Array.new(klass, hash: hash, model_class: attr.model_class.last))
57
+ end
58
+ define_method "#{name}=" do |val|
59
+ unless val.class == FormObj::Array
60
+ raise ArgumentError.new(":#{name} attribute value should be of class #{self.class.name}::Array while attempt to assign value of class #{val.class.name}")
61
+ end
62
+ unless val.item_class == klass
63
+ raise ArgumentError.new(":#{name} attribute value should be a form array with items of class #{klass.name} attempt to assign a form array with items of class #{val.item_class.name}")
64
+ end
65
+
66
+ @persisted = false
67
+ instance_variable_set("@#{name}", val)
68
+ end
69
+
70
+ else
71
+ define_method name do
72
+ instance_variable_get("@#{name}") || instance_variable_set("@#{name}", klass.new({}, hash: hash))
73
+ end
74
+ define_method "#{name}=" do |val|
75
+ unless val.class == klass
76
+ raise ArgumentError.new(":#{name} attribute value should be of class #{klass.name} while attempt to assign value of class #{val.class.name}")
77
+ end
78
+
79
+ @persisted = false
80
+ instance_variable_set("@#{name}", val)
81
+ end
82
+ end
83
+
84
+ else
85
+ attr_reader name
86
+ define_method "#{name}=" do |val|
87
+ @persisted = false
88
+ instance_variable_set("@#{name}", val)
89
+ end
90
+ self.primary_key = name.to_sym if primary_key
91
+ end
92
+ end
93
+
94
+ def initialize(models = {}, opts = { hash: false })
95
+ @errors = ActiveModel::Errors.new(self)
96
+ @persisted = false
97
+ @hash = opts[:hash]
98
+ load_from_models(models) if models.present?
99
+ end
100
+
101
+ def persisted?
102
+ @persisted
103
+ end
104
+
105
+ def primary_key
106
+ send(self.class.primary_key)
107
+ end
108
+
109
+ def primary_key=(val)
110
+ send("#{self.class.primary_key}=", val)
111
+ end
112
+
113
+ def load_from_models(models)
114
+ attributes.each { |attribute| load_attribute_from_model(attribute, models) }
115
+ @persisted = true
116
+ self
117
+ end
118
+
119
+ def save_to_models(models)
120
+ attributes.each { |attribute | save_attribute_to_model(attribute, models) }
121
+ @persisted = true
122
+ self
123
+ end
124
+
125
+ def load_from_model(model)
126
+ load_from_models(default: model)
127
+ end
128
+
129
+ def save_to_model(model)
130
+ save_to_models(default: model)
131
+ end
132
+
133
+ def update_attributes(new_attrs, raise_if_not_found: true)
134
+ @persisted = false
135
+ new_attrs.each_pair do |new_attr, new_val|
136
+ attr = attributes.find { |attr| attr.name == new_attr.to_s }
137
+ if attr.nil?
138
+ raise UnknownAttributeError.new(new_attr) if raise_if_not_found
139
+ else
140
+ if attr.subform && attr.array?
141
+ if new_val.is_a?(Enumerable)
142
+ self.send(new_attr).update_attributes(new_val)
143
+ else
144
+ raise WrongArrayAttributeValue.new("#{new_attr}: #{new_val.inspect}")
145
+ end
146
+ elsif attr.subform
147
+ if new_val.is_a? Hash
148
+ self.send(new_attr).update_attributes(new_val)
149
+ else
150
+ raise WrongHashAttributeValue.new("#{new_attr}: #{new_val.inspect}")
151
+ end
152
+ else
153
+ self.send("#{new_attr}=", new_val)
154
+ end
155
+ end
156
+ end
157
+ self
158
+ end
159
+
160
+ def to_hash
161
+ Hash[attributes.map { |attribute| [attribute.name.to_sym, attribute.subform ? send(attribute.name).to_hash : send(attribute.name)] }]
162
+ end
163
+
164
+ def to_model_hash(model = :default)
165
+ export_to_model_hash(model => (hash = {}))
166
+ hash
167
+ end
168
+
169
+ def export_to_model_hash(models)
170
+ attributes.each do |attribute|
171
+ if attribute.array?
172
+ value = []
173
+ if models[attribute.model]
174
+ val = if attribute.model_attributes.present?
175
+ attribute
176
+ .model_attributes
177
+ .map { |ma| ma.to_s[0] == ':' ? ma[1..-1] : ma }
178
+ .reverse
179
+ .reduce(value) { |h, k| { k.to_sym => h } }
180
+ else
181
+ { self: value }
182
+ end
183
+ models[attribute.model].merge!(val)
184
+ end
185
+ nested_models = models.merge(default: value)
186
+ send(attribute.name).export_to_model_hash(nested_models)
187
+
188
+ elsif attribute.subform # && !attribute.array?
189
+ value = {}
190
+ if models[attribute.model]
191
+ if attribute.model_attributes.present?
192
+ val = attribute
193
+ .model_attributes
194
+ .map { |ma| ma.to_s[0] == ':' ? ma[1..-1] : ma }
195
+ .reverse
196
+ .reduce(value) { |h, k| { k.to_sym => h } }
197
+ models[attribute.model].merge!(val)
198
+ nested_models = models.merge(default: value)
199
+ else
200
+ if attribute.model == :default
201
+ nested_models = models
202
+ else
203
+ nested_models = models.merge(default: models[attribute.model])
204
+ end
205
+ end
206
+ else
207
+ nested_models = models.merge(default: value)
208
+ end
209
+ send(attribute.name).export_to_model_hash(nested_models)
210
+
211
+ else
212
+ if models[attribute.model]
213
+ if attribute.model_attributes.present?
214
+ value = send(attribute.name)
215
+ val = attribute
216
+ .model_attributes
217
+ .map { |ma| ma.to_s[0] == ':' ? ma[1..-1] : ma }
218
+ .reverse
219
+ .reduce(value) { |h, k| { k.to_sym => h } }
220
+ models[attribute.model].merge!(val)
221
+ end
222
+ end
223
+ end
224
+ end
225
+ models
226
+ end
227
+
228
+ def copy_errors_from_model(model)
229
+ copy_errors_from_models(default: model)
230
+ end
231
+
232
+ def copy_errors_from_models(models)
233
+ attributes.each do |attribute|
234
+ if attribute.subform
235
+ else
236
+ @errors[attribute.name].push(*read_attribute_errors_from_model(attribute, models[attribute.model]))
237
+ end
238
+ end
239
+ self
240
+ end
241
+
242
+ private
243
+
244
+ def attributes
245
+ self.class._attributes.clone
246
+ end
247
+
248
+ def load_attribute_from_model(attribute, models)
249
+ if attribute.subform
250
+ if attribute.array?
251
+ self.send(attribute.name).clear
252
+ if (model_array = if attribute.model_attributes.present?
253
+ read_attribute_from_model(attribute, models[attribute.model])
254
+ else
255
+ models[attribute.model]
256
+ end)
257
+ model_array.each do |model|
258
+ self.send(attribute.name).create.load_from_models(models.merge(default: model))
259
+ end
260
+ end
261
+ else
262
+ if attribute.model_attributes.present?
263
+ self.send(attribute.name).load_from_models(models.merge(default: read_attribute_from_model(attribute, models[attribute.model])))
264
+ else
265
+ self.send(attribute.name).load_from_models(models.merge(default: models[attribute.model]))
266
+ end
267
+ end
268
+ else
269
+ if attribute.model_attributes.present?
270
+ self.send("#{attribute.name}=", read_attribute_from_model(attribute, models[attribute.model]))
271
+ end
272
+ end
273
+ end
274
+
275
+ def read_attribute_from_model(attribute, model, create_nested_form_if_nil: false)
276
+ m = attribute
277
+ .model_attributes[1..-1]
278
+ .reduce(
279
+ {
280
+ index: 0,
281
+ model: _read_attribute(
282
+ model: model,
283
+ model_attr: attribute.model_attributes.first,
284
+ hash: @hash,
285
+ nested_form_class: create_nested_form_if_nil ? ((attribute.model_class.size == 1 && attribute.array?) ? ::Array : attribute.model_class.first) : nil
286
+ )
287
+ }
288
+ ) { |a, m_attr|
289
+ {
290
+ index: a[:index] + 1,
291
+ model: _read_attribute(
292
+ model: a[:model],
293
+ model_attr: m_attr,
294
+ nested_form_class: create_nested_form_if_nil ? ((attribute.model_class.size == a[:index] + 2 && attribute.array?) ? ::Array : attribute.model_class[a[:index] + 1]) : nil
295
+ )
296
+ }
297
+ }[:model]
298
+ end
299
+
300
+ def _read_attribute(model_attr:, model:, hash: false, nested_form_class: nil)
301
+ return nil if model.nil?
302
+
303
+ result = if hash
304
+ model[model_attr.to_sym].nil? ? model[model_attr.to_s] : model[model_attr.to_sym]
305
+ elsif model_attr.to_s[0] == ':'
306
+ model[model_attr[1..-1].to_sym].nil? ? model[model_attr[1..-1].to_s] : model[model_attr[1..-1].to_sym]
307
+ else
308
+ model.send(model_attr)
309
+ end
310
+
311
+ if result.nil? && nested_form_class
312
+ result = (nested_form_class.is_a?(String) ? nested_form_class.constantize : nested_form_class).try(:new)
313
+ if hash
314
+ model[model_attr.to_sym] = result
315
+ elsif model_attr.to_s[0] == ':'
316
+ model[model_attr[1..-1].to_sym] = result
317
+ else
318
+ model.send("#{model_attr}=", result)
319
+ end
320
+ end
321
+
322
+ result
323
+ end
324
+
325
+ def read_attribute_errors_from_model(attribute, model)
326
+ m = if attribute.model_attributes.size > 1
327
+ attribute
328
+ .model_attributes[1..-2]
329
+ .reduce(
330
+ {
331
+ index: 0,
332
+ model: _read_attribute(
333
+ model: model,
334
+ model_attr: attribute.model_attributes.first,
335
+ hash: @hash,
336
+ )
337
+ }
338
+ ) { |a, m_attr|
339
+ {
340
+ index: a[:index] + 1,
341
+ model: _read_attribute(
342
+ model: a[:model],
343
+ model_attr: m_attr,
344
+ )
345
+ }
346
+ }[:model]
347
+
348
+ elsif attribute.model_attributes.size == 1
349
+ model
350
+ else
351
+ nil
352
+ end
353
+
354
+ if m.nil?
355
+ []
356
+ else
357
+ _read_attribute_error(model_attr: attribute.model_attributes.last, model: m, hash: (attribute.model_attributes.size == 1) && @hash)
358
+ end
359
+ end
360
+
361
+ def _read_attribute_error(model_attr:, model:, hash: false)
362
+ if hash || model_attr.to_s[0] == ':'
363
+ []
364
+ else
365
+ model.errors[model_attr.to_sym]
366
+ end
367
+ end
368
+
369
+ def save_attribute_to_model(attribute, models)
370
+ if attribute.subform
371
+ if attribute.array?
372
+ if attribute.model_attributes.present?
373
+ self.send(attribute.name).save_to_models(models.merge(default: read_attribute_from_model(attribute, models[attribute.model], create_nested_form_if_nil: true)))
374
+ else
375
+ self.send(attribute.name).save_to_models(models.merge(default: models[attribute.model]))
376
+ end
377
+ else
378
+ if attribute.model_attributes.present?
379
+ self.send(attribute.name).save_to_models(models.merge(default: read_attribute_from_model(attribute, models[attribute.model], create_nested_form_if_nil: true)))
380
+ else
381
+ self.send(attribute.name).save_to_models(models.merge(default: models[attribute.model]))
382
+ end
383
+ end
384
+ else
385
+ if attribute.model_attributes.present?
386
+ write_attribute_to_model(attribute, models[attribute.model], self.send(attribute.name))
387
+ end
388
+ end
389
+ end
390
+
391
+ def write_attribute_to_model(attribute, model, value)
392
+ path = attribute.model_attributes[0..-2]
393
+ model_attr = attribute.model_attributes.last
394
+
395
+ if path.present?
396
+ _write_attribute(
397
+ model: path[1..-1]
398
+ .reduce({
399
+ index: 0,
400
+ model: _read_attribute(
401
+ model: model,
402
+ model_attr: attribute.model_attributes.first,
403
+ hash: @hash,
404
+ nested_form_class: attribute.model_class.first
405
+ )
406
+ }) { |a, m_attr|
407
+ {
408
+ index: a[:index] + 1,
409
+ model: _read_attribute(
410
+ model: a[:model],
411
+ model_attr: m_attr,
412
+ nested_form_class: attribute.model_class[a[:index] + 1]
413
+ )
414
+ }
415
+ }[:model],
416
+ model_attr: model_attr,
417
+ value: value
418
+ )
419
+ else
420
+ _write_attribute(model: model, model_attr: model_attr, hash: @hash, value: value)
421
+ end
422
+ end
423
+
424
+ def _write_attribute(model_attr:, model:, hash: false, value:)
425
+ if hash
426
+ model[(model.key?(model_attr.to_s) && !model.key?(model_attr.to_sym)) ? model_attr.to_s : model_attr.to_sym] = value
427
+ elsif model_attr.to_s[0] == ':'
428
+ model[(model.key?(model_attr[1..-1].to_s) && !model.key?(model_attr[1..-1].to_sym)) ? model_attr[1..-1].to_s : model_attr[1..-1].to_sym] = value
429
+ else
430
+ model.send("#{model_attr}=", value)
431
+ end
432
+ end
433
+ end
data/lib/result_obj.rb ADDED
@@ -0,0 +1,9 @@
1
+ module ResultObj
2
+ def failure?
3
+ @errors.present?
4
+ end
5
+
6
+ def success?
7
+ !failure?
8
+ end
9
+ end
metadata ADDED
@@ -0,0 +1,150 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: form_obj
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Alexander Koltun
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2018-04-12 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: typed_array
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: 1.0.0
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: 1.0.0
27
+ - !ruby/object:Gem::Dependency
28
+ name: activemodel
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '3.2'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '3.2'
41
+ - !ruby/object:Gem::Dependency
42
+ name: bundler
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '1.16'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '1.16'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rake
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '10.0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '10.0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: rspec
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '3.0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '3.0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: actionpack
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ">="
88
+ - !ruby/object:Gem::Version
89
+ version: '3.2'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ">="
95
+ - !ruby/object:Gem::Version
96
+ version: '3.2'
97
+ description: |-
98
+ Form Object with simple DSL which allows nested Form Objects and arrays of Form Objects. Form
99
+ Object is compatible with Rails form builders, can update its attributes from a hash and serialize them to a hash.
100
+ Form Object attributes could be mapped to models attributes and Form Object can be loaded from and saved to models as
101
+ well as serialized to a hash which reflects a model. ActiveModel::Errors could be copied from a model to Form Object.
102
+ email:
103
+ - alexander.koltun@gmail.com
104
+ executables: []
105
+ extensions: []
106
+ extra_rdoc_files: []
107
+ files:
108
+ - ".gitignore"
109
+ - ".rspec"
110
+ - ".ruby-version"
111
+ - ".travis.yml"
112
+ - Gemfile
113
+ - Gemfile.lock
114
+ - LICENSE.txt
115
+ - README.md
116
+ - Rakefile
117
+ - bin/console
118
+ - bin/setup
119
+ - form_obj.gemspec
120
+ - lib/form_obj.rb
121
+ - lib/form_obj/array.rb
122
+ - lib/form_obj/attribute.rb
123
+ - lib/form_obj/version.rb
124
+ - lib/result_obj.rb
125
+ homepage: https://github.com/akoltun/form_obj
126
+ licenses:
127
+ - MIT
128
+ metadata:
129
+ allowed_push_host: https://rubygems.org
130
+ post_install_message:
131
+ rdoc_options: []
132
+ require_paths:
133
+ - lib
134
+ required_ruby_version: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - ">="
137
+ - !ruby/object:Gem::Version
138
+ version: '0'
139
+ required_rubygems_version: !ruby/object:Gem::Requirement
140
+ requirements:
141
+ - - ">="
142
+ - !ruby/object:Gem::Version
143
+ version: '0'
144
+ requirements: []
145
+ rubyforge_project:
146
+ rubygems_version: 2.4.5.3
147
+ signing_key:
148
+ specification_version: 4
149
+ summary: Simple but powerful form object compatible with Rails form builders.
150
+ test_files: []