agi_active_model_serializers 0.10.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (220) hide show
  1. checksums.yaml +7 -0
  2. data/.github/ISSUE_TEMPLATE.md +29 -0
  3. data/.github/PULL_REQUEST_TEMPLATE.md +15 -0
  4. data/.gitignore +35 -0
  5. data/.rubocop.yml +102 -0
  6. data/.simplecov +110 -0
  7. data/.travis.yml +51 -0
  8. data/CHANGELOG.md +612 -0
  9. data/CODE_OF_CONDUCT.md +74 -0
  10. data/CONTRIBUTING.md +105 -0
  11. data/Gemfile +56 -0
  12. data/MIT-LICENSE +22 -0
  13. data/README.md +307 -0
  14. data/Rakefile +103 -0
  15. data/active_model_serializers.gemspec +63 -0
  16. data/appveyor.yml +24 -0
  17. data/bin/bench +171 -0
  18. data/bin/bench_regression +316 -0
  19. data/bin/serve_benchmark +39 -0
  20. data/docs/README.md +41 -0
  21. data/docs/STYLE.md +58 -0
  22. data/docs/general/adapters.md +247 -0
  23. data/docs/general/caching.md +58 -0
  24. data/docs/general/configuration_options.md +169 -0
  25. data/docs/general/deserialization.md +100 -0
  26. data/docs/general/fields.md +31 -0
  27. data/docs/general/getting_started.md +133 -0
  28. data/docs/general/instrumentation.md +40 -0
  29. data/docs/general/key_transforms.md +40 -0
  30. data/docs/general/logging.md +14 -0
  31. data/docs/general/rendering.md +279 -0
  32. data/docs/general/serializers.md +461 -0
  33. data/docs/how-open-source-maintained.jpg +0 -0
  34. data/docs/howto/add_pagination_links.md +138 -0
  35. data/docs/howto/add_relationship_links.md +137 -0
  36. data/docs/howto/add_root_key.md +55 -0
  37. data/docs/howto/grape_integration.md +42 -0
  38. data/docs/howto/outside_controller_use.md +65 -0
  39. data/docs/howto/passing_arbitrary_options.md +27 -0
  40. data/docs/howto/serialize_poro.md +32 -0
  41. data/docs/howto/test.md +154 -0
  42. data/docs/howto/upgrade_from_0_8_to_0_10.md +265 -0
  43. data/docs/integrations/ember-and-json-api.md +144 -0
  44. data/docs/integrations/grape.md +19 -0
  45. data/docs/jsonapi/errors.md +56 -0
  46. data/docs/jsonapi/schema.md +151 -0
  47. data/docs/jsonapi/schema/schema.json +366 -0
  48. data/docs/rfcs/0000-namespace.md +106 -0
  49. data/docs/rfcs/template.md +15 -0
  50. data/lib/action_controller/serialization.rb +66 -0
  51. data/lib/active_model/serializable_resource.rb +11 -0
  52. data/lib/active_model/serializer.rb +231 -0
  53. data/lib/active_model/serializer/adapter.rb +24 -0
  54. data/lib/active_model/serializer/adapter/attributes.rb +15 -0
  55. data/lib/active_model/serializer/adapter/base.rb +18 -0
  56. data/lib/active_model/serializer/adapter/json.rb +15 -0
  57. data/lib/active_model/serializer/adapter/json_api.rb +15 -0
  58. data/lib/active_model/serializer/adapter/null.rb +15 -0
  59. data/lib/active_model/serializer/array_serializer.rb +12 -0
  60. data/lib/active_model/serializer/association.rb +34 -0
  61. data/lib/active_model/serializer/attribute.rb +25 -0
  62. data/lib/active_model/serializer/belongs_to_reflection.rb +7 -0
  63. data/lib/active_model/serializer/collection_reflection.rb +7 -0
  64. data/lib/active_model/serializer/collection_serializer.rb +87 -0
  65. data/lib/active_model/serializer/concerns/associations.rb +102 -0
  66. data/lib/active_model/serializer/concerns/attributes.rb +82 -0
  67. data/lib/active_model/serializer/concerns/caching.rb +292 -0
  68. data/lib/active_model/serializer/concerns/configuration.rb +59 -0
  69. data/lib/active_model/serializer/concerns/links.rb +35 -0
  70. data/lib/active_model/serializer/concerns/meta.rb +29 -0
  71. data/lib/active_model/serializer/concerns/type.rb +25 -0
  72. data/lib/active_model/serializer/error_serializer.rb +14 -0
  73. data/lib/active_model/serializer/errors_serializer.rb +32 -0
  74. data/lib/active_model/serializer/field.rb +90 -0
  75. data/lib/active_model/serializer/fieldset.rb +31 -0
  76. data/lib/active_model/serializer/has_many_reflection.rb +7 -0
  77. data/lib/active_model/serializer/has_one_reflection.rb +7 -0
  78. data/lib/active_model/serializer/lint.rb +150 -0
  79. data/lib/active_model/serializer/null.rb +17 -0
  80. data/lib/active_model/serializer/reflection.rb +163 -0
  81. data/lib/active_model/serializer/singular_reflection.rb +7 -0
  82. data/lib/active_model/serializer/version.rb +5 -0
  83. data/lib/active_model_serializers.rb +53 -0
  84. data/lib/active_model_serializers/adapter.rb +98 -0
  85. data/lib/active_model_serializers/adapter/attributes.rb +13 -0
  86. data/lib/active_model_serializers/adapter/base.rb +83 -0
  87. data/lib/active_model_serializers/adapter/json.rb +21 -0
  88. data/lib/active_model_serializers/adapter/json_api.rb +517 -0
  89. data/lib/active_model_serializers/adapter/json_api/deserialization.rb +213 -0
  90. data/lib/active_model_serializers/adapter/json_api/error.rb +96 -0
  91. data/lib/active_model_serializers/adapter/json_api/jsonapi.rb +49 -0
  92. data/lib/active_model_serializers/adapter/json_api/link.rb +83 -0
  93. data/lib/active_model_serializers/adapter/json_api/meta.rb +37 -0
  94. data/lib/active_model_serializers/adapter/json_api/pagination_links.rb +69 -0
  95. data/lib/active_model_serializers/adapter/json_api/relationship.rb +63 -0
  96. data/lib/active_model_serializers/adapter/json_api/resource_identifier.rb +51 -0
  97. data/lib/active_model_serializers/adapter/null.rb +9 -0
  98. data/lib/active_model_serializers/callbacks.rb +55 -0
  99. data/lib/active_model_serializers/deprecate.rb +54 -0
  100. data/lib/active_model_serializers/deserialization.rb +15 -0
  101. data/lib/active_model_serializers/json_pointer.rb +14 -0
  102. data/lib/active_model_serializers/logging.rb +122 -0
  103. data/lib/active_model_serializers/lookup_chain.rb +80 -0
  104. data/lib/active_model_serializers/model.rb +71 -0
  105. data/lib/active_model_serializers/railtie.rb +48 -0
  106. data/lib/active_model_serializers/register_jsonapi_renderer.rb +78 -0
  107. data/lib/active_model_serializers/serializable_resource.rb +82 -0
  108. data/lib/active_model_serializers/serialization_context.rb +39 -0
  109. data/lib/active_model_serializers/test.rb +7 -0
  110. data/lib/active_model_serializers/test/schema.rb +138 -0
  111. data/lib/active_model_serializers/test/serializer.rb +125 -0
  112. data/lib/generators/rails/USAGE +6 -0
  113. data/lib/generators/rails/resource_override.rb +10 -0
  114. data/lib/generators/rails/serializer_generator.rb +36 -0
  115. data/lib/generators/rails/templates/serializer.rb.erb +15 -0
  116. data/lib/grape/active_model_serializers.rb +16 -0
  117. data/lib/grape/formatters/active_model_serializers.rb +32 -0
  118. data/lib/grape/helpers/active_model_serializers.rb +17 -0
  119. data/test/action_controller/adapter_selector_test.rb +53 -0
  120. data/test/action_controller/explicit_serializer_test.rb +135 -0
  121. data/test/action_controller/json/include_test.rb +246 -0
  122. data/test/action_controller/json_api/deserialization_test.rb +112 -0
  123. data/test/action_controller/json_api/errors_test.rb +40 -0
  124. data/test/action_controller/json_api/fields_test.rb +66 -0
  125. data/test/action_controller/json_api/linked_test.rb +202 -0
  126. data/test/action_controller/json_api/pagination_test.rb +116 -0
  127. data/test/action_controller/json_api/transform_test.rb +189 -0
  128. data/test/action_controller/lookup_proc_test.rb +49 -0
  129. data/test/action_controller/namespace_lookup_test.rb +232 -0
  130. data/test/action_controller/serialization_scope_name_test.rb +229 -0
  131. data/test/action_controller/serialization_test.rb +472 -0
  132. data/test/active_model_serializers/adapter_for_test.rb +208 -0
  133. data/test/active_model_serializers/json_pointer_test.rb +22 -0
  134. data/test/active_model_serializers/logging_test.rb +77 -0
  135. data/test/active_model_serializers/model_test.rb +69 -0
  136. data/test/active_model_serializers/railtie_test_isolated.rb +63 -0
  137. data/test/active_model_serializers/register_jsonapi_renderer_test_isolated.rb +161 -0
  138. data/test/active_model_serializers/serialization_context_test_isolated.rb +71 -0
  139. data/test/active_model_serializers/test/schema_test.rb +131 -0
  140. data/test/active_model_serializers/test/serializer_test.rb +62 -0
  141. data/test/active_record_test.rb +9 -0
  142. data/test/adapter/attributes_test.rb +43 -0
  143. data/test/adapter/deprecation_test.rb +100 -0
  144. data/test/adapter/json/belongs_to_test.rb +45 -0
  145. data/test/adapter/json/collection_test.rb +104 -0
  146. data/test/adapter/json/has_many_test.rb +45 -0
  147. data/test/adapter/json/transform_test.rb +93 -0
  148. data/test/adapter/json_api/belongs_to_test.rb +155 -0
  149. data/test/adapter/json_api/collection_test.rb +96 -0
  150. data/test/adapter/json_api/errors_test.rb +76 -0
  151. data/test/adapter/json_api/fields_test.rb +96 -0
  152. data/test/adapter/json_api/has_many_embed_ids_test.rb +43 -0
  153. data/test/adapter/json_api/has_many_explicit_serializer_test.rb +96 -0
  154. data/test/adapter/json_api/has_many_test.rb +165 -0
  155. data/test/adapter/json_api/has_one_test.rb +80 -0
  156. data/test/adapter/json_api/include_data_if_sideloaded_test.rb +168 -0
  157. data/test/adapter/json_api/json_api_test.rb +33 -0
  158. data/test/adapter/json_api/linked_test.rb +413 -0
  159. data/test/adapter/json_api/links_test.rb +95 -0
  160. data/test/adapter/json_api/pagination_links_test.rb +193 -0
  161. data/test/adapter/json_api/parse_test.rb +137 -0
  162. data/test/adapter/json_api/relationship_test.rb +397 -0
  163. data/test/adapter/json_api/resource_identifier_test.rb +110 -0
  164. data/test/adapter/json_api/resource_meta_test.rb +100 -0
  165. data/test/adapter/json_api/toplevel_jsonapi_test.rb +82 -0
  166. data/test/adapter/json_api/transform_test.rb +512 -0
  167. data/test/adapter/json_api/type_test.rb +61 -0
  168. data/test/adapter/json_test.rb +46 -0
  169. data/test/adapter/null_test.rb +22 -0
  170. data/test/adapter/polymorphic_test.rb +171 -0
  171. data/test/adapter_test.rb +67 -0
  172. data/test/array_serializer_test.rb +22 -0
  173. data/test/benchmark/app.rb +65 -0
  174. data/test/benchmark/benchmarking_support.rb +67 -0
  175. data/test/benchmark/bm_active_record.rb +81 -0
  176. data/test/benchmark/bm_adapter.rb +38 -0
  177. data/test/benchmark/bm_caching.rb +119 -0
  178. data/test/benchmark/bm_lookup_chain.rb +83 -0
  179. data/test/benchmark/bm_transform.rb +45 -0
  180. data/test/benchmark/config.ru +3 -0
  181. data/test/benchmark/controllers.rb +83 -0
  182. data/test/benchmark/fixtures.rb +219 -0
  183. data/test/cache_test.rb +595 -0
  184. data/test/collection_serializer_test.rb +123 -0
  185. data/test/fixtures/active_record.rb +113 -0
  186. data/test/fixtures/poro.rb +232 -0
  187. data/test/generators/scaffold_controller_generator_test.rb +24 -0
  188. data/test/generators/serializer_generator_test.rb +74 -0
  189. data/test/grape_test.rb +178 -0
  190. data/test/lint_test.rb +49 -0
  191. data/test/logger_test.rb +20 -0
  192. data/test/poro_test.rb +9 -0
  193. data/test/serializable_resource_test.rb +79 -0
  194. data/test/serializers/association_macros_test.rb +37 -0
  195. data/test/serializers/associations_test.rb +383 -0
  196. data/test/serializers/attribute_test.rb +153 -0
  197. data/test/serializers/attributes_test.rb +52 -0
  198. data/test/serializers/caching_configuration_test_isolated.rb +170 -0
  199. data/test/serializers/configuration_test.rb +32 -0
  200. data/test/serializers/fieldset_test.rb +14 -0
  201. data/test/serializers/meta_test.rb +202 -0
  202. data/test/serializers/options_test.rb +32 -0
  203. data/test/serializers/read_attribute_for_serialization_test.rb +79 -0
  204. data/test/serializers/root_test.rb +21 -0
  205. data/test/serializers/serialization_test.rb +55 -0
  206. data/test/serializers/serializer_for_test.rb +136 -0
  207. data/test/serializers/serializer_for_with_namespace_test.rb +88 -0
  208. data/test/support/custom_schemas/active_model_serializers/test/schema_test/my/index.json +6 -0
  209. data/test/support/isolated_unit.rb +82 -0
  210. data/test/support/rails5_shims.rb +53 -0
  211. data/test/support/rails_app.rb +36 -0
  212. data/test/support/schemas/active_model_serializers/test/schema_test/my/index.json +6 -0
  213. data/test/support/schemas/active_model_serializers/test/schema_test/my/show.json +6 -0
  214. data/test/support/schemas/custom/show.json +7 -0
  215. data/test/support/schemas/hyper_schema.json +93 -0
  216. data/test/support/schemas/render_using_json_api.json +43 -0
  217. data/test/support/schemas/simple_json_pointers.json +10 -0
  218. data/test/support/serialization_testing.rb +71 -0
  219. data/test/test_helper.rb +58 -0
  220. metadata +602 -0
