Skip to content

Commit

Permalink
Merge records in apply_join
Browse files Browse the repository at this point in the history
  • Loading branch information
lgebhardt committed Jan 25, 2021
1 parent 9d30ea3 commit 58bdf36
Show file tree
Hide file tree
Showing 6 changed files with 85 additions and 26 deletions.
5 changes: 5 additions & 0 deletions lib/jsonapi/active_relation_resource.rb
Original file line number Diff line number Diff line change
Expand Up @@ -338,6 +338,11 @@ def apply_join(records:, relationship:, resource_type:, join_type:, options:)
records = records.joins_left(relation_name)
end
end

if relationship.merge_resource_records
records = records.merge(self.records(options))
end

records
end

Expand Down
4 changes: 2 additions & 2 deletions lib/jsonapi/relationship.rb
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ class Relationship
attr_reader :acts_as_set, :foreign_key, :options, :name,
:class_name, :polymorphic, :always_include_optional_linkage_data,
:parent_resource, :eager_load_on_include, :custom_methods,
:inverse_relationship, :allow_include
:inverse_relationship, :allow_include, :merge_resource_records

attr_writer :allow_include

Expand All @@ -22,7 +22,7 @@ def initialize(name, options = {})
ActiveSupport::Deprecation.warn('Use polymorphic_types instead of polymorphic_relations')
@polymorphic_types ||= options[:polymorphic_relations]
end

@merge_resource_records = options.fetch(:merge_resource_records, options[:relation_name].nil?) == true
@always_include_optional_linkage_data = options.fetch(:always_include_optional_linkage_data, false) == true
@eager_load_on_include = options.fetch(:eager_load_on_include, true) == true
@allow_include = options[:allow_include]
Expand Down
28 changes: 10 additions & 18 deletions test/fixtures/active_record.rb
Original file line number Diff line number Diff line change
Expand Up @@ -1413,7 +1413,7 @@ class PostResource < JSONAPI::Resource

has_one :author, class_name: 'Person'
has_one :section
has_many :tags, acts_as_set: true, inverse_relationship: :posts, eager_load_on_include: false
has_many :tags, acts_as_set: true, inverse_relationship: :posts
has_many :comments, acts_as_set: false, inverse_relationship: :post

# Not needed - just for testing
Expand Down Expand Up @@ -1937,16 +1937,7 @@ class AuthorResource < JSONAPI::Resource
model_name 'Person'
attributes :name

has_many :books, inverse_relationship: :authors, relation_name: -> (options) {
book_admin = options[:context][:book_admin] || options[:context][:current_user].try(:book_admin)

if book_admin
:books
else
:not_banned_books
end
}

has_many :books
has_many :book_comments
end

Expand Down Expand Up @@ -1986,6 +1977,9 @@ class BookResource < JSONAPI::Resource
}

filter :banned, apply: :apply_filter_banned
filter :title, apply: ->(records, value, options) {
records.where('books.title LIKE ?', "#{value[0]}%")
}

class << self
def books
Expand All @@ -1997,10 +1991,9 @@ def not_banned_books
end

def records(options = {})
context = options[:context]
current_user = context ? context[:current_user] : nil
current_user = options.dig(:context, :current_user)

records = _model_class.all
records = super
# Hide the banned books from people who are not book admins
unless current_user && current_user.book_admin
records = records.where(not_banned_books)
Expand All @@ -2009,8 +2002,7 @@ def records(options = {})
end

def apply_filter_banned(records, value, options)
context = options[:context]
current_user = context ? context[:current_user] : nil
current_user = options.dig(:context, :current_user)

# Only book admins might filter for banned books
if current_user && current_user.book_admin
Expand Down Expand Up @@ -2050,7 +2042,7 @@ def approved_comments(approved = true)
end

def records(options = {})
current_user = options[:context][:current_user]
current_user = options.dig(:context, :current_user)
_model_class.for_user(current_user)
end
end
Expand Down Expand Up @@ -2105,7 +2097,7 @@ class PostResource < JSONAPI::Resource

has_one :author, class_name: 'Person', exclude_links: [:self, "related"]
has_one :section, exclude_links: [:self, :related]
has_many :tags, acts_as_set: true, inverse_relationship: :posts, eager_load_on_include: false, exclude_links: :default
has_many :tags, acts_as_set: true, inverse_relationship: :posts, exclude_links: :default
has_many :comments, acts_as_set: false, inverse_relationship: :post, exclude_links: ["self", :related]
end

Expand Down
38 changes: 38 additions & 0 deletions test/integration/book_authorization_test.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
require File.expand_path('../../test_helper', __FILE__)

class BookAuthorizationTest < ActionDispatch::IntegrationTest
def setup
DatabaseCleaner.start
JSONAPI.configuration.json_key_format = :underscored_key
JSONAPI.configuration.route_format = :underscored_route
Api::V2::BookResource.paginator :offset
end

def test_restricted_records_primary
Api::V2::BookResource.paginator :none

# Not a book admin
$test_user = Person.find(1001)
assert_cacheable_jsonapi_get '/api/v2/books?filter[title]=Book%206'
assert_equal 12, json_response['data'].size

# book_admin
$test_user = Person.find(1005)
assert_cacheable_jsonapi_get '/api/v2/books?filter[title]=Book%206'
assert_equal 111, json_response['data'].size
end

def test_restricted_records_related
Api::V2::BookResource.paginator :none

# Not a book admin
$test_user = Person.find(1001)
assert_cacheable_jsonapi_get '/api/v2/authors/1002/books'
assert_equal 1, json_response['data'].size

# book_admin
$test_user = Person.find(1005)
assert_cacheable_jsonapi_get '/api/v2/authors/1002/books'
assert_equal 2, json_response['data'].size
end
end
12 changes: 6 additions & 6 deletions test/unit/active_relation_resource_finder/join_manager_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -218,18 +218,18 @@ def test_add_joins_with_sub_relationship

if (Rails::VERSION::MAJOR == 6 && Rails::VERSION::MINOR >= 1) || Rails::VERSION::MAJOR > 6
sql = 'SELECT "posts".* FROM "posts" INNER JOIN "comments" ON "comments"."post_id" = "posts"."id" LEFT OUTER JOIN "people" author ON author."id" = "comments"."author_id" LEFT OUTER JOIN "comments_tags" ON "comments_tags"."comment_id" = "comments"."id" LEFT OUTER JOIN "tags" ON "tags"."id" = "comments_tags"."tag_id" LEFT OUTER JOIN "comments" "comments_people" ON "comments_people"."author_id" = "people"."id" WHERE "comments"."approved" = ' + db_true + ' AND "author"."special" = ' + db_true
assert_hash_equals({alias: 'author', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:author)))
assert_hash_equals({alias: 'author', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:author)).except!(:join_options))
else
sql = 'SELECT "posts".* FROM "posts" INNER JOIN "comments" ON "comments"."post_id" = "posts"."id" LEFT OUTER JOIN "people" ON "people"."id" = "comments"."author_id" LEFT OUTER JOIN "comments_tags" ON "comments_tags"."comment_id" = "comments"."id" LEFT OUTER JOIN "tags" ON "tags"."id" = "comments_tags"."tag_id" LEFT OUTER JOIN "comments" "comments_people" ON "comments_people"."author_id" = "people"."id" WHERE "comments"."approved" = ' + db_true + ' AND "author"."special" = ' + db_true
assert_hash_equals({alias: 'people', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:author)))
assert_hash_equals({alias: 'people', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:author)).except!(:join_options))
end

assert_equal sql, records.to_sql

assert_hash_equals({alias: 'comments', join_type: :inner}, join_manager.source_join_details)
assert_hash_equals({alias: 'comments', join_type: :inner}, join_manager.join_details_by_relationship(Api::V10::PostResource._relationship(:comments)))
assert_hash_equals({alias: 'tags', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:tags)))
assert_hash_equals({alias: 'comments_people', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::PersonResource._relationship(:comments)))
assert_hash_equals({alias: 'comments', join_type: :inner}, join_manager.source_join_details.except!(:join_options))
assert_hash_equals({alias: 'comments', join_type: :inner}, join_manager.join_details_by_relationship(Api::V10::PostResource._relationship(:comments)).except!(:join_options))
assert_hash_equals({alias: 'tags', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::CommentResource._relationship(:tags)).except!(:join_options))
assert_hash_equals({alias: 'comments_people', join_type: :left}, join_manager.join_details_by_relationship(Api::V10::PersonResource._relationship(:comments)).except!(:join_options))
end

def test_add_joins_with_sub_relationship_and_filters
Expand Down
24 changes: 24 additions & 0 deletions test/unit/resource/relationship_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,30 @@ def self.is_admin(context)
end
end

class TestRelationshipOptionsPostsResource < JSONAPI::Resource
model_name 'Post'
has_one :author, allow_include: :is_admin, merge_resource_records: false
end

class RelationshipTest < ActiveSupport::TestCase
def test_merge_resource_records_enabled_by_default
relationship = JSONAPI::Relationship::ToOne.new(:author)
assert relationship.merge_resource_records
end

def test_merge_resource_records_is_disabled_by_deafult_with_relation_name
relationship = JSONAPI::Relationship::ToOne.new(:author,
relation_name: "foo" )
refute relationship.merge_resource_records
end

def test_merge_resource_records_can_be_disabled
relationship = JSONAPI::Relationship::ToOne.new(:author,
merge_resource_records: false )
refute relationship.merge_resource_records
end
end

class HasOneRelationshipTest < ActiveSupport::TestCase

def test_polymorphic_type
Expand Down

0 comments on commit 58bdf36

Please sign in to comment.