mirror of
https://github.com/zeek/zeek.git
synced 2025-10-02 06:38:20 +00:00
670 lines
20 KiB
C++
670 lines
20 KiB
C++
//
|
|
// See the file "COPYING" in the main distribution directory for copyright.
|
|
//
|
|
|
|
#include "zeek/Reporter.h"
|
|
|
|
#include <syslog.h>
|
|
#include <unistd.h>
|
|
|
|
#include "zeek/Conn.h"
|
|
#include "zeek/Desc.h"
|
|
#include "zeek/Event.h"
|
|
#include "zeek/EventHandler.h"
|
|
#include "zeek/Expr.h"
|
|
#include "zeek/Frame.h"
|
|
#include "zeek/ID.h"
|
|
#include "zeek/RunState.h"
|
|
#include "zeek/Timer.h"
|
|
#include "zeek/file_analysis/File.h"
|
|
#include "zeek/input.h"
|
|
#include "zeek/plugin/Manager.h"
|
|
#include "zeek/plugin/Plugin.h"
|
|
|
|
#include "zeek/3rdparty/doctest.h"
|
|
|
|
#ifdef SYSLOG_INT
|
|
extern "C" {
|
|
int openlog(const char* ident, int logopt, int facility);
|
|
int syslog(int priority, const char* message_fmt, ...);
|
|
int closelog();
|
|
}
|
|
#endif
|
|
|
|
zeek::Reporter* zeek::reporter = nullptr;
|
|
zeek::Reporter*& reporter = zeek::reporter;
|
|
|
|
namespace zeek {
|
|
|
|
Reporter::Reporter(bool arg_abort_on_scripting_errors) {
|
|
abort_on_scripting_errors = arg_abort_on_scripting_errors;
|
|
errors = 0;
|
|
via_events = false;
|
|
in_error_handler = 0;
|
|
|
|
// Always use stderr at startup/init before scripts have been fully parsed
|
|
// and zeek_init() processed.
|
|
// Messages may otherwise be missed if an error occurs that prevents events
|
|
// from ever being dispatched.
|
|
info_to_stderr = true;
|
|
warnings_to_stderr = true;
|
|
errors_to_stderr = true;
|
|
|
|
weird_count = 0;
|
|
weird_sampling_rate = 0;
|
|
weird_sampling_duration = 0;
|
|
weird_sampling_threshold = 0;
|
|
|
|
ignore_deprecations = false;
|
|
|
|
syslog_open = false;
|
|
}
|
|
|
|
Reporter::~Reporter() { closelog(); }
|
|
|
|
void Reporter::InitOptions() {
|
|
info_to_stderr = id::find_val("Reporter::info_to_stderr")->AsBool();
|
|
warnings_to_stderr = id::find_val("Reporter::warnings_to_stderr")->AsBool();
|
|
errors_to_stderr = id::find_val("Reporter::errors_to_stderr")->AsBool();
|
|
weird_sampling_rate = id::find_val("Weird::sampling_rate")->AsCount();
|
|
weird_sampling_threshold = id::find_val("Weird::sampling_threshold")->AsCount();
|
|
weird_sampling_duration = id::find_val("Weird::sampling_duration")->AsInterval();
|
|
|
|
auto init_weird_set = [](WeirdSet* set, const char* name) {
|
|
auto wl_val = id::find_val(name)->AsTableVal();
|
|
auto wl_table = wl_val->AsTable();
|
|
|
|
for ( const auto& wle : *wl_table ) {
|
|
auto k = wle.GetHashKey();
|
|
auto index = wl_val->RecreateIndex(*k);
|
|
std::string key = index->Idx(0)->AsString()->CheckString();
|
|
set->emplace(std::move(key));
|
|
}
|
|
};
|
|
|
|
init_weird_set(&weird_sampling_whitelist, "Weird::sampling_whitelist");
|
|
init_weird_set(&weird_sampling_global_list, "Weird::sampling_global_list");
|
|
}
|
|
|
|
void Reporter::Info(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(info_to_stderr) ? stderr : nullptr;
|
|
DoLog("", reporter_info, out, nullptr, nullptr, true, true, nullptr, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::Warning(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(warnings_to_stderr) ? stderr : nullptr;
|
|
DoLog("warning", reporter_warning, out, nullptr, nullptr, true, true, nullptr, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::Error(const char* fmt, ...) {
|
|
++errors;
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(errors_to_stderr) ? stderr : nullptr;
|
|
DoLog("error", reporter_error, out, nullptr, nullptr, true, true, nullptr, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::FatalError(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
|
|
// Always log to stderr.
|
|
DoLog("fatal error", nullptr, stderr, nullptr, nullptr, true, false, nullptr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
util::detail::set_processing_status("TERMINATED", "fatal_error");
|
|
fflush(stderr);
|
|
fflush(stdout);
|
|
_exit(1);
|
|
}
|
|
|
|
void Reporter::FatalErrorWithCore(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
|
|
// Always log to stderr.
|
|
DoLog("fatal error", nullptr, stderr, nullptr, nullptr, true, false, nullptr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
util::detail::set_processing_status("TERMINATED", "fatal_error");
|
|
abort();
|
|
}
|
|
|
|
void Reporter::ExprRuntimeError(const detail::Expr* expr, const char* fmt, ...) {
|
|
++errors;
|
|
|
|
ODesc d;
|
|
expr->Describe(&d);
|
|
|
|
PushLocation(expr->GetLocationInfo());
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(errors_to_stderr) ? stderr : nullptr;
|
|
DoLog("expression error", reporter_error, out, nullptr, nullptr, true, true, d.Description(), fmt, ap);
|
|
va_end(ap);
|
|
PopLocation();
|
|
|
|
if ( abort_on_scripting_errors )
|
|
abort();
|
|
|
|
throw InterpreterException();
|
|
}
|
|
|
|
void Reporter::ExprRuntimeWarning(const detail::Expr* expr, const char* fmt, ...) {
|
|
ODesc d;
|
|
expr->Describe(&d);
|
|
|
|
PushLocation(expr->GetLocationInfo());
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(warnings_to_stderr) ? stderr : nullptr;
|
|
DoLog("expression warning", reporter_warning, out, nullptr, nullptr, true, true, d.Description(), fmt, ap);
|
|
va_end(ap);
|
|
PopLocation();
|
|
}
|
|
|
|
void Reporter::RuntimeError(const detail::Location* location, const char* fmt, ...) {
|
|
++errors;
|
|
PushLocation(location);
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(errors_to_stderr) ? stderr : nullptr;
|
|
DoLog("runtime error", reporter_error, out, nullptr, nullptr, true, true, "", fmt, ap);
|
|
va_end(ap);
|
|
PopLocation();
|
|
|
|
if ( abort_on_scripting_errors )
|
|
abort();
|
|
|
|
throw InterpreterException();
|
|
}
|
|
|
|
void Reporter::CPPRuntimeError(const char* fmt, ...) {
|
|
++errors;
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(errors_to_stderr) ? stderr : nullptr;
|
|
DoLog("runtime error in compiled code", reporter_error, out, nullptr, nullptr, true, true, "", fmt, ap);
|
|
va_end(ap);
|
|
|
|
if ( abort_on_scripting_errors )
|
|
abort();
|
|
|
|
throw InterpreterException();
|
|
}
|
|
|
|
void Reporter::CPPRuntimeWarning(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(warnings_to_stderr) ? stderr : nullptr;
|
|
DoLog("runtime warning in compiled code", reporter_error, out, nullptr, nullptr, true, true, "", fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::InternalError(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
|
|
// Always log to stderr.
|
|
DoLog("internal error", nullptr, stderr, nullptr, nullptr, true, false, nullptr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
util::detail::set_processing_status("TERMINATED", "internal_error");
|
|
abort();
|
|
}
|
|
|
|
void Reporter::AnalyzerError(analyzer::Analyzer* a, const char* fmt, ...) {
|
|
if ( a )
|
|
a->SetSkip(true);
|
|
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
// Always log to stderr.
|
|
// TODO: would be nice to also log a call stack.
|
|
DoLog("analyzer error", reporter_error, stderr, nullptr, nullptr, true, true, nullptr, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::InternalWarning(const char* fmt, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
FILE* out = EmitToStderr(warnings_to_stderr) ? stderr : nullptr;
|
|
// TODO: would be nice to also log a call stack.
|
|
DoLog("internal warning", reporter_warning, out, nullptr, nullptr, true, true, nullptr, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::Syslog(const char* fmt, ...) {
|
|
if ( run_state::reading_traces )
|
|
return;
|
|
|
|
if ( ! syslog_open ) {
|
|
openlog("bro", 0, LOG_LOCAL5);
|
|
syslog_open = true;
|
|
}
|
|
|
|
va_list ap;
|
|
va_start(ap, fmt);
|
|
vsyslog(LOG_NOTICE, fmt, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::WeirdHelper(EventHandlerPtr event, ValPList vl, const char* fmt_name, ...) {
|
|
va_list ap;
|
|
va_start(ap, fmt_name);
|
|
DoLog("weird", event, nullptr, nullptr, &vl, false, false, nullptr, fmt_name, ap);
|
|
va_end(ap);
|
|
}
|
|
|
|
void Reporter::UpdateWeirdStats(const char* name) {
|
|
++weird_count;
|
|
++weird_count_by_type[name];
|
|
}
|
|
|
|
class NetWeirdTimer final : public detail::Timer {
|
|
public:
|
|
NetWeirdTimer(double t, const char* name, double timeout)
|
|
: detail::Timer(t + timeout, detail::TIMER_NET_WEIRD_EXPIRE), weird_name(name) {}
|
|
|
|
void Dispatch(double t, bool is_expire) override { reporter->ResetNetWeird(weird_name); }
|
|
|
|
std::string weird_name;
|
|
};
|
|
|
|
class FlowWeirdTimer final : public detail::Timer {
|
|
public:
|
|
using IPPair = std::pair<IPAddr, IPAddr>;
|
|
|
|
FlowWeirdTimer(double t, IPPair p, double timeout)
|
|
: detail::Timer(t + timeout, detail::TIMER_FLOW_WEIRD_EXPIRE), endpoints(std::move(p)) {}
|
|
|
|
void Dispatch(double t, bool is_expire) override { reporter->ResetFlowWeird(endpoints.first, endpoints.second); }
|
|
|
|
IPPair endpoints;
|
|
};
|
|
|
|
class ConnTupleWeirdTimer final : public detail::Timer {
|
|
public:
|
|
using ConnTuple = Reporter::ConnTuple;
|
|
|
|
ConnTupleWeirdTimer(double t, ConnTuple id, double timeout)
|
|
: detail::Timer(t + timeout, detail::TIMER_CONN_TUPLE_WEIRD_EXPIRE), conn_id(std::move(id)) {}
|
|
|
|
void Dispatch(double t, bool is_expire) override { reporter->ResetExpiredConnWeird(conn_id); }
|
|
|
|
ConnTuple conn_id;
|
|
};
|
|
|
|
void Reporter::ResetNetWeird(const std::string& name) { net_weird_state.erase(name); }
|
|
|
|
void Reporter::ResetFlowWeird(const IPAddr& orig, const IPAddr& resp) {
|
|
flow_weird_state.erase(std::make_pair(orig, resp));
|
|
}
|
|
|
|
void Reporter::ResetExpiredConnWeird(const ConnTuple& id) { expired_conn_weird_state.erase(id); }
|
|
|
|
Reporter::PermitWeird Reporter::CheckGlobalWeirdLists(const char* name) {
|
|
if ( WeirdOnSamplingWhiteList(name) )
|
|
return PermitWeird::Allow;
|
|
|
|
if ( WeirdOnGlobalList(name) )
|
|
// We track weirds on the global list through the "net_weird" table.
|
|
return PermitNetWeird(name) ? PermitWeird::Allow : PermitWeird::Deny;
|
|
|
|
return PermitWeird::Unknown;
|
|
}
|
|
|
|
bool Reporter::PermitNetWeird(const char* name) {
|
|
auto& count = net_weird_state[name];
|
|
++count;
|
|
|
|
if ( count == 1 )
|
|
detail::timer_mgr->Add(new NetWeirdTimer(run_state::network_time, name, weird_sampling_duration));
|
|
|
|
if ( count <= weird_sampling_threshold )
|
|
return true;
|
|
|
|
auto num_above_threshold = count - weird_sampling_threshold;
|
|
if ( weird_sampling_rate )
|
|
return num_above_threshold % weird_sampling_rate == 0;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool Reporter::PermitFlowWeird(const char* name, const IPAddr& orig, const IPAddr& resp) {
|
|
auto endpoints = std::make_pair(orig, resp);
|
|
auto& map = flow_weird_state[endpoints];
|
|
|
|
if ( map.empty() )
|
|
detail::timer_mgr->Add(new FlowWeirdTimer(run_state::network_time, endpoints, weird_sampling_duration));
|
|
|
|
auto& count = map[name];
|
|
++count;
|
|
|
|
if ( count <= weird_sampling_threshold )
|
|
return true;
|
|
|
|
auto num_above_threshold = count - weird_sampling_threshold;
|
|
if ( weird_sampling_rate )
|
|
return num_above_threshold % weird_sampling_rate == 0;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
bool Reporter::PermitExpiredConnWeird(const char* name, const RecordVal& conn_id) {
|
|
if ( ! conn_id.HasField("orig_h") || ! conn_id.HasField("resp_h") || ! conn_id.HasField("orig_p") ||
|
|
! conn_id.HasField("resp_p") )
|
|
return false;
|
|
|
|
auto conn_tuple =
|
|
std::make_tuple(conn_id.GetFieldAs<AddrVal>("orig_h"), conn_id.GetFieldAs<AddrVal>("resp_h"),
|
|
conn_id.GetFieldAs<PortVal>("orig_p")->Port(), conn_id.GetFieldAs<PortVal>("resp_p")->Port(),
|
|
conn_id.GetFieldAs<PortVal>("resp_p")->PortType());
|
|
|
|
auto& map = expired_conn_weird_state[conn_tuple];
|
|
|
|
if ( map.empty() )
|
|
detail::timer_mgr->Add(
|
|
new ConnTupleWeirdTimer(run_state::network_time, std::move(conn_tuple), weird_sampling_duration));
|
|
|
|
auto& count = map[name];
|
|
++count;
|
|
|
|
if ( count <= weird_sampling_threshold )
|
|
return true;
|
|
|
|
auto num_above_threshold = count - weird_sampling_threshold;
|
|
|
|
if ( weird_sampling_rate )
|
|
return num_above_threshold % weird_sampling_rate == 0;
|
|
else
|
|
return false;
|
|
}
|
|
|
|
void Reporter::Weird(const char* name, const char* addl, const char* source) {
|
|
UpdateWeirdStats(name);
|
|
|
|
if ( ! WeirdOnSamplingWhiteList(name) ) {
|
|
if ( ! PermitNetWeird(name) )
|
|
return;
|
|
}
|
|
|
|
WeirdHelper(net_weird, {new StringVal(addl), new StringVal(source)}, "%s", name);
|
|
}
|
|
|
|
void Reporter::Weird(file_analysis::File* f, const char* name, const char* addl, const char* source) {
|
|
UpdateWeirdStats(name);
|
|
|
|
switch ( CheckGlobalWeirdLists(name) ) {
|
|
case PermitWeird::Allow: break;
|
|
case PermitWeird::Deny: return;
|
|
case PermitWeird::Unknown:
|
|
if ( ! f->PermitWeird(name, weird_sampling_threshold, weird_sampling_rate, weird_sampling_duration) )
|
|
return;
|
|
}
|
|
|
|
WeirdHelper(file_weird, {f->ToVal()->Ref(), new StringVal(addl), new StringVal(source)}, "%s", name);
|
|
}
|
|
|
|
void Reporter::Weird(Connection* conn, const char* name, const char* addl, const char* source) {
|
|
UpdateWeirdStats(name);
|
|
|
|
switch ( CheckGlobalWeirdLists(name) ) {
|
|
case PermitWeird::Allow: break;
|
|
case PermitWeird::Deny: return;
|
|
case PermitWeird::Unknown:
|
|
if ( ! conn->PermitWeird(name, weird_sampling_threshold, weird_sampling_rate, weird_sampling_duration) )
|
|
return;
|
|
}
|
|
|
|
WeirdHelper(conn_weird, {conn->GetVal()->Ref(), new StringVal(addl), new StringVal(source)}, "%s", name);
|
|
}
|
|
|
|
void Reporter::Weird(RecordValPtr conn_id, StringValPtr uid, const char* name, const char* addl, const char* source) {
|
|
UpdateWeirdStats(name);
|
|
|
|
switch ( CheckGlobalWeirdLists(name) ) {
|
|
case PermitWeird::Allow: break;
|
|
case PermitWeird::Deny: return;
|
|
case PermitWeird::Unknown:
|
|
if ( ! PermitExpiredConnWeird(name, *conn_id) )
|
|
return;
|
|
}
|
|
|
|
WeirdHelper(expired_conn_weird, {conn_id.release(), uid.release(), new StringVal(addl), new StringVal(source)},
|
|
"%s", name);
|
|
}
|
|
|
|
void Reporter::Weird(const IPAddr& orig, const IPAddr& resp, const char* name, const char* addl, const char* source) {
|
|
UpdateWeirdStats(name);
|
|
|
|
switch ( CheckGlobalWeirdLists(name) ) {
|
|
case PermitWeird::Allow: break;
|
|
case PermitWeird::Deny: return;
|
|
case PermitWeird::Unknown:
|
|
if ( ! PermitFlowWeird(name, orig, resp) )
|
|
return;
|
|
}
|
|
|
|
WeirdHelper(flow_weird, {new AddrVal(orig), new AddrVal(resp), new StringVal(addl), new StringVal(source)}, "%s",
|
|
name);
|
|
}
|
|
|
|
void Reporter::Deprecation(std::string_view msg, const detail::Location* loc1, const detail::Location* loc2) {
|
|
if ( ignore_deprecations )
|
|
return;
|
|
|
|
if ( loc1 || loc2 )
|
|
PushLocation(loc1, loc2);
|
|
|
|
Warning("%.*s", static_cast<int>(msg.size()), msg.data());
|
|
|
|
if ( loc1 || loc2 )
|
|
PopLocation();
|
|
}
|
|
|
|
void Reporter::DoLog(const char* prefix, EventHandlerPtr event, FILE* out, Connection* conn, ValPList* addl,
|
|
bool location, bool time, const char* postfix, const char* fmt, va_list ap) {
|
|
constexpr size_t DEFAULT_BUFFER_SIZE = 512;
|
|
static char tmp[DEFAULT_BUFFER_SIZE];
|
|
|
|
int size = sizeof(tmp);
|
|
char* buffer = tmp;
|
|
char* allocated = nullptr;
|
|
|
|
std::string loc_str;
|
|
|
|
if ( location ) {
|
|
int loc_line = 0;
|
|
|
|
if ( locations.size() ) {
|
|
ODesc d;
|
|
|
|
std::pair<const detail::Location*, const detail::Location*> locs = locations.back();
|
|
|
|
if ( locs.first ) {
|
|
if ( locs.first != &detail::no_location )
|
|
locs.first->Describe(&d);
|
|
|
|
else
|
|
d.Add("<no location>");
|
|
|
|
if ( locs.second ) {
|
|
d.Add(" and ");
|
|
|
|
if ( locs.second != &detail::no_location )
|
|
locs.second->Describe(&d);
|
|
|
|
else
|
|
d.Add("<no location>");
|
|
}
|
|
}
|
|
|
|
loc_str = d.Description();
|
|
}
|
|
|
|
else if ( filename && *filename ) {
|
|
// Take from globals.
|
|
loc_str = filename;
|
|
char tmp[32];
|
|
snprintf(tmp, 32, "%d", line_number);
|
|
loc_str += std::string(", line ") + std::string(tmp);
|
|
}
|
|
}
|
|
|
|
// Figure out how big of a buffer is needed here.
|
|
va_list ap_copy;
|
|
va_copy(ap_copy, ap);
|
|
int req_buffer_size = vsnprintf(nullptr, 0, fmt, ap_copy);
|
|
|
|
if ( req_buffer_size < 0 ) {
|
|
va_end(ap_copy);
|
|
FatalError("out of memory in Reporter");
|
|
}
|
|
|
|
size_t new_buffer_size = req_buffer_size;
|
|
|
|
if ( postfix && *postfix )
|
|
new_buffer_size += strlen(postfix) + 10;
|
|
|
|
// Add one byte for a null terminator.
|
|
new_buffer_size++;
|
|
|
|
if ( new_buffer_size > DEFAULT_BUFFER_SIZE ) {
|
|
buffer = (char*)malloc(new_buffer_size);
|
|
if ( ! buffer ) {
|
|
va_end(ap_copy);
|
|
FatalError("out of memory in Reporter");
|
|
}
|
|
|
|
allocated = buffer;
|
|
}
|
|
|
|
va_end(ap_copy);
|
|
|
|
req_buffer_size = vsnprintf(buffer, new_buffer_size, fmt, ap);
|
|
if ( req_buffer_size < 0 )
|
|
FatalError("out of memory in Reporter");
|
|
|
|
if ( postfix && *postfix )
|
|
// Note, if you change this fmt string, adjust the additional
|
|
// buffer size above.
|
|
snprintf(buffer + strlen(buffer), size - strlen(buffer), " (%s)", postfix);
|
|
|
|
bool raise_event = true;
|
|
|
|
if ( via_events && ! in_error_handler ) {
|
|
if ( locations.size() ) {
|
|
auto locs = locations.back();
|
|
raise_event = PLUGIN_HOOK_WITH_RESULT(HOOK_REPORTER,
|
|
HookReporter(prefix, event, conn, addl, location, locs.first,
|
|
locs.second, time, buffer),
|
|
true);
|
|
}
|
|
else
|
|
raise_event = PLUGIN_HOOK_WITH_RESULT(HOOK_REPORTER,
|
|
HookReporter(prefix, event, conn, addl, location, nullptr, nullptr,
|
|
time, buffer),
|
|
true);
|
|
}
|
|
|
|
if ( raise_event && event && via_events && ! in_error_handler ) {
|
|
auto vl_size = 1 + (bool)time + (bool)location + (bool)conn + (addl ? addl->length() : 0);
|
|
|
|
Args vl;
|
|
vl.reserve(vl_size);
|
|
|
|
if ( time )
|
|
vl.emplace_back(
|
|
make_intrusive<TimeVal>(run_state::network_time ? run_state::network_time : util::current_time()));
|
|
|
|
vl.emplace_back(make_intrusive<StringVal>(buffer));
|
|
|
|
if ( location )
|
|
vl.emplace_back(make_intrusive<StringVal>(loc_str.c_str()));
|
|
|
|
if ( conn )
|
|
vl.emplace_back(conn->GetVal());
|
|
|
|
if ( addl )
|
|
for ( auto v : *addl )
|
|
vl.emplace_back(AdoptRef{}, v);
|
|
|
|
if ( conn )
|
|
conn->EnqueueEvent(event, nullptr, std::move(vl));
|
|
else
|
|
event_mgr.Enqueue(event, std::move(vl));
|
|
}
|
|
else {
|
|
if ( addl ) {
|
|
for ( const auto& av : *addl )
|
|
Unref(av);
|
|
}
|
|
}
|
|
|
|
if ( out ) {
|
|
std::string s = "";
|
|
|
|
if ( run_state::zeek_start_network_time != 0.0 ) {
|
|
char tmp[32];
|
|
snprintf(tmp, 32, "%.6f", run_state::network_time);
|
|
s += std::string(tmp) + " ";
|
|
}
|
|
|
|
if ( prefix && *prefix ) {
|
|
if ( loc_str != "" )
|
|
s += std::string(prefix) + " in " + loc_str + ": ";
|
|
else
|
|
s += std::string(prefix) + ": ";
|
|
}
|
|
|
|
else {
|
|
if ( loc_str != "" )
|
|
s += loc_str + ": ";
|
|
}
|
|
|
|
s += buffer;
|
|
|
|
#ifdef ENABLE_ZEEK_UNIT_TESTS
|
|
if ( doctest::is_running_in_test ) {
|
|
try {
|
|
MESSAGE(s);
|
|
} catch ( const doctest::detail::TestFailureException& e ) {
|
|
// If doctest throws an exception, just write the string out to stdout
|
|
// like normal, just so it's captured somewhere.
|
|
fprintf(out, "%s\n", s.c_str());
|
|
}
|
|
}
|
|
else {
|
|
#endif
|
|
s += "\n";
|
|
fprintf(out, "%s", s.c_str());
|
|
#ifdef ENABLE_ZEEK_UNIT_TESTS
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// If the buffer got reallocated because it was too small, free the reallocated one.
|
|
if ( allocated )
|
|
free(allocated);
|
|
}
|
|
|
|
bool Reporter::EmitToStderr(bool flag) { return flag || ! run_state::detail::zeek_init_done; }
|
|
|
|
ScriptLocationScope::ScriptLocationScope(const zeek::detail::Frame* frame) {
|
|
zeek::reporter->PushLocation(frame->GetCallLocation());
|
|
}
|
|
|
|
ScriptLocationScope::~ScriptLocationScope() { zeek::reporter->PopLocation(); }
|
|
|
|
} // namespace zeek
|