Commit 6bd072e5 authored by Dmitry Gruzd's avatar Dmitry Gruzd

Add caching for ES migration_has_finished?

This MR adds a caching layer to the
`Elastic::DataMigrationService.migration_has_finished?` method. We plan
to call it each time we build an Elasticsearch query when we have before
and after migration states, so it needs to be performant.
parent 4fb2e5cd
...@@ -20,18 +20,36 @@ module Elastic ...@@ -20,18 +20,36 @@ module Elastic
migrations.sort_by(&:version) migrations.sort_by(&:version)
end end
def drop_migration_has_finished_cache!(migration)
name = migration.name.underscore
Rails.cache.delete cache_key(:migration_has_finished, name)
end
def migration_has_finished?(name) def migration_has_finished?(name)
migration = migrations.find { |migration| migration.name == name.camelize } Rails.cache.fetch cache_key(:migration_has_finished, name), expires_in: 30.minutes do
migration_has_finished_uncached?(name)
end
end
def migration_has_finished_uncached?(name)
migration = migrations.find { |migration| migration.name == name.to_s.camelize }
!!migration&.load_from_index&.dig('_source', 'completed') !!migration&.load_from_index&.dig('_source', 'completed')
end end
def mark_all_as_completed! def mark_all_as_completed!
migrations.each { |migration| migration.save!(completed: true) } migrations.each do |migration|
migration.save!(completed: true)
drop_migration_has_finished_cache!(migration)
end
end end
private private
def cache_key(method_name, *additional_key)
[name, method_name, *additional_key]
end
def parse_migration_filename(filename) def parse_migration_filename(filename)
File.basename(filename).scan(MIGRATION_REGEXP).first File.basename(filename).scan(MIGRATION_REGEXP).first
end end
......
...@@ -34,6 +34,8 @@ module Elastic ...@@ -34,6 +34,8 @@ module Elastic
completed = migration.completed? completed = migration.completed?
logger.info "MigrationWorker: migration[#{migration.name}] updating with completed: #{completed}" logger.info "MigrationWorker: migration[#{migration.name}] updating with completed: #{completed}"
migration.save!(completed: completed) migration.save!(completed: completed)
Elastic::DataMigrationService.drop_migration_has_finished_cache!(migration)
end end
end end
......
...@@ -29,26 +29,44 @@ RSpec.describe Elastic::DataMigrationService, :elastic do ...@@ -29,26 +29,44 @@ RSpec.describe Elastic::DataMigrationService, :elastic do
end end
end end
describe '.migration_has_finished?' do describe '.migration_has_finished_uncached?' do
let(:migration) { subject.migrations.first } let(:migration) { subject.migrations.first }
let(:migration_name) { migration.name.underscore } let(:migration_name) { migration.name.underscore }
it 'returns true if migration has finished' do it 'returns true if migration has finished' do
expect(subject.migration_has_finished?(migration_name)).to eq(false) expect(subject.migration_has_finished_uncached?(migration_name)).to eq(false)
migration.save!(completed: false) migration.save!(completed: false)
refresh_index! refresh_index!
expect(subject.migration_has_finished?(migration_name)).to eq(false) expect(subject.migration_has_finished_uncached?(migration_name)).to eq(false)
migration.save!(completed: true) migration.save!(completed: true)
refresh_index! refresh_index!
expect(subject.migration_has_finished?(migration_name)).to eq(true) expect(subject.migration_has_finished_uncached?(migration_name)).to eq(true)
end
end end
describe '.migration_has_finished?' do
let(:migration) { subject.migrations.first }
let(:migration_name) { migration.name.underscore }
let(:finished) { false }
before do
allow(Rails).to receive(:cache).and_return(ActiveSupport::Cache::MemoryStore.new)
allow(subject).to receive(:migration_has_finished_uncached?).with(migration_name).and_return(finished)
end end
describe 'mark_all_as_completed!' do it 'calls the uncached method only once' do
expect(subject).to receive(:migration_has_finished_uncached?).once
expect(subject.migration_has_finished?(migration_name)).to eq(finished)
expect(subject.migration_has_finished?(migration_name)).to eq(finished)
end
end
describe '.mark_all_as_completed!' do
it 'creates all migration versions' do it 'creates all migration versions' do
expect(Elastic::MigrationRecord.persisted_versions(completed: true).count).to eq(0) expect(Elastic::MigrationRecord.persisted_versions(completed: true).count).to eq(0)
...@@ -57,5 +75,41 @@ RSpec.describe Elastic::DataMigrationService, :elastic do ...@@ -57,5 +75,41 @@ RSpec.describe Elastic::DataMigrationService, :elastic do
expect(Elastic::MigrationRecord.persisted_versions(completed: true).count).to eq(subject.migrations.count) expect(Elastic::MigrationRecord.persisted_versions(completed: true).count).to eq(subject.migrations.count)
end end
it 'drops all cache keys' do
allow(subject).to receive(:migrations).and_return(
[
Elastic::MigrationRecord.new(version: 100, name: 'SomeMigration', filename: nil),
Elastic::MigrationRecord.new(version: 200, name: 'SomeOtherMigration', filename: nil)
]
)
subject.migrations.each do |migration|
expect(subject).to receive(:drop_migration_has_finished_cache!).with(migration)
end
subject.mark_all_as_completed!
end
end
describe '.drop_migration_has_finished_cache!' do
let(:migration) { subject.migrations.first }
let(:migration_name) { migration.name.underscore }
let(:finished) { true }
before do
allow(Rails).to receive(:cache).and_return(ActiveSupport::Cache::MemoryStore.new)
allow(subject).to receive(:migration_has_finished_uncached?).with(migration_name).and_return(finished)
end
it 'drops cache' do
expect(subject).to receive(:migration_has_finished_uncached?).twice
expect(subject.migration_has_finished?(migration_name)).to eq(finished)
subject.drop_migration_has_finished_cache!(migration)
expect(subject.migration_has_finished?(migration_name)).to eq(finished)
end
end end
end end
...@@ -67,6 +67,7 @@ RSpec.describe Elastic::MigrationWorker, :elastic do ...@@ -67,6 +67,7 @@ RSpec.describe Elastic::MigrationWorker, :elastic do
end end
expect(migration).to receive(:save!).with(completed: completed) expect(migration).to receive(:save!).with(completed: completed)
expect(Elastic::DataMigrationService).to receive(:drop_migration_has_finished_cache!).with(migration)
subject.perform subject.perform
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment