Skip to content

Commit

Permalink
new(falco): add json_include_message_property option
Browse files Browse the repository at this point in the history
Signed-off-by: Luca Guerra <[email protected]>
  • Loading branch information
LucaGuerra committed Sep 11, 2024
1 parent 0f26e3c commit 4a33e78
Show file tree
Hide file tree
Showing 8 changed files with 99 additions and 44 deletions.
9 changes: 9 additions & 0 deletions falco.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -518,6 +518,15 @@ json_output: false
# case.
json_include_output_property: true

# [Incubating] `json_include_message_property`
#
# When using JSON output in Falco, you have the option to include the formatted
# rule output without timestamp or priority. For instance, if a rule specifies
# an "output" property like "Opened process %proc.name" the "message" field will
# only contain "Opened process bash" whereas the "output" field will contain more
# information.
json_include_message_property: false

# [Stable] `json_include_tags_property`
#
# When using JSON output in Falco, you have the option to include the "tags"
Expand Down
90 changes: 72 additions & 18 deletions userspace/engine/formats.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -22,10 +22,14 @@ limitations under the License.

falco_formats::falco_formats(std::shared_ptr<const falco_engine> engine,
bool json_include_output_property,
bool json_include_tags_property)
bool json_include_tags_property,
bool json_include_message_property,
bool time_format_iso_8601)
: m_falco_engine(engine),
m_json_include_output_property(json_include_output_property),
m_json_include_tags_property(json_include_tags_property)
m_json_include_tags_property(json_include_tags_property),
m_json_include_message_property(json_include_message_property),
m_time_format_iso_8601(time_format_iso_8601)
{
}

