我有一个rails 5.1应用程序,它使用Devise来处理我的用户模型的身份验证.这个应用程序有一个Oracle数据库后端,需要在执行任何查询之前使用登录用户设置系统上下文变量,所以我希望在ConnectionAdapter的:checkout回调中执行此操作.
class ApplicationController < ActionController::Base
before_action :log_user
ActiveRecord::ConnectionAdapters::OracleEnhancedAdapter.set_callback :checkout, :after do
# Would like to get the logged-in user's username here so I can apply
# it to the oracle sys_context.
# The below throws "undefined method 'user_signed_in?'"
username = current_user.username if user_signed_in?
end
def log_user
# When in this method, user_signed_in? and current_user work fine.
puts "User is #{current_user.username}" if user_signed_in?
end
end
user_signed_in?在:checkout回调块中运行时找不到方法,尽管它通常在控制器中可用.为什么?
此外,块中的current_user似乎评估为ConnectionAdapter中定义的current_user方法,而不是Devise定义的方法.如何访问Devise的current_user?
如何在此回调中使用这些Devise提供的方法?
最佳答案 您不能使用checkout回调,在执行它时,它没有与控制器上下文的连接.您在ApplicationController中定义它的事实与它实际执行的上下文无关.
您需要在before_action中设置连接选项,以便在控制器上下文中运行.就像是:
before_action :set_user_context
def set_user_context
if current_user
ApplicationRecord.connection.execute "DBMS_SESSION.SET_CONTEXT('whatever', 'goes', 'here', '#{current_user.username}')"
end
end
…或类似的东西.请注意,您可能需要添加checkin回调以在连接完成时清除该值.
顺便说一下,我几天前回答了一个几乎完全相同的问题:https://stackoverflow.com/a/54837596/152786不同的命令,但可能有所帮助.