mirror of https://github.com/mastodon/mastodon
Experimental Async Refreshes API (#34918)
parent
825312d4b0
commit
319fbbbfac
@ -0,0 +1,16 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
class Api::V1Alpha::AsyncRefreshesController < Api::BaseController
|
||||
before_action -> { doorkeeper_authorize! :read }
|
||||
before_action :require_user!
|
||||
|
||||
def show
|
||||
async_refresh = AsyncRefresh.find(params[:id])
|
||||
|
||||
if async_refresh
|
||||
render json: async_refresh
|
||||
else
|
||||
not_found
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,11 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
module AsyncRefreshesConcern
|
||||
private
|
||||
|
||||
def add_async_refresh_header(async_refresh, retry_seconds: 3)
|
||||
return unless async_refresh.running?
|
||||
|
||||
response.headers['Mastodon-Async-Refresh'] = "id=\"#{async_refresh.id}\", retry=#{retry_seconds}"
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,76 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
class AsyncRefresh
|
||||
extend Redisable
|
||||
include Redisable
|
||||
|
||||
NEW_REFRESH_EXPIRATION = 1.day
|
||||
FINISHED_REFRESH_EXPIRATION = 1.hour
|
||||
|
||||
def self.find(id)
|
||||
redis_key = Rails.application.message_verifier('async_refreshes').verify(id)
|
||||
new(redis_key) if redis.exists?(redis_key)
|
||||
rescue ActiveSupport::MessageVerifier::InvalidSignature
|
||||
nil
|
||||
end
|
||||
|
||||
def self.create(redis_key, count_results: false)
|
||||
data = { 'status' => 'running' }
|
||||
data['result_count'] = 0 if count_results
|
||||
redis.hset(redis_key, data)
|
||||
redis.expire(redis_key, NEW_REFRESH_EXPIRATION)
|
||||
new(redis_key)
|
||||
end
|
||||
|
||||
attr_reader :status, :result_count
|
||||
|
||||
def initialize(redis_key)
|
||||
@redis_key = redis_key
|
||||
fetch_data_from_redis
|
||||
end
|
||||
|
||||
def id
|
||||
Rails.application.message_verifier('async_refreshes').generate(@redis_key)
|
||||
end
|
||||
|
||||
def running?
|
||||
@status == 'running'
|
||||
end
|
||||
|
||||
def finished?
|
||||
@status == 'finished'
|
||||
end
|
||||
|
||||
def finish!
|
||||
redis.pipelined do |pipeline|
|
||||
pipeline.hset(@redis_key, { 'status' => 'finished' })
|
||||
pipeline.expire(@redis_key, FINISHED_REFRESH_EXPIRATION)
|
||||
end
|
||||
@status = 'finished'
|
||||
end
|
||||
|
||||
def reload
|
||||
fetch_data_from_redis
|
||||
self
|
||||
end
|
||||
|
||||
def to_json(_options)
|
||||
{
|
||||
async_refresh: {
|
||||
id:,
|
||||
status:,
|
||||
result_count:,
|
||||
},
|
||||
}.to_json
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def fetch_data_from_redis
|
||||
@status, @result_count = redis.pipelined do |pipeline|
|
||||
pipeline.hget(@redis_key, 'status')
|
||||
pipeline.hget(@redis_key, 'result_count')
|
||||
end
|
||||
@result_count = @result_count.presence&.to_i
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,174 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'rails_helper'
|
||||
|
||||
RSpec.describe AsyncRefresh do
|
||||
subject { described_class.new(redis_key) }
|
||||
|
||||
let(:redis_key) { 'testjob:key' }
|
||||
let(:status) { 'running' }
|
||||
let(:job_hash) { { 'status' => status, 'result_count' => 23 } }
|
||||
|
||||
describe '::find' do
|
||||
context 'when a matching job in redis exists' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
it 'returns a new instance' do
|
||||
id = Rails.application.message_verifier('async_refreshes').generate(redis_key)
|
||||
async_refresh = described_class.find(id)
|
||||
|
||||
expect(async_refresh).to be_a described_class
|
||||
end
|
||||
end
|
||||
|
||||
context 'when no matching job in redis exists' do
|
||||
it 'returns `nil`' do
|
||||
id = Rails.application.message_verifier('async_refreshes').generate('non_existent')
|
||||
expect(described_class.find(id)).to be_nil
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe '::create' do
|
||||
it 'inserts the given key into redis' do
|
||||
described_class.create(redis_key)
|
||||
|
||||
expect(redis.exists?(redis_key)).to be true
|
||||
end
|
||||
|
||||
it 'sets the status to `running`' do
|
||||
async_refresh = described_class.create(redis_key)
|
||||
|
||||
expect(async_refresh.status).to eq 'running'
|
||||
end
|
||||
|
||||
context 'with `count_results`' do
|
||||
it 'set `result_count` to 0' do
|
||||
async_refresh = described_class.create(redis_key, count_results: true)
|
||||
|
||||
expect(async_refresh.result_count).to eq 0
|
||||
end
|
||||
end
|
||||
|
||||
context 'without `count_results`' do
|
||||
it 'does not set `result_count`' do
|
||||
async_refresh = described_class.create(redis_key)
|
||||
|
||||
expect(async_refresh.result_count).to be_nil
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe '#id' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
it "returns a signed version of the job's redis key" do
|
||||
id = subject.id
|
||||
key_name = Base64.decode64(id.split('-').first)
|
||||
|
||||
expect(key_name).to include redis_key
|
||||
end
|
||||
end
|
||||
|
||||
describe '#status' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
context 'when the job is running' do
|
||||
it "returns 'running'" do
|
||||
expect(subject.status).to eq 'running'
|
||||
end
|
||||
end
|
||||
|
||||
context 'when the job is finished' do
|
||||
let(:status) { 'finished' }
|
||||
|
||||
it "returns 'finished'" do
|
||||
expect(subject.status).to eq 'finished'
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe '#running?' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
context 'when the job is running' do
|
||||
it 'returns `true`' do
|
||||
expect(subject.running?).to be true
|
||||
end
|
||||
end
|
||||
|
||||
context 'when the job is finished' do
|
||||
let(:status) { 'finished' }
|
||||
|
||||
it 'returns `false`' do
|
||||
expect(subject.running?).to be false
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe '#finished?' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
context 'when the job is running' do
|
||||
it 'returns `false`' do
|
||||
expect(subject.finished?).to be false
|
||||
end
|
||||
end
|
||||
|
||||
context 'when the job is finished' do
|
||||
let(:status) { 'finished' }
|
||||
|
||||
it 'returns `true`' do
|
||||
expect(subject.finished?).to be true
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe '#finish!' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
it 'sets the status to `finished`' do
|
||||
subject.finish!
|
||||
|
||||
expect(subject).to be_finished
|
||||
end
|
||||
end
|
||||
|
||||
describe '#result_count' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
it 'returns the result count from redis' do
|
||||
expect(subject.result_count).to eq 23
|
||||
end
|
||||
end
|
||||
|
||||
describe '#reload' do
|
||||
before do
|
||||
redis.hset(redis_key, job_hash)
|
||||
end
|
||||
|
||||
it 'reloads the current data from redis and returns itself' do
|
||||
expect(subject).to be_running
|
||||
redis.hset(redis_key, { 'status' => 'finished' })
|
||||
expect(subject).to be_running
|
||||
|
||||
expect(subject.reload).to eq subject
|
||||
|
||||
expect(subject).to be_finished
|
||||
end
|
||||
end
|
||||
end
|
||||
@ -0,0 +1,70 @@
|
||||
# frozen_string_literal: true
|
||||
|
||||
require 'rails_helper'
|
||||
|
||||
RSpec.describe 'AsyncRefreshes' do
|
||||
let(:user) { Fabricate(:user) }
|
||||
let(:token) { Fabricate(:accessible_access_token, resource_owner_id: user.id, scopes: scopes) }
|
||||
let(:headers) { { 'Authorization' => "Bearer #{token.token}" } }
|
||||
let(:job) { AsyncRefresh.new('test_job') }
|
||||
|
||||
describe 'GET /api/v1_alpha/async_refreshes/:id' do
|
||||
context 'when not authorized' do
|
||||
it 'returns http unauthorized' do
|
||||
get api_v1_alpha_async_refresh_path(job.id)
|
||||
|
||||
expect(response)
|
||||
.to have_http_status(401)
|
||||
expect(response.content_type)
|
||||
.to start_with('application/json')
|
||||
end
|
||||
end
|
||||
|
||||
context 'with wrong scope' do
|
||||
before do
|
||||
get api_v1_alpha_async_refresh_path(job.id), headers: headers
|
||||
end
|
||||
|
||||
it_behaves_like 'forbidden for wrong scope', 'write write:accounts'
|
||||
end
|
||||
|
||||
context 'with correct scope' do
|
||||
let(:scopes) { 'read' }
|
||||
|
||||
context 'when job exists' do
|
||||
before do
|
||||
redis.hset('test_job', { 'status' => 'running', 'result_count' => 10 })
|
||||
end
|
||||
|
||||
after do
|
||||
redis.del('test_job')
|
||||
end
|
||||
|
||||
it 'returns http success' do
|
||||
get api_v1_alpha_async_refresh_path(job.id), headers: headers
|
||||
|
||||
expect(response)
|
||||
.to have_http_status(200)
|
||||
|
||||
expect(response.content_type)
|
||||
.to start_with('application/json')
|
||||
|
||||
parsed_response = response.parsed_body
|
||||
expect(parsed_response)
|
||||
.to be_present
|
||||
expect(parsed_response['async_refresh'])
|
||||
.to include('status' => 'running', 'result_count' => 10)
|
||||
end
|
||||
end
|
||||
|
||||
context 'when job does not exist' do
|
||||
it 'returns not found' do
|
||||
get api_v1_alpha_async_refresh_path(job.id), headers: headers
|
||||
|
||||
expect(response)
|
||||
.to have_http_status(404)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
Loading…
Reference in New Issue