Expand All @@ -37,21 +41,53 @@ std::string falco_formats::format_event(sinsp_evt *evt, const std::string &rule,
const std::string &level, const std::string &format, const std::set<std::string> &tags,
const std::string &hostname, const extra_output_field_t &extra_fields) const
{
std::string line;
std::string prefix_format;
std::string message_format = format;

if(m_time_format_iso_8601)
{
prefix_format = "*%evt.time.iso8601: ";
}
else
{
prefix_format = "*%evt.time: ";
}
prefix_format += level;

if(message_format[0] != '*')
{
message_format = "*" + message_format;
}

std::shared_ptr<sinsp_evt_formatter> formatter;

formatter = m_falco_engine->create_formatter(source, format);
auto prefix_formatter = m_falco_engine->create_formatter(source, prefix_format);
auto message_formatter = m_falco_engine->create_formatter(source, message_format);

// Format the original output string, regardless of output format
formatter->tostring_withformat(evt, line, sinsp_evt_formatter::OF_NORMAL);
// The classic Falco output prefix with time and priority e.g. "13:53:31.726060287: Critical"
std::string prefix;
prefix_formatter->tostring_withformat(evt, prefix, sinsp_evt_formatter::OF_NORMAL);

// The formatted rule message/output
std::string message;
message_formatter->tostring_withformat(evt, message, sinsp_evt_formatter::OF_NORMAL);

// The complete Falco output, e.g. "13:53:31.726060287: Critical Some Event Description (proc_exe=bash)..."
std::string output = prefix + " " + message;

if(formatter->get_output_format() == sinsp_evt_formatter::OF_JSON)
if(message_formatter->get_output_format() == sinsp_evt_formatter::OF_NORMAL)
{
std::string json_fields;
return output;
}
else if(message_formatter->get_output_format() == sinsp_evt_formatter::OF_JSON)
{
std::string json_fields_message;
std::string json_fields_prefix;

// Format the event into a json object with all fields resolved
formatter->tostring(evt, json_fields);
// Resolve message fields
message_formatter->tostring(evt, json_fields_message);
// Resolve prefix (e.g. time) fields
prefix_formatter->tostring(evt, json_fields_prefix);

// For JSON output, the formatter returned a json-as-text
// object containing all the fields in the original format
Expand All @@ -78,16 +114,27 @@ std::string falco_formats::format_event(sinsp_evt *evt, const std::string &rule,

if(m_json_include_output_property)
{
// This is the filled-in output line.
event["output"] = line;
event["output"] = output;
}

if(m_json_include_tags_property)
{
event["tags"] = tags;
}

event["output_fields"] = nlohmann::json::parse(json_fields);
if(m_json_include_message_property)
{
event["message"] = message;
}

event["output_fields"] = nlohmann::json::parse(json_fields_message);

auto prefix_fields = nlohmann::json::parse(json_fields_prefix);
if (prefix_fields.is_object()) {
for (auto const& el : prefix_fields.items()) {
event["output_fields"][el.key()] = el.value();
}
}

for (auto const& ef : extra_fields)
{
Expand All @@ -105,8 +152,8 @@ std::string falco_formats::format_event(sinsp_evt *evt, const std::string &rule,
if(ef.second.second) // raw field
{
std::string json_field_map;
formatter = m_falco_engine->create_formatter(source, fformat);
formatter->tostring_withformat(evt, json_field_map, sinsp_evt_formatter::OF_JSON);
auto field_formatter = m_falco_engine->create_formatter(source, fformat);
field_formatter->tostring_withformat(evt, json_field_map, sinsp_evt_formatter::OF_JSON);
auto json_obj = nlohmann::json::parse(json_field_map);
event["output_fields"][ef.first] = json_obj[ef.first];
} else
Expand All @@ -115,10 +162,11 @@ std::string falco_formats::format_event(sinsp_evt *evt, const std::string &rule,
}
}

line = event.dump();
return event.dump();
}

return line;
// should never get here until we only have OF_NORMAL and OF_JSON
return "INVALID_OUTPUT_FORMAT";
}

std::string falco_formats::format_string(sinsp_evt *evt, const std::string &format, const std::string &source) const
Expand All @@ -137,7 +185,13 @@ std::map<std::string, std::string> falco_formats::get_field_values(sinsp_evt *ev
{
std::shared_ptr<sinsp_evt_formatter> formatter;

formatter = m_falco_engine->create_formatter(source, format);
std::string fformat = format;
if(fformat[0] != '*')
{
fformat = "*" + fformat;
}

formatter = m_falco_engine->create_formatter(source, fformat);

std::map<std::string, std::string> ret;

Expand Down
6 changes: 5 additions & 1 deletion userspace/engine/formats.h
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,9 @@ class falco_formats
public:
falco_formats(std::shared_ptr<const falco_engine> engine,
bool json_include_output_property,
bool json_include_tags_property);
bool json_include_tags_property,
bool json_include_message_property,
bool time_format_iso_8601);
virtual ~falco_formats();

std::string format_event(sinsp_evt *evt, const std::string &rule, const std::string &source,
Expand All @@ -42,4 +44,6 @@ class falco_formats
std::shared_ptr<const falco_engine> m_falco_engine;
bool m_json_include_output_property;
bool m_json_include_tags_property;
bool m_json_include_message_property;
bool m_time_format_iso_8601;
};
1 change: 1 addition & 0 deletions userspace/falco/app/actions/init_outputs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,7 @@ falco::app::run_result falco::app::actions::init_outputs(falco::app::state& s)
s.config->m_json_output,
s.config->m_json_include_output_property,
s.config->m_json_include_tags_property,
s.config->m_json_include_message_property,
s.config->m_output_timeout,
s.config->m_buffered_outputs,
s.config->m_outputs_queue_capacity,
Expand Down
4 changes: 3 additions & 1 deletion userspace/falco/configuration.cpp

Large diffs are not rendered by default.

1 change: 1 addition & 0 deletions userspace/falco/configuration.h
Original file line number Diff line number Diff line change
Expand Up @@ -149,6 +149,7 @@ class falco_configuration
bool m_json_output;
bool m_json_include_output_property;
bool m_json_include_tags_property;
bool m_json_include_message_property;
std::string m_log_level;
std::vector<falco::outputs::config> m_outputs;

Expand Down
31 changes: 7 additions & 24 deletions userspace/falco/falco_outputs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -45,12 +45,16 @@ falco_outputs::falco_outputs(
bool json_output,
bool json_include_output_property,
bool json_include_tags_property,
bool json_include_message_property,
uint32_t timeout,
bool buffered,
size_t outputs_queue_capacity,
bool time_format_iso_8601,
const std::string& hostname)
: m_formats(std::make_unique<falco_formats>(engine, json_include_output_property, json_include_tags_property)),
: m_formats(std::make_unique<falco_formats>(
engine,
json_include_output_property, json_include_tags_property, json_include_message_property,
time_format_iso_8601)),
m_buffered(buffered),
m_json_output(json_output),
m_time_format_iso_8601(time_format_iso_8601),
Expand Down Expand Up @@ -136,32 +140,11 @@ void falco_outputs::handle_event(sinsp_evt *evt, const std::string &rule, const
cmsg.source = source;
cmsg.rule = rule;

std::string sformat;
if(m_time_format_iso_8601)
{
sformat = "*%evt.time.iso8601: ";
}
else
{
sformat = "*%evt.time: ";
}
sformat += falco_common::format_priority(priority);

// if format starts with a *, remove it, as we added our own prefix
if(format[0] == '*')
{
sformat += " " + format.substr(1, format.length() - 1);
}
else
{
sformat += " " + format;
}

cmsg.msg = m_formats->format_event(
evt, rule, source, falco_common::format_priority(priority), sformat, tags, m_hostname, extra_fields
evt, rule, source, falco_common::format_priority(priority), format, tags, m_hostname, extra_fields
);

auto fields = m_formats->get_field_values(evt, source, sformat);
auto fields = m_formats->get_field_values(evt, source, format);
for (auto const& ef : extra_fields)
{
// when formatting for the control message we always want strings,
Expand Down
1 change: 1 addition & 0 deletions userspace/falco/falco_outputs.h
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ class falco_outputs
bool json_output,
bool json_include_output_property,
bool json_include_tags_property,
bool json_include_message_property,
uint32_t timeout,
bool buffered,
size_t outputs_queue_capacity,
Expand Down

0 comments on commit 4a33e78

Please sign in to comment.