active_model_serializers 0.10.0 → 0.10.9
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/.rubocop.yml +10 -5
- data/.travis.yml +41 -21
- data/CHANGELOG.md +200 -2
- data/CODE_OF_CONDUCT.md +74 -0
- data/Gemfile +25 -4
- data/README.md +166 -28
- data/Rakefile +5 -32
- data/active_model_serializers.gemspec +23 -25
- data/appveyor.yml +10 -6
- data/bin/rubocop +38 -0
- data/docs/README.md +2 -1
- data/docs/general/adapters.md +35 -11
- data/docs/general/caching.md +7 -1
- data/docs/general/configuration_options.md +86 -1
- data/docs/general/deserialization.md +1 -1
- data/docs/general/fields.md +31 -0
- data/docs/general/getting_started.md +1 -1
- data/docs/general/logging.md +7 -0
- data/docs/general/rendering.md +63 -25
- data/docs/general/serializers.md +137 -14
- data/docs/howto/add_pagination_links.md +16 -17
- data/docs/howto/add_relationship_links.md +140 -0
- data/docs/howto/add_root_key.md +11 -0
- data/docs/howto/grape_integration.md +42 -0
- data/docs/howto/outside_controller_use.md +12 -4
- data/docs/howto/passing_arbitrary_options.md +2 -2
- data/docs/howto/serialize_poro.md +46 -5
- data/docs/howto/test.md +2 -0
- data/docs/howto/upgrade_from_0_8_to_0_10.md +265 -0
- data/docs/integrations/ember-and-json-api.md +67 -32
- data/docs/jsonapi/schema.md +1 -1
- data/lib/action_controller/serialization.rb +15 -3
- data/lib/active_model/serializable_resource.rb +2 -0
- data/lib/active_model/serializer/adapter/attributes.rb +2 -0
- data/lib/active_model/serializer/adapter/base.rb +4 -0
- data/lib/active_model/serializer/adapter/json.rb +2 -0
- data/lib/active_model/serializer/adapter/json_api.rb +2 -0
- data/lib/active_model/serializer/adapter/null.rb +2 -0
- data/lib/active_model/serializer/adapter.rb +2 -0
- data/lib/active_model/serializer/array_serializer.rb +10 -5
- data/lib/active_model/serializer/association.rb +64 -10
- data/lib/active_model/serializer/attribute.rb +2 -0
- data/lib/active_model/serializer/belongs_to_reflection.rb +6 -3
- data/lib/active_model/serializer/collection_serializer.rb +39 -13
- data/lib/active_model/serializer/{caching.rb → concerns/caching.rb} +87 -116
- data/lib/active_model/serializer/error_serializer.rb +13 -7
- data/lib/active_model/serializer/errors_serializer.rb +27 -20
- data/lib/active_model/serializer/field.rb +2 -0
- data/lib/active_model/serializer/fieldset.rb +2 -0
- data/lib/active_model/serializer/has_many_reflection.rb +5 -3
- data/lib/active_model/serializer/has_one_reflection.rb +3 -4
- data/lib/active_model/serializer/lazy_association.rb +99 -0
- data/lib/active_model/serializer/link.rb +23 -0
- data/lib/active_model/serializer/lint.rb +136 -130
- data/lib/active_model/serializer/null.rb +2 -0
- data/lib/active_model/serializer/reflection.rb +132 -67
- data/lib/active_model/serializer/version.rb +3 -1
- data/lib/active_model/serializer.rb +308 -82
- data/lib/active_model_serializers/adapter/attributes.rb +5 -66
- data/lib/active_model_serializers/adapter/base.rb +41 -39
- data/lib/active_model_serializers/adapter/json.rb +2 -0
- data/lib/active_model_serializers/adapter/json_api/deserialization.rb +4 -2
- data/lib/active_model_serializers/adapter/json_api/error.rb +2 -0
- data/lib/active_model_serializers/adapter/json_api/jsonapi.rb +2 -0
- data/lib/active_model_serializers/adapter/json_api/link.rb +3 -1
- data/lib/active_model_serializers/adapter/json_api/meta.rb +2 -0
- data/lib/active_model_serializers/adapter/json_api/pagination_links.rb +49 -21
- data/lib/active_model_serializers/adapter/json_api/relationship.rb +77 -23
- data/lib/active_model_serializers/adapter/json_api/resource_identifier.rb +41 -10
- data/lib/active_model_serializers/adapter/json_api.rb +84 -65
- data/lib/active_model_serializers/adapter/null.rb +2 -0
- data/lib/active_model_serializers/adapter.rb +9 -1
- data/lib/active_model_serializers/callbacks.rb +2 -0
- data/lib/active_model_serializers/deprecate.rb +3 -2
- data/lib/active_model_serializers/deserialization.rb +4 -0
- data/lib/active_model_serializers/json_pointer.rb +2 -0
- data/lib/active_model_serializers/logging.rb +2 -0
- data/lib/active_model_serializers/lookup_chain.rb +82 -0
- data/lib/active_model_serializers/model.rb +111 -28
- data/lib/active_model_serializers/railtie.rb +7 -1
- data/lib/active_model_serializers/register_jsonapi_renderer.rb +46 -31
- data/lib/active_model_serializers/serializable_resource.rb +10 -7
- data/lib/active_model_serializers/serialization_context.rb +12 -3
- data/lib/active_model_serializers/test/schema.rb +4 -2
- data/lib/active_model_serializers/test/serializer.rb +2 -0
- data/lib/active_model_serializers/test.rb +2 -0
- data/lib/active_model_serializers.rb +35 -10
- data/lib/generators/rails/resource_override.rb +3 -1
- data/lib/generators/rails/serializer_generator.rb +6 -4
- data/lib/grape/active_model_serializers.rb +9 -5
- data/lib/grape/formatters/active_model_serializers.rb +21 -2
- data/lib/grape/helpers/active_model_serializers.rb +3 -0
- data/lib/tasks/rubocop.rake +55 -0
- data/test/action_controller/adapter_selector_test.rb +16 -5
- data/test/action_controller/explicit_serializer_test.rb +7 -4
- data/test/action_controller/json/include_test.rb +108 -27
- data/test/action_controller/json_api/deserialization_test.rb +3 -1
- data/test/action_controller/json_api/errors_test.rb +10 -9
- data/test/action_controller/json_api/fields_test.rb +68 -0
- data/test/action_controller/json_api/linked_test.rb +31 -24
- data/test/action_controller/json_api/pagination_test.rb +33 -23
- data/test/action_controller/json_api/transform_test.rb +13 -3
- data/test/action_controller/lookup_proc_test.rb +51 -0
- data/test/action_controller/namespace_lookup_test.rb +234 -0
- data/test/action_controller/serialization_scope_name_test.rb +14 -6
- data/test/action_controller/serialization_test.rb +23 -12
- data/test/active_model_serializers/adapter_for_test.rb +2 -0
- data/test/active_model_serializers/json_pointer_test.rb +17 -13
- data/test/active_model_serializers/logging_test.rb +2 -0
- data/test/active_model_serializers/model_test.rb +139 -4
- data/test/active_model_serializers/railtie_test_isolated.rb +14 -7
- data/test/active_model_serializers/register_jsonapi_renderer_test_isolated.rb +163 -0
- data/test/active_model_serializers/serialization_context_test_isolated.rb +25 -10
- data/test/active_model_serializers/test/schema_test.rb +5 -2
- data/test/active_model_serializers/test/serializer_test.rb +2 -0
- data/test/active_record_test.rb +2 -0
- data/test/adapter/attributes_test.rb +42 -0
- data/test/adapter/deprecation_test.rb +2 -0
- data/test/adapter/json/belongs_to_test.rb +2 -0
- data/test/adapter/json/collection_test.rb +16 -0
- data/test/adapter/json/has_many_test.rb +12 -2
- data/test/adapter/json/transform_test.rb +17 -15
- data/test/adapter/json_api/belongs_to_test.rb +2 -0
- data/test/adapter/json_api/collection_test.rb +6 -3
- data/test/adapter/json_api/errors_test.rb +19 -19
- data/test/adapter/json_api/fields_test.rb +14 -3
- data/test/adapter/json_api/has_many_explicit_serializer_test.rb +2 -0
- data/test/adapter/json_api/has_many_test.rb +51 -20
- data/test/adapter/json_api/has_one_test.rb +2 -0
- data/test/adapter/json_api/include_data_if_sideloaded_test.rb +215 -0
- data/test/adapter/json_api/json_api_test.rb +7 -7
- data/test/adapter/json_api/linked_test.rb +35 -12
- data/test/adapter/json_api/links_test.rb +22 -3
- data/test/adapter/json_api/pagination_links_test.rb +55 -13
- data/test/adapter/json_api/parse_test.rb +3 -1
- data/test/adapter/json_api/relationship_test.rb +311 -73
- data/test/adapter/json_api/resource_meta_test.rb +5 -3
- data/test/adapter/json_api/toplevel_jsonapi_test.rb +2 -0
- data/test/adapter/json_api/transform_test.rb +265 -253
- data/test/adapter/json_api/type_test.rb +170 -36
- data/test/adapter/json_test.rb +10 -7
- data/test/adapter/null_test.rb +3 -2
- data/test/adapter/polymorphic_test.rb +54 -5
- data/test/adapter_test.rb +3 -1
- data/test/array_serializer_test.rb +2 -0
- data/test/benchmark/app.rb +3 -1
- data/test/benchmark/benchmarking_support.rb +3 -1
- data/test/benchmark/bm_active_record.rb +83 -0
- data/test/benchmark/bm_adapter.rb +40 -0
- data/test/benchmark/bm_caching.rb +18 -16
- data/test/benchmark/bm_lookup_chain.rb +85 -0
- data/test/benchmark/bm_transform.rb +23 -10
- data/test/benchmark/controllers.rb +18 -17
- data/test/benchmark/fixtures.rb +74 -72
- data/test/cache_test.rb +301 -69
- data/test/collection_serializer_test.rb +33 -14
- data/test/fixtures/active_record.rb +47 -10
- data/test/fixtures/poro.rb +128 -183
- data/test/generators/scaffold_controller_generator_test.rb +2 -0
- data/test/generators/serializer_generator_test.rb +25 -5
- data/test/grape_test.rb +172 -56
- data/test/lint_test.rb +3 -1
- data/test/logger_test.rb +15 -11
- data/test/poro_test.rb +2 -0
- data/test/serializable_resource_test.rb +20 -22
- data/test/serializers/association_macros_test.rb +5 -2
- data/test/serializers/associations_test.rb +274 -49
- data/test/serializers/attribute_test.rb +7 -3
- data/test/serializers/attributes_test.rb +3 -1
- data/test/serializers/caching_configuration_test_isolated.rb +8 -6
- data/test/serializers/configuration_test.rb +2 -0
- data/test/serializers/fieldset_test.rb +3 -1
- data/test/serializers/meta_test.rb +14 -6
- data/test/serializers/options_test.rb +19 -6
- data/test/serializers/read_attribute_for_serialization_test.rb +5 -3
- data/test/serializers/reflection_test.rb +481 -0
- data/test/serializers/root_test.rb +3 -1
- data/test/serializers/serialization_test.rb +4 -2
- data/test/serializers/serializer_for_test.rb +14 -10
- data/test/serializers/serializer_for_with_namespace_test.rb +90 -0
- data/test/support/isolated_unit.rb +11 -4
- data/test/support/rails5_shims.rb +10 -2
- data/test/support/rails_app.rb +4 -9
- data/test/support/serialization_testing.rb +33 -5
- data/test/test_helper.rb +15 -0
- metadata +126 -46
- data/.rubocop_todo.yml +0 -167
- data/docs/ARCHITECTURE.md +0 -126
- data/lib/active_model/serializer/associations.rb +0 -100
- data/lib/active_model/serializer/attributes.rb +0 -82
- data/lib/active_model/serializer/collection_reflection.rb +0 -7
- data/lib/active_model/serializer/configuration.rb +0 -35
- data/lib/active_model/serializer/include_tree.rb +0 -111
- data/lib/active_model/serializer/links.rb +0 -35
- data/lib/active_model/serializer/meta.rb +0 -29
- data/lib/active_model/serializer/singular_reflection.rb +0 -7
- data/lib/active_model/serializer/type.rb +0 -25
- data/lib/active_model_serializers/key_transform.rb +0 -70
- data/test/active_model_serializers/key_transform_test.rb +0 -263
- data/test/adapter/json_api/has_many_embed_ids_test.rb +0 -43
- data/test/adapter/json_api/relationships_test.rb +0 -199
- data/test/adapter/json_api/resource_identifier_test.rb +0 -85
- data/test/include_tree/from_include_args_test.rb +0 -26
- data/test/include_tree/from_string_test.rb +0 -94
- data/test/include_tree/include_args_to_hash_test.rb +0 -64
@@ -17,25 +17,75 @@ To solve this, in Ember, both the adapter and the serializer will need some modi
|
|
17
17
|
|
18
18
|
### Server-Side Changes
|
19
19
|
|
20
|
-
|
21
|
-
|
20
|
+
First, set the adapter type in an initializer file:
|
21
|
+
|
22
|
+
```ruby
|
23
|
+
# config/initializers/active_model_serializers.rb
|
24
|
+
ActiveModelSerializers.config.adapter = :json_api
|
25
|
+
```
|
26
|
+
|
27
|
+
or:
|
28
|
+
|
29
|
+
```ruby
|
30
|
+
# config/initializers/active_model_serializers.rb
|
31
|
+
ActiveModelSerializers.config.adapter = ActiveModelSerializers::Adapter::JsonApi
|
32
|
+
```
|
33
|
+
|
34
|
+
You will also want to set the `key_transform` to `:unaltered` since you will adjust the attributes in your Ember serializer to use underscores instead of dashes later. You could also use `:underscore`, but `:unaltered` is better for performance.
|
35
|
+
|
36
|
+
```ruby
|
37
|
+
# config/initializers/active_model_serializers.rb
|
38
|
+
ActiveModelSerializers.config.key_transform = :unaltered
|
39
|
+
```
|
40
|
+
|
41
|
+
In order to properly handle JSON API responses, we need to register a JSON API renderer, like so:
|
42
|
+
|
43
|
+
```ruby
|
44
|
+
# config/initializers/active_model_serializers.rb
|
45
|
+
ActiveSupport.on_load(:action_controller) do
|
46
|
+
require 'active_model_serializers/register_jsonapi_renderer'
|
47
|
+
end
|
48
|
+
```
|
49
|
+
Rails also requires your controller to tell it that you accept and generate JSONAPI data. To do that, you use `respond_to` in your controller handlers to tell rails you are consuming and returning jsonapi format data. Without this, Rails will refuse to parse the request body into params. You can add `ActionController::MimeResponds` to your application controller to enable this:
|
50
|
+
|
51
|
+
```ruby
|
52
|
+
class ApplicationController < ActionController::API
|
53
|
+
include ActionController::MimeResponds
|
54
|
+
end
|
55
|
+
```
|
56
|
+
Then, in your controller you can tell rails you're accepting and rendering the jsonapi format:
|
22
57
|
```ruby
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
58
|
+
# POST /post
|
59
|
+
def create
|
60
|
+
@post = Post.new(post_params)
|
61
|
+
respond_to do |format|
|
62
|
+
if @post.save
|
63
|
+
format.jsonapi { render jsonapi: @post, status: :created, location: @post }
|
64
|
+
else
|
65
|
+
format.jsonapi { render jsonapi: @post.errors, status: :unprocessable_entity }
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
# Only allow a trusted parameter "white list" through.
|
71
|
+
def post_params
|
72
|
+
ActiveModelSerializers::Deserialization.jsonapi_parse!(params, only: [:title, :body] )
|
73
|
+
end
|
74
|
+
end
|
31
75
|
```
|
32
76
|
|
77
|
+
#### Note:
|
78
|
+
In Rails 5, the "unsafe" method ( `jsonapi_parse!` vs the safe `jsonapi_parse`) throws an `InvalidDocument` exception when the payload does not meet basic criteria for JSON API deserialization.
|
79
|
+
|
80
|
+
|
33
81
|
### Adapter Changes
|
34
82
|
|
35
83
|
```javascript
|
36
84
|
// app/adapters/application.js
|
85
|
+
import Ember from 'ember';
|
37
86
|
import DS from 'ember-data';
|
38
87
|
import ENV from "../config/environment";
|
88
|
+
const { underscore, pluralize } = Ember.String;
|
39
89
|
|
40
90
|
export default DS.JSONAPIAdapter.extend({
|
41
91
|
namespace: 'api',
|
@@ -46,22 +96,10 @@ export default DS.JSONAPIAdapter.extend({
|
|
46
96
|
|
47
97
|
// allows the multiword paths in urls to be underscored
|
48
98
|
pathForType: function(type) {
|
49
|
-
let underscored =
|
50
|
-
return
|
99
|
+
let underscored = underscore(type);
|
100
|
+
return pluralize(underscored);
|
51
101
|
},
|
52
102
|
|
53
|
-
// allows queries to be sent along with a findRecord
|
54
|
-
// hopefully Ember / EmberData will soon have this built in
|
55
|
-
// ember-data issue tracked here:
|
56
|
-
// https://github.com/emberjs/data/issues/3596
|
57
|
-
urlForFindRecord(id, modelName, snapshot) {
|
58
|
-
let url = this._super(...arguments);
|
59
|
-
let query = Ember.get(snapshot, 'adapterOptions.query');
|
60
|
-
if(query) {
|
61
|
-
url += '?' + Ember.$.param(query);
|
62
|
-
}
|
63
|
-
return url;
|
64
|
-
}
|
65
103
|
});
|
66
104
|
```
|
67
105
|
|
@@ -85,22 +123,19 @@ export default DS.JSONAPISerializer.extend({
|
|
85
123
|
|
86
124
|
```
|
87
125
|
|
88
|
-
## Including Nested Resources
|
89
126
|
|
90
|
-
|
91
|
-
The ActiveModelSerializers default for the `include` parameter is to be `nil` meaning that if any associations are defined in your serializer, only the `id` and `type` will be in the `relationships` structure of the JSON API response.
|
92
|
-
For more on `include` usage, see: [The JSON API include examples](./../general/adapters.md#JSON-API)
|
127
|
+
## Including Nested Resources
|
93
128
|
|
94
|
-
|
129
|
+
Ember Data can request related records by using `include`. Below are some examples of how to make Ember Data request the inclusion of related records. For more on `include` usage, see: [The JSON API include examples](./../general/adapters.md#JSON-API)
|
95
130
|
|
96
131
|
```javascript
|
97
|
-
store.findRecord('post', postId, {
|
132
|
+
store.findRecord('post', postId, { include: 'comments' } );
|
98
133
|
```
|
99
|
-
will generate the path
|
134
|
+
which will generate the path /posts/{postId}?include='comments'
|
100
135
|
|
101
136
|
So then in your controller, you'll want to be sure to have something like:
|
102
137
|
```ruby
|
103
|
-
render
|
138
|
+
render jsonapi: @post, include: params[:include]
|
104
139
|
```
|
105
140
|
|
106
141
|
If you want to use `include` on a collection, you'd write something like this:
|
data/docs/jsonapi/schema.md
CHANGED
@@ -28,7 +28,7 @@ Example supported requests
|
|
28
28
|
- Relationships
|
29
29
|
- GET /articles/1/relationships/comments
|
30
30
|
- GET /articles/1/relationships/author
|
31
|
-
- Optional: [Inclusion of related resources](http://jsonapi.org/format/#fetching-includes) `
|
31
|
+
- Optional: [Inclusion of related resources](http://jsonapi.org/format/#fetching-includes) `JSONAPI::IncludeDirective`
|
32
32
|
- GET /articles/1?`include`=comments
|
33
33
|
- GET /articles/1?`include`=comments.author
|
34
34
|
- GET /articles/1?`include`=author,comments.author
|
@@ -1,3 +1,5 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'active_support/core_ext/class/attribute'
|
2
4
|
require 'active_model_serializers/serialization_context'
|
3
5
|
|
@@ -16,19 +18,29 @@ module ActionController
|
|
16
18
|
included do
|
17
19
|
class_attribute :_serialization_scope
|
18
20
|
self._serialization_scope = :current_user
|
21
|
+
|
22
|
+
attr_writer :namespace_for_serializer
|
23
|
+
end
|
24
|
+
|
25
|
+
def namespace_for_serializer
|
26
|
+
@namespace_for_serializer ||= self.class.parent unless self.class.parent == Object
|
19
27
|
end
|
20
28
|
|
21
29
|
def serialization_scope
|
22
|
-
|
23
|
-
|
30
|
+
return unless _serialization_scope && respond_to?(_serialization_scope, true)
|
31
|
+
|
32
|
+
send(_serialization_scope)
|
24
33
|
end
|
25
34
|
|
26
35
|
def get_serializer(resource, options = {})
|
27
|
-
|
36
|
+
unless use_adapter?
|
28
37
|
warn 'ActionController::Serialization#use_adapter? has been removed. '\
|
29
38
|
"Please pass 'adapter: false' or see ActiveSupport::SerializableResource.new"
|
30
39
|
options[:adapter] = false
|
31
40
|
end
|
41
|
+
|
42
|
+
options.fetch(:namespace) { options[:namespace] = namespace_for_serializer }
|
43
|
+
|
32
44
|
serializable_resource = ActiveModelSerializers::SerializableResource.new(resource, options)
|
33
45
|
serializable_resource.serialization_scope ||= options.fetch(:scope) { serialization_scope }
|
34
46
|
serializable_resource.serialization_scope_name = options.fetch(:scope_name) { _serialization_scope }
|
@@ -1,3 +1,5 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveModel
|
2
4
|
class Serializer
|
3
5
|
module Adapter
|
@@ -7,9 +9,11 @@ module ActiveModel
|
|
7
9
|
deprecate :inherited, 'ActiveModelSerializers::Adapter::Base.'
|
8
10
|
end
|
9
11
|
|
12
|
+
# :nocov:
|
10
13
|
def initialize(serializer, options = {})
|
11
14
|
super(ActiveModelSerializers::Adapter::Base.new(serializer, options))
|
12
15
|
end
|
16
|
+
# :nocov:
|
13
17
|
end
|
14
18
|
end
|
15
19
|
end
|
@@ -1,9 +1,14 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'active_model/serializer/collection_serializer'
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
6
|
-
|
4
|
+
|
5
|
+
module ActiveModel
|
6
|
+
class Serializer
|
7
|
+
class ArraySerializer < CollectionSerializer
|
8
|
+
class << self
|
9
|
+
extend ActiveModelSerializers::Deprecate
|
10
|
+
deprecate :new, 'ActiveModel::Serializer::CollectionSerializer.'
|
11
|
+
end
|
7
12
|
end
|
8
13
|
end
|
9
14
|
end
|
@@ -1,19 +1,73 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'active_model/serializer/lazy_association'
|
4
|
+
|
1
5
|
module ActiveModel
|
2
6
|
class Serializer
|
3
|
-
# This class
|
7
|
+
# This class holds all information about serializer's association.
|
4
8
|
#
|
5
|
-
# @
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
9
|
+
# @api private
|
10
|
+
Association = Struct.new(:reflection, :association_options) do
|
11
|
+
attr_reader :lazy_association
|
12
|
+
delegate :object, :include_data?, :virtual_value, :collection?, to: :lazy_association
|
13
|
+
|
14
|
+
def initialize(*)
|
15
|
+
super
|
16
|
+
@lazy_association = LazyAssociation.new(reflection, association_options)
|
17
|
+
end
|
18
|
+
|
19
|
+
# @return [Symbol]
|
20
|
+
delegate :name, to: :reflection
|
21
|
+
|
13
22
|
# @return [Symbol]
|
14
23
|
def key
|
15
|
-
|
24
|
+
reflection_options.fetch(:key, name)
|
25
|
+
end
|
26
|
+
|
27
|
+
# @return [True,False]
|
28
|
+
def key?
|
29
|
+
reflection_options.key?(:key)
|
30
|
+
end
|
31
|
+
|
32
|
+
# @return [Hash]
|
33
|
+
def links
|
34
|
+
reflection_options.fetch(:links) || {}
|
35
|
+
end
|
36
|
+
|
37
|
+
# @return [Hash, nil]
|
38
|
+
# This gets mutated, so cannot use the cached reflection_options
|
39
|
+
def meta
|
40
|
+
reflection.options[:meta]
|
41
|
+
end
|
42
|
+
|
43
|
+
def belongs_to?
|
44
|
+
reflection.foreign_key_on == :self
|
45
|
+
end
|
46
|
+
|
47
|
+
def polymorphic?
|
48
|
+
true == reflection_options[:polymorphic]
|
49
|
+
end
|
50
|
+
|
51
|
+
# @api private
|
52
|
+
def serializable_hash(adapter_options, adapter_instance)
|
53
|
+
association_serializer = lazy_association.serializer
|
54
|
+
return virtual_value if virtual_value
|
55
|
+
association_object = association_serializer && association_serializer.object
|
56
|
+
return unless association_object
|
57
|
+
|
58
|
+
serialization = association_serializer.serializable_hash(adapter_options, {}, adapter_instance)
|
59
|
+
|
60
|
+
if polymorphic? && serialization
|
61
|
+
polymorphic_type = association_object.class.name.underscore
|
62
|
+
serialization = { type: polymorphic_type, polymorphic_type.to_sym => serialization }
|
63
|
+
end
|
64
|
+
|
65
|
+
serialization
|
16
66
|
end
|
67
|
+
|
68
|
+
private
|
69
|
+
|
70
|
+
delegate :reflection_options, to: :lazy_association
|
17
71
|
end
|
18
72
|
end
|
19
73
|
end
|
@@ -1,9 +1,12 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveModel
|
2
4
|
class Serializer
|
3
5
|
# @api private
|
4
|
-
class BelongsToReflection <
|
5
|
-
|
6
|
-
|
6
|
+
class BelongsToReflection < Reflection
|
7
|
+
# @api private
|
8
|
+
def foreign_key_on
|
9
|
+
:self
|
7
10
|
end
|
8
11
|
end
|
9
12
|
end
|
@@ -1,7 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
module ActiveModel
|
2
4
|
class Serializer
|
3
5
|
class CollectionSerializer
|
4
|
-
NoSerializerError = Class.new(StandardError)
|
5
6
|
include Enumerable
|
6
7
|
delegate :each, to: :@serializers
|
7
8
|
|
@@ -11,22 +12,22 @@ module ActiveModel
|
|
11
12
|
@object = resources
|
12
13
|
@options = options
|
13
14
|
@root = options[:root]
|
14
|
-
|
15
|
-
@serializers = resources.map do |resource|
|
16
|
-
serializer_class = options.fetch(:serializer) { serializer_context_class.serializer_for(resource) }
|
17
|
-
|
18
|
-
if serializer_class.nil? # rubocop:disable Style/GuardClause
|
19
|
-
fail NoSerializerError, "No serializer found for resource: #{resource.inspect}"
|
20
|
-
else
|
21
|
-
serializer_class.new(resource, options.except(:serializer))
|
22
|
-
end
|
23
|
-
end
|
15
|
+
@serializers = serializers_from_resources
|
24
16
|
end
|
25
17
|
|
26
18
|
def success?
|
27
19
|
true
|
28
20
|
end
|
29
21
|
|
22
|
+
# @api private
|
23
|
+
def serializable_hash(adapter_options, options, adapter_instance)
|
24
|
+
options[:include_directive] ||= ActiveModel::Serializer.include_directive_from_options(adapter_options)
|
25
|
+
options[:cached_attributes] ||= ActiveModel::Serializer.cache_read_multi(self, adapter_instance, options[:include_directive])
|
26
|
+
serializers.map do |serializer|
|
27
|
+
serializer.serializable_hash(adapter_options, options, adapter_instance)
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
30
31
|
# TODO: unify naming of root, json_key, and _type. Right now, a serializer's
|
31
32
|
# json_key comes from the root option or the object's model name, by default.
|
32
33
|
# But, if a dev defines a custom `json_key` method with an explicit value,
|
@@ -46,12 +47,15 @@ module ActiveModel
|
|
46
47
|
# 3. get from collection name, if a named collection
|
47
48
|
key ||= object.respond_to?(:name) ? object.name && object.name.underscore : nil
|
48
49
|
# 4. key may be nil for empty collection and no serializer option
|
49
|
-
key
|
50
|
+
key &&= key.pluralize
|
51
|
+
# 5. fail if the key cannot be determined
|
52
|
+
key || fail(ArgumentError, 'Cannot infer root key from collection type. Please specify the root or each_serializer option, or render a JSON String')
|
50
53
|
end
|
51
54
|
# rubocop:enable Metrics/CyclomaticComplexity
|
52
55
|
|
53
56
|
def paginated?
|
54
|
-
|
57
|
+
ActiveModelSerializers.config.jsonapi_pagination_links_enabled &&
|
58
|
+
object.respond_to?(:current_page) &&
|
55
59
|
object.respond_to?(:total_pages) &&
|
56
60
|
object.respond_to?(:size)
|
57
61
|
end
|
@@ -59,6 +63,28 @@ module ActiveModel
|
|
59
63
|
protected
|
60
64
|
|
61
65
|
attr_reader :serializers, :options
|
66
|
+
|
67
|
+
private
|
68
|
+
|
69
|
+
def serializers_from_resources
|
70
|
+
serializer_context_class = options.fetch(:serializer_context_class, ActiveModel::Serializer)
|
71
|
+
object.map do |resource|
|
72
|
+
serializer_from_resource(resource, serializer_context_class, options)
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
def serializer_from_resource(resource, serializer_context_class, options)
|
77
|
+
serializer_class = options.fetch(:serializer) do
|
78
|
+
serializer_context_class.serializer_for(resource, namespace: options[:namespace])
|
79
|
+
end
|
80
|
+
|
81
|
+
if serializer_class.nil?
|
82
|
+
ActiveModelSerializers.logger.debug "No serializer found for resource: #{resource.inspect}"
|
83
|
+
throw :no_serializer
|
84
|
+
else
|
85
|
+
serializer_class.new(resource, options.except(:serializer))
|
86
|
+
end
|
87
|
+
end
|
62
88
|
end
|
63
89
|
end
|
64
90
|
end
|