@@ -8,10 +8,10 @@ info: To determine the technical writer assigned to the Stage/Group associated w
...
@@ -8,10 +8,10 @@ info: To determine the technical writer assigned to the Stage/Group associated w
> This doc refers to <https://gitlab.com/gitlab-org/gitlab/blob/master/app/models/concerns/reactive_caching.rb>.
> This doc refers to <https://gitlab.com/gitlab-org/gitlab/blob/master/app/models/concerns/reactive_caching.rb>.
The `ReactiveCaching` concern is used for fetching some data in the background and store it
The `ReactiveCaching` concern is used for fetching some data in the background and storing it
in the Rails cache, keeping it up-to-date for as long as it is being requested. If the
in the Rails cache, keeping it up-to-date for as long as it is being requested. If the
data hasn't been requested for `reactive_cache_lifetime`, it will stop being refreshed,
data hasn't been requested for `reactive_cache_lifetime`, it stops being refreshed,
and then be removed.
and is removed.
## Examples
## Examples
...
@@ -35,38 +35,38 @@ class Foo < ApplicationRecord
...
@@ -35,38 +35,38 @@ class Foo < ApplicationRecord
end
end
```
```
In this example, the first time `#result` is called, it will return`nil`. However,
In this example, the first time `#result` is called, it returns`nil`. However,
it will enqueue a background worker to call `#calculate_reactive_cache` and set an
it enqueues a background worker to call `#calculate_reactive_cache` and set an
initial cache lifetime of 10 min.
initial cache lifetime of 10 minutes.
## How it works
## How it works
The first time `#with_reactive_cache` is called, a background job is enqueued and
The first time `#with_reactive_cache` is called, a background job is enqueued and
`with_reactive_cache` returns `nil`. The background job calls `#calculate_reactive_cache`
`with_reactive_cache` returns `nil`. The background job calls `#calculate_reactive_cache`
and stores its return value. It also re-enqueues the background job to run again after
and stores its return value. It also re-enqueues the background job to run again after
`reactive_cache_refresh_interval`. Therefore, it will keep the stored value up to date.
`reactive_cache_refresh_interval`. Therefore, it keeps the stored value up to date.
Calculations never run concurrently.
Calculations never run concurrently.
Calling `#with_reactive_cache` while a value is cached will call the block given to
Calling `#with_reactive_cache` while a value is cached calls the block given to
`#with_reactive_cache`, yielding the cached value. It will also extend the lifetime
`#with_reactive_cache`, yielding the cached value. It also extends the lifetime
of the cache by the `reactive_cache_lifetime` value.
of the cache by the `reactive_cache_lifetime` value.
Once the lifetime has expired, no more background jobs will be enqueued and calling
After the lifetime has expired, no more background jobs are enqueued and calling
`#with_reactive_cache`will again return `nil` - starting the process all over again.
`#with_reactive_cache`again returns `nil`, starting the process all over again.
### 1 MB hard limit
### 1 MB hard limit
`ReactiveCaching` has a 1 megabyte default limit. [This value is configurable](#selfreactive_cache_worker_finder).
`ReactiveCaching` has a 1 megabyte default limit. [This value is configurable](#selfreactive_cache_worker_finder).
If the data we're trying to cache has over 1 megabyte, it will not be cached and a handled `ReactiveCaching::ExceededReactiveCacheLimit` will be notified on Sentry.
If the data we're trying to cache has over 1 megabyte, it is not cached and a handled `ReactiveCaching::ExceededReactiveCacheLimit` is notified on Sentry.
## When to use
## When to use
- If we need to make a request to an external API (for example, requests to the k8s API).
- If we need to make a request to an external API (for example, requests to the k8s API).
It is not advisable to keep the application server worker blocked for the duration of
It is not advisable to keep the application server worker blocked for the duration of
the external request.
the external request.
- If a model needs to perform a lot of database calls or other time consuming
- If a model needs to perform a lot of database calls or other time consuming
calculations.
calculations.
## How to use
## How to use
...
@@ -99,13 +99,13 @@ Controller endpoints that call a model or service method that uses `ReactiveCach
...
@@ -99,13 +99,13 @@ Controller endpoints that call a model or service method that uses `ReactiveCach
not wait until the background worker completes.
not wait until the background worker completes.
- An API that calls a model or service method that uses `ReactiveCaching` should return
- An API that calls a model or service method that uses `ReactiveCaching` should return
`202 accepted` when the cache is being calculated (when `#with_reactive_cache` returns `nil`).
`202 accepted` when the cache is being calculated (when `#with_reactive_cache` returns `nil`).
- It should also
- It should also
[set the polling interval header](fe_guide/performance.md#realtime-components) with
[set the polling interval header](fe_guide/performance.md#realtime-components) with
`Gitlab::PollingInterval.set_header`.
`Gitlab::PollingInterval.set_header`.
- The consumer of the API is expected to poll the API.
- The consumer of the API is expected to poll the API.
- You can also consider implementing [ETag caching](polling.md) to reduce the server
- You can also consider implementing [ETag caching](polling.md) to reduce the server
load caused by polling.
load caused by polling.
### Methods to implement in a model or service
### Methods to implement in a model or service
...
@@ -113,17 +113,17 @@ These are methods that should be implemented in the model/service that includes
...
@@ -113,17 +113,17 @@ These are methods that should be implemented in the model/service that includes
#### `#calculate_reactive_cache` (required)
#### `#calculate_reactive_cache` (required)
- This method must be implemented. Its return value will be cached.
- This method must be implemented. Its return value is cached.
- It will be called by `ReactiveCaching` when it needs to populate the cache.
- It is called by `ReactiveCaching` when it needs to populate the cache.
- Any arguments passed to `with_reactive_cache`will also be passed to `calculate_reactive_cache`.
- Any arguments passed to `with_reactive_cache`are also passed to `calculate_reactive_cache`.
#### `#reactive_cache_updated` (optional)
#### `#reactive_cache_updated` (optional)
- This method can be implemented if needed.
- This method can be implemented if needed.
- It is called by the `ReactiveCaching` concern whenever the cache is updated.
- It is called by the `ReactiveCaching` concern whenever the cache is updated.
If the cache is being refreshed and the new cache value is the same as the old cache
If the cache is being refreshed and the new cache value is the same as the old cache
value, this method will not be called. It is only called if a new value is stored in
value, this method is not called. It is only called if a new value is stored in
the cache.
the cache.
- It can be used to perform an action whenever the cache is updated.
- It can be used to perform an action whenever the cache is updated.
### Methods called by a model or service
### Methods called by a model or service
...
@@ -134,22 +134,22 @@ the model/service.
...
@@ -134,22 +134,22 @@ the model/service.
#### `#with_reactive_cache` (required)
#### `#with_reactive_cache` (required)
-`with_reactive_cache` must be called where the result of `calculate_reactive_cache`
-`with_reactive_cache` must be called where the result of `calculate_reactive_cache`
is required.
is required.
- A block can be given to `with_reactive_cache`. `with_reactive_cache` can also take
- A block can be given to `with_reactive_cache`. `with_reactive_cache` can also take
any number of arguments. Any arguments passed to `with_reactive_cache` will be
any number of arguments. Any arguments passed to `with_reactive_cache` will be
passed to `calculate_reactive_cache`. The arguments passed to `with_reactive_cache`
passed to `calculate_reactive_cache`. The arguments passed to `with_reactive_cache`
will be appended to the cache key name.
are appended to the cache key name.
- If `with_reactive_cache` is called when the result has already been cached, the
- If `with_reactive_cache` is called when the result has already been cached, the
block will be called, yielding the cached value and the return value of the block
block is called, yielding the cached value and the return value of the block
will be returned by `with_reactive_cache`. It will also reset the timeout of the
is returned by `with_reactive_cache`. It also resets the timeout of the
cache to the `reactive_cache_lifetime` value.
cache to the `reactive_cache_lifetime` value.
- If the result has not been cached as yet, `with_reactive_cache`will return nil.
- If the result has not been cached as yet, `with_reactive_cache`return `nil`.
It will also enqueue a background job, which will call`calculate_reactive_cache`
It also enqueues a background job, which calls`calculate_reactive_cache`
and cache the result.
and caches the result.
-Once the background job has completed and the result is cached, the next call
-After the background job has completed and the result is cached, the next call
to `with_reactive_cache` will pick up the cached value.
to `with_reactive_cache` picks up the cached value.
- In the example below, `data` is the cached value which is yielded to the block
- In the example below, `data` is the cached value which is yielded to the block
given to `with_reactive_cache`.
given to `with_reactive_cache`.
```ruby
```ruby
classFoo<ApplicationRecord
classFoo<ApplicationRecord
...
@@ -170,16 +170,16 @@ given to `with_reactive_cache`.
...
@@ -170,16 +170,16 @@ given to `with_reactive_cache`.
#### `#clear_reactive_cache!` (optional)
#### `#clear_reactive_cache!` (optional)
- This method can be called when the cache needs to be expired/cleared. For example,
- This method can be called when the cache needs to be expired/cleared. For example,
it can be called in an `after_save` callback in a model so that the cache is
it can be called in an `after_save` callback in a model so that the cache is
cleared after the model is modified.
cleared after the model is modified.
- This method should be called with the same parameters that are passed to
- This method should be called with the same parameters that are passed to
`with_reactive_cache` because the parameters are part of the cache key.
`with_reactive_cache` because the parameters are part of the cache key.
#### `#without_reactive_cache` (optional)
#### `#without_reactive_cache` (optional)
- This is a convenience method that can be used for debugging purposes.
- This is a convenience method that can be used for debugging purposes.
- This method calls `calculate_reactive_cache` in the current process instead of
- This method calls `calculate_reactive_cache` in the current process instead of
in a background worker.
in a background worker.
### Configurable options
### Configurable options
...
@@ -188,19 +188,19 @@ There are some `class_attribute` options which can be tweaked.
...
@@ -188,19 +188,19 @@ There are some `class_attribute` options which can be tweaked.
#### `self.reactive_cache_key`
#### `self.reactive_cache_key`
- The value of this attribute is the prefix to the `data` and `alive` cache key names.
- The value of this attribute is the prefix to the `data` and `alive` cache key names.
The parameters passed to `with_reactive_cache` form the rest of the cache key names.
The parameters passed to `with_reactive_cache` form the rest of the cache key names.
- By default, this key uses the model's name and the ID of the record.
- By default, this key uses the model's name and the ID of the record.