@@ -0,0 +1,15 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Adapter
4
+ class Attributes < DelegateClass(ActiveModelSerializers::Adapter::Attributes)
5
+ def initialize(serializer, options = {})
6
+ super(ActiveModelSerializers::Adapter::Attributes.new(serializer, options))
7
+ end
8
+ class << self
9
+ extend ActiveModelSerializers::Deprecate
10
+ deprecate :new, 'ActiveModelSerializers::Adapter::Json.'
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,18 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Adapter
4
+ class Base < DelegateClass(ActiveModelSerializers::Adapter::Base)
5
+ class << self
6
+ extend ActiveModelSerializers::Deprecate
7
+ deprecate :inherited, 'ActiveModelSerializers::Adapter::Base.'
8
+ end
9
+
10
+ # :nocov:
11
+ def initialize(serializer, options = {})
12
+ super(ActiveModelSerializers::Adapter::Base.new(serializer, options))
13
+ end
14
+ # :nocov:
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,15 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Adapter
4
+ class Json < DelegateClass(ActiveModelSerializers::Adapter::Json)
5
+ def initialize(serializer, options = {})
6
+ super(ActiveModelSerializers::Adapter::Json.new(serializer, options))
7
+ end
8
+ class << self
9
+ extend ActiveModelSerializers::Deprecate
10
+ deprecate :new, 'ActiveModelSerializers::Adapter::Json.new'
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,15 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Adapter
4
+ class JsonApi < DelegateClass(ActiveModelSerializers::Adapter::JsonApi)
5
+ def initialize(serializer, options = {})
6
+ super(ActiveModelSerializers::Adapter::JsonApi.new(serializer, options))
7
+ end
8
+ class << self
9
+ extend ActiveModelSerializers::Deprecate
10
+ deprecate :new, 'ActiveModelSerializers::Adapter::JsonApi.new'
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,15 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Adapter
4
+ class Null < DelegateClass(ActiveModelSerializers::Adapter::Null)
5
+ def initialize(serializer, options = {})
6
+ super(ActiveModelSerializers::Adapter::Null.new(serializer, options))
7
+ end
8
+ class << self
9
+ extend ActiveModelSerializers::Deprecate
10
+ deprecate :new, 'ActiveModelSerializers::Adapter::Null.new'
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,12 @@
1
+ require 'active_model/serializer/collection_serializer'
2
+
3
+ module ActiveModel
4
+ class Serializer
5
+ class ArraySerializer < CollectionSerializer
6
+ class << self
7
+ extend ActiveModelSerializers::Deprecate
8
+ deprecate :new, 'ActiveModel::Serializer::CollectionSerializer.'
9
+ end
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,34 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ # This class holds all information about serializer's association.
4
+ #
5
+ # @attr [Symbol] name
6
+ # @attr [Hash{Symbol => Object}] options
7
+ # @attr [block]
8
+ #
9
+ # @example
10
+ # Association.new(:comments, { serializer: CommentSummarySerializer })
11
+ #
12
+ class Association < Field
13
+ # @return [Symbol]
14
+ def key
15
+ options.fetch(:key, name)
16
+ end
17
+
18
+ # @return [ActiveModel::Serializer, nil]
19
+ def serializer
20
+ options[:serializer]
21
+ end
22
+
23
+ # @return [Hash]
24
+ def links
25
+ options.fetch(:links) || {}
26
+ end
27
+
28
+ # @return [Hash, nil]
29
+ def meta
30
+ options[:meta]
31
+ end
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,25 @@
1
+ require 'active_model/serializer/field'
2
+
3
+ module ActiveModel
4
+ class Serializer
5
+ # Holds all the meta-data about an attribute as it was specified in the
6
+ # ActiveModel::Serializer class.
7
+ #
8
+ # @example
9
+ # class PostSerializer < ActiveModel::Serializer
10
+ # attribute :content
11
+ # attribute :name, key: :title
12
+ # attribute :email, key: :author_email, if: :user_logged_in?
13
+ # attribute :preview do
14
+ # truncate(object.content)
15
+ # end
16
+ #
17
+ # def user_logged_in?
18
+ # current_user.logged_in?
19
+ # end
20
+ # end
21
+ #
22
+ class Attribute < Field
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,7 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ # @api private
4
+ class BelongsToReflection < SingularReflection
5
+ end
6
+ end
7
+ end
@@ -0,0 +1,7 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ # @api private
4
+ class CollectionReflection < Reflection
5
+ end
6
+ end
7
+ end
@@ -0,0 +1,87 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ class CollectionSerializer
4
+ include Enumerable
5
+ delegate :each, to: :@serializers
6
+
7
+ attr_reader :object, :root
8
+
9
+ def initialize(resources, options = {})
10
+ @object = resources
11
+ @options = options
12
+ @root = options[:root]
13
+ @serializers = serializers_from_resources
14
+ end
15
+
16
+ def success?
17
+ true
18
+ end
19
+
20
+ # @api private
21
+ def serializable_hash(adapter_options, options, adapter_instance)
22
+ include_directive = ActiveModel::Serializer.include_directive_from_options(adapter_options)
23
+ adapter_options[:cached_attributes] ||= ActiveModel::Serializer.cache_read_multi(self, adapter_instance, include_directive)
24
+ adapter_opts = adapter_options.merge(include_directive: include_directive)
25
+ serializers.map do |serializer|
26
+ serializer.serializable_hash(adapter_opts, options, adapter_instance)
27
+ end
28
+ end
29
+
30
+ # TODO: unify naming of root, json_key, and _type. Right now, a serializer's
31
+ # json_key comes from the root option or the object's model name, by default.
32
+ # But, if a dev defines a custom `json_key` method with an explicit value,
33
+ # we have no simple way to know that it is safe to call that instance method.
34
+ # (which is really a class property at this point, anyhow).
35
+ # rubocop:disable Metrics/CyclomaticComplexity
36
+ # Disabling cop since it's good to highlight the complexity of this method by
37
+ # including all the logic right here.
38
+ def json_key
39
+ return root if root
40
+ # 1. get from options[:serializer] for empty resource collection
41
+ key = object.empty? &&
42
+ (explicit_serializer_class = options[:serializer]) &&
43
+ explicit_serializer_class._type
44
+ # 2. get from first serializer instance in collection
45
+ key ||= (serializer = serializers.first) && serializer.json_key
46
+ # 3. get from collection name, if a named collection
47
+ key ||= object.respond_to?(:name) ? object.name && object.name.underscore : nil
48
+ # 4. key may be nil for empty collection and no serializer option
49
+ key && key.pluralize
50
+ end
51
+ # rubocop:enable Metrics/CyclomaticComplexity
52
+
53
+ def paginated?
54
+ ActiveModelSerializers.config.jsonapi_pagination_links_enabled &&
55
+ object.respond_to?(:current_page) &&
56
+ object.respond_to?(:total_pages) &&
57
+ object.respond_to?(:size)
58
+ end
59
+
60
+ protected
61
+
62
+ attr_reader :serializers, :options
63
+
64
+ private
65
+
66
+ def serializers_from_resources
67
+ serializer_context_class = options.fetch(:serializer_context_class, ActiveModel::Serializer)
68
+ object.map do |resource|
69
+ serializer_from_resource(resource, serializer_context_class, options)
70
+ end
71
+ end
72
+
73
+ def serializer_from_resource(resource, serializer_context_class, options)
74
+ serializer_class = options.fetch(:serializer) do
75
+ serializer_context_class.serializer_for(resource, namespace: options[:namespace])
76
+ end
77
+
78
+ if serializer_class.nil?
79
+ ActiveModelSerializers.logger.debug "No serializer found for resource: #{resource.inspect}"
80
+ throw :no_serializer
81
+ else
82
+ serializer_class.new(resource, options.except(:serializer))
83
+ end
84
+ end
85
+ end
86
+ end
87
+ end
@@ -0,0 +1,102 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ # Defines an association in the object should be rendered.
4
+ #
5
+ # The serializer object should implement the association name
6
+ # as a method which should return an array when invoked. If a method
7
+ # with the association name does not exist, the association name is
8
+ # dispatched to the serialized object.
9
+ #
10
+ module Associations
11
+ extend ActiveSupport::Concern
12
+
13
+ included do
14
+ with_options instance_writer: false, instance_reader: true do |serializer|
15
+ serializer.class_attribute :_reflections
16
+ self._reflections ||= {}
17
+ end
18
+
19
+ extend ActiveSupport::Autoload
20
+ autoload :Association
21
+ autoload :Reflection
22
+ autoload :SingularReflection
23
+ autoload :CollectionReflection
24
+ autoload :BelongsToReflection
25
+ autoload :HasOneReflection
26
+ autoload :HasManyReflection
27
+ end
28
+
29
+ module ClassMethods
30
+ def inherited(base)
31
+ super
32
+ base._reflections = _reflections.dup
33
+ end
34
+
35
+ # @param [Symbol] name of the association
36
+ # @param [Hash<Symbol => any>] options for the reflection
37
+ # @return [void]
38
+ #
39
+ # @example
40
+ # has_many :comments, serializer: CommentSummarySerializer
41
+ #
42
+ def has_many(name, options = {}, &block) # rubocop:disable Style/PredicateName
43
+ associate(HasManyReflection.new(name, options, block))
44
+ end
45
+
46
+ # @param [Symbol] name of the association
47
+ # @param [Hash<Symbol => any>] options for the reflection
48
+ # @return [void]
49
+ #
50
+ # @example
51
+ # belongs_to :author, serializer: AuthorSerializer
52
+ #
53
+ def belongs_to(name, options = {}, &block)
54
+ associate(BelongsToReflection.new(name, options, block))
55
+ end
56
+
57
+ # @param [Symbol] name of the association
58
+ # @param [Hash<Symbol => any>] options for the reflection
59
+ # @return [void]
60
+ #
61
+ # @example
62
+ # has_one :author, serializer: AuthorSerializer
63
+ #
64
+ def has_one(name, options = {}, &block) # rubocop:disable Style/PredicateName
65
+ associate(HasOneReflection.new(name, options, block))
66
+ end
67
+
68
+ private
69
+
70
+ # Add reflection and define {name} accessor.
71
+ # @param [ActiveModel::Serializer::Reflection] reflection
72
+ # @return [void]
73
+ #
74
+ # @api private
75
+ #
76
+ def associate(reflection)
77
+ key = reflection.options[:key] || reflection.name
78
+ self._reflections[key] = reflection
79
+ end
80
+ end
81
+
82
+ # @param [JSONAPI::IncludeDirective] include_directive (defaults to the
83
+ # +default_include_directive+ config value when not provided)
84
+ # @return [Enumerator<Association>]
85
+ #
86
+ def associations(include_directive = ActiveModelSerializers.default_include_directive, include_slice = nil)
87
+ include_slice ||= include_directive
88
+ return unless object
89
+
90
+ Enumerator.new do |y|
91
+ self.class._reflections.values.each do |reflection|
92
+ next if reflection.excluded?(self)
93
+ key = reflection.options.fetch(:key, reflection.name)
94
+ next unless include_directive.key?(key)
95
+
96
+ y.yield reflection.build_association(self, instance_options, include_slice)
97
+ end
98
+ end
99
+ end
100
+ end
101
+ end
102
+ end
@@ -0,0 +1,82 @@
1
+ module ActiveModel
2
+ class Serializer
3
+ module Attributes
4
+ extend ActiveSupport::Concern
5
+
6
+ included do
7
+ with_options instance_writer: false, instance_reader: false do |serializer|
8
+ serializer.class_attribute :_attributes_data # @api private
9
+ self._attributes_data ||= {}
10
+ end
11
+
12
+ extend ActiveSupport::Autoload
13
+ autoload :Attribute
14
+
15
+ # Return the +attributes+ of +object+ as presented
16
+ # by the serializer.
17
+ def attributes(requested_attrs = nil, reload = false)
18
+ @attributes = nil if reload
19
+ @attributes ||= self.class._attributes_data.each_with_object({}) do |(key, attr), hash|
20
+ next if attr.excluded?(self)
21
+ next unless requested_attrs.nil? || requested_attrs.include?(key)
22
+ hash[key] = attr.value(self)
23
+ end
24
+ end
25
+ end
26
+
27
+ module ClassMethods
28
+ def inherited(base)
29
+ super
30
+ base._attributes_data = _attributes_data.dup
31
+ end
32
+
33
+ # @example
34
+ # class AdminAuthorSerializer < ActiveModel::Serializer
35
+ # attributes :id, :name, :recent_edits
36
+ def attributes(*attrs)
37
+ attrs = attrs.first if attrs.first.class == Array
38
+
39
+ attrs.each do |attr|
40
+ attribute(attr)
41
+ end
42
+ end
43
+
44
+ # @example
45
+ # class AdminAuthorSerializer < ActiveModel::Serializer
46
+ # attributes :id, :recent_edits
47
+ # attribute :name, key: :title
48
+ #
49
+ # attribute :full_name do
50
+ # "#{object.first_name} #{object.last_name}"
51
+ # end
52
+ #
53
+ # def recent_edits
54
+ # object.edits.last(5)
55
+ # end
56
+ def attribute(attr, options = {}, &block)
57
+ key = options.fetch(:key, attr)
58
+ _attributes_data[key] = Attribute.new(attr, options, block)
59
+ end
60
+
61
+ # @api private
62
+ # keys of attributes
63
+ # @see Serializer::attribute
64
+ def _attributes
65
+ _attributes_data.keys
66
+ end
67
+
68
+ # @api private
69
+ # maps attribute value to explicit key name
70
+ # @see Serializer::attribute
71
+ # @see FragmentCache#fragment_serializer
72
+ def _attributes_keys
73
+ _attributes_data
74
+ .each_with_object({}) do |(key, attr), hash|
75
+ next if key == attr.name
76
+ hash[attr.name] = { key: key }
77
+ end
78
+ end
79
+ end
80
+ end
81
+ end
82
+ end