active_model_serializers 0.8.3 → 0.10.0

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 (232) hide show
  1. checksums.yaml +4 -4
  2. data/.github/ISSUE_TEMPLATE.md +29 -0
  3. data/.github/PULL_REQUEST_TEMPLATE.md +15 -0
  4. data/.gitignore +17 -0
  5. data/.rubocop.yml +104 -0
  6. data/.rubocop_todo.yml +167 -0
  7. data/.simplecov +110 -0
  8. data/.travis.yml +39 -24
  9. data/CHANGELOG.md +465 -6
  10. data/CONTRIBUTING.md +105 -0
  11. data/Gemfile +50 -1
  12. data/{MIT-LICENSE.txt → MIT-LICENSE} +3 -2
  13. data/README.md +102 -590
  14. data/Rakefile +93 -8
  15. data/active_model_serializers.gemspec +65 -23
  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/ARCHITECTURE.md +126 -0
  21. data/docs/README.md +40 -0
  22. data/docs/STYLE.md +58 -0
  23. data/docs/general/adapters.md +245 -0
  24. data/docs/general/caching.md +52 -0
  25. data/docs/general/configuration_options.md +100 -0
  26. data/docs/general/deserialization.md +100 -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 +255 -0
  32. data/docs/general/serializers.md +372 -0
  33. data/docs/how-open-source-maintained.jpg +0 -0
  34. data/docs/howto/add_pagination_links.md +139 -0
  35. data/docs/howto/add_root_key.md +51 -0
  36. data/docs/howto/outside_controller_use.md +58 -0
  37. data/docs/howto/passing_arbitrary_options.md +27 -0
  38. data/docs/howto/serialize_poro.md +32 -0
  39. data/docs/howto/test.md +152 -0
  40. data/docs/integrations/ember-and-json-api.md +112 -0
  41. data/docs/integrations/grape.md +19 -0
  42. data/docs/jsonapi/errors.md +56 -0
  43. data/docs/jsonapi/schema/schema.json +366 -0
  44. data/docs/jsonapi/schema.md +151 -0
  45. data/docs/rfcs/0000-namespace.md +106 -0
  46. data/docs/rfcs/template.md +15 -0
  47. data/lib/action_controller/serialization.rb +31 -36
  48. data/lib/active_model/serializable_resource.rb +11 -0
  49. data/lib/active_model/serializer/adapter/attributes.rb +15 -0
  50. data/lib/active_model/serializer/adapter/base.rb +16 -0
  51. data/lib/active_model/serializer/adapter/json.rb +15 -0
  52. data/lib/active_model/serializer/adapter/json_api.rb +15 -0
  53. data/lib/active_model/serializer/adapter/null.rb +15 -0
  54. data/lib/active_model/serializer/adapter.rb +24 -0
  55. data/lib/active_model/serializer/array_serializer.rb +9 -0
  56. data/lib/active_model/serializer/association.rb +19 -0
  57. data/lib/active_model/serializer/associations.rb +87 -220
  58. data/lib/active_model/serializer/attribute.rb +25 -0
  59. data/lib/active_model/serializer/attributes.rb +82 -0
  60. data/lib/active_model/serializer/belongs_to_reflection.rb +10 -0
  61. data/lib/active_model/serializer/caching.rb +333 -0
  62. data/lib/active_model/serializer/collection_reflection.rb +7 -0
  63. data/lib/active_model/serializer/collection_serializer.rb +64 -0
  64. data/lib/active_model/serializer/configuration.rb +35 -0
  65. data/lib/active_model/serializer/error_serializer.rb +10 -0
  66. data/lib/active_model/serializer/errors_serializer.rb +27 -0
  67. data/lib/active_model/serializer/field.rb +90 -0
  68. data/lib/active_model/serializer/fieldset.rb +31 -0
  69. data/lib/active_model/serializer/has_many_reflection.rb +10 -0
  70. data/lib/active_model/serializer/has_one_reflection.rb +10 -0
  71. data/lib/active_model/serializer/include_tree.rb +111 -0
  72. data/lib/active_model/serializer/links.rb +35 -0
  73. data/lib/active_model/serializer/lint.rb +146 -0
  74. data/lib/active_model/serializer/meta.rb +29 -0
  75. data/lib/active_model/serializer/null.rb +17 -0
  76. data/lib/active_model/serializer/reflection.rb +147 -0
  77. data/lib/active_model/serializer/singular_reflection.rb +7 -0
  78. data/lib/active_model/serializer/type.rb +25 -0
  79. data/lib/active_model/{serializers → serializer}/version.rb +1 -1
  80. data/lib/active_model/serializer.rb +158 -481
  81. data/lib/active_model_serializers/adapter/attributes.rb +76 -0
  82. data/lib/active_model_serializers/adapter/base.rb +83 -0
  83. data/lib/active_model_serializers/adapter/json.rb +21 -0
  84. data/lib/active_model_serializers/adapter/json_api/deserialization.rb +213 -0
  85. data/lib/active_model_serializers/adapter/json_api/error.rb +96 -0
  86. data/lib/active_model_serializers/adapter/json_api/jsonapi.rb +49 -0
  87. data/lib/active_model_serializers/adapter/json_api/link.rb +83 -0
  88. data/lib/active_model_serializers/adapter/json_api/meta.rb +37 -0
  89. data/lib/active_model_serializers/adapter/json_api/pagination_links.rb +62 -0
  90. data/lib/active_model_serializers/adapter/json_api/relationship.rb +52 -0
  91. data/lib/active_model_serializers/adapter/json_api/resource_identifier.rb +37 -0
  92. data/lib/active_model_serializers/adapter/json_api.rb +516 -0
  93. data/lib/active_model_serializers/adapter/null.rb +9 -0
  94. data/lib/active_model_serializers/adapter.rb +92 -0
  95. data/lib/active_model_serializers/callbacks.rb +55 -0
  96. data/lib/active_model_serializers/deprecate.rb +55 -0
  97. data/lib/active_model_serializers/deserialization.rb +13 -0
  98. data/lib/active_model_serializers/json_pointer.rb +14 -0
  99. data/lib/active_model_serializers/key_transform.rb +70 -0
  100. data/lib/active_model_serializers/logging.rb +122 -0
  101. data/lib/active_model_serializers/model.rb +49 -0
  102. data/lib/active_model_serializers/railtie.rb +46 -0
  103. data/lib/active_model_serializers/register_jsonapi_renderer.rb +65 -0
  104. data/lib/active_model_serializers/serializable_resource.rb +81 -0
  105. data/lib/active_model_serializers/serialization_context.rb +32 -0
  106. data/lib/active_model_serializers/test/schema.rb +138 -0
  107. data/lib/active_model_serializers/test/serializer.rb +125 -0
  108. data/lib/active_model_serializers/test.rb +7 -0
  109. data/lib/active_model_serializers.rb +32 -89
  110. data/lib/generators/rails/USAGE +6 -0
  111. data/lib/generators/rails/resource_override.rb +10 -0
  112. data/lib/generators/rails/serializer_generator.rb +36 -0
  113. data/lib/generators/rails/templates/serializer.rb.erb +8 -0
  114. data/lib/grape/active_model_serializers.rb +14 -0
  115. data/lib/grape/formatters/active_model_serializers.rb +15 -0
  116. data/lib/grape/helpers/active_model_serializers.rb +16 -0
  117. data/test/action_controller/adapter_selector_test.rb +53 -0
  118. data/test/action_controller/explicit_serializer_test.rb +134 -0
  119. data/test/action_controller/json/include_test.rb +167 -0
  120. data/test/action_controller/json_api/deserialization_test.rb +112 -0
  121. data/test/action_controller/json_api/errors_test.rb +41 -0
  122. data/test/action_controller/json_api/linked_test.rb +197 -0
  123. data/test/action_controller/json_api/pagination_test.rb +116 -0
  124. data/test/action_controller/json_api/transform_test.rb +181 -0
  125. data/test/action_controller/serialization_scope_name_test.rb +229 -0
  126. data/test/action_controller/serialization_test.rb +469 -0
  127. data/test/active_model_serializers/adapter_for_test.rb +208 -0
  128. data/test/active_model_serializers/json_pointer_test.rb +20 -0
  129. data/test/active_model_serializers/key_transform_test.rb +263 -0
  130. data/test/active_model_serializers/logging_test.rb +77 -0
  131. data/test/active_model_serializers/model_test.rb +9 -0
  132. data/test/active_model_serializers/railtie_test_isolated.rb +63 -0
  133. data/test/active_model_serializers/serialization_context_test_isolated.rb +58 -0
  134. data/test/active_model_serializers/test/schema_test.rb +130 -0
  135. data/test/active_model_serializers/test/serializer_test.rb +62 -0
  136. data/test/active_record_test.rb +9 -0
  137. data/test/adapter/deprecation_test.rb +100 -0
  138. data/test/adapter/json/belongs_to_test.rb +45 -0
  139. data/test/adapter/json/collection_test.rb +90 -0
  140. data/test/adapter/json/has_many_test.rb +45 -0
  141. data/test/adapter/json/transform_test.rb +93 -0
  142. data/test/adapter/json_api/belongs_to_test.rb +155 -0
  143. data/test/adapter/json_api/collection_test.rb +95 -0
  144. data/test/adapter/json_api/errors_test.rb +78 -0
  145. data/test/adapter/json_api/fields_test.rb +87 -0
  146. data/test/adapter/json_api/has_many_embed_ids_test.rb +43 -0
  147. data/test/adapter/json_api/has_many_explicit_serializer_test.rb +96 -0
  148. data/test/adapter/json_api/has_many_test.rb +144 -0
  149. data/test/adapter/json_api/has_one_test.rb +80 -0
  150. data/test/adapter/json_api/json_api_test.rb +35 -0
  151. data/test/adapter/json_api/linked_test.rb +392 -0
  152. data/test/adapter/json_api/links_test.rb +93 -0
  153. data/test/adapter/json_api/pagination_links_test.rb +166 -0
  154. data/test/adapter/json_api/parse_test.rb +137 -0
  155. data/test/adapter/json_api/relationship_test.rb +161 -0
  156. data/test/adapter/json_api/relationships_test.rb +199 -0
  157. data/test/adapter/json_api/resource_identifier_test.rb +85 -0
  158. data/test/adapter/json_api/resource_meta_test.rb +100 -0
  159. data/test/adapter/json_api/toplevel_jsonapi_test.rb +82 -0
  160. data/test/adapter/json_api/transform_test.rb +502 -0
  161. data/test/adapter/json_api/type_test.rb +61 -0
  162. data/test/adapter/json_test.rb +45 -0
  163. data/test/adapter/null_test.rb +23 -0
  164. data/test/adapter/polymorphic_test.rb +171 -0
  165. data/test/adapter_test.rb +67 -0
  166. data/test/array_serializer_test.rb +20 -73
  167. data/test/benchmark/app.rb +65 -0
  168. data/test/benchmark/benchmarking_support.rb +67 -0
  169. data/test/benchmark/bm_caching.rb +119 -0
  170. data/test/benchmark/bm_transform.rb +34 -0
  171. data/test/benchmark/config.ru +3 -0
  172. data/test/benchmark/controllers.rb +84 -0
  173. data/test/benchmark/fixtures.rb +219 -0
  174. data/test/cache_test.rb +485 -0
  175. data/test/collection_serializer_test.rb +110 -0
  176. data/test/fixtures/active_record.rb +78 -0
  177. data/test/fixtures/poro.rb +282 -0
  178. data/test/generators/scaffold_controller_generator_test.rb +24 -0
  179. data/test/generators/serializer_generator_test.rb +57 -0
  180. data/test/grape_test.rb +82 -0
  181. data/test/include_tree/from_include_args_test.rb +26 -0
  182. data/test/include_tree/from_string_test.rb +94 -0
  183. data/test/include_tree/include_args_to_hash_test.rb +64 -0
  184. data/test/lint_test.rb +49 -0
  185. data/test/logger_test.rb +18 -0
  186. data/test/poro_test.rb +9 -0
  187. data/test/serializable_resource_test.rb +83 -0
  188. data/test/serializers/association_macros_test.rb +36 -0
  189. data/test/serializers/associations_test.rb +295 -0
  190. data/test/serializers/attribute_test.rb +151 -0
  191. data/test/serializers/attributes_test.rb +52 -0
  192. data/test/serializers/caching_configuration_test_isolated.rb +170 -0
  193. data/test/serializers/configuration_test.rb +32 -0
  194. data/test/serializers/fieldset_test.rb +14 -0
  195. data/test/serializers/meta_test.rb +196 -0
  196. data/test/serializers/options_test.rb +21 -0
  197. data/test/serializers/read_attribute_for_serialization_test.rb +79 -0
  198. data/test/serializers/root_test.rb +21 -0
  199. data/test/serializers/serialization_test.rb +55 -0
  200. data/test/serializers/serializer_for_test.rb +134 -0
  201. data/test/support/custom_schemas/active_model_serializers/test/schema_test/my/index.json +6 -0
  202. data/test/support/isolated_unit.rb +79 -0
  203. data/test/support/rails5_shims.rb +47 -0
  204. data/test/support/rails_app.rb +45 -0
  205. data/test/support/schemas/active_model_serializers/test/schema_test/my/index.json +6 -0
  206. data/test/support/schemas/active_model_serializers/test/schema_test/my/show.json +6 -0
  207. data/test/support/schemas/custom/show.json +7 -0
  208. data/test/support/schemas/hyper_schema.json +93 -0
  209. data/test/support/schemas/render_using_json_api.json +43 -0
  210. data/test/support/schemas/simple_json_pointers.json +10 -0
  211. data/test/support/serialization_testing.rb +53 -0
  212. data/test/test_helper.rb +48 -23
  213. metadata +449 -43
  214. data/DESIGN.textile +0 -586
  215. data/Gemfile.edge +0 -9
  216. data/bench/perf.rb +0 -43
  217. data/cruft.md +0 -19
  218. data/lib/active_model/array_serializer.rb +0 -104
  219. data/lib/active_record/serializer_override.rb +0 -16
  220. data/lib/generators/resource_override.rb +0 -13
  221. data/lib/generators/serializer/USAGE +0 -9
  222. data/lib/generators/serializer/serializer_generator.rb +0 -42
  223. data/lib/generators/serializer/templates/serializer.rb +0 -19
  224. data/test/association_test.rb +0 -592
  225. data/test/caching_test.rb +0 -96
  226. data/test/generators_test.rb +0 -85
  227. data/test/no_serialization_scope_test.rb +0 -34
  228. data/test/serialization_scope_name_test.rb +0 -67
  229. data/test/serialization_test.rb +0 -392
  230. data/test/serializer_support_test.rb +0 -51
  231. data/test/serializer_test.rb +0 -1465
  232. data/test/test_fakes.rb +0 -217
