134 lines
3.2 KiB
Ruby
134 lines
3.2 KiB
Ruby
|
require 'spec_helper'
|
||
|
|
||
|
describe Gitlab::Cache::RequestCache do
|
||
|
let(:klass) do
|
||
|
Class.new do
|
||
|
extend Gitlab::Cache::RequestCache
|
||
|
|
||
|
attr_accessor :id, :name, :result, :extra
|
||
|
|
||
|
def self.name
|
||
|
'ExpensiveAlgorithm'
|
||
|
end
|
||
|
|
||
|
def initialize(id, name, result, extra = nil)
|
||
|
self.id = id
|
||
|
self.name = name
|
||
|
self.result = result
|
||
|
self.extra = nil
|
||
|
end
|
||
|
|
||
|
request_cache def compute(arg)
|
||
|
result << arg
|
||
|
end
|
||
|
|
||
|
request_cache def repute(arg)
|
||
|
result << arg
|
||
|
end
|
||
|
|
||
|
def dispute(arg)
|
||
|
result << arg
|
||
|
end
|
||
|
request_cache(:dispute) { extra }
|
||
|
end
|
||
|
end
|
||
|
|
||
|
let(:algorithm) { klass.new('id', 'name', []) }
|
||
|
|
||
|
shared_examples 'cache for the same instance' do
|
||
|
it 'does not compute twice for the same argument' do
|
||
|
algorithm.compute(true)
|
||
|
result = algorithm.compute(true)
|
||
|
|
||
|
expect(result).to eq([true])
|
||
|
end
|
||
|
|
||
|
it 'computes twice for the different argument' do
|
||
|
algorithm.compute(true)
|
||
|
result = algorithm.compute(false)
|
||
|
|
||
|
expect(result).to eq([true, false])
|
||
|
end
|
||
|
|
||
|
it 'computes twice for the different class name' do
|
||
|
algorithm.compute(true)
|
||
|
allow(klass).to receive(:name).and_return('CheapAlgo')
|
||
|
result = algorithm.compute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
|
||
|
it 'computes twice for the different method' do
|
||
|
algorithm.compute(true)
|
||
|
result = algorithm.repute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
|
||
|
context 'when request_cache_key is provided' do
|
||
|
before do
|
||
|
klass.request_cache_key do
|
||
|
[id, name]
|
||
|
end
|
||
|
end
|
||
|
|
||
|
it 'computes twice for the different keys, id' do
|
||
|
algorithm.compute(true)
|
||
|
algorithm.id = 'ad'
|
||
|
result = algorithm.compute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
|
||
|
it 'computes twice for the different keys, name' do
|
||
|
algorithm.compute(true)
|
||
|
algorithm.name = 'same'
|
||
|
result = algorithm.compute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
|
||
|
it 'uses extra method cache key if provided' do
|
||
|
algorithm.dispute(true) # miss
|
||
|
algorithm.extra = true
|
||
|
algorithm.dispute(true) # miss
|
||
|
result = algorithm.dispute(true) # hit
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when RequestStore is active', :request_store do
|
||
|
it_behaves_like 'cache for the same instance'
|
||
|
|
||
|
it 'computes once for different instances when keys are the same' do
|
||
|
algorithm.compute(true)
|
||
|
result = klass.new('id', 'name', algorithm.result).compute(true)
|
||
|
|
||
|
expect(result).to eq([true])
|
||
|
end
|
||
|
|
||
|
it 'computes twice if RequestStore starts over' do
|
||
|
algorithm.compute(true)
|
||
|
RequestStore.end!
|
||
|
RequestStore.clear!
|
||
|
RequestStore.begin!
|
||
|
result = algorithm.compute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
end
|
||
|
|
||
|
context 'when RequestStore is inactive' do
|
||
|
it_behaves_like 'cache for the same instance'
|
||
|
|
||
|
it 'computes twice for different instances even if keys are the same' do
|
||
|
algorithm.compute(true)
|
||
|
result = klass.new('id', 'name', algorithm.result).compute(true)
|
||
|
|
||
|
expect(result).to eq([true, true])
|
||
|
end
|
||
|
end
|
||
|
end
|