154 lines
5.8 KiB
Ruby
154 lines
5.8 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'active_support/concern'
|
|
require 'fast_jsonapi/multi_to_json'
|
|
|
|
module FastJsonapi
|
|
MandatoryField = Class.new(StandardError)
|
|
|
|
module SerializationCore
|
|
extend ActiveSupport::Concern
|
|
|
|
included do
|
|
class << self
|
|
attr_accessor :attributes_to_serialize,
|
|
:relationships_to_serialize,
|
|
:cachable_relationships_to_serialize,
|
|
:uncachable_relationships_to_serialize,
|
|
:transform_method,
|
|
:record_type,
|
|
:record_id,
|
|
:cache_length,
|
|
:race_condition_ttl,
|
|
:cached
|
|
end
|
|
end
|
|
|
|
class_methods do
|
|
def id_hash(id, record_type, default_return=false)
|
|
if id.present?
|
|
{ id: id.to_s, type: record_type }
|
|
else
|
|
default_return ? { id: nil, type: record_type } : nil
|
|
end
|
|
end
|
|
|
|
def ids_hash(ids, record_type)
|
|
return ids.map { |id| id_hash(id, record_type) } if ids.respond_to? :map
|
|
id_hash(ids, record_type) # ids variable is just a single id here
|
|
end
|
|
|
|
def id_hash_from_record(record, record_types)
|
|
# memoize the record type within the record_types dictionary, then assigning to record_type:
|
|
record_type = record_types[record.class] ||= record.class.name.underscore.to_sym
|
|
id_hash(record.id, record_type)
|
|
end
|
|
|
|
def ids_hash_from_record_and_relationship(record, relationship)
|
|
polymorphic = relationship[:polymorphic]
|
|
|
|
return ids_hash(
|
|
fetch_id(record, relationship),
|
|
relationship[:record_type]
|
|
) unless polymorphic
|
|
|
|
return unless associated_object = fetch_associated_object(record, relationship)
|
|
|
|
return associated_object.map do |object|
|
|
id_hash_from_record object, polymorphic
|
|
end if associated_object.respond_to? :map
|
|
|
|
id_hash_from_record associated_object, polymorphic
|
|
end
|
|
|
|
def attributes_hash(record, params = {})
|
|
attributes_to_serialize.each_with_object({}) do |(key, method), attr_hash|
|
|
attr_hash[key] = if method.is_a?(Proc)
|
|
method.arity == 1 ? method.call(record) : method.call(record, params)
|
|
else
|
|
record.public_send(method)
|
|
end
|
|
end
|
|
end
|
|
|
|
def relationships_hash(record, relationships = nil)
|
|
relationships = relationships_to_serialize if relationships.nil?
|
|
|
|
relationships.each_with_object({}) do |(_k, relationship), hash|
|
|
name = relationship[:key]
|
|
empty_case = relationship[:relationship_type] == :has_many ? [] : nil
|
|
hash[name] = {
|
|
data: ids_hash_from_record_and_relationship(record, relationship) || empty_case
|
|
}
|
|
end
|
|
end
|
|
|
|
def record_hash(record, params = {})
|
|
if cached
|
|
record_hash = Rails.cache.fetch(record.cache_key, expires_in: cache_length, race_condition_ttl: race_condition_ttl) do
|
|
temp_hash = id_hash(id_from_record(record), record_type, true)
|
|
temp_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present?
|
|
temp_hash[:relationships] = {}
|
|
temp_hash[:relationships] = relationships_hash(record, cachable_relationships_to_serialize) if cachable_relationships_to_serialize.present?
|
|
temp_hash
|
|
end
|
|
record_hash[:relationships] = record_hash[:relationships].merge(relationships_hash(record, uncachable_relationships_to_serialize)) if uncachable_relationships_to_serialize.present?
|
|
record_hash
|
|
else
|
|
record_hash = id_hash(id_from_record(record), record_type, true)
|
|
record_hash[:attributes] = attributes_hash(record, params) if attributes_to_serialize.present?
|
|
record_hash[:relationships] = relationships_hash(record) if relationships_to_serialize.present?
|
|
record_hash
|
|
end
|
|
end
|
|
|
|
def id_from_record(record)
|
|
return record.send(record_id) if record_id
|
|
raise MandatoryField, 'id is a mandatory field in the jsonapi spec' unless record.respond_to?(:id)
|
|
record.id
|
|
end
|
|
|
|
# Override #to_json for alternative implementation
|
|
def to_json(payload)
|
|
FastJsonapi::MultiToJson.to_json(payload) if payload.present?
|
|
end
|
|
|
|
# includes handler
|
|
|
|
def get_included_records(record, includes_list, known_included_objects)
|
|
includes_list.each_with_object([]) do |item, included_records|
|
|
included_objects = fetch_associated_object(record, @relationships_to_serialize[item])
|
|
next if included_objects.blank?
|
|
|
|
record_type = @relationships_to_serialize[item][:record_type]
|
|
serializer = @relationships_to_serialize[item][:serializer].to_s.constantize
|
|
relationship_type = @relationships_to_serialize[item][:relationship_type]
|
|
included_objects = [included_objects] unless relationship_type == :has_many
|
|
included_objects.each do |inc_obj|
|
|
code = "#{record_type}_#{inc_obj.id}"
|
|
next if known_included_objects.key?(code)
|
|
known_included_objects[code] = inc_obj
|
|
included_records << serializer.record_hash(inc_obj)
|
|
end
|
|
end
|
|
end
|
|
|
|
def fetch_associated_object(record, relationship)
|
|
return relationship[:object_block].call(record) unless relationship[:object_block].nil?
|
|
record.send(relationship[:object_method_name])
|
|
end
|
|
|
|
def fetch_id(record, relationship)
|
|
unless relationship[:object_block].nil?
|
|
object = relationship[:object_block].call(record)
|
|
|
|
return object.map(&:id) if object.respond_to? :map
|
|
return object.id
|
|
end
|
|
|
|
record.public_send(relationship[:id_method_name])
|
|
end
|
|
end
|
|
end
|
|
end
|