@@ -0,0 +1,32 @@
1
+ module ActiveModelSerializers
2
+ class SerializationContext
3
+ class << self
4
+ attr_writer :url_helpers, :default_url_options
5
+ def url_helpers
6
+ @url_helpers ||= Module.new
7
+ end
8
+
9
+ def default_url_options
10
+ @default_url_options ||= {}
11
+ end
12
+ end
13
+ module UrlHelpers
14
+ def self.included(base)
15
+ base.send(:include, SerializationContext.url_helpers)
16
+ end
17
+
18
+ def default_url_options
19
+ SerializationContext.default_url_options
20
+ end
21
+ end
22
+
23
+ attr_reader :request_url, :query_parameters, :key_transform
24
+
25
+ def initialize(request, options = {})
26
+ @request_url = request.original_url[/\A[^?]+/]
27
+ @query_parameters = request.query_parameters
28
+ @url_helpers = options.delete(:url_helpers) || self.class.url_helpers
29
+ @default_url_options = options.delete(:default_url_options) || self.class.default_url_options
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,138 @@
1
+ module ActiveModelSerializers
2
+ module Test
3
+ module Schema
4
+ # A Minitest Assertion that test the response is valid against a schema.
5
+ # @param schema_path [String] a custom schema path
6
+ # @param message [String] a custom error message
7
+ # @return [Boolean] true when the response is valid
8
+ # @return [Minitest::Assertion] when the response is invalid
9
+ # @example
10
+ # get :index
11
+ # assert_response_schema
12
+ def assert_response_schema(schema_path = nil, message = nil)
13
+ matcher = AssertResponseSchema.new(schema_path, request, response, message)
14
+ assert(matcher.call, matcher.message)
15
+ end
16
+
17
+ def assert_request_schema(schema_path = nil, message = nil)
18
+ matcher = AssertRequestSchema.new(schema_path, request, response, message)
19
+ assert(matcher.call, matcher.message)
20
+ end
21
+
22
+ # May be renamed
23
+ def assert_request_response_schema(schema_path = nil, message = nil)
24
+ assert_request_schema(schema_path, message)
25
+ assert_response_schema(schema_path, message)
26
+ end
27
+
28
+ def assert_schema(payload, schema_path = nil, message = nil)
29
+ matcher = AssertSchema.new(schema_path, request, response, message, payload)
30
+ assert(matcher.call, matcher.message)
31
+ end
32
+
33
+ MissingSchema = Class.new(Minitest::Assertion)
34
+ InvalidSchemaError = Class.new(Minitest::Assertion)
35
+
36
+ class AssertSchema
37
+ attr_reader :schema_path, :request, :response, :message, :payload
38
+
39
+ # Interface may change.
40
+ def initialize(schema_path, request, response, message, payload = nil)
41
+ require_json_schema!
42
+ @request = request
43
+ @response = response
44
+ @payload = payload
45
+ @schema_path = schema_path || schema_path_default
46
+ @message = message
47
+ @document_store = JsonSchema::DocumentStore.new
48
+ add_schema_to_document_store
49
+ end
50
+
51
+ def call
52
+ json_schema.expand_references!(store: document_store)
53
+ status, errors = json_schema.validate(response_body)
54
+ @message = [message, errors.map(&:to_s).to_sentence].compact.join(': ')
55
+ status
56
+ end
57
+
58
+ protected
59
+
60
+ attr_reader :document_store
61
+
62
+ def controller_path
63
+ request.filtered_parameters[:controller]
64
+ end
65
+
66
+ def action
67
+ request.filtered_parameters[:action]
68
+ end
69
+
70
+ def schema_directory
71
+ ActiveModelSerializers.config.schema_path
72
+ end
73
+
74
+ def schema_full_path
75
+ "#{schema_directory}/#{schema_path}"
76
+ end
77
+
78
+ def schema_path_default
79
+ "#{controller_path}/#{action}.json"
80
+ end
81
+
82
+ def schema_data
83
+ load_json_file(schema_full_path)
84
+ end
85
+
86
+ def response_body
87
+ load_json(response.body)
88
+ end
89
+
90
+ def request_params
91
+ request.env['action_dispatch.request.request_parameters']
92
+ end
93
+
94
+ def json_schema
95
+ @json_schema ||= JsonSchema.parse!(schema_data)
96
+ end
97
+
98
+ def add_schema_to_document_store
99
+ Dir.glob("#{schema_directory}/**/*.json").each do |path|
100
+ schema_data = load_json_file(path)
101
+ extra_schema = JsonSchema.parse!(schema_data)
102
+ document_store.add_schema(extra_schema)
103
+ end
104
+ end
105
+
106
+ def load_json(json)
107
+ JSON.parse(json)
108
+ rescue JSON::ParserError => ex
109
+ raise InvalidSchemaError, ex.message
110
+ end
111
+
112
+ def load_json_file(path)
113
+ load_json(File.read(path))
114
+ rescue Errno::ENOENT
115
+ raise MissingSchema, "No Schema file at #{schema_full_path}"
116
+ end
117
+
118
+ def require_json_schema!
119
+ require 'json_schema'
120
+ rescue LoadError
121
+ raise LoadError, "You don't have json_schema installed in your application. Please add it to your Gemfile and run bundle install"
122
+ end
123
+ end
124
+ class AssertResponseSchema < AssertSchema
125
+ def initialize(*)
126
+ super
127
+ @payload = response_body
128
+ end
129
+ end
130
+ class AssertRequestSchema < AssertSchema
131
+ def initialize(*)
132
+ super
133
+ @payload = request_params
134
+ end
135
+ end
136
+ end
137
+ end
138
+ end
@@ -0,0 +1,125 @@
1
+ require 'set'
2
+ module ActiveModelSerializers
3
+ module Test
4
+ module Serializer
5
+ extend ActiveSupport::Concern
6
+
7
+ included do
8
+ setup :setup_serialization_subscriptions
9
+ teardown :teardown_serialization_subscriptions
10
+ end
11
+
12
+ # Asserts that the request was rendered with the appropriate serializers.
13
+ #
14
+ # # assert that the "PostSerializer" serializer was rendered
15
+ # assert_serializer "PostSerializer"
16
+ #
17
+ # # return a custom error message
18
+ # assert_serializer "PostSerializer", "PostSerializer not rendered"
19
+ #
20
+ # # assert that the instance of PostSerializer was rendered
21
+ # assert_serializer PostSerializer
22
+ #
23
+ # # assert that the "PostSerializer" serializer was rendered
24
+ # assert_serializer :post_serializer
25
+ #
26
+ # # assert that the rendered serializer starts with "Post"
27
+ # assert_serializer %r{\APost.+\Z}
28
+ #
29
+ # # assert that no serializer was rendered
30
+ # assert_serializer nil
31
+ #
32
+ def assert_serializer(expectation, message = nil)
33
+ @assert_serializer.expectation = expectation
34
+ @assert_serializer.message = message
35
+ @assert_serializer.response = response
36
+ assert(@assert_serializer.matches?, @assert_serializer.message)
37
+ end
38
+
39
+ class AssertSerializer
40
+ attr_reader :serializers, :message
41
+ attr_accessor :response, :expectation
42
+
43
+ def initialize
44
+ @serializers = Set.new
45
+ @_subscribers = []
46
+ end
47
+
48
+ def message=(message)
49
+ @message = message || "expecting <#{expectation.inspect}> but rendering with <#{serializers.to_a}>"
50
+ end
51
+
52
+ def matches?
53
+ # Force body to be read in case the template is being streamed.
54
+ response.body
55
+
56
+ case expectation
57
+ when a_serializer? then matches_class?
58
+ when Symbol then matches_symbol?
59
+ when String then matches_string?
60
+ when Regexp then matches_regexp?
61
+ when NilClass then matches_nil?
62
+ else fail ArgumentError, 'assert_serializer only accepts a String, Symbol, Regexp, ActiveModel::Serializer, or nil'
63
+ end
64
+ end
65
+
66
+ def subscribe
67
+ @_subscribers << ActiveSupport::Notifications.subscribe(event_name) do |_name, _start, _finish, _id, payload|
68
+ serializer = payload[:serializer].name
69
+ serializers << serializer
70
+ end
71
+ end
72
+
73
+ def unsubscribe
74
+ @_subscribers.each do |subscriber|
75
+ ActiveSupport::Notifications.unsubscribe(subscriber)
76
+ end
77
+ end
78
+
79
+ private
80
+
81
+ def matches_class?
82
+ serializers.include?(expectation.name)
83
+ end
84
+
85
+ def matches_symbol?
86
+ camelize_expectation = expectation.to_s.camelize
87
+ serializers.include?(camelize_expectation)
88
+ end
89
+
90
+ def matches_string?
91
+ !expectation.empty? && serializers.include?(expectation)
92
+ end
93
+
94
+ def matches_regexp?
95
+ serializers.any? do |serializer|
96
+ serializer.match(expectation)
97
+ end
98
+ end
99
+
100
+ def matches_nil?
101
+ serializers.empty?
102
+ end
103
+
104
+ def a_serializer?
105
+ ->(exp) { exp.is_a?(Class) && exp < ActiveModel::Serializer }
106
+ end
107
+
108
+ def event_name
109
+ ::ActiveModelSerializers::Logging::RENDER_EVENT
110
+ end
111
+ end
112
+
113
+ private
114
+
115
+ def setup_serialization_subscriptions
116
+ @assert_serializer = AssertSerializer.new
117
+ @assert_serializer.subscribe
118
+ end
119
+
120
+ def teardown_serialization_subscriptions
121
+ @assert_serializer.unsubscribe
122
+ end
123
+ end
124
+ end
125
+ end
@@ -0,0 +1,7 @@
1
+ module ActiveModelSerializers
2
+ module Test
3
+ extend ActiveSupport::Autoload
4
+ autoload :Serializer
5
+ autoload :Schema
6
+ end
7
+ end
@@ -1,95 +1,38 @@
1
- require "active_support"
2
- require "active_support/core_ext/string/inflections"
3
- require "active_support/notifications"
4
- require "active_model"
5
- require "active_model/array_serializer"
6
- require "active_model/serializer"
7
- require "active_model/serializer/associations"
8
- require "set"
9
-
10
- if defined?(Rails)
11
- module ActiveModel
12
- class Railtie < Rails::Railtie
13
- generators do |app|
14
- app ||= Rails.application # Rails 3.0.x does not yield `app`
15
-
16
- Rails::Generators.configure!(app.config.generators)
17
- Rails::Generators.hidden_namespaces.uniq!
18
- require_relative "generators/resource_override"
19
- end
20
-
21
- initializer "include_routes.active_model_serializer" do |app|
22
- ActiveSupport.on_load(:active_model_serializers) do
23
- include app.routes.url_helpers
24
- end
25
- end
26
-
27
- initializer "caching.active_model_serializer" do |app|
28
- ActiveModel::Serializer.perform_caching = app.config.action_controller.perform_caching
29
- ActiveModel::ArraySerializer.perform_caching = app.config.action_controller.perform_caching
30
-
31
- ActiveModel::Serializer.cache = Rails.cache
32
- ActiveModel::ArraySerializer.cache = Rails.cache
33
- end
34
- end
35
- end
36
- end
37
-
38
- module ActiveModel::SerializerSupport
39
- extend ActiveSupport::Concern
40
-
41
- module ClassMethods #:nodoc:
42
- if "".respond_to?(:safe_constantize)
43
- def active_model_serializer
44
- "#{self.name}Serializer".safe_constantize
45
- end
46
- else
47
- def active_model_serializer
48
- begin
49
- "#{self.name}Serializer".constantize
50
- rescue NameError => e
51
- raise unless e.message =~ /uninitialized constant/
52
- end
53
- end
54
- end
55
- end
56
-
57
- # Returns a model serializer for this object considering its namespace.
58
- def active_model_serializer
59
- self.class.active_model_serializer
60
- end
61
-
62
- alias :read_attribute_for_serialization :send
63
- end
64
-
65
- module ActiveModel::ArraySerializerSupport
66
- def active_model_serializer
67
- ActiveModel::ArraySerializer
1
+ require 'active_model'
2
+ require 'active_support'
3
+ require 'active_support/core_ext/object/with_options'
4
+ require 'active_support/core_ext/string/inflections'
5
+ require 'active_support/json'
6
+ module ActiveModelSerializers
7
+ extend ActiveSupport::Autoload
8
+ autoload :Model
9
+ autoload :Callbacks
10
+ autoload :Deserialization
11
+ autoload :SerializableResource
12
+ autoload :Logging
13
+ autoload :Test
14
+ autoload :Adapter
15
+ autoload :JsonPointer
16
+ autoload :Deprecate
17
+
18
+ class << self; attr_accessor :logger; end
19
+ self.logger = ActiveSupport::TaggedLogging.new(ActiveSupport::Logger.new(STDOUT))
20
+
21
+ def self.config
22
+ ActiveModel::Serializer.config
68
23
  end
69
- end
70
24
 
71
- Array.send(:include, ActiveModel::ArraySerializerSupport)
72
- Set.send(:include, ActiveModel::ArraySerializerSupport)
25
+ # The file name and line number of the caller of the caller of this method.
26
+ def self.location_of_caller
27
+ caller[1] =~ /(.*?):(\d+).*?$/i
28
+ file = Regexp.last_match(1)
29
+ lineno = Regexp.last_match(2).to_i
73
30
 
74
- {
75
- :active_record => 'ActiveRecord::Relation',
76
- :mongoid => 'Mongoid::Criteria'
77
- }.each do |orm, rel_class|
78
- ActiveSupport.on_load(orm) do
79
- include ActiveModel::SerializerSupport
80
- rel_class.constantize.send(:include, ActiveModel::ArraySerializerSupport)
31
+ [file, lineno]
81
32
  end
82
- end
83
-
84
- begin
85
- require 'action_controller'
86
- require 'action_controller/serialization'
87
33
 
88
- ActiveSupport.on_load(:action_controller) do
89
- include ::ActionController::Serialization
90
- end
91
- rescue LoadError => ex
92
- # rails on installed, continuing
34
+ require 'active_model/serializer/version'
35
+ require 'active_model/serializer'
36
+ require 'active_model/serializable_resource'
37
+ require 'active_model_serializers/railtie' if defined?(::Rails)
93
38
  end
94
-
95
- ActiveSupport.run_load_hooks(:active_model_serializers, ActiveModel::Serializer)
@@ -0,0 +1,6 @@
1
+ Description:
2
+ Generates a serializer for the given resource.
3
+
4
+ Example:
5
+ `rails generate serializer Account name created_at`
6
+
@@ -0,0 +1,10 @@
1
+ require 'rails/generators'
2
+ require 'rails/generators/rails/resource/resource_generator'
3
+
4
+ module Rails
5
+ module Generators
6
+ class ResourceGenerator
7
+ hook_for :serializer, default: true, boolean: true
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,36 @@
1
+ module Rails
2
+ module Generators
3
+ class SerializerGenerator < NamedBase
4
+ source_root File.expand_path('../templates', __FILE__)
5
+ check_class_collision :suffix => 'Serializer'
6
+
7
+ argument :attributes, :type => :array, :default => [], :banner => 'field:type field:type'
8
+
9
+ class_option :parent, :type => :string, :desc => 'The parent class for the generated serializer'
10
+
11
+ def create_serializer_file
12
+ template 'serializer.rb.erb', File.join('app/serializers', class_path, "#{file_name}_serializer.rb")
13
+ end
14
+
15
+ private
16
+
17
+ def attributes_names
18
+ [:id] + attributes.reject(&:reference?).map! { |a| a.name.to_sym }
19
+ end
20
+
21
+ def association_names
22
+ attributes.select(&:reference?).map! { |a| a.name.to_sym }
23
+ end
24
+
25
+ def parent_class_name
26
+ if options[:parent]
27
+ options[:parent]
28
+ elsif defined?(::ApplicationSerializer)
29
+ 'ApplicationSerializer'
30
+ else
31
+ 'ActiveModel::Serializer'
32
+ end
33
+ end
34
+ end
35
+ end
36
+ end
@@ -0,0 +1,8 @@
1
+ <% module_namespacing do -%>
2
+ class <%= class_name %>Serializer < <%= parent_class_name %>
3
+ attributes <%= attributes_names.map(&:inspect).join(", ") %>
4
+ <% association_names.each do |attribute| -%>
5
+ has_one :<%= attribute %>
6
+ <% end -%>
7
+ end
8
+ <% end -%>
@@ -0,0 +1,14 @@
1
+ # To add Grape support, require 'grape/active_model_serializers' in the base of your Grape endpoints
2
+ # Then add 'include Grape::ActiveModelSerializers' to enable the formatter and helpers
3
+ require 'active_model_serializers'
4
+ require 'grape/formatters/active_model_serializers'
5
+ require 'grape/helpers/active_model_serializers'
6
+
7
+ module Grape::ActiveModelSerializers
8
+ extend ActiveSupport::Concern
9
+
10
+ included do
11
+ formatter :json, Grape::Formatters::ActiveModelSerializers
12
+ helpers Grape::Helpers::ActiveModelSerializers
13
+ end
14
+ end
@@ -0,0 +1,15 @@
1
+ # A Grape response formatter that can be used as 'formatter :json, Grape::Formatters::ActiveModelSerializers'
2
+ #
3
+ # Serializer options can be passed as a hash from your Grape endpoint using env[:active_model_serializer_options],
4
+ # or better yet user the render helper in Grape::Helpers::ActiveModelSerializers
5
+ module Grape
6
+ module Formatters
7
+ module ActiveModelSerializers
8
+ def self.call(resource, env)
9
+ serializer_options = {}
10
+ serializer_options.merge!(env[:active_model_serializer_options]) if env[:active_model_serializer_options]
11
+ ::ActiveModelSerializers::SerializableResource.new(resource, serializer_options).to_json
12
+ end
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,16 @@
1
+ # Helpers can be included in your Grape endpoint as: helpers Grape::Helpers::ActiveModelSerializers
2
+ module Grape
3
+ module Helpers
4
+ module ActiveModelSerializers
5
+ # A convenience method for passing ActiveModelSerializers serializer options
6
+ #
7
+ # Example: To include relationships in the response: render(post, include: ['comments'])
8
+ #
9
+ # Example: To include pagination meta data: render(posts, meta: { page: posts.page, total_pages: posts.total_pages })
10
+ def render(resource, active_model_serializer_options = {})
11
+ env[:active_model_serializer_options] = active_model_serializer_options
12
+ resource
13
+ end
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,53 @@
1
+ require 'test_helper'
2
+
3
+ module ActionController
4
+ module Serialization
5
+ class AdapterSelectorTest < ActionController::TestCase
6
+ class AdapterSelectorTestController < ActionController::Base
7
+ def render_using_default_adapter
8
+ @profile = Profile.new({ name: 'Name 1', description: 'Description 1', comments: 'Comments 1' })
9
+ render json: @profile
10
+ end
11
+
12
+ def render_using_adapter_override
13
+ @profile = Profile.new({ name: 'Name 1', description: 'Description 1', comments: 'Comments 1' })
14
+ render json: @profile, adapter: :json_api
15
+ end
16
+
17
+ def render_skipping_adapter
18
+ @profile = Profile.new({ name: 'Name 1', description: 'Description 1', comments: 'Comments 1' })
19
+ render json: @profile, adapter: false
20
+ end
21
+ end
22
+
23
+ tests AdapterSelectorTestController
24
+
25
+ def test_render_using_default_adapter
26
+ get :render_using_default_adapter
27
+ assert_equal '{"name":"Name 1","description":"Description 1"}', response.body
28
+ end
29
+
30
+ def test_render_using_adapter_override
31
+ get :render_using_adapter_override
32
+
33
+ expected = {
34
+ data: {
35
+ id: assigns(:profile).id.to_s,
36
+ type: 'profiles',
37
+ attributes: {
38
+ name: 'Name 1',
39
+ description: 'Description 1'
40
+ }
41
+ }
42
+ }
43
+
44
+ assert_equal expected.to_json, response.body
45
+ end
46
+
47
+ def test_render_skipping_adapter
48
+ get :render_skipping_adapter
49
+ assert_equal '{"name":"Name 1","description":"Description 1","comments":"Comments 1"}', response.body
50
+ end
51
+ end
52
+ end
53
+ end