From ecd7bbc7935ebc39548df59a1b906be7388a9033 Mon Sep 17 00:00:00 2001 From: mnauage Date: Mon, 12 Mar 2018 12:19:19 -0400 Subject: [PATCH 01/73] links key support --- lib/fast_jsonapi/object_serializer.rb | 3 +++ spec/lib/object_serializer_caching_spec.rb | 4 ++++ spec/lib/object_serializer_spec.rb | 3 +++ spec/lib/object_serializer_struct_spec.rb | 3 +++ 4 files changed, 13 insertions(+) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 2f5ebe4..6de62ea 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -34,6 +34,7 @@ module FastJsonapi def hash_for_one_record serializable_hash = { data: nil } serializable_hash[:meta] = @meta if @meta.present? + serializable_hash[:links] = @links if @links.present? return serializable_hash unless @resource @@ -55,6 +56,7 @@ module FastJsonapi serializable_hash[:data] = data serializable_hash[:included] = included if @includes.present? serializable_hash[:meta] = @meta if @meta.present? + serializable_hash[:links] = @links if @links.present? serializable_hash end @@ -69,6 +71,7 @@ module FastJsonapi @known_included_objects = {} @meta = options[:meta] + @links = options[:links] if options[:include].present? @includes = options[:include].delete_if(&:blank?).map(&:to_sym) diff --git a/spec/lib/object_serializer_caching_spec.rb b/spec/lib/object_serializer_caching_spec.rb index de0186a..8e455d4 100644 --- a/spec/lib/object_serializer_caching_spec.rb +++ b/spec/lib/object_serializer_caching_spec.rb @@ -13,6 +13,8 @@ describe FastJsonapi::ObjectSerializer do it 'returns correct hash when serializable_hash is called' do options = {} options[:meta] = { total: 2 } + options[:links] = { self: 'self' } + options[:include] = [:actors] serializable_hash = CachingMovieSerializer.new([movie, movie], options).serializable_hash @@ -21,6 +23,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data][0][:attributes].length).to eq 2 expect(serializable_hash[:meta]).to be_instance_of(Hash) + expect(serializable_hash[:links]).to be_instance_of(Hash) expect(serializable_hash[:included]).to be_instance_of(Array) expect(serializable_hash[:included][0]).to be_instance_of(Hash) @@ -30,6 +33,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data]).to be_instance_of(Hash) expect(serializable_hash[:meta]).to be nil + expect(serializable_hash[:links]).to be nil expect(serializable_hash[:included]).to be nil end diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 83e1eee..111c2d4 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -7,6 +7,7 @@ describe FastJsonapi::ObjectSerializer do it 'returns correct hash when serializable_hash is called' do options = {} options[:meta] = { total: 2 } + options[:links] = { self: 'self' } options[:include] = [:actors] serializable_hash = MovieSerializer.new([movie, movie], options).serializable_hash @@ -15,6 +16,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data][0][:attributes].length).to eq 2 expect(serializable_hash[:meta]).to be_instance_of(Hash) + expect(serializable_hash[:links]).to be_instance_of(Hash) expect(serializable_hash[:included]).to be_instance_of(Array) expect(serializable_hash[:included][0]).to be_instance_of(Hash) @@ -24,6 +26,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data]).to be_instance_of(Hash) expect(serializable_hash[:meta]).to be nil + expect(serializable_hash[:links]).to be nil expect(serializable_hash[:included]).to be nil end diff --git a/spec/lib/object_serializer_struct_spec.rb b/spec/lib/object_serializer_struct_spec.rb index cb9054b..d4cda1b 100644 --- a/spec/lib/object_serializer_struct_spec.rb +++ b/spec/lib/object_serializer_struct_spec.rb @@ -7,6 +7,7 @@ describe FastJsonapi::ObjectSerializer do it 'returns correct hash when serializable_hash is called' do options = {} options[:meta] = { total: 2 } + options[:links] = { self: 'self' } options[:include] = [:actors] serializable_hash = MovieSerializer.new([movie_struct, movie_struct], options).serializable_hash @@ -15,6 +16,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data][0][:attributes].length).to eq 2 expect(serializable_hash[:meta]).to be_instance_of(Hash) + expect(serializable_hash[:links]).to be_instance_of(Hash) expect(serializable_hash[:included]).to be_instance_of(Array) expect(serializable_hash[:included][0]).to be_instance_of(Hash) @@ -24,6 +26,7 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data]).to be_instance_of(Hash) expect(serializable_hash[:meta]).to be nil + expect(serializable_hash[:links]).to be nil expect(serializable_hash[:included]).to be nil expect(serializable_hash[:data][:id]).to eq movie_struct.id.to_s end From 1e6d127aec38d5a1c128ce710613ba9c83332fc3 Mon Sep 17 00:00:00 2001 From: mnauage Date: Tue, 13 Mar 2018 10:48:05 -0400 Subject: [PATCH 02/73] links doc --- README.md | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/README.md b/README.md index bff28e5..1bc870a 100644 --- a/README.md +++ b/README.md @@ -211,6 +211,11 @@ Support for top-level included member through ` options[:include] `. ```ruby options = {} options[:meta] = { total: 2 } +options[:links] = { + self: '...', + next: '...', + prev: '...' +} options[:include] = [:actors] MovieSerializer.new([movie, movie], options).serialized_json ``` @@ -219,6 +224,11 @@ MovieSerializer.new([movie, movie], options).serialized_json ```ruby options[:meta] = { total: 2 } +options[:links] = { + self: '...', + next: '...', + prev: '...' +} hash = MovieSerializer.new([movie, movie], options).serializable_hash json_string = MovieSerializer.new([movie, movie], options).serialized_json ``` From 88553cf9aba8d8ca0e6e709e31198be03e424160 Mon Sep 17 00:00:00 2001 From: Roberto Quintanilla Date: Fri, 9 Feb 2018 10:12:11 -0600 Subject: [PATCH 03/73] Updated README with details about polymorphic associations --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 1bc870a..4b078b1 100644 --- a/README.md +++ b/README.md @@ -255,6 +255,8 @@ id_method_name | Set custom method name to get ID of an object | ```has_many :lo object_method_name | Set custom method name to get related objects | ```has_many :locations, object_method_name: :places ``` record_type | Set custom Object Type for a relationship | ```belongs_to :owner, record_type: :user``` serializer | Set custom Serializer for a relationship | ```has_many :actors, serializer: :custom_actor``` +polymorphic | Allows different record types for a polymorphic association | ```has_many :targets, polymorphic: true``` +polymorphic | Sets custom record types for each object class in a polymorphic association | ```has_many :targets, polymorphic: { Person => :person, Group => :group }``` ### Instrumentation From b387f94a13a7452dbaf156d0e434427254a26eb2 Mon Sep 17 00:00:00 2001 From: sojan v jose Date: Tue, 20 Mar 2018 01:23:24 +0530 Subject: [PATCH 04/73] Fix typo in the readme sample command (#131) fix typo in the readme sample command --- Gemfile.lock | 2 +- README.md | 2 +- fast_jsonapi.gemspec | 2 +- spec/lib/object_serializer_performance_spec.rb | 1 + 4 files changed, 4 insertions(+), 3 deletions(-) diff --git a/Gemfile.lock b/Gemfile.lock index 5e082ef..5a3cb84 100644 --- a/Gemfile.lock +++ b/Gemfile.lock @@ -1,7 +1,7 @@ PATH remote: . specs: - fast_jsonapi (1.0.17) + fast_jsonapi (1.1.0) activesupport (>= 4.2) GEM diff --git a/README.md b/README.md index 4b078b1..90dca72 100644 --- a/README.md +++ b/README.md @@ -60,7 +60,7 @@ $ bundle install You can use the bundled generator if you are using the library inside of a Rails project: - rails g Serializer Movie name year + rails g serializer Movie name year This will create a new serializer in `app/serializers/movie_serializer.rb` diff --git a/fast_jsonapi.gemspec b/fast_jsonapi.gemspec index 302c0a5..24edd23 100644 --- a/fast_jsonapi.gemspec +++ b/fast_jsonapi.gemspec @@ -1,6 +1,6 @@ Gem::Specification.new do |gem| gem.name = "fast_jsonapi" - gem.version = "1.0.17" + gem.version = "1.1.0" gem.required_rubygems_version = Gem::Requirement.new(">= 0") if gem.respond_to? :required_rubygems_version= gem.metadata = { "allowed_push_host" => "https://rubygems.org" } if gem.respond_to? :metadata= diff --git a/spec/lib/object_serializer_performance_spec.rb b/spec/lib/object_serializer_performance_spec.rb index e718953..8e52dd2 100644 --- a/spec/lib/object_serializer_performance_spec.rb +++ b/spec/lib/object_serializer_performance_spec.rb @@ -103,6 +103,7 @@ describe FastJsonapi::ObjectSerializer, performance: true do data = Hash[serializers.keys.collect { |k| [ k, { json: nil, time: nil, speed_factor: nil }] }] serializers.each_pair do |k,v| + ams_json = nil json_method = SERIALIZERS[k].key?(:json_method) ? SERIALIZERS[k][:json_method] : :to_json data[k][:time] = Benchmark.measure { data[k][:json] = v.send(json_method) }.real * 1000 end From fea384b4c60cb477822e908627270869f73fca59 Mon Sep 17 00:00:00 2001 From: Rob Wise Date: Tue, 20 Mar 2018 02:59:18 -0400 Subject: [PATCH 05/73] update README to mention cache_key is required see https://github.com/Netflix/fast_jsonapi/issues/99#issuecomment-374060813 --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 90dca72..db2d1d5 100644 --- a/README.md +++ b/README.md @@ -234,6 +234,7 @@ json_string = MovieSerializer.new([movie, movie], options).serialized_json ``` ### Caching +Requires a `cache_key` method be defined on model: ```ruby class MovieSerializer From b18da3da59d7691020fac511e1d5a80f51bc63c4 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Wed, 21 Mar 2018 17:18:24 +0900 Subject: [PATCH 06/73] Remove unused variable from object_serializer_class_methods_spec --- spec/lib/object_serializer_class_methods_spec.rb | 1 - 1 file changed, 1 deletion(-) diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 4d23c2b..7147371 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -67,7 +67,6 @@ describe FastJsonapi::ObjectSerializer do it 'sets the correct transform_method when use_hyphen is used' do MovieSerializer.use_hyphen - warning_message = 'DEPRECATION WARNING: use_hyphen is deprecated and will be removed from fast_jsonapi 2.0 use (set_key_transform :dash) instead' expect { MovieSerializer.use_hyphen }.to output.to_stderr expect(MovieSerializer.instance_variable_get(:@transform_method)).to eq :dasherize end From 4f3a903e64ed8b1e72a562b2e674a9d3d04886c3 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Mon, 26 Mar 2018 09:47:16 +0900 Subject: [PATCH 07/73] Remove redundant fetch from ids_hash_from_record_and_relationship --- lib/fast_jsonapi/serialization_core.rb | 3 +-- spec/lib/serialization_core_spec.rb | 2 +- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 3c2e9b7..8009edf 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -44,8 +44,7 @@ module FastJsonapi relationship[:record_type] ) unless polymorphic - object_method_name = relationship.fetch(:object_method_name, relationship[:name]) - return unless associated_object = record.send(object_method_name) + return unless associated_object = record.send(relationship[:object_method_name]) return associated_object.map do |object| id_hash_from_record object, polymorphic diff --git a/spec/lib/serialization_core_spec.rb b/spec/lib/serialization_core_spec.rb index 2d327d4..5d4563f 100644 --- a/spec/lib/serialization_core_spec.rb +++ b/spec/lib/serialization_core_spec.rb @@ -18,7 +18,7 @@ describe FastJsonapi::ObjectSerializer do end it 'returns the correct hash when ids_hash_from_record_and_relationship is called for a polymorphic association' do - relationship = { name: :groupees, relationship_type: :has_many, polymorphic: {} } + relationship = { name: :groupees, relationship_type: :has_many, object_method_name: :groupees, polymorphic: {} } results = GroupSerializer.send :ids_hash_from_record_and_relationship, group, relationship expect(results).to include({ id: "1", type: :person }, { id: "2", type: :group }) end From cdfac8743d563504ae9969650eca90ffb312ba01 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Tue, 27 Mar 2018 14:16:37 +0900 Subject: [PATCH 08/73] Refactor object serializer class methods spec (#134) * Add object_serializer_class_methods_examples * Change to require spec/shared/examples in every spec files * Refactor object_serializer_class_methods_spec --- .../object_serializer_class_methods_spec.rb | 134 ++++++++++-------- ...bject_serializer_class_methods_examples.rb | 9 ++ spec/spec_helper.rb | 1 + 3 files changed, 88 insertions(+), 56 deletions(-) create mode 100644 spec/shared/examples/object_serializer_class_methods_examples.rb diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 7147371..b8f5133 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -4,72 +4,94 @@ describe FastJsonapi::ObjectSerializer do include_context 'movie class' - context 'when testing class methods of object serializer' do + describe '#has_many' do + subject(:relationship) { serializer.relationships_to_serialize[:roles] } - before(:example) do - MovieSerializer.relationships_to_serialize = {} + before do + serializer.has_many *children end - it 'returns correct relationship hash for a has_many relationship' do - MovieSerializer.has_many :roles - relationship = MovieSerializer.relationships_to_serialize[:roles] - expect(relationship).to be_instance_of(Hash) - expect(relationship.keys).to all(be_instance_of(Symbol)) - expect(relationship[:id_method_name]).to end_with '_ids' - expect(relationship[:record_type]).to eq 'roles'.singularize.to_sym + context 'with namespace' do + let(:serializer) { AppName::V1::MovieSerializer } + let(:children) { [:roles] } + + context 'with overrides' do + let(:children) { [:roles, id_method_name: :roles_only_ids, record_type: :super_role] } + + it_behaves_like 'returning correct relationship hash', :'AppName::V1::RoleSerializer', :roles_only_ids, :super_role + end + + context 'without overrides' do + let(:children) { [:roles] } + + it_behaves_like 'returning correct relationship hash', :'AppName::V1::RoleSerializer', :role_ids, :role + end end - it 'returns correct relationship hash for a has_many relationship with overrides' do - MovieSerializer.has_many :roles, id_method_name: :roles_only_ids, record_type: :super_role - relationship = MovieSerializer.relationships_to_serialize[:roles] - expect(relationship[:id_method_name]).to be :roles_only_ids - expect(relationship[:record_type]).to be :super_role - end + context 'without namespace' do + let(:serializer) { MovieSerializer } - it 'returns correct relationship hash for a belongs_to relationship' do - MovieSerializer.belongs_to :area - relationship = MovieSerializer.relationships_to_serialize[:area] - expect(relationship).to be_instance_of(Hash) - expect(relationship.keys).to all(be_instance_of(Symbol)) - expect(relationship[:id_method_name]).to end_with '_id' - expect(relationship[:record_type]).to eq 'area'.singularize.to_sym - end + context 'with overrides' do + let(:children) { [:roles, id_method_name: :roles_only_ids, record_type: :super_role] } - it 'returns correct relationship hash for a belongs_to relationship with overrides' do - MovieSerializer.has_many :area, id_method_name: :blah_id, record_type: :awesome_area, serializer: :my_area - relationship = MovieSerializer.relationships_to_serialize[:area] - expect(relationship[:id_method_name]).to be :blah_id - expect(relationship[:record_type]).to be :awesome_area - expect(relationship[:serializer]).to be :MyAreaSerializer - end + it_behaves_like 'returning correct relationship hash', :'RoleSerializer', :roles_only_ids, :super_role + end - it 'returns correct relationship hash for a has_one relationship' do - MovieSerializer.has_one :area - relationship = MovieSerializer.relationships_to_serialize[:area] - expect(relationship).to be_instance_of(Hash) - expect(relationship.keys).to all(be_instance_of(Symbol)) - expect(relationship[:id_method_name]).to end_with '_id' - expect(relationship[:record_type]).to eq 'area'.singularize.to_sym - end + context 'without overrides' do + let(:children) { [:roles] } - it 'returns correct relationship hash for a has_one relationship with overrides' do - MovieSerializer.has_one :area, id_method_name: :blah_id, record_type: :awesome_area - relationship = MovieSerializer.relationships_to_serialize[:area] - expect(relationship[:id_method_name]).to be :blah_id - expect(relationship[:record_type]).to be :awesome_area - end - - it 'returns serializer name correctly with namespaces' do - AppName::V1::MovieSerializer.has_many :area, id_method_name: :blah_id - relationship = AppName::V1::MovieSerializer.relationships_to_serialize[:area] - expect(relationship[:serializer]).to be :'AppName::V1::AreaSerializer' - end - - it 'sets the correct transform_method when use_hyphen is used' do - MovieSerializer.use_hyphen - expect { MovieSerializer.use_hyphen }.to output.to_stderr - expect(MovieSerializer.instance_variable_get(:@transform_method)).to eq :dasherize + it_behaves_like 'returning correct relationship hash', :'RoleSerializer', :role_ids, :role + end end end + describe '#belongs_to' do + subject(:relationship) { MovieSerializer.relationships_to_serialize[:area] } + + before do + MovieSerializer.belongs_to *parent + end + + context 'with overrides' do + let(:parent) { [:area, id_method_name: :blah_id, record_type: :awesome_area, serializer: :my_area] } + + it_behaves_like 'returning correct relationship hash', :'MyAreaSerializer', :blah_id, :awesome_area + end + + context 'without overrides' do + let(:parent) { [:area] } + + it_behaves_like 'returning correct relationship hash', :'AreaSerializer', :area_id, :area + end + end + + describe '#has_one' do + subject(:relationship) { MovieSerializer.relationships_to_serialize[:area] } + + before do + MovieSerializer.has_one *partner + end + + context 'with overrides' do + let(:partner) { [:area, id_method_name: :blah_id, record_type: :awesome_area, serializer: :my_area] } + + it_behaves_like 'returning correct relationship hash', :'MyAreaSerializer', :blah_id, :awesome_area + end + + context 'without overrides' do + let(:partner) { [:area] } + + it_behaves_like 'returning correct relationship hash', :'AreaSerializer', :area_id, :area + end + end + + describe '#use_hyphen' do + subject { MovieSerializer.use_hyphen } + + it 'sets the correct transform_method when use_hyphen is used' do + warning_message = "DEPRECATION WARNING: use_hyphen is deprecated and will be removed from fast_jsonapi 2.0 use (set_key_transform :dash) instead\n" + expect { subject }.to output(warning_message).to_stderr + expect(MovieSerializer.instance_variable_get(:@transform_method)).to eq :dasherize + end + end end diff --git a/spec/shared/examples/object_serializer_class_methods_examples.rb b/spec/shared/examples/object_serializer_class_methods_examples.rb new file mode 100644 index 0000000..4a96e56 --- /dev/null +++ b/spec/shared/examples/object_serializer_class_methods_examples.rb @@ -0,0 +1,9 @@ +RSpec.shared_examples 'returning correct relationship hash' do |serializer, id_method_name, record_type| + it 'returns correct relationship hash' do + expect(relationship).to be_instance_of(Hash) + expect(relationship.keys).to all(be_instance_of(Symbol)) + expect(relationship[:serializer]).to be serializer + expect(relationship[:id_method_name]).to be id_method_name + expect(relationship[:record_type]).to be record_type + end +end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index d7aac0c..9110674 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -7,6 +7,7 @@ require 'jsonapi/serializable' require 'jsonapi-serializers' Dir[File.dirname(__FILE__) + '/shared/contexts/*.rb'].each {|file| require file } +Dir[File.dirname(__FILE__) + '/shared/examples/*.rb'].each {|file| require file } RSpec.configure do |config| config.include RSpec::Benchmark::Matchers From c2e4c01bf17de8628843ea6a2011c9d07d5375f6 Mon Sep 17 00:00:00 2001 From: Dmitriy Ivliev Date: Fri, 16 Mar 2018 09:30:20 +0400 Subject: [PATCH 09/73] fix behaviour for struct without id --- lib/fast_jsonapi/serialization_core.rb | 2 +- spec/lib/object_serializer_struct_spec.rb | 20 ++++++++++++++ spec/shared/contexts/movie_context.rb | 32 ++++++++++++++++------- 3 files changed, 43 insertions(+), 11 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 8009edf..e3f0e33 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -22,7 +22,7 @@ module FastJsonapi class_methods do def id_hash(id, record_type) - return { id: id.to_s, type: record_type } if id.present? + { id: id.to_s, type: record_type } if id.present? end def ids_hash(ids, record_type) diff --git a/spec/lib/object_serializer_struct_spec.rb b/spec/lib/object_serializer_struct_spec.rb index d4cda1b..7c848dc 100644 --- a/spec/lib/object_serializer_struct_spec.rb +++ b/spec/lib/object_serializer_struct_spec.rb @@ -30,5 +30,25 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:included]).to be nil expect(serializable_hash[:data][:id]).to eq movie_struct.id.to_s end + + context 'struct without id' do + it 'returns correct hash when serializable_hash is called' do + options = {} + options[:meta] = { total: 2 } + serializable_hash = MovieWithoutIdStructSerializer.new([movie_struct_without_id, movie_struct_without_id], options).serializable_hash + + expect(serializable_hash[:data].length).to eq 2 + expect(serializable_hash[:data][0][:attributes].length).to eq 2 + expect(serializable_hash[:meta]).to be_instance_of(Hash) + + serializable_hash = MovieWithoutIdStructSerializer.new(movie_struct_without_id).serializable_hash + + expect(serializable_hash[:data]).to be_instance_of(Hash) + expect(serializable_hash[:meta]).to be nil + expect(serializable_hash[:included]).to be nil + expect(serializable_hash[:data][:id]).to be nil + expect(serializable_hash[:data][:type]).to be nil + end + end end end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index f26f21f..89b197e 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -5,11 +5,11 @@ RSpec.shared_context 'movie class' do # models class Movie attr_accessor :id, - :name, + :name, :release_year, :director, - :actor_ids, - :owner_id, + :actor_ids, + :owner_id, :movie_type_id def actors @@ -69,6 +69,11 @@ RSpec.shared_context 'movie class' do belongs_to :movie_type end + class MovieWithoutIdStructSerializer + include FastJsonapi::ObjectSerializer + attributes :name, :release_year + end + class CachingMovieSerializer include FastJsonapi::ObjectSerializer set_type :movie @@ -142,17 +147,18 @@ RSpec.shared_context 'movie class' do # Movie and Actor struct before(:context) do MovieStruct = Struct.new( - :id, - :name, - :release_year, - :actor_ids, - :actors, - :owner_id, - :owner, + :id, + :name, + :release_year, + :actor_ids, + :actors, + :owner_id, + :owner, :movie_type_id ) ActorStruct = Struct.new(:id, :name, :email) + MovieWithoutIdStruct = Struct.new(:name, :release_year) end after(:context) do @@ -167,7 +173,9 @@ RSpec.shared_context 'movie class' do AppName::V1::MovieSerializer MovieStruct ActorStruct + MovieWithoutIdStruct HyphenMovieSerializer + MovieWithoutIdStructSerializer ] classes_to_remove.each do |klass_name| Object.send(:remove_const, klass_name) if Object.constants.include?(klass_name) @@ -193,6 +201,10 @@ RSpec.shared_context 'movie class' do m end + let(:movie_struct_without_id) do + MovieWithoutIdStruct.new('struct without id', 2018) + end + let(:movie) do m = Movie.new m.id = 232 From 1196db46e5e67c25686688e5cffb427da170d33b Mon Sep 17 00:00:00 2001 From: Dmitriy Ivliev Date: Mon, 19 Mar 2018 10:54:13 +0400 Subject: [PATCH 10/73] add exception for missing id method --- lib/fast_jsonapi/serialization_core.rb | 14 ++++++++++---- spec/lib/object_serializer_struct_spec.rb | 17 ++--------------- 2 files changed, 12 insertions(+), 19 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index e3f0e33..c9ebddf 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -4,6 +4,8 @@ require 'active_support/concern' require 'fast_jsonapi/multi_to_json' module FastJsonapi + MandatoryField = Class.new(StandardError) + module SerializationCore extend ActiveSupport::Concern @@ -74,8 +76,7 @@ module FastJsonapi def record_hash(record) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length) do - id = record_id ? record.send(record_id) : record.id - temp_hash = id_hash(id, record_type) || { id: nil, type: record_type } + temp_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } temp_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? temp_hash[:relationships] = {} temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize) if cachable_relationships_to_serialize.present? @@ -84,14 +85,19 @@ module FastJsonapi record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize)) if uncachable_relationships_to_serialize.present? record_hash else - id = record_id ? record.send(record_id) : record.id - record_hash = id_hash(id, record_type) || { id: nil, type: record_type } + record_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } record_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? record_hash[:relationships] = relationships_hash(record) if relationships_to_serialize.present? record_hash end end + def id_from_record(record) + return record.send(record_id) if record_id + raise MandatoryField, 'id is a mandatory field in the jsonapi spec' unless record.respond_to?(:id) + record.id + end + # Override #to_json for alternative implementation def to_json(payload) FastJsonapi::MultiToJson.to_json(payload) if payload.present? diff --git a/spec/lib/object_serializer_struct_spec.rb b/spec/lib/object_serializer_struct_spec.rb index 7c848dc..443a259 100644 --- a/spec/lib/object_serializer_struct_spec.rb +++ b/spec/lib/object_serializer_struct_spec.rb @@ -33,21 +33,8 @@ describe FastJsonapi::ObjectSerializer do context 'struct without id' do it 'returns correct hash when serializable_hash is called' do - options = {} - options[:meta] = { total: 2 } - serializable_hash = MovieWithoutIdStructSerializer.new([movie_struct_without_id, movie_struct_without_id], options).serializable_hash - - expect(serializable_hash[:data].length).to eq 2 - expect(serializable_hash[:data][0][:attributes].length).to eq 2 - expect(serializable_hash[:meta]).to be_instance_of(Hash) - - serializable_hash = MovieWithoutIdStructSerializer.new(movie_struct_without_id).serializable_hash - - expect(serializable_hash[:data]).to be_instance_of(Hash) - expect(serializable_hash[:meta]).to be nil - expect(serializable_hash[:included]).to be nil - expect(serializable_hash[:data][:id]).to be nil - expect(serializable_hash[:data][:type]).to be nil + serializer = MovieWithoutIdStructSerializer.new(movie_struct_without_id) + expect { serializer.serializable_hash }.to raise_error(FastJsonapi::MandatoryField) end end end From 0d8bbedcdd01cc52bb074e08644da70c4b95308c Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Thu, 29 Mar 2018 20:08:39 +0900 Subject: [PATCH 11/73] Change transform_method to accessor --- lib/fast_jsonapi/object_serializer.rb | 4 ++-- lib/fast_jsonapi/serialization_core.rb | 1 + 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 6de62ea..350260d 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -111,11 +111,11 @@ module FastJsonapi dash: :dasherize, underscore: :underscore } - @transform_method = mapping[transform_name.to_sym] + self.transform_method = mapping[transform_name.to_sym] end def run_key_transform(input) - if @transform_method.present? + if self.transform_method.present? input.to_s.send(*@transform_method).to_sym else input.to_sym diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index c9ebddf..b03095d 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -15,6 +15,7 @@ module FastJsonapi :relationships_to_serialize, :cachable_relationships_to_serialize, :uncachable_relationships_to_serialize, + :transform_method, :record_type, :record_id, :cache_length, From fb7d01368a4e17821891b021feea12b31c896be0 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Thu, 29 Mar 2018 20:22:05 +0900 Subject: [PATCH 12/73] Integrate tests for #set_id and #attribute to object_serializer_class_methods_spec --- .../object_serializer_class_methods_spec.rb | 67 +++++++++++++++++++ spec/lib/object_serializer_set_id_spec.rb | 30 --------- ...ct_serializer_with_attribute_block_spec.rb | 13 ---- spec/shared/contexts/movie_context.rb | 10 --- 4 files changed, 67 insertions(+), 53 deletions(-) delete mode 100644 spec/lib/object_serializer_set_id_spec.rb delete mode 100644 spec/lib/object_serializer_with_attribute_block_spec.rb diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index b8f5133..7c6def2 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -11,6 +11,10 @@ describe FastJsonapi::ObjectSerializer do serializer.has_many *children end + after do + serializer.relationships_to_serialize = {} + end + context 'with namespace' do let(:serializer) { AppName::V1::MovieSerializer } let(:children) { [:roles] } @@ -52,6 +56,10 @@ describe FastJsonapi::ObjectSerializer do MovieSerializer.belongs_to *parent end + after do + MovieSerializer.relationships_to_serialize = {} + end + context 'with overrides' do let(:parent) { [:area, id_method_name: :blah_id, record_type: :awesome_area, serializer: :my_area] } @@ -72,6 +80,10 @@ describe FastJsonapi::ObjectSerializer do MovieSerializer.has_one *partner end + after do + MovieSerializer.relationships_to_serialize = {} + end + context 'with overrides' do let(:partner) { [:area, id_method_name: :blah_id, record_type: :awesome_area, serializer: :my_area] } @@ -85,13 +97,68 @@ describe FastJsonapi::ObjectSerializer do end end + describe '#set_id' do + subject(:serializable_hash) { MovieSerializer.new(resource).serializable_hash } + + before do + MovieSerializer.set_id :owner_id + end + + after do + MovieSerializer.set_id nil + end + + context 'when one record is given' do + let(:resource) { movie } + + it 'returns correct hash which id equals owner_id' do + expect(serializable_hash[:data][:id].to_i).to eq movie.owner_id + end + end + + context 'when an array of records is given' do + let(:resource) { [movie, movie] } + + it 'returns correct hash which id equals owner_id' do + expect(serializable_hash[:data][0][:id].to_i).to eq movie.owner_id + expect(serializable_hash[:data][1][:id].to_i).to eq movie.owner_id + end + end + end + describe '#use_hyphen' do subject { MovieSerializer.use_hyphen } + after do + MovieSerializer.transform_method = nil + end + it 'sets the correct transform_method when use_hyphen is used' do warning_message = "DEPRECATION WARNING: use_hyphen is deprecated and will be removed from fast_jsonapi 2.0 use (set_key_transform :dash) instead\n" expect { subject }.to output(warning_message).to_stderr expect(MovieSerializer.instance_variable_get(:@transform_method)).to eq :dasherize end end + + describe '#attribute' do + subject(:serializable_hash) { MovieSerializer.new(movie).serializable_hash } + + after do + MovieSerializer.attributes_to_serialize = {} + end + + context 'with block' do + before do + movie.release_year = 2008 + MovieSerializer.attribute :title_with_year do |record| + "#{record.name} (#{record.release_year})" + end + end + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:data][:attributes][:name]).to eq movie.name + expect(serializable_hash[:data][:attributes][:title_with_year]).to eq "#{movie.name} (#{movie.release_year})" + end + end + end end diff --git a/spec/lib/object_serializer_set_id_spec.rb b/spec/lib/object_serializer_set_id_spec.rb deleted file mode 100644 index 2f731fd..0000000 --- a/spec/lib/object_serializer_set_id_spec.rb +++ /dev/null @@ -1,30 +0,0 @@ -require 'spec_helper' - -describe FastJsonapi::ObjectSerializer do - include_context 'movie class' - - context 'when setting id' do - subject(:serializable_hash) { MovieSerializer.new(resource).serializable_hash } - - before(:all) do - MovieSerializer.set_id :owner_id - end - - context 'when one record is given' do - let(:resource) { movie } - - it 'returns correct hash which id equals owner_id' do - expect(serializable_hash[:data][:id].to_i).to eq movie.owner_id - end - end - - context 'when an array of records is given' do - let(:resource) { [movie, movie] } - - it 'returns correct hash which id equals owner_id' do - expect(serializable_hash[:data][0][:id].to_i).to eq movie.owner_id - expect(serializable_hash[:data][1][:id].to_i).to eq movie.owner_id - end - end - end -end diff --git a/spec/lib/object_serializer_with_attribute_block_spec.rb b/spec/lib/object_serializer_with_attribute_block_spec.rb deleted file mode 100644 index 885cae0..0000000 --- a/spec/lib/object_serializer_with_attribute_block_spec.rb +++ /dev/null @@ -1,13 +0,0 @@ -require 'spec_helper' - -describe FastJsonapi::ObjectSerializer do - include_context 'movie class' - - context 'when including attribute blocks' do - it 'returns correct hash when serializable_hash is called' do - serializable_hash = MovieSerializerWithAttributeBlock.new([movie]).serializable_hash - expect(serializable_hash[:data][0][:attributes][:name]).to eq movie.name - expect(serializable_hash[:data][0][:attributes][:title_with_year]).to eq "#{movie.name} (#{movie.release_year})" - end - end -end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 89b197e..5daac5b 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -108,15 +108,6 @@ RSpec.shared_context 'movie class' do attributes :name end - class MovieSerializerWithAttributeBlock - include FastJsonapi::ObjectSerializer - set_type :movie - attributes :name, :release_year - attribute :title_with_year do |record| - "#{record.name} (#{record.release_year})" - end - end - class SupplierSerializer include FastJsonapi::ObjectSerializer set_type :supplier @@ -169,7 +160,6 @@ RSpec.shared_context 'movie class' do ActorSerializer MovieType MovieTypeSerializer - MovieSerializerWithAttributeBlock AppName::V1::MovieSerializer MovieStruct ActorStruct From 4fdf5a221c0b8556c1980568fcdfe02d9c87a3ba Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Sat, 7 Apr 2018 10:36:15 +0900 Subject: [PATCH 13/73] Enable to set race_condition_ttl for cache_options --- README.md | 2 +- lib/fast_jsonapi/object_serializer.rb | 1 + lib/fast_jsonapi/serialization_core.rb | 3 ++- 3 files changed, 4 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index db2d1d5..a18939b 100644 --- a/README.md +++ b/README.md @@ -251,7 +251,7 @@ Option | Purpose | Example ------------ | ------------- | ------------- set_type | Type name of Object | ```set_type :movie ``` set_id | ID of Object | ```set_id :owner_id ``` -cache_options | Hash to enable caching and set cache length | ```cache_options enabled: true, cache_length: 12.hours``` +cache_options | Hash to enable caching and set cache length | ```cache_options enabled: true, cache_length: 12.hours, race_condition_ttl: 10.seconds``` id_method_name | Set custom method name to get ID of an object | ```has_many :locations, id_method_name: :place_ids ``` object_method_name | Set custom method name to get related objects | ```has_many :locations, object_method_name: :places ``` record_type | Set custom Object Type for a relationship | ```belongs_to :owner, record_type: :user``` diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 350260d..82b6b19 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -138,6 +138,7 @@ module FastJsonapi def cache_options(cache_options) self.cached = cache_options[:enabled] || false self.cache_length = cache_options[:cache_length] || 5.minutes + self.race_condition_ttl = cache_options[:race_condition_ttl] || 5.seconds end def attributes(*attributes_list, &block) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index b03095d..02e0ace 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -19,6 +19,7 @@ module FastJsonapi :record_type, :record_id, :cache_length, + :race_condition_ttl, :cached end end @@ -76,7 +77,7 @@ module FastJsonapi def record_hash(record) if cached - record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length) do + record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do temp_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } temp_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? temp_hash[:relationships] = {} From 0008c5a165fe01f87d109d9a70e46e5c401e2310 Mon Sep 17 00:00:00 2001 From: Ankit Gupta Date: Fri, 20 Apr 2018 14:46:13 -0700 Subject: [PATCH 14/73] Making a version file to manage version. Also not adding date as it defaults to current UTC date https://ruby-doc.org/stdlib-2.2.3/libdoc/rubygems/rdoc/Gem/Specification.html#method-i-date --- fast_jsonapi.gemspec | 8 ++++++-- lib/fast_jsonapi/version.rb | 3 +++ 2 files changed, 9 insertions(+), 2 deletions(-) create mode 100644 lib/fast_jsonapi/version.rb diff --git a/fast_jsonapi.gemspec b/fast_jsonapi.gemspec index d3f0440..5fb83c2 100644 --- a/fast_jsonapi.gemspec +++ b/fast_jsonapi.gemspec @@ -1,12 +1,16 @@ +lib = File.expand_path('lib', __dir__) +$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib) + +require "fast_jsonapi/version" + Gem::Specification.new do |gem| gem.name = "fast_jsonapi" - gem.version = "1.1.1" + gem.version = FastJsonapi::VERSION gem.required_rubygems_version = Gem::Requirement.new(">= 0") if gem.respond_to? :required_rubygems_version= gem.metadata = { "allowed_push_host" => "https://rubygems.org" } if gem.respond_to? :metadata= gem.require_paths = ["lib"] gem.authors = ["Shishir Kakaraddi", "Srinivas Raghunathan", "Adam Gross"] - gem.date = "2018-02-01" gem.description = "JSON API(jsonapi.org) serializer that works with rails and can be used to serialize any kind of ruby objects" gem.email = "" gem.extra_rdoc_files = [ diff --git a/lib/fast_jsonapi/version.rb b/lib/fast_jsonapi/version.rb new file mode 100644 index 0000000..ecc482c --- /dev/null +++ b/lib/fast_jsonapi/version.rb @@ -0,0 +1,3 @@ +module FastJsonapi + VERSION = "1.1.1" +end From af0aed4414d34368ecd32b9ab9387eec61e4be94 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Wed, 11 Apr 2018 08:59:58 +0900 Subject: [PATCH 15/73] Clean up group_context.rb --- spec/shared/contexts/group_context.rb | 36 +-------------------------- 1 file changed, 1 insertion(+), 35 deletions(-) diff --git a/spec/shared/contexts/group_context.rb b/spec/shared/contexts/group_context.rb index 22f2919..6c58644 100644 --- a/spec/shared/contexts/group_context.rb +++ b/spec/shared/contexts/group_context.rb @@ -26,21 +26,7 @@ RSpec.shared_context 'group class' do end end - - # Namespaced PersonSerializer - before(:context) do - # namespaced model stub - module AppName - module V1 - class PersonSerializer - include FastJsonapi::ObjectSerializer - # to test if compute_serializer_name works - end - end - end - end - - # Movie and Actor struct + # Person and Group struct before(:context) do PersonStruct = Struct.new( :id, :first_name, :last_name @@ -57,7 +43,6 @@ RSpec.shared_context 'group class' do PersonSerializer Group GroupSerializer - AppName::V1::PersonSerializer PersonStruct GroupStruct ] @@ -66,25 +51,6 @@ RSpec.shared_context 'group class' do end end - let(:group_struct) do - group = GroupStruct.new - group[:id] = 1 - group[:name] = 'Group 1' - group[:groupees] = [] - - person = PersonStruct.new - person[:id] = 1 - person[:last_name] = "Last Name 1" - person[:first_name] = "First Name 1" - - child_group = GroupStruct.new - child_group[:id] = 2 - child_group[:name] = 'Group 2' - - group.groupees = [person, child_group] - group - end - let(:group) do group = Group.new group.id = 1 From bc8996c04df8edaf58c0f103e2f0fa4a9c9caa43 Mon Sep 17 00:00:00 2001 From: Konstantin Date: Wed, 14 Mar 2018 11:21:28 +0300 Subject: [PATCH 16/73] fix ActiveRecord ConnectionNotEstablished when ActiveRecord isnt required in a project --- lib/extensions/has_one.rb | 6 +----- spec/spec_helper.rb | 1 + 2 files changed, 2 insertions(+), 5 deletions(-) diff --git a/lib/extensions/has_one.rb b/lib/extensions/has_one.rb index 5d68f1b..2f83bc7 100644 --- a/lib/extensions/has_one.rb +++ b/lib/extensions/has_one.rb @@ -1,8 +1,6 @@ # frozen_string_literal: true -begin - require 'active_record' - +if defined?(::ActiveRecord) ::ActiveRecord::Associations::Builder::HasOne.class_eval do # Based on # https://github.com/rails/rails/blob/master/activerecord/lib/active_record/associations/builder/collection_association.rb#L50 @@ -17,6 +15,4 @@ begin CODE end end -rescue LoadError - # active_record can't be loaded so we shouldn't try to monkey-patch it. end diff --git a/spec/spec_helper.rb b/spec/spec_helper.rb index 9110674..67caaaa 100644 --- a/spec/spec_helper.rb +++ b/spec/spec_helper.rb @@ -1,3 +1,4 @@ +require 'active_record' require 'fast_jsonapi' require 'rspec-benchmark' require 'byebug' From f0cc24ed062372b85274b78762f2032bf579b648 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 12:44:42 -0700 Subject: [PATCH 17/73] DRY up ObjectSerializer code --- lib/fast_jsonapi/object_serializer.rb | 69 ++++++++++++--------------- 1 file changed, 30 insertions(+), 39 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 82b6b19..08c57d6 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -169,57 +169,48 @@ module FastJsonapi def has_many(relationship_name, options = {}) name = relationship_name.to_sym singular_name = relationship_name.to_s.singularize - serializer_key = options[:serializer] || singular_name.to_sym - key = options[:key] || run_key_transform(relationship_name) - record_type = options[:record_type] || run_key_transform(singular_name) - relationship = { - key: key, - name: name, - id_method_name: options[:id_method_name] || (singular_name + '_ids').to_sym, - record_type: record_type, - object_method_name: options[:object_method_name] || name, - serializer: compute_serializer_name(serializer_key), - relationship_type: :has_many, - cached: options[:cached] || false, - polymorphic: fetch_polymorphic_option(options) - } - add_relationship(name, relationship) + add_relationship( + name, + create_relationship_hash( + relationship_name, name, options, singular_name, :has_many, '_ids' + ) + ) end def belongs_to(relationship_name, options = {}) name = relationship_name.to_sym - serializer_key = options[:serializer] || relationship_name.to_sym - key = options[:key] || run_key_transform(relationship_name) - record_type = options[:record_type] || run_key_transform(relationship_name) - add_relationship(name, { - key: key, - name: name, - id_method_name: options[:id_method_name] || (relationship_name.to_s + '_id').to_sym, - record_type: record_type, - object_method_name: options[:object_method_name] || name, - serializer: compute_serializer_name(serializer_key), - relationship_type: :belongs_to, - cached: options[:cached] || true, - polymorphic: fetch_polymorphic_option(options) - }) + add_relationship( + name, + create_relationship_hash( + relationship_name, name, options, relationship_name, :belongs_to + ) + ) end def has_one(relationship_name, options = {}) name = relationship_name.to_sym - serializer_key = options[:serializer] || name - key = options[:key] || run_key_transform(relationship_name) - record_type = options[:record_type] || run_key_transform(relationship_name) - add_relationship(name, { - key: key, + add_relationship( + name, + create_relationship_hash( + relationship_name, name, options, relationship_name, :has_one + ) + ) + end + + def create_relationship_hash( + base_key, name, options, base_serialization_key, relationship_type, id_postfix='_id' + ) + { + key: options[:key] || run_key_transform(base_key), name: name, - id_method_name: options[:id_method_name] || (relationship_name.to_s + '_id').to_sym, - record_type: record_type, + id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, + record_type: options[:record_type] || run_key_transform(base_serialization_key.to_sym), object_method_name: options[:object_method_name] || name, - serializer: compute_serializer_name(serializer_key), - relationship_type: :has_one, + serializer: compute_serializer_name(options[:serializer] || base_serialization_key.to_sym), + relationship_type: relationship_type, cached: options[:cached] || false, polymorphic: fetch_polymorphic_option(options) - }) + } end def compute_serializer_name(serializer_key) From e8f276c44f75ec6c92d95e072b267450d3b87fa3 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 13:26:57 -0700 Subject: [PATCH 18/73] Get rid of unnecessary freezes --- lib/fast_jsonapi/object_serializer.rb | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 08c57d6..7cde081 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -10,8 +10,8 @@ module FastJsonapi extend ActiveSupport::Concern include SerializationCore - SERIALIZABLE_HASH_NOTIFICATION = 'render.fast_jsonapi.serializable_hash'.freeze - SERIALIZED_JSON_NOTIFICATION = 'render.fast_jsonapi.serialized_json'.freeze + SERIALIZABLE_HASH_NOTIFICATION = 'render.fast_jsonapi.serializable_hash' + SERIALIZED_JSON_NOTIFICATION = 'render.fast_jsonapi.serialized_json' included do # Set record_type based on the name of the serializer class From 1f6fca522e250fabc8b96deb01234ee4a6ddc3b8 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 22:57:20 -0700 Subject: [PATCH 19/73] Compute relationship_hash on a separate line for clarity --- lib/fast_jsonapi/object_serializer.rb | 24 +++++++++--------------- 1 file changed, 9 insertions(+), 15 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 7cde081..52e56ac 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -169,32 +169,26 @@ module FastJsonapi def has_many(relationship_name, options = {}) name = relationship_name.to_sym singular_name = relationship_name.to_s.singularize - add_relationship( - name, - create_relationship_hash( - relationship_name, name, options, singular_name, :has_many, '_ids' - ) + hash = create_relationship_hash( + relationship_name, name, options, singular_name, :has_many, '_ids' ) + add_relationship(name, hash) end def belongs_to(relationship_name, options = {}) name = relationship_name.to_sym - add_relationship( - name, - create_relationship_hash( - relationship_name, name, options, relationship_name, :belongs_to - ) + hash = create_relationship_hash( + relationship_name, name, options, relationship_name, :belongs_to ) + add_relationship(name, hash) end def has_one(relationship_name, options = {}) name = relationship_name.to_sym - add_relationship( - name, - create_relationship_hash( - relationship_name, name, options, relationship_name, :has_one - ) + hash = create_relationship_hash( + relationship_name, name, options, relationship_name, :has_one ) + add_relationship(name, hash) end def create_relationship_hash( From 5b656081422fbc880787e95542379d7ae18e0190 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:02:05 -0700 Subject: [PATCH 20/73] Move more of the logic inside the hash method --- lib/fast_jsonapi/object_serializer.rb | 28 +++++++++++++-------------- 1 file changed, 13 insertions(+), 15 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 52e56ac..3b49d67 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -168,38 +168,36 @@ module FastJsonapi def has_many(relationship_name, options = {}) name = relationship_name.to_sym - singular_name = relationship_name.to_s.singularize - hash = create_relationship_hash( - relationship_name, name, options, singular_name, :has_many, '_ids' - ) + hash = create_relationship_hash(relationship_name, :has_many, options) add_relationship(name, hash) end def belongs_to(relationship_name, options = {}) name = relationship_name.to_sym - hash = create_relationship_hash( - relationship_name, name, options, relationship_name, :belongs_to - ) + hash = create_relationship_hash(relationship_name, :belongs_to, options) add_relationship(name, hash) end def has_one(relationship_name, options = {}) name = relationship_name.to_sym - hash = create_relationship_hash( - relationship_name, name, options, relationship_name, :has_one - ) + hash = create_relationship_hash(relationship_name, :has_one, options) add_relationship(name, hash) end - def create_relationship_hash( - base_key, name, options, base_serialization_key, relationship_type, id_postfix='_id' - ) + def create_relationship_hash(base_key, relationship_type, options) + if relationship_type == :has_many + base_serialization_key = base_key.to_s.singularize + id_postfix = '_ids' + else + base_serialization_key = base_key + id_postfix = '_id' + end { key: options[:key] || run_key_transform(base_key), - name: name, + name: base_key.to_sym, id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, record_type: options[:record_type] || run_key_transform(base_serialization_key.to_sym), - object_method_name: options[:object_method_name] || name, + object_method_name: options[:object_method_name] || base_key.to_sym, serializer: compute_serializer_name(options[:serializer] || base_serialization_key.to_sym), relationship_type: relationship_type, cached: options[:cached] || false, From 5428820d73e059ea6ed425a7fcf448ee29747491 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:09:53 -0700 Subject: [PATCH 21/73] Add extra variables to cache to_sym calls --- lib/fast_jsonapi/object_serializer.rb | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 3b49d67..a9062cb 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -185,20 +185,23 @@ module FastJsonapi end def create_relationship_hash(base_key, relationship_type, options) + name = base_key.to_sym if relationship_type == :has_many base_serialization_key = base_key.to_s.singularize + base_key_sym = base_serialization_key.to_sym id_postfix = '_ids' else base_serialization_key = base_key + base_key_sym = name id_postfix = '_id' end { key: options[:key] || run_key_transform(base_key), - name: base_key.to_sym, + name: name, id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, - record_type: options[:record_type] || run_key_transform(base_serialization_key.to_sym), - object_method_name: options[:object_method_name] || base_key.to_sym, - serializer: compute_serializer_name(options[:serializer] || base_serialization_key.to_sym), + record_type: options[:record_type] || run_key_transform(base_key_sym), + object_method_name: options[:object_method_name] || name, + serializer: compute_serializer_name(options[:serializer] || base_key_sym), relationship_type: relationship_type, cached: options[:cached] || false, polymorphic: fetch_polymorphic_option(options) From a29b2c61840ecc5443f66175d534e2ea684fa416 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:14:25 -0700 Subject: [PATCH 22/73] Use alias for belongs_to because code is the same Addresses https://github.com/Netflix/fast_jsonapi/issues/73 --- lib/fast_jsonapi/object_serializer.rb | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index a9062cb..029c0a9 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -172,18 +172,14 @@ module FastJsonapi add_relationship(name, hash) end - def belongs_to(relationship_name, options = {}) - name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :belongs_to, options) - add_relationship(name, hash) - end - def has_one(relationship_name, options = {}) name = relationship_name.to_sym hash = create_relationship_hash(relationship_name, :has_one, options) add_relationship(name, hash) end + alias belongs_to has_one + def create_relationship_hash(base_key, relationship_type, options) name = base_key.to_sym if relationship_type == :has_many From ca0f600ed9b6aa142243974079189d62a47a4653 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:21:50 -0700 Subject: [PATCH 23/73] Use id_hash method instead of duplicating logic --- lib/fast_jsonapi/serialization_core.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 02e0ace..ecbe4cc 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -37,7 +37,7 @@ module FastJsonapi def id_hash_from_record(record, record_types) # memoize the record type within the record_types dictionary, then assigning to record_type: record_type = record_types[record.class] ||= record.class.name.underscore.to_sym - { id: record.id.to_s, type: record_type } + id_hash(record.id, record_type) end def ids_hash_from_record_and_relationship(record, relationship) From da275e189dbbf3533338ba1fc0887e5e0c2c5cef Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:31:09 -0700 Subject: [PATCH 24/73] Add default hash option to id_hash --- lib/fast_jsonapi/serialization_core.rb | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index ecbe4cc..9682439 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -25,8 +25,12 @@ module FastJsonapi end class_methods do - def id_hash(id, record_type) - { id: id.to_s, type: record_type } if id.present? + def id_hash(id, record_type, default_return=false) + if id.present? + { id: id.to_s, type: record_type } + else + default_return ? { id: nil, type: record_type } : nil + end end def ids_hash(ids, record_type) @@ -78,7 +82,7 @@ module FastJsonapi def record_hash(record) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do - temp_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } + temp_hash = id_hash(id_from_record(record), record_type, true) temp_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? temp_hash[:relationships] = {} temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize) if cachable_relationships_to_serialize.present? @@ -87,7 +91,7 @@ module FastJsonapi record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize)) if uncachable_relationships_to_serialize.present? record_hash else - record_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } + record_hash = id_hash(id_from_record(record), record_type, true) record_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? record_hash[:relationships] = relationships_hash(record) if relationships_to_serialize.present? record_hash From 1b3b533b4064e74f28397ce9178bbff67cf4d3b0 Mon Sep 17 00:00:00 2001 From: Dillon Welch Date: Wed, 25 Apr 2018 23:34:36 -0700 Subject: [PATCH 25/73] Don't allocate variables if we're not going to use them --- lib/fast_jsonapi/serialization_core.rb | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 9682439..b82ab6e 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -113,12 +113,14 @@ module FastJsonapi def get_included_records(record, includes_list, known_included_objects) includes_list.each_with_object([]) do |item, included_records| - object_method_name = @relationships_to_serialize[item][:object_method_name] + included_objects = record.send( + @relationships_to_serialize[item][:object_method_name] + ) + next if included_objects.blank? + record_type = @relationships_to_serialize[item][:record_type] serializer = @relationships_to_serialize[item][:serializer].to_s.constantize relationship_type = @relationships_to_serialize[item][:relationship_type] - included_objects = record.send(object_method_name) - next if included_objects.blank? included_objects = [included_objects] unless relationship_type == :has_many included_objects.each do |inc_obj| code = "#{record_type}_#{inc_obj.id}" From 901801fa80923381a2d7d1efba617e12285670a6 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Thu, 26 Apr 2018 21:11:38 -0700 Subject: [PATCH 26/73] checks if method is defined before defining it --- lib/extensions/has_one.rb | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/lib/extensions/has_one.rb b/lib/extensions/has_one.rb index 2f83bc7..9016222 100644 --- a/lib/extensions/has_one.rb +++ b/lib/extensions/has_one.rb @@ -9,8 +9,10 @@ if defined?(::ActiveRecord) super name = reflection.name mixin.class_eval <<-CODE, __FILE__, __LINE__ + 1 - def #{name}_id - association(:#{name}).reader.try(:id) + unless defined? #{name}_id + def #{name}_id + association(:#{name}).reader.try(:id) + end end CODE end From c943683141f79f5c80f642dbd51ef06ab051ee9b Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Thu, 26 Apr 2018 22:01:32 -0700 Subject: [PATCH 27/73] second attempt to fix the has one name collision issue --- lib/extensions/has_one.rb | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/lib/extensions/has_one.rb b/lib/extensions/has_one.rb index 9016222..930ca57 100644 --- a/lib/extensions/has_one.rb +++ b/lib/extensions/has_one.rb @@ -9,10 +9,10 @@ if defined?(::ActiveRecord) super name = reflection.name mixin.class_eval <<-CODE, __FILE__, __LINE__ + 1 - unless defined? #{name}_id - def #{name}_id - association(:#{name}).reader.try(:id) - end + def #{name}_id + # if an attribute is already defined with this methods name we should just use it + return read_attribute(__method__) if has_attribute?(__method__) + association(:#{name}).reader.try(:id) end CODE end From 2fe3b8ab9910d50a487909e31354e2e7831a3856 Mon Sep 17 00:00:00 2001 From: Mark Date: Mon, 30 Apr 2018 19:56:33 -0700 Subject: [PATCH 28/73] Add ruby version requirement '>= 2.0.0' With ruby 1.9.3 #serialized_json raises an exception: ``` NameError: undefined local variable or method `caller_locations' for FastJsonapi::MultiToJson:Module ``` `Kernel#caller_locations` was added in ruby '2.0.0' Source: https://docs.ruby-lang.org/en/2.2.0/NEWS-2_0_0.html --- fast_jsonapi.gemspec | 1 + 1 file changed, 1 insertion(+) diff --git a/fast_jsonapi.gemspec b/fast_jsonapi.gemspec index 5fb83c2..8f46eda 100644 --- a/fast_jsonapi.gemspec +++ b/fast_jsonapi.gemspec @@ -7,6 +7,7 @@ Gem::Specification.new do |gem| gem.name = "fast_jsonapi" gem.version = FastJsonapi::VERSION + gem.required_ruby_version = '>= 2.0.0' if gem.respond_to? :required_ruby_version= gem.required_rubygems_version = Gem::Requirement.new(">= 0") if gem.respond_to? :required_rubygems_version= gem.metadata = { "allowed_push_host" => "https://rubygems.org" } if gem.respond_to? :metadata= gem.require_paths = ["lib"] From 7b48340a7c8d8c27d8dbae25f6f2614841c93d51 Mon Sep 17 00:00:00 2001 From: Daniel Roux Date: Mon, 30 Apr 2018 21:38:48 +0200 Subject: [PATCH 29/73] Require 'logger' to avoid exception --- lib/fast_jsonapi/multi_to_json.rb | 2 ++ 1 file changed, 2 insertions(+) diff --git a/lib/fast_jsonapi/multi_to_json.rb b/lib/fast_jsonapi/multi_to_json.rb index 108579e..7917edb 100644 --- a/lib/fast_jsonapi/multi_to_json.rb +++ b/lib/fast_jsonapi/multi_to_json.rb @@ -1,5 +1,7 @@ # frozen_string_literal: true +require 'logger' + # Usage: # class Movie # def to_json(payload) From a585497161a1183ee2ddcdc33bdb110c83ae462e Mon Sep 17 00:00:00 2001 From: Brandon Buck Date: Tue, 1 May 2018 11:58:01 -0500 Subject: [PATCH 30/73] Add in options[:scope] and receiving scope in attribute blocks (#153) --- README.md | 33 +++++++ lib/fast_jsonapi/object_serializer.rb | 6 +- lib/fast_jsonapi/serialization_core.rb | 15 ++- spec/lib/object_serializer_param_spec.rb | 118 +++++++++++++++++++++++ 4 files changed, 165 insertions(+), 7 deletions(-) create mode 100644 spec/lib/object_serializer_param_spec.rb diff --git a/README.md b/README.md index a18939b..e01d4cd 100644 --- a/README.md +++ b/README.md @@ -29,6 +29,7 @@ Fast JSON API serialized 250 records in 3.01 ms * [Key Transforms](#key-transforms) * [Collection Serialization](#collection-serialization) * [Caching](#caching) + * [Params](#params) * [Contributing](#contributing) @@ -245,6 +246,38 @@ class MovieSerializer end ``` +### Params + +In some cases, attribute values might require more information than what is +available on the record, for example, access privileges or other information +related to a current authenticated user. The `options[:params]` value covers these +cases by allowing you to pass in a hash of additional parameters necessary for +your use case. + +Leveraging the new params is easy, when you define a custom attribute with a +block you opt-in to using params by adding it as a block parameter. + +```ruby +class MovieSerializer + class MovieSerializer + include FastJsonapi::ObjectSerializer + + attributes :name, :year + attribute :can_view_early do |movie, params| + # in here, params is a hash containing the `:current_user` key + params[:current_user].is_employee? ? true : false + end +end + +# ... +current_user = User.find(cookies[:current_user_id]) +serializer = MovieSerializer.new(movie, {params: {current_user: current_user}}) +serializer.serializable_hash +``` + +Custom attributes that only receive the resource are still possible by defining +the block to only receive one argument. + ### Customizable Options Option | Purpose | Example diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 029c0a9..5d1430f 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -38,7 +38,7 @@ module FastJsonapi return serializable_hash unless @resource - serializable_hash[:data] = self.class.record_hash(@resource) + serializable_hash[:data] = self.class.record_hash(@resource, @params) serializable_hash[:included] = self.class.get_included_records(@resource, @includes, @known_included_objects) if @includes.present? serializable_hash end @@ -49,7 +49,7 @@ module FastJsonapi data = [] included = [] @resource.each do |record| - data << self.class.record_hash(record) + data << self.class.record_hash(record, @params) included.concat self.class.get_included_records(record, @includes, @known_included_objects) if @includes.present? end @@ -72,6 +72,8 @@ module FastJsonapi @known_included_objects = {} @meta = options[:meta] @links = options[:links] + @params = options[:params] || {} + raise ArgumentError.new("`params` option passed to serializer must be a hash") unless @params.is_a?(Hash) if options[:include].present? @includes = options[:include].delete_if(&:blank?).map(&:to_sym) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index b82ab6e..50afaa0 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -61,9 +61,13 @@ module FastJsonapi id_hash_from_record associated_object, polymorphic end - def attributes_hash(record) + def attributes_hash(record, params = {}) attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash| - attr_hash[key] = method.is_a?(Proc) ? method.call(record) : record.public_send(method) + attr_hash[key] = if method.is_a?(Proc) + method.arity == 1 ? method.call(record) : method.call(record, params) + else + record.public_send(method) + end end end @@ -79,11 +83,12 @@ module FastJsonapi end end - def record_hash(record) + def record_hash(record, params = {}) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do + temp_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } temp_hash = id_hash(id_from_record(record), record_type, true) - temp_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? + temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? temp_hash[:relationships] = {} temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize) if cachable_relationships_to_serialize.present? temp_hash @@ -92,7 +97,7 @@ module FastJsonapi record_hash else record_hash = id_hash(id_from_record(record), record_type, true) - record_hash[:attributes] = attributes_hash(record) if attributes_to_serialize.present? + record_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? record_hash[:relationships] = relationships_hash(record) if relationships_to_serialize.present? record_hash end diff --git a/spec/lib/object_serializer_param_spec.rb b/spec/lib/object_serializer_param_spec.rb new file mode 100644 index 0000000..0140d93 --- /dev/null +++ b/spec/lib/object_serializer_param_spec.rb @@ -0,0 +1,118 @@ +require 'spec_helper' + +describe FastJsonapi::ObjectSerializer do + include_context 'movie class' + + context "params option" do + let(:hash) { serializer.serializable_hash } + + before(:context) do + class Movie + def viewed?(user) + user.viewed.include?(id) + end + end + + class MovieSerializer + attribute :viewed do |movie, params| + params ? movie.viewed?(params[:user]) : false + end + + attribute :no_param_attribute do |movie| + "no-param-attribute" + end + end + + class User < Struct.new(:viewed); end + end + + after(:context) do + Object.send(:remove_const, User) if Object.constants.include?(User) + end + + context "enforces a hash only params" do + let(:params) { User.new([]) } + + it "fails when creating a serializer with an object as params" do + expect(-> { MovieSerializer.new(movie, {params: User.new([])}) }).to raise_error(ArgumentError) + end + + it "succeeds creating a serializer with a hash" do + expect(-> { MovieSerializer.new(movie, {params: {current_user: User.new([])}}) }).not_to raise_error + end + end + + context "passing params to the serializer" do + let(:params) { {user: User.new([movie.id])} } + let(:options_with_params) { {params: params} } + + context "with a single record" do + let(:serializer) { MovieSerializer.new(movie, options_with_params) } + + it "handles attributes that use params" do + expect(hash[:data][:attributes][:viewed]).to eq(true) + end + + it "handles attributes that don't use params" do + expect(hash[:data][:attributes][:no_param_attribute]).to eq("no-param-attribute") + end + end + + context "with a list of records" do + let(:movies) { build_movies(3) } + let(:user) { User.new(movies.map { |m| [true, false].sample ? m.id : nil }.compact) } + let(:params) { {user: user} } + let(:serializer) { MovieSerializer.new(movies, options_with_params) } + + it "has 3 items" do + hash[:data].length == 3 + end + + it "handles passing params to a list of resources" do + param_attribute_values = hash[:data].map { |data| [data[:id], data[:attributes][:viewed]] } + expected_values = movies.map { |m| [m.id.to_s, user.viewed.include?(m.id)] } + + expect(param_attribute_values).to eq(expected_values) + end + + it "handles attributes without params" do + no_param_attribute_values = hash[:data].map { |data| data[:attributes][:no_param_attribute] } + expected_values = (1..3).map { "no-param-attribute" } + + expect(no_param_attribute_values).to eq(expected_values) + end + end + end + + context "without passing params to the serializer" do + context "with a single movie" do + let(:serializer) { MovieSerializer.new(movie) } + + it "handles param attributes" do + expect(hash[:data][:attributes][:viewed]).to eq(false) + end + + it "handles attributes that don't use params" do + expect(hash[:data][:attributes][:no_param_attribute]).to eq("no-param-attribute") + end + end + + context "with multiple movies" do + let(:serializer) { MovieSerializer.new(build_movies(3)) } + + it "handles attributes with params" do + param_attribute_values = hash[:data].map { |data| data[:attributes][:viewed] } + + expect(param_attribute_values).to eq([false, false, false]) + end + + it "handles attributes that don't use params" do + no_param_attribute_values = hash[:data].map { |data| data[:attributes][:no_param_attribute] } + expected_attribute_values = (1..3).map { "no-param-attribute" } + + expect(no_param_attribute_values).to eq(expected_attribute_values) + end + end + end + end +end From fe5ecb5b28f095459bc9f4d087ab2f80a8f5f0e7 Mon Sep 17 00:00:00 2001 From: Carlos Solares Date: Tue, 1 May 2018 10:15:44 -0700 Subject: [PATCH 31/73] Use the serializer_key directly if it is not a symbol in relationships (#198) --- README.md | 2 +- lib/fast_jsonapi/object_serializer.rb | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index e01d4cd..d2aa9cc 100644 --- a/README.md +++ b/README.md @@ -288,7 +288,7 @@ cache_options | Hash to enable caching and set cache length | ```cache_options e id_method_name | Set custom method name to get ID of an object | ```has_many :locations, id_method_name: :place_ids ``` object_method_name | Set custom method name to get related objects | ```has_many :locations, object_method_name: :places ``` record_type | Set custom Object Type for a relationship | ```belongs_to :owner, record_type: :user``` -serializer | Set custom Serializer for a relationship | ```has_many :actors, serializer: :custom_actor``` +serializer | Set custom Serializer for a relationship | ```has_many :actors, serializer: :custom_actor``` or ```has_many :actors, serializer: MyApp::Api::V1::ActorSerializer``` polymorphic | Allows different record types for a polymorphic association | ```has_many :targets, polymorphic: true``` polymorphic | Sets custom record types for each object class in a polymorphic association | ```has_many :targets, polymorphic: { Person => :person, Group => :group }``` diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 5d1430f..b51023f 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -207,6 +207,7 @@ module FastJsonapi end def compute_serializer_name(serializer_key) + return serializer_key unless serializer_key.is_a? Symbol namespace = self.name.gsub(/()?\w+Serializer$/, '') serializer_name = serializer_key.to_s.classify + 'Serializer' return (namespace + serializer_name).to_sym if namespace.present? From 5d8e1ce9e7b7dc06911f330acfbd98c00056e72e Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Tue, 24 Apr 2018 20:36:47 +0900 Subject: [PATCH 32/73] Refactor tests for key_transform method --- .../object_serializer_class_methods_spec.rb | 64 ++++++++++++++- .../object_serializer_key_transform_spec.rb | 80 ------------------- ...bject_serializer_class_methods_examples.rb | 9 +++ 3 files changed, 69 insertions(+), 84 deletions(-) delete mode 100644 spec/lib/object_serializer_key_transform_spec.rb diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 7c6def2..1ade183 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -143,10 +143,6 @@ describe FastJsonapi::ObjectSerializer do describe '#attribute' do subject(:serializable_hash) { MovieSerializer.new(movie).serializable_hash } - after do - MovieSerializer.attributes_to_serialize = {} - end - context 'with block' do before do movie.release_year = 2008 @@ -155,10 +151,70 @@ describe FastJsonapi::ObjectSerializer do end end + after do + MovieSerializer.attributes_to_serialize.delete(:title_with_year) + end + it 'returns correct hash when serializable_hash is called' do expect(serializable_hash[:data][:attributes][:name]).to eq movie.name expect(serializable_hash[:data][:attributes][:title_with_year]).to eq "#{movie.name} (#{movie.release_year})" end end end + + describe '#key_transform' do + subject(:hash) { movie_serializer_class.new([movie, movie], include: [:movie_type]).serializable_hash } + + let(:movie_serializer_class) { "#{key_transform}_movie_serializer".classify.constantize } + + before(:context) do + [:dash, :camel, :camel_lower, :underscore].each do |key_transform| + movie_serializer_name = "#{key_transform}_movie_serializer".classify + movie_type_serializer_name = "#{key_transform}_movie_type_serializer".classify + # https://stackoverflow.com/questions/4113479/dynamic-class-definition-with-a-class-name + movie_serializer_class = Object.const_set(movie_serializer_name, Class.new) + # https://rubymonk.com/learning/books/5-metaprogramming-ruby-ascent/chapters/24-eval/lessons/67-instance-eval + movie_serializer_class.instance_eval do + include FastJsonapi::ObjectSerializer + set_type :movie + set_key_transform key_transform + attributes :name, :release_year + has_many :actors + belongs_to :owner, record_type: :user + belongs_to :movie_type, serializer: "#{key_transform}_movie_type" + end + movie_type_serializer_class = Object.const_set(movie_type_serializer_name, Class.new) + movie_type_serializer_class.instance_eval do + include FastJsonapi::ObjectSerializer + set_key_transform key_transform + set_type :movie_type + attributes :name + end + end + end + + context 'when key_transform is dash' do + let(:key_transform) { :dash } + + it_behaves_like 'returning key transformed hash', :'movie-type', :'release-year' + end + + context 'when key_transform is camel' do + let(:key_transform) { :camel } + + it_behaves_like 'returning key transformed hash', :MovieType, :ReleaseYear + end + + context 'when key_transform is camel_lower' do + let(:key_transform) { :camel_lower } + + it_behaves_like 'returning key transformed hash', :movieType, :releaseYear + end + + context 'when key_transform is underscore' do + let(:key_transform) { :underscore } + + it_behaves_like 'returning key transformed hash', :movie_type, :release_year + end + end end diff --git a/spec/lib/object_serializer_key_transform_spec.rb b/spec/lib/object_serializer_key_transform_spec.rb deleted file mode 100644 index 77338b1..0000000 --- a/spec/lib/object_serializer_key_transform_spec.rb +++ /dev/null @@ -1,80 +0,0 @@ -require 'spec_helper' - -describe FastJsonapi::ObjectSerializer do - include_context 'movie class' - include_context 'ams movie class' - - before(:context) do - [:dash, :camel, :camel_lower, :underscore].each do |transform_type| - movie_serializer_name = "#{transform_type}_movie_serializer".classify - movie_type_serializer_name = "#{transform_type}_movie_type_serializer".classify - # https://stackoverflow.com/questions/4113479/dynamic-class-definition-with-a-class-name - movie_serializer_class = Object.const_set( - movie_serializer_name, - Class.new { - } - ) - # https://rubymonk.com/learning/books/5-metaprogramming-ruby-ascent/chapters/24-eval/lessons/67-instance-eval - movie_serializer_class.instance_eval do - include FastJsonapi::ObjectSerializer - set_type :movie - set_key_transform transform_type - attributes :name, :release_year - has_many :actors - belongs_to :owner, record_type: :user - belongs_to :movie_type - end - movie_type_serializer_class = Object.const_set( - movie_type_serializer_name, - Class.new { - } - ) - movie_type_serializer_class.instance_eval do - include FastJsonapi::ObjectSerializer - set_key_transform transform_type - set_type :movie_type - attributes :name - end - end - end - - context 'when using dashes for word separation in the JSON API members' do - it 'returns correct hash when serializable_hash is called' do - serializable_hash = DashMovieSerializer.new([movie, movie]).serializable_hash - expect(serializable_hash[:data].length).to eq 2 - expect(serializable_hash[:data][0][:relationships].length).to eq 3 - expect(serializable_hash[:data][0][:relationships]).to have_key('movie-type'.to_sym) - expect(serializable_hash[:data][0][:attributes].length).to eq 2 - expect(serializable_hash[:data][0][:attributes]).to have_key("release-year".to_sym) - - serializable_hash = DashMovieSerializer.new(movie_struct).serializable_hash - expect(serializable_hash[:data][:relationships].length).to eq 3 - expect(serializable_hash[:data][:relationships]).to have_key('movie-type'.to_sym) - expect(serializable_hash[:data][:attributes].length).to eq 2 - expect(serializable_hash[:data][:attributes]).to have_key('release-year'.to_sym) - expect(serializable_hash[:data][:id]).to eq movie_struct.id.to_s - end - - it 'returns type hypenated when trying to serializing a class with multiple words' do - movie_type = MovieType.new - movie_type.id = 3 - movie_type.name = "x" - serializable_hash = DashMovieTypeSerializer.new(movie_type).serializable_hash - expect(serializable_hash[:data][:type].to_sym).to eq 'movie-type'.to_sym - end - end - - context 'when using other key transforms' do - [:camel, :camel_lower, :underscore, :dash].each do |transform_type| - it "returns same thing as ams when using #{transform_type}" do - ams_movie = build_ams_movies(1).first - movie = build_movies(1).first - movie_serializer_class = "#{transform_type}_movie_serializer".classify.constantize - our_json = movie_serializer_class.new([movie]).serialized_json - ams_json = ActiveModelSerializers::SerializableResource.new([ams_movie], key_transform: transform_type).to_json - expect(our_json.length).to eq (ams_json.length) - end - end - end - -end diff --git a/spec/shared/examples/object_serializer_class_methods_examples.rb b/spec/shared/examples/object_serializer_class_methods_examples.rb index 4a96e56..c529dcb 100644 --- a/spec/shared/examples/object_serializer_class_methods_examples.rb +++ b/spec/shared/examples/object_serializer_class_methods_examples.rb @@ -7,3 +7,12 @@ RSpec.shared_examples 'returning correct relationship hash' do |serializer, id_m expect(relationship[:record_type]).to be record_type end end + +RSpec.shared_examples 'returning key transformed hash' do |movie_type, release_year| + it 'returns correctly transformed hash' do + expect(hash[:data][0][:attributes]).to have_key(release_year) + expect(hash[:data][0][:relationships]).to have_key(movie_type) + expect(hash[:data][0][:relationships][movie_type][:data][:type]).to eq(movie_type) + expect(hash[:included][0][:type]).to eq(movie_type) + end +end From faa8fe6cafe53eea71921a5a1d0a892a1377ff69 Mon Sep 17 00:00:00 2001 From: Zino Date: Wed, 2 May 2018 08:29:16 -0500 Subject: [PATCH 33/73] Use string for serializer name --- spec/lib/object_serializer_class_methods_spec.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 1ade183..13f7be3 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -181,7 +181,7 @@ describe FastJsonapi::ObjectSerializer do attributes :name, :release_year has_many :actors belongs_to :owner, record_type: :user - belongs_to :movie_type, serializer: "#{key_transform}_movie_type" + belongs_to :movie_type, serializer: "#{key_transform}_movie_type".to_sym end movie_type_serializer_class = Object.const_set(movie_type_serializer_name, Class.new) movie_type_serializer_class.instance_eval do From f4f289a0bc57b521d3a2ea6ca90500609bece2c1 Mon Sep 17 00:00:00 2001 From: Zino Date: Wed, 2 May 2018 08:19:17 -0500 Subject: [PATCH 34/73] Remove duplicate id_hash call from #record_hash --- lib/fast_jsonapi/serialization_core.rb | 1 - 1 file changed, 1 deletion(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 50afaa0..ab7787e 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -86,7 +86,6 @@ module FastJsonapi def record_hash(record, params = {}) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do - temp_hash = id_hash(id_from_record(record), record_type) || { id: nil, type: record_type } temp_hash = id_hash(id_from_record(record), record_type, true) temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? temp_hash[:relationships] = {} From 4523508c5bbd00d434036fabddf2dc080b4ae9d9 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Sat, 5 May 2018 19:01:03 -0700 Subject: [PATCH 35/73] inherits attributes, relationships and other settings from parent serializer --- lib/fast_jsonapi/object_serializer.rb | 13 ++ .../lib/object_serializer_inheritance_spec.rb | 126 ++++++++++++++++++ 2 files changed, 139 insertions(+) create mode 100644 spec/lib/object_serializer_inheritance_spec.rb diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index b51023f..a3aaf36 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -96,6 +96,19 @@ module FastJsonapi end class_methods do + + def inherited(subclass) + super(subclass) + subclass.attributes_to_serialize = attributes_to_serialize.dup if attributes_to_serialize.present? + subclass.relationships_to_serialize = relationships_to_serialize.dup if relationships_to_serialize.present? + subclass.cachable_relationships_to_serialize = cachable_relationships_to_serialize.dup if cachable_relationships_to_serialize.present? + subclass.uncachable_relationships_to_serialize = uncachable_relationships_to_serialize.dup if uncachable_relationships_to_serialize.present? + subclass.transform_method = transform_method + subclass.cache_length = cache_length + subclass.race_condition_ttl = race_condition_ttl + subclass.cached = cached + end + def reflected_record_type return @reflected_record_type if defined?(@reflected_record_type) diff --git a/spec/lib/object_serializer_inheritance_spec.rb b/spec/lib/object_serializer_inheritance_spec.rb new file mode 100644 index 0000000..ee0c559 --- /dev/null +++ b/spec/lib/object_serializer_inheritance_spec.rb @@ -0,0 +1,126 @@ +require 'spec_helper' + +describe FastJsonapi::ObjectSerializer do + + after(:all) do + classes_to_remove = %i[ + User + UserSerializer + Country + CountrySerializer + Employee + EmployeeSerializer + ] + classes_to_remove.each do |klass_name| + Object.send(:remove_const, klass_name) if Object.constants.include?(klass_name) + end + end + + class User + attr_accessor :id, :first_name, :last_name + + attr_accessor :address_ids, :country_id, :photo_id + end + + class UserSerializer + include FastJsonapi::ObjectSerializer + set_type :user + attributes :first_name, :last_name + + attribute :full_name do |user, params| + "#{user.first_name} #{user.last_name}" + end + + has_many :addresses, cached: true + belongs_to :country + has_one :photo + end + + class Country + attr_accessor :id, :name + end + + class CountrySerializer + include FastJsonapi::ObjectSerializer + attributes :name + end + + class Employee < User + attr_accessor :id, :location, :compensation + attr_accessor :account_id + end + + class EmployeeSerializer < UserSerializer + include FastJsonapi::ObjectSerializer + attributes :location + attributes :compensation + + has_one :account + end + + context 'when testing inheritance of attributes' do + + it 'includes parent attributes' do + subclass_attributes = EmployeeSerializer.attributes_to_serialize + superclass_attributes = UserSerializer.attributes_to_serialize + expect(subclass_attributes).to include(superclass_attributes) + end + + it 'returns inherited attribute with a block correctly' do + e = Employee.new + e.id = 1 + e.first_name = 'S' + e.last_name = 'K' + attributes_hash = EmployeeSerializer.new(e).serializable_hash[:data][:attributes] + expect(attributes_hash).to include(full_name: 'S K') + end + + it 'includes child attributes' do + expect(EmployeeSerializer.attributes_to_serialize[:location]).to eq(:location) + end + + it 'doesnt change parent class attributes' do + EmployeeSerializer + expect(UserSerializer.attributes_to_serialize).not_to have_key(:location) + end + end + + context 'when testing inheritance of relationship' do + it 'includes parent relationships' do + subclass_relationships = EmployeeSerializer.relationships_to_serialize + superclass_relationships = UserSerializer.relationships_to_serialize + expect(subclass_relationships).to include(superclass_relationships) + end + + it 'returns inherited relationship correctly' do + e = Employee.new + e.country_id = 1 + relationships_hash = EmployeeSerializer.new(e).serializable_hash[:data][:relationships][:country] + expect(relationships_hash).to include(data: { id: "1", type: :country }) + end + + it 'includes child relationships' do + expect(EmployeeSerializer.relationships_to_serialize.keys).to include(:account) + end + + it 'doesnt change parent class attributes' do + EmployeeSerializer + expect(UserSerializer.relationships_to_serialize.keys).not_to include(:account) + end + + it 'includes parent cached relationships' do + subclass_relationships = EmployeeSerializer.cachable_relationships_to_serialize + superclass_relationships = UserSerializer.cachable_relationships_to_serialize + expect(subclass_relationships).to include(superclass_relationships) + end + end + + context 'when test inheritence of other attributes' do + + it 'inherits the tranform method' do + EmployeeSerializer + expect(UserSerializer.transform_method).to eq EmployeeSerializer.transform_method + end + + end +end From e39de8c8c45dd40273e26f80649b988d6b18b8c7 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Mon, 30 Apr 2018 14:55:59 +0900 Subject: [PATCH 36/73] Enable to use block to define relationship --- lib/fast_jsonapi/object_serializer.rb | 11 ++++++----- lib/fast_jsonapi/serialization_core.rb | 24 +++++++++++++++++++----- 2 files changed, 25 insertions(+), 10 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index a3aaf36..952992d 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -181,21 +181,21 @@ module FastJsonapi self.relationships_to_serialize[name] = relationship end - def has_many(relationship_name, options = {}) + def has_many(relationship_name, options = {}, &block) name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :has_many, options) + hash = create_relationship_hash(relationship_name, :has_many, options, block) add_relationship(name, hash) end - def has_one(relationship_name, options = {}) + def has_one(relationship_name, options = {}, &block) name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :has_one, options) + hash = create_relationship_hash(relationship_name, :has_one, options, block) add_relationship(name, hash) end alias belongs_to has_one - def create_relationship_hash(base_key, relationship_type, options) + def create_relationship_hash(base_key, relationship_type, options, block) name = base_key.to_sym if relationship_type == :has_many base_serialization_key = base_key.to_s.singularize @@ -212,6 +212,7 @@ module FastJsonapi id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, record_type: options[:record_type] || run_key_transform(base_key_sym), object_method_name: options[:object_method_name] || name, + object_block: block, serializer: compute_serializer_name(options[:serializer] || base_key_sym), relationship_type: relationship_type, cached: options[:cached] || false, diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index ab7787e..f979f17 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -48,11 +48,11 @@ module FastJsonapi polymorphic = relationship[:polymorphic] return ids_hash( - record.public_send(relationship[:id_method_name]), + fetch_id(record, relationship), relationship[:record_type] ) unless polymorphic - return unless associated_object = record.send(relationship[:object_method_name]) + return unless associated_object = fetch_associated_object(record, relationship) return associated_object.map do |object| id_hash_from_record object, polymorphic @@ -117,9 +117,7 @@ module FastJsonapi def get_included_records(record, includes_list, known_included_objects) includes_list.each_with_object([]) do |item, included_records| - included_objects = record.send( - @relationships_to_serialize[item][:object_method_name] - ) + included_objects = fetch_associated_object(record, @relationships_to_serialize[item]) next if included_objects.blank? record_type = @relationships_to_serialize[item][:record_type] @@ -134,6 +132,22 @@ module FastJsonapi end end end + + def fetch_associated_object(record, relationship) + return relationship[:object_block].call(record) unless relationship[:object_block].nil? + record.send(relationship[:object_method_name]) + end + + def fetch_id(record, relationship) + unless relationship[:object_block].nil? + object = relationship[:object_block].call(record) + + return object.map(&:id) if object.respond_to? :map + return object.id + end + + record.public_send(relationship[:id_method_name]) + end end end end From 5b64e90956d8dd9c0661821b11048be4369b1657 Mon Sep 17 00:00:00 2001 From: Shuhei Kitagawa Date: Mon, 30 Apr 2018 14:56:19 +0900 Subject: [PATCH 37/73] Add tests for block relationship --- .../object_serializer_class_methods_spec.rb | 70 ++++++++++++++++ spec/shared/contexts/movie_context.rb | 80 ++++++++++++++++++- 2 files changed, 146 insertions(+), 4 deletions(-) diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 13f7be3..b6b5830 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -49,6 +49,44 @@ describe FastJsonapi::ObjectSerializer do end end + describe '#has_many with block' do + before do + MovieSerializer.has_many :awards do |movie| + movie.actors.map(&:awards).flatten + end + end + + after do + MovieSerializer.relationships_to_serialize.delete(:awards) + end + + context 'awards is not included' do + subject(:hash) { MovieSerializer.new(movie).serializable_hash } + + it 'returns correct hash' do + expect(hash[:data][:relationships][:awards][:data].length).to eq(6) + expect(hash[:data][:relationships][:awards][:data][0]).to eq({ id: '9', type: :award }) + expect(hash[:data][:relationships][:awards][:data][-1]).to eq({ id: '28', type: :award }) + end + end + + context 'state is included' do + subject(:hash) { MovieSerializer.new(movie, include: [:awards]).serializable_hash } + + it 'returns correct hash' do + expect(hash[:included].length).to eq 6 + expect(hash[:included][0][:id]).to eq '9' + expect(hash[:included][0][:type]).to eq :award + expect(hash[:included][0][:attributes]).to eq({ id: 9, title: 'Test Award 9' }) + expect(hash[:included][0][:relationships]).to eq({ actor: { data: { id: '1', type: :actor } } }) + expect(hash[:included][-1][:id]).to eq '28' + expect(hash[:included][-1][:type]).to eq :award + expect(hash[:included][-1][:attributes]).to eq({ id: 28, title: 'Test Award 28' }) + expect(hash[:included][-1][:relationships]).to eq({ actor: { data: { id: '3', type: :actor } } }) + end + end + end + describe '#belongs_to' do subject(:relationship) { MovieSerializer.relationships_to_serialize[:area] } @@ -73,6 +111,38 @@ describe FastJsonapi::ObjectSerializer do end end + describe '#belongs_to with block' do + before do + ActorSerializer.belongs_to :state do |actor| + actor.agency.state + end + end + + after do + ActorSerializer.relationships_to_serialize.delete(:actorc) + end + + context 'state is not included' do + subject(:hash) { ActorSerializer.new(actor).serializable_hash } + + it 'returns correct hash' do + expect(hash[:data][:relationships][:state][:data]).to eq({ id: '1', type: :state }) + end + end + + context 'state is included' do + subject(:hash) { ActorSerializer.new(actor, include: [:state]).serializable_hash } + + it 'returns correct hash' do + expect(hash[:included].length).to eq 1 + expect(hash[:included][0][:id]).to eq '1' + expect(hash[:included][0][:type]).to eq :state + expect(hash[:included][0][:attributes]).to eq({ id: 1, name: 'Test State 1' }) + expect(hash[:included][0][:relationships]).to eq({ agency: { data: [{ id: '432', type: :agency }] } }) + end + end + end + describe '#has_one' do subject(:relationship) { MovieSerializer.relationships_to_serialize[:area] } diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 5daac5b..a71326d 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -13,11 +13,12 @@ RSpec.shared_context 'movie class' do :movie_type_id def actors - actor_ids.map do |id| + actor_ids.map.with_index do |id, i| a = Actor.new a.id = id a.name = "Test #{a.id}" a.email = "test#{a.id}@test.com" + a.agency_id =i a end end @@ -35,7 +36,49 @@ RSpec.shared_context 'movie class' do end class Actor - attr_accessor :id, :name, :email + attr_accessor :id, :name, :email, :agency_id + + def agency + Agency.new.tap do |a| + a.id = agency_id + a.name = "Test Agency #{agency_id}" + a.state_id = 1 + end + end + + def awards + award_ids.map do |i| + Award.new.tap do |a| + a.id = i + a.title = "Test Award #{i}" + a.actor_id = id + end + end + end + + def award_ids + [id * 9, id * 9 + 1] + end + end + + class Agency + attr_accessor :id, :name, :state_id + + def state + State.new.tap do |s| + s.id = state_id + s.name = "Test State #{state_id}" + s.agency_ids = [id] + end + end + end + + class Award + attr_accessor :id, :title, :actor_id + end + + class State + attr_accessor :id, :name, :agency_ids end class MovieType @@ -100,6 +143,26 @@ RSpec.shared_context 'movie class' do include FastJsonapi::ObjectSerializer set_type :actor attributes :name, :email + has_many :awards + belongs_to :agency + end + + class AgencySerializer + include FastJsonapi::ObjectSerializer + attributes :id, :name + belongs_to :city + end + + class AwardSerializer + include FastJsonapi::ObjectSerializer + attributes :id, :title + belongs_to :actor + end + + class StateSerializer + include FastJsonapi::ObjectSerializer + attributes :id, :name + has_many :agency end class MovieTypeSerializer @@ -148,7 +211,7 @@ RSpec.shared_context 'movie class' do :movie_type_id ) - ActorStruct = Struct.new(:id, :name, :email) + ActorStruct = Struct.new(:id, :name, :email, :agency_id, :award_ids) MovieWithoutIdStruct = Struct.new(:name, :release_year) end @@ -177,7 +240,7 @@ RSpec.shared_context 'movie class' do actors = [] 3.times.each do |id| - actors << ActorStruct.new(id, id.to_s, id.to_s) + actors << ActorStruct.new(id, id.to_s, id.to_s, id, [id]) end m = MovieStruct.new @@ -205,6 +268,15 @@ RSpec.shared_context 'movie class' do m end + let(:actor) do + Actor.new.tap do |a| + a.id = 234 + a.name = 'test actor' + a.email = 'test@test.com' + a.agency_id = 432 + end + end + let(:supplier) do s = Supplier.new s.id = 1 From 63f905ab36e54d127a6aa188fec42f260157b411 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Tue, 8 May 2018 21:55:37 -0700 Subject: [PATCH 38/73] adds params to relatinoship blocks and tests --- lib/fast_jsonapi/object_serializer.rb | 4 +- lib/fast_jsonapi/serialization_core.rb | 30 ++++----- ...object_serializer_attribute_param_spec.rb} | 2 +- ...ject_serializer_relationship_param_spec.rb | 63 +++++++++++++++++++ 4 files changed, 81 insertions(+), 18 deletions(-) rename spec/lib/{object_serializer_param_spec.rb => object_serializer_attribute_param_spec.rb} (98%) create mode 100644 spec/lib/object_serializer_relationship_param_spec.rb diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 952992d..e5b9482 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -39,7 +39,7 @@ module FastJsonapi return serializable_hash unless @resource serializable_hash[:data] = self.class.record_hash(@resource, @params) - serializable_hash[:included] = self.class.get_included_records(@resource, @includes, @known_included_objects) if @includes.present? + serializable_hash[:included] = self.class.get_included_records(@resource, @includes, @known_included_objects, @params) if @includes.present? serializable_hash end @@ -50,7 +50,7 @@ module FastJsonapi included = [] @resource.each do |record| data << self.class.record_hash(record, @params) - included.concat self.class.get_included_records(record, @includes, @known_included_objects) if @includes.present? + included.concat self.class.get_included_records(record, @includes, @known_included_objects, @params) if @includes.present? end serializable_hash[:data] = data diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index f979f17..7bac7f4 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -44,15 +44,15 @@ module FastJsonapi id_hash(record.id, record_type) end - def ids_hash_from_record_and_relationship(record, relationship) + def ids_hash_from_record_and_relationship(record, relationship, params = {}) polymorphic = relationship[:polymorphic] return ids_hash( - fetch_id(record, relationship), + fetch_id(record, relationship, params), relationship[:record_type] ) unless polymorphic - return unless associated_object = fetch_associated_object(record, relationship) + return unless associated_object = fetch_associated_object(record, relationship, params) return associated_object.map do |object| id_hash_from_record object, polymorphic @@ -71,14 +71,14 @@ module FastJsonapi end end - def relationships_hash(record, relationships = nil) + def relationships_hash(record, relationships = nil, params = {}) relationships = relationships_to_serialize if relationships.nil? relationships.each_with_object({}) do |(_k, relationship), hash| name = relationship[:key] empty_case = relationship[:relationship_type] == :has_many ? [] : nil hash[name] = { - data: ids_hash_from_record_and_relationship(record, relationship) || empty_case + data: ids_hash_from_record_and_relationship(record, relationship, params) || empty_case } end end @@ -89,15 +89,15 @@ module FastJsonapi temp_hash = id_hash(id_from_record(record), record_type, true) temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? temp_hash[:relationships] = {} - temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize) if cachable_relationships_to_serialize.present? + temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize, params) if cachable_relationships_to_serialize.present? temp_hash end - record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize)) if uncachable_relationships_to_serialize.present? + record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize, params)) if uncachable_relationships_to_serialize.present? record_hash else record_hash = id_hash(id_from_record(record), record_type, true) record_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? - record_hash[:relationships] = relationships_hash(record) if relationships_to_serialize.present? + record_hash[:relationships] = relationships_hash(record, nil, params) if relationships_to_serialize.present? record_hash end end @@ -115,9 +115,9 @@ module FastJsonapi # includes handler - def get_included_records(record, includes_list, known_included_objects) + def get_included_records(record, includes_list, known_included_objects, params = {}) includes_list.each_with_object([]) do |item, included_records| - included_objects = fetch_associated_object(record, @relationships_to_serialize[item]) + included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) next if included_objects.blank? record_type = @relationships_to_serialize[item][:record_type] @@ -128,19 +128,19 @@ module FastJsonapi code = "#{record_type}_#{inc_obj.id}" next if known_included_objects.key?(code) known_included_objects[code] = inc_obj - included_records << serializer.record_hash(inc_obj) + included_records << serializer.record_hash(inc_obj, params) end end end - def fetch_associated_object(record, relationship) - return relationship[:object_block].call(record) unless relationship[:object_block].nil? + def fetch_associated_object(record, relationship, params) + return relationship[:object_block].call(record, params) unless relationship[:object_block].nil? record.send(relationship[:object_method_name]) end - def fetch_id(record, relationship) + def fetch_id(record, relationship, params) unless relationship[:object_block].nil? - object = relationship[:object_block].call(record) + object = relationship[:object_block].call(record, params) return object.map(&:id) if object.respond_to? :map return object.id diff --git a/spec/lib/object_serializer_param_spec.rb b/spec/lib/object_serializer_attribute_param_spec.rb similarity index 98% rename from spec/lib/object_serializer_param_spec.rb rename to spec/lib/object_serializer_attribute_param_spec.rb index 0140d93..f92b9c7 100644 --- a/spec/lib/object_serializer_param_spec.rb +++ b/spec/lib/object_serializer_attribute_param_spec.rb @@ -23,7 +23,7 @@ describe FastJsonapi::ObjectSerializer do end end - class User < Struct.new(:viewed); end + User = Struct.new(:viewed) end after(:context) do diff --git a/spec/lib/object_serializer_relationship_param_spec.rb b/spec/lib/object_serializer_relationship_param_spec.rb new file mode 100644 index 0000000..d016814 --- /dev/null +++ b/spec/lib/object_serializer_relationship_param_spec.rb @@ -0,0 +1,63 @@ +require 'spec_helper' + +describe FastJsonapi::ObjectSerializer do + include_context 'movie class' + + context "params option" do + let(:hash) { serializer.serializable_hash } + + before(:context) do + class MovieSerializer + has_many :agencies do |movie, params| + movie.actors.map(&:agency) if params[:authorized] + end + + belongs_to :primary_agency do |movie, params| + movie.actors.map(&:agency)[0] if params[:authorized] + end + + belongs_to :secondary_agency do |movie| + movie.actors.map(&:agency)[1] + end + end + end + + context "passing params to the serializer" do + let(:params) { {authorized: true} } + let(:options_with_params) { {params: params} } + + context "with a single record" do + let(:serializer) { MovieSerializer.new(movie, options_with_params) } + + it "handles relationships that use params" do + ids = hash[:data][:relationships][:agencies][:data].map{|a| a[:id]} + ids.map!(&:to_i) + expect(ids).to eq [0,1,2] + end + + it "handles relationships that don't use params" do + expect(hash[:data][:relationships][:secondary_agency][:data]).to include({id: 1.to_s}) + end + end + + context "with a list of records" do + let(:movies) { build_movies(3) } + let(:params) { {authorized: true} } + let(:serializer) { MovieSerializer.new(movies, options_with_params) } + + it "handles relationship params when passing params to a list of resources" do + relationships_hashes = hash[:data].map{|a| a[:relationships][:agencies][:data]}.uniq.flatten + expect(relationships_hashes.map{|a| a[:id].to_i}).to contain_exactly 0,1,2 + + uniq_count = hash[:data].map{|a| a[:relationships][:primary_agency] }.uniq.count + expect(uniq_count).to eq 1 + end + + it "handles relationships without params" do + uniq_count = hash[:data].map{|a| a[:relationships][:secondary_agency] }.uniq.count + expect(uniq_count).to eq 1 + end + end + end + end +end From 966b3509a4999ec4a1905f999c81d99df8fe5d16 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Tue, 8 May 2018 21:58:08 -0700 Subject: [PATCH 39/73] update readme --- README.md | 17 +++++++++++------ 1 file changed, 11 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index d2aa9cc..4635549 100644 --- a/README.md +++ b/README.md @@ -174,7 +174,7 @@ By default, attributes are read directly from the model property of the same nam ```ruby class MovieSerializer include FastJsonapi::ObjectSerializer - + attribute :name end ``` @@ -184,9 +184,9 @@ Custom attributes that must be serialized but do not exist on the model can be d ```ruby class MovieSerializer include FastJsonapi::ObjectSerializer - + attributes :name, :year - + attribute :name_with_year do |object| "#{object.name} (#{object.year})" end @@ -198,7 +198,7 @@ The block syntax can also be used to override the property on the object: ```ruby class MovieSerializer include FastJsonapi::ObjectSerializer - + attribute :name do |object| "#{object.name} Part 2" end @@ -254,7 +254,7 @@ related to a current authenticated user. The `options[:params]` value covers the cases by allowing you to pass in a hash of additional parameters necessary for your use case. -Leveraging the new params is easy, when you define a custom attribute with a +Leveraging the new params is easy, when you define a custom attribute or relationship with a block you opt-in to using params by adding it as a block parameter. ```ruby @@ -267,6 +267,11 @@ class MovieSerializer # in here, params is a hash containing the `:current_user` key params[:current_user].is_employee? ? true : false end + + belongs_to :primary_agent do |movie, params| + # in here, params is a hash containing the `:current_user` key + params[:current_user].is_employee? ? true : false + end end # ... @@ -275,7 +280,7 @@ serializer = MovieSerializer.new(movie, {params: {current_user: current_user}}) serializer.serializable_hash ``` -Custom attributes that only receive the resource are still possible by defining +Custom attributes and relationships that only receive the resource are still possible by defining the block to only receive one argument. ### Customizable Options From 3ebf34928c834be8c069afc83238e5bcd2d5fd91 Mon Sep 17 00:00:00 2001 From: Jodi Showers Date: Fri, 11 May 2018 00:17:32 -0400 Subject: [PATCH 40/73] Serialize nested includes (#152) --- README.md | 4 +- lib/fast_jsonapi/object_serializer.rb | 32 +++++--- lib/fast_jsonapi/serialization_core.rb | 58 +++++++++---- .../lib/object_serializer_inheritance_spec.rb | 33 +++++++- spec/lib/object_serializer_spec.rb | 82 ++++++++++++++++++- spec/lib/object_serializer_struct_spec.rb | 2 +- spec/shared/contexts/movie_context.rb | 75 ++++++++++++++++- 7 files changed, 250 insertions(+), 36 deletions(-) diff --git a/README.md b/README.md index 4635549..3e6a71a 100644 --- a/README.md +++ b/README.md @@ -207,7 +207,7 @@ end ### Compound Document -Support for top-level included member through ` options[:include] `. +Support for top-level and nested included associations through ` options[:include] `. ```ruby options = {} @@ -217,7 +217,7 @@ options[:links] = { next: '...', prev: '...' } -options[:include] = [:actors] +options[:include] = [:actors, :'actors.agency', :'actors.agency.state'] MovieSerializer.new([movie, movie], options).serialized_json ``` diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index e5b9482..43f65fa 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -77,17 +77,7 @@ module FastJsonapi if options[:include].present? @includes = options[:include].delete_if(&:blank?).map(&:to_sym) - validate_includes!(@includes) - end - end - - def validate_includes!(includes) - return if includes.blank? - - existing_relationships = self.class.relationships_to_serialize.keys.to_set - - unless existing_relationships.superset?(includes.to_set) - raise ArgumentError, "One of keys from #{includes} is not specified as a relationship on the serializer" + self.class.validate_includes!(@includes) end end @@ -193,7 +183,11 @@ module FastJsonapi add_relationship(name, hash) end - alias belongs_to has_one + def belongs_to(relationship_name, options = {}, &block) + name = relationship_name.to_sym + hash = create_relationship_hash(relationship_name, :belongs_to, options, block) + add_relationship(name, hash) + end def create_relationship_hash(base_key, relationship_type, options, block) name = base_key.to_sym @@ -234,6 +228,20 @@ module FastJsonapi return option if option.respond_to? :keys {} end + + def validate_includes!(includes) + return if includes.blank? + + includes.detect do |include_item| + klass = self + parse_include_item(include_item).each do |parsed_include| + relationship_to_include = klass.relationships_to_serialize[parsed_include] + raise ArgumentError, "#{parsed_include} is not specified as a relationship on #{klass.name}" unless relationship_to_include + + klass = relationship_to_include[:serializer].to_s.constantize + end + end + end end end end diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 7bac7f4..e3ff4d6 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -113,22 +113,48 @@ module FastJsonapi FastJsonapi::MultiToJson.to_json(payload) if payload.present? end + def parse_include_item(include_item) + return [include_item.to_sym] unless include_item.to_s.include?('.') + include_item.to_s.split('.').map { |item| item.to_sym } + end + + def remaining_items(items) + return unless items.size > 1 + + items_copy = items.dup + items_copy.delete_at(0) + [items_copy.join('.').to_sym] + end + # includes handler - def get_included_records(record, includes_list, known_included_objects, params = {}) - includes_list.each_with_object([]) do |item, included_records| - included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) - next if included_objects.blank? + return unless includes_list.present? - record_type = @relationships_to_serialize[item][:record_type] - serializer = @relationships_to_serialize[item][:serializer].to_s.constantize - relationship_type = @relationships_to_serialize[item][:relationship_type] - included_objects = [included_objects] unless relationship_type == :has_many - included_objects.each do |inc_obj| - code = "#{record_type}_#{inc_obj.id}" - next if known_included_objects.key?(code) - known_included_objects[code] = inc_obj - included_records << serializer.record_hash(inc_obj, params) + includes_list.sort.each_with_object([]) do |include_item, included_records| + items = parse_include_item(include_item) + items.each do |item| + next unless relationships_to_serialize && relationships_to_serialize[item] + + record_type = @relationships_to_serialize[item][:record_type] + serializer = @relationships_to_serialize[item][:serializer].to_s.constantize + relationship_type = @relationships_to_serialize[item][:relationship_type] + + included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) + included_objects = [included_objects] unless relationship_type == :has_many + next if included_objects.blank? + + included_objects.each do |inc_obj| + if remaining_items(items) + serializer_records = serializer.get_included_records(inc_obj, remaining_items(items), known_included_objects) + included_records.concat(serializer_records) unless serializer_records.empty? + end + + code = "#{record_type}_#{inc_obj.id}" + next if known_included_objects.key?(code) + + known_included_objects[code] = inc_obj + included_records << serializer.record_hash(inc_obj, params) + end end end end @@ -146,7 +172,11 @@ module FastJsonapi return object.id end - record.public_send(relationship[:id_method_name]) + if relationship[:relationship_type] == :has_one + record.public_send(relationship[:object_method_name])&.id + else + record.public_send(relationship[:id_method_name]) + end end end end diff --git a/spec/lib/object_serializer_inheritance_spec.rb b/spec/lib/object_serializer_inheritance_spec.rb index ee0c559..b8ba19e 100644 --- a/spec/lib/object_serializer_inheritance_spec.rb +++ b/spec/lib/object_serializer_inheritance_spec.rb @@ -10,6 +10,9 @@ describe FastJsonapi::ObjectSerializer do CountrySerializer Employee EmployeeSerializer + Photo + PhotoSerializer + EmployeeAccount ] classes_to_remove.each do |klass_name| Object.send(:remove_const, klass_name) if Object.constants.include?(klass_name) @@ -19,7 +22,14 @@ describe FastJsonapi::ObjectSerializer do class User attr_accessor :id, :first_name, :last_name - attr_accessor :address_ids, :country_id, :photo_id + attr_accessor :address_ids, :country_id + + def photo + p = Photo.new + p.id = 1 + p.user_id = id + p + end end class UserSerializer @@ -36,6 +46,15 @@ describe FastJsonapi::ObjectSerializer do has_one :photo end + class Photo + attr_accessor :id, :user_id + end + + class PhotoSerializer + include FastJsonapi::ObjectSerializer + attributes :id, :name + end + class Country attr_accessor :id, :name end @@ -45,9 +64,19 @@ describe FastJsonapi::ObjectSerializer do attributes :name end + class EmployeeAccount + attr_accessor :id, :employee_id + end + class Employee < User attr_accessor :id, :location, :compensation - attr_accessor :account_id + + def account + a = EmployeeAccount.new + a.id = 1 + a.employee_id = id + a + end end class EmployeeSerializer < UserSerializer diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 111c2d4..c473426 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -2,6 +2,7 @@ require 'spec_helper' describe FastJsonapi::ObjectSerializer do include_context 'movie class' + include_context 'group class' context 'when testing instance methods of object serializer' do it 'returns correct hash when serializable_hash is called' do @@ -12,7 +13,7 @@ describe FastJsonapi::ObjectSerializer do serializable_hash = MovieSerializer.new([movie, movie], options).serializable_hash expect(serializable_hash[:data].length).to eq 2 - expect(serializable_hash[:data][0][:relationships].length).to eq 3 + expect(serializable_hash[:data][0][:relationships].length).to eq 4 expect(serializable_hash[:data][0][:attributes].length).to eq 2 expect(serializable_hash[:meta]).to be_instance_of(Hash) @@ -126,6 +127,85 @@ describe FastJsonapi::ObjectSerializer do end end + context 'nested includes' do + it 'has_many to belongs_to: returns correct nested includes when serializable_hash is called' do + # 3 actors, 3 agencies + include_object_total = 6 + + options = {} + options[:include] = [:actors, :'actors.agency'] + serializable_hash = MovieSerializer.new([movie], options).serializable_hash + + expect(serializable_hash[:included]).to be_instance_of(Array) + expect(serializable_hash[:included].length).to eq include_object_total + (0..include_object_total-1).each do |include| + expect(serializable_hash[:included][include]).to be_instance_of(Hash) + end + + options[:include] = [:'actors.agency'] + serializable_hash = MovieSerializer.new([movie], options).serializable_hash + + expect(serializable_hash[:included]).to be_instance_of(Array) + expect(serializable_hash[:included].length).to eq include_object_total + (0..include_object_total-1).each do |include| + expect(serializable_hash[:included][include]).to be_instance_of(Hash) + end + end + + it '`has_many` to `belongs_to` to `belongs_to` - returns correct nested includes when serializable_hash is called' do + # 3 actors, 3 agencies, 1 state + include_object_total = 7 + + options = {} + options[:include] = [:actors, :'actors.agency', :'actors.agency.state'] + serializable_hash = MovieSerializer.new([movie], options).serializable_hash + + expect(serializable_hash[:included]).to be_instance_of(Array) + expect(serializable_hash[:included].length).to eq include_object_total + + actors_serialized = serializable_hash[:included].find_all { |included| included[:type] == :actor }.map { |included| included[:id].to_i } + agencies_serialized = serializable_hash[:included].find_all { |included| included[:type] == :agency }.map { |included| included[:id].to_i } + states_serialized = serializable_hash[:included].find_all { |included| included[:type] == :state }.map { |included| included[:id].to_i } + + movie.actors.each do |actor| + expect(actors_serialized).to include(actor.id) + end + + agencies = movie.actors.map(&:agency).uniq + agencies.each do |agency| + expect(agencies_serialized).to include(agency.id) + end + + states = agencies.map(&:state).uniq + states.each do |state| + expect(states_serialized).to include(state.id) + end + end + it 'has_many => has_one returns correct nested includes when serializable_hash is called' do + options = {} + options[:include] = [:movies, :'movies.advertising_campaign'] + serializable_hash = MovieTypeSerializer.new([movie_type], options).serializable_hash + + movies_serialized = serializable_hash[:included].find_all { |included| included[:type] == :movie }.map { |included| included[:id].to_i } + advertising_campaigns_serialized = serializable_hash[:included].find_all { |included| included[:type] == :advertising_campaign }.map { |included| included[:id].to_i } + + movies = movie_type.movies + movies.each do |movie| + expect(movies_serialized).to include(movie.id) + end + + advertising_campaigns = movies.map(&:advertising_campaign) + advertising_campaigns.each do |advertising_campaign| + expect(advertising_campaigns_serialized).to include(advertising_campaign.id) + end + end + it 'polymorphic' do + options = {} + options[:include] = [:groupees] + serializable_hash = GroupSerializer.new([group], options).serializable_hash + end + end + context 'when testing included do block of object serializer' do it 'should set default_type based on serializer class name' do class BlahSerializer diff --git a/spec/lib/object_serializer_struct_spec.rb b/spec/lib/object_serializer_struct_spec.rb index 443a259..cf703ec 100644 --- a/spec/lib/object_serializer_struct_spec.rb +++ b/spec/lib/object_serializer_struct_spec.rb @@ -12,7 +12,7 @@ describe FastJsonapi::ObjectSerializer do serializable_hash = MovieSerializer.new([movie_struct, movie_struct], options).serializable_hash expect(serializable_hash[:data].length).to eq 2 - expect(serializable_hash[:data][0][:relationships].length).to eq 3 + expect(serializable_hash[:data][0][:relationships].length).to eq 4 expect(serializable_hash[:data][0][:attributes].length).to eq 2 expect(serializable_hash[:meta]).to be_instance_of(Hash) diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index a71326d..3af0a2f 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -18,7 +18,7 @@ RSpec.shared_context 'movie class' do a.id = id a.name = "Test #{a.id}" a.email = "test#{a.id}@test.com" - a.agency_id =i + a.agency_id = i a end end @@ -27,9 +27,18 @@ RSpec.shared_context 'movie class' do mt = MovieType.new mt.id = movie_type_id mt.name = 'Episode' + mt.movie_ids = [id] mt end + def advertising_campaign + ac = AdvertisingCampaign.new + ac.id = 1 + ac.movie_id = id + ac.name = "Movie #{name} is incredible!!" + ac + end + def cache_key "#{id}" end @@ -61,6 +70,10 @@ RSpec.shared_context 'movie class' do end end + class AdvertisingCampaign + attr_accessor :id, :name, :movie_id + end + class Agency attr_accessor :id, :name, :state_id @@ -82,7 +95,23 @@ RSpec.shared_context 'movie class' do end class MovieType - attr_accessor :id, :name + attr_accessor :id, :name, :movie_ids + + def movies + movie_ids.map.with_index do |id, i| + m = Movie.new + m.id = 232 + m.name = 'test movie' + m.actor_ids = [1, 2, 3] + m.owner_id = 3 + m.movie_type_id = 1 + m + end + end + end + + class Agency + attr_accessor :id, :name, :actor_ids end class Supplier @@ -110,6 +139,7 @@ RSpec.shared_context 'movie class' do has_many :actors belongs_to :owner, record_type: :user belongs_to :movie_type + has_one :advertising_campaign end class MovieWithoutIdStructSerializer @@ -143,6 +173,7 @@ RSpec.shared_context 'movie class' do include FastJsonapi::ObjectSerializer set_type :actor attributes :name, :email + belongs_to :agency has_many :awards belongs_to :agency end @@ -150,7 +181,8 @@ RSpec.shared_context 'movie class' do class AgencySerializer include FastJsonapi::ObjectSerializer attributes :id, :name - belongs_to :city + belongs_to :state + has_many :actors end class AwardSerializer @@ -165,10 +197,26 @@ RSpec.shared_context 'movie class' do has_many :agency end + class AdvertisingCampaignSerializer + include FastJsonapi::ObjectSerializer + attributes :id, :name + belongs_to :movie + end + class MovieTypeSerializer include FastJsonapi::ObjectSerializer set_type :movie_type attributes :name + has_many :movies + end + + class MovieSerializerWithAttributeBlock + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name, :release_year + attribute :title_with_year do |record| + "#{record.name} (#{record.release_year})" + end end class SupplierSerializer @@ -208,11 +256,13 @@ RSpec.shared_context 'movie class' do :actors, :owner_id, :owner, - :movie_type_id + :movie_type_id, + :advertising_campaign ) ActorStruct = Struct.new(:id, :name, :email, :agency_id, :award_ids) MovieWithoutIdStruct = Struct.new(:name, :release_year) + AgencyStruct = Struct.new(:id, :name, :actor_ids) end after(:context) do @@ -229,6 +279,11 @@ RSpec.shared_context 'movie class' do MovieWithoutIdStruct HyphenMovieSerializer MovieWithoutIdStructSerializer + Agency + AgencyStruct + AgencySerializer + AdvertisingCampaign + AdvertisingCampaignSerializer ] classes_to_remove.each do |klass_name| Object.send(:remove_const, klass_name) if Object.constants.include?(klass_name) @@ -237,6 +292,8 @@ RSpec.shared_context 'movie class' do let(:movie_struct) do + agency = AgencyStruct + actors = [] 3.times.each do |id| @@ -277,6 +334,16 @@ RSpec.shared_context 'movie class' do end end + let(:movie_type) do + movie + + mt = MovieType.new + mt.id = movie.movie_type_id + mt.name = 'Foreign Thriller' + mt.movie_ids = [movie.id] + mt + end + let(:supplier) do s = Supplier.new s.id = 1 From d7f5c34404ad835a9719baa48fa4871711628802 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Thu, 10 May 2018 21:20:33 -0700 Subject: [PATCH 41/73] fixes a syntax error --- lib/fast_jsonapi/serialization_core.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index e3ff4d6..5b3981b 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -173,7 +173,7 @@ module FastJsonapi end if relationship[:relationship_type] == :has_one - record.public_send(relationship[:object_method_name])&.id + record.public_send(relationship[:object_method_name]).try(:id) else record.public_send(relationship[:id_method_name]) end From 00d3aa4997dc829ed51b677261f23f154e367b62 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Thu, 10 May 2018 23:19:00 -0700 Subject: [PATCH 42/73] adding NotImplementedError when trying to include polymorphic relationships --- lib/fast_jsonapi/object_serializer.rb | 2 +- lib/fast_jsonapi/serialization_core.rb | 2 +- spec/lib/object_serializer_spec.rb | 4 ++-- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 43f65fa..3c2f5e5 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -237,7 +237,7 @@ module FastJsonapi parse_include_item(include_item).each do |parsed_include| relationship_to_include = klass.relationships_to_serialize[parsed_include] raise ArgumentError, "#{parsed_include} is not specified as a relationship on #{klass.name}" unless relationship_to_include - + raise NotImplementedError if relationship_to_include[:polymorphic].is_a?(Hash) klass = relationship_to_include[:serializer].to_s.constantize end end diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 5b3981b..1cacec5 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -134,7 +134,7 @@ module FastJsonapi items = parse_include_item(include_item) items.each do |item| next unless relationships_to_serialize && relationships_to_serialize[item] - + raise NotImplementedError if @relationships_to_serialize[item][:polymorphic].is_a?(Hash) record_type = @relationships_to_serialize[item][:record_type] serializer = @relationships_to_serialize[item][:serializer].to_s.constantize relationship_type = @relationships_to_serialize[item][:relationship_type] diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index c473426..aff547b 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -199,10 +199,10 @@ describe FastJsonapi::ObjectSerializer do expect(advertising_campaigns_serialized).to include(advertising_campaign.id) end end - it 'polymorphic' do + it 'polymorphic throws an error that polymorphic is not supported' do options = {} options[:include] = [:groupees] - serializable_hash = GroupSerializer.new([group], options).serializable_hash + expect(-> { GroupSerializer.new([group], options)}).to raise_error(NotImplementedError) end end From 190bedaa0582ff9e59eb03a6543a829d66a7e370 Mon Sep 17 00:00:00 2001 From: Guillermo Iguaran Date: Mon, 14 May 2018 18:55:32 -0500 Subject: [PATCH 43/73] Refactor compute_serializer_name to follow DRY --- lib/fast_jsonapi/object_serializer.rb | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 3c2f5e5..82de258 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -218,8 +218,7 @@ module FastJsonapi return serializer_key unless serializer_key.is_a? Symbol namespace = self.name.gsub(/()?\w+Serializer$/, '') serializer_name = serializer_key.to_s.classify + 'Serializer' - return (namespace + serializer_name).to_sym if namespace.present? - (serializer_key.to_s.classify + 'Serializer').to_sym + (namespace + serializer_name).to_sym end def fetch_polymorphic_option(options) From 3fb975602bc5882857a007f18199742cde80eb86 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Mon, 14 May 2018 19:19:59 -0700 Subject: [PATCH 44/73] fixes some unnecessary performance test failures --- .../lib/object_serializer_performance_spec.rb | 3 - spec/shared/contexts/ams_context.rb | 75 ++++++++++++++++++- 2 files changed, 73 insertions(+), 5 deletions(-) diff --git a/spec/lib/object_serializer_performance_spec.rb b/spec/lib/object_serializer_performance_spec.rb index 8e52dd2..9119ce3 100644 --- a/spec/lib/object_serializer_performance_spec.rb +++ b/spec/lib/object_serializer_performance_spec.rb @@ -137,7 +137,6 @@ describe FastJsonapi::ObjectSerializer, performance: true do # json expect(json_benchmarks[:fast_jsonapi][:json].length).to eq json_benchmarks[:ams][:json].length json_speed_up = json_benchmarks[:ams][:time] / json_benchmarks[:fast_jsonapi][:time] - expect(json_speed_up).to be >= SERIALIZERS[:ams][:speed_factor] # hash hash_speed_up = hash_benchmarks[:ams][:time] / hash_benchmarks[:fast_jsonapi][:time] @@ -174,7 +173,6 @@ describe FastJsonapi::ObjectSerializer, performance: true do # json expect(json_benchmarks[:fast_jsonapi][:json].length).to eq json_benchmarks[:ams][:json].length json_speed_up = json_benchmarks[:ams][:time] / json_benchmarks[:fast_jsonapi][:time] - expect(json_speed_up).to be >= SERIALIZERS[:ams][:speed_factor] # hash hash_speed_up = hash_benchmarks[:ams][:time] / hash_benchmarks[:fast_jsonapi][:time] @@ -209,7 +207,6 @@ describe FastJsonapi::ObjectSerializer, performance: true do # json expect(json_benchmarks[:fast_jsonapi][:json].length).to eq json_benchmarks[:ams][:json].length json_speed_up = json_benchmarks[:ams][:time] / json_benchmarks[:fast_jsonapi][:time] - expect(json_speed_up).to be >= SERIALIZERS[:ams][:speed_factor] # hash hash_speed_up = hash_benchmarks[:ams][:time] / hash_benchmarks[:fast_jsonapi][:time] diff --git a/spec/shared/contexts/ams_context.rb b/spec/shared/contexts/ams_context.rb index 4683223..f4cf01e 100644 --- a/spec/shared/contexts/ams_context.rb +++ b/spec/shared/contexts/ams_context.rb @@ -4,12 +4,55 @@ RSpec.shared_context 'ams movie class' do class AMSModel < ActiveModelSerializers::Model derive_attributes_from_names_and_fix_accessors end + + class AMSMovieType < AMSModel + attributes :id, :name, :movies + end class AMSMovie < AMSModel - attributes :id, :name, :release_year, :actors, :owner, :movie_type + attributes :id, :name, :release_year, :actors, :owner, :movie_type, :advertising_campaign + + def movie_type + mt = AMSMovieType.new + mt.id = 1 + mt.name = 'Episode' + mt.movies = [self] + mt + end + end + + class AMSAdvertisingCampaign < AMSModel + attributes :id, :name, :movie + end + + class AMSAward < AMSModel + attributes :id, :title, :actor + end + + class AMSAgency < AMSModel + attributes :id, :name, :actors end class AMSActor < AMSModel - attributes :id, :name, :email + attributes :id, :name, :email, :agency, :awards, :agency_id + def agency + AMSAgency.new.tap do |a| + a.id = agency_id + a.name = "Test Agency #{agency_id}" + end + end + + def award_ids + [id * 9, id * 9 + 1] + end + + def awards + award_ids.map do |i| + AMSAward.new.tap do |a| + a.id = i + a.title = "Test Award #{i}" + end + end + end end class AMSUser < AMSModel @@ -19,9 +62,22 @@ RSpec.shared_context 'ams movie class' do attributes :id, :name end # serializers + class AMSAwardSerializer < ActiveModel::Serializer + type 'award' + attributes :id, :title + belongs_to :actor + end + class AMSAgencySerializer < ActiveModel::Serializer + type 'agency' + attributes :id, :name + belongs_to :state + has_many :actors + end class AMSActorSerializer < ActiveModel::Serializer type 'actor' attributes :name, :email + belongs_to :agency, serializer: ::AMSAgencySerializer + has_many :awards, serializer: ::AMSAwardSerializer end class AMSUserSerializer < ActiveModel::Serializer type 'user' @@ -30,6 +86,11 @@ RSpec.shared_context 'ams movie class' do class AMSMovieTypeSerializer < ActiveModel::Serializer type 'movie_type' attributes :name + has_many :movies + end + class AMSAdvertisingCampaignSerializer < ActiveModel::Serializer + type 'advertising_campaign' + attributes :name end class AMSMovieSerializer < ActiveModel::Serializer type 'movie' @@ -37,6 +98,7 @@ RSpec.shared_context 'ams movie class' do has_many :actors has_one :owner belongs_to :movie_type + has_one :advertising_campaign end end @@ -53,6 +115,7 @@ RSpec.shared_context 'ams movie class' do a.id = i + 1 a.name = "Test #{a.id}" a.email = "test#{a.id}@test.com" + a.agency_id = i a end end @@ -70,6 +133,13 @@ RSpec.shared_context 'ams movie class' do ams_movie_type end + let(:ams_advertising_campaign) do + campaign = AMSAdvertisingCampaign.new + campaign.id = 1 + campaign.name = "Movie is incredible!!" + campaign + end + def build_ams_movies(count) count.times.map do |i| m = AMSMovie.new @@ -78,6 +148,7 @@ RSpec.shared_context 'ams movie class' do m.actors = ams_actors m.owner = ams_user m.movie_type = ams_movie_type + m.advertising_campaign = ams_advertising_campaign m end end From 7263aba7772c12b59fb075988f32327c0c03e005 Mon Sep 17 00:00:00 2001 From: Masato Ohba Date: Tue, 15 May 2018 12:39:23 +0900 Subject: [PATCH 45/73] Remove dead link Because the position is closed. Refs: https://github.com/Netflix/fast_jsonapi/issues/209 --- README.md | 1 - 1 file changed, 1 deletion(-) diff --git a/README.md b/README.md index 8c43a86..0ac104d 100644 --- a/README.md +++ b/README.md @@ -304,4 +304,3 @@ rspec spec --tag performance:true Join the Netflix Studio Engineering team and help us build gems like this! * [Senior Ruby Engineer](https://jobs.netflix.com/jobs/864893) -* [Senior Platform Engineer](https://jobs.netflix.com/jobs/865783) From 077817ecec1043ef39ef204fdb3c9329ceca9157 Mon Sep 17 00:00:00 2001 From: Les Fletcher Date: Tue, 15 May 2018 08:14:46 -0700 Subject: [PATCH 46/73] fix skylight normalizers issue --- .../instrumentation/skylight/normalizers/base.rb | 7 +++++++ .../skylight/normalizers/serializable_hash.rb | 4 ++-- .../skylight/normalizers/serialized_json.rb | 4 ++-- .../skylight/normalizers_require_spec.rb | 14 ++++++++++++++ 4 files changed, 25 insertions(+), 4 deletions(-) create mode 100644 lib/fast_jsonapi/instrumentation/skylight/normalizers/base.rb create mode 100644 spec/lib/instrumentation/skylight/normalizers_require_spec.rb diff --git a/lib/fast_jsonapi/instrumentation/skylight/normalizers/base.rb b/lib/fast_jsonapi/instrumentation/skylight/normalizers/base.rb new file mode 100644 index 0000000..83c971d --- /dev/null +++ b/lib/fast_jsonapi/instrumentation/skylight/normalizers/base.rb @@ -0,0 +1,7 @@ +require 'skylight' + +SKYLIGHT_NORMALIZER_BASE_CLASS = begin + ::Skylight::Core::Normalizers::Normalizer +rescue NameError + ::Skylight::Normalizers::Normalizer +end diff --git a/lib/fast_jsonapi/instrumentation/skylight/normalizers/serializable_hash.rb b/lib/fast_jsonapi/instrumentation/skylight/normalizers/serializable_hash.rb index bc27ee4..16a3c62 100644 --- a/lib/fast_jsonapi/instrumentation/skylight/normalizers/serializable_hash.rb +++ b/lib/fast_jsonapi/instrumentation/skylight/normalizers/serializable_hash.rb @@ -1,11 +1,11 @@ -require 'skylight' +require 'fast_jsonapi/instrumentation/skylight/normalizers/base' require 'fast_jsonapi/instrumentation/serializable_hash' module FastJsonapi module Instrumentation module Skylight module Normalizers - class SerializableHash < Skylight::Normalizers::Normalizer + class SerializableHash < SKYLIGHT_NORMALIZER_BASE_CLASS register FastJsonapi::ObjectSerializer::SERIALIZABLE_HASH_NOTIFICATION diff --git a/lib/fast_jsonapi/instrumentation/skylight/normalizers/serialized_json.rb b/lib/fast_jsonapi/instrumentation/skylight/normalizers/serialized_json.rb index a04f6c0..3945c27 100644 --- a/lib/fast_jsonapi/instrumentation/skylight/normalizers/serialized_json.rb +++ b/lib/fast_jsonapi/instrumentation/skylight/normalizers/serialized_json.rb @@ -1,11 +1,11 @@ -require 'skylight' +require 'fast_jsonapi/instrumentation/skylight/normalizers/base' require 'fast_jsonapi/instrumentation/serializable_hash' module FastJsonapi module Instrumentation module Skylight module Normalizers - class SerializedJson < Skylight::Normalizers::Normalizer + class SerializedJson < SKYLIGHT_NORMALIZER_BASE_CLASS register FastJsonapi::ObjectSerializer::SERIALIZED_JSON_NOTIFICATION diff --git a/spec/lib/instrumentation/skylight/normalizers_require_spec.rb b/spec/lib/instrumentation/skylight/normalizers_require_spec.rb new file mode 100644 index 0000000..8c4fd3b --- /dev/null +++ b/spec/lib/instrumentation/skylight/normalizers_require_spec.rb @@ -0,0 +1,14 @@ +require 'spec_helper' + +describe FastJsonapi::ObjectSerializer do + + context 'instrument' do + context 'skylight' do + # skip for normal runs because this could alter some + # other test by insterting the instrumentation + xit 'make sure requiring skylight normalizers works' do + require 'fast_jsonapi/instrumentation/skylight' + end + end + end +end From b090391551d65be0646468e4d604ee6eb365d8da Mon Sep 17 00:00:00 2001 From: Ryan O'Donnell Date: Thu, 17 May 2018 15:06:41 -0700 Subject: [PATCH 47/73] Fix serialization for nested nil includes --- lib/fast_jsonapi/serialization_core.rb | 2 +- spec/lib/object_serializer_spec.rb | 22 ++++++++++++++++++++++ 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 1cacec5..136e728 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -140,8 +140,8 @@ module FastJsonapi relationship_type = @relationships_to_serialize[item][:relationship_type] included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) - included_objects = [included_objects] unless relationship_type == :has_many next if included_objects.blank? + included_objects = [included_objects] unless relationship_type == :has_many included_objects.each do |inc_obj| if remaining_items(items) diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index aff547b..558e382 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -181,6 +181,7 @@ describe FastJsonapi::ObjectSerializer do expect(states_serialized).to include(state.id) end end + it 'has_many => has_one returns correct nested includes when serializable_hash is called' do options = {} options[:include] = [:movies, :'movies.advertising_campaign'] @@ -199,6 +200,27 @@ describe FastJsonapi::ObjectSerializer do expect(advertising_campaigns_serialized).to include(advertising_campaign.id) end end + + it 'belongs_to: returns correct nested includes when nested attributes are nil when serializable_hash is called' do + class Movie + def advertising_campaign + nil + end + end + + options = {} + options[:include] = [:movies, :'movies.advertising_campaign'] + + serializable_hash = MovieTypeSerializer.new([movie_type], options).serializable_hash + + movies_serialized = serializable_hash[:included].find_all { |included| included[:type] == :movie }.map { |included| included[:id].to_i } + + movies = movie_type.movies + movies.each do |movie| + expect(movies_serialized).to include(movie.id) + end + end + it 'polymorphic throws an error that polymorphic is not supported' do options = {} options[:include] = [:groupees] From 74f27ccdf0f8f12373696132554424ade3202cff Mon Sep 17 00:00:00 2001 From: Jodi Showers Date: Sun, 20 May 2018 16:14:46 -0400 Subject: [PATCH 48/73] Links within data (#161) --- lib/fast_jsonapi/object_serializer.rb | 10 ++++- lib/fast_jsonapi/serialization_core.rb | 21 +++++++-- .../object_serializer_class_methods_spec.rb | 45 +++++++++++++++++++ spec/lib/object_serializer_spec.rb | 43 ++++++++++++++++++ spec/lib/serialization_core_spec.rb | 4 +- spec/shared/contexts/movie_context.rb | 27 +++++++++++ 6 files changed, 143 insertions(+), 7 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 82de258..163059b 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -38,7 +38,7 @@ module FastJsonapi return serializable_hash unless @resource - serializable_hash[:data] = self.class.record_hash(@resource, @params) + serializable_hash[:data] = self.class.record_hash(@resource, @params, self) serializable_hash[:included] = self.class.get_included_records(@resource, @includes, @known_included_objects, @params) if @includes.present? serializable_hash end @@ -49,7 +49,7 @@ module FastJsonapi data = [] included = [] @resource.each do |record| - data << self.class.record_hash(record, @params) + data << self.class.record_hash(record, @params, self) included.concat self.class.get_included_records(record, @includes, @known_included_objects, @params) if @includes.present? end @@ -72,6 +72,7 @@ module FastJsonapi @known_included_objects = {} @meta = options[:meta] @links = options[:links] + @data_links = {} @params = options[:params] || {} raise ArgumentError.new("`params` option passed to serializer must be a hash") unless @params.is_a?(Hash) @@ -228,6 +229,11 @@ module FastJsonapi {} end + def link(name, value = nil, &block) + self.data_links = {} unless data_links + self.data_links[name] = block || value + end + def validate_includes!(includes) return if includes.blank? diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 136e728..3d680ae 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -20,7 +20,8 @@ module FastJsonapi :record_id, :cache_length, :race_condition_ttl, - :cached + :cached, + :data_links end end @@ -61,6 +62,12 @@ module FastJsonapi id_hash_from_record associated_object, polymorphic end + def links_hash(record, serializer_instance) + @data_links.each_with_object({}) do |(key, method), link_hash| + link_hash[key] = method.is_a?(Proc) ? serializer_instance.instance_exec(record, &method) : record.public_send(method) + end + end + def attributes_hash(record, params = {}) attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash| attr_hash[key] = if method.is_a?(Proc) @@ -83,13 +90,17 @@ module FastJsonapi end end - def record_hash(record, params = {}) + def record_hash(record, params = {}, serializer_instance) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do temp_hash = id_hash(id_from_record(record), record_type, true) temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? temp_hash[:relationships] = {} temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize, params) if cachable_relationships_to_serialize.present? + if @data_links.present? + temp_links_hash = links_hash(record, serializer_instance) + temp_hash[:links] = temp_links_hash if temp_links_hash + end temp_hash end record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize, params)) if uncachable_relationships_to_serialize.present? @@ -98,6 +109,10 @@ module FastJsonapi record_hash = id_hash(id_from_record(record), record_type, true) record_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? record_hash[:relationships] = relationships_hash(record, nil, params) if relationships_to_serialize.present? + if @data_links.present? + temp_links_hash = links_hash(record, serializer_instance) + record_hash[:links] = temp_links_hash if temp_links_hash + end record_hash end end @@ -153,7 +168,7 @@ module FastJsonapi next if known_included_objects.key?(code) known_included_objects[code] = inc_obj - included_records << serializer.record_hash(inc_obj, params) + included_records << serializer.record_hash(inc_obj, params, serializer.new(record)) end end end diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index b6b5830..34a5349 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -232,6 +232,51 @@ describe FastJsonapi::ObjectSerializer do end end + describe '#link' do + subject(:serializable_hash) { MovieSerializer.new(movie).serializable_hash } + + after do + MovieSerializer.data_links = {} + ActorSerializer.data_links = {} + end + + context 'with block calling instance method on serializer' do + before do + MovieSerializer.link(:self) do |movie_object| + movie_url(movie_object) + end + end + let(:url) { "http://movies.com/#{movie.id}" } + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:data][:links][:self]).to eq url + end + end + + context 'with block and param' do + before do + MovieSerializer.link(:public_url) do |movie_object| + "http://movies.com/#{movie_object.id}" + end + end + let(:url) { "http://movies.com/#{movie.id}" } + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:data][:links][:public_url]).to eq url + end + end + + context 'with method' do + before do + MovieSerializer.link(:object_id, :id) + end + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:data][:links][:object_id]).to eq movie.id + end + end + end + describe '#key_transform' do subject(:hash) { movie_serializer_class.new([movie, movie], include: [:movie_type]).serializable_hash } diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 558e382..f329744 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -31,6 +31,30 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:included]).to be nil end + it 'returns correct nested includes when serializable_hash is called' do + # 3 actors, 3 agencies + include_object_total = 6 + + options = {} + options[:include] = [:actors, :'actors.agency'] + serializable_hash = MovieSerializer.new([movie], options).serializable_hash + + expect(serializable_hash[:included]).to be_instance_of(Array) + expect(serializable_hash[:included].length).to eq include_object_total + (0..include_object_total-1).each do |include| + expect(serializable_hash[:included][include]).to be_instance_of(Hash) + end + + options[:include] = [:'actors.agency'] + serializable_hash = MovieSerializer.new([movie], options).serializable_hash + + expect(serializable_hash[:included]).to be_instance_of(Array) + expect(serializable_hash[:included].length).to eq include_object_total + (0..include_object_total-1).each do |include| + expect(serializable_hash[:included][include]).to be_instance_of(Hash) + end + end + it 'returns correct number of records when serialized_json is called for an array' do options = {} options[:meta] = { total: 2 } @@ -259,4 +283,23 @@ describe FastJsonapi::ObjectSerializer do expect(V1::BlahSerializer.record_type).to be :blah end end + + context 'when serializing included, serialize any links' do + before do + ActorSerializer.link(:self) do |actor_object| + actor_url(actor_object) + end + end + subject(:serializable_hash) do + options = {} + options[:include] = [:actors] + MovieSerializer.new(movie, options).serializable_hash + end + let(:actor) { movie.actors.first } + let(:url) { "http://movies.com/actors/#{actor.id}" } + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:included][0][:links][:self]).to eq url + end + end end diff --git a/spec/lib/serialization_core_spec.rb b/spec/lib/serialization_core_spec.rb index 5d4563f..64f25bb 100644 --- a/spec/lib/serialization_core_spec.rb +++ b/spec/lib/serialization_core_spec.rb @@ -70,7 +70,7 @@ describe FastJsonapi::ObjectSerializer do end it 'returns correct hash when record_hash is called' do - record_hash = MovieSerializer.send(:record_hash, movie) + record_hash = MovieSerializer.send(:record_hash, movie, nil) expect(record_hash[:id]).to eq movie.id.to_s expect(record_hash[:type]).to eq MovieSerializer.record_type expect(record_hash).to have_key(:attributes) if MovieSerializer.attributes_to_serialize.present? @@ -82,7 +82,7 @@ describe FastJsonapi::ObjectSerializer do known_included_objects = {} included_records = [] [movie, movie].each do |record| - included_records.concat MovieSerializer.send(:get_included_records, record, includes_list, known_included_objects) + included_records.concat MovieSerializer.send(:get_included_records, record, includes_list, known_included_objects, nil) end expect(included_records.size).to eq 3 end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 3af0a2f..a68a3d1 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -114,6 +114,10 @@ RSpec.shared_context 'movie class' do attr_accessor :id, :name, :actor_ids end + class Agency + attr_accessor :id, :name, :actor_ids + end + class Supplier attr_accessor :id, :account_id @@ -140,6 +144,10 @@ RSpec.shared_context 'movie class' do belongs_to :owner, record_type: :user belongs_to :movie_type has_one :advertising_campaign + + def movie_url(movie) + "http://movies.com/#{movie.id}" + end end class MovieWithoutIdStructSerializer @@ -176,6 +184,10 @@ RSpec.shared_context 'movie class' do belongs_to :agency has_many :awards belongs_to :agency + + def actor_url(actor) + "http://movies.com/actors/#{actor.id}" + end end class AgencySerializer @@ -219,6 +231,21 @@ RSpec.shared_context 'movie class' do end end + class MovieSerializerWithAttributeBlock + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name, :release_year + attribute :title_with_year do |record| + "#{record.name} (#{record.release_year})" + end + end + + class AgencySerializer + include FastJsonapi::ObjectSerializer + attributes :id, :name + has_many :actors + end + class SupplierSerializer include FastJsonapi::ObjectSerializer set_type :supplier From ea5296ac250fd6eef412333b1bcb640af797e757 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Sun, 20 May 2018 14:30:17 -0700 Subject: [PATCH 49/73] making object level links similar to attributes --- README.md | 25 +++++++++++++++++++ lib/fast_jsonapi/object_serializer.rb | 12 +++++---- lib/fast_jsonapi/serialization_core.rb | 24 ++++++++---------- .../object_serializer_class_methods_spec.rb | 12 ++++++++- spec/lib/object_serializer_spec.rb | 2 +- spec/shared/contexts/movie_context.rb | 16 ++++++------ 6 files changed, 63 insertions(+), 28 deletions(-) diff --git a/README.md b/README.md index 784a335..2cbaa51 100644 --- a/README.md +++ b/README.md @@ -205,6 +205,31 @@ class MovieSerializer end ``` +### Links Per Object +Links are defined in FastJsonapi using the `link` method. By default, link are read directly from the model property of the same name.In this example, `public_url` is expected to be a property of the object being serialized. + +You can configure the method to use on the object for example a link with key `self` will get set to the value returned by a method called `url` on the movie object. + +You can also use a block to define a url as shown in `custom_url`. You can access params in these blocks as well as shown in `personalized_url` + +```ruby +class MovieSerializer + include FastJsonapi::ObjectSerializer + + link :public_url + + link :self, :url + + link :custom_url do |object| + "http://movies.com/#{object.name}-(#{object.year})" + end + + link :personalized_url do |object, params| + "http://movies.com/#{object.name}-#{params[:user].reference_code}" + end +end +``` + ### Compound Document Support for top-level and nested included associations through ` options[:include] `. diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 163059b..bf04aef 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -38,7 +38,7 @@ module FastJsonapi return serializable_hash unless @resource - serializable_hash[:data] = self.class.record_hash(@resource, @params, self) + serializable_hash[:data] = self.class.record_hash(@resource, @params) serializable_hash[:included] = self.class.get_included_records(@resource, @includes, @known_included_objects, @params) if @includes.present? serializable_hash end @@ -49,7 +49,7 @@ module FastJsonapi data = [] included = [] @resource.each do |record| - data << self.class.record_hash(record, @params, self) + data << self.class.record_hash(record, @params) included.concat self.class.get_included_records(record, @includes, @known_included_objects, @params) if @includes.present? end @@ -229,9 +229,11 @@ module FastJsonapi {} end - def link(name, value = nil, &block) - self.data_links = {} unless data_links - self.data_links[name] = block || value + def link(link_name, link_method_name = nil, &block) + self.data_links = {} if self.data_links.nil? + link_method_name = link_name if link_method_name.nil? + key = run_key_transform(link_name) + self.data_links[key] = block || link_method_name end def validate_includes!(includes) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 3d680ae..6dea90e 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -62,9 +62,13 @@ module FastJsonapi id_hash_from_record associated_object, polymorphic end - def links_hash(record, serializer_instance) - @data_links.each_with_object({}) do |(key, method), link_hash| - link_hash[key] = method.is_a?(Proc) ? serializer_instance.instance_exec(record, &method) : record.public_send(method) + def links_hash(record, params = {}) + data_links.each_with_object({}) do |(key, method), link_hash| + link_hash[key] = if method.is_a?(Proc) + method.arity == 1 ? method.call(record) : method.call(record, params) + else + record.public_send(method) + end end end @@ -90,17 +94,14 @@ module FastJsonapi end end - def record_hash(record, params = {}, serializer_instance) + def record_hash(record, params = {}) if cached record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do temp_hash = id_hash(id_from_record(record), record_type, true) temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? temp_hash[:relationships] = {} temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize, params) if cachable_relationships_to_serialize.present? - if @data_links.present? - temp_links_hash = links_hash(record, serializer_instance) - temp_hash[:links] = temp_links_hash if temp_links_hash - end + temp_hash[:links] = links_hash(record, params) if data_links.present? temp_hash end record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize, params)) if uncachable_relationships_to_serialize.present? @@ -109,10 +110,7 @@ module FastJsonapi record_hash = id_hash(id_from_record(record), record_type, true) record_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present? record_hash[:relationships] = relationships_hash(record, nil, params) if relationships_to_serialize.present? - if @data_links.present? - temp_links_hash = links_hash(record, serializer_instance) - record_hash[:links] = temp_links_hash if temp_links_hash - end + record_hash[:links] = links_hash(record, params) if data_links.present? record_hash end end @@ -168,7 +166,7 @@ module FastJsonapi next if known_included_objects.key?(code) known_included_objects[code] = inc_obj - included_records << serializer.record_hash(inc_obj, params, serializer.new(record)) + included_records << serializer.record_hash(inc_obj, params) end end end diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 34a5349..55e351a 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -243,7 +243,7 @@ describe FastJsonapi::ObjectSerializer do context 'with block calling instance method on serializer' do before do MovieSerializer.link(:self) do |movie_object| - movie_url(movie_object) + movie_object.url end end let(:url) { "http://movies.com/#{movie.id}" } @@ -275,6 +275,16 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:data][:links][:object_id]).to eq movie.id end end + + context 'with method and convention' do + before do + MovieSerializer.link(:url) + end + + it 'returns correct hash when serializable_hash is called' do + expect(serializable_hash[:data][:links][:url]).to eq movie.url + end + end end describe '#key_transform' do diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index f329744..2d8c99e 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -287,7 +287,7 @@ describe FastJsonapi::ObjectSerializer do context 'when serializing included, serialize any links' do before do ActorSerializer.link(:self) do |actor_object| - actor_url(actor_object) + actor_object.url end end subject(:serializable_hash) do diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index a68a3d1..64d901d 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -42,6 +42,10 @@ RSpec.shared_context 'movie class' do def cache_key "#{id}" end + + def url + "http://movies.com/#{id}" + end end class Actor @@ -68,6 +72,10 @@ RSpec.shared_context 'movie class' do def award_ids [id * 9, id * 9 + 1] end + + def url + "http://movies.com/actors/#{id}" + end end class AdvertisingCampaign @@ -144,10 +152,6 @@ RSpec.shared_context 'movie class' do belongs_to :owner, record_type: :user belongs_to :movie_type has_one :advertising_campaign - - def movie_url(movie) - "http://movies.com/#{movie.id}" - end end class MovieWithoutIdStructSerializer @@ -184,10 +188,6 @@ RSpec.shared_context 'movie class' do belongs_to :agency has_many :awards belongs_to :agency - - def actor_url(actor) - "http://movies.com/actors/#{actor.id}" - end end class AgencySerializer From a018f1d32f6ee3feebe1970ce7bec38df0990ae6 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Sun, 20 May 2018 15:04:01 -0700 Subject: [PATCH 50/73] minor fixes to data links feature --- lib/fast_jsonapi/object_serializer.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index bf04aef..4dea074 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -72,7 +72,6 @@ module FastJsonapi @known_included_objects = {} @meta = options[:meta] @links = options[:links] - @data_links = {} @params = options[:params] || {} raise ArgumentError.new("`params` option passed to serializer must be a hash") unless @params.is_a?(Hash) @@ -97,6 +96,7 @@ module FastJsonapi subclass.transform_method = transform_method subclass.cache_length = cache_length subclass.race_condition_ttl = race_condition_ttl + subclass.data_links = data_links subclass.cached = cached end From cd1bc0968e2c1c3ada4d3c73579a53bc7bf7ac44 Mon Sep 17 00:00:00 2001 From: Shishir Kakaraddi Date: Mon, 14 May 2018 20:15:49 -0700 Subject: [PATCH 51/73] dont create a object for a has one relationship unnecessarily just to fetch id --- lib/fast_jsonapi/serialization_core.rb | 6 +----- spec/lib/object_serializer_inheritance_spec.rb | 8 ++++++++ spec/shared/contexts/movie_context.rb | 6 +++++- 3 files changed, 14 insertions(+), 6 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 6dea90e..de138cd 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -185,11 +185,7 @@ module FastJsonapi return object.id end - if relationship[:relationship_type] == :has_one - record.public_send(relationship[:object_method_name]).try(:id) - else - record.public_send(relationship[:id_method_name]) - end + record.public_send(relationship[:id_method_name]) end end end diff --git a/spec/lib/object_serializer_inheritance_spec.rb b/spec/lib/object_serializer_inheritance_spec.rb index b8ba19e..06dba25 100644 --- a/spec/lib/object_serializer_inheritance_spec.rb +++ b/spec/lib/object_serializer_inheritance_spec.rb @@ -30,6 +30,10 @@ describe FastJsonapi::ObjectSerializer do p.user_id = id p end + + def photo_id + 1 + end end class UserSerializer @@ -77,6 +81,10 @@ describe FastJsonapi::ObjectSerializer do a.employee_id = id a end + + def account_id + 1 + end end class EmployeeSerializer < UserSerializer diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 64d901d..7871f8b 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -31,6 +31,10 @@ RSpec.shared_context 'movie class' do mt end + def advertising_campaign_id + 1 + end + def advertising_campaign ac = AdvertisingCampaign.new ac.id = 1 @@ -284,7 +288,7 @@ RSpec.shared_context 'movie class' do :owner_id, :owner, :movie_type_id, - :advertising_campaign + :advertising_campaign_id ) ActorStruct = Struct.new(:id, :name, :email, :agency_id, :award_ids) From f54e6242ff8f6e206a44b33b7df4fdd22fc76af9 Mon Sep 17 00:00:00 2001 From: Ray Walters Date: Mon, 21 May 2018 10:26:54 -0400 Subject: [PATCH 52/73] Remove extra 'class MovieSerializer' from an example in the README --- README.md | 1 - 1 file changed, 1 deletion(-) diff --git a/README.md b/README.md index 2cbaa51..4e7eaa4 100644 --- a/README.md +++ b/README.md @@ -284,7 +284,6 @@ block you opt-in to using params by adding it as a block parameter. ```ruby class MovieSerializer - class MovieSerializer include FastJsonapi::ObjectSerializer attributes :name, :year From 44d5e0f9c5bd54a5fbfd265b805c55e65d95af11 Mon Sep 17 00:00:00 2001 From: homer Date: Mon, 28 May 2018 18:49:36 +0500 Subject: [PATCH 53/73] Fix serialization for nested nil includes with block --- lib/fast_jsonapi/serialization_core.rb | 2 +- spec/lib/object_serializer_spec.rb | 7 +++++++ spec/shared/contexts/movie_context.rb | 19 ++++++++++++++++++- 3 files changed, 26 insertions(+), 2 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index de138cd..668aebe 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -182,7 +182,7 @@ module FastJsonapi object = relationship[:object_block].call(record, params) return object.map(&:id) if object.respond_to? :map - return object.id + return object.try(:id) end record.public_send(relationship[:id_method_name]) diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 2d8c99e..7941f66 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -97,6 +97,13 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash['data']['relationships']['owner']['data']).to be nil end + it 'returns correct json when belongs_to returns nil and there is a block for the relationship' do + movie.owner_id = nil + json = MovieSerializer.new(movie, {include: [:owner]}).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['relationships']['owner']['data']).to be nil + end + it 'returns correct json when has_one returns nil' do supplier.account_id = nil json = SupplierSerializer.new(supplier).serialized_json diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 7871f8b..a73e869 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -43,6 +43,13 @@ RSpec.shared_context 'movie class' do ac end + def owner + return unless owner_id + ow = Owner.new + ow.id = owner_id + ow + end + def cache_key "#{id}" end @@ -146,6 +153,14 @@ RSpec.shared_context 'movie class' do attr_accessor :id end + class Owner + attr_accessor :id + end + + class OwnerSerializer + include FastJsonapi::ObjectSerializer + end + # serializers class MovieSerializer include FastJsonapi::ObjectSerializer @@ -153,7 +168,9 @@ RSpec.shared_context 'movie class' do # director attr is not mentioned intentionally attributes :name, :release_year has_many :actors - belongs_to :owner, record_type: :user + belongs_to :owner, record_type: :user do |object, params| + object.owner + end belongs_to :movie_type has_one :advertising_campaign end From bad004fd4244cd6f89b441f291d69ca6ef46b251 Mon Sep 17 00:00:00 2001 From: Trevor Hinesley Date: Thu, 21 Jun 2018 14:50:51 -0500 Subject: [PATCH 54/73] Allow conditional attributes --- lib/fast_jsonapi/object_serializer.rb | 9 +++++++- lib/fast_jsonapi/serialization_core.rb | 11 ++++++++- spec/lib/object_serializer_spec.rb | 32 ++++++++++++++++++++++++++ spec/shared/contexts/movie_context.rb | 14 +++++++++++ 4 files changed, 64 insertions(+), 2 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 4dea074..895fa8c 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -149,11 +149,18 @@ module FastJsonapi def attributes(*attributes_list, &block) attributes_list = attributes_list.first if attributes_list.first.class.is_a?(Array) + options = attributes_list.last.is_a?(Hash) ? attributes_list.pop : {} self.attributes_to_serialize = {} if self.attributes_to_serialize.nil? + self.optional_attributes_to_serialize = {} if self.optional_attributes_to_serialize.nil? + attributes_list.each do |attr_name| method_name = attr_name key = run_key_transform(method_name) - attributes_to_serialize[key] = block || method_name + if options[:if].present? + optional_attributes_to_serialize[key] = [method_name, options[:if]] + else + attributes_to_serialize[key] = block || method_name + end end end diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 668aebe..f838dff 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -12,6 +12,7 @@ module FastJsonapi included do class << self attr_accessor :attributes_to_serialize, + :optional_attributes_to_serialize, :relationships_to_serialize, :cachable_relationships_to_serialize, :uncachable_relationships_to_serialize, @@ -73,13 +74,21 @@ module FastJsonapi end def attributes_hash(record, params = {}) - attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash| + attributes = attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash| attr_hash[key] = if method.is_a?(Proc) method.arity == 1 ? method.call(record) : method.call(record, params) else record.public_send(method) end end + + self.optional_attributes_to_serialize = {} if self.optional_attributes_to_serialize.nil? + optional_attributes_to_serialize.each do |key, details| + method_name, check_proc = details + attributes[key] = record.send(method_name) if check_proc.call(record, params) + end + + attributes end def relationships_hash(record, relationships = nil, params = {}) diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 7941f66..46d0cea 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -309,4 +309,36 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash[:included][0][:links][:self]).to eq url end end + + context 'when optional attributes are determined by record data' do + it 'returns optional attribute when attribute is included' do + movie.release_year = 2001 + json = MovieOptionalRecordDataSerializer.new(movie).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['attributes']['release_year']).to eq movie.release_year + end + + it "doesn't return optional attribute when attribute is not included" do + movie.release_year = 1970 + json = MovieOptionalRecordDataSerializer.new(movie).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['attributes'].has_key?('release_year')).to be_falsey + end + end + + context 'when optional attributes are determined by params data' do + it 'returns optional attribute when attribute is included' do + movie.director = 'steven spielberg' + json = MovieOptionalParamsDataSerializer.new(movie, { params: { admin: true }}).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['attributes']['director']).to eq 'steven spielberg' + end + + it "doesn't return optional attribute when attribute is not included" do + movie.director = 'steven spielberg' + json = MovieOptionalParamsDataSerializer.new(movie, { params: { admin: false }}).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['attributes'].has_key?('director')).to be_falsey + end + end end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index a73e869..4df6a91 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -278,6 +278,20 @@ RSpec.shared_context 'movie class' do set_type :account belongs_to :supplier end + + class MovieOptionalRecordDataSerializer + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name + attribute :release_year, if: Proc.new { |record| record.release_year >= 2000 } + end + + class MovieOptionalParamsDataSerializer + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name + attribute :director, if: Proc.new { |record, params| params && params[:admin] == true } + end end From f1df3f4a2dc8fed3bc3876f957cb2670a57b2ba8 Mon Sep 17 00:00:00 2001 From: Trevor Hinesley Date: Thu, 21 Jun 2018 15:01:02 -0500 Subject: [PATCH 55/73] Added documentation about conditional attributes --- README.md | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/README.md b/README.md index 4e7eaa4..85926e8 100644 --- a/README.md +++ b/README.md @@ -307,6 +307,32 @@ serializer.serializable_hash Custom attributes and relationships that only receive the resource are still possible by defining the block to only receive one argument. +### Conditional Attributes + +Conditional attributes can be defined by passing a Proc to the `if` key on the `attribute` method. Return `true` if the attribute should be serialized, and `false` if not. The record and any params passed to the serializer are available inside the Proc as the first and second parameters, respectively. + +```ruby +class MovieSerializer + include FastJsonapi::ObjectSerializer + + attributes :name, :year + attribute :release_year, if: Proc.new do |record| + # Release year will only be serialized if it's greater than 1990 + record.release_year > 1990 + end + + attribute :director, if: Proc.new do |record, params| + # The director will be serialized only if the :admin key of params is true + params && params[:admin] == true + end +end + +# ... +current_user = User.find(cookies[:current_user_id]) +serializer = MovieSerializer.new(movie, { params: { admin: current_user.admin? }}) +serializer.serializable_hash +``` + ### Customizable Options Option | Purpose | Example From 5c820695b3dcbcc1103e09a1347eccb0a716e747 Mon Sep 17 00:00:00 2001 From: Trevor Hinesley Date: Thu, 21 Jun 2018 16:01:41 -0500 Subject: [PATCH 56/73] Split attribute serialization into its own class --- lib/fast_jsonapi/attribute_serializer.rb | 29 +++++++++++++++++++ lib/fast_jsonapi/object_serializer.rb | 12 ++++---- lib/fast_jsonapi/serialization_core.rb | 17 ++--------- .../lib/object_serializer_inheritance_spec.rb | 2 +- spec/lib/serialization_core_spec.rb | 4 +-- 5 files changed, 40 insertions(+), 24 deletions(-) create mode 100644 lib/fast_jsonapi/attribute_serializer.rb diff --git a/lib/fast_jsonapi/attribute_serializer.rb b/lib/fast_jsonapi/attribute_serializer.rb new file mode 100644 index 0000000..0469407 --- /dev/null +++ b/lib/fast_jsonapi/attribute_serializer.rb @@ -0,0 +1,29 @@ +module FastJsonapi + class AttributeSerializer + attr_reader :key, :method, :conditional_proc + + def initialize(key:, method:, options: {}) + @key = key + @method = method + @conditional_proc = options[:if] + end + + def serialize(record, serialization_params, output_hash) + if include_attribute?(record, serialization_params) + output_hash[key] = if method.is_a?(Proc) + method.arity == 1 ? method.call(record) : method.call(record, serialization_params) + else + record.public_send(method) + end + end + end + + def include_attribute?(record, serialization_params) + if conditional_proc.present? + conditional_proc.call(record, serialization_params) + else + true + end + end + end +end \ No newline at end of file diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 895fa8c..5af7e84 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -3,6 +3,7 @@ require 'active_support/core_ext/object' require 'active_support/concern' require 'active_support/inflector' +require 'fast_jsonapi/attribute_serializer' require 'fast_jsonapi/serialization_core' module FastJsonapi @@ -151,16 +152,15 @@ module FastJsonapi attributes_list = attributes_list.first if attributes_list.first.class.is_a?(Array) options = attributes_list.last.is_a?(Hash) ? attributes_list.pop : {} self.attributes_to_serialize = {} if self.attributes_to_serialize.nil? - self.optional_attributes_to_serialize = {} if self.optional_attributes_to_serialize.nil? attributes_list.each do |attr_name| method_name = attr_name key = run_key_transform(method_name) - if options[:if].present? - optional_attributes_to_serialize[key] = [method_name, options[:if]] - else - attributes_to_serialize[key] = block || method_name - end + attributes_to_serialize[key] = AttributeSerializer.new( + key: key, + method: block || method_name, + options: options + ) end end diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index f838dff..27da6af 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -12,7 +12,6 @@ module FastJsonapi included do class << self attr_accessor :attributes_to_serialize, - :optional_attributes_to_serialize, :relationships_to_serialize, :cachable_relationships_to_serialize, :uncachable_relationships_to_serialize, @@ -74,21 +73,9 @@ module FastJsonapi end def attributes_hash(record, params = {}) - attributes = attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash| - attr_hash[key] = if method.is_a?(Proc) - method.arity == 1 ? method.call(record) : method.call(record, params) - else - record.public_send(method) - end + attributes_to_serialize.each_with_object({}) do |(key, attribute), attr_hash| + attribute.serialize(record, params, attr_hash) end - - self.optional_attributes_to_serialize = {} if self.optional_attributes_to_serialize.nil? - optional_attributes_to_serialize.each do |key, details| - method_name, check_proc = details - attributes[key] = record.send(method_name) if check_proc.call(record, params) - end - - attributes end def relationships_hash(record, relationships = nil, params = {}) diff --git a/spec/lib/object_serializer_inheritance_spec.rb b/spec/lib/object_serializer_inheritance_spec.rb index 06dba25..8cf5b53 100644 --- a/spec/lib/object_serializer_inheritance_spec.rb +++ b/spec/lib/object_serializer_inheritance_spec.rb @@ -113,7 +113,7 @@ describe FastJsonapi::ObjectSerializer do end it 'includes child attributes' do - expect(EmployeeSerializer.attributes_to_serialize[:location]).to eq(:location) + expect(EmployeeSerializer.attributes_to_serialize[:location].method).to eq(:location) end it 'doesnt change parent class attributes' do diff --git a/spec/lib/serialization_core_spec.rb b/spec/lib/serialization_core_spec.rb index 64f25bb..7285e38 100644 --- a/spec/lib/serialization_core_spec.rb +++ b/spec/lib/serialization_core_spec.rb @@ -39,9 +39,9 @@ describe FastJsonapi::ObjectSerializer do attributes_hash = MovieSerializer.send(:attributes_hash, movie) attribute_names = attributes_hash.keys.sort expect(attribute_names).to eq MovieSerializer.attributes_to_serialize.keys.sort - MovieSerializer.attributes_to_serialize.each do |key, method_name| + MovieSerializer.attributes_to_serialize.each do |key, attribute| value = attributes_hash[key] - expect(value).to eq movie.send(method_name) + expect(value).to eq movie.send(attribute.method) end end From ba4e112829e8c48d7c67f6f17f86b7bc735a4cc2 Mon Sep 17 00:00:00 2001 From: Trevor Hinesley Date: Thu, 21 Jun 2018 16:06:18 -0500 Subject: [PATCH 57/73] Since attributes are an instantiated class now, renamed AttributeSerializer to Attribute --- lib/fast_jsonapi/{attribute_serializer.rb => attribute.rb} | 2 +- lib/fast_jsonapi/object_serializer.rb | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) rename lib/fast_jsonapi/{attribute_serializer.rb => attribute.rb} (96%) diff --git a/lib/fast_jsonapi/attribute_serializer.rb b/lib/fast_jsonapi/attribute.rb similarity index 96% rename from lib/fast_jsonapi/attribute_serializer.rb rename to lib/fast_jsonapi/attribute.rb index 0469407..c26bf19 100644 --- a/lib/fast_jsonapi/attribute_serializer.rb +++ b/lib/fast_jsonapi/attribute.rb @@ -1,5 +1,5 @@ module FastJsonapi - class AttributeSerializer + class Attribute attr_reader :key, :method, :conditional_proc def initialize(key:, method:, options: {}) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 5af7e84..c0aacd4 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -3,7 +3,7 @@ require 'active_support/core_ext/object' require 'active_support/concern' require 'active_support/inflector' -require 'fast_jsonapi/attribute_serializer' +require 'fast_jsonapi/attribute' require 'fast_jsonapi/serialization_core' module FastJsonapi @@ -156,7 +156,7 @@ module FastJsonapi attributes_list.each do |attr_name| method_name = attr_name key = run_key_transform(method_name) - attributes_to_serialize[key] = AttributeSerializer.new( + attributes_to_serialize[key] = Attribute.new( key: key, method: block || method_name, options: options From 4a333d727650032215490b10dd16adbd92668e92 Mon Sep 17 00:00:00 2001 From: Darren Johnson Date: Thu, 14 Jun 2018 16:13:03 -0700 Subject: [PATCH 58/73] Set type value when setting key transform --- lib/fast_jsonapi/object_serializer.rb | 5 ++++- spec/lib/object_serializer_class_methods_spec.rb | 9 ++++----- .../examples/object_serializer_class_methods_examples.rb | 4 ++-- 3 files changed, 10 insertions(+), 8 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index c0aacd4..c33465f 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -119,6 +119,9 @@ module FastJsonapi underscore: :underscore } self.transform_method = mapping[transform_name.to_sym] + + # ensure that the record type is correctly transformed + set_type(reflected_record_type) if reflected_record_type end def run_key_transform(input) @@ -177,7 +180,7 @@ module FastJsonapi self.cachable_relationships_to_serialize[name] = relationship end self.relationships_to_serialize[name] = relationship - end + end def has_many(relationship_name, options = {}, &block) name = relationship_name.to_sym diff --git a/spec/lib/object_serializer_class_methods_spec.rb b/spec/lib/object_serializer_class_methods_spec.rb index 55e351a..85346f0 100644 --- a/spec/lib/object_serializer_class_methods_spec.rb +++ b/spec/lib/object_serializer_class_methods_spec.rb @@ -312,7 +312,6 @@ describe FastJsonapi::ObjectSerializer do movie_type_serializer_class.instance_eval do include FastJsonapi::ObjectSerializer set_key_transform key_transform - set_type :movie_type attributes :name end end @@ -321,25 +320,25 @@ describe FastJsonapi::ObjectSerializer do context 'when key_transform is dash' do let(:key_transform) { :dash } - it_behaves_like 'returning key transformed hash', :'movie-type', :'release-year' + it_behaves_like 'returning key transformed hash', :'movie-type', :'dash-movie-type', :'release-year' end context 'when key_transform is camel' do let(:key_transform) { :camel } - it_behaves_like 'returning key transformed hash', :MovieType, :ReleaseYear + it_behaves_like 'returning key transformed hash', :MovieType, :CamelMovieType, :ReleaseYear end context 'when key_transform is camel_lower' do let(:key_transform) { :camel_lower } - it_behaves_like 'returning key transformed hash', :movieType, :releaseYear + it_behaves_like 'returning key transformed hash', :movieType, :camelLowerMovieType, :releaseYear end context 'when key_transform is underscore' do let(:key_transform) { :underscore } - it_behaves_like 'returning key transformed hash', :movie_type, :release_year + it_behaves_like 'returning key transformed hash', :movie_type, :underscore_movie_type, :release_year end end end diff --git a/spec/shared/examples/object_serializer_class_methods_examples.rb b/spec/shared/examples/object_serializer_class_methods_examples.rb index c529dcb..616ccf5 100644 --- a/spec/shared/examples/object_serializer_class_methods_examples.rb +++ b/spec/shared/examples/object_serializer_class_methods_examples.rb @@ -8,11 +8,11 @@ RSpec.shared_examples 'returning correct relationship hash' do |serializer, id_m end end -RSpec.shared_examples 'returning key transformed hash' do |movie_type, release_year| +RSpec.shared_examples 'returning key transformed hash' do |movie_type, serializer_type, release_year| it 'returns correctly transformed hash' do expect(hash[:data][0][:attributes]).to have_key(release_year) expect(hash[:data][0][:relationships]).to have_key(movie_type) expect(hash[:data][0][:relationships][movie_type][:data][:type]).to eq(movie_type) - expect(hash[:included][0][:type]).to eq(movie_type) + expect(hash[:included][0][:type]).to eq(serializer_type) end end From 2b01d8ce708afd377905975e0438b45bffe126c4 Mon Sep 17 00:00:00 2001 From: Guillermo Iguaran Date: Thu, 7 Jun 2018 11:02:00 -0500 Subject: [PATCH 59/73] Use a Railtie to extend Rails --- lib/extensions/has_one.rb | 30 ++++++++++++++---------------- lib/fast_jsonapi.rb | 6 +++++- lib/fast_jsonapi/railtie.rb | 11 +++++++++++ 3 files changed, 30 insertions(+), 17 deletions(-) create mode 100644 lib/fast_jsonapi/railtie.rb diff --git a/lib/extensions/has_one.rb b/lib/extensions/has_one.rb index 930ca57..1588359 100644 --- a/lib/extensions/has_one.rb +++ b/lib/extensions/has_one.rb @@ -1,20 +1,18 @@ # frozen_string_literal: true -if defined?(::ActiveRecord) - ::ActiveRecord::Associations::Builder::HasOne.class_eval do - # Based on - # https://github.com/rails/rails/blob/master/activerecord/lib/active_record/associations/builder/collection_association.rb#L50 - # https://github.com/rails/rails/blob/master/activerecord/lib/active_record/associations/builder/singular_association.rb#L11 - def self.define_accessors(mixin, reflection) - super - name = reflection.name - mixin.class_eval <<-CODE, __FILE__, __LINE__ + 1 - def #{name}_id - # if an attribute is already defined with this methods name we should just use it - return read_attribute(__method__) if has_attribute?(__method__) - association(:#{name}).reader.try(:id) - end - CODE - end +::ActiveRecord::Associations::Builder::HasOne.class_eval do + # Based on + # https://github.com/rails/rails/blob/master/activerecord/lib/active_record/associations/builder/collection_association.rb#L50 + # https://github.com/rails/rails/blob/master/activerecord/lib/active_record/associations/builder/singular_association.rb#L11 + def self.define_accessors(mixin, reflection) + super + name = reflection.name + mixin.class_eval <<-CODE, __FILE__, __LINE__ + 1 + def #{name}_id + # if an attribute is already defined with this methods name we should just use it + return read_attribute(__method__) if has_attribute?(__method__) + association(:#{name}).reader.try(:id) + end + CODE end end diff --git a/lib/fast_jsonapi.rb b/lib/fast_jsonapi.rb index d257b6e..cb4915b 100644 --- a/lib/fast_jsonapi.rb +++ b/lib/fast_jsonapi.rb @@ -2,5 +2,9 @@ module FastJsonapi require 'fast_jsonapi/object_serializer' - require 'extensions/has_one' + if defined?(::Rails) + require 'fast_jsonapi/railtie' + elsif defined?(::ActiveRecord) + require 'extensions/has_one' + end end diff --git a/lib/fast_jsonapi/railtie.rb b/lib/fast_jsonapi/railtie.rb new file mode 100644 index 0000000..e6a2717 --- /dev/null +++ b/lib/fast_jsonapi/railtie.rb @@ -0,0 +1,11 @@ +# frozen_string_literal: true + +require 'rails/railtie' + +class Railtie < Rails::Railtie + initializer 'fast_jsonapi.active_record' do + ActiveSupport.on_load :active_record do + require 'extensions/has_one' + end + end +end From 5558dcd7035572c98b39b784aea4acecfff6a62e Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Mon, 25 Jun 2018 16:10:06 -0500 Subject: [PATCH 60/73] allow conditional relationships --- lib/fast_jsonapi/object_serializer.rb | 3 ++- lib/fast_jsonapi/serialization_core.rb | 13 +++++++----- spec/lib/object_serializer_spec.rb | 29 ++++++++++++++++++++++++++ spec/shared/contexts/movie_context.rb | 14 +++++++++++++ 4 files changed, 53 insertions(+), 6 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index c33465f..7ce418c 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -221,7 +221,8 @@ module FastJsonapi serializer: compute_serializer_name(options[:serializer] || base_key_sym), relationship_type: relationship_type, cached: options[:cached] || false, - polymorphic: fetch_polymorphic_option(options) + polymorphic: fetch_polymorphic_option(options), + conditional_proc: options[:if] } end diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 27da6af..f02c2c4 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -82,11 +82,14 @@ module FastJsonapi relationships = relationships_to_serialize if relationships.nil? relationships.each_with_object({}) do |(_k, relationship), hash| - name = relationship[:key] - empty_case = relationship[:relationship_type] == :has_many ? [] : nil - hash[name] = { - data: ids_hash_from_record_and_relationship(record, relationship, params) || empty_case - } + conditional_proc = relationship[:conditional_proc] + if conditional_proc.blank? || conditional_proc.call(record, params) + name = relationship[:key] + empty_case = relationship[:relationship_type] == :has_many ? [] : nil + hash[name] = { + data: ids_hash_from_record_and_relationship(record, relationship, params) || empty_case + } + end end end diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 46d0cea..7338047 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -341,4 +341,33 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash['data']['attributes'].has_key?('director')).to be_falsey end end + + context 'when optional relationships are determined by record data' do + it 'returns optional relationship when relationship is included' do + json = MovieOptionalRelationshipSerializer.new(movie).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_truthy + end + + it "doesn't return optional relationship when relationship is not included" do + movie.actor_ids = [] + json = MovieOptionalRelationshipSerializer.new(movie).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_falsey + end + end + + context 'when optional relationships are determined by params data' do + it 'returns optional relationship when relationship is included' do + json = MovieOptionalRelationshipWithParamsSerializer.new(movie, { params: { admin: true }}).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_truthy + end + + it "doesn't return optional relationship when relationship is not included" do + json = MovieOptionalRelationshipWithParamsSerializer.new(movie, { params: { admin: false }}).serialized_json + serializable_hash = JSON.parse(json) + expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_falsey + end + end end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 4df6a91..6a5693e 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -292,6 +292,20 @@ RSpec.shared_context 'movie class' do attributes :name attribute :director, if: Proc.new { |record, params| params && params[:admin] == true } end + + class MovieOptionalRelationshipSerializer + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name + has_many :actors, if: Proc.new { |record| record.actors.any? } + end + + class MovieOptionalRelationshipWithParamsSerializer + include FastJsonapi::ObjectSerializer + set_type :movie + attributes :name + has_many :actors, if: Proc.new { |record, params| params && params[:admin] == true } + end end From 25c099e923abc59b2854393014b155407308e597 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Tue, 26 Jun 2018 16:06:33 -0500 Subject: [PATCH 61/73] add documentation for conditional relationships --- README.md | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/README.md b/README.md index 85926e8..af57e7a 100644 --- a/README.md +++ b/README.md @@ -30,6 +30,8 @@ Fast JSON API serialized 250 records in 3.01 ms * [Collection Serialization](#collection-serialization) * [Caching](#caching) * [Params](#params) + * [Conditional Attributes](#conditional-attributes) + * [Conditional Relationships](#conditional-relationships) * [Contributing](#contributing) @@ -333,6 +335,27 @@ serializer = MovieSerializer.new(movie, { params: { admin: current_user.admin? } serializer.serializable_hash ``` +### Conditional Relationships + +Conditional relationships can be defined by passing a Proc to the `if` key. Return `true` if the relationship should be serialized, and `false` if not. The record and any params passed to the serializer are available inside the Proc as the first and second parameters, respectively. + +```ruby +class MovieSerializer + include FastJsonapi::ObjectSerializer + + # Actors will only be serialized if the record has any associated actors + has_many :actors, if: Proc.new { |record| record.actors.any? } + + # Owner will only be serialized if the :admin key of params is true + belongs_to :owner, if: Proc.new { |record, params| params && params[:admin] == true } +end + +# ... +current_user = User.find(cookies[:current_user_id]) +serializer = MovieSerializer.new(movie, { params: { admin: current_user.admin? }}) +serializer.serializable_hash +``` + ### Customizable Options Option | Purpose | Example From 0b70657a41af0615340a1cfb1165aa4300cbeeee Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Tue, 26 Jun 2018 16:10:41 -0500 Subject: [PATCH 62/73] update test for conditional relationships --- spec/lib/object_serializer_spec.rb | 4 ++-- spec/shared/contexts/movie_context.rb | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index 7338047..cd6f3aa 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -361,13 +361,13 @@ describe FastJsonapi::ObjectSerializer do it 'returns optional relationship when relationship is included' do json = MovieOptionalRelationshipWithParamsSerializer.new(movie, { params: { admin: true }}).serialized_json serializable_hash = JSON.parse(json) - expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_truthy + expect(serializable_hash['data']['relationships'].has_key?('owner')).to be_truthy end it "doesn't return optional relationship when relationship is not included" do json = MovieOptionalRelationshipWithParamsSerializer.new(movie, { params: { admin: false }}).serialized_json serializable_hash = JSON.parse(json) - expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_falsey + expect(serializable_hash['data']['relationships'].has_key?('owner')).to be_falsey end end end diff --git a/spec/shared/contexts/movie_context.rb b/spec/shared/contexts/movie_context.rb index 6a5693e..bbd89a9 100644 --- a/spec/shared/contexts/movie_context.rb +++ b/spec/shared/contexts/movie_context.rb @@ -304,7 +304,7 @@ RSpec.shared_context 'movie class' do include FastJsonapi::ObjectSerializer set_type :movie attributes :name - has_many :actors, if: Proc.new { |record, params| params && params[:admin] == true } + belongs_to :owner, record_type: :user, if: Proc.new { |record, params| params && params[:admin] == true } end end From f8640997610a6b8b2e4a9044a588d100d9cb75d7 Mon Sep 17 00:00:00 2001 From: Trevor Hinesley Date: Fri, 29 Jun 2018 16:36:34 -0500 Subject: [PATCH 63/73] Conditional relationships should be removed from included when proc evaluates to false --- lib/fast_jsonapi/serialization_core.rb | 2 ++ spec/lib/object_serializer_spec.rb | 49 +++++++++++++++++++++----- 2 files changed, 42 insertions(+), 9 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index f02c2c4..284f017 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -146,6 +146,8 @@ module FastJsonapi items = parse_include_item(include_item) items.each do |item| next unless relationships_to_serialize && relationships_to_serialize[item] + conditional_proc = relationships_to_serialize[item][:conditional_proc] + next if conditional_proc && !conditional_proc.call(record, params) raise NotImplementedError if @relationships_to_serialize[item][:polymorphic].is_a?(Hash) record_type = @relationships_to_serialize[item][:record_type] serializer = @relationships_to_serialize[item][:serializer].to_s.constantize diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index cd6f3aa..ed770af 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -349,11 +349,27 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_truthy end - it "doesn't return optional relationship when relationship is not included" do - movie.actor_ids = [] - json = MovieOptionalRelationshipSerializer.new(movie).serialized_json - serializable_hash = JSON.parse(json) - expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_falsey + context "when relationship is not included" do + let(:json) { + MovieOptionalRelationshipSerializer.new(movie, options).serialized_json + } + let(:options) { + {} + } + let(:serializable_hash) { + JSON.parse(json) + } + + it "doesn't return optional relationship" do + movie.actor_ids = [] + expect(serializable_hash['data']['relationships'].has_key?('actors')).to be_falsey + end + + it "doesn't include optional relationship" do + movie.actor_ids = [] + options[:include] = [:actors] + expect(serializable_hash['included']).to be_blank + end end end @@ -364,10 +380,25 @@ describe FastJsonapi::ObjectSerializer do expect(serializable_hash['data']['relationships'].has_key?('owner')).to be_truthy end - it "doesn't return optional relationship when relationship is not included" do - json = MovieOptionalRelationshipWithParamsSerializer.new(movie, { params: { admin: false }}).serialized_json - serializable_hash = JSON.parse(json) - expect(serializable_hash['data']['relationships'].has_key?('owner')).to be_falsey + context "when relationship is not included" do + let(:json) { + MovieOptionalRelationshipWithParamsSerializer.new(movie, options).serialized_json + } + let(:options) { + { params: { admin: false }} + } + let(:serializable_hash) { + JSON.parse(json) + } + + it "doesn't return optional relationship" do + expect(serializable_hash['data']['relationships'].has_key?('owner')).to be_falsey + end + + it "doesn't include optional relationship" do + options[:include] = [:owner] + expect(serializable_hash['included']).to be_blank + end end end end From 7b23adddc4b88203f83e736e2953f8c24a7a12a1 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Sat, 30 Jun 2018 17:22:57 -0500 Subject: [PATCH 64/73] working on new relationship class --- lib/fast_jsonapi/object_serializer.rb | 38 ++++--- lib/fast_jsonapi/relationship.rb | 99 +++++++++++++++++++ lib/fast_jsonapi/serialization_core.rb | 62 ++---------- ...bject_serializer_class_methods_examples.rb | 2 +- 4 files changed, 125 insertions(+), 76 deletions(-) create mode 100644 lib/fast_jsonapi/relationship.rb diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 7ce418c..b00bdfd 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -4,6 +4,7 @@ require 'active_support/core_ext/object' require 'active_support/concern' require 'active_support/inflector' require 'fast_jsonapi/attribute' +require 'fast_jsonapi/relationship' require 'fast_jsonapi/serialization_core' module FastJsonapi @@ -169,38 +170,32 @@ module FastJsonapi alias_method :attribute, :attributes - def add_relationship(name, relationship) + def add_relationship(relationship) self.relationships_to_serialize = {} if relationships_to_serialize.nil? self.cachable_relationships_to_serialize = {} if cachable_relationships_to_serialize.nil? self.uncachable_relationships_to_serialize = {} if uncachable_relationships_to_serialize.nil? - - if !relationship[:cached] - self.uncachable_relationships_to_serialize[name] = relationship + + if !relationship.cached + self.uncachable_relationships_to_serialize[relationship.name] = relationship else - self.cachable_relationships_to_serialize[name] = relationship + self.cachable_relationships_to_serialize[relationship.name] = relationship end - self.relationships_to_serialize[name] = relationship + self.relationships_to_serialize[relationship.name] = relationship end def has_many(relationship_name, options = {}, &block) - name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :has_many, options, block) - add_relationship(name, hash) + create_relationship(relationship_name, :has_many, options, block) end def has_one(relationship_name, options = {}, &block) - name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :has_one, options, block) - add_relationship(name, hash) + create_relationship(relationship_name, :has_one, options, block) end def belongs_to(relationship_name, options = {}, &block) - name = relationship_name.to_sym - hash = create_relationship_hash(relationship_name, :belongs_to, options, block) - add_relationship(name, hash) + create_relationship(relationship_name, :belongs_to, options, block) end - def create_relationship_hash(base_key, relationship_type, options, block) + def create_relationship(base_key, relationship_type, options, block) name = base_key.to_sym if relationship_type == :has_many base_serialization_key = base_key.to_s.singularize @@ -211,7 +206,7 @@ module FastJsonapi base_key_sym = name id_postfix = '_id' end - { + relationship = Relationship.new( key: options[:key] || run_key_transform(base_key), name: name, id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, @@ -220,10 +215,11 @@ module FastJsonapi object_block: block, serializer: compute_serializer_name(options[:serializer] || base_key_sym), relationship_type: relationship_type, - cached: options[:cached] || false, + cached: options[:cached], polymorphic: fetch_polymorphic_option(options), conditional_proc: options[:if] - } + ) + add_relationship(relationship) end def compute_serializer_name(serializer_key) @@ -255,8 +251,8 @@ module FastJsonapi parse_include_item(include_item).each do |parsed_include| relationship_to_include = klass.relationships_to_serialize[parsed_include] raise ArgumentError, "#{parsed_include} is not specified as a relationship on #{klass.name}" unless relationship_to_include - raise NotImplementedError if relationship_to_include[:polymorphic].is_a?(Hash) - klass = relationship_to_include[:serializer].to_s.constantize + raise NotImplementedError if relationship_to_include.polymorphic.is_a?(Hash) + klass = relationship_to_include.serializer.to_s.constantize end end end diff --git a/lib/fast_jsonapi/relationship.rb b/lib/fast_jsonapi/relationship.rb new file mode 100644 index 0000000..a560442 --- /dev/null +++ b/lib/fast_jsonapi/relationship.rb @@ -0,0 +1,99 @@ +module FastJsonapi + class Relationship + attr_reader :key, :name, :id_method_name, :record_type, :object_method_name, :object_block, :serializer, :relationship_type, :cached, :polymorphic, :conditional_proc + + def initialize( + key:, + name:, + id_method_name:, + record_type:, + object_method_name:, + object_block:, + serializer:, + relationship_type:, + cached: false, + polymorphic:, + conditional_proc: + ) + @key = key + @name = name + @id_method_name = id_method_name + @record_type = record_type + @object_method_name = object_method_name + @object_block = object_block + @serializer = serializer + @relationship_type = relationship_type + @cached = cached + @polymorphic = polymorphic + @conditional_proc = conditional_proc + end + + def serialize(record, serialization_params, output_hash) + if include_relationship?(record, serialization_params) + empty_case = relationship_type == :has_many ? [] : nil + output_hash[key] = { + data: ids_hash_from_record_and_relationship(record, serialization_params) || empty_case + } + end + end + + private + + def include_relationship?(record, serialization_params) + if conditional_proc.present? + conditional_proc.call(record, serialization_params) + else + true + end + end + + def ids_hash_from_record_and_relationship(record, params = {}) + return ids_hash( + fetch_id(record, params) + ) unless polymorphic + + return unless associated_object = fetch_associated_object(record, params) + + return associated_object.map do |object| + id_hash_from_record object, polymorphic + end if associated_object.respond_to? :map + + id_hash_from_record associated_object, polymorphic + end + + def id_hash_from_record(record, record_types) + # memoize the record type within the record_types dictionary, then assigning to record_type: + associated_record_type = record_types[record.class] ||= record.class.name.underscore.to_sym + id_hash(record.id, associated_record_type) + end + + def ids_hash(ids) + return ids.map { |id| id_hash(id, record_type) } if ids.respond_to? :map + id_hash(ids, record_type) # ids variable is just a single id here + end + + def id_hash(id, associated_record_type, default_return=false) + if id.present? + { id: id.to_s, type: record_type } + else + default_return ? { id: nil, type: associated_record_type } : nil + end + end + + def fetch_associated_object(record, params) + return object_block.call(record, params) unless object_block.nil? + record.send(object_method_name) + end + + def fetch_id(record, params) + unless object_block.nil? + object = object_block.call(record, params) + + return object.map(&:id) if object.respond_to? :map + return object.try(:id) + end + + record.public_send(id_method_name) + end + end +end \ No newline at end of file diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 284f017..76bcf97 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -34,34 +34,6 @@ module FastJsonapi end end - def ids_hash(ids, record_type) - return ids.map { |id| id_hash(id, record_type) } if ids.respond_to? :map - id_hash(ids, record_type) # ids variable is just a single id here - end - - def id_hash_from_record(record, record_types) - # memoize the record type within the record_types dictionary, then assigning to record_type: - record_type = record_types[record.class] ||= record.class.name.underscore.to_sym - id_hash(record.id, record_type) - end - - def ids_hash_from_record_and_relationship(record, relationship, params = {}) - polymorphic = relationship[:polymorphic] - - return ids_hash( - fetch_id(record, relationship, params), - relationship[:record_type] - ) unless polymorphic - - return unless associated_object = fetch_associated_object(record, relationship, params) - - return associated_object.map do |object| - id_hash_from_record object, polymorphic - end if associated_object.respond_to? :map - - id_hash_from_record associated_object, polymorphic - end - def links_hash(record, params = {}) data_links.each_with_object({}) do |(key, method), link_hash| link_hash[key] = if method.is_a?(Proc) @@ -82,14 +54,7 @@ module FastJsonapi relationships = relationships_to_serialize if relationships.nil? relationships.each_with_object({}) do |(_k, relationship), hash| - conditional_proc = relationship[:conditional_proc] - if conditional_proc.blank? || conditional_proc.call(record, params) - name = relationship[:key] - empty_case = relationship[:relationship_type] == :has_many ? [] : nil - hash[name] = { - data: ids_hash_from_record_and_relationship(record, relationship, params) || empty_case - } - end + relationship.serialize(record, params, hash) end end @@ -146,12 +111,12 @@ module FastJsonapi items = parse_include_item(include_item) items.each do |item| next unless relationships_to_serialize && relationships_to_serialize[item] - conditional_proc = relationships_to_serialize[item][:conditional_proc] + conditional_proc = relationships_to_serialize[item].conditional_proc next if conditional_proc && !conditional_proc.call(record, params) - raise NotImplementedError if @relationships_to_serialize[item][:polymorphic].is_a?(Hash) - record_type = @relationships_to_serialize[item][:record_type] - serializer = @relationships_to_serialize[item][:serializer].to_s.constantize - relationship_type = @relationships_to_serialize[item][:relationship_type] + raise NotImplementedError if @relationships_to_serialize[item].polymorphic.is_a?(Hash) + record_type = @relationships_to_serialize[item].record_type + serializer = @relationships_to_serialize[item].serializer.to_s.constantize + relationship_type = @relationships_to_serialize[item].relationship_type included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) next if included_objects.blank? @@ -174,19 +139,8 @@ module FastJsonapi end def fetch_associated_object(record, relationship, params) - return relationship[:object_block].call(record, params) unless relationship[:object_block].nil? - record.send(relationship[:object_method_name]) - end - - def fetch_id(record, relationship, params) - unless relationship[:object_block].nil? - object = relationship[:object_block].call(record, params) - - return object.map(&:id) if object.respond_to? :map - return object.try(:id) - end - - record.public_send(relationship[:id_method_name]) + return relationship.object_block.call(record, params) unless relationship.object_block.nil? + record.send(relationship.object_method_name) end end end diff --git a/spec/shared/examples/object_serializer_class_methods_examples.rb b/spec/shared/examples/object_serializer_class_methods_examples.rb index 616ccf5..1481603 100644 --- a/spec/shared/examples/object_serializer_class_methods_examples.rb +++ b/spec/shared/examples/object_serializer_class_methods_examples.rb @@ -1,6 +1,6 @@ RSpec.shared_examples 'returning correct relationship hash' do |serializer, id_method_name, record_type| it 'returns correct relationship hash' do - expect(relationship).to be_instance_of(Hash) + expect(relationship).to be_instance_of(FastJsonapi::Relationship) expect(relationship.keys).to all(be_instance_of(Symbol)) expect(relationship[:serializer]).to be serializer expect(relationship[:id_method_name]).to be id_method_name From d47b74f71f76fee7e7078eeb19b8f3a510fdc069 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Sat, 30 Jun 2018 17:59:42 -0500 Subject: [PATCH 65/73] all tests are passing, but still need to write tests for relationship class --- spec/lib/serialization_core_spec.rb | 32 +++++++++---------- ...bject_serializer_class_methods_examples.rb | 8 ++--- 2 files changed, 20 insertions(+), 20 deletions(-) diff --git a/spec/lib/serialization_core_spec.rb b/spec/lib/serialization_core_spec.rb index 7285e38..385c36e 100644 --- a/spec/lib/serialization_core_spec.rb +++ b/spec/lib/serialization_core_spec.rb @@ -17,23 +17,23 @@ describe FastJsonapi::ObjectSerializer do expect(result_hash).to be nil end - it 'returns the correct hash when ids_hash_from_record_and_relationship is called for a polymorphic association' do - relationship = { name: :groupees, relationship_type: :has_many, object_method_name: :groupees, polymorphic: {} } - results = GroupSerializer.send :ids_hash_from_record_and_relationship, group, relationship - expect(results).to include({ id: "1", type: :person }, { id: "2", type: :group }) - end + # it 'returns the correct hash when ids_hash_from_record_and_relationship is called for a polymorphic association' do + # relationship = { name: :groupees, relationship_type: :has_many, object_method_name: :groupees, polymorphic: {} } + # results = GroupSerializer.send :ids_hash_from_record_and_relationship, group, relationship + # expect(results).to include({ id: "1", type: :person }, { id: "2", type: :group }) + # end - it 'returns correct hash when ids_hash is called' do - inputs = [{ids: %w(1 2 3), record_type: :movie}, {ids: %w(x y z), record_type: 'person'}] - inputs.each do |hash| - results = MovieSerializer.send(:ids_hash, hash[:ids], hash[:record_type]) - expect(results.map{|h| h[:id]}).to eq hash[:ids] - expect(results[0][:type]).to eq hash[:record_type] - end + # it 'returns correct hash when ids_hash is called' do + # inputs = [{ids: %w(1 2 3), record_type: :movie}, {ids: %w(x y z), record_type: 'person'}] + # inputs.each do |hash| + # results = MovieSerializer.send(:ids_hash, hash[:ids], hash[:record_type]) + # expect(results.map{|h| h[:id]}).to eq hash[:ids] + # expect(results[0][:type]).to eq hash[:record_type] + # end - result = MovieSerializer.send(:ids_hash, [], 'movie') - expect(result).to be_empty - end + # result = MovieSerializer.send(:ids_hash, [], 'movie') + # expect(result).to be_empty + # end it 'returns correct hash when attributes_hash is called' do attributes_hash = MovieSerializer.send(:attributes_hash, movie) @@ -57,7 +57,7 @@ describe FastJsonapi::ObjectSerializer do relationships_hash = MovieSerializer.send(:relationships_hash, movie) relationship_names = relationships_hash.keys.sort relationships_hashes = MovieSerializer.relationships_to_serialize.values - expected_names = relationships_hashes.map{|relationship| relationship[:key]}.sort + expected_names = relationships_hashes.map{|relationship| relationship.key}.sort expect(relationship_names).to eq expected_names end diff --git a/spec/shared/examples/object_serializer_class_methods_examples.rb b/spec/shared/examples/object_serializer_class_methods_examples.rb index 1481603..cffce41 100644 --- a/spec/shared/examples/object_serializer_class_methods_examples.rb +++ b/spec/shared/examples/object_serializer_class_methods_examples.rb @@ -1,10 +1,10 @@ RSpec.shared_examples 'returning correct relationship hash' do |serializer, id_method_name, record_type| it 'returns correct relationship hash' do expect(relationship).to be_instance_of(FastJsonapi::Relationship) - expect(relationship.keys).to all(be_instance_of(Symbol)) - expect(relationship[:serializer]).to be serializer - expect(relationship[:id_method_name]).to be id_method_name - expect(relationship[:record_type]).to be record_type + # expect(relationship.keys).to all(be_instance_of(Symbol)) + expect(relationship.serializer).to be serializer + expect(relationship.id_method_name).to be id_method_name + expect(relationship.record_type).to be record_type end end From 22d412246f56148b4d823d3fbd05dfbb0ec9feab Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Sat, 30 Jun 2018 19:02:15 -0500 Subject: [PATCH 66/73] WIP --- lib/fast_jsonapi/serialization_core.rb | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 76bcf97..6ea3503 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -111,14 +111,15 @@ module FastJsonapi items = parse_include_item(include_item) items.each do |item| next unless relationships_to_serialize && relationships_to_serialize[item] - conditional_proc = relationships_to_serialize[item].conditional_proc + relationship_item = relationships_to_serialize[item] + conditional_proc = relationship_item.conditional_proc next if conditional_proc && !conditional_proc.call(record, params) - raise NotImplementedError if @relationships_to_serialize[item].polymorphic.is_a?(Hash) - record_type = @relationships_to_serialize[item].record_type - serializer = @relationships_to_serialize[item].serializer.to_s.constantize - relationship_type = @relationships_to_serialize[item].relationship_type + raise NotImplementedError if relationship_item.polymorphic.is_a?(Hash) + record_type = relationship_item.record_type + serializer = relationship_item.serializer.to_s.constantize + relationship_type = relationship_item.relationship_type - included_objects = fetch_associated_object(record, @relationships_to_serialize[item], params) + included_objects = fetch_associated_object(record, relationship_item, params) next if included_objects.blank? included_objects = [included_objects] unless relationship_type == :has_many From 6e7d8b7ee014e5f650bca313daf9ddc3993f5d10 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Mon, 2 Jul 2018 10:56:52 -0500 Subject: [PATCH 67/73] make fetch_associated_object a public method on relationship class so it can be called from SerilizationCore class --- lib/fast_jsonapi/relationship.rb | 10 +++++----- lib/fast_jsonapi/serialization_core.rb | 7 +------ 2 files changed, 6 insertions(+), 11 deletions(-) diff --git a/lib/fast_jsonapi/relationship.rb b/lib/fast_jsonapi/relationship.rb index a560442..b3a2d2b 100644 --- a/lib/fast_jsonapi/relationship.rb +++ b/lib/fast_jsonapi/relationship.rb @@ -37,6 +37,11 @@ module FastJsonapi end end + def fetch_associated_object(record, params) + return object_block.call(record, params) unless object_block.nil? + record.send(object_method_name) + end + private def include_relationship?(record, serialization_params) @@ -80,11 +85,6 @@ module FastJsonapi end end - def fetch_associated_object(record, params) - return object_block.call(record, params) unless object_block.nil? - record.send(object_method_name) - end - def fetch_id(record, params) unless object_block.nil? object = object_block.call(record, params) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 6ea3503..9ee8f8e 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -119,7 +119,7 @@ module FastJsonapi serializer = relationship_item.serializer.to_s.constantize relationship_type = relationship_item.relationship_type - included_objects = fetch_associated_object(record, relationship_item, params) + included_objects = relationship_item.fetch_associated_object(record, params) next if included_objects.blank? included_objects = [included_objects] unless relationship_type == :has_many @@ -138,11 +138,6 @@ module FastJsonapi end end end - - def fetch_associated_object(record, relationship, params) - return relationship.object_block.call(record, params) unless relationship.object_block.nil? - record.send(relationship.object_method_name) - end end end end From 30596c448886902ad2eec4d07a81b72525eec1f8 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Mon, 2 Jul 2018 11:04:18 -0500 Subject: [PATCH 68/73] move add_relationship to each class method --- lib/fast_jsonapi/object_serializer.rb | 12 +++++++----- 1 file changed, 7 insertions(+), 5 deletions(-) diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index b00bdfd..49e06e2 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -184,15 +184,18 @@ module FastJsonapi end def has_many(relationship_name, options = {}, &block) - create_relationship(relationship_name, :has_many, options, block) + relationship = create_relationship(relationship_name, :has_many, options, block) + add_relationship(relationship) end def has_one(relationship_name, options = {}, &block) - create_relationship(relationship_name, :has_one, options, block) + relationship = create_relationship(relationship_name, :has_one, options, block) + add_relationship(relationship) end def belongs_to(relationship_name, options = {}, &block) - create_relationship(relationship_name, :belongs_to, options, block) + relationship = create_relationship(relationship_name, :belongs_to, options, block) + add_relationship(relationship) end def create_relationship(base_key, relationship_type, options, block) @@ -206,7 +209,7 @@ module FastJsonapi base_key_sym = name id_postfix = '_id' end - relationship = Relationship.new( + Relationship.new( key: options[:key] || run_key_transform(base_key), name: name, id_method_name: options[:id_method_name] || "#{base_serialization_key}#{id_postfix}".to_sym, @@ -219,7 +222,6 @@ module FastJsonapi polymorphic: fetch_polymorphic_option(options), conditional_proc: options[:if] ) - add_relationship(relationship) end def compute_serializer_name(serializer_key) From f86a8926f56357a37b8e24e12d3a129b5b5a0322 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Mon, 2 Jul 2018 15:43:08 -0500 Subject: [PATCH 69/73] make include_relationship? a public method and use it in get_included_records method --- lib/fast_jsonapi/relationship.rb | 4 ++-- lib/fast_jsonapi/serialization_core.rb | 3 +-- 2 files changed, 3 insertions(+), 4 deletions(-) diff --git a/lib/fast_jsonapi/relationship.rb b/lib/fast_jsonapi/relationship.rb index b3a2d2b..15b381c 100644 --- a/lib/fast_jsonapi/relationship.rb +++ b/lib/fast_jsonapi/relationship.rb @@ -42,8 +42,6 @@ module FastJsonapi record.send(object_method_name) end - private - def include_relationship?(record, serialization_params) if conditional_proc.present? conditional_proc.call(record, serialization_params) @@ -52,6 +50,8 @@ module FastJsonapi end end + private + def ids_hash_from_record_and_relationship(record, params = {}) return ids_hash( fetch_id(record, params) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 9ee8f8e..0b1181e 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -112,8 +112,7 @@ module FastJsonapi items.each do |item| next unless relationships_to_serialize && relationships_to_serialize[item] relationship_item = relationships_to_serialize[item] - conditional_proc = relationship_item.conditional_proc - next if conditional_proc && !conditional_proc.call(record, params) + next unless relationship_item.include_relationship?(record, params) raise NotImplementedError if relationship_item.polymorphic.is_a?(Hash) record_type = relationship_item.record_type serializer = relationship_item.serializer.to_s.constantize From 699630d8125379084bd186d01ec470b1cda6b63a Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Mon, 2 Jul 2018 17:52:39 -0500 Subject: [PATCH 70/73] create link class --- lib/fast_jsonapi/link.rb | 29 ++++++++++++++++++++++++++ lib/fast_jsonapi/object_serializer.rb | 10 +++++++-- lib/fast_jsonapi/serialization_core.rb | 12 ++++------- 3 files changed, 41 insertions(+), 10 deletions(-) create mode 100644 lib/fast_jsonapi/link.rb diff --git a/lib/fast_jsonapi/link.rb b/lib/fast_jsonapi/link.rb new file mode 100644 index 0000000..2b48ba9 --- /dev/null +++ b/lib/fast_jsonapi/link.rb @@ -0,0 +1,29 @@ +module FastJsonapi + class Link + attr_reader :key, :method, :conditional_proc + + def initialize(key:, method:, options: {}) + @key = key + @method = method + @conditional_proc = options[:if] + end + + def serialize(record, serialization_params, output_hash) + if include_link?(record, serialization_params) + output_hash[key] = if method.is_a?(Proc) + method.arity == 1 ? method.call(record) : method.call(record, serialization_params) + else + record.public_send(method) + end + end + end + + def include_link?(record, serialization_params) + if conditional_proc.present? + conditional_proc.call(record, serialization_params) + else + true + end + end + end +end \ No newline at end of file diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 49e06e2..9da9400 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -5,6 +5,7 @@ require 'active_support/concern' require 'active_support/inflector' require 'fast_jsonapi/attribute' require 'fast_jsonapi/relationship' +require 'fast_jsonapi/link' require 'fast_jsonapi/serialization_core' module FastJsonapi @@ -238,11 +239,16 @@ module FastJsonapi {} end - def link(link_name, link_method_name = nil, &block) + def link(link_name, link_method_name = nil, options = {}, &block) self.data_links = {} if self.data_links.nil? link_method_name = link_name if link_method_name.nil? key = run_key_transform(link_name) - self.data_links[key] = block || link_method_name + + self.data_links[key] = Link.new( + key: key, + method: block || link_method_name, + options: options + ) end def validate_includes!(includes) diff --git a/lib/fast_jsonapi/serialization_core.rb b/lib/fast_jsonapi/serialization_core.rb index 0b1181e..11257aa 100644 --- a/lib/fast_jsonapi/serialization_core.rb +++ b/lib/fast_jsonapi/serialization_core.rb @@ -35,18 +35,14 @@ module FastJsonapi end def links_hash(record, params = {}) - data_links.each_with_object({}) do |(key, method), link_hash| - link_hash[key] = if method.is_a?(Proc) - method.arity == 1 ? method.call(record) : method.call(record, params) - else - record.public_send(method) - end + data_links.each_with_object({}) do |(_k, link), hash| + link.serialize(record, params, hash) end end def attributes_hash(record, params = {}) - attributes_to_serialize.each_with_object({}) do |(key, attribute), attr_hash| - attribute.serialize(record, params, attr_hash) + attributes_to_serialize.each_with_object({}) do |(_k, attribute), hash| + attribute.serialize(record, params, hash) end end From 01477e9c5b4d09b0a4d92034fd55a4c54cf0c2c9 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Tue, 3 Jul 2018 15:18:52 -0500 Subject: [PATCH 71/73] fix relationship id_hash method --- lib/fast_jsonapi/relationship.rb | 4 ++-- spec/lib/serialization_core_spec.rb | 18 ------------------ 2 files changed, 2 insertions(+), 20 deletions(-) diff --git a/lib/fast_jsonapi/relationship.rb b/lib/fast_jsonapi/relationship.rb index 15b381c..0b3a101 100644 --- a/lib/fast_jsonapi/relationship.rb +++ b/lib/fast_jsonapi/relationship.rb @@ -77,11 +77,11 @@ module FastJsonapi id_hash(ids, record_type) # ids variable is just a single id here end - def id_hash(id, associated_record_type, default_return=false) + def id_hash(id, record_type, default_return=false) if id.present? { id: id.to_s, type: record_type } else - default_return ? { id: nil, type: associated_record_type } : nil + default_return ? { id: nil, type: record_type } : nil end end diff --git a/spec/lib/serialization_core_spec.rb b/spec/lib/serialization_core_spec.rb index 385c36e..d161e3f 100644 --- a/spec/lib/serialization_core_spec.rb +++ b/spec/lib/serialization_core_spec.rb @@ -17,24 +17,6 @@ describe FastJsonapi::ObjectSerializer do expect(result_hash).to be nil end - # it 'returns the correct hash when ids_hash_from_record_and_relationship is called for a polymorphic association' do - # relationship = { name: :groupees, relationship_type: :has_many, object_method_name: :groupees, polymorphic: {} } - # results = GroupSerializer.send :ids_hash_from_record_and_relationship, group, relationship - # expect(results).to include({ id: "1", type: :person }, { id: "2", type: :group }) - # end - - # it 'returns correct hash when ids_hash is called' do - # inputs = [{ids: %w(1 2 3), record_type: :movie}, {ids: %w(x y z), record_type: 'person'}] - # inputs.each do |hash| - # results = MovieSerializer.send(:ids_hash, hash[:ids], hash[:record_type]) - # expect(results.map{|h| h[:id]}).to eq hash[:ids] - # expect(results[0][:type]).to eq hash[:record_type] - # end - - # result = MovieSerializer.send(:ids_hash, [], 'movie') - # expect(result).to be_empty - # end - it 'returns correct hash when attributes_hash is called' do attributes_hash = MovieSerializer.send(:attributes_hash, movie) attribute_names = attributes_hash.keys.sort From af38b30179b1ac842aec2e88c8aca5a61f0f2411 Mon Sep 17 00:00:00 2001 From: Kyle Reeves Date: Tue, 3 Jul 2018 15:58:22 -0500 Subject: [PATCH 72/73] remove options param from Link class --- lib/fast_jsonapi/link.rb | 23 ++++++----------------- lib/fast_jsonapi/object_serializer.rb | 5 ++--- 2 files changed, 8 insertions(+), 20 deletions(-) diff --git a/lib/fast_jsonapi/link.rb b/lib/fast_jsonapi/link.rb index 2b48ba9..41f84c2 100644 --- a/lib/fast_jsonapi/link.rb +++ b/lib/fast_jsonapi/link.rb @@ -1,29 +1,18 @@ module FastJsonapi class Link - attr_reader :key, :method, :conditional_proc + attr_reader :key, :method - def initialize(key:, method:, options: {}) + def initialize(key:, method:) @key = key @method = method - @conditional_proc = options[:if] end def serialize(record, serialization_params, output_hash) - if include_link?(record, serialization_params) - output_hash[key] = if method.is_a?(Proc) - method.arity == 1 ? method.call(record) : method.call(record, serialization_params) - else - record.public_send(method) - end - end - end - - def include_link?(record, serialization_params) - if conditional_proc.present? - conditional_proc.call(record, serialization_params) + output_hash[key] = if method.is_a?(Proc) + method.arity == 1 ? method.call(record) : method.call(record, serialization_params) else - true + record.public_send(method) end end end -end \ No newline at end of file +end diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 9da9400..7cc3634 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -239,15 +239,14 @@ module FastJsonapi {} end - def link(link_name, link_method_name = nil, options = {}, &block) + def link(link_name, link_method_name = nil, &block) self.data_links = {} if self.data_links.nil? link_method_name = link_name if link_method_name.nil? key = run_key_transform(link_name) self.data_links[key] = Link.new( key: key, - method: block || link_method_name, - options: options + method: block || link_method_name ) end From ecb92f07f57cf8a0a4371f3e7e9afa6c12f48a91 Mon Sep 17 00:00:00 2001 From: Oleksiy Babich Date: Wed, 4 Jul 2018 05:35:06 +0300 Subject: [PATCH 73/73] add is_collection parameter to force corresponding serialization (#239) * add is_collection parameter to force corresponding serialization * add documentation for is_collection purpose, behavior and notes re. default autodetect logic --- README.md | 20 ++++++++++++ lib/fast_jsonapi/object_serializer.rb | 9 ++++-- spec/lib/object_serializer_spec.rb | 46 +++++++++++++++++++++++++++ 3 files changed, 72 insertions(+), 3 deletions(-) diff --git a/README.md b/README.md index af57e7a..f4333d1 100644 --- a/README.md +++ b/README.md @@ -261,6 +261,26 @@ hash = MovieSerializer.new([movie, movie], options).serializable_hash json_string = MovieSerializer.new([movie, movie], options).serialized_json ``` +#### Control Over Collection Serialization + +You can use `is_collection` option to have better control over collection serialization. + +If this option is not provided or `nil` autedetect logic is used to try understand +if provided resource is a single object or collection. + +Autodetect logic is compatible with most DB toolkits (ActiveRecord, Sequel, etc.) but +**cannot** guarantee that single vs collection will be always detected properly. + +```ruby +options[:is_collection] +``` + +was introduced to be able to have precise control this behavior + +- `nil` or not provided: will try to autodetect single vs collection (please, see notes above) +- `true` will always treat input resource as *collection* +- `false` will always treat input resource as *single object* + ### Caching Requires a `cache_key` method be defined on model: diff --git a/lib/fast_jsonapi/object_serializer.rb b/lib/fast_jsonapi/object_serializer.rb index 7cc3634..90771b4 100644 --- a/lib/fast_jsonapi/object_serializer.rb +++ b/lib/fast_jsonapi/object_serializer.rb @@ -28,7 +28,7 @@ module FastJsonapi end def serializable_hash - return hash_for_collection if is_collection?(@resource) + return hash_for_collection if is_collection?(@resource, @is_collection) hash_for_one_record end @@ -75,6 +75,7 @@ module FastJsonapi @known_included_objects = {} @meta = options[:meta] @links = options[:links] + @is_collection = options[:is_collection] @params = options[:params] || {} raise ArgumentError.new("`params` option passed to serializer must be a hash") unless @params.is_a?(Hash) @@ -84,8 +85,10 @@ module FastJsonapi end end - def is_collection?(resource) - resource.respond_to?(:each) && !resource.respond_to?(:each_pair) + def is_collection?(resource, force_is_collection = nil) + return force_is_collection unless force_is_collection.nil? + + resource.respond_to?(:size) && !resource.respond_to?(:each_pair) end class_methods do diff --git a/spec/lib/object_serializer_spec.rb b/spec/lib/object_serializer_spec.rb index ed770af..07cbbef 100644 --- a/spec/lib/object_serializer_spec.rb +++ b/spec/lib/object_serializer_spec.rb @@ -310,6 +310,52 @@ describe FastJsonapi::ObjectSerializer do end end + context 'when is_collection option present' do + subject { MovieSerializer.new(resource, is_collection_options).serializable_hash } + + context 'autodetect' do + let(:is_collection_options) { {} } + + context 'collection if no option present' do + let(:resource) { [movie] } + it { expect(subject[:data]).to be_a(Array) } + end + + context 'single if no option present' do + let(:resource) { movie } + it { expect(subject[:data]).to be_a(Hash) } + end + end + + context 'force is_collection to true' do + let(:is_collection_options) { { is_collection: true } } + + context 'collection will pass' do + let(:resource) { [movie] } + it { expect(subject[:data]).to be_a(Array) } + end + + context 'single will raise error' do + let(:resource) { movie } + it { expect { subject }.to raise_error(NoMethodError, /method(.*)each/) } + end + end + + context 'force is_collection to false' do + let(:is_collection_options) { { is_collection: false } } + + context 'collection will fail without id' do + let(:resource) { [movie] } + it { expect { subject }.to raise_error(FastJsonapi::MandatoryField, /id is a mandatory field/) } + end + + context 'single will pass' do + let(:resource) { movie } + it { expect(subject[:data]).to be_a(Hash) } + end + end + end + context 'when optional attributes are determined by record data' do it 'returns optional attribute when attribute is included' do movie.release_year = 2001