Skip to content

Commit 5ab6685

Browse files
committed
WIP #1105 fix bug with inclusion and linkage of has_one polymorphic
add tests for link/unlink polymorphic relationships with include them in response add tests for include not linked relationships
1 parent d185ebb commit 5ab6685

File tree

5 files changed

+150
-2
lines changed

5 files changed

+150
-2
lines changed

lib/jsonapi/relationship.rb

+3
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,9 @@ def self.polymorphic_types(name)
4141
klass.reflect_on_all_associations(:has_many).select{|r| r.options[:as] }.each do |reflection|
4242
(hash[reflection.options[:as]] ||= []) << klass.name.downcase
4343
end
44+
klass.reflect_on_all_associations(:has_one).select{|r| r.options[:as] }.each do |reflection|
45+
(hash[reflection.options[:as]] ||= []) << klass.name.downcase
46+
end
4447
end
4548
end
4649
end

lib/jsonapi/resource.rb

+4-2
Original file line numberDiff line numberDiff line change
@@ -1039,8 +1039,10 @@ def define_relationship_methods(relationship_name, relationship_klass, options)
10391039
def define_foreign_key_setter(relationship)
10401040
if relationship.polymorphic?
10411041
define_on_resource "#{relationship.foreign_key}=" do |v|
1042-
_model.method("#{relationship.foreign_key}=").call(v[:id])
1043-
_model.public_send("#{relationship.polymorphic_type}=", v[:type])
1042+
model_id = v.nil? ? nil : v[:id]
1043+
model_type = v.nil? ? nil : self.class.resource_klass_for(v[:type].to_s)._model_class.to_s
1044+
_model.method("#{relationship.foreign_key}=").call(model_id)
1045+
_model.public_send("#{relationship.polymorphic_type}=", model_type)
10441046
end
10451047
else
10461048
define_on_resource "#{relationship.foreign_key}=" do |value|

test/fixtures/active_record.rb

+49
Original file line numberDiff line numberDiff line change
@@ -356,6 +356,25 @@
356356
t.integer :version
357357
t.timestamps null: false
358358
end
359+
360+
create_table :options, force: true do |t|
361+
t.integer :optionable_id
362+
t.string :optionable_type
363+
t.integer :maintainer_id
364+
t.boolean :enabled, default: false, null: false
365+
t.timestamps null: false
366+
end
367+
368+
create_table :androids, force: true do |t|
369+
t.string :version_name
370+
t.timestamps null: false
371+
end
372+
373+
create_table :maintainers, force: true do |t|
374+
t.string :name
375+
t.timestamps null: false
376+
end
377+
359378
end
360379

361380
### MODELS
@@ -734,6 +753,19 @@ class Widget < ActiveRecord::Base
734753
class Robot < ActiveRecord::Base
735754
end
736755

756+
class Option < ActiveRecord::Base
757+
belongs_to :optionable, polymorphic: true, required: false
758+
belongs_to :maintainer, required: false
759+
end
760+
761+
class Android < ActiveRecord::Base
762+
has_one :option, as: :optionable
763+
end
764+
765+
class Maintainer < ActiveRecord::Base
766+
has_one :maintained_option
767+
end
768+
737769
### CONTROLLERS
738770
class AuthorsController < JSONAPI::ResourceControllerMetal
739771
end
@@ -1033,6 +1065,9 @@ class IndicatorsController < JSONAPI::ResourceController
10331065
class RobotsController < JSONAPI::ResourceController
10341066
end
10351067

1068+
class OptionsController < JSONAPI::ResourceController
1069+
end
1070+
10361071
### RESOURCES
10371072
class BaseResource < JSONAPI::Resource
10381073
abstract
@@ -2207,6 +2242,20 @@ class RobotResource < ::JSONAPI::Resource
22072242
end
22082243
end
22092244

2245+
class OptionResource < JSONAPI::Resource
2246+
attribute :enabled
2247+
has_one :optionable, polymorphic: true, class_name: 'Android'
2248+
has_one :maintainer, class_name: 'Maintainer'
2249+
end
2250+
2251+
class AndroidResource < JSONAPI::Resource
2252+
attribute :version_name
2253+
end
2254+
2255+
class MaintainerResource < JSONAPI::Resource
2256+
attribute :name
2257+
end
2258+
22102259
### PORO Data - don't do this in a production app
22112260
$breed_data = BreedData.new
22122261
$breed_data.add(Breed.new(0, 'persian'))

test/integration/requests/request_test.rb

+93
Original file line numberDiff line numberDiff line change
@@ -1162,4 +1162,97 @@ def test_get_resource_with_belongs_to_relationship_and_changed_primary_key
11621162
assert_equal 'access_cards', included.first['type']
11631163
assert_equal access_card.token, included.first['id']
11641164
end
1165+
1166+
def test_update_option_link_with_optionable_include_optionable
1167+
android = Android.create! version_name: '1.0'
1168+
option = Option.create!
1169+
json_request = {
1170+
data: {
1171+
id: option.id.to_s,
1172+
type: 'options',
1173+
relationships: {
1174+
optionable: {
1175+
data: {
1176+
id: android.id.to_s,
1177+
type: 'androids'
1178+
}
1179+
}
1180+
}
1181+
}
1182+
}
1183+
json_api_headers = {
1184+
'CONTENT_TYPE' => JSONAPI::MEDIA_TYPE,
1185+
'Accept' => JSONAPI::MEDIA_TYPE
1186+
}
1187+
patch "/options/#{option.id}?include=optionable", params: json_request.to_json, headers: json_api_headers
1188+
assert_jsonapi_response 200
1189+
relationship_data = {'id' => android.id.to_s, 'type' => 'androids'}
1190+
assert_equal relationship_data, json_response['data']['relationships']['optionable']['data']
1191+
assert_equal relationship_data, json_response['included'].first.slice('id', 'type')
1192+
assert_equal android, option.reload.optionable
1193+
end
1194+
1195+
def test_update_option_unlink_from_optionable_include_optionable
1196+
android = Android.create! version_name: '1.0'
1197+
option = Option.create! optionable: android
1198+
json_request = {
1199+
data: {
1200+
id: option.id.to_s,
1201+
type: 'options',
1202+
relationships: {
1203+
optionable: {
1204+
data: nil
1205+
}
1206+
}
1207+
}
1208+
}
1209+
json_api_headers = {
1210+
'CONTENT_TYPE' => JSONAPI::MEDIA_TYPE,
1211+
'Accept' => JSONAPI::MEDIA_TYPE
1212+
}
1213+
patch "/options/#{option.id}?include=optionable", params: json_request.to_json, headers: json_api_headers
1214+
assert_jsonapi_response 200
1215+
assert_equal true, json_response['data']['relationships']['optionable'].has_key?('data')
1216+
assert_nil json_response['data']['relationships']['optionable']['data']
1217+
assert_equal false, json_response.has_key?('included')
1218+
assert_nil option.reload.optionable
1219+
end
1220+
1221+
def test_fetch_option_linked_with_optionable_include_optionable
1222+
android = Android.create! version_name: '1.0'
1223+
option = Option.create! optionable: android
1224+
assert_cacheable_jsonapi_get "/options/#{option.id}?include=optionable"
1225+
assert_jsonapi_response 200
1226+
relationship_data = {'id' => android.id.to_s, 'type' => 'androids'}
1227+
assert_equal relationship_data, json_response['data']['relationships']['optionable']['data']
1228+
assert_equal relationship_data, json_response['included'].first.slice('id', 'type')
1229+
end
1230+
1231+
def test_fetch_option_not_linked_with_optionable_include_optionable
1232+
option = Option.create!
1233+
assert_cacheable_jsonapi_get "/options/#{option.id}?include=optionable"
1234+
assert_jsonapi_response 200
1235+
assert_equal true, json_response['data']['relationships']['optionable'].has_key?('data')
1236+
assert_nil json_response['data']['relationships']['optionable']['data']
1237+
assert_equal false, json_response.has_key?('included')
1238+
end
1239+
1240+
def test_fetch_option_not_linked_with_maintainer_include_maintainer
1241+
option = Option.create!
1242+
assert_cacheable_jsonapi_get "/options/#{option.id}?include=maintainer"
1243+
assert_jsonapi_response 200
1244+
assert_equal true, json_response['data']['relationships']['maintainer'].has_key?('data')
1245+
assert_nil json_response['data']['relationships']['maintainer']['data']
1246+
assert_equal false, json_response.has_key?('included')
1247+
end
1248+
1249+
def test_fetch_option_linked_with_maintainer_include_maintainer
1250+
maintainer = Maintainer.create! name: 'John Doe'
1251+
option = Option.create! maintainer: maintainer
1252+
assert_cacheable_jsonapi_get "/options/#{option.id}?include=maintainer"
1253+
assert_jsonapi_response 200
1254+
relationship_data = {'id' => maintainer.id.to_s, 'type' => 'maintainers'}
1255+
assert_equal relationship_data, json_response['data']['relationships']['maintainer']['data']
1256+
assert_equal relationship_data, json_response['included'].first.slice('id', 'type')
1257+
end
11651258
end

test/test_helper.rb

+1
Original file line numberDiff line numberDiff line change
@@ -401,6 +401,7 @@ class CatResource < JSONAPI::Resource
401401
jsonapi_resources :widgets, only: [:index]
402402
jsonapi_resources :indicators, only: [:index]
403403
jsonapi_resources :robots, only: [:index]
404+
jsonapi_resources :options, only: [:show, :update]
404405

405406
mount MyEngine::Engine => "/boomshaka", as: :my_engine
406407
mount ApiV2Engine::Engine => "/api_v2", as: :api_v2_engine

0 commit comments

Comments
 (0)