При попытке миграции базы данных я постоянно получаю эту ошибку Redis. Это установка Discourse через Docker. Пожалуйста, помогите.
У вас установка на 2 контейнера или Redis размещён где-то ещё? У вас Redis версии 6.2?
Это моя версия Redis 6.2.6, и у меня есть только один контейнер Redis в Docker, а также несколько контейнеров discourse_dev.
Я также запросил помощь на GitHub проекта sidekiq.
Вот что выводится при запуске в моём Docker-контейнере:
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 перемещены из настроек сайта в глобальные настройки. Удалите переопределение в интерфейсе настроек сайта и используйте файл конфигурации или переменные окружения для установки глобальных настроек. (удаление в 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')
Пакетное выполнение команд в экземпляре 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!
Redis::CommandError: ERR неизвестная команда `synchronize`, аргументы начинаются с:
/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>'
Задачи: TOP => db:migrate
(Полный трассировочный вывод можно получить, запустив задачу с флагом --trace)
natashaselvidge@Natashas-MacBook-Pro discourse %
Похоже, вы вручную обновили gem Redis в вашей среде разработки с версии 4.5.1 до 4.6.0. Discourse не работает с этой версией gem, поэтому вам нужно откатиться к версии, указанной в файле Gemfile.lock.
Мы работаем над тем, чтобы в будущем перейти на redis-rb 4.6.0, но это требует значительных усилий:
Ох, хорошо, большое спасибо! Я попробую это ![]()
Обновление
Для тех, кому это может понадобиться в будущем ![]()
Огромное спасибо, всё заработало! Я обновил Discourse, но в файле gem версия не была указана. Я добавил номер версии в конец следующим образом: gem ‘redis’, “~> 4.5.1”
Мы указываем это в файле Gemfile.lock:
Думаю, это потому что я удалил Gemfile.lock и сделал bundle install. Затем я общался с разработчиками, занимающимися проблемами Redis и Sidekiq на GitHub, и они сказали, что мне нужно использовать обновлённую версию. Спасибо ещё раз!
