-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserializable.rb
129 lines (120 loc) · 5.08 KB
/
serializable.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
module Shark
module Serializable
# Return a Hash representation (serialization) of this Object.
def to_h nested: false
normalize_configuration!
# Determine the set of attributes to include in the serialization
attribute_list = if nested
configuration.nested_serialized_attributes
else
configuration.serialized_attributes
end
# Create a hash including all of the requested attributes
hash = attribute_list.each_with_object({}) do |name, h|
h[name] = serialize_attribute(name, send(name), nested: nested)
end
# Only embed associated objects on top level objects, or those which
# specify it in their configuration.
# TODO: this currently ignores `configuration.embed_nested_associated_objects`.
hash[:associated_objects] = serialized_associated_objects if !nested
hash
end
# Create a JSON representation of this Object by creating the Hash and
# calling `to_json` on it.
def to_json opts={}
to_h.to_json(opts)
end
private
# Because object associations are stored as identifiers, embedding them
# requires looking up their actual value in the storage adapter.
def lookup_object identifier
Storage.adapter.find(identifier)
end
# Interpret the current serialization configuration for this Object,
# ensuring that all values have their required properties (e.g.,
# `serialized_attributes` must include `identifier`).
def normalize_configuration!
configuration.validate!(context: self.class)
configuration.serialized_attributes |= [:identifier]
configuration.nested_serialized_attributes |= [:identifier]
configuration.embed_associated_objects = begin
case configuration.embed_associated_objects
when true then self.associated_objects.keys
when false then []
when Array then configuration.embed_associated_objects
end
end
configuration.embed_nested_associated_objects = begin
case configuration.embed_nested_associated_objects
when true then self.associated_objects.keys
when false then []
when Array then configuration.embed_nested_associated_objects
end
end
configuration
end
# Return the serialization of the Object with the given identifier. If the
# Object does not exist, return the identifier string.
#
# If `nested` is true, the nested serialization will be returned instead.
def serialization_for_identifier identifier, nested: false
object = lookup_object(identifier)
object ? object.to_h(nested: nested) : identifier
end
# Return true if the given Object attribute should be embedded in the
# serialization of the Object, based on the configuration and the
# given context.
def should_embed_object object_name, nested: false
if nested
configuration.nested_embedded_objects.include?(object_name)
else
configuration.embedded_objects.include?(object_name)
end
end
# Return the serialized form of the given attribute.
# For simple types like String and Numeric, this will do nothing.
# For container types like Array and Hash, each element they contain will
# be serialized individually.
# For Shark::Object instances, they will be serialized as nested embeds,
# according to their configuration and the configuration of this Object.
def serialize_attribute name, attribute, nested: false
case attribute
# Iteratively serialize container types
when Array
attribute.map{ |element| serialize_attribute(name, element, nested: nested) }
when Hash
attribute.map{ |key, val| [key, serialize_attribute(name, val, nested: nested)] }.to_h
# Strings should check if they are identifiers (but not this Object's
# identifier). If so, follow the configuration options for embedding.
# Otherwise, just pass the string through.
when String
if attribute.identifier? && attribute != self.identifier
if should_embed_object(name, nested: nested)
serialization_for_identifier(attribute, nested: true)
else
attribute
end
else
attribute
end
# Everything else gets passed through
else
attribute
end
end
# Return a Hash representing the serialized version of `associated_objects`
# for the Object.
# The hash contents are set by `configuration.embed_associated_objects`.
# If the value is false, this method will return the unmodified
# `associated_objects` hash.
def serialized_associated_objects
associated_objects.map do |klass, set|
if configuration.embed_associated_objects.include?(klass)
[klass, set.map{ |ident| serialization_for_identifier(ident, nested: true) }]
else
[klass, set.to_a]
end
end.to_h
end
end
end