我在Rails 3中使用Devise。我想在production.log中看到current_user的名称。
我想像这样配置rails:
config.log_tags = [:user_name]
答案 0 :(得分:21)
我发现这个有点棘手但有效的解决方案。
Warden在会话中存储用户信息,但req.session不可用于记录。
所以你必须将它添加到config / application.rb
config.middleware.delete(ActionDispatch::Cookies)
config.middleware.delete(ActionDispatch::Session::CookieStore)
config.middleware.insert_before(Rails::Rack::Logger, ActionDispatch::Session::CookieStore)
config.middleware.insert_before(ActionDispatch::Session::CookieStore, ActionDispatch::Cookies)
然后创建文件config / initializers / logging.rb
Rails.configuration.log_tags = [
proc do |req|
if req.session["warden.user.user.key"].nil?
"Anonym"
else
"user_id:#{req.session["warden.user.user.key"][0][0]}"
end
end
]
现在我看到Anonymous:
[Anonym] Served asset ...
这对于用户来说:
[user_id:1] Served asset ...
答案 1 :(得分:13)
我使用了Wojtek Kruszewski的这个解决方案:https://gist.github.com/WojtekKruszewski
我为我的项目调整了一点,只包含id,但基本相同。
# config/application.rb
config.log_tags = [
->(req){
if user_id = WardenTaggedLogger.extract_user_id_from_request(req)
user_id.to_s
else
"?"
end
}
]
并创建此初始化程序
# initializers/warden_tagged_logger.rb
module WardenTaggedLogger
def self.extract_user_id_from_request(req)
session_key = Rails.application.config.session_options[:key]
session_data = req.cookie_jar.encrypted[session_key]
warden_data = session_data["warden.user.user.key"]
warden_data[0][0]
rescue
nil
end
end
答案 2 :(得分:5)
在Rails 4中,encrypted_cookie_store是默认的会话存储。这是您访问会话数据的方式:
session_data = req.cookie_jar.signed[ "_qnaire_session" ]
看起来warden_data在我的新应用中看起来有所不同,例如:[[542], "$2a$10$e5aYxr/PIp6OOj8jzE7mke"]
,其中第一项是用户ID。
这是我目前的代码段:https://gist.github.com/wojt-eu/5109643
这就是我提出的:
config.log_tags = [
:remote_ip,
->(req){
session_data = req.cookie_jar.signed[ "_qnaire_session" ]
warden_data = session_data["warden.user.provider_user.key"]
if warden_data
'#' + warden_data[1][0].to_s
else
"guest"
end
}
]
_qnaire_session
可以替换为instance.config.session_options[:key]
或通过单身取代:Rails.application.config.session_options[:key]
我有ProviderUser模型,因此warden.user.provider_user.key
。我想用User模型这将是warden.user.user.key
。
这很麻烦,但它不会影响正常的身份验证过程,中间件堆栈顺序等。如果在某些更新过程中它只会中断标记日志会受到影响,我应该在查看开发日志时快速注意到这一点。
答案 3 :(得分:2)
不幸的是,日志标记仅在请求委派的最开始时评估一次(在Rails::Rack::Logger
中间件中)。在此阶段没有控制器,因此任何current_user帮助程序都不可用。还没有设置warden甚至会话,但至少有一个cookiejar,所以如果你存储session_id,你可以直接恢复会话或记录session_id。
config.log_tags = [ lambda { |req| req.cookie_jar["_session_id"].to_s } ]
我认为最好的选择是直接在log_in中将用户名存储在cookie中,并使用会话将其销毁。
config.log_tags = [ lambda { |req| req.cookie_jar["user_name"] || 'Noone' } ]
不工作:
但是如果你使用设计,它使用warden raack中间件,所以env['warden']
应该可用,你可以试试吗?
config.log_tags = [ lambda { |req| user = req.env['warden'].user; user && user.name || 'Noone'; } ]
即使没有看门人,因为您确实通过env['rack.session']
提供了会话,如果您在会话中存储用户ID,您可以执行类似
config.log_tags = [ lambda { |req| user = User.find_by_id(req.env['rack.session']['user_id']); user && user.name || 'Noone'; }
答案 4 :(得分:2)
以下是我刚刚添加到config/initializers/logging.rb
的内容:
Rails.configuration.log_tags = [
:uuid, # request UUID
lambda { |req|
# Credentials are (currently) in the format of:
#
# <session_hash>::<user_id>
#
# So we have to split by '::' to obtain the user_id for logging.
#
# This will just output "User: nil" if there is no current session.
"User: #{req.cookies['user_credentials'].to_s.split('::')[1]}"
}
]
这适用于Authlogic。您需要做的事情可能会有所不同,因此您应该深入挖掘并了解您的数据已经暴露给您的信息。
第1步:
查看req
对象的可用内容。将其添加到config/initializers/logging.rb
:
Rails.configuration.log_tags = [
lambda { |req|
req.inspect
}
]
然后点击页面,看看会被转储的内容。
步骤2:使用相同的技术查看您的cookie jar是否有足够的信息:
Rails.configuration.log_tags = [
lambda { |req|
req.cookies.inspect
}
]
(点击请求)
抛开:不要担心将用户名/电子邮件放入日志中 - 用户ID已经足够好了,您可以在数据库中查找它以获取所需的任何额外元数据。
答案 5 :(得分:2)
对我来说几乎有用的东西(Rails 3.2.22.2)就是这里的答案:http://benjit.com/rails/logger/2016/02/26/getting-admin-user-into-rails-logfile/
这假定cookie_jar
对象响应encrypted
。但事实并非如此。最终对我有用的是:
config/initializers/logging.rb
:
Rails.configuration.log_tags = [
lambda { |req|
session_key = Rails.application.config.session_options[:key]
session_data = req.cookie_jar.signed[Rails.application.config.session_options[:key] ]
warden_data = ( session_data["warden.user.user.key"]|| [[]])
admin_user = warden_data[0][0]
"u: #{admin_user || 0}"
}
]
答案 6 :(得分:1)
正如@viktortron在log_tags
初始化session
时说的那样,我们没有适当的session_id
对象可用,但session = ActiveRecord::SessionStore::Session.find_by_session_id(request.cookie_jar["_session_id"])
在请求中。
如果您正在使用_database session_store_,就像我的情况一样,您可以重建会话 ad-hoc:
log_tags
这就是# config/initializers/rails_log.rb
def get_user_id(req)
session = ActiveRecord::SessionStore::Session.find_by_session_id(req.cookie_jar["_session_id"])
result = session ? session.data["user_id"] : 0
"%07d" % result
end
log_tags = []
log_tags << lambda { |req| Time.now.strftime("%F %T.%L") }
log_tags << lambda { |req| req.uuid.first(8) }
log_tags << lambda { |req| get_user_id(req) }
Rails.configuration.log_tags = log_tags
的定义方式:
[2013-01-22 13:51:36.659] [e52435d1] [0036484] <the log line>
结果如下:
{{1}}
答案 7 :(得分:1)
现在完全不同......
https://github.com/roidrage/lograge/issues/23#issuecomment-11709861
我刚刚尝试使用Rails 4,最新的Devise和Lograge。
答案 8 :(得分:1)
我使用Swards solution,它就像一个魅力。但是使用begin..rescue
而不是Hash#has_key?是一个表演杀手
require 'benchmark/ips'
good_hash = { 'warden.user.admin_user.key' => [[1]]}
bad_hash = {}
Benchmark.ips do |x|
x.report('begin+rescue good hash') { good_hash['warden.user.admin_user.key'][0][0] }
x.report('has_key good hash') do
good_hash.has_key?('warden.user.admin_user.key') &&
good_hash['warden.user.admin_user.key'][0][0]
end
x.report('begin+rescue bad hash') { bad_hash['warden.user.admin_user.key'][0][0] rescue nil }
x.report('has_key bad hash') do
if bad_hash.has_key?('warden.user.admin_user.key')
bad_hash['warden.user.admin_user.key'][0][0]
end
end
# Compare the iterations per second of the various reports!
x.compare!
end
结果不言而喻
Comparison:
has_key bad hash: 4870164.1 i/s
begin+rescue good hash: 3544134.7 i/s - 1.37x slower
has_key good hash: 2127500.6 i/s - 2.29x slower
begin+rescue bad hash: 4468.2 i/s - 1089.95x slower
答案 9 :(得分:1)
这是我在Rails 5.1中使用的cookie解密器,用user_id
标记日志,该session[:user_id]
存储在cookie中。它基本上允许我从原始cookie
config.log_tags = [
:request_id,
:remote_ip,
lambda do |req|
session_data = CookieDecrypter.new(req).session_data
"user_id:#{session_data['user_id']}"
end
]
的控制器
环境/ production.rb:
class CookieDecrypter
attr_reader :request
def initialize(request)
@request = request
end
def session_data
cookie = request.cookies[session_key]
return {} unless cookie.present?
cookie = CGI::unescape(cookie)
key_generator = ActiveSupport::KeyGenerator.new(secret_key_base, iterations: 1000)
secret = key_generator.generate_key(salt)[0, ActiveSupport::MessageEncryptor.key_len]
sign_secret = key_generator.generate_key(signed_salt)
encryptor = ActiveSupport::MessageEncryptor.new(secret, sign_secret, serializer: JSON)
encryptor.decrypt_and_verify(cookie) || {}
end
private
def session_key
Rails.application.config.session_options[:key]
end
def secret_key_base
Rails.application.secrets[:secret_key_base]
end
def salt
Rails.application.config.action_dispatch.encrypted_cookie_salt
end
def signed_salt
Rails.application.config.action_dispatch.encrypted_signed_cookie_salt
end
end
应用程序/模型/ cookie_decrypter.rb:
{{1}}
答案 10 :(得分:0)
作为一种快速而丑陋的解决方法,也许可以在处理完请求后记录第二行。
此500 ServerError已通过以下方式显示:#{username}
答案 11 :(得分:-2)
对于任何使用Redis :: Store @ fjuillen的人来说,答案如下:
redis = Redis::Store.new
redis.select 3 # only if you use a different database
result = redis.get req.cookie_jar["_session_id"]
在导轨4上测试。