zeek/src/cluster/BifSupport.cc
Arne Welzel 210b54799e cluster: Move publish_hrw() and publish_rr() to cluster.bif
From this point on, Cluster::publish_hrw() and Cluster::publish_rr()
go through cluster/Backend.cc code.
2024-12-10 20:33:02 +01:00

148 lines
5.6 KiB
C++

// See the file "COPYING" in the main distribution directory for copyright.
#include "zeek/cluster/BifSupport.h"
#include "zeek/Desc.h"
#include "zeek/Event.h"
#include "zeek/EventRegistry.h"
#include "zeek/Frame.h"
#include "zeek/Func.h"
#include "zeek/IntrusivePtr.h"
#include "zeek/Reporter.h"
#include "zeek/Type.h"
#include "zeek/Val.h"
#include "zeek/broker/Manager.h" // For publishing to broker_mgr directly.
#include "zeek/cluster/Backend.h"
namespace {
// Convert a script-level Cluster::Event to a cluster::detail::Event.
std::optional<zeek::cluster::detail::Event> to_cluster_event(const zeek::RecordValPtr& rec) {
const auto& func = rec->GetField<zeek::FuncVal>(0);
const auto& vargs = rec->GetField<zeek::VectorVal>(1);
const auto& eh = zeek::event_registry->Lookup(func->AsFuncPtr()->GetName());
if ( ! eh ) {
zeek::emit_builtin_error(
zeek::util::fmt("event registry lookup of '%s' failed", zeek::obj_desc_short(func.get()).c_str()));
return std::nullopt;
}
// Need to copy from VectorVal to zeek::Args
zeek::Args args(vargs->Size());
for ( size_t i = 0; i < vargs->Size(); i++ )
args[i] = vargs->ValAt(i);
// TODO: Support configurable timestamps or custom metadata on the record.
auto timestamp = zeek::event_mgr.CurrentEventTime();
return zeek::cluster::detail::Event(eh, std::move(args), timestamp);
}
} // namespace
namespace zeek::cluster::detail::bif {
zeek::RecordValPtr make_event(zeek::ArgsSpan args) {
static const auto& any_vec_type = zeek::id::find_type<zeek::VectorType>("any_vec");
static const auto& event_record_type = zeek::id::find_type<zeek::RecordType>("Cluster::Event");
auto rec = zeek::make_intrusive<zeek::RecordVal>(event_record_type);
if ( args.empty() ) {
zeek::emit_builtin_error("not enough arguments");
return rec;
}
const auto& maybe_func_val = args[0];
if ( maybe_func_val->GetType()->Tag() != zeek::TYPE_FUNC ) {
zeek::emit_builtin_error(
zeek::util::fmt("got non-event type '%s'", zeek::obj_desc_short(maybe_func_val->GetType().get()).c_str()));
return rec;
}
const auto func = zeek::FuncValPtr{zeek::NewRef{}, maybe_func_val->AsFuncVal()};
auto checked_args = cluster::detail::check_args(func, args.subspan(1));
if ( ! checked_args )
return rec;
// Making a copy from zeek::Args to a VectorVal and then back again on publish.
auto vec = zeek::make_intrusive<zeek::VectorVal>(any_vec_type);
vec->Reserve(checked_args->size());
rec->Assign(0, maybe_func_val);
for ( const auto& arg : *checked_args )
vec->Append(arg);
rec->Assign(1, vec); // Args
return rec;
}
zeek::ValPtr publish_event(const zeek::ValPtr& topic, zeek::ArgsSpan args) {
static const auto& cluster_event_type = zeek::id::find_type<zeek::RecordType>("Cluster::Event");
static const auto& broker_event_type = zeek::id::find_type<zeek::RecordType>("Broker::Event");
if ( args.empty() ) {
zeek::emit_builtin_error("no event arguments given");
return zeek::val_mgr->False();
}
if ( topic->GetType()->Tag() != zeek::TYPE_STRING ) {
zeek::emit_builtin_error("topic is not a string");
return zeek::val_mgr->False();
}
const auto topic_str = topic->AsStringVal()->ToStdString();
auto timestamp = zeek::event_mgr.CurrentEventTime();
if ( args[0]->GetType()->Tag() == zeek::TYPE_FUNC ) {
auto event = zeek::cluster::backend->MakeClusterEvent({zeek::NewRef{}, args[0]->AsFuncVal()}, args.subspan(1),
timestamp);
if ( event )
return zeek::val_mgr->Bool(zeek::cluster::backend->PublishEvent(topic_str, *event));
return zeek::val_mgr->False();
}
else if ( args[0]->GetType()->Tag() == zeek::TYPE_RECORD ) {
if ( args[0]->GetType() == cluster_event_type ) { // Handling Cluster::Event record type
auto ev = to_cluster_event(cast_intrusive<zeek::RecordVal>(args[0]));
if ( ! ev )
return zeek::val_mgr->False();
return zeek::val_mgr->Bool(zeek::cluster::backend->PublishEvent(topic_str, *ev));
}
else if ( args[0]->GetType() == broker_event_type ) {
// Handling Broker::Event record type created by Broker::make_event()
// only works if the backend is broker_mgr!
if ( zeek::cluster::backend != zeek::broker_mgr ) {
zeek::emit_builtin_error(
zeek::util::fmt("Publish of Broker::Event record instance with type '%s' to a non-Broker backend",
zeek::obj_desc_short(args[0]->GetType().get()).c_str()));
return zeek::val_mgr->False();
}
return zeek::val_mgr->Bool(zeek::broker_mgr->PublishEvent(topic_str, args[0]->AsRecordVal()));
}
else {
zeek::emit_builtin_error(zeek::util::fmt("Publish of unknown record type '%s'",
zeek::obj_desc_short(args[0]->GetType().get()).c_str()));
return zeek::val_mgr->False();
}
}
zeek::emit_builtin_error(zeek::util::fmt("expected function or record as first argument, got %s",
zeek::obj_desc_short(args[0]->GetType().get()).c_str()));
return zeek::val_mgr->False();
}
bool is_cluster_pool(const zeek::Val* pool) {
static zeek::RecordTypePtr pool_type = nullptr;
if ( ! pool_type )
pool_type = zeek::id::find_type<zeek::RecordType>("Cluster::Pool");
return pool->GetType() == pool_type;
}
} // namespace zeek::cluster::detail::bif