Ruby on rails Rails控制台-重新加载!模块中的第三方服务
我的应用程序连接到一些第三方API 我有几个APIconnector模块单例,它们在应用程序启动时只初始化一次(初始化意味着客户端使用从secrets检索的凭据实例化一次) 当我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
重新加载时代码>我的控制台中的应用程序,我正在丢失这些服务,我必须退出并重新启动控制台
基本上,我所有的连接器都包括一个像这样的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