2021-11-28 15:29:26 +08:00
|
|
|
// Copyright 2021, Roman Gershman. All rights reserved.
|
|
|
|
// See LICENSE for licensing terms.
|
2021-11-17 18:51:47 +08:00
|
|
|
//
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2021-12-22 23:17:52 +08:00
|
|
|
#include <absl/container/flat_hash_map.h>
|
|
|
|
#include <absl/container/flat_hash_set.h>
|
|
|
|
|
|
|
|
#include "core/intent_lock.h"
|
2021-12-21 04:46:51 +08:00
|
|
|
#include "core/op_status.h"
|
2021-12-22 23:17:52 +08:00
|
|
|
#include "server/common_types.h"
|
2021-12-20 17:42:55 +08:00
|
|
|
#include "server/table.h"
|
2021-11-17 18:51:47 +08:00
|
|
|
|
|
|
|
namespace util {
|
|
|
|
class ProactorBase;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace dfly {
|
|
|
|
|
|
|
|
class DbSlice {
|
|
|
|
struct InternalDbStats {
|
|
|
|
// Object memory usage besides hash-table capacity.
|
|
|
|
size_t obj_memory_usage = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
DbSlice(uint32_t index, EngineShard* owner);
|
|
|
|
~DbSlice();
|
|
|
|
|
|
|
|
// Activates `db_ind` database if it does not exist (see ActivateDb below).
|
|
|
|
void Reserve(DbIndex db_ind, size_t key_size);
|
|
|
|
|
2021-12-20 17:42:55 +08:00
|
|
|
//! UpdateExpireClock updates the expire clock for this db slice.
|
|
|
|
//! Must be a wall clock so we could replicate it betweeen machines.
|
|
|
|
void UpdateExpireClock(uint64_t now_ms) {
|
|
|
|
now_ms_ = now_ms;
|
|
|
|
}
|
|
|
|
|
|
|
|
// returns wall clock in millis as it has been set via UpdateExpireClock.
|
|
|
|
uint64_t Now() const {
|
|
|
|
return now_ms_;
|
|
|
|
}
|
|
|
|
|
2022-01-02 16:40:16 +08:00
|
|
|
OpResult<MainIterator> Find(DbIndex db_index, std::string_view key, unsigned obj_type) const;
|
2021-11-17 18:51:47 +08:00
|
|
|
|
2021-12-20 17:42:55 +08:00
|
|
|
// Returns (value, expire) dict entries if key exists, null if it does not exist or has expired.
|
|
|
|
std::pair<MainIterator, ExpireIterator> FindExt(DbIndex db_ind, std::string_view key) const;
|
|
|
|
|
2021-11-17 18:51:47 +08:00
|
|
|
// Return .second=true if insertion ocurred, false if we return the existing key.
|
|
|
|
std::pair<MainIterator, bool> AddOrFind(DbIndex db_ind, std::string_view key);
|
|
|
|
|
2021-12-20 17:42:55 +08:00
|
|
|
// Either adds or removes (if at == 0) expiry. Returns true if a change was made.
|
|
|
|
// Does not change expiry if at != 0 and expiry already exists.
|
|
|
|
bool Expire(DbIndex db_ind, MainIterator main_it, uint64_t at);
|
|
|
|
|
2021-11-17 18:51:47 +08:00
|
|
|
// Adds a new entry. Requires: key does not exist in this slice.
|
|
|
|
void AddNew(DbIndex db_ind, std::string_view key, MainValue obj, uint64_t expire_at_ms);
|
|
|
|
|
|
|
|
// Adds a new entry if a key does not exists. Returns true if insertion took place,
|
|
|
|
// false otherwise. expire_at_ms equal to 0 - means no expiry.
|
|
|
|
bool AddIfNotExist(DbIndex db_ind, std::string_view key, MainValue obj, uint64_t expire_at_ms);
|
|
|
|
|
|
|
|
// Creates a database with index `db_ind`. If such database exists does nothing.
|
|
|
|
void ActivateDb(DbIndex db_ind);
|
|
|
|
|
2021-12-26 23:25:49 +08:00
|
|
|
bool Del(DbIndex db_ind, const MainIterator& it);
|
|
|
|
|
2021-12-22 23:17:52 +08:00
|
|
|
ShardId shard_id() const {
|
|
|
|
return shard_id_;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Acquire(IntentLock::Mode m, const KeyLockArgs& lock_args);
|
|
|
|
|
|
|
|
void Release(IntentLock::Mode m, const KeyLockArgs& lock_args);
|
|
|
|
void Release(IntentLock::Mode m, DbIndex db_index, std::string_view key, unsigned count);
|
|
|
|
|
|
|
|
// Returns true if all keys can be locked under m. Does not lock them though.
|
|
|
|
bool CheckLock(IntentLock::Mode m, const KeyLockArgs& lock_args) const;
|
2021-12-20 17:42:55 +08:00
|
|
|
|
2021-11-17 18:51:47 +08:00
|
|
|
size_t db_array_size() const {
|
|
|
|
return db_arr_.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsDbValid(DbIndex id) const {
|
2021-12-22 23:17:52 +08:00
|
|
|
return id < db_arr_.size() && bool(db_arr_[id]);
|
2021-11-17 18:51:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Returns existing keys count in the db.
|
|
|
|
size_t DbSize(DbIndex db_ind) const;
|
|
|
|
|
|
|
|
private:
|
2021-12-22 23:17:52 +08:00
|
|
|
void CreateDb(DbIndex index);
|
2021-11-17 18:51:47 +08:00
|
|
|
|
|
|
|
ShardId shard_id_;
|
|
|
|
|
|
|
|
EngineShard* owner_;
|
|
|
|
|
2021-12-20 17:42:55 +08:00
|
|
|
uint64_t now_ms_ = 0; // Used for expire logic, represents a real clock.
|
|
|
|
|
2021-12-22 23:17:52 +08:00
|
|
|
using LockTable = absl::flat_hash_map<std::string, IntentLock>;
|
|
|
|
|
|
|
|
struct DbWrapper {
|
|
|
|
MainTable main_table;
|
|
|
|
ExpireTable expire_table;
|
|
|
|
LockTable lock_table;
|
2021-12-20 17:42:55 +08:00
|
|
|
|
2021-11-17 18:51:47 +08:00
|
|
|
mutable InternalDbStats stats;
|
|
|
|
};
|
|
|
|
|
2021-12-22 23:17:52 +08:00
|
|
|
std::vector<std::unique_ptr<DbWrapper>> db_arr_;
|
|
|
|
|
|
|
|
// Used in temporary computations in Acquire/Release.
|
|
|
|
absl::flat_hash_set<std::string_view> uniq_keys_;
|
2021-11-17 18:51:47 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace dfly
|