首页 > 解决方案 > 使用 redis 进行易碎测试

问题描述

我有一个用于在 redis 中缓存 API 密钥的单元测试(在 rails 应用程序中)。

redis 使用gem设置为全局变量 ( $redis) 。所以我确实得到了redis连接。connection_pool$redis.with do |redis|

问题是,我的测试非常不稳定。它有两个断言:一个用于测试获取预期值,另一个用于测试该值是否被缓存(10 分钟)。

jwt = JWT.encode payload, private_key, "RS256"
assert_equal jwt, Foo.app_token
travel 5.minutes
assert_equal jwt, Foo.app_token

我无法弄清楚为什么测试如此不稳定。它可能每运行 3-5 次就会失败。任何帮助将不胜感激。我从来没有真正在连接池中使用过 redis,所以很可能是在做一些基本的错误。

方法的实现:

def app_token
  find_app_token || create_app_token
end

private

def find_app_token
  $redis.with { |redis| redis.get APP_KEY }
end

def create_app_token
  payload = # some payload
  JWT.encode(payload, private_key, "RS256").tap do |jwt|
     $redis.with do |redis|
       redis.set APP_KEY, jwt
       redis.expireat APP_KEY, payload[:exp]
     end
  end
end

标签: ruby-on-railsrubymultithreadingtestingredis

解决方案


在这里将其作为答案提出以提高知名度。

travel 5.minutes存根 theTime.now和其他相关方法以按所需的持续时间增加/减少时间。存根在测试结束时被移除。我认为他们的意思是在整个测试套件执行结束时。这可能是片状测试的可能原因。

travel还提供了一个选项来提供一个块,然后重置存根。因此,像

travel 5.minutes do
 assert_equal jwt, Foo.app_token
end

可能是一种更合适的方法,因为它会在检查所需的断言后立即删除存根。更多信息: https ://api.rubyonrails.org/v6.0.0/classes/ActiveSupport/Testing/TimeHelpers.html#method-i-travel


推荐阅读