Skip to content

Commit d55f3b3

Browse files
committed
Merge pull request #811 from mateomurphy/scope
Reimplement serialization scope and scope_name
2 parents 3389218 + 232e367 commit d55f3b3

File tree

3 files changed

+99
-4
lines changed

3 files changed

+99
-4
lines changed

lib/action_controller/serialization.rb

Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,16 @@ module Serialization
88

99
ADAPTER_OPTION_KEYS = [:include, :fields, :root, :adapter]
1010

11+
included do
12+
class_attribute :_serialization_scope
13+
self._serialization_scope = :current_user
14+
end
15+
16+
def serialization_scope
17+
send(_serialization_scope) if _serialization_scope &&
18+
respond_to?(_serialization_scope, true)
19+
end
20+
1121
def get_serializer(resource)
1222
@_serializer ||= @_serializer_opts.delete(:serializer)
1323
@_serializer ||= ActiveModel::Serializer.serializer_for(resource)
@@ -29,6 +39,10 @@ def use_adapter?
2939
options.partition { |k, _| ADAPTER_OPTION_KEYS.include? k }.map { |h| Hash[h] }
3040

3141
if use_adapter? && (serializer = get_serializer(resource))
42+
43+
@_serializer_opts[:scope] ||= serialization_scope
44+
@_serializer_opts[:scope_name] = _serialization_scope
45+
3246
# omg hax
3347
object = serializer.new(resource, @_serializer_opts)
3448
adapter = ActiveModel::Serializer::Adapter.create(object, @_adapter_opts)
@@ -38,5 +52,11 @@ def use_adapter?
3852
end
3953
end
4054
end
55+
56+
module ClassMethods
57+
def serialization_scope(scope)
58+
self._serialization_scope = scope
59+
end
60+
end
4161
end
4262
end

lib/active_model/serializer.rb

Lines changed: 12 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -136,13 +136,21 @@ def self.root_name
136136
name.demodulize.underscore.sub(/_serializer$/, '') if name
137137
end
138138

139-
attr_accessor :object, :root, :meta, :meta_key
139+
attr_accessor :object, :root, :meta, :meta_key, :scope
140140

141141
def initialize(object, options = {})
142-
@object = object
143-
@root = options[:root] || (self.class._root ? self.class.root_name : false)
144-
@meta = options[:meta]
142+
@object = object
143+
@root = options[:root] || (self.class._root ? self.class.root_name : false)
144+
@meta = options[:meta]
145145
@meta_key = options[:meta_key]
146+
@scope = options[:scope]
147+
148+
scope_name = options[:scope_name]
149+
if scope_name && !respond_to?(scope_name)
150+
self.class.class_eval do
151+
define_method scope_name, lambda { scope }
152+
end
153+
end
146154
end
147155

148156
def json_key
Lines changed: 67 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
require 'test_helper'
2+
require 'pathname'
3+
4+
class DefaultScopeNameTest < ActionController::TestCase
5+
TestUser = Struct.new(:name, :admin)
6+
7+
class UserSerializer < ActiveModel::Serializer
8+
attributes :admin?
9+
def admin?
10+
current_user.admin
11+
end
12+
end
13+
14+
class UserTestController < ActionController::Base
15+
protect_from_forgery
16+
17+
before_filter { request.format = :json }
18+
19+
def current_user
20+
TestUser.new('Pete', false)
21+
end
22+
23+
def render_new_user
24+
render json: TestUser.new('pete', false), serializer: UserSerializer, adapter: :json_api
25+
end
26+
end
27+
28+
tests UserTestController
29+
30+
def test_default_scope_name
31+
get :render_new_user
32+
assert_equal '{"users":{"admin?":false}}', @response.body
33+
end
34+
end
35+
36+
class SerializationScopeNameTest < ActionController::TestCase
37+
TestUser = Struct.new(:name, :admin)
38+
39+
class AdminUserSerializer < ActiveModel::Serializer
40+
attributes :admin?
41+
def admin?
42+
current_admin.admin
43+
end
44+
end
45+
46+
class AdminUserTestController < ActionController::Base
47+
protect_from_forgery
48+
49+
serialization_scope :current_admin
50+
before_filter { request.format = :json }
51+
52+
def current_admin
53+
TestUser.new('Bob', true)
54+
end
55+
56+
def render_new_user
57+
render json: TestUser.new('pete', false), serializer: AdminUserSerializer, adapter: :json_api
58+
end
59+
end
60+
61+
tests AdminUserTestController
62+
63+
def test_override_scope_name_with_controller
64+
get :render_new_user
65+
assert_equal '{"admin_users":{"admin?":true}}', @response.body
66+
end
67+
end

0 commit comments

Comments
 (0)