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