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

Reliable storage lock #2014

Merged
merged 6 commits into from
Dec 9, 2024
Merged

Reliable storage lock #2014

merged 6 commits into from
Dec 9, 2024

Conversation

IvoDD
Copy link
Collaborator

@IvoDD IvoDD commented Nov 25, 2024

What does this implement or fix?

Introduces a reliable storage lock

Introduces a new ReliableStorageLock and ReliableStorageLockGuard to
be used as a slower but more reliable alternative to the existing
StorageLock.

It uses the new If-None-Match atomic put operations in S3.

First commit (Reliable storage lock):

  • Upgrades the aws-sdk-cpp in vcpkg (which needed a few additions
    because of some problematic dependencies)
  • Adds write_if_none_match capability to AsyncStore's S3 and to
    InMemoryStore
  • Logic for ReliableStorageLock
  • C++ tests using the InMemoryStore

Second commit (Real S3 storage python tests for ReliableStorageLock)

  • Adds a ReliableStorageLockManager which exposes functions to aquire and free the lock in python. (The guard structure is unusable for python)
  • Adds a new storage lock test with the existing real_s3_version_store fixture.

Any other comments?

Checklist

Checklist for code changes...
  • Have you updated the relevant docstrings, documentation and copyright notice?
  • Is this contribution tested against all ArcticDB's features?
  • Do all exceptions introduced raise appropriate error messages?
  • Are API changes highlighted in the PR description?
  • Is the PR labelled as enhancement or bug so it appears in autogenerated release notes?

@IvoDD IvoDD force-pushed the reliable-storage-lock branch 5 times, most recently from 776bc2d to e140bca Compare November 25, 2024 18:41
@IvoDD
Copy link
Collaborator Author

IvoDD commented Nov 26, 2024

Evidence of real s3 storage tests passing here

@IvoDD IvoDD force-pushed the reliable-storage-lock branch from 3add5e2 to 7f4fac4 Compare November 26, 2024 14:24
@@ -60,6 +60,7 @@ KeyData get_key_data(KeyType key_type) {
STRING_REF(KeyType::APPEND_REF, aref, 'a')
STRING_KEY(KeyType::MULTI_KEY, mref, 'm')
STRING_REF(KeyType::LOCK, lref, 'x')
STRING_REF(KeyType::SLOW_LOCK, lref, 'x')
Copy link
Collaborator

@poodlewars poodlewars Nov 27, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This clashes with the character and prefix for LOCK above which is very odd. Is this intentional? Doesn't it mean we'd list out both sorts of lock when we iterate type?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They all have to be unique, otherwise bad things will happen. Also, I'm not sure why we're characterising this as a slow lock, since if anything it will be faster than the other lock which required variable (and ultimately unknowable) wait periods to ensure that it really was the holder of the lock.

My preference would be that the eventual lock key (the copy destination) is just a normal KeyType::LOCK, and the one that gets copied to it (the first one written is called a PENDING_LOCK or a LOCK_ATTEMPT or something like that

Copy link
Collaborator Author

@IvoDD IvoDD Dec 2, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, renaming it to ATOMIC_LOCK (with alref and A) as it relies on storage specific atomic operations. I will eventually replace it with the If-Match atomic aws s3 operations which would make it more meaningful.

I thought the letter here is just for users to easily identify key types from their name. Listing works by putting all keys from same key type in a directory (e.g. "tdata") I think, so I don't this this would cause issues.

If this is indeed an issue (notice that both the existing TOMBSTONE and LOCK use the same x) we should fix them as well.

@@ -57,7 +57,8 @@ class S3ClientWrapper {
virtual S3Result<std::monostate> put_object(
const std::string& s3_object_name,
Segment&& segment,
const std::string& bucket_name) = 0;
const std::string& bucket_name,
bool if_none_match = false) = 0;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I hate bools on public APIs. Could we have two different public methods without the bool that push down to a single private method with the bool or something?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

also bools tend to multiply out of control, it's preferable to use an enum, that way callers have to have a name on them (WriteConditions::IF_NONE_MATCH is way more self-explanatory at the call site than 'true')

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, will use a PutHeader::IF_NONE_MATCH enum

@@ -83,7 +83,8 @@ S3Result<Segment> MockS3Client::get_object(
S3Result<std::monostate> MockS3Client::put_object(
const std::string &s3_object_name,
Segment &&segment,
const std::string &bucket_name) {
const std::string &bucket_name,
bool if_none_match[[maybe_unused]]) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We should extend the mock client so it can give us precondition failed response codes shouldn't we?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah agreed, I'll fix that. We didn't use it for tests but thinking more it might have been better to test with a MockS3Client rather than with the InMemoryStore but it would require some extra work to protect the in memory nature of it with some mutexes.

Will add this to a combined ticket for rework the lock to use If-Match.

@@ -14,4 +16,11 @@ struct PilotedClock {
}
};

struct PilotedClockNoAutoIncrement {
Copy link
Collaborator

@poodlewars poodlewars Nov 27, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think there's already a clock type in the codebase that works like this - ManualClock?

@@ -1,3 +1,5 @@
#pragma once
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yuck

// provide a slower but more reliable lock than the StorageLock. It should be completely consistent unless a process
// holding a lock get's paused for times comparable to the lock timeout.
// It lock follows the algorithm described here:
// https://www.morling.dev/blog/leader-election-with-s3-conditional-writes/
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe we should save of copy of this blog somewhere, will suck to have no docs of how this works if it vanishes

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given that we'll soon replace the locking with If-Match based locks, we can probably skip this.

@IvoDD IvoDD force-pushed the reliable-storage-lock branch 4 times, most recently from 3b41a46 to 614d8c2 Compare November 27, 2024 18:32

// The ReliableStorageLock is a storage lock which relies on atomic If-None-Match Put and ListObject operations to
// provide a slower but more reliable lock than the StorageLock. It should be completely consistent unless a process
// holding a lock get's paused for times comparable to the lock timeout.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

get's should be gets


}

#include "arcticdb/util/reliable_storage_lock.tpp"
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We normally put all the templated definitions straight in the .hpp file. It's an arbitrary choice, so I think we should be consistent with the rest of the codebase here.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The other alternative is to put them in reliable_storage_lock-inl.hpp, which is how we usually indicate that really this is implementation but it's templated implementation. You can look at any of the other -inl.hpp files for the usual mechanism

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As in normally we include them at the bottom of the header file, and ensure that they're not included anywhere else, that way you get file separation, but you don't need to know about the additional file in your cpp files, you just include the header as normal

template <class ClockType>
ReliableStorageLock<ClockType>::ReliableStorageLock(const std::string &base_name, const std::shared_ptr<Store> store, timestamp timeout) :
base_name_(base_name), store_(store), timeout_(timeout) {
auto s3_timeout = ConfigsMap::instance()->get_int("S3Storage.RequestTimeoutMs", 200000) * ONE_MILLISECOND;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems quite weird to have references to S3 in this layer of the code. Also didn't we discover some quite odd properties of the s3 request timeout in the sense that it does not prevent s3 requests taking longer than the value set but instead definitions a window of time in which we expect a certain number of bytes to be transferred?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah we'll want to use it for azure etc which already has similar operations, it would be better to pass it in as a policy from the storage where we just expect a function that is void() that does whatever we need

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed, it would be best to pass some sort of MaxTimeoutPolicy which allows us to configure the timeout of the storage. And yeah the RequestTimeout was indeed some period of time to determine a threshold for transfer speed.

I've just removed the check for now and will add to the follow up ticket to be able to configure and enforce a certain maximum timeout for all stores.


namespace lock {

using Epoch = uint64_t;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I know it's the nomenclature that the blog used, but I find this Epoch name really confusing. I think most people would expect it to be some sort of timestamp, but it's a counter (and then we have timestamps in the segment itself, to add to the confusion).

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, used AcquiredLockId

return;
}
auto lock_stream_id = get_stream_id(held_lock_epoch);
auto expiration = ClockType::nanos_since_epoch(); // Write current time to mark lock as expired as of now
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Managing the expiry with this process' clock isn't right is it? The blog uses the last modification time on S3. This is probably something we can improve afterwards.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah you're right. The difference is not big though. Just 2x the clock skew (if we e.g. expect clock skew to be 100ms for a process this would enstill a 200ms total skew between the two processes, otherwise if we use the s3 time as a source of truth we would suffer only the 100ms skew).

Let's do this together with the If-Match rework (which will also require passing in the ETag from s3 to the Store API and we can do a similar change to use the last-modified time)

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think that's reasonable on the basis that clock drift can still cause locking failures even if we make the change I suggested.

Clock::time_ = 10;
ASSERT_EQ(lock2.try_take_lock(), std::nullopt);
Clock::time_ = 19;
ASSERT_EQ(lock1.try_take_lock(), std::nullopt);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The std::nullopt assertions should all be for both lock1 and lock2, right? We should document that the lock API is not re-entrant

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks, agreed done

that->lock_lost_ = true;
});
auto value_before_sleep = cnt_;
// std::cout<<"Taken a lock with "<<value_before_sleep<<std::endl;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Remove the print statements

cpp/vcpkg.json Outdated
{
"name": "aws-sdk-cpp",
"$version reason": "Minimum version in the baseline that works with aws-c-io above.",
"version>=": "1.11.405",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why isn't the version down in the overrides section like all the other packages?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I needed to enforce fairly new versions of all aws packages because otherwise newer versions of aws-sdk-cpp would not compile.

To do this I used version>= so in the future if aws packages fix their vcpkg dependencies we would not have to update all aws overrides when we want o upgrade the aws-sdk-cpp.

I think if we use an override we don't allow vcpkg to resolve dependency versions for us, but given it's currently broken either way I'll just add an override only for aws-sdk-cpp. I'll leave the other aws packages with version>= to not require upgrading all overrides when we only want to upgrade the aws-sdk-cpp


TEST(ReliableStorageLock, StressMultiThreaded) {
// It is hard to use a piloted clock for these tests because the folly::FunctionScheduler we use for the lock
// extensions doesn't support a custom clock. Thus this test will need to run for about 2 minutes.
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's useful to keep test_unit_arcticdb running fast. Might be better to add this stress test to a new cmake target. Can be done as a follow up.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree it's useful to separate slower stress tests. I rememember other stress tests e.g. some of the tests in test_slab_alloc.cpp were also quite slow, we should probably move them together.

Adding a note to fix in the If-match cleanup.


read_df = lib.read(symbol).data
expected_df = pd.DataFrame({"col": [num_processes]})
assert_frame_equal(read_df, expected_df)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also assert on the version number after all these writes?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've noticed when running these that the persistent storage might not get cleaned up after a previous test run especially when running locally. So I think it might be possible to do these writes on a library which already has the counter written with some arbitrary version. I.e. it's possible we get a version 200 because the test was run before.

I think the counter being equal to 100 is essentially the same assertions as writing 100 versions, so there isn't much benefit and it makes local testing slightly more annoying. This ifine for the CI I think. What do you think?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Would be simple to do a force delete symbol at the start of the test?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah didn't know about the force_delete_symbol API. Done, thanks!

@@ -60,6 +60,7 @@ KeyData get_key_data(KeyType key_type) {
STRING_REF(KeyType::APPEND_REF, aref, 'a')
STRING_KEY(KeyType::MULTI_KEY, mref, 'm')
STRING_REF(KeyType::LOCK, lref, 'x')
STRING_REF(KeyType::SLOW_LOCK, lref, 'x')
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

They all have to be unique, otherwise bad things will happen. Also, I'm not sure why we're characterising this as a slow lock, since if anything it will be faster than the other lock which required variable (and ultimately unknowable) wait periods to ensure that it really was the holder of the lock.

My preference would be that the eventual lock key (the copy destination) is just a normal KeyType::LOCK, and the one that gets copied to it (the first one written is called a PENDING_LOCK or a LOCK_ATTEMPT or something like that

@@ -57,7 +57,8 @@ class S3ClientWrapper {
virtual S3Result<std::monostate> put_object(
const std::string& s3_object_name,
Segment&& segment,
const std::string& bucket_name) = 0;
const std::string& bucket_name,
bool if_none_match = false) = 0;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

also bools tend to multiply out of control, it's preferable to use an enum, that way callers have to have a name on them (WriteConditions::IF_NONE_MATCH is way more self-explanatory at the call site than 'true')


}

#include "arcticdb/util/reliable_storage_lock.tpp"
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The other alternative is to put them in reliable_storage_lock-inl.hpp, which is how we usually indicate that really this is implementation but it's templated implementation. You can look at any of the other -inl.hpp files for the usual mechanism


}

#include "arcticdb/util/reliable_storage_lock.tpp"
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As in normally we include them at the bottom of the header file, and ensure that they're not included anywhere else, that way you get file separation, but you don't need to know about the additional file in your cpp files, you just include the header as normal

const auto EXTENDS_PER_TIMEOUT = 5u;

inline StreamDescriptor lock_stream_descriptor(const StreamId &stream_id) {
return StreamDescriptor{stream_descriptor(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think it needs the other constructor as the function already returns a stream descriptor

template <class ClockType>
ReliableStorageLock<ClockType>::ReliableStorageLock(const std::string &base_name, const std::shared_ptr<Store> store, timestamp timeout) :
base_name_(base_name), store_(store), timeout_(timeout) {
auto s3_timeout = ConfigsMap::instance()->get_int("S3Storage.RequestTimeoutMs", 200000) * ONE_MILLISECOND;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah we'll want to use it for azure etc which already has similar operations, it would be better to pass it in as a policy from the storage where we just expect a function that is void() that does whatever we need

}

template <class ClockType>
StreamId ReliableStorageLock<ClockType>::get_stream_id(Epoch e) const {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

e is commonly used for error, no harm in calling it 'epoch'

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Used lock_id because I renamed Epoch to AcquiredLockId

@IvoDD
Copy link
Collaborator Author

IvoDD commented Dec 2, 2024

Some of the discussed follow up fixes will be done when addressing #2028

@IvoDD IvoDD force-pushed the reliable-storage-lock branch 4 times, most recently from e21f7a9 to 1bc9ddb Compare December 2, 2024 14:28
@IvoDD IvoDD force-pushed the reliable-storage-lock branch 2 times, most recently from 6843bd9 to dc7204a Compare December 3, 2024 09:04
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wextra")
# Required to be able to include headers from glog since glog 0.7
# See: https://github.com/google/glog/pull/1030
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wextra -DGLOG_USE_GLOG_EXPORT")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You don't need to change this but setting the CMAKE_CXX_FLAGS is an outdated approach. Unfortunately most of our code is setup like this :(

The correct thing to do is to use target_compile_definitions

@IvoDD IvoDD force-pushed the reliable-storage-lock branch 2 times, most recently from d7ed925 to 39b460d Compare December 4, 2024 16:05
@IvoDD IvoDD mentioned this pull request Dec 6, 2024
5 tasks
@IvoDD IvoDD force-pushed the reliable-storage-lock branch 2 times, most recently from ea698e2 to 92c203a Compare December 6, 2024 12:08
IvoDD added 6 commits December 9, 2024 14:28
Introduces a new `ReliableStorageLock` and `ReliableStorageLockGuard` to
be used as a slower but more reliable alternative to the existing
`StorageLock`.

It uses the new If-None-Match atomic put operations in S3.

This commit:
- Upgrades the aws-sdk-cpp in vcpkg (which needed a few additions
  because of some problematic dependencies)
- Adds `write_if_none_match` capability to `AsyncStore`'s S3 and to
  `InMemoryStore`
- Logic for `ReliableStorageLock`
- C++ tests using the `InMemoryStore`

Follow up commit will introduce a python integration test with real aws
s3.
Adds a real s3 storage test (currently to be run with persistent storage
tests mark) for the lock.
Currently all backends are unsupported apart from S3 (for which only some providers like AWS support it).

Unfotrunately it's impossible to differentiate between aws and e.g. vast
backends apart from looking at the endpoint which can be subject to
rerouting etc.

This commit also reworks the Guard to work only with aquired locks.
- Addressing review comments on ReliableStorageLock pr
- Mostly renames
- Longer timeout before lock cleanup
- Fixes mamba dependencies
This will be useful if we want to construct a ReliableStorageLockGuard
but decide later what to do in case a lock is lost.

Also changes ReliableStorageLockGuard to hold a copy of the lock instead
of just a reference. The lock is a cheap object to copy, so it makes
easier to avoid errors where lock is destructed before the guard for it.
Because aws-sdk-cpp transforms PreconditionFailed errors into
S3Error::Unkown we do string munging on the exception name to identify
it.

Also fixes some block version ref key conflicts
@IvoDD IvoDD force-pushed the reliable-storage-lock branch from 92c203a to a025892 Compare December 9, 2024 12:30
@IvoDD IvoDD merged commit 2a6b3cd into master Dec 9, 2024
137 of 138 checks passed
@IvoDD IvoDD deleted the reliable-storage-lock branch December 9, 2024 14:17
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants