mirror of
https://github.com/zeek/zeek.git
synced 2025-10-02 14:48:21 +00:00
Prevent SQLite storage backend from serving expired entries
The SQLite storage backend implements expiration by hand and garbage collection is done in `DoExpire`. This previously relied exclusively on gets not running within `Storage::expire_interval` of the put, otherwise we would potentially serve expired entries. With this patch we explictly check that entries are not expired before serving them so that the SQLite backend should never serve expired entries.
This commit is contained in:
parent
dc5d7c3fc9
commit
56f5cafa98
3 changed files with 70 additions and 8 deletions
|
@ -17,6 +17,11 @@
|
|||
|
||||
using namespace std::chrono_literals;
|
||||
|
||||
namespace {
|
||||
// Helper to check whether a database-stored `expire_time` is considered expired.
|
||||
bool is_expired(double expire_time) { return expire_time != 0 && expire_time < zeek::run_state::network_time; }
|
||||
} // namespace
|
||||
|
||||
namespace zeek::storage::backend::sqlite {
|
||||
|
||||
OperationResult SQLite::RunPragma(std::string_view name, std::optional<std::string_view> value) {
|
||||
|
@ -190,7 +195,7 @@ OperationResult SQLite::DoOpen(OpenResultCallback* cb, RecordValPtr options) {
|
|||
"DO UPDATE SET value_str=?, expire_time=?",
|
||||
table_name.c_str()),
|
||||
db),
|
||||
std::make_pair(util::fmt("select value_str from %s where key_str=?", table_name.c_str()), db),
|
||||
std::make_pair(util::fmt("select value_str, expire_time from %s where key_str=?", table_name.c_str()), db),
|
||||
std::make_pair(util::fmt("delete from %s where key_str=?", table_name.c_str()), db),
|
||||
|
||||
std::make_pair(
|
||||
|
@ -497,15 +502,19 @@ OperationResult SQLite::Step(sqlite3_stmt* stmt, bool parse_value) {
|
|||
int step_status = sqlite3_step(stmt);
|
||||
if ( step_status == SQLITE_ROW ) {
|
||||
if ( parse_value ) {
|
||||
auto blob = static_cast<const std::byte*>(sqlite3_column_blob(stmt, 0));
|
||||
size_t blob_size = sqlite3_column_bytes(stmt, 0);
|
||||
if ( sqlite3_column_type(stmt, 1) != SQLITE_NULL && is_expired(sqlite3_column_double(stmt, 1)) )
|
||||
ret = {ReturnCode::KEY_NOT_FOUND, ""};
|
||||
else {
|
||||
auto blob = static_cast<const std::byte*>(sqlite3_column_blob(stmt, 0));
|
||||
size_t blob_size = sqlite3_column_bytes(stmt, 0);
|
||||
|
||||
auto val = serializer->Unserialize({blob, blob_size}, val_type);
|
||||
auto val = serializer->Unserialize({blob, blob_size}, val_type);
|
||||
|
||||
if ( val )
|
||||
ret = {ReturnCode::SUCCESS, "", val.value()};
|
||||
else
|
||||
ret = {ReturnCode::OPERATION_FAILED, val.error()};
|
||||
if ( val )
|
||||
ret = {ReturnCode::SUCCESS, "", val.value()};
|
||||
else
|
||||
ret = {ReturnCode::OPERATION_FAILED, val.error()};
|
||||
}
|
||||
}
|
||||
else {
|
||||
ret = {ReturnCode::OPERATION_FAILED, "sqlite3_step should not have returned a value"};
|
||||
|
|
|
@ -0,0 +1,3 @@
|
|||
### BTest baseline data generated by btest-diff. Do not edit. Use "btest -U/-u" to update. Requires BTest >= 0.63.
|
||||
BEFORE, [code=Storage::SUCCESS, error_str=<uninitialized>, value=v]
|
||||
AFTER, [code=Storage::KEY_NOT_FOUND, error_str=<uninitialized>, value=<uninitialized>]
|
|
@ -0,0 +1,50 @@
|
|||
# @TEST-EXEC: zeek %INPUT 2>&1 >output
|
||||
# @TEST-EXEC: btest-diff output
|
||||
|
||||
@load base/frameworks/storage/sync
|
||||
@load policy/frameworks/storage/backend/sqlite
|
||||
|
||||
# Manually control the clock.
|
||||
redef allow_network_time_forward = F;
|
||||
|
||||
redef Storage::expire_interval = 1secs;
|
||||
|
||||
event zeek_init()
|
||||
{
|
||||
local opts = Storage::BackendOptions(
|
||||
$serializer=Storage::STORAGE_SERIALIZER_JSON,
|
||||
$sqlite=Storage::Backend::SQLite::Options(
|
||||
$database_path="test.sqlite",
|
||||
$table_name="testing"));
|
||||
|
||||
local open_res = Storage::Sync::open_backend(
|
||||
Storage::STORAGE_BACKEND_SQLITE,
|
||||
opts,
|
||||
string,
|
||||
string);
|
||||
local h = open_res$value;
|
||||
|
||||
local key="k";
|
||||
local value="v";
|
||||
|
||||
# Expire entries well within `Storage::expire_interval` so `DoExpire` does not kick in.
|
||||
local expire = Storage::expire_interval / 10;
|
||||
local expire_time = network_time() + expire;
|
||||
Storage::Sync::put(
|
||||
h,
|
||||
Storage::PutArgs(
|
||||
$key=key,
|
||||
$value=value,
|
||||
$expire_time=expire));
|
||||
|
||||
# The entry we just put in exists.
|
||||
local get = Storage::Sync::get(h, key);
|
||||
print "BEFORE", get;
|
||||
|
||||
# Advance time.
|
||||
set_network_time(network_time() + expire * 2);
|
||||
|
||||
# An expired value does not exist.
|
||||
get = Storage::Sync::get(h, key);
|
||||
print "AFTER", get;
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue