form_core_v1 0.0.14

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 87a636a0d0d56a208122563acd0ef6d26a6a00fe
4
+ data.tar.gz: 70c4bd33de24eb514df38432678be77e21a9f567
5
+ SHA512:
6
+ metadata.gz: 36cb5e15b9fa07d70ff05c5731f9d0c9dc902581d436e0252dd7e78468876df4471d369e82a2554527cf189e32028b6d6b166eb9f8461a53ebea93fe88e77a2c
7
+ data.tar.gz: 93bc33f7b006822a1d09da402af63b4b5d778a4d6cb1102e569ecaa6d45260bc51a3049a2dbd7254de6d5b49e60845954c1568dd8588d592186aef2a53e5a032
@@ -0,0 +1,20 @@
1
+ Copyright 2017 Jun Jiang
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,97 @@
1
+ Form Core
2
+ ====
3
+
4
+ A Rails engine providing ability to generate dynamic form.
5
+
6
+ ## Requirements
7
+
8
+ - MRI 2.3+
9
+ - Rails 5.0+
10
+
11
+ ## Usage
12
+
13
+ See demo for now.
14
+
15
+ ## Installation
16
+
17
+ Add this line to your Gemfile:
18
+
19
+ ```ruby
20
+ gem 'form_core'
21
+ ```
22
+
23
+ Or you may want to include the gem directly from GitHub:
24
+
25
+ ```ruby
26
+ gem 'form_core', github: 'jasl-lab/form_core'
27
+ ```
28
+
29
+ And then execute:
30
+
31
+ ```sh
32
+ $ bundle
33
+ ```
34
+
35
+ Copy migrations
36
+
37
+ ```sh
38
+ $ bin/rails form_core:install:migrations
39
+ ```
40
+
41
+ Then do migrate
42
+
43
+ ```sh
44
+ $ bin/rails db:migrate
45
+ ```
46
+
47
+ ## Demo
48
+
49
+ Clone the repository.
50
+
51
+ ```sh
52
+ $ git clone https://github.com/jasl-lab/form_core.git
53
+ ```
54
+
55
+ Change directory
56
+
57
+ ```sh
58
+ $ cd form_core
59
+ ```
60
+
61
+ Run bundler
62
+
63
+ ```sh
64
+ $ bundle install
65
+ ```
66
+
67
+ Preparing database
68
+
69
+ ```sh
70
+ $ bin/rails db:migrate
71
+ ```
72
+
73
+ Start the Rails server
74
+
75
+ ```sh
76
+ $ bin/rails s
77
+ ```
78
+
79
+ Open your browser, and visit `http://localhost:3000`
80
+
81
+ ## Contributing
82
+
83
+ Bug report or pull request are welcome.
84
+
85
+ ### Make a pull request
86
+
87
+ 1. Fork it
88
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
89
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
90
+ 4. Push to the branch (`git push origin my-new-feature`)
91
+ 5. Create new Pull Request
92
+
93
+ Please write unit test with your code if necessary.
94
+
95
+ ## License
96
+
97
+ The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
@@ -0,0 +1,34 @@
1
+ # frozen_string_literal: true
2
+
3
+ begin
4
+ require "bundler/setup"
5
+ rescue LoadError
6
+ puts "You must `gem install bundler` and `bundle install` to run rake tasks"
7
+ end
8
+
9
+ require "rdoc/task"
10
+
11
+ RDoc::Task.new(:rdoc) do |rdoc|
12
+ rdoc.rdoc_dir = "rdoc"
13
+ rdoc.title = "FormCore"
14
+ rdoc.options << "--line-numbers"
15
+ rdoc.rdoc_files.include("README.md")
16
+ rdoc.rdoc_files.include("lib/**/*.rb")
17
+ end
18
+
19
+ APP_RAKEFILE = File.expand_path("../test/dummy/Rakefile", __FILE__)
20
+ load "rails/tasks/engine.rake"
21
+
22
+ load "rails/tasks/statistics.rake"
23
+
24
+ require "bundler/gem_tasks"
25
+
26
+ require "rake/testtask"
27
+
28
+ Rake::TestTask.new(:test) do |t|
29
+ t.libs << "test"
30
+ t.pattern = "test/**/*_test.rb"
31
+ t.verbose = false
32
+ end
33
+
34
+ task default: :test
@@ -0,0 +1,48 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "form_core/engine"
4
+ require "form_core/errors"
5
+
6
+ require "form_core/coder"
7
+ require "form_core/coders/hash_coder"
8
+ require "form_core/coders/yaml_coder"
9
+
10
+ require "form_core/virtual_model"
11
+ require "form_core/concerns/models/form"
12
+ require "form_core/concerns/models/field"
13
+
14
+ module FormCore
15
+ class << self
16
+ def virtual_model_class
17
+ @virtual_model_class ||= VirtualModel
18
+ end
19
+
20
+ def virtual_model_class=(klass)
21
+ unless klass && klass < VirtualModel
22
+ raise ArgumentError, "#{klass} should be sub-class of #{VirtualModel}."
23
+ end
24
+
25
+ @reserved_names = nil
26
+ @virtual_model_class = klass
27
+ end
28
+
29
+ def reserved_names
30
+ @reserved_names ||= Set.new(
31
+ %i(def class module private public protected allocate new parent superclass) +
32
+ virtual_model_class.instance_methods(true)
33
+ )
34
+ end
35
+
36
+ def virtual_model_coder_class
37
+ @virtual_model_coder_class ||= HashCoder
38
+ end
39
+
40
+ def virtual_model_coder_class=(klass)
41
+ unless klass && klass < Coder
42
+ raise ArgumentError, "#{klass} should be sub-class of #{Coder}."
43
+ end
44
+
45
+ @virtual_model_coder_class = klass
46
+ end
47
+ end
48
+ end
@@ -0,0 +1,43 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore
4
+ class Coder
5
+ cattr_accessor :strict
6
+
7
+ attr_reader :object_class
8
+
9
+ def initialize(object_class)
10
+ @object_class = object_class
11
+ end
12
+
13
+ def strict?
14
+ Coder.strict
15
+ end
16
+
17
+ def dump(_obj)
18
+ raise NotImplementedError
19
+ end
20
+
21
+ def load(_src)
22
+ raise NotImplementedError
23
+ end
24
+
25
+ private
26
+
27
+ def new_or_raise_decoding_error
28
+ if strict?
29
+ raise DecodingDataCorrupted
30
+ else
31
+ object_class.new
32
+ end
33
+ end
34
+
35
+ def valid_attribute_names
36
+ object_class.attribute_names + object_class._embeds_reflections.keys
37
+ end
38
+
39
+ def valid_attributes(hash)
40
+ hash.slice(*valid_attribute_names)
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "yaml"
4
+
5
+ module FormCore
6
+ class HashCoder < FormCore::Coder # :nodoc:
7
+ def dump(obj)
8
+ obj&.serializable_hash || {}
9
+ end
10
+
11
+ def load(hash)
12
+ if hash.nil? || !hash.respond_to?(:to_h)
13
+ return new_or_raise_decoding_error
14
+ end
15
+
16
+ object_class.new valid_attributes(hash)
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,43 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "yaml"
4
+
5
+ module FormCore
6
+ class YAMLCoder < FormCore::Coder # :nodoc:
7
+ cattr_accessor :safe_mode
8
+
9
+ def self.whitelist_classes
10
+ @whitelist_classes ||= []
11
+ end
12
+
13
+ def safe_mode?
14
+ YAMLCoder.safe_mode
15
+ end
16
+
17
+ def dump(obj)
18
+ return YAML.dump({}) unless obj
19
+
20
+ YAML.dump obj.serializable_hash
21
+ end
22
+
23
+ def load(yaml)
24
+ return object_class.new if yaml.blank?
25
+
26
+ unless yaml.is_a?(String) && /^---/.match?(yaml)
27
+ return new_or_raise_decoding_error
28
+ end
29
+
30
+ decoded =
31
+ if safe_mode?
32
+ YAML.safe_load(yaml, YAMLCoder.whitelist_classes)
33
+ else
34
+ YAML.load(yaml)
35
+ end
36
+ unless decoded.is_a? Hash
37
+ return new_or_raise_decoding_error
38
+ end
39
+
40
+ object_class.new valid_attributes(decoded)
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,72 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore::Concerns
4
+ module Models
5
+ module Field
6
+ extend ActiveSupport::Concern
7
+
8
+ NAME_REGEX = /\A[a-z_][a-z_0-9]*\z/
9
+
10
+ included do
11
+
12
+ serialize :validations
13
+ serialize :options
14
+
15
+ validates :name,
16
+ presence: true,
17
+ uniqueness: {scope: :form},
18
+ exclusion: {in: FormCore.reserved_names},
19
+ format: {with: NAME_REGEX}
20
+
21
+ after_initialize do
22
+ self.validations ||= {}
23
+ self.options ||= {}
24
+ end
25
+ end
26
+
27
+ def name
28
+ self[:name]&.to_sym
29
+ end
30
+
31
+ def stored_type
32
+ raise NotImplementedError
33
+ end
34
+
35
+ def default_value
36
+ nil
37
+ end
38
+
39
+ def interpret_to(model, overrides: {})
40
+ check_model_validity!(model)
41
+
42
+ default_value = overrides.fetch(:default_value, self.default_value)
43
+ model.attribute name, stored_type, default: default_value
44
+
45
+ interpret_validations_to model, overrides
46
+ interpret_extra_to model, overrides
47
+
48
+ model
49
+ end
50
+
51
+ protected
52
+
53
+ def interpret_validations_to(model, overrides = {})
54
+ validations = overrides.fetch(:validations, (self.validations || {}))
55
+ validation_options = overrides.fetch(:validation_options) { self.options.fetch(:validation, {}) }
56
+
57
+ if validations.present?
58
+ model.validates name, **validations, **validation_options
59
+ end
60
+ end
61
+
62
+ def interpret_extra_to(_model, _overrides = {})
63
+ end
64
+
65
+ def check_model_validity!(model)
66
+ unless model.is_a?(Class) && model < ::FormCore::VirtualModel
67
+ raise ArgumentError, "#{model} must be a #{::FormCore::VirtualModel}'s subclass"
68
+ end
69
+ end
70
+ end
71
+ end
72
+ end
@@ -0,0 +1,38 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore::Concerns
4
+ module Models
5
+ module Form
6
+ extend ActiveSupport::Concern
7
+
8
+ def to_virtual_model(model_name: "Form",
9
+ fields_scope: proc { |fields| fields },
10
+ overrides: {})
11
+ model = FormCore.virtual_model_class.build model_name
12
+
13
+ append_to_virtual_model(model, fields_scope: fields_scope, overrides: overrides)
14
+ end
15
+
16
+ def append_to_virtual_model(model,
17
+ fields_scope: proc { |fields| fields },
18
+ overrides: {})
19
+ check_model_validity! model
20
+
21
+ global_overrides = overrides.fetch(:_global, {})
22
+ fields_scope.call(fields).each do |f|
23
+ f.interpret_to model, overrides: global_overrides.merge(overrides.fetch(f.name, {}))
24
+ end
25
+
26
+ model
27
+ end
28
+
29
+ private
30
+
31
+ def check_model_validity!(model)
32
+ unless model.is_a?(Class) && model < ::FormCore::VirtualModel
33
+ raise ArgumentError, "#{model} must be a #{::FormCore::VirtualModel}'s subclass"
34
+ end
35
+ end
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore
4
+ class Engine < ::Rails::Engine
5
+ isolate_namespace FormCore
6
+ end
7
+ end
@@ -0,0 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore
4
+ # = Form Core Errors
5
+ #
6
+ # Generic Form Core exception class.
7
+ class FormCoreError < StandardError
8
+ end
9
+
10
+ class DecodingDataCorrupted < FormCoreError
11
+ end
12
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FormCore
4
+ VERSION = "0.0.14"
5
+ end
@@ -0,0 +1,77 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "duck_record"
4
+
5
+ module FormCore
6
+ class VirtualModel < ::DuckRecord::Base
7
+ # Returns the contents of the record as a nicely formatted string.
8
+ def inspect
9
+ # We check defined?(@attributes) not to issue warnings if the object is
10
+ # allocated but not initialized.
11
+ inspection =
12
+ if defined?(@attributes) && @attributes
13
+ self.class.attribute_names.collect do |name|
14
+ if has_attribute?(name)
15
+ "#{name}: #{attribute_for_inspect(name)}"
16
+ end
17
+ end.compact.join(", ")
18
+ else
19
+ "not initialized"
20
+ end
21
+
22
+ "#<VirtualModel:#{self.class.name}:#{object_id} #{inspection}>"
23
+ end
24
+
25
+ def serializable_hash(options = {})
26
+ options = (options || {}).reverse_merge include: self.class._embeds_reflections.keys
27
+ super options
28
+ end
29
+
30
+ def dump
31
+ self.class.dump(self)
32
+ end
33
+
34
+ class << self
35
+ def name
36
+ @_name ||= "Form"
37
+ end
38
+
39
+ def name=(value)
40
+ value = value.classify
41
+ raise ArgumentError, "`value` isn't a valid class name" if value.blank?
42
+
43
+ @_name = value
44
+ end
45
+
46
+ def coder
47
+ @_coder ||= FormCore.virtual_model_coder_class.new(self)
48
+ end
49
+
50
+ def coder=(klass)
51
+ unless klass && klass < Coder
52
+ raise ArgumentError, "#{klass} should be sub-class of #{Coder}."
53
+ end
54
+
55
+ @_coder = klass.new(self)
56
+ end
57
+
58
+ delegate :dump, :load, to: :coder, allow_nil: false
59
+
60
+ def build(name = nil)
61
+ klass = Class.new(self)
62
+ klass.name = name
63
+ klass
64
+ end
65
+
66
+ # Returns a string like "Post(id:integer, title:string, body:text)"
67
+ def inspect
68
+ attr_list = attribute_types.map { |name, type| "#{name}: #{type.type}" } * ", "
69
+ "#<VirtualModel:#{name}:#{object_id} #{attr_list}>"
70
+ end
71
+
72
+ def _embeds_reflections
73
+ _reflections.select { |_, v| v.is_a? DuckRecord::Reflection::EmbedsAssociationReflection }
74
+ end
75
+ end
76
+ end
77
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+ # desc "Explaining what the task does"
3
+ # task :form_core do
4
+ # # Task goes here
5
+ # end
metadata ADDED
@@ -0,0 +1,88 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: form_core_v1
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.14
5
+ platform: ruby
6
+ authors:
7
+ - jasl
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2018-02-24 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rails
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '5.0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '5.0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: duck_record
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ description: |
42
+ A Rails engine providing ability to generate dynamic form.
43
+ It's would make such as dynamic fields of model or questionnaire easily.
44
+ email:
45
+ - jasl9187@hotmail.com
46
+ executables: []
47
+ extensions: []
48
+ extra_rdoc_files: []
49
+ files:
50
+ - MIT-LICENSE
51
+ - README.md
52
+ - Rakefile
53
+ - lib/form_core.rb
54
+ - lib/form_core/coder.rb
55
+ - lib/form_core/coders/hash_coder.rb
56
+ - lib/form_core/coders/yaml_coder.rb
57
+ - lib/form_core/concerns/models/field.rb
58
+ - lib/form_core/concerns/models/form.rb
59
+ - lib/form_core/engine.rb
60
+ - lib/form_core/errors.rb
61
+ - lib/form_core/version.rb
62
+ - lib/form_core/virtual_model.rb
63
+ - lib/tasks/form_core_tasks.rake
64
+ homepage: https://github.com/jasl-lab/form_core
65
+ licenses:
66
+ - MIT
67
+ metadata: {}
68
+ post_install_message:
69
+ rdoc_options: []
70
+ require_paths:
71
+ - lib
72
+ required_ruby_version: !ruby/object:Gem::Requirement
73
+ requirements:
74
+ - - ">="
75
+ - !ruby/object:Gem::Version
76
+ version: '0'
77
+ required_rubygems_version: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - ">="
80
+ - !ruby/object:Gem::Version
81
+ version: '0'
82
+ requirements: []
83
+ rubyforge_project:
84
+ rubygems_version: 2.6.11
85
+ signing_key:
86
+ specification_version: 4
87
+ summary: A Rails engine providing ability to generate dynamic form.
88
+ test_files: []