[Data Buckets] Implement scoped data buckets (#3498)

* [Data Buckets] Implement scoped data buckets

* Update database_update_manifest.cpp

* Update data_bucket.cpp

* Update data_bucket.cpp

* Update data_bucket.cpp

* Update data_bucket.h

* Update database_update_manifest.cpp

* Add GetScopedDbFilters references

* Scope transfer
This commit is contained in:
Chris Miles 2023-07-16 13:52:28 -05:00 committed by GitHub
parent 59537ae977
commit 3f3bbe98b5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 350 additions and 259 deletions

View File

@ -4804,6 +4804,27 @@ UNIQUE KEY `name` (`name`)
PRIMARY KEY (`id`), PRIMARY KEY (`id`),
UNIQUE INDEX `command`(`parent_command`, `sub_command`) UNIQUE INDEX `command`(`parent_command`, `sub_command`)
) )
)"
},
ManifestEntry{
.version = 9233,
.description = "2023_07_16_scoped_data_buckets.sql",
.check = "SHOW COLUMNS FROM `data_buckets` LIKE 'character_id'",
.condition = "empty",
.match = "",
.sql = R"(
ALTER TABLE `data_buckets`
ADD COLUMN `character_id` bigint(11) NOT NULL DEFAULT 0 AFTER `expires`,
ADD COLUMN `npc_id` bigint(11) NOT NULL DEFAULT 0 AFTER `character_id`,
ADD COLUMN `bot_id` bigint(11) NOT NULL DEFAULT 0 AFTER `npc_id`,
DROP INDEX `key_index`,
ADD UNIQUE INDEX `keys`(`key`,`character_id`,`npc_id`,`bot_id`);
UPDATE data_buckets SET character_id = SUBSTRING_INDEX(SUBSTRING_INDEX( `key`, '-', 2 ), '-', -1), `key` = SUBSTR(SUBSTRING_INDEX(`key`, SUBSTRING_INDEX( `key`, '-', 2 ), -1), 2) WHERE `key` LIKE 'character-%';
UPDATE data_buckets SET npc_id = SUBSTRING_INDEX(SUBSTRING_INDEX( `key`, '-', 2 ), '-', -1), `key` = SUBSTR(SUBSTRING_INDEX(`key`, SUBSTRING_INDEX( `key`, '-', 2 ), -1), 2) WHERE `key` LIKE 'npc-%';
UPDATE data_buckets SET bot_id = SUBSTRING_INDEX(SUBSTRING_INDEX( `key`, '-', 2 ), '-', -1), `key` = SUBSTR(SUBSTRING_INDEX(`key`, SUBSTRING_INDEX( `key`, '-', 2 ), -1), 2) WHERE `key` LIKE 'bot-%';
)" )"
}, },

View File

@ -24,6 +24,9 @@ public:
std::string key_; std::string key_;
std::string value; std::string value;
uint32_t expires; uint32_t expires;
int64_t character_id;
int64_t npc_id;
int64_t bot_id;
}; };
static std::string PrimaryKey() static std::string PrimaryKey()
@ -38,6 +41,9 @@ public:
"`key`", "`key`",
"value", "value",
"expires", "expires",
"character_id",
"npc_id",
"bot_id",
}; };
} }
@ -48,6 +54,9 @@ public:
"`key`", "`key`",
"value", "value",
"expires", "expires",
"character_id",
"npc_id",
"bot_id",
}; };
} }
@ -88,10 +97,13 @@ public:
{ {
DataBuckets e{}; DataBuckets e{};
e.id = 0; e.id = 0;
e.key_ = ""; e.key_ = "";
e.value = ""; e.value = "";
e.expires = 0; e.expires = 0;
e.character_id = 0;
e.npc_id = 0;
e.bot_id = 0;
return e; return e;
} }
@ -128,10 +140,13 @@ public:
if (results.RowCount() == 1) { if (results.RowCount() == 1) {
DataBuckets e{}; DataBuckets e{};
e.id = strtoull(row[0], nullptr, 10); e.id = strtoull(row[0], nullptr, 10);
e.key_ = row[1] ? row[1] : ""; e.key_ = row[1] ? row[1] : "";
e.value = row[2] ? row[2] : ""; e.value = row[2] ? row[2] : "";
e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10)); e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
e.character_id = strtoll(row[4], nullptr, 10);
e.npc_id = strtoll(row[5], nullptr, 10);
e.bot_id = strtoll(row[6], nullptr, 10);
return e; return e;
} }
@ -168,6 +183,9 @@ public:
v.push_back(columns[1] + " = '" + Strings::Escape(e.key_) + "'"); v.push_back(columns[1] + " = '" + Strings::Escape(e.key_) + "'");
v.push_back(columns[2] + " = '" + Strings::Escape(e.value) + "'"); v.push_back(columns[2] + " = '" + Strings::Escape(e.value) + "'");
v.push_back(columns[3] + " = " + std::to_string(e.expires)); v.push_back(columns[3] + " = " + std::to_string(e.expires));
v.push_back(columns[4] + " = " + std::to_string(e.character_id));
v.push_back(columns[5] + " = " + std::to_string(e.npc_id));
v.push_back(columns[6] + " = " + std::to_string(e.bot_id));
auto results = db.QueryDatabase( auto results = db.QueryDatabase(
fmt::format( fmt::format(
@ -193,6 +211,9 @@ public:
v.push_back("'" + Strings::Escape(e.key_) + "'"); v.push_back("'" + Strings::Escape(e.key_) + "'");
v.push_back("'" + Strings::Escape(e.value) + "'"); v.push_back("'" + Strings::Escape(e.value) + "'");
v.push_back(std::to_string(e.expires)); v.push_back(std::to_string(e.expires));
v.push_back(std::to_string(e.character_id));
v.push_back(std::to_string(e.npc_id));
v.push_back(std::to_string(e.bot_id));
auto results = db.QueryDatabase( auto results = db.QueryDatabase(
fmt::format( fmt::format(
@ -226,6 +247,9 @@ public:
v.push_back("'" + Strings::Escape(e.key_) + "'"); v.push_back("'" + Strings::Escape(e.key_) + "'");
v.push_back("'" + Strings::Escape(e.value) + "'"); v.push_back("'" + Strings::Escape(e.value) + "'");
v.push_back(std::to_string(e.expires)); v.push_back(std::to_string(e.expires));
v.push_back(std::to_string(e.character_id));
v.push_back(std::to_string(e.npc_id));
v.push_back(std::to_string(e.bot_id));
insert_chunks.push_back("(" + Strings::Implode(",", v) + ")"); insert_chunks.push_back("(" + Strings::Implode(",", v) + ")");
} }
@ -259,10 +283,13 @@ public:
for (auto row = results.begin(); row != results.end(); ++row) { for (auto row = results.begin(); row != results.end(); ++row) {
DataBuckets e{}; DataBuckets e{};
e.id = strtoull(row[0], nullptr, 10); e.id = strtoull(row[0], nullptr, 10);
e.key_ = row[1] ? row[1] : ""; e.key_ = row[1] ? row[1] : "";
e.value = row[2] ? row[2] : ""; e.value = row[2] ? row[2] : "";
e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10)); e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
e.character_id = strtoll(row[4], nullptr, 10);
e.npc_id = strtoll(row[5], nullptr, 10);
e.bot_id = strtoll(row[6], nullptr, 10);
all_entries.push_back(e); all_entries.push_back(e);
} }
@ -287,10 +314,13 @@ public:
for (auto row = results.begin(); row != results.end(); ++row) { for (auto row = results.begin(); row != results.end(); ++row) {
DataBuckets e{}; DataBuckets e{};
e.id = strtoull(row[0], nullptr, 10); e.id = strtoull(row[0], nullptr, 10);
e.key_ = row[1] ? row[1] : ""; e.key_ = row[1] ? row[1] : "";
e.value = row[2] ? row[2] : ""; e.value = row[2] ? row[2] : "";
e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10)); e.expires = static_cast<uint32_t>(strtoul(row[3], nullptr, 10));
e.character_id = strtoll(row[4], nullptr, 10);
e.npc_id = strtoll(row[5], nullptr, 10);
e.bot_id = strtoll(row[6], nullptr, 10);
all_entries.push_back(e); all_entries.push_back(e);
} }

View File

@ -42,7 +42,7 @@
* Manifest: https://github.com/EQEmu/Server/blob/master/utils/sql/db_update_manifest.txt * Manifest: https://github.com/EQEmu/Server/blob/master/utils/sql/db_update_manifest.txt
*/ */
#define CURRENT_BINARY_DATABASE_VERSION 9232 #define CURRENT_BINARY_DATABASE_VERSION 9233
#define CURRENT_BINARY_BOTS_DATABASE_VERSION 9039 #define CURRENT_BINARY_BOTS_DATABASE_VERSION 9039

View File

@ -6033,14 +6033,10 @@ float Mob::CheckHeroicBonusesDataBuckets(std::string bucket_name)
{ {
std::string bucket_value; std::string bucket_value;
if (!bucket_name.empty()) { if (!bucket_name.empty()) {
const auto full_name = fmt::format( DataBucketKey k = GetScopedBucketKeys();
"{}-{}", k.key = bucket_name;
GetBucketKey(),
bucket_name
);
if (IsOfClientBot()) { if (IsOfClientBot()) {
bucket_value = DataBucket::CheckBucketKey(this, full_name); bucket_value = DataBucket::CheckBucketKey(this, k);
} }
if (bucket_value.empty() || !Strings::IsNumber(bucket_value)) { if (bucket_value.empty() || !Strings::IsNumber(bucket_value)) {

View File

@ -8193,21 +8193,16 @@ void Bot::OwnerMessage(const std::string& message)
bool Bot::CheckDataBucket(std::string bucket_name, const std::string& bucket_value, uint8 bucket_comparison) bool Bot::CheckDataBucket(std::string bucket_name, const std::string& bucket_value, uint8 bucket_comparison)
{ {
if (!bucket_name.empty() && !bucket_value.empty()) { if (!bucket_name.empty() && !bucket_value.empty()) {
auto full_name = fmt::format( // try to fetch from bot first
"{}-{}", DataBucketKey k = GetScopedBucketKeys();
GetBucketKey(), k.key = bucket_name;
bucket_name
);
auto player_value = DataBucket::CheckBucketKey(this, full_name); auto player_value = DataBucket::CheckBucketKey(this, k);
if (player_value.empty() && GetBotOwner()) { if (player_value.empty() && GetBotOwner()) {
full_name = fmt::format( // fetch from owner
"{}-{}", k = GetBotOwner()->GetScopedBucketKeys();
GetBotOwner()->GetBucketKey(),
bucket_name
);
player_value = DataBucket::CheckBucketKey(GetBotOwner(), full_name); player_value = DataBucket::CheckBucketKey(GetBotOwner(), k);
if (player_value.empty()) { if (player_value.empty()) {
return false; return false;
} }

View File

@ -853,13 +853,11 @@ void Client::BulkSendMerchantInventory(int merchant_id, int npcid) {
auto bucket_name = ml.bucket_name; auto bucket_name = ml.bucket_name;
auto const& bucket_value = ml.bucket_value; auto const& bucket_value = ml.bucket_value;
if (!bucket_name.empty() && !bucket_value.empty()) { if (!bucket_name.empty() && !bucket_value.empty()) {
auto full_name = fmt::format(
"{}-{}",
GetBucketKey(),
bucket_name
);
auto const& player_value = DataBucket::CheckBucketKey(this, full_name); DataBucketKey k = GetScopedBucketKeys();
k.key = bucket_name;
auto const& player_value = DataBucket::CheckBucketKey(this, k);
if (player_value.empty()) { if (player_value.empty()) {
continue; continue;
} }

View File

@ -1,173 +1,130 @@
#include "data_bucket.h" #include "data_bucket.h"
#include "zonedb.h" #include "zonedb.h"
#include "mob.h"
#include <ctime> #include <ctime>
#include <cctype> #include <cctype>
#include "../common/repositories/data_buckets_repository.h"
/** void DataBucket::SetData(const std::string &bucket_key, const std::string &bucket_value, std::string expires_time)
* Persists data via bucket_name as key {
* @param bucket_key auto k = DataBucketKey{
* @param bucket_value .key = bucket_key,
* @param expires_time .value = bucket_value,
*/ .expires = expires_time,
void DataBucket::SetData(const std::string& bucket_key, const std::string& bucket_value, std::string expires_time) { .character_id = 0,
uint64 bucket_id = DataBucket::DoesBucketExist(bucket_key); .npc_id = 0,
.bot_id = 0
};
std::string query; DataBucket::SetData(k);
}
void DataBucket::SetData(const DataBucketKey &k)
{
auto r = DataBucketsRepository::GetWhere(
database,
fmt::format(
"{} `key` = '{}' AND (`expires` > {} OR `expires` = 0) LIMIT 1",
DataBucket::GetScopedDbFilters(k),
Strings::Escape(k.key),
(long long) std::time(nullptr)
)
);
// if we have an entry, use it
auto b = DataBucketsRepository::NewEntity();
if (!r.empty()) {
b = r[0];
}
if (k.character_id > 0) {
b.character_id = k.character_id;
} else if (k.npc_id > 0) {
b.npc_id = k.npc_id;
} else if (k.bot_id > 0) {
b.bot_id = k.bot_id;
}
uint64 bucket_id = b.id;
long long expires_time_unix = 0; long long expires_time_unix = 0;
if (!expires_time.empty()) { if (!k.expires.empty()) {
if (isalpha(expires_time[0]) || isalpha(expires_time[expires_time.length() - 1])) { expires_time_unix = (long long) std::time(nullptr) + Strings::ToInt(k.expires);
expires_time_unix = (long long) std::time(nullptr) + Strings::TimeToSeconds(expires_time); if (isalpha(k.expires[0]) || isalpha(k.expires[k.expires.length() - 1])) {
} else { expires_time_unix = (long long) std::time(nullptr) + Strings::TimeToSeconds(k.expires);
expires_time_unix = (long long) std::time(nullptr) + Strings::ToInt(expires_time);
} }
} }
if (bucket_id > 0) { if (bucket_id > 0) {
std::string update_expired_time; b.expires = expires_time_unix;
if (expires_time_unix > 0) { b.value = k.value;
update_expired_time = StringFormat(", `expires` = %lld ", expires_time_unix); DataBucketsRepository::UpdateOne(database, b);
}
query = StringFormat(
"UPDATE `data_buckets` SET `value` = '%s' %s WHERE `id` = %i",
Strings::Escape(bucket_value).c_str(),
Strings::Escape(update_expired_time).c_str(),
bucket_id
);
} }
else { else {
query = StringFormat( b.expires = expires_time_unix;
"INSERT INTO `data_buckets` (`key`, `value`, `expires`) VALUES ('%s', '%s', '%lld')", b.key_ = k.key;
Strings::Escape(bucket_key).c_str(), b.value = k.value;
Strings::Escape(bucket_value).c_str(), DataBucketsRepository::InsertOne(database, b);
expires_time_unix
);
} }
database.QueryDatabase(query);
} }
/** std::string DataBucket::GetData(const std::string &bucket_key)
* Retrieves data via bucket_name as key
* @param bucket_key
* @return
*/
std::string DataBucket::GetData(const std::string& bucket_key) {
std::string query = StringFormat(
"SELECT `value` from `data_buckets` WHERE `key` = '%s' AND (`expires` > %lld OR `expires` = 0) LIMIT 1",
bucket_key.c_str(),
(long long) std::time(nullptr)
);
auto results = database.QueryDatabase(query);
if (!results.Success()) {
return std::string();
}
if (results.RowCount() != 1)
return std::string();
auto row = results.begin();
return std::string(row[0]);
}
/**
* Retrieves data expires time via bucket_name as key
* @param bucket_key
* @return
*/
std::string DataBucket::GetDataExpires(const std::string& bucket_key) {
std::string query = StringFormat(
"SELECT `expires` from `data_buckets` WHERE `key` = '%s' AND (`expires` > %lld OR `expires` = 0) LIMIT 1",
bucket_key.c_str(),
(long long) std::time(nullptr)
);
auto results = database.QueryDatabase(query);
if (!results.Success()) {
return std::string();
}
if (results.RowCount() != 1)
return std::string();
auto row = results.begin();
return std::string(row[0]);
}
std::string DataBucket::GetDataRemaining(const std::string& bucket_key) {
if (DataBucket::GetDataExpires(bucket_key).empty()) {
return "0";
}
std::string query = fmt::format(
"SELECT (`expires` - UNIX_TIMESTAMP()) AS `remaining` from `data_buckets` WHERE `key` = '{}' AND (`expires` > {} OR `expires` = 0) LIMIT 1",
bucket_key.c_str(),
(long long) std::time(nullptr)
);
auto results = database.QueryDatabase(query);
if (!results.Success()) {
return std::string();
}
if (results.RowCount() != 1)
return std::string();
auto row = results.begin();
return std::string(row[0]);
}
/**
* Checks for bucket existence by bucket_name key
* @param bucket_key
* @return
*/
uint64 DataBucket::DoesBucketExist(const std::string& bucket_key) {
std::string query = StringFormat(
"SELECT `id` from `data_buckets` WHERE `key` = '%s' AND (`expires` > %lld OR `expires` = 0) LIMIT 1",
Strings::Escape(bucket_key).c_str(),
(long long) std::time(nullptr)
);
auto results = database.QueryDatabase(query);
if (!results.Success()) {
return 0;
}
auto row = results.begin();
if (results.RowCount() != 1)
return 0;
return Strings::ToUnsignedBigInt(row[0]);
}
/**
* Deletes data bucket by key
* @param bucket_key
* @return
*/
bool DataBucket::DeleteData(const std::string& bucket_key) {
std::string query = StringFormat(
"DELETE FROM `data_buckets` WHERE `key` = '%s'",
Strings::Escape(bucket_key).c_str()
);
auto results = database.QueryDatabase(query);
return results.Success();
}
bool DataBucket::GetDataBuckets(Mob* mob)
{ {
auto l = BaseDataBucketsRepository::GetWhere( DataBucketKey k = {};
k.key = bucket_key;
return GetData(k);
}
std::string DataBucket::GetData(const DataBucketKey &k)
{
auto r = DataBucketsRepository::GetWhere(
database, database,
fmt::format( fmt::format(
"`key` LIKE '{}-%'", "{} `key` = '{}' AND (`expires` > {} OR `expires` = 0) LIMIT 1",
Strings::Escape(mob->GetBucketKey()) DataBucket::GetScopedDbFilters(k),
k.key,
(long long) std::time(nullptr)
)
);
if (r.empty()) {
return {};
}
return r[0].value;
}
std::string DataBucket::GetDataExpires(const std::string &bucket_key)
{
DataBucketKey k = {};
k.key = bucket_key;
return GetDataExpires(k);
}
std::string DataBucket::GetDataRemaining(const std::string &bucket_key)
{
DataBucketKey k = {};
k.key = bucket_key;
return GetDataRemaining(k);
}
bool DataBucket::DeleteData(const std::string &bucket_key)
{
DataBucketKey r = {};
r.key = bucket_key;
return DeleteData(r);
}
bool DataBucket::GetDataBuckets(Mob *mob)
{
DataBucketKey k = mob->GetScopedBucketKeys();
auto l = BaseDataBucketsRepository::GetWhere(
database,
fmt::format(
"{} (`expires` > {} OR `expires` = 0)",
DataBucket::GetScopedDbFilters(k),
(long long) std::time(nullptr)
) )
); );
@ -179,10 +136,10 @@ bool DataBucket::GetDataBuckets(Mob* mob)
DataBucketCache d; DataBucketCache d;
for (const auto& e : l) { for (const auto &e: l) {
d.bucket_id = e.id; d.bucket_id = e.id;
d.bucket_key = e.key_; d.bucket_key = e.key_;
d.bucket_value = e.value; d.bucket_value = e.value;
d.bucket_expires = e.expires; d.bucket_expires = e.expires;
mob->m_data_bucket_cache.emplace_back(d); mob->m_data_bucket_cache.emplace_back(d);
@ -191,11 +148,11 @@ bool DataBucket::GetDataBuckets(Mob* mob)
return true; return true;
} }
std::string DataBucket::CheckBucketKey(const Mob* mob, std::string_view full_name) std::string DataBucket::CheckBucketKey(const Mob *mob, const DataBucketKey &k)
{ {
std::string bucket_value; std::string bucket_value;
for (const auto &d : mob->m_data_bucket_cache) { for (const auto &d: mob->m_data_bucket_cache) {
if (d.bucket_key == full_name) { if (d.bucket_key == k.key) {
bucket_value = d.bucket_value; bucket_value = d.bucket_value;
break; break;
} }
@ -203,3 +160,72 @@ std::string DataBucket::CheckBucketKey(const Mob* mob, std::string_view full_nam
return bucket_value; return bucket_value;
} }
bool DataBucket::DeleteData(const DataBucketKey &k)
{
return DataBucketsRepository::DeleteWhere(
database,
fmt::format(
"{} `key` = '{}'",
DataBucket::GetScopedDbFilters(k),
k.key
)
);
}
std::string DataBucket::GetDataExpires(const DataBucketKey &k)
{
auto r = DataBucketsRepository::GetWhere(
database,
fmt::format(
"{} `key` = '{}' AND (`expires` > {} OR `expires` = 0) LIMIT 1",
DataBucket::GetScopedDbFilters(k),
k.key,
(long long) std::time(nullptr)
)
);
if (r.empty()) {
return {};
}
return fmt::format("{}", r[0].expires);
}
std::string DataBucket::GetDataRemaining(const DataBucketKey &k)
{
auto r = DataBucketsRepository::GetWhere(
database,
fmt::format(
"{} `key` = '{}' AND (`expires` > {} OR `expires` = 0) LIMIT 1",
DataBucket::GetScopedDbFilters(k),
k.key,
(long long) std::time(nullptr)
)
);
if (r.empty()) {
return "0";
}
return fmt::format("{}", r[0].expires - (long long) std::time(nullptr));
}
std::string DataBucket::GetScopedDbFilters(const DataBucketKey &k)
{
std::vector<std::string> query = {};
if (k.character_id > 0) {
query.emplace_back(fmt::format("character_id = {}", k.character_id));
}
else if (k.npc_id > 0) {
query.emplace_back(fmt::format("npc_id = {}", k.npc_id));
}
else if (k.bot_id > 0) {
query.emplace_back(fmt::format("bot_id = {}", k.bot_id));
}
return fmt::format(
"{} {}",
Strings::Join(query, " AND "),
!query.empty() ? "AND" : ""
);
}

View File

@ -10,18 +10,35 @@
#include "../common/repositories/data_buckets_repository.h" #include "../common/repositories/data_buckets_repository.h"
#include "mob.h" #include "mob.h"
struct DataBucketKey {
std::string key;
std::string value;
std::string expires;
int64_t character_id;
int64_t npc_id;
int64_t bot_id;
};
class DataBucket { class DataBucket {
public: public:
// non-scoped bucket methods (for global buckets)
static void SetData(const std::string& bucket_key, const std::string& bucket_value, std::string expires_time = ""); static void SetData(const std::string& bucket_key, const std::string& bucket_value, std::string expires_time = "");
static bool DeleteData(const std::string& bucket_key); static bool DeleteData(const std::string& bucket_key);
static std::string GetData(const std::string& bucket_key); static std::string GetData(const std::string& bucket_key);
static std::string GetDataExpires(const std::string& bucket_key); static std::string GetDataExpires(const std::string& bucket_key);
static std::string GetDataRemaining(const std::string& bucket_key); static std::string GetDataRemaining(const std::string& bucket_key);
static bool GetDataBuckets(Mob* mob);
static std::string CheckBucketKey(const Mob* mob, std::string_view full_name);
private: static bool GetDataBuckets(Mob* mob);
static uint64 DoesBucketExist(const std::string& bucket_key);
// scoped bucket methods
static void SetData(const DataBucketKey& k);
static bool DeleteData(const DataBucketKey& k);
static std::string GetData(const DataBucketKey& k);
static std::string GetDataExpires(const DataBucketKey& k);
static std::string GetDataRemaining(const DataBucketKey& k);
static std::string CheckBucketKey(const Mob* mob, const DataBucketKey& k);
static std::string GetScopedDbFilters(const DataBucketKey& k);
}; };
#endif //EQEMU_DATABUCKET_H #endif //EQEMU_DATABUCKET_H

View File

@ -1273,24 +1273,10 @@ uint8 Client::GetCharMaxLevelFromQGlobal() {
uint8 Client::GetCharMaxLevelFromBucket() uint8 Client::GetCharMaxLevelFromBucket()
{ {
auto new_bucket_name = fmt::format( DataBucketKey k = GetScopedBucketKeys();
"{}-CharMaxLevel", k.key = "CharMaxLevel";
GetBucketKey()
);
auto bucket_value = DataBucket::GetData(new_bucket_name); auto bucket_value = DataBucket::GetData(k);
if (!bucket_value.empty()) {
if (Strings::IsNumber(bucket_value)) {
return static_cast<uint8>(Strings::ToUnsignedInt(bucket_value));
}
}
auto old_bucket_name = fmt::format(
"{}-CharMaxLevel",
CharacterID()
);
bucket_value = DataBucket::GetData(old_bucket_name);
if (!bucket_value.empty()) { if (!bucket_value.empty()) {
if (Strings::IsNumber(bucket_value)) { if (Strings::IsNumber(bucket_value)) {
return static_cast<uint8>(Strings::ToUnsignedInt(bucket_value)); return static_cast<uint8>(Strings::ToUnsignedInt(bucket_value));

View File

@ -2353,7 +2353,7 @@ std::string Lua_Mob::GetBucketExpires(std::string bucket_name)
std::string Lua_Mob::GetBucketKey() std::string Lua_Mob::GetBucketKey()
{ {
Lua_Safe_Call_String(); Lua_Safe_Call_String();
return self->GetBucketKey(); return {};
} }
std::string Lua_Mob::GetBucketRemaining(std::string bucket_name) std::string Lua_Mob::GetBucketRemaining(std::string bucket_name)

View File

@ -8049,54 +8049,63 @@ void Mob::SetCanOpenDoors(bool can_open)
m_can_open_doors = can_open; m_can_open_doors = can_open;
} }
void Mob::DeleteBucket(std::string bucket_name) { void Mob::DeleteBucket(std::string bucket_name)
std::string full_bucket_name = fmt::format("{}-{}", GetBucketKey(), bucket_name); {
DataBucket::DeleteData(full_bucket_name); DataBucketKey k = GetScopedBucketKeys();
k.key = bucket_name;
DataBucket::DeleteData(k);
} }
std::string Mob::GetBucket(std::string bucket_name) { std::string Mob::GetBucket(std::string bucket_name)
std::string full_bucket_name = fmt::format("{}-{}", GetBucketKey(), bucket_name); {
std::string bucket_value = DataBucket::GetData(full_bucket_name); DataBucketKey k = GetScopedBucketKeys();
k.key = bucket_name;
std::string bucket_value = DataBucket::GetData(k);
if (!bucket_value.empty()) { if (!bucket_value.empty()) {
return bucket_value; return bucket_value;
} }
return std::string(); return {};
} }
std::string Mob::GetBucketExpires(std::string bucket_name) { std::string Mob::GetBucketExpires(std::string bucket_name)
std::string full_bucket_name = fmt::format("{}-{}", GetBucketKey(), bucket_name); {
std::string bucket_expiration = DataBucket::GetDataExpires(full_bucket_name); DataBucketKey k = GetScopedBucketKeys();
k.key = bucket_name;
std::string bucket_expiration = DataBucket::GetDataExpires(k);
if (!bucket_expiration.empty()) { if (!bucket_expiration.empty()) {
return bucket_expiration; return bucket_expiration;
} }
return std::string();
return {};
} }
std::string Mob::GetBucketKey() { std::string Mob::GetBucketRemaining(std::string bucket_name)
if (IsClient()) { {
return fmt::format("character-{}", CastToClient()->CharacterID()); DataBucketKey k = GetScopedBucketKeys();
} else if (IsNPC()) { k.key = bucket_name;
return fmt::format("npc-{}", GetNPCTypeID());
} else if (IsBot()) {
return fmt::format("bot-{}", CastToBot()->GetBotID());
}
return std::string();
}
std::string Mob::GetBucketRemaining(std::string bucket_name) { std::string bucket_remaining = DataBucket::GetDataRemaining(k);
std::string full_bucket_name = fmt::format("{}-{}", GetBucketKey(), bucket_name);
std::string bucket_remaining = DataBucket::GetDataRemaining(full_bucket_name);
if (!bucket_remaining.empty() && Strings::ToInt(bucket_remaining) > 0) { if (!bucket_remaining.empty() && Strings::ToInt(bucket_remaining) > 0) {
return bucket_remaining; return bucket_remaining;
} else if (Strings::ToInt(bucket_remaining) == 0) { }
else if (Strings::ToInt(bucket_remaining) == 0) {
return "0"; return "0";
} }
return std::string();
return {};
} }
void Mob::SetBucket(std::string bucket_name, std::string bucket_value, std::string expiration) { void Mob::SetBucket(std::string bucket_name, std::string bucket_value, std::string expiration)
std::string full_bucket_name = fmt::format("{}-{}", GetBucketKey(), bucket_name); {
DataBucket::SetData(full_bucket_name, bucket_value, expiration); DataBucketKey k = GetScopedBucketKeys();
k.key = bucket_name;
k.expires = expiration;
k.value = bucket_value;
DataBucket::SetData(k);
} }
std::string Mob::GetMobDescription() std::string Mob::GetMobDescription()
@ -8329,3 +8338,17 @@ std::string Mob::GetClassPlural()
return "Classes"; return "Classes";
} }
} }
DataBucketKey Mob::GetScopedBucketKeys()
{
DataBucketKey k = {};
if (IsClient()) {
k.character_id = CastToClient()->CharacterID();
} else if (IsNPC()) {
k.npc_id = GetNPCTypeID();
} else if (IsBot()) {
k.bot_id = CastToBot()->GetBotID();
}
return k;
}

View File

@ -69,6 +69,7 @@ enum class eSpecialAttacks : int {
ChaoticStab ChaoticStab
}; };
class DataBucketKey;
class Mob : public Entity { class Mob : public Entity {
public: public:
enum CLIENT_CONN_STATUS { CLIENT_CONNECTING, CLIENT_CONNECTED, CLIENT_LINKDEAD, enum CLIENT_CONN_STATUS { CLIENT_CONNECTING, CLIENT_CONNECTED, CLIENT_LINKDEAD,
@ -1413,7 +1414,6 @@ public:
void DeleteBucket(std::string bucket_name); void DeleteBucket(std::string bucket_name);
std::string GetBucket(std::string bucket_name); std::string GetBucket(std::string bucket_name);
std::string GetBucketExpires(std::string bucket_name); std::string GetBucketExpires(std::string bucket_name);
std::string GetBucketKey();
std::string GetBucketRemaining(std::string bucket_name); std::string GetBucketRemaining(std::string bucket_name);
void SetBucket(std::string bucket_name, std::string bucket_value, std::string expiration = ""); void SetBucket(std::string bucket_name, std::string bucket_value, std::string expiration = "");
@ -1443,6 +1443,8 @@ public:
void CalcHeroicBonuses(StatBonuses* newbon); void CalcHeroicBonuses(StatBonuses* newbon);
DataBucketKey GetScopedBucketKeys();
protected: protected:
void CommonDamage(Mob* other, int64 &damage, const uint16 spell_id, const EQ::skills::SkillType attack_skill, bool &avoidable, const int8 buffslot, const bool iBuffTic, eSpecialAttacks specal = eSpecialAttacks::None); void CommonDamage(Mob* other, int64 &damage, const uint16 spell_id, const EQ::skills::SkillType attack_skill, bool &avoidable, const int8 buffslot, const bool iBuffTic, eSpecialAttacks specal = eSpecialAttacks::None);
static uint16 GetProcID(uint16 spell_id, uint8 effect_index); static uint16 GetProcID(uint16 spell_id, uint8 effect_index);

View File

@ -2420,7 +2420,7 @@ std::string Perl_Mob_GetBucketExpires(Mob* self, std::string bucket_name) // @ca
std::string Perl_Mob_GetBucketKey(Mob* self) // @categories Script Utility std::string Perl_Mob_GetBucketKey(Mob* self) // @categories Script Utility
{ {
return self->GetBucketKey(); return {};
} }
std::string Perl_Mob_GetBucketRemaining(Mob* self, std::string bucket_name) // @categories Script Utility std::string Perl_Mob_GetBucketRemaining(Mob* self, std::string bucket_name) // @categories Script Utility

View File

@ -5877,13 +5877,10 @@ bool Client::SpellBucketCheck(uint16 spell_id, uint32 character_id) {
return true; // If the entry in the spell_buckets table has nothing set for the qglobal name, allow scribing. return true; // If the entry in the spell_buckets table has nothing set for the qglobal name, allow scribing.
} }
auto new_bucket_name = fmt::format( DataBucketKey k = GetScopedBucketKeys();
"{}-{}", k.key = spell_bucket_name;
GetBucketKey(),
spell_bucket_name
);
auto bucket_value = DataBucket::GetData(new_bucket_name); auto bucket_value = DataBucket::GetData(k);
if (!bucket_value.empty()) { if (!bucket_value.empty()) {
if (Strings::IsNumber(bucket_value) && Strings::IsNumber(spell_bucket_value)) { if (Strings::IsNumber(bucket_value) && Strings::IsNumber(spell_bucket_value)) {
if (Strings::ToInt(bucket_value) >= Strings::ToInt(spell_bucket_value)) { if (Strings::ToInt(bucket_value) >= Strings::ToInt(spell_bucket_value)) {