65 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			65 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
# frozen_string_literal: true
 | 
						|
 | 
						|
class Mastodon::SidekiqMiddleware
 | 
						|
  BACKTRACE_LIMIT = 3
 | 
						|
 | 
						|
  def call(*, &block)
 | 
						|
    Chewy.strategy(:mastodon, &block)
 | 
						|
  rescue Mastodon::HostValidationError
 | 
						|
    # Do not retry
 | 
						|
  rescue => e
 | 
						|
    clean_up_elasticsearch_connections!
 | 
						|
    limit_backtrace_and_raise(e)
 | 
						|
  ensure
 | 
						|
    clean_up_sockets!
 | 
						|
  end
 | 
						|
 | 
						|
  private
 | 
						|
 | 
						|
  def limit_backtrace_and_raise(exception)
 | 
						|
    exception.set_backtrace(exception.backtrace.first(BACKTRACE_LIMIT)) unless ENV['BACKTRACE']
 | 
						|
    raise exception
 | 
						|
  end
 | 
						|
 | 
						|
  def clean_up_sockets!
 | 
						|
    clean_up_redis_socket!
 | 
						|
    clean_up_statsd_socket!
 | 
						|
  end
 | 
						|
 | 
						|
  # This is a hack to immediately free up unused Elasticsearch connections.
 | 
						|
  #
 | 
						|
  # Indeed, Chewy creates one `Elasticsearch::Client` instance per thread,
 | 
						|
  # and each such client manages its long-lasting connection to
 | 
						|
  # Elasticsearch.
 | 
						|
  #
 | 
						|
  # As far as I know, neither `chewy`,  `elasticsearch-transport` or even
 | 
						|
  # `faraday` provide a reliable way to immediately close a connection, and
 | 
						|
  # rely on the underlying object to be garbage-collected instead.
 | 
						|
  #
 | 
						|
  # Furthermore, `sidekiq` creates a new thread each time a job throws an
 | 
						|
  # exception, meaning that each failure will create a new connection, and
 | 
						|
  # the old one will only be closed on full garbage collection.
 | 
						|
  def clean_up_elasticsearch_connections!
 | 
						|
    return unless Chewy.enabled? && Chewy.current[:chewy_client].present?
 | 
						|
 | 
						|
    Chewy.client.transport.transport.connections.each do |connection|
 | 
						|
      # NOTE: This bit of code is tailored for the HTTPClient Faraday adapter
 | 
						|
      connection.connection.app.instance_variable_get(:@client)&.reset_all
 | 
						|
    end
 | 
						|
 | 
						|
    Chewy.current.delete(:chewy_client)
 | 
						|
  rescue
 | 
						|
    nil
 | 
						|
  end
 | 
						|
 | 
						|
  def clean_up_redis_socket!
 | 
						|
    RedisConfiguration.pool.checkin if Thread.current[:redis]
 | 
						|
    Thread.current[:redis] = nil
 | 
						|
  end
 | 
						|
 | 
						|
  def clean_up_statsd_socket!
 | 
						|
    Thread.current[:statsd_socket]&.close
 | 
						|
    Thread.current[:statsd_socket] = nil
 | 
						|
  end
 | 
						|
end
 |