Ruby on rails Rails控制台-重新加载!模块中的第三方服务

Ruby on rails Rails控制台-重新加载!模块中的第三方服务,ruby-on-rails,ruby,ruby-on-rails-5,Ruby On Rails,Ruby,Ruby On Rails 5,我的应用程序连接到一些第三方API 我有几个APIconnector模块单例,它们在应用程序启动时只初始化一次(初始化意味着客户端使用从secrets检索的凭据实例化一次) 当我重新加载时我的控制台中的应用程序,我正在丢失这些服务,我必须退出并重新启动控制台 基本上,我所有的连接器都包括一个像这样的ServiceConnector模块 module ServiceConnector extend ActiveSupport::Concern included do @activ

我的应用程序连接到一些第三方API

我有几个APIconnector模块单例,它们在应用程序启动时只初始化一次(初始化意味着客户端使用从secrets检索的凭据实例化一次)

当我
重新加载时我的控制台中的应用程序,我正在丢失这些服务,我必须退出并重新启动控制台

基本上,我所有的连接器都包括一个像这样的ServiceConnector模块

module ServiceConnector
  extend ActiveSupport::Concern

  included do
    @activated = false
    @activation_attempt = false
    @client = nil

    attr_reader :client, :activated

    def self.client
      @client ||= service_client
    end

    def self.service_name
      name.gsub('Connector', '')
    end

    def self.activate
      @activation_attempt = true
      if credentials_present?
        @client = service_client
        @activated = true
      end
    end
下面是一个服务实现的示例

module My Connector
  include ServiceConnector

  @app_id = nil
  @api_key = nil

  def self.set_credentials(id, key)
    @app_id = id
    @api_key = key
  end

  def self.credentials_present?
    @app_id.present? and @api_key.present?
  end

  def self.service_client
    ::SomeAPI::Client.new(
      app_id: @app_id,
      api_key: @api_key
    )
  end
end
我使用这种模式,可以重用Rails之外的服务(例如Capistrano、worker without Rails等)。在Rails中,我将以这种方式加载服务

# config/initializers/my_service.rb
if my_service_should_be_activated?
  my_service.set_credentials(
    Rails.application.secrets.my_service_app_id,
    Rails.application.secrets.my_service_app_key
  )
  my_service.activate
end
我猜执行<代码>重新加载
似乎清除了我的所有实例变量,包括
@client
@app\u id
@api\u key


是否可以在
重新加载后添加要执行的代码?在我的情况下,我需要重新运行初始值设定项。或者有没有办法确保我的服务的实例变量不会被重载清除

因此,我提出了一个涉及两个初始值设定项的解决方案

首先是一个000_初始值设定项,它将报告成功加载了哪些机密

module SecretChecker
  module_function

  # Return true if all secrets are present
  def secrets?(secret_list, under:)
    secret_root = Rails.application.secrets
    if under
      if under.is_a?(Array)
        secret_root = secret_root.public_send(under.shift)&.dig(*under.map(&:to_s))
      else
        secret_root = secret_root.public_send(under)
      end
      secret_list.map do |secret|
        secret_root&.dig(secret.to_s).present?
      end
    else
      secret_list.map do |secret|
        secret_root&.public_send(secret.to_s).present?
      end
    end.reduce(:&)
  end

  def check_secrets(theme, secret_list, under: nil)
    return if secrets?(secret_list, under: under)
    message = "WARNING - Missing secrets for #{theme} - #{yield}"
    puts message and Rails.logger.warn(message)
  end
end

SecretChecker.check_secrets('Slack', %i[martine], under: [:slack, :webhooks]) do
  'Slack Notifications will not work'
end

SecretChecker.check_secrets('MongoDB', %i[user password], under: :mongodb) do
  'No Database Connection if auth is activated'
end
然后,使用ActiveSupport::Reloader重新加载服务的模块(以Slack为特征的示例)


也许这可以给你一些关于从哪里开始的灵感:哦,对不起,这个问题很老了,我已经找到了一些介于两者之间的方法。很像你的,我会贴出来的。
# config/initializers/0_service_activation.rb
module ServiceActivation
  def self.with_reload
    ActiveSupport::Reloader.to_prepare do
      yield
    end
  end

  module Slack
    def self.service
      ::SlackConnector
    end

    def self.should_be_activated?
      Rails.env.production? ||
      Rails.env.staging? ||
      (Rails.env.development? && ENV['ENABLE_SLACK'] == 'true')
    end

    def self.activate
      slack = service
      slack.webhook = Rails.application.secrets.slack&.dig('webhooks', 'my_webhook')
      ENV['SLACK_INTERCEPT_CHANNEL'].try do |channel|
        slack.intercept_channel = channel if channel.present?
      end
      slack.activate
      slack
    end
  end
end

[
  ...,
  ServiceActivation::Slack
] .each do |activator|
  ServiceActivation.with_reload do
    activator.activate if activator.should_be_activated?
    activator.service.status_report
  end
end