Skip to content

Commit 975793f

Browse files
committed
Simplify jsonapi_class configuration
Adds • JSONAPI::Rails::SerializableClassMapping class Overriding Hash’s lookup can be confusing without creating an descendent class. - Old behavior inferrer.class == Hash Doesn’t make it obvious that there’s custom behavior - New behavior inferrer.class == JSONAPI::Rails::SerializableClassMapping Now it’s obvious where to look for the unusual behavior This setup also allows us to define the default mappings and the lookup behavior in separate configuration options • configuration options for 1. jsonapi_class_mapper 2. jsonapi_class_mappings 3. jsonapi_errors_class_mapper (fallback to jsonapi_class_mapper if nil) 4. jsonapi_errors_class_mappings Removes • configration options for 1. jsonapi_class 2. jsonapi_errors_class
1 parent 14a9421 commit 975793f

File tree

4 files changed

+61
-27
lines changed

4 files changed

+61
-27
lines changed

lib/generators/jsonapi/initializer/templates/initializer.rb

+24-12
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,31 @@
11
JSONAPI::Rails.configure do |config|
2-
# # Set a default serializable class mapping.
3-
# config.jsonapi_class = Hash.new { |h, k|
4-
# names = k.to_s.split('::')
2+
# # Set a default serializable class mapper
3+
# # Can be a Proc or any object that responds to #call(class_name)
4+
# # e.g. MyCustomMapper.call('User::Article') => User::SerializableArticle
5+
# config.jsonapi_class_mapper = -> (class_name) do
6+
# names = class_name.to_s.split('::')
57
# klass = names.pop
6-
# h[k] = [*names, "Serializable#{klass}"].join('::').safe_constantize
8+
# [*names, "Serializable#{klass}"].join('::').safe_constantize
9+
# end
10+
#
11+
# # Set any default serializable class mappings
12+
# config.jsonapi_class_mappings = {
13+
# :Car => SerializableVehicle,
14+
# :Boat => SerializableVehicle
715
# }
816
#
9-
# # Set a default serializable class mapping for errors.
10-
# config.jsonapi_errors_class = Hash.new { |h, k|
11-
# names = k.to_s.split('::')
12-
# klass = names.pop
13-
# h[k] = [*names, "Serializable#{klass}"].join('::').safe_constantize
14-
# }.tap { |h|
15-
# h[:'ActiveModel::Errors'] = JSONAPI::Rails::SerializableActiveModelErrors
16-
# h[:Hash] = JSONAPI::Rails::SerializableErrorHash
17+
# # Set a default serializable class mapper for errors.
18+
# # Can be a Proc or any object that responds to #call(class_name)
19+
# # e.g. MyCustomMapper.call('PORO::Error') => PORO::SerializableError
20+
# # If no jsonapi_errors_class_mapper is configured jsonapi_class_mapper will
21+
# # be used
22+
# config.jsonapi_errors_class_mapper = config.jsonapi_class_mapper.dup
23+
#
24+
# # Set any default serializable class errors mappings
25+
# config.jsonapi_errors_class_mappings = {
26+
# :'MyCustomModule::ErrorObject' => MyCustomModule::SerializableErrorObject,
27+
# :'ActiveModel::Errors' => JSONAPI::Rails::SerializableActiveModelErrors,
28+
# :Hash => JSONAPI::Rails::SerializableErrorHash
1729
# }
1830
#
1931
# # Set a default JSON API object.

lib/jsonapi/rails/configuration.rb

+14-13
Original file line numberDiff line numberDiff line change
@@ -7,22 +7,21 @@ class Configuration < ActiveSupport::InheritableOptions; end
77

88
# @private
99
module Configurable
10-
DEFAULT_JSONAPI_CLASS = Hash.new do |h, k|
11-
names = k.to_s.split('::')
10+
DEFAULT_JSONAPI_CLASS_MAPPER = -> (class_name) do
11+
names = class_name.to_s.split('::')
1212
klass = names.pop
13-
h[k] = [*names, "Serializable#{klass}"].join('::').safe_constantize
14-
end.freeze
13+
[*names, "Serializable#{klass}"].join('::').safe_constantize
14+
end
1515

16-
DEFAULT_JSONAPI_ERRORS_CLASS = DEFAULT_JSONAPI_CLASS.dup.merge!(
17-
'ActiveModel::Errors'.to_sym =>
18-
JSONAPI::Rails::SerializableActiveModelErrors,
19-
'Hash'.to_sym => JSONAPI::Rails::SerializableErrorHash
20-
).freeze
16+
DEFAULT_JSONAPI_CLASS_MAPPINGS = {}.freeze
2117

22-
DEFAULT_JSONAPI_OBJECT = {
23-
version: '1.0'
18+
DEFAULT_JSONAPI_ERROR_CLASS_MAPPINGS = {
19+
:'ActiveModel::Errors' => JSONAPI::Rails::SerializableActiveModelErrors,
20+
:Hash => JSONAPI::Rails::SerializableErrorHash
2421
}.freeze
2522

23+
DEFAULT_JSONAPI_OBJECT = { version: '1.0' }.freeze
24+
2625
DEFAULT_JSONAPI_CACHE = ->() { nil }
2726

2827
DEFAULT_JSONAPI_EXPOSE = lambda {
@@ -42,8 +41,10 @@ module Configurable
4241
DEFAULT_LOGGER = Logger.new(STDERR)
4342

4443
DEFAULT_CONFIG = {
45-
jsonapi_class: DEFAULT_JSONAPI_CLASS,
46-
jsonapi_errors_class: DEFAULT_JSONAPI_ERRORS_CLASS,
44+
jsonapi_class_mapper: DEFAULT_JSONAPI_CLASS_MAPPER,
45+
jsonapi_class_mappings: DEFAULT_JSONAPI_CLASS_MAPPINGS,
46+
jsonapi_errors_class_mapper: nil,
47+
jsonapi_errors_class_mappings: DEFAULT_JSONAPI_ERROR_CLASS_MAPPINGS,
4748
jsonapi_cache: DEFAULT_JSONAPI_CACHE,
4849
jsonapi_expose: DEFAULT_JSONAPI_EXPOSE,
4950
jsonapi_fields: DEFAULT_JSONAPI_FIELDS,

lib/jsonapi/rails/controller/hooks.rb

+10-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
require 'jsonapi/rails/configuration'
2+
require 'jsonapi/rails/serializable_class_mapping'
23

34
module JSONAPI
45
module Rails
@@ -11,14 +12,21 @@ module Hooks
1112
# Overridden by the `class` renderer option.
1213
# @return [Hash{Symbol=>Class}]
1314
def jsonapi_class
14-
JSONAPI::Rails.config[:jsonapi_class].dup
15+
JSONAPI::Rails::SerializableClassMapping.new(
16+
JSONAPI::Rails.config[:jsonapi_class_mapper],
17+
JSONAPI::Rails.config[:jsonapi_class_mappings].dup
18+
)
1519
end
1620

1721
# Hook for serializable class mapping (for errors).
1822
# Overridden by the `class` renderer option.
1923
# @return [Hash{Symbol=>Class}]
2024
def jsonapi_errors_class
21-
JSONAPI::Rails.config[:jsonapi_errors_class].dup
25+
mapper = JSONAPI::Rails.config[:jsonapi_errors_class_mapper] ||
26+
JSONAPI::Rails.config[:jsonapi_class_mapper]
27+
JSONAPI::Rails::SerializableClassMapping.new(
28+
mapper, JSONAPI::Rails.config[:jsonapi_errors_class_mappings].dup
29+
)
2230
end
2331

2432
# Hook for the jsonapi object.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
module JSONAPI
2+
module Rails
3+
# @private
4+
class SerializableClassMapping < Hash
5+
def initialize(mapper, default_mappings = {})
6+
super() do |hash, class_name_sym|
7+
hash[class_name_sym] =
8+
mapper.call(class_name_sym.to_s)
9+
end.merge!(default_mappings)
10+
end
11+
end
12+
end
13+
end

0 commit comments

Comments
 (0)