Project

General

Profile

Actions

Patch #8222

open

Setting model should use Rails.cache instead of class variable

Added by Jan from Planio www.plan.io almost 13 years ago. Updated almost 8 years ago.

Status:
Needs feedback
Priority:
Normal
Assignee:
-
Category:
Rails support
Start date:
2011-04-25
Due date:
% Done:

0%

Estimated time:

Description

The Setting model uses two local class variables called @cached_settings and @cached_cleared_on to prevent reloading the settings from database and/or settings.yml at every request.

Rails has been supporting different cache implementations for some time now, why not let the Setting model use them as well? Besides the fact that this will help with tenant switching which we've implemented for Planio, it should also improve Redmine's performance when running it with more than processes and a custom cache mechanism, like memcached.

A patch is attached.


Files

Actions #1

Updated by Etienne Massip almost 13 years ago

  • Target version set to Candidate for next major release
Actions #2

Updated by Anuj Sapkota almost 13 years ago

  • Status changed from New to Resolved
  • Assignee set to Jim Mulholland

(spam)

Actions #3

Updated by Jan from Planio www.plan.io almost 13 years ago

?? :-)

how were you even able to update this ticket?

Actions #4

Updated by Etienne Massip almost 13 years ago

  • Status changed from Resolved to New
  • Assignee deleted (Jim Mulholland)
Actions #5

Updated by Toshi MARUYAMA almost 13 years ago

This is patch series from https://www.chiliproject.org/issues/350

Actions #6

Updated by Etienne Massip almost 13 years ago

Chiliproject looks to add a lot of code complexity and we should not need to change tests, Jan's patch, on the other side, is pretty clean.

Should settings be cached separately or could the full Hash of Settings be cached instead ?

Actions #7

Updated by Jean-Baptiste Barth over 9 years ago

  • Status changed from New to Needs feedback
  • Assignee set to Jean-Baptiste Barth

There's at list one possible gotcha with this: Rails.cache defaults to a file backend which leads to potential problems if permissions are not set correctly, or, say, the production server disk is full (which never happens hey?).

In those cases you'll have this behaviour:

>> Rails.cache.write("foo", "bar")
Errno::EACCES: Permission denied - /Users/jbbarth/Projects/redmine/redmine-test/tmp/cache/145
...

... which will break the application completely, whereas the current code would work.

Also it seems even Rails will issue a "read" operation on the file each time we call Rails.cache.read/fetch. Maybe not a good idea on instances with lots of users, this will lead to unecessary IOs.

Otoh I admit it would be nice to rely on Rails.cache if we configure a memory store or a smart database (e.g. memcached or redis).

Ideally, I'd like to make this optional. What do you think ?

Actions #8

Updated by Jan from Planio www.plan.io over 9 years ago

Jean-Baptiste Barth wrote:

There's at list one possible gotcha with this: Rails.cache defaults to a file backend which leads to potential problems if permissions are not set correctly, or, say, the production server disk is full (which never happens hey?).

In those cases you'll have this behaviour: [...]
... which will break the application completely, whereas the current code would work.

Also it seems even Rails will issue a "read" operation on the file each time we call Rails.cache.read/fetch. Maybe not a good idea on instances with lots of users, this will lead to unecessary IOs.

Oh, I was under the impression that the default store would be MemoryStore, but I can confirm it's FileStore in a vanilla Redmine - you're absolutely right. However, Redmine already uses Rails Caching since r10844. So, I would say, it wouldn't hurt to make more use of it.

Otoh I admit it would be nice to rely on Rails.cache if we configure a memory store or a smart database (e.g. memcached or redis).

Ideally, I'd like to make this optional. What do you think ?

Hmmmm, personally I would not make this optional. I think the current caching mechanism of the Setting model might be a relic from times where Rails Caching wasn't available, and that should be replaced. To make it more obvious that caching is used there and less confusing for cases where in-memory caching is not the best option, I would really like to move this to Rails.cache.

I agree that a default to MemoryStore for Redmine in general may be a good idea. But then again, it may also be nice just to follow Rails' defaults. I think users who deploy Redmine in production will have their own opinion on Caching anyways and will want to define their favorite cache store. (And if they do, I think they would be confused by the fact that the current Setting cache does not follow their configuration for Rails.cache but uses its own in-memory mechanism...)

Actions #9

Updated by Gregor Schmidt almost 8 years ago

I just stumbled upon this issue and had to rebase the change onto the lastest trunk anyway. So I thought, I'd give this a bump.

Attached you may find an updated version of Jan's patch based on r15494.


Addendum: The code currently used at Planio also dups the value fetched from the cache using the following code:

  def self.[](name)
    v = Rails.cache.fetch("redmine-settings/#{name}") { find_or_default(name).value }
    v.dup rescue v
  end

Talking to coworkers and inspecting the history of the file suggest, that this was done to avoid getting frozen objects out of the cache. Something similar was discussed in the matching ChiliProject issue.

I was trying to verify that, but I wasn't able not reproduce that behavior. FileStore, MemoryStore and MemCacheStore all seem to return the objects unfrozen. (FileStore and MemCacheStore return new instances for each each call, MemoryStore returns the same object - similar to the ad-hoc cache currently in place.) But since this patch is already 5 years old, it might as well be the case, that implementations have changed since then.

Nevertheless, I wanted to make sure, that everybody is aware of that.

Actions #10

Updated by Jan from Planio www.plan.io almost 8 years ago

  • Assignee deleted (Jean-Baptiste Barth)
  • Target version changed from Candidate for next major release to Candidate for next minor release
Actions #11

Updated by Jan from Planio www.plan.io almost 8 years ago

  • File deleted (setting_rb_use_rails_cache.patch)
Actions #12

Updated by Jean-Philippe Lang almost 8 years ago

The Rails default :file_store seems to be OK for most use cases but I agree with Jean-Baptiste and would definitely not use it to cache settings that are heavily used (some requests generate > 1000 setting reads).

And the problem with the proposed patch is that it doesn't work if the cache is not shared between Redmine processes (eg. if using :memory_store):
redmine-settings-cleared-on is not initialized and Setting.check_cache never clears the cache (redmine-settings-cleared-on would only be set in Setting.clear_cache).

Actions #13

Updated by Jean-Philippe Lang almost 8 years ago

From a performance point of view:

With patch applied:

irb(main):004:0> Benchmark.ms {1000.times {Setting.app_title}}
=> 218.401

With patch applied and Rails cache store set to :memory_store

irb(main):004:0> Benchmark.ms {1000.times {Setting.app_title}}
=> 31.2

With current implementation:

irb(main):004:0> Benchmark.ms {1000.times {Setting.app_title}}
=> 0.0
Actions #14

Updated by Gregor Schmidt almost 8 years ago

Thanks a lot for reviewing this discussion and the patch.

Jean-Philippe Lang wrote:

And the problem with the proposed patch is that it doesn't work if the cache is not shared between Redmine processes (eg. if using :memory_store):
redmine-settings-cleared-on is not initialized and Setting.check_cache never clears the cache (redmine-settings-cleared-on would only be set in Setting.clear_cache).

Attached you may find an updated patch, which addresses the bug you described.

Actions #15

Updated by Jan from Planio www.plan.io almost 8 years ago

Jean-Philippe Lang wrote:

From a performance point of view: ...

Cannot argue with that. 0.0ms is a very convincing argument ;-) In that case, using Rails.cache would maybe only be beneficial if it's in fact being shared between several processes or app servers. In that case, maybe coming back to Jean-Baptiste's proposal to make this optional could be a way forward.

Of course, if you feel that this doesn't really benefit Redmine, we can also just close this issue...

Actions

Also available in: Atom PDF