From 2dfd149c8bb2e8f852da11268964d09f239ea4f5 Mon Sep 17 00:00:00 2001 From: Attila Szakacs Date: Mon, 4 Mar 2024 19:12:42 +0100 Subject: [PATCH] xml: add filterx parser Signed-off-by: Attila Szakacs --- lib/filterx/expr-function.h | 6 +- modules/xml/CMakeLists.txt | 2 + modules/xml/Makefile.am | 4 +- modules/xml/filterx-parse-xml.c | 605 ++++++++++++++++++++++++++++++++ modules/xml/filterx-parse-xml.h | 33 ++ modules/xml/xml-plugin.c | 4 + tests/copyright/policy | 1 + 7 files changed, 651 insertions(+), 4 deletions(-) create mode 100644 modules/xml/filterx-parse-xml.c create mode 100644 modules/xml/filterx-parse-xml.h diff --git a/lib/filterx/expr-function.h b/lib/filterx/expr-function.h index a1a5c40ef6..71b97323ed 100644 --- a/lib/filterx/expr-function.h +++ b/lib/filterx/expr-function.h @@ -146,9 +146,9 @@ FilterXExpr *filterx_generator_function_lookup(GlobalConfig *cfg, const gchar *f .construct = filterx_function_ ## func_name ## _construct, \ } -#define FILTERX_GENERATOR_FUNCTION_PROTOTYPE(func_name) \ - gpointer \ - filterx_function_ ## func_name ## _construct(Plugin *self) +#define FILTERX_GENERATOR_FUNCTION_PROTOTYPE(func_name) \ + gpointer \ + filterx_generator_function_ ## func_name ## _construct(Plugin *self) #define FILTERX_GENERATOR_FUNCTION_DECLARE(func_name) \ FILTERX_GENERATOR_FUNCTION_PROTOTYPE(func_name); diff --git a/modules/xml/CMakeLists.txt b/modules/xml/CMakeLists.txt index a9f048feba..ea4f29b68e 100644 --- a/modules/xml/CMakeLists.txt +++ b/modules/xml/CMakeLists.txt @@ -9,11 +9,13 @@ set(xml_SOURCES "xml.h" "xml-private.h" "windows-eventlog-xml-parser.h" + "filterx-parse-xml.h" "xml-plugin.c" "xml-parser.c" "xml.c" "windows-eventlog-xml-parser.c" + "filterx-parse-xml.c" ) diff --git a/modules/xml/Makefile.am b/modules/xml/Makefile.am index 565b6fbd1c..c4af2ef716 100644 --- a/modules/xml/Makefile.am +++ b/modules/xml/Makefile.am @@ -8,7 +8,9 @@ modules_xml_libxml_la_SOURCES = \ modules/xml/xml.c \ modules/xml/xml-private.h \ modules/xml/windows-eventlog-xml-parser.h \ - modules/xml/windows-eventlog-xml-parser.c + modules/xml/windows-eventlog-xml-parser.c \ + modules/xml/filterx-parse-xml.h \ + modules/xml/filterx-parse-xml.c BUILT_SOURCES += \ diff --git a/modules/xml/filterx-parse-xml.c b/modules/xml/filterx-parse-xml.c new file mode 100644 index 0000000000..dcb77b7e60 --- /dev/null +++ b/modules/xml/filterx-parse-xml.c @@ -0,0 +1,605 @@ +/* + * Copyright (c) 2024 Axoflow + * Copyright (c) 2024 Attila Szakacs + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 as published + * by the Free Software Foundation, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + * + * As an additional exemption you are allowed to compile & link against the + * OpenSSL libraries as published by the OpenSSL project. See the file + * COPYING for details. + * + */ + +#include "filterx-parse-xml.h" +#include "filterx/object-extractor.h" +#include "filterx/object-string.h" +#include "filterx/object-list-interface.h" +#include "filterx/object-dict-interface.h" +#include "filterx/filterx-eval.h" +#include "scratch-buffers.h" + +#include + +#define FILTERX_FUNC_PARSE_XML_USAGE "Usage: parse_xml(raw_xml)" + + +static void _set_error(GError **error, const gchar *format, ...) G_GNUC_PRINTF(2, 0); +static void +_set_error(GError **error, const gchar *format, ...) +{ + if (!error) + return; + + va_list va; + va_start(va, format); + *error = g_error_new_valist(g_quark_from_static_string("filterx-parse-xml"), 0, format, va); + va_end(va); +} + + +typedef struct XmlElemContext_ +{ + FilterXObject *current_obj; + FilterXObject *parent_obj; +} XmlElemContext; + +static void +_elem_context_set_current_obj(XmlElemContext *self, FilterXObject *current_obj) +{ + filterx_object_unref(self->current_obj); + self->current_obj = filterx_object_ref(current_obj); +} + +static void +_elem_context_set_parent_obj(XmlElemContext *self, FilterXObject *parent_obj) +{ + filterx_object_unref(self->parent_obj); + self->parent_obj = filterx_object_ref(parent_obj); +} + +static void +_elem_context_free(XmlElemContext *self) +{ + _elem_context_set_current_obj(self, NULL); + _elem_context_set_parent_obj(self, NULL); + g_free(self); +} + +static XmlElemContext * +_elem_context_new(FilterXObject *parent_obj, FilterXObject *current_obj) +{ + XmlElemContext *self = g_new0(XmlElemContext, 1); + _elem_context_set_parent_obj(self, parent_obj); + _elem_context_set_current_obj(self, current_obj); + return self; +} + + +typedef struct FilterXGeneratorFunctionParseXml_ +{ + FilterXGeneratorFunction super; + FilterXExpr *xml_expr; +} FilterXGeneratorFunctionParseXml; + +static FilterXObject * +_create_object_for_new_elem(FilterXObject *parent_obj, gboolean has_attrs, const gchar **new_elem_repr) +{ + /* + * If the new element has attributes, we create a dict for them, and we will either set "#text" if this is a leaf + * or we will create inner dicts otherwise. + */ + + if (has_attrs) + { + *new_elem_repr = "{}"; + return filterx_object_create_dict(parent_obj); + } + + /* + * If the new element does not have attributes, we want to see an empty string for it, as it might be a leaf. + * However, at this point we cannot know wheter this elem is a leaf for sure, so there is a logic that converts + * this empty string to a dict in _convert_to_inner_node(). + */ + + *new_elem_repr = "\"\""; + return filterx_string_new("", 0); +} + +static void +_store_first_elem(XmlElemContext *new_elem_context, FilterXObject *new_elem_key, + const gchar *new_elem_name, const gchar *new_elem_repr, GError **error) +{ + if (!filterx_object_set_subscript(new_elem_context->parent_obj, new_elem_key, &new_elem_context->current_obj)) + _set_error(error, "failed to store first element: \"%s\"=%s", new_elem_name, new_elem_repr); +} + +static gboolean +_is_obj_storing_a_single_elem(FilterXObject *obj) +{ + return filterx_object_is_type(obj, &FILTERX_TYPE_NAME(string)) || + filterx_object_is_type(obj, &FILTERX_TYPE_NAME(dict)); +} + +static void +_store_second_elem(XmlElemContext *new_elem_context, FilterXObject **existing_obj, FilterXObject *new_elem_key, + const gchar *new_elem_name, const gchar *new_elem_repr, GError **error) +{ + FilterXObject *list_obj = filterx_object_create_list(new_elem_context->parent_obj); + if (!list_obj) + goto fail; + + if (!filterx_list_append(list_obj, existing_obj)) + goto fail; + + if (!filterx_list_append(list_obj, &new_elem_context->current_obj)) + goto fail; + + if (!filterx_object_set_subscript(new_elem_context->parent_obj, new_elem_key, &list_obj)) + goto fail; + + _elem_context_set_parent_obj(new_elem_context, list_obj); + filterx_object_unref(list_obj); + return; + +fail: + _set_error(error, "failed to store second element: \"%s\"=[..., %s]", new_elem_name, new_elem_repr); + filterx_object_unref(list_obj); +} + +static gboolean +_is_obj_storing_multiple_elems(FilterXObject *obj) +{ + return filterx_object_is_type(obj, &FILTERX_TYPE_NAME(list)); +} + +static void +_store_nth_elem(XmlElemContext *new_elem_context, FilterXObject *existing_obj, FilterXObject *new_elem_key, + const gchar *new_elem_name, const gchar *new_elem_repr, GError **error) +{ + if (!filterx_list_append(existing_obj, &new_elem_context->current_obj)) + { + _set_error(error, "failed to store nth element: \"%s\"=[..., %s]", new_elem_name, new_elem_repr); + return; + } + + _elem_context_set_parent_obj(new_elem_context, existing_obj); +} + +static XmlElemContext * +_prepare_elem(const gchar *new_elem_name, XmlElemContext *last_elem_context, gboolean has_attrs, GError **error) +{ + g_assert(filterx_object_is_type(last_elem_context->current_obj, &FILTERX_TYPE_NAME(dict))); + + const gchar *new_elem_repr; + FilterXObject *new_elem_obj = _create_object_for_new_elem(last_elem_context->current_obj, has_attrs, &new_elem_repr); + XmlElemContext *new_elem_context = _elem_context_new(last_elem_context->current_obj, new_elem_obj); + + FilterXObject *new_elem_key = filterx_string_new(new_elem_name, -1); + FilterXObject *existing_obj = NULL; + + if (!filterx_object_is_key_set(new_elem_context->parent_obj, new_elem_key)) + { + _store_first_elem(new_elem_context, new_elem_key, new_elem_name, new_elem_repr, error); + goto exit; + } + + existing_obj = filterx_object_get_subscript(new_elem_context->parent_obj, new_elem_key); + + if (_is_obj_storing_a_single_elem(existing_obj)) + { + _store_second_elem(new_elem_context, &existing_obj, new_elem_key, new_elem_name, new_elem_repr, error); + goto exit; + } + + if (_is_obj_storing_multiple_elems(existing_obj)) + { + _store_nth_elem(new_elem_context, existing_obj, new_elem_key, new_elem_name, new_elem_repr, error); + goto exit; + } + + msg_debug("FilterX: parse_xml(): Unexpected node type, overwriting", evt_tag_str("type", existing_obj->type->name)); + _elem_context_free(new_elem_context); + new_elem_context = _prepare_elem(new_elem_name, last_elem_context, has_attrs, error); + +exit: + filterx_object_unref(new_elem_key); + filterx_object_unref(new_elem_obj); + filterx_object_unref(existing_obj); + + if (*error) + { + _elem_context_free(new_elem_context); + new_elem_context = NULL; + } + + return new_elem_context; +} + +static void +_collect_attrs(const gchar *element_name, XmlElemContext *elem_context, + const gchar **attribute_names, const gchar **attribute_values, + GError **error) +{ + /* Ensured by _prepare_elem() and _create_object_for_new_elem(). */ + g_assert(filterx_object_is_type(elem_context->current_obj, &FILTERX_TYPE_NAME(dict))); + + ScratchBuffersMarker marker; + GString *attr_key = scratch_buffers_alloc_and_mark(&marker); + g_string_assign(attr_key, "@"); + + for (gint i = 0; attribute_names[i]; i++) + { + g_string_truncate(attr_key, 1); + g_string_append(attr_key, attribute_names[i]); + + const gchar *attr_value = attribute_values[i]; + + FilterXObject *key = filterx_string_new(attr_key->str, attr_key->len); + FilterXObject *value = filterx_string_new(attr_value, -1); + + gboolean success = filterx_object_set_subscript(elem_context->current_obj, key, &value); + + filterx_object_unref(key); + filterx_object_unref(value); + + if (!success) + { + _set_error(error, "failed to store attribute: \"%s\"=\"%s\"", attribute_names[i], attr_value); + break; + } + } + + scratch_buffers_reclaim_marked(marker); +} + +static gboolean +_convert_to_dict(GMarkupParseContext *context, XmlElemContext *elem_context, GError **error) +{ + const gchar *parent_elem_name = (const gchar *) g_markup_parse_context_get_element_stack(context)->next->data; + FilterXObject *key = filterx_string_new(parent_elem_name, -1); + + gsize existing_value_len; + const gchar *existing_value; + filterx_object_extract_string(elem_context->current_obj, &existing_value, &existing_value_len); + + if (G_UNLIKELY(debug_flag) && !existing_value) + { + GString *repr = scratch_buffers_alloc(); + filterx_object_repr(elem_context->current_obj, repr); + msg_debug("FilterX: parse_xml(): unexpected node, overwriting", + evt_tag_str("type", elem_context->current_obj->type->name), + evt_tag_str("value", repr->str)); + } + + FilterXObject *dict_obj = filterx_object_create_dict(elem_context->parent_obj); + if (!dict_obj) + goto exit; + + if (existing_value && existing_value_len > 0) + { + FilterXObject *existing_value_key = filterx_string_new("#text", 5); + gboolean success = filterx_object_set_subscript(dict_obj, existing_value_key, &elem_context->current_obj); + filterx_object_unref(existing_value_key); + + if (!success) + { + _set_error(error, "failed to store leaf node value in a new dict: \"%s\"={\"#text\": \"%s\"}", + parent_elem_name, existing_value); + goto exit; + } + } + + if (filterx_object_is_type(elem_context->parent_obj, &FILTERX_TYPE_NAME(dict))) + { + if (!filterx_object_set_subscript(elem_context->parent_obj, key, &dict_obj)) + _set_error(error, "failed to replace leaf node object with: \"%s\"={}", parent_elem_name); + goto exit; + } + + if (filterx_object_is_type(elem_context->parent_obj, &FILTERX_TYPE_NAME(list))) + { + if (!filterx_list_set_subscript(elem_context->parent_obj, -1, &dict_obj)) + _set_error(error, "failed to replace leaf node object with: {}"); + goto exit; + } + + g_assert_not_reached(); + +exit: + gboolean success = !(*error); + if (success) + _elem_context_set_current_obj(elem_context, dict_obj); + + filterx_object_unref(key); + filterx_object_unref(dict_obj); + return success; +} + +static void +_start_elem_cb(GMarkupParseContext *context, const gchar *element_name, + const gchar **attribute_names, const gchar **attribute_values, + gpointer user_data, GError **error) +{ + GQueue *obj_stack = (GQueue *) user_data; + XmlElemContext *last_elem_context = g_queue_peek_head(obj_stack); + + if (!filterx_object_is_type(last_elem_context->current_obj, &FILTERX_TYPE_NAME(dict))) + { + /* + * We need the last node to be a dict, so we can start a new inner element in it. + * It can already be a dict, if we already stored an inner element in it, + * or if it had attributes. + */ + if (!_convert_to_dict(context, last_elem_context, error)) + return; + } + + gboolean has_attrs = !!attribute_names[0]; + XmlElemContext *new_elem_context = _prepare_elem(element_name, last_elem_context, has_attrs, error); + if (!new_elem_context) + return; + + g_queue_push_head(obj_stack, new_elem_context); + + if (has_attrs) + _collect_attrs(element_name, new_elem_context, attribute_names, attribute_values, error); +} + +void +_end_elem_cb(GMarkupParseContext *context, const gchar *element_name, gpointer user_data, GError **error) +{ + GQueue *obj_stack = (GQueue *) user_data; + XmlElemContext *elem_context = g_queue_pop_head(obj_stack); + _elem_context_free(elem_context); +} + +static FilterXObject * +_strip_and_create_object(const gchar *text, gsize text_len) +{ + gchar *stripped_text = g_strndup(text, text_len); + g_strstrip(stripped_text); + + gsize stripped_text_len = strlen(stripped_text); + if (!stripped_text_len) + { + g_free(stripped_text); + return NULL; + } + + FilterXObject *result = filterx_string_new(stripped_text, stripped_text_len); + g_free(stripped_text); + return result; +} + +static void +_replace_text(FilterXObject *parent_obj, const gchar *element_name, FilterXObject **new_text_obj, const gchar *raw_text, + GError **error) +{ + if (filterx_object_is_type(parent_obj, &FILTERX_TYPE_NAME(dict))) + { + FilterXObject *key = filterx_string_new(element_name, -1); + gboolean result = filterx_object_set_subscript(parent_obj, key, new_text_obj); + filterx_object_unref(key); + + if (!result) + _set_error(error, "failed to add text to dict: \"%s\"=\"%s\"", element_name, raw_text); + return; + } + + if (filterx_object_is_type(parent_obj, &FILTERX_TYPE_NAME(list))) + { + if (!filterx_list_set_subscript(parent_obj, -1, new_text_obj)) + _set_error(error, "failed to add text to list: \"%s\"", raw_text); + return; + } + + g_assert_not_reached(); +} + +static gboolean +_add_text(FilterXObject *dict_obj, FilterXObject **text_obj, const gchar *text, GError **error) +{ + FilterXObject *key = filterx_string_new("#text", 5); + + if (!filterx_object_set_subscript(dict_obj, key, text_obj)) + _set_error(error, "failed to add text to dict: \"#text\"=\"%s\"", text); + + filterx_object_unref(key); + return !(*error); +} + +static void +_text_cb(GMarkupParseContext *context, const gchar *text, gsize text_len, gpointer user_data, GError **error) +{ + GQueue *obj_stack = (GQueue *) user_data; + XmlElemContext *elem_context = g_queue_peek_head(obj_stack); + const gchar *element_name = g_markup_parse_context_get_element(context); + + FilterXObject *text_obj = _strip_and_create_object(text, text_len); + if (!text_obj) + { + /* + * We already stored an empty value in _prepare_elem() and _create_object_for_new_elem(). + * There's nothing to do here. + */ + return; + } + + if (filterx_object_is_type(elem_context->current_obj, &FILTERX_TYPE_NAME(string))) + { + _replace_text(elem_context->parent_obj, element_name, &text_obj, text, error); + goto exit; + } + + if (filterx_object_is_type(elem_context->current_obj, &FILTERX_TYPE_NAME(dict))) + { + if (_add_text(elem_context->current_obj, &text_obj, text, error)) + _elem_context_set_parent_obj(elem_context, elem_context->current_obj); + goto exit; + } + + g_assert_not_reached(); + +exit: + if (!(*error)) + _elem_context_set_current_obj(elem_context, text_obj); + filterx_object_unref(text_obj); +} + +static gboolean +_validate_fillable(FilterXGeneratorFunctionParseXml *self, FilterXObject *fillable) +{ + if (!filterx_object_is_type(fillable, &FILTERX_TYPE_NAME(dict))) + { + filterx_eval_push_error_info("fillable must be dict", &self->super.super.super, + g_strdup_printf("got %s instead", fillable->type->name), TRUE); + return FALSE; + } + return TRUE; +} + +static const gchar * +_extract_raw_xml(FilterXGeneratorFunctionParseXml *self, FilterXObject *xml_obj, gsize *len) +{ + const gchar *raw_xml; + if (!filterx_object_extract_string(xml_obj, &raw_xml, len)) + { + filterx_eval_push_error_info("input must be string", &self->super.super.super, + g_strdup_printf("got %s instead", xml_obj->type->name), TRUE); + filterx_object_unref(xml_obj); + return NULL; + } + + return raw_xml; +} + +static gboolean +_parse(FilterXGeneratorFunctionParseXml *self, const gchar *raw_xml, gsize raw_xml_len, FilterXObject *fillable) +{ + static GMarkupParser scanner_callbacks = + { + .start_element = _start_elem_cb, + .end_element = _end_elem_cb, + .text = _text_cb, + }; + + GQueue *obj_stack = g_queue_new(); + XmlElemContext *root_elem_context = _elem_context_new(NULL, fillable); + g_queue_push_head(obj_stack, root_elem_context); + GMarkupParseContext *context = g_markup_parse_context_new(&scanner_callbacks, 0, obj_stack, NULL); + + GError *error = NULL; + gboolean success = g_markup_parse_context_parse(context, raw_xml, raw_xml_len, &error) && + g_markup_parse_context_end_parse(context, &error); + if (!success) + { + gchar *error_info = g_strdup(error ? error->message : "unknown error"); + filterx_eval_push_error_info("failed to parse xml", &self->super.super.super, error_info, TRUE); + if (error) + g_error_free(error); + goto exit; + } + +exit: + _elem_context_free(root_elem_context); + g_queue_free(obj_stack); + g_markup_parse_context_free(context); + return success; +} + +static gboolean +_generate(FilterXExprGenerator *s, FilterXObject *fillable) +{ + FilterXGeneratorFunctionParseXml *self = (FilterXGeneratorFunctionParseXml *) s; + + if (!_validate_fillable(self, fillable)) + return FALSE; + + FilterXObject *xml_obj = filterx_expr_eval(self->xml_expr); + if (!xml_obj) + return FALSE; + + gboolean success = FALSE; + + gsize raw_xml_len; + const gchar *raw_xml = _extract_raw_xml(self, xml_obj, &raw_xml_len); + if (!raw_xml) + goto exit; + + success = _parse(self, raw_xml, raw_xml_len, fillable); + +exit: + filterx_object_unref(xml_obj); + return success; +} + +static FilterXObject * +_create_container(FilterXExprGenerator *s, FilterXExpr *fillable_parent) +{ + FilterXObject *fillable_parent_obj = filterx_expr_eval_typed(fillable_parent); + if (!fillable_parent_obj) + return NULL; + + FilterXObject *result = filterx_object_create_dict(fillable_parent_obj); + filterx_object_unref(fillable_parent_obj); + return result; +} + +static gboolean +_extract_args(FilterXGeneratorFunctionParseXml *self, FilterXFunctionArgs *args, GError **error) +{ + if (filterx_function_args_len(args) != 1) + { + g_set_error(error, FILTERX_FUNCTION_ERROR, FILTERX_FUNCTION_ERROR_CTOR_FAIL, + "invalid number of arguments. " FILTERX_FUNC_PARSE_XML_USAGE); + return FALSE; + } + + self->xml_expr = filterx_function_args_get_expr(args, 0); + return TRUE; +} + +static void +_free(FilterXExpr *s) +{ + FilterXGeneratorFunctionParseXml *self = (FilterXGeneratorFunctionParseXml *) s; + + filterx_expr_unref(self->xml_expr); + filterx_generator_function_free_method(&self->super); +} + +FilterXExpr * +filterx_generator_function_parse_xml_new(const gchar *func_name, FilterXFunctionArgs *args, GError **error) +{ + FilterXGeneratorFunctionParseXml *self = g_new0(FilterXGeneratorFunctionParseXml, 1); + + filterx_generator_function_init_instance(&self->super, func_name); + self->super.super.generate = _generate; + self->super.super.create_container = _create_container; + self->super.super.super.free_fn = _free; + + if (!_extract_args(self, args, error) || + !filterx_function_args_check(args, error)) + goto fail; + + filterx_function_args_free(args); + return &self->super.super.super; + +fail: + filterx_function_args_free(args); + filterx_expr_unref(&self->super.super.super); + return NULL; +} + +FILTERX_GENERATOR_FUNCTION(parse_xml, filterx_generator_function_parse_xml_new); diff --git a/modules/xml/filterx-parse-xml.h b/modules/xml/filterx-parse-xml.h new file mode 100644 index 0000000000..8a9574b0fa --- /dev/null +++ b/modules/xml/filterx-parse-xml.h @@ -0,0 +1,33 @@ +/* + * Copyright (c) 2024 Attila Szakacs + * + * This program is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 as published + * by the Free Software Foundation, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + * + * As an additional exemption you are allowed to compile & link against the + * OpenSSL libraries as published by the OpenSSL project. See the file + * COPYING for details. + * + */ + +#ifndef FILTERX_PARSE_XML_H_INCLUDED +#define FILTERX_PARSE_XML_H_INCLUDED + +#include "filterx/expr-function.h" + +FILTERX_GENERATOR_FUNCTION_DECLARE(parse_xml); + +FilterXExpr *filterx_generator_function_parse_xml_new(const gchar *func_name, FilterXFunctionArgs *args, + GError **error); + +#endif diff --git a/modules/xml/xml-plugin.c b/modules/xml/xml-plugin.c index 6d38bcf1ac..f60ab1cdfd 100644 --- a/modules/xml/xml-plugin.c +++ b/modules/xml/xml-plugin.c @@ -20,6 +20,9 @@ * */ +#include "filterx-parse-xml.h" +#include "filterx/expr-function.h" + #include "cfg-parser.h" #include "plugin.h" #include "plugin-types.h" @@ -38,6 +41,7 @@ static Plugin xml_plugins[] = .name = "windows-eventlog-xml-parser", .parser = &xml_parser, }, + FILTERX_GENERATOR_FUNCTION_PLUGIN(parse_xml), }; gboolean diff --git a/tests/copyright/policy b/tests/copyright/policy index 13babd6f75..1829275a4c 100644 --- a/tests/copyright/policy +++ b/tests/copyright/policy @@ -273,6 +273,7 @@ tests/light/src/syslog_ng_config/statements/__init__\.py modules/correlation/id-counter\.[ch]$ modules/correlation/group-lines.h modules/xml/windows-eventlog-xml-parser\.h +modules/xml/filterx-parse-xml\.[ch]$ modules/xml/tests/test_windows_eventlog_xml_parser\.c modules/examples/filterx/example-filterx-func/example-filterx-func-plugin\.[ch] modules/grpc/otel/filterx