Cache JSON of immutable ActivityPub representations (#6171)
This commit is contained in:
		
							parent
							
								
									d907d4352e
								
							
						
					
					
						commit
						c10f4bdb03
					
				
					 5 changed files with 48 additions and 32 deletions
				
			
		| 
						 | 
					@ -2,7 +2,8 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class AccountsController < ApplicationController
 | 
					class AccountsController < ApplicationController
 | 
				
			||||||
  include AccountControllerConcern
 | 
					  include AccountControllerConcern
 | 
				
			||||||
  include SignatureVerification
 | 
					
 | 
				
			||||||
 | 
					  before_action :set_cache_headers
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def show
 | 
					  def show
 | 
				
			||||||
    respond_to do |format|
 | 
					    respond_to do |format|
 | 
				
			||||||
| 
						 | 
					@ -26,10 +27,11 @@ class AccountsController < ApplicationController
 | 
				
			||||||
      end
 | 
					      end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      format.json do
 | 
					      format.json do
 | 
				
			||||||
        render json: @account,
 | 
					        skip_session!
 | 
				
			||||||
               serializer: ActivityPub::ActorSerializer,
 | 
					
 | 
				
			||||||
               adapter: ActivityPub::Adapter,
 | 
					        render_cached_json(['activitypub', 'actor', @account.cache_key], content_type: 'application/activity+json') do
 | 
				
			||||||
               content_type: 'application/activity+json'
 | 
					          ActiveModelSerializers::SerializableResource.new(@account, serializer: ActivityPub::ActorSerializer, adapter: ActivityPub::Adapter)
 | 
				
			||||||
 | 
					        end
 | 
				
			||||||
      end
 | 
					      end
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -4,15 +4,19 @@ class ActivityPub::FollowsController < Api::BaseController
 | 
				
			||||||
  include SignatureVerification
 | 
					  include SignatureVerification
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def show
 | 
					  def show
 | 
				
			||||||
    render(
 | 
					    render json: follow_request,
 | 
				
			||||||
      json: FollowRequest.includes(:account).references(:account).find_by!(
 | 
					 | 
				
			||||||
        id: params.require(:id),
 | 
					 | 
				
			||||||
        accounts: { domain: nil, username: params.require(:account_username) },
 | 
					 | 
				
			||||||
        target_account: signed_request_account
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
           serializer: ActivityPub::FollowSerializer,
 | 
					           serializer: ActivityPub::FollowSerializer,
 | 
				
			||||||
           adapter: ActivityPub::Adapter,
 | 
					           adapter: ActivityPub::Adapter,
 | 
				
			||||||
           content_type: 'application/activity+json'
 | 
					           content_type: 'application/activity+json'
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  private
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  def follow_request
 | 
				
			||||||
 | 
					    FollowRequest.includes(:account).references(:account).find_by!(
 | 
				
			||||||
 | 
					      id: params.require(:id),
 | 
				
			||||||
 | 
					      accounts: { domain: nil, username: params.require(:account_username) },
 | 
				
			||||||
 | 
					      target_account: signed_request_account
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
end
 | 
					end
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -123,11 +123,23 @@ class ApplicationController < ActionController::Base
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def render_cached_json(cache_key, **options)
 | 
					  def render_cached_json(cache_key, **options)
 | 
				
			||||||
 | 
					    options[:expires_in] ||= 3.minutes
 | 
				
			||||||
 | 
					    cache_key              = cache_key.join(':') if cache_key.is_a?(Enumerable)
 | 
				
			||||||
 | 
					    content_type           = options.delete(:content_type) || 'application/json'
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    data = Rails.cache.fetch(cache_key, { raw: true }.merge(options)) do
 | 
					    data = Rails.cache.fetch(cache_key, { raw: true }.merge(options)) do
 | 
				
			||||||
      yield.to_json
 | 
					      yield.to_json
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    expires_in options[:expires_in], public: true
 | 
					    expires_in options[:expires_in], public: true
 | 
				
			||||||
    render json: data
 | 
					    render json: data, content_type: content_type
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  def set_cache_headers
 | 
				
			||||||
 | 
					    response.headers['Vary'] = 'Accept'
 | 
				
			||||||
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  def skip_session!
 | 
				
			||||||
 | 
					    request.session_options[:skip] = true
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
end
 | 
					end
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,14 +2,16 @@
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class EmojisController < ApplicationController
 | 
					class EmojisController < ApplicationController
 | 
				
			||||||
  before_action :set_emoji
 | 
					  before_action :set_emoji
 | 
				
			||||||
 | 
					  before_action :set_cache_headers
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def show
 | 
					  def show
 | 
				
			||||||
    respond_to do |format|
 | 
					    respond_to do |format|
 | 
				
			||||||
      format.json do
 | 
					      format.json do
 | 
				
			||||||
        render json: @emoji,
 | 
					        skip_session!
 | 
				
			||||||
               serializer: ActivityPub::EmojiSerializer,
 | 
					
 | 
				
			||||||
               adapter: ActivityPub::Adapter,
 | 
					        render_cached_json(['activitypub', 'emoji', @emoji.cache_key], content_type: 'application/activity+json') do
 | 
				
			||||||
               content_type: 'application/activity+json'
 | 
					          ActiveModelSerializers::SerializableResource.new(@emoji, serializer: ActivityPub::EmojiSerializer, adapter: ActivityPub::Adapter)
 | 
				
			||||||
 | 
					        end
 | 
				
			||||||
      end
 | 
					      end
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -10,7 +10,7 @@ class StatusesController < ApplicationController
 | 
				
			||||||
  before_action :set_link_headers
 | 
					  before_action :set_link_headers
 | 
				
			||||||
  before_action :check_account_suspension
 | 
					  before_action :check_account_suspension
 | 
				
			||||||
  before_action :redirect_to_original, only: [:show]
 | 
					  before_action :redirect_to_original, only: [:show]
 | 
				
			||||||
  before_action { response.headers['Vary'] = 'Accept' }
 | 
					  before_action :set_cache_headers
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def show
 | 
					  def show
 | 
				
			||||||
    respond_to do |format|
 | 
					    respond_to do |format|
 | 
				
			||||||
| 
						 | 
					@ -22,25 +22,21 @@ class StatusesController < ApplicationController
 | 
				
			||||||
      end
 | 
					      end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      format.json do
 | 
					      format.json do
 | 
				
			||||||
        render json: @status,
 | 
					        skip_session! unless @stream_entry.hidden?
 | 
				
			||||||
               serializer: ActivityPub::NoteSerializer,
 | 
					 | 
				
			||||||
               adapter: ActivityPub::Adapter,
 | 
					 | 
				
			||||||
               content_type: 'application/activity+json'
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # Allow HTTP caching for 3 minutes if the status is public
 | 
					        render_cached_json(['activitypub', 'note', @status.cache_key], content_type: 'application/activity+json') do
 | 
				
			||||||
        unless @stream_entry.hidden?
 | 
					          ActiveModelSerializers::SerializableResource.new(@status, serializer: ActivityPub::NoteSerializer, adapter: ActivityPub::Adapter)
 | 
				
			||||||
          request.session_options[:skip] = true
 | 
					 | 
				
			||||||
          expires_in(3.minutes, public: true)
 | 
					 | 
				
			||||||
        end
 | 
					        end
 | 
				
			||||||
      end
 | 
					      end
 | 
				
			||||||
    end
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def activity
 | 
					  def activity
 | 
				
			||||||
    render json: @status,
 | 
					    skip_session!
 | 
				
			||||||
           serializer: ActivityPub::ActivitySerializer,
 | 
					
 | 
				
			||||||
           adapter: ActivityPub::Adapter,
 | 
					    render_cached_json(['activitypub', 'activity', @status.cache_key], content_type: 'application/activity+json') do
 | 
				
			||||||
           content_type: 'application/activity+json'
 | 
					      ActiveModelSerializers::SerializableResource.new(@status, serializer: ActivityPub::ActivitySerializer, adapter: ActivityPub::Adapter)
 | 
				
			||||||
 | 
					    end
 | 
				
			||||||
  end
 | 
					  end
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  def embed
 | 
					  def embed
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		
		Reference in a new issue