zeek/src/file_analysis/File.cc

536 lines
17 KiB
C++

// See the file "COPYING" in the main distribution directory for copyright.
#include "zeek/file_analysis/File.h"
#include <utility>
#include "zeek/Conn.h"
#include "zeek/Event.h"
#include "zeek/Reporter.h"
#include "zeek/RuleMatcher.h"
#include "zeek/Type.h"
#include "zeek/Val.h"
#include "zeek/analyzer/Analyzer.h"
#include "zeek/file_analysis/Analyzer.h"
#include "zeek/file_analysis/FileReassembler.h"
#include "zeek/file_analysis/FileTimer.h"
#include "zeek/file_analysis/Manager.h"
#include "zeek/file_analysis/analyzer/extract/Extract.h"
#include "zeek/file_analysis/analyzer/extract/events.bif.h"
namespace zeek::file_analysis {
static TableValPtr empty_connection_table() {
auto tbl_index = make_intrusive<TypeList>(id::conn_id);
tbl_index->Append(id::conn_id);
auto tbl_type = make_intrusive<TableType>(std::move(tbl_index), id::connection);
return make_intrusive<TableVal>(std::move(tbl_type));
}
static RecordValPtr get_conn_id_val(const Connection* conn) {
auto v = make_intrusive<RecordVal>(id::conn_id);
v->Assign(0, make_intrusive<AddrVal>(conn->OrigAddr()));
v->Assign(1, val_mgr->Port(ntohs(conn->OrigPort()), conn->ConnTransport()));
v->Assign(2, make_intrusive<AddrVal>(conn->RespAddr()));
v->Assign(3, val_mgr->Port(ntohs(conn->RespPort()), conn->ConnTransport()));
v->Assign(4, conn->KeyProto());
return v;
}
int File::id_idx = -1;
int File::parent_id_idx = -1;
int File::source_idx = -1;
int File::is_orig_idx = -1;
int File::conns_idx = -1;
int File::last_active_idx = -1;
int File::seen_bytes_idx = -1;
int File::total_bytes_idx = -1;
int File::missing_bytes_idx = -1;
int File::overflow_bytes_idx = -1;
int File::timeout_interval_idx = -1;
int File::bof_buffer_size_idx = -1;
int File::bof_buffer_idx = -1;
int File::meta_mime_type_idx = -1;
int File::meta_mime_types_idx = -1;
int File::meta_inferred_idx = -1;
void File::StaticInit() {
if ( id_idx != -1 )
return;
id_idx = Idx("id", id::fa_file);
parent_id_idx = Idx("parent_id", id::fa_file);
source_idx = Idx("source", id::fa_file);
is_orig_idx = Idx("is_orig", id::fa_file);
conns_idx = Idx("conns", id::fa_file);
last_active_idx = Idx("last_active", id::fa_file);
seen_bytes_idx = Idx("seen_bytes", id::fa_file);
total_bytes_idx = Idx("total_bytes", id::fa_file);
missing_bytes_idx = Idx("missing_bytes", id::fa_file);
overflow_bytes_idx = Idx("overflow_bytes", id::fa_file);
timeout_interval_idx = Idx("timeout_interval", id::fa_file);
bof_buffer_size_idx = Idx("bof_buffer_size", id::fa_file);
bof_buffer_idx = Idx("bof_buffer", id::fa_file);
meta_mime_type_idx = Idx("mime_type", id::fa_metadata);
meta_mime_types_idx = Idx("mime_types", id::fa_metadata);
meta_inferred_idx = Idx("inferred", id::fa_metadata);
}
File::File(const std::string& file_id, const std::string& source_name, Connection* conn, zeek::Tag tag, bool is_orig)
: id(file_id), val(nullptr), analyzers(this) {
StaticInit();
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Creating new File object", file_id.c_str());
val = make_intrusive<RecordVal>(id::fa_file);
val->Assign(id_idx, file_id);
SetSource(source_name);
if ( conn ) {
val->Assign(is_orig_idx, is_orig);
UpdateConnectionFields(conn, is_orig);
}
UpdateLastActivityTime();
}
File::~File() {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Destroying File object", id.c_str());
delete file_reassembler;
for ( auto a : done_analyzers )
delete a;
}
void File::UpdateLastActivityTime() { val->AssignTime(last_active_idx, run_state::network_time); }
double File::GetLastActivityTime() const { return val->GetFieldAs<TimeVal>(last_active_idx); }
bool File::UpdateConnectionFields(Connection* conn, bool is_orig) {
if ( ! conn )
return false;
auto conns = val->GetField(conns_idx);
if ( ! conns ) {
auto empty_conn_table = empty_connection_table();
conns = empty_conn_table;
val->Assign(conns_idx, std::move(empty_conn_table));
}
auto idx = get_conn_id_val(conn);
if ( conns->AsTableVal()->FindOrDefault(idx) )
return false;
conns->AsTableVal()->Assign(std::move(idx), conn->GetVal());
return true;
}
void File::RaiseFileOverNewConnection(Connection* conn, bool is_orig) {
if ( conn && FileEventAvailable(file_over_new_connection) ) {
FileEvent(file_over_new_connection, {
val,
conn->GetVal(),
val_mgr->Bool(is_orig),
});
}
}
uint64_t File::LookupFieldDefaultCount(int idx) const {
if ( val->HasRawField(idx) )
return val->GetFieldAs<zeek::CountVal>(idx);
auto v = val->GetFieldOrDefault(idx);
return v->AsCount();
}
double File::LookupFieldDefaultInterval(int idx) const {
auto v = val->GetFieldOrDefault(idx);
return v->AsInterval();
}
int File::Idx(const std::string& field, const RecordType* type) {
int rval = type->FieldOffset(field.c_str());
if ( rval < 0 )
reporter->InternalError("Unknown %s field: %s", type->GetName().c_str(), field.c_str());
return rval;
}
std::string File::GetSource() const {
const auto& v = val->GetField(source_idx);
return v ? v->AsString()->CheckString() : std::string();
}
void File::SetSource(const std::string& source) { val->Assign(source_idx, source); }
double File::GetTimeoutInterval() const { return LookupFieldDefaultInterval(timeout_interval_idx); }
void File::SetTimeoutInterval(double interval) { val->AssignInterval(timeout_interval_idx, interval); }
bool File::SetExtractionLimit(RecordValPtr args, uint64_t bytes) {
Analyzer* a = analyzers.Find(file_mgr->GetComponentTag("EXTRACT"), std::move(args));
if ( ! a )
return false;
auto* e = dynamic_cast<file_analysis::detail::Extract*>(a);
if ( ! e )
return false;
e->SetLimit(bytes);
return true;
}
void File::SetTotalBytes(uint64_t size) {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Total bytes %" PRIu64, id.c_str(), size);
val->Assign(total_bytes_idx, size);
}
bool File::IsComplete() const {
// If total_bytes hasn't been initialized yet, file is certainly not complete.
if ( ! val->HasRawField(total_bytes_idx) )
return false;
auto total = val->GetFieldAs<zeek::CountVal>(total_bytes_idx);
if ( stream_offset >= total )
return true;
return false;
}
void File::ScheduleInactivityTimer() const {
zeek::detail::timer_mgr->Add(new detail::FileTimer(run_state::network_time, id, GetTimeoutInterval()));
}
bool File::AddAnalyzer(zeek::Tag tag, RecordValPtr args) {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Queuing addition of %s analyzer", id.c_str(),
file_mgr->GetComponentName(tag).c_str());
if ( done )
return false;
return analyzers.QueueAdd(tag, std::move(args)) != nullptr;
}
bool File::RemoveAnalyzer(zeek::Tag tag, RecordValPtr args) {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Queuing remove of %s analyzer", id.c_str(),
file_mgr->GetComponentName(tag).c_str());
return done ? false : analyzers.QueueRemove(tag, std::move(args));
}
void File::EnableReassembly() { reassembly_enabled = true; }
void File::DisableReassembly() {
reassembly_enabled = false;
delete file_reassembler;
file_reassembler = nullptr;
}
void File::SetReassemblyBuffer(uint64_t max) { reassembly_max_buffer = max; }
bool File::SetMime(const std::string& mime_type) {
if ( mime_type.empty() || bof_buffer.size != 0 || did_metadata_inference )
return false;
did_metadata_inference = true;
bof_buffer.full = true;
if ( ! FileEventAvailable(file_sniff) )
return false;
auto meta = make_intrusive<RecordVal>(id::fa_metadata);
meta->Assign(meta_mime_type_idx, mime_type);
meta->Assign(meta_inferred_idx, false);
FileEvent(file_sniff, {val, std::move(meta)});
return true;
}
void File::InferMetadata() {
did_metadata_inference = true;
auto bof_buffer_val = val->GetField(bof_buffer_idx);
if ( ! bof_buffer_val ) {
if ( bof_buffer.size == 0 )
return;
String* bs = concatenate(bof_buffer.chunks);
val->Assign(bof_buffer_idx, bs);
bof_buffer_val = val->GetField(bof_buffer_idx);
}
if ( ! FileEventAvailable(file_sniff) )
return;
zeek::detail::RuleMatcher::MIME_Matches matches;
const u_char* data = bof_buffer_val->AsString()->Bytes();
uint64_t len = bof_buffer_val->AsString()->Len();
len = std::min(len, LookupFieldDefaultCount(bof_buffer_size_idx));
file_mgr->DetectMIME(data, len, &matches);
auto meta = make_intrusive<RecordVal>(id::fa_metadata);
if ( ! matches.empty() ) {
meta->Assign(meta_mime_type_idx, *(matches.begin()->second.begin()));
meta->Assign(meta_mime_types_idx, file_analysis::GenMIMEMatchesVal(matches));
}
FileEvent(file_sniff, {val, std::move(meta)});
}
bool File::BufferBOF(const u_char* data, uint64_t len) {
if ( bof_buffer.full )
return false;
uint64_t desired_size = LookupFieldDefaultCount(bof_buffer_size_idx);
bof_buffer.chunks.push_back(new String(data, len, false));
bof_buffer.size += len;
if ( bof_buffer.size < desired_size )
return true;
bof_buffer.full = true;
if ( bof_buffer.size > 0 ) {
String* bs = concatenate(bof_buffer.chunks);
val->Assign(bof_buffer_idx, bs);
}
return false;
}
void File::DeliverStream(const u_char* data, uint64_t len) {
bool bof_was_full = bof_buffer.full;
// Buffer enough data for the BOF buffer
BufferBOF(data, len);
if ( ! did_metadata_inference && bof_buffer.full && LookupFieldDefaultCount(missing_bytes_idx) == 0 )
InferMetadata();
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] %" PRIu64 " stream bytes in at offset %" PRIu64 "; %s [%s%s]", id.c_str(), len,
stream_offset, IsComplete() ? "complete" : "incomplete",
util::fmt_bytes((const char*)data, std::min((uint64_t)40, len)), len > 40 ? "..." : "");
for ( const auto& entry : analyzers ) {
auto* a = entry.value;
DBG_LOG(DBG_FILE_ANALYSIS, "stream delivery to analyzer %s", file_mgr->GetComponentName(a->Tag()).c_str());
if ( ! a->GotStreamDelivery() ) {
DBG_LOG(DBG_FILE_ANALYSIS, "skipping stream delivery to analyzer %s",
file_mgr->GetComponentName(a->Tag()).c_str());
int num_bof_chunks_behind = bof_buffer.chunks.size();
if ( ! bof_was_full )
// We just added a chunk to the BOF buffer, don't count it
// as it will get delivered on its own.
num_bof_chunks_behind -= 1;
uint64_t bytes_delivered = 0;
// Catch this analyzer up with the BOF buffer.
for ( int i = 0; i < num_bof_chunks_behind; ++i ) {
if ( ! a->Skipping() ) {
if ( ! a->DeliverStream(bof_buffer.chunks[i]->Bytes(), bof_buffer.chunks[i]->Len()) ) {
a->SetSkip(true);
analyzers.QueueRemove(a->Tag(), a->GetArgs());
}
}
bytes_delivered += bof_buffer.chunks[i]->Len();
}
a->SetGotStreamDelivery();
// May need to catch analyzer up on missed gap?
// Analyzer should be fully caught up to stream_offset now.
}
if ( ! a->Skipping() ) {
if ( ! a->DeliverStream(data, len) ) {
a->SetSkip(true);
analyzers.QueueRemove(a->Tag(), a->GetArgs());
}
}
}
stream_offset += len;
seen_bytes += len;
}
void File::DeliverChunk(const u_char* data, uint64_t len, uint64_t offset) {
if ( std::numeric_limits<uint64_t>::max() - offset < len ) {
reporter->Weird(this, "file_offset_overflow", zeek::util::fmt("offset=%" PRIx64 " len=%" PRIx64, offset, len),
GetSource().c_str());
len = std::numeric_limits<uint64_t>::max() - offset;
}
// Potentially handle reassembly and deliver to the stream analyzers.
if ( file_reassembler ) {
if ( reassembly_max_buffer > 0 && reassembly_max_buffer < file_reassembler->TotalSize() ) {
uint64_t current_offset = stream_offset;
uint64_t gap_bytes = file_reassembler->Flush();
overflow_bytes += gap_bytes;
if ( FileEventAvailable(file_reassembly_overflow) ) {
FileEvent(file_reassembly_overflow, {val, val_mgr->Count(current_offset), val_mgr->Count(gap_bytes)});
}
}
// Forward data to the reassembler.
file_reassembler->NewBlock(run_state::network_time, offset, len, data);
}
else if ( stream_offset == offset ) {
// This is the normal case where a file is transferred linearly.
// Nothing special should be done here.
DeliverStream(data, len);
}
else if ( reassembly_enabled ) {
// This is data that doesn't match the offset and the reassembler
// needs to be enabled.
file_reassembler = new FileReassembler(this, stream_offset);
file_reassembler->NewBlock(run_state::network_time, offset, len, data);
}
else {
// We can't reassemble so we throw out the data for streaming.
overflow_bytes += len;
}
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] %" PRIu64 " chunk bytes in at offset %" PRIu64 "; %s [%s%s]", id.c_str(), len,
offset, IsComplete() ? "complete" : "incomplete",
util::fmt_bytes((const char*)data, std::min((uint64_t)40, len)), len > 40 ? "..." : "");
for ( const auto& entry : analyzers ) {
auto* a = entry.value;
DBG_LOG(DBG_FILE_ANALYSIS, "chunk delivery to analyzer %s", file_mgr->GetComponentName(a->Tag()).c_str());
if ( ! a->Skipping() ) {
if ( ! a->DeliverChunk(data, len, offset) ) {
a->SetSkip(true);
analyzers.QueueRemove(a->Tag(), a->GetArgs());
}
}
}
if ( IsComplete() )
EndOfFile();
}
void File::DoneWithAnalyzer(Analyzer* analyzer) { done_analyzers.push_back(analyzer); }
void File::DataIn(const u_char* data, uint64_t len, uint64_t offset) {
analyzers.DrainModifications();
DeliverChunk(data, len, offset);
analyzers.DrainModifications();
}
void File::DataIn(const u_char* data, uint64_t len) {
analyzers.DrainModifications();
DeliverChunk(data, len, stream_offset);
analyzers.DrainModifications();
}
void File::EndOfFile() {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] End of file", id.c_str());
if ( done )
return;
if ( file_reassembler ) {
file_reassembler->Flush();
}
// Mark the bof_buffer as full in case it isn't yet
// so that the whole thing can be flushed out to
// any stream analyzers.
if ( ! bof_buffer.full ) {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] File over but bof_buffer not full.", id.c_str());
bof_buffer.full = true;
DeliverStream((const u_char*)"", 0);
}
analyzers.DrainModifications();
done = true;
for ( const auto& entry : analyzers ) {
auto* a = entry.value;
if ( ! a->EndOfFile() )
analyzers.QueueRemove(a->Tag(), a->GetArgs());
}
FileEvent(file_state_remove);
analyzers.DrainModifications();
}
void File::Gap(uint64_t offset, uint64_t len) {
DBG_LOG(DBG_FILE_ANALYSIS, "[%s] Gap of size %" PRIu64 " at offset %" PRIu64, id.c_str(), len, offset);
if ( file_reassembler && ! file_reassembler->IsCurrentlyFlushing() ) {
file_reassembler->FlushTo(offset + len);
// The reassembler will call us back with all the gaps we need to know.
return;
}
if ( ! bof_buffer.full ) {
DBG_LOG(DBG_FILE_ANALYSIS,
"[%s] File gap before bof_buffer filled, continued without attempting to fill "
"bof_buffer.",
id.c_str());
bof_buffer.full = true;
DeliverStream((const u_char*)"", 0);
}
for ( const auto& entry : analyzers ) {
auto* a = entry.value;
if ( ! a->Undelivered(offset, len) )
analyzers.QueueRemove(a->Tag(), a->GetArgs());
}
if ( FileEventAvailable(file_gap) )
FileEvent(file_gap, {val, val_mgr->Count(offset), val_mgr->Count(len)});
analyzers.DrainModifications();
stream_offset += len;
missing_bytes += len;
}
bool File::FileEventAvailable(EventHandlerPtr h) { return h && ! file_mgr->IsIgnored(id); }
void File::FileEvent(EventHandlerPtr h) {
if ( ! FileEventAvailable(h) )
return;
FileEvent(h, Args{val});
}
void File::FileEvent(EventHandlerPtr h, Args args) {
val->Assign(seen_bytes_idx, seen_bytes);
val->Assign(missing_bytes_idx, missing_bytes);
val->Assign(overflow_bytes_idx, overflow_bytes);
event_mgr.Enqueue(h, std::move(args));
if ( h == file_new || h == file_over_new_connection || h == file_sniff || h == file_timeout ||
h == file_extraction_limit ) {
// immediate feedback is required for these events.
event_mgr.Drain();
analyzers.DrainModifications();
}
}
bool File::PermitWeird(const char* name, uint64_t threshold, uint64_t rate, double duration) {
return zeek::detail::PermitWeird(weird_state, name, threshold, rate, duration);
}
} // namespace zeek::file_analysis