Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Force snapshot under memory pressure #25685

Closed
Tracked by #25683
pauldix opened this issue Dec 19, 2024 · 8 comments · Fixed by #25765 or #25767
Closed
Tracked by #25683

Force snapshot under memory pressure #25685

pauldix opened this issue Dec 19, 2024 · 8 comments · Fixed by #25765 or #25767
Assignees
Labels

Comments

@pauldix
Copy link
Member

pauldix commented Dec 19, 2024

By default, the server will only snapshot after it receives 900 WAL files and it only tries to snapshot 600 of them. Under high load this can lead to runaway memory use in the buffer.

We'll need a configuration option to set a memory threshold for when we should force a snapshot. Ideally, the threshold would be based on the size of the QueryableBuffer, but that might be inaccurate and expensive to measure. We could also use the process memory size. Let's try the QueryableBuffer option and see how it works.

Set the default value to 70% of whatever the detected system memory is.

We should have a background task that checks every 10s to see if the queryable buffer has hit this threshold. If it has, we should force a snapshot in which everything in the queryable buffer will be persisted to Parquet and all WAL files will be cleared out.

We can develop something a little more precise and less like a sledgehammer later.

@praveen-influx
Copy link
Contributor

@pauldix few thoughts/queries,

Ideally, the threshold would be based on the size of the QueryableBuffer, but that might be inaccurate and expensive to measure. We could also use the process memory size. Let's try the QueryableBuffer option and see how it works.

At a high level we have 2 options as you suggested there,

  • Use QueryableBuffer mem usage
  • Use process memory usage

I'm wondering if we could use process memory usage (because it's less intrusive) - I think this probably masks real buffer usage and it could be the caches using most of the memory, however it's fairly straight forward to trigger the snapshot based on whole process memory usage.

If we'd like to still proceed with the memory tracking for QueryableBuffer, I need to introduce a method to find the size of fields embedded in QueryableBuffer, mainly BufferState, LastCache, MetaCache, Persister and PersistedFiles (I think). This will be a more intrusive approach but possibly will give a more accurate reading. I also wonder if there's a way to track num bytes (at a high level within QueryableBuffer itself) added to each of these structs when a buffer op is played, so that we don't need to recursively go check the bytes consumed by caches for example. Also, deduct that usage when buffer is flushed or cache evicts entries.

Would you still like to proceed with tracking memory for QueryableBuffer and also do you think tracking bytes (and/or rows) will be better (like avoiding locks whilst calculating size of nested data structures)?

@pauldix
Copy link
Member Author

pauldix commented Dec 24, 2024

So the TableBuffer already has a function to calculate its size here: https://github.com/influxdata/influxdb/blob/main/influxdb3_write/src/write_buffer/table_buffer.rs#L165-L177

So you could get a size by walking the databases and tables and summing it all together. That might be a good place to start. The tricky bit is it won't be exact and ultimately it's the process memory that matters since that's what'll trigger an OOM kill.

So I'm honestly not sure what would be the best approach here. The other thing about process memory is that it can spike if there's some expensive query and we wouldn't want that triggering a premature persistence. So you'd need to track some sort of moving average if using that.

I think, use the QueryableBuffer size as the thing that triggers it for now and we can adjust after some testing.

@praveen-influx
Copy link
Contributor

Ok - thanks, do we need to check the size of last cache/meta cache as well as QueryableBuffer has pointers to them? But given flushing the buffer (for snapshot) doesn't change the size of those caches I'm assuming we don't need to take them into account?

@pauldix
Copy link
Member Author

pauldix commented Dec 27, 2024

We don't need to take the size of those caches into account for this.

@praveen-influx
Copy link
Contributor

Is it OK to flush the WAL buffer when we force the snapshot? I can then follow the same path as background_wal_flush but not sure if I should leave that alone and wire in forcing snapshot directly by getting the snapshot details from SnapshotTracker directly. Using the same path as background_wal_flush looks like the simplest thing to do but I'm not sure if I'm overlooking any details.

@pauldix
Copy link
Member Author

pauldix commented Dec 27, 2024

I think forcing a wal flush is fine

praveen-influx added a commit that referenced this issue Dec 31, 2024
praveen-influx added a commit that referenced this issue Dec 31, 2024
- The core of the change is to introduce another method
  `force_flush_buffer` in `Wal` trait. This gives a handle to choose
  when to kick off snapshot.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `force_flush_buffer`. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of the code went on to make sure the calls made
  via `Wal` trait to flush buffer and cleanup any snapshot is reused
  across both branches (forcing snapshot and normal wal buffer flush)

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_file_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_file_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_file_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_file_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_sequence_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 2, 2025
- The main change is to detach wal and snapshot, in a way all 3 of the
  following things can happen
    - flush the wal buffer only (already handled, before this commit)
    - flush wal buffer and snapshot (already handled, before this commit)
    - snapshot without flushing wal buffer (introduced in this commit)
  This is achieved by introducing another method `snapshot` in
  `WalFileNotifier` trait. The main dependency between wal and snapshot
  is the `wal_file_number`, since this is tracked in `SnapshotTracker`
  separately we can switch to using `SnapshotTracker`'s
  `last_wal_sequence_number` instead of the one that comes through the
  `WalContents`.
- A higher level background loop is introduced that checks the overall
  table buffer size every `N` seconds and if it is greater than a
  threshold (`X`) then it calls `snapshot` method. Both `N` and `X` are
  configurable through cli. `N` defaults to 10s and `X` defaults to 70%
- Some refactoring of code so that existing methods can be reused when
  only snapshotting

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
@praveen-influx
Copy link
Contributor

@pauldix - just wondering if instead of trying to force snapshotting outside of flush interval, would it be better to expose something like is_query_buff_at_max_threshold or something along those lines that returns a bool in WalFileNotifier directly. Then internally query buffer (impl WalFileNotifier) can have a separate loop (running in background) that updates an atomic bool whenever it's over the threshold.

The issue is there is an additional atomic bool load for every flush interval but when it's in contention this might be cheaper. Just a thought - we can explore this even later

praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 8, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit checks the memory consumed by query buffer (buffer state),
and if it's over the threshold forces a buffer flush which in turn
triggers the snapshotting process.

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
- This commit changes the functionality to allow snapshots to happen even when
  wal buffer is empty. For snapshots wal periods are still required but
  not the wal buffer. To allow this, we write a no-op into wal file with
  snapshot details. This enables force snapshotting functionality

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit checks the memory consumed by query buffer (buffer state),
and if it's over the threshold forces a buffer flush which in turn
triggers the snapshotting process.

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit allows checking memory in the background and force
snapshotting if query buffer size is > mem threshold. This hooks into
the function (`force_flush_buffer`) to achieve it.

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit allows checking memory in the background and force
snapshotting if query buffer size is > mem threshold. This hooks into
the function (`force_flush_buffer`) to achieve it.

closes: #25685
@pauldix
Copy link
Member Author

pauldix commented Jan 9, 2025

@praveen-influx I anticipate that the computation of the size might be a little expensive, so we should be doing that only while holding a read lock on the buffer. I'm not sure there's any difference between having a background loop check that occasionally and call force snapshot vs. having it update some protected counter or flag that gets checked on flush.

Something to revisit when we have time during the alpha & beta period to do some real refactoring and more extensive testing in this part of the code base

praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit allows checking memory in the background and force
snapshotting if query buffer size is > mem threshold. This hooks into
the function (`force_flush_buffer`) to achieve it.

closes: #25685
praveen-influx added a commit that referenced this issue Jan 9, 2025
This commit allows checking memory in the background and force
snapshotting if query buffer size is > mem threshold. This hooks into
the function (`force_flush_buffer`) to achieve it.

closes: #25685
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment