jsonapi-serializer/spec/lib/object_serializer_performance_spec.rb
2018-02-13 23:06:47 -08:00

163 lines
8.1 KiB
Ruby

require 'spec_helper'
describe FastJsonapi::ObjectSerializer, performance: true do
include_context 'movie class'
include_context 'ams movie class'
include_context 'jsonapi movie class'
include_context 'jsonapi-serializers movie class'
include_context 'group class'
include_context 'ams group class'
include_context 'jsonapi group class'
include_context 'jsonapi-serializers group class'
before(:all) { GC.disable }
after(:all) { GC.enable }
context 'when testing performance of serialization' do
it 'should create a hash of 1000 records in less than 50 ms' do
movies = 1000.times.map { |_i| movie }
expect { MovieSerializer.new(movies).serializable_hash }.to perform_under(50).ms
end
it 'should serialize 1000 records to jsonapi in less than 60 ms' do
movies = 1000.times.map { |_i| movie }
expect { MovieSerializer.new(movies).serialized_json }.to perform_under(60).ms
end
it 'should create a hash of 1000 records with includes and meta in less than 75 ms' do
count = 1000
movies = count.times.map { |_i| movie }
options = {}
options[:meta] = { total: count }
options[:include] = [:actors]
expect { MovieSerializer.new(movies, options).serializable_hash }.to perform_under(75).ms
end
it 'should serialize 1000 records to jsonapi with includes and meta in less than 75 ms' do
count = 1000
movies = count.times.map { |_i| movie }
options = {}
options[:meta] = { total: count }
options[:include] = [:actors]
expect { MovieSerializer.new(movies, options).serialized_json }.to perform_under(75).ms
end
end
def print_stats(message, count, ams_time, jsonapi_time, jsonapis_time, our_time)
format = '%-15s %-10s %s'
puts ''
puts message
puts format(format, 'Serializer', 'Records', 'Time')
puts format(format, 'AMS serializer', count, ams_time.round(2).to_s + ' ms')
puts format(format, 'jsonapi-rb serializer', count, jsonapi_time.round(2).to_s + ' ms')
puts format(format, 'jsonapi-serializers', count, jsonapis_time.round(2).to_s + ' ms')
puts format(format, 'Fast serializer', count, our_time.round(2).to_s + ' ms')
end
def run_hash_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
our_time = Benchmark.measure { our_serializer.serializable_hash }.real * 1000
ams_time = Benchmark.measure { ams_serializer.as_json }.real * 1000
jsonapi_time = Benchmark.measure { jsonapi_serializer.to_hash }.real * 1000
jsonapis_time = Benchmark.measure { jsonapis_serializer.to_hash }.real * 1000
print_stats(message, movie_count, ams_time, jsonapi_time, jsonapis_time, our_time)
end
def run_json_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
our_json = nil
ams_json = nil
jsonapi_json = nil
jsonapis_json = nil
our_time = Benchmark.measure { our_json = our_serializer.serialized_json }.real * 1000
ams_time = Benchmark.measure { ams_json = ams_serializer.to_json }.real * 1000
jsonapi_time = Benchmark.measure { jsonapi_json = jsonapi_serializer.to_json }.real * 1000
jsonapis_time = Benchmark.measure { jsonapis_json = jsonapis_serializer.to_json }.real * 1000
print_stats(message, movie_count, ams_time, jsonapi_time, jsonapis_time, our_time)
return our_json, ams_json, jsonapi_json, jsonapis_json
end
context 'when comparing with AMS 0.10.x' do
[1, 25, 250, 1000].each do |movie_count|
speed_factor = 25
it "should serialize #{movie_count} records atleast #{speed_factor} times faster than AMS" do
ams_movies = build_ams_movies(movie_count)
movies = build_movies(movie_count)
jsonapi_movies = build_jsonapi_movies(movie_count)
jsonapis_movies = build_js_movies(movie_count)
our_serializer = MovieSerializer.new(movies)
ams_serializer = ActiveModelSerializers::SerializableResource.new(ams_movies)
jsonapi_serializer = JSONAPISerializer.new(jsonapi_movies)
jsonapis_serializer = JSONAPISSerializer.new(jsonapis_movies)
message = "Serialize to JSON string #{movie_count} records"
our_json, ams_json, _, _ = run_json_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
message = "Serialize to Ruby Hash #{movie_count} records"
run_hash_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
expect(our_json.length).to eq ams_json.length
expect { our_serializer.serialized_json }.to perform_faster_than { ams_serializer.to_json }.at_least(speed_factor).times
expect { our_serializer.serializable_hash }.to perform_faster_than { ams_serializer.as_json }.at_least(speed_factor).times
end
end
end
context 'when comparing with AMS 0.10.x and with includes and meta' do
[1, 25, 250, 1000].each do |movie_count|
speed_factor = 25
it "should serialize #{movie_count} records atleast #{speed_factor} times faster than AMS" do
ams_movies = build_ams_movies(movie_count)
movies = build_movies(movie_count)
jsonapi_movies = build_jsonapi_movies(movie_count)
jsonapis_movies = build_js_movies(movie_count)
options = {}
options[:meta] = { total: movie_count }
options[:include] = [:actors, :movie_type]
our_serializer = MovieSerializer.new(movies, options)
ams_serializer = ActiveModelSerializers::SerializableResource.new(ams_movies, include: options[:include], meta: options[:meta])
jsonapi_serializer = JSONAPISerializer.new(jsonapi_movies, include: options[:include], meta: options[:meta])
jsonapis_serializer = JSONAPISSerializer.new(jsonapis_movies, include: options[:include].map{|i| i.to_s.dasherize}, meta: options[:meta])
message = "Serialize to JSON string #{movie_count} with includes and meta"
our_json, ams_json = run_json_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
message = "Serialize to Ruby Hash #{movie_count} with includes and meta"
run_hash_benchmark(message, movie_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
expect(our_json.length).to eq ams_json.length
expect { our_serializer.serialized_json }.to perform_faster_than { ams_serializer.to_json }.at_least(speed_factor).times
expect { our_serializer.serializable_hash }.to perform_faster_than { ams_serializer.as_json }.at_least(speed_factor).times
end
end
end
context 'when comparing with AMS 0.10.x and with polymorphic has_many' do
[1, 25, 250, 1000].each do |group_count|
speed_factor = 25
it "should serialize #{group_count} records at least #{speed_factor} times faster than AMS" do
ams_groups = build_ams_groups(group_count)
groups = build_groups(group_count)
jsonapi_groups = build_jsonapi_groups(group_count)
jsonapis_groups = build_jsonapis_groups(group_count)
options = {}
our_serializer = GroupSerializer.new(groups, options)
ams_serializer = ActiveModelSerializers::SerializableResource.new(ams_groups)
jsonapi_serializer = JSONAPISerializerB.new(jsonapi_groups)
jsonapis_serializer = JSONAPISSerializerB.new(jsonapis_groups)
message = "Serialize to JSON string #{group_count} with polymorphic has_many"
our_json, ams_json, _, _ = run_json_benchmark(message, group_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
message = "Serialize to Ruby Hash #{group_count} with polymorphic has_many"
run_hash_benchmark(message, group_count, our_serializer, ams_serializer, jsonapi_serializer, jsonapis_serializer)
expect(our_json.length).to eq ams_json.length
expect { our_serializer.serialized_json }.to perform_faster_than { ams_serializer.to_json }.at_least(speed_factor).times
expect { our_serializer.serializable_hash }.to perform_faster_than { ams_serializer.as_json }.at_least(speed_factor).times
end
end
end
end