Redis 更新后错误

当我尝试迁移我的数据库时,我一直收到这个 Redis 错误。这是一个 discourse docker 安装。请帮忙。

1 个赞

您是否有双容器安装或在其他地方托管的 Redis?您有 Redis 6.2 吗?

2 个赞

这是我的 Redis 6.2.6 版本,我的 docker 中只有一个 redis 容器,还有几个 discourse_dev 容器。

2 个赞

我已在 sidekiq github 上寻求帮助。
当我使用 docker 容器运行时,会出现以下情况:
d/bundle show redis

/Users/natashaselvidge/discourse/vendor/bundle/ruby/2.6.0/gems/redis-4.6.0

这是我的 discourse_redis.rb 文件:

#
#  一个包装器,围绕 redis,用当前站点 ID 对键进行命名空间
#

class DiscourseRedis
  def self.raw_connection(config = nil)
    config ||= self.config
    Redis.new(config)
  end

  def self.config
    GlobalSetting.redis_config
  end

  def initialize(config = nil, namespace: true)
    @config = config || DiscourseRedis.config
    @redis = DiscourseRedis.raw_connection(@config.dup)
    @namespace = namespace
  end

  def without_namespace
    # 仅当您要存储和检索跨站点共享的数据时才使用此方法
    @redis
  end

  def self.ignore_readonly
    yield
  rescue Redis::CommandError => ex
    if ex.message =~ /READONLY/
      Discourse.received_redis_readonly!
      nil
    else
      raise ex
    end
  end

  # 使用命名空间为键添加前缀
  def method_missing(meth, *args, **kwargs, &block)
    if @redis.respond_to?(meth)
      DiscourseRedis.ignore_readonly { @redis.public_send(meth, *args, **kwargs, &block) }
    else
      super
    end
  end

  # 通过代理键方法,但使用命名空间为键添加前缀
  [:append, :blpop, :brpop, :brpoplpush, :decr, :decrby, :expire, :expireat, :get, :getbit, :getrange, :getset,
   :hdel, :hexists, :hget, :hgetall, :hincrby, :hincrbyfloat, :hkeys, :hlen, :hmget, :hmset, :hset, :hsetnx, :hvals, :incr,
   :incrby, :incrbyfloat, :lindex, :linsert, :llen, :lpop, :lpush, :lpushx, :lrange, :lrem, :lset, :ltrim,
   :mapped_hmset, :mapped_hmget, :mapped_mget, :mapped_mset, :mapped_msetnx, :move, :mset,
   :msetnx, :persist, :pexpire, :pexpireat, :psetex, :pttl, :rename, :renamenx, :rpop, :rpoplpush, :rpush, :rpushx, :sadd, :scard,
   :sdiff, :set, :setbit, :setex, :setnx, :setrange, :sinter, :sismember, :smembers, :sort, :spop, :srandmember, :srem, :strlen,
   :sunion, :ttl, :type, :watch, :zadd, :zcard, :zcount, :zincrby, :zrange, :zrangebyscore, :zrank, :zrem, :zremrangebyrank,
   :zremrangebyscore, :zrevrange, :zrevrangebyscore, :zrevrank, :zrangebyscore,
   :dump, :restore].each do |m|
    define_method m do |*args, **kwargs|
      args[0] = "#{namespace}:#{args[0]}" if @namespace
      DiscourseRedis.ignore_readonly { @redis.public_send(m, *args, **kwargs) }
    end
  end

  def exists(*args)
    args.map! { |a| "#{namespace}:#{a}" } if @namespace
    DiscourseRedis.ignore_readonly { @redis.exists(*args) }
  end

  def exists?(*args)
    args.map! { |a| "#{namespace}:#{a}" } if @namespace
    DiscourseRedis.ignore_readonly { @redis.exists?(*args) }
  end

  def mget(*args)
    args.map! { |a| "#{namespace}:#{a}" }  if @namespace
    DiscourseRedis.ignore_readonly { @redis.mget(*args) }
  end

  def del(*keys)
    DiscourseRedis.ignore_readonly do
      keys = keys.flatten(1)
      keys.map! { |k| "#{namespace}:#{k}" } if @namespace
      @redis.del(*keys)
    end
  end

  def scan_each(options = {}, &block)
    DiscourseRedis.ignore_readonly do
      match = options[:match].presence || '*'

      options[:match] =
        if @namespace
          "#{namespace}:#{match}"
        else
          match
        end

      if block
        @redis.scan_each(**options) do |key|
          key = remove_namespace(key) if @namespace
          block.call(key)
        end
      else
        @redis.scan_each(**options).map do |key|
          key = remove_namespace(key) if @namespace
          key
        end
      end
    end
  end

  def keys(pattern = nil)
    DiscourseRedis.ignore_readonly do
      pattern = pattern || '*'
      pattern = "#{namespace}:#{pattern}" if @namespace
      keys = @redis.keys(pattern)

      if @namespace
        len = namespace.length + 1
        keys.map! { |k| k[len..-1] }
      end

      keys
    end
  end

  def delete_prefixed(prefix)
    DiscourseRedis.ignore_readonly do
      keys("#{prefix}*").each { |k| Discourse.redis.del(k) }
    end
  end

  def reconnect
    @redis._client.reconnect
  end

  def namespace_key(key)
    if @namespace
      "#{namespace}:#{key}"
    else
      key
    end
  end

  def namespace
    RailsMultisite::ConnectionManagement.current_db
  end

  def self.new_redis_store
    Cache.new
  end

  private

  def remove_namespace(key)
    key[(namespace.length + 1)..-1]
  end

  class EvalHelper
    def initialize(script)
      @script = script
      @sha1 = Digest::SHA1.hexdigest(script)
    end

    def eval(redis, *args, **kwargs)
      redis.evalsha @sha1, *args, **kwargs
    rescue ::Redis::CommandError => e
      if e.to_s =~ /^NOSCRIPT/
        redis.eval @script, *args, **kwargs
      else
        raise
      end
    end
  end
end
natashaselvidge@Natashas-MacBook-Pro discourse % d/rake db:migrate
弃用通知:enable_long_polling/long_polling_interval 已从站点设置切换到全局设置。请从“站点设置”UI 中移除覆盖,并使用配置文件或环境变量来设置全局设置。(将在 Discourse 2.9.0 中移除)
在 /usr/local/lib/ruby/gems/2.7.0/gems/bootsnap-1.10.3/lib/bootsnap/load_path_cache/core_ext/kernel_require.rb:48:in `load`
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
正在忽略 /src/db/schema_cache.yml,因为它已过期。当前架构版本是 20220124003259,但缓存中的版本是 20211011123651。
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
在 Redis 实例上进行管道命令已弃用,将在 Redis 5.0.0 中移除。

redis.multi do
  redis.get("key")
end

应替换为

redis.multi do |pipeline|
  pipeline.get("key")
end

(在 /src/lib/discourse_redis.rb:42:in `public_send` 中调用)
rake aborted!
Redis::CommandError: ERR unknown command `synchronize`, with args beginning with:
/src/lib/distributed_mutex.rb:84:in `try_to_get_lock'
/src/lib/distributed_mutex.rb:61:in `get_lock'
/src/lib/distributed_mutex.rb:30:in `block in synchronize'
/src/lib/distributed_mutex.rb:29:in `synchronize'
/src/lib/distributed_mutex.rb:29:in `synchronize'
/src/lib/distributed_mutex.rb:14:in `synchronize'
/src/lib/tasks/db.rake:210:in `block in <main>'
Tasks: TOP => db:migrate
(通过运行带 --trace 的任务可查看完整跟踪)
natashaselvidge@Natashas-MacBook-Pro discourse % 

看起来您在开发环境中手动将 Redis gem 从 4.5.1 更新到了 4.6.0。Discourse 不支持该 gem 版本,因此您需要恢复到 Gemfile.lock 文件中使用的版本。

我们正在努力最终迁移到 redis-rb 4.6.0,但这需要大量工作:

4 个赞

哦哦,好的,非常感谢!我会试试的 :folded_hands:t3:

1 个赞

更新
供将来可能需要此信息的人参考 :hugs:
非常感谢,这解决了问题!我更新了我的 discourse,但 gem 文件中未指定版本。我像这样在末尾添加了版本号: gem ‘redis’, “~> 4.5.1”

我们在 Gemfile.lock 中指定它:

1 个赞

我认为是因为我删除了我的 Gemfile.lock 并进行了 bundle。然后我在 Github 上与 Redis 和 Sidekiq 的问题人员交谈,他们告诉我必须使用更新的版本。再次感谢!

This topic was automatically closed 30 days after the last reply. New replies are no longer allowed.