Cache inconsistencies when using RDB

Created on 12 June 2023, almost 2 years ago
Updated 17 July 2023, over 1 year ago

Problem/Motivation

We recently encountered a problem on a server using RDB.
Redis crashed and was then restarted by the hosting provider.
When restarted, it loaded a RDB snapshot from several weeks before.
Because of this, every Drupal call to read config was getting outdated config from the cache, so it made Drupal behave erratically.

Because the _redis_last_delete_all is stored in Redis, it was also restored from the backup.
So Drupal didn't know it was using outdated cache.

Running drush cr manually fixed the problem (and we also asked the hosting provider to disable RDB).

(I see this problem was also mentioned here .)

Steps to reproduce

In theory:

  1. Enable RDB.
  2. Wait for Redis save a RDB snapshot.
  3. Change some Drupal config.
  4. Restore cache from the RDB snapshot.
  5. Drupal now uses outdated config from the cache.

Proposed resolution

I'm not sure but I guess the _redis_last_delete_all value could be stored in the SQL database (for example in a state).
Or maybe a hook_requirements() could warn that using RDB can lead to outdated cache?

🐛 Bug report
Status

Needs work

Version

1.5

Component

Code

Created by

🇫🇷France prudloff Lille

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

  • Issue created by @prudloff
  • 🇨🇭Switzerland berdir Switzerland

    Why are you using a persistent state for redis cache at all?

    FWIW, if you'd import an old database dump you would also have a mess, if you have bad data, you get bad results, I don't really see a way around that.

  • 🇫🇷France prudloff Lille

    We did not make this choice and we do recommend to have RDB disabled.
    But some hosting providers have it enabled by default (and sometimes don't advertise it), that's how we got bit by that.

    Also, I guess there can be some legitimate use cases. For example, if the Drupal site uses the same Redis server as another website that does need to store persistent data in Redis (some other CMSs store queues in Redis for example).
    And according to this doc, it seems that RDB is enabled by default if the save option is not explicitly defined, so other people could be impacted by this and not know it.

    We mitigated this issue by adding a custom hook_requirements() that checks if RDB is enabled:

    /** @var \Drupal\redis\ClientFactory $redisFactory */
    $redisFactory = \Drupal::service('redis.factory');
    
    /** @var \Redis $client */
    $client = $redisFactory::getClient();
    
    /** @var string[] $config */
    $config = $client->config('GET', 'save');
    
    if (!empty($config['save'])) {
      // Add warning.
    }
    
  • 🇨🇭Switzerland berdir Switzerland

    We have a status page that already has some checks and recommendations, patches welcome to check for the save setting and add a warning that recommends to disable it. That's the only actionable option that I see here.

    If you have other things you do need to store persistently, I'd set up separate server for that.

  • @prudloff opened merge request.
  • Status changed to Needs work over 1 year ago
  • 🇫🇷France prudloff Lille

    The MR adds a warning when using PhpRedis. We don't use Predis so I'm not sure how to check the config with it.

Production build 0.71.5 2024