summaryrefslogtreecommitdiffstats
path: root/fluent-bit/plugins/in_tail
diff options
context:
space:
mode:
Diffstat (limited to 'fluent-bit/plugins/in_tail')
-rw-r--r--fluent-bit/plugins/in_tail/CMakeLists.txt37
-rw-r--r--fluent-bit/plugins/in_tail/tail.c783
-rw-r--r--fluent-bit/plugins/in_tail/tail.h45
-rw-r--r--fluent-bit/plugins/in_tail/tail_config.c472
-rw-r--r--fluent-bit/plugins/in_tail/tail_config.h168
-rw-r--r--fluent-bit/plugins/in_tail/tail_db.c277
-rw-r--r--fluent-bit/plugins/in_tail/tail_db.h43
-rw-r--r--fluent-bit/plugins/in_tail/tail_dockermode.c459
-rw-r--r--fluent-bit/plugins/in_tail/tail_dockermode.h38
-rw-r--r--fluent-bit/plugins/in_tail/tail_file.c1860
-rw-r--r--fluent-bit/plugins/in_tail/tail_file.h137
-rw-r--r--fluent-bit/plugins/in_tail/tail_file_internal.h130
-rw-r--r--fluent-bit/plugins/in_tail/tail_fs.h96
-rw-r--r--fluent-bit/plugins/in_tail/tail_fs_inotify.c433
-rw-r--r--fluent-bit/plugins/in_tail/tail_fs_inotify.h37
-rw-r--r--fluent-bit/plugins/in_tail/tail_fs_stat.c253
-rw-r--r--fluent-bit/plugins/in_tail/tail_fs_stat.h37
-rw-r--r--fluent-bit/plugins/in_tail/tail_multiline.c606
-rw-r--r--fluent-bit/plugins/in_tail/tail_multiline.h57
-rw-r--r--fluent-bit/plugins/in_tail/tail_scan.c71
-rw-r--r--fluent-bit/plugins/in_tail/tail_scan.h29
-rw-r--r--fluent-bit/plugins/in_tail/tail_scan_glob.c278
-rw-r--r--fluent-bit/plugins/in_tail/tail_scan_win32.c245
-rw-r--r--fluent-bit/plugins/in_tail/tail_signal.h98
-rw-r--r--fluent-bit/plugins/in_tail/tail_sql.h65
-rw-r--r--fluent-bit/plugins/in_tail/win32.h67
-rw-r--r--fluent-bit/plugins/in_tail/win32/interface.h44
-rw-r--r--fluent-bit/plugins/in_tail/win32/io.c47
-rw-r--r--fluent-bit/plugins/in_tail/win32/stat.c332
29 files changed, 0 insertions, 7244 deletions
diff --git a/fluent-bit/plugins/in_tail/CMakeLists.txt b/fluent-bit/plugins/in_tail/CMakeLists.txt
deleted file mode 100644
index 31d865218..000000000
--- a/fluent-bit/plugins/in_tail/CMakeLists.txt
+++ /dev/null
@@ -1,37 +0,0 @@
-set(src
- tail_file.c
- tail_dockermode.c
- tail_scan.c
- tail_config.c
- tail_fs_stat.c
- tail.c)
-
-if(FLB_HAVE_INOTIFY)
-set(src
- ${src}
- tail_fs_inotify.c)
-endif()
-
-if(FLB_SQLDB)
-set(src
- ${src}
- tail_db.c)
-endif()
-
-if(FLB_PARSER)
- set(src
- ${src}
- tail_multiline.c
- )
-endif()
-
-if(MSVC)
- set(src
- ${src}
- win32/stat.c
- win32/io.c
- )
- FLB_PLUGIN(in_tail "${src}" "Shlwapi")
-else()
- FLB_PLUGIN(in_tail "${src}" "")
-endif()
diff --git a/fluent-bit/plugins/in_tail/tail.c b/fluent-bit/plugins/in_tail/tail.c
deleted file mode 100644
index 34a0fec3d..000000000
--- a/fluent-bit/plugins/in_tail/tail.c
+++ /dev/null
@@ -1,783 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_compat.h>
-#include <fluent-bit/flb_input.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_config.h>
-#include <fluent-bit/flb_config_map.h>
-#include <fluent-bit/flb_error.h>
-#include <fluent-bit/flb_utils.h>
-
-#include "tail.h"
-#include "tail_fs.h"
-#include "tail_db.h"
-#include "tail_file.h"
-#include "tail_scan.h"
-#include "tail_signal.h"
-#include "tail_config.h"
-#include "tail_dockermode.h"
-#include "tail_multiline.h"
-
-static inline int consume_byte(flb_pipefd_t fd)
-{
- int ret;
- uint64_t val;
-
- /* We need to consume the byte */
- ret = flb_pipe_r(fd, (char *) &val, sizeof(val));
- if (ret <= 0) {
- flb_errno();
- return -1;
- }
-
- return 0;
-}
-
-/* cb_collect callback */
-static int in_tail_collect_pending(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- int ret;
- int active = 0;
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_config *ctx = in_context;
- struct flb_tail_file *file;
- struct stat st;
- uint64_t pre;
- uint64_t total_processed = 0;
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
-
- if (file->watch_fd == -1 ||
- (file->offset >= file->size)) {
- ret = fstat(file->fd, &st);
-
- if (ret == -1) {
- flb_errno();
- flb_tail_file_remove(file);
- continue;
- }
-
- file->size = st.st_size;
- file->pending_bytes = (file->size - file->offset);
- }
- else {
- memset(&st, 0, sizeof(struct stat));
- }
-
- if (file->pending_bytes <= 0) {
- file->pending_bytes = (file->size - file->offset);
- }
-
- if (file->pending_bytes <= 0) {
- continue;
- }
-
- if (ctx->event_batch_size > 0 &&
- total_processed >= ctx->event_batch_size) {
- break;
- }
-
- /* get initial offset to calculate the number of processed bytes later */
- pre = file->offset;
-
- ret = flb_tail_file_chunk(file);
-
- /* Update the total number of bytes processed */
- if (file->offset > pre) {
- total_processed += (file->offset - pre);
- }
-
- switch (ret) {
- case FLB_TAIL_ERROR:
- /* Could not longer read the file */
- flb_tail_file_remove(file);
- break;
- case FLB_TAIL_OK:
- case FLB_TAIL_BUSY:
- /*
- * Adjust counter to verify if we need a further read(2) later.
- * For more details refer to tail_fs_inotify.c:96.
- */
- if (file->offset < file->size) {
- file->pending_bytes = (file->size - file->offset);
- active++;
- }
- else {
- file->pending_bytes = 0;
- }
- break;
- }
- }
-
- /* If no more active files, consume pending signal so we don't get called again. */
- if (active == 0) {
- tail_consume_pending(ctx);
- }
-
- return 0;
-}
-
-/* cb_collect callback */
-static int in_tail_collect_static(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- int ret;
- int active = 0;
- int pre_size;
- int pos_size;
- int alter_size = 0;
- int completed = FLB_FALSE;
- char s_size[32];
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_config *ctx = in_context;
- struct flb_tail_file *file;
- uint64_t pre;
- uint64_t total_processed = 0;
-
- /* Do a data chunk collection for each file */
- mk_list_foreach_safe(head, tmp, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
-
- /*
- * The list 'files_static' represents all the files that were discovered
- * on startup that already contains data: these are called 'static files'.
- *
- * When processing static files, we don't know what kind of content they
- * have and what kind of 'latency' might add to process all of them in
- * a row. Despite we always 'try' to do a full round and process a
- * fraction of them on every invocation of this function if we have a
- * huge number of files we will face latency and make the main pipeline
- * to degrade performance.
- *
- * In order to avoid this situation, we added a new option to the plugin
- * called 'static_batch_size' which basically defines how many bytes can
- * be processed on every invocation to process the static files.
- *
- * When the limit is reached, we just break the loop and as a side effect
- * we allow other events keep processing.
- */
- if (ctx->static_batch_size > 0 &&
- total_processed >= ctx->static_batch_size) {
- break;
- }
-
- /* get initial offset to calculate the number of processed bytes later */
- pre = file->offset;
-
- /* Process the file */
- ret = flb_tail_file_chunk(file);
-
- /* Update the total number of bytes processed */
- if (file->offset > pre) {
- total_processed += (file->offset - pre);
- }
-
- switch (ret) {
- case FLB_TAIL_ERROR:
- /* Could not longer read the file */
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" collect static ERROR",
- file->inode);
- flb_tail_file_remove(file);
- break;
- case FLB_TAIL_OK:
- case FLB_TAIL_BUSY:
- active++;
- break;
- case FLB_TAIL_WAIT:
- if (file->config->exit_on_eof) {
- flb_plg_info(ctx->ins, "inode=%"PRIu64" file=%s ended, stop",
- file->inode, file->name);
- if (ctx->files_static_count == 1) {
- flb_engine_exit(config);
- }
- }
- /* Promote file to 'events' type handler */
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" file=%s promote to TAIL_EVENT",
- file->inode, file->name);
-
- /*
- * When promoting a file from 'static' to 'event' mode, the promoter
- * will check if the file has been rotated while it was being
- * processed on this function, if so, it will try to check for the
- * following condition:
- *
- * "discover a new possible file created due to rotation"
- *
- * If the condition above is met, a new file entry will be added to
- * the list that we are processing and a 'new signal' will be send
- * to the signal manager. But the signal manager will trigger the
- * message only if no pending messages exists (to avoid queue size
- * exhaustion).
- *
- * All good, but there is a corner case where if no 'active' files
- * exists, the signal will be read and this function will not be
- * called again and since the signal did not triggered the
- * message, the 'new file' enqueued by the nested function
- * might stay in stale mode (note that altering the length of this
- * list will not be reflected yet)
- *
- * To fix the corner case, we use a variable called 'alter_size'
- * that determinate if the size of the list keeps the same after
- * a rotation, so it means: a new file was added.
- *
- * We use 'alter_size' as a helper in the conditional below to know
- * when to stop processing the static list.
- */
- if (alter_size == 0) {
- pre_size = ctx->files_static_count;
- }
- ret = flb_tail_file_to_event(file);
- if (ret == -1) {
- flb_plg_debug(ctx->ins, "file=%s cannot promote, unregistering",
- file->name);
- flb_tail_file_remove(file);
- }
-
- if (alter_size == 0) {
- pos_size = ctx->files_static_count;
- if (pre_size == pos_size) {
- alter_size++;
- }
- }
- break;
- }
- }
-
- /*
- * If there are no more active static handlers, we consume the 'byte' that
- * triggered this event so this is not longer called again.
- */
- if (active == 0 && alter_size == 0) {
- consume_byte(ctx->ch_manager[0]);
- ctx->ch_reads++;
- completed = FLB_TRUE;
- }
-
- /* Debugging number of processed bytes */
- if (flb_log_check_level(ctx->ins->log_level, FLB_LOG_DEBUG)) {
- flb_utils_bytes_to_human_readable_size(total_processed,
- s_size, sizeof(s_size));
- if (completed) {
- flb_plg_debug(ctx->ins, "[static files] processed %s, done", s_size);
- }
- else {
- flb_plg_debug(ctx->ins, "[static files] processed %s", s_size);
- }
- }
-
- return 0;
-}
-
-static int in_tail_watcher_callback(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- int ret = 0;
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_config *ctx = context;
- struct flb_tail_file *file;
- (void) config;
-
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- if (file->is_link == FLB_TRUE) {
- ret = flb_tail_file_is_rotated(ctx, file);
- if (ret == FLB_FALSE) {
- continue;
- }
-
- /* The symbolic link name has been rotated */
- flb_tail_file_rotated(file);
- }
- }
- return ret;
-}
-
-int in_tail_collect_event(void *file, struct flb_config *config)
-{
- int ret;
- struct stat st;
- struct flb_tail_file *f = file;
-
- ret = fstat(f->fd, &st);
- if (ret == -1) {
- flb_tail_file_remove(f);
- return 0;
- }
-
- ret = flb_tail_file_chunk(f);
- switch (ret) {
- case FLB_TAIL_ERROR:
- /* Could not longer read the file */
- flb_tail_file_remove(f);
- break;
- case FLB_TAIL_OK:
- case FLB_TAIL_WAIT:
- break;
- }
-
- return 0;
-}
-
-/* Initialize plugin */
-static int in_tail_init(struct flb_input_instance *in,
- struct flb_config *config, void *data)
-{
- int ret = -1;
- struct flb_tail_config *ctx = NULL;
-
- /* Allocate space for the configuration */
- ctx = flb_tail_config_create(in, config);
- if (!ctx) {
- return -1;
- }
- ctx->ins = in;
-
- /* Initialize file-system watcher */
- ret = flb_tail_fs_init(in, ctx, config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
-
- /* Scan path */
- flb_tail_scan(ctx->path_list, ctx);
-
- /*
- * After the first scan (on start time), all new files discovered needs to be
- * read from head, so we switch the 'read_from_head' flag to true so any
- * other file discovered after a scan or a rotation are read from the
- * beginning.
- */
- ctx->read_from_head = FLB_TRUE;
-
- /* Set plugin context */
- flb_input_set_context(in, ctx);
-
- /* Register an event collector */
- ret = flb_input_set_collector_event(in, in_tail_collect_static,
- ctx->ch_manager[0], config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_static = ret;
-
- /* Register re-scan: time managed by 'refresh_interval' property */
- ret = flb_input_set_collector_time(in, flb_tail_scan_callback,
- ctx->refresh_interval_sec,
- ctx->refresh_interval_nsec,
- config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_scan = ret;
-
- /* Register watcher, interval managed by 'watcher_interval' property */
- ret = flb_input_set_collector_time(in, in_tail_watcher_callback,
- ctx->watcher_interval, 0,
- config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_watcher = ret;
-
- /* Register callback to purge rotated files */
- ret = flb_input_set_collector_time(in, flb_tail_file_purge,
- ctx->rotate_wait, 0,
- config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_rotated = ret;
-
- /* Register callback to process pending bytes in promoted files */
- ret = flb_input_set_collector_event(in, in_tail_collect_pending,
- ctx->ch_pending[0], config);//1, 0, config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_pending = ret;
-
-
- if (ctx->multiline == FLB_TRUE && ctx->parser) {
- ctx->parser = NULL;
- flb_plg_warn(in, "on multiline mode 'Parser' is not allowed "
- "(parser disabled)");
- }
-
- /* Register callback to process docker mode queued buffer */
- if (ctx->docker_mode == FLB_TRUE) {
- ret = flb_input_set_collector_time(in, flb_tail_dmode_pending_flush,
- ctx->docker_mode_flush, 0,
- config);
- if (ret == -1) {
- ctx->docker_mode = FLB_FALSE;
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_dmode_flush = ret;
- }
-
-#ifdef FLB_HAVE_PARSER
- /* Register callback to process multiline queued buffer */
- if (ctx->multiline == FLB_TRUE) {
- ret = flb_input_set_collector_time(in, flb_tail_mult_pending_flush,
- ctx->multiline_flush, 0,
- config);
- if (ret == -1) {
- ctx->multiline = FLB_FALSE;
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_mult_flush = ret;
- }
-#endif
-
- return 0;
-}
-
-/* Pre-run callback / before the event loop */
-static int in_tail_pre_run(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- struct flb_tail_config *ctx = in_context;
- (void) ins;
-
- return tail_signal_manager(ctx);
-}
-
-static int in_tail_exit(void *data, struct flb_config *config)
-{
- (void) *config;
- struct flb_tail_config *ctx = data;
-
- flb_tail_file_remove_all(ctx);
- flb_tail_fs_exit(ctx);
- flb_tail_config_destroy(ctx);
-
- return 0;
-}
-
-static void in_tail_pause(void *data, struct flb_config *config)
-{
- struct flb_tail_config *ctx = data;
-
- /*
- * Pause general collectors:
- *
- * - static : static files lookup before promotion
- */
- flb_input_collector_pause(ctx->coll_fd_static, ctx->ins);
- flb_input_collector_pause(ctx->coll_fd_pending, ctx->ins);
-
- if (ctx->docker_mode == FLB_TRUE) {
- flb_input_collector_pause(ctx->coll_fd_dmode_flush, ctx->ins);
- if (config->is_ingestion_active == FLB_FALSE) {
- flb_plg_info(ctx->ins, "flushing pending docker mode data...");
- flb_tail_dmode_pending_flush_all(ctx);
- }
- }
-
- if (ctx->multiline == FLB_TRUE) {
- flb_input_collector_pause(ctx->coll_fd_mult_flush, ctx->ins);
- if (config->is_ingestion_active == FLB_FALSE) {
- flb_plg_info(ctx->ins, "flushing pending multiline data...");
- flb_tail_mult_pending_flush_all(ctx);
- }
- }
-
- /* Pause file system backend handlers */
- flb_tail_fs_pause(ctx);
-}
-
-static void in_tail_resume(void *data, struct flb_config *config)
-{
- struct flb_tail_config *ctx = data;
-
- flb_input_collector_resume(ctx->coll_fd_static, ctx->ins);
- flb_input_collector_resume(ctx->coll_fd_pending, ctx->ins);
-
- if (ctx->docker_mode == FLB_TRUE) {
- flb_input_collector_resume(ctx->coll_fd_dmode_flush, ctx->ins);
- }
-
- if (ctx->multiline == FLB_TRUE) {
- flb_input_collector_resume(ctx->coll_fd_mult_flush, ctx->ins);
- }
-
- /* Pause file system backend handlers */
- flb_tail_fs_resume(ctx);
-}
-
-/* Configuration properties map */
-static struct flb_config_map config_map[] = {
- {
- FLB_CONFIG_MAP_CLIST, "path", NULL,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, path_list),
- "pattern specifying log files or multiple ones through "
- "the use of common wildcards."
- },
- {
- FLB_CONFIG_MAP_CLIST, "exclude_path", NULL,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, exclude_list),
- "Set one or multiple shell patterns separated by commas to exclude "
- "files matching a certain criteria, e.g: 'exclude_path *.gz,*.zip'"
- },
- {
- FLB_CONFIG_MAP_STR, "key", "log",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, key),
- "when a message is unstructured (no parser applied), it's appended "
- "as a string under the key name log. This option allows to define an "
- "alternative name for that key."
- },
- {
- FLB_CONFIG_MAP_BOOL, "read_from_head", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, read_from_head),
- "For new discovered files on start (without a database offset/position), read the "
- "content from the head of the file, not tail."
- },
- {
- FLB_CONFIG_MAP_STR, "refresh_interval", "60",
- 0, FLB_FALSE, 0,
- "interval to refresh the list of watched files expressed in seconds."
- },
- {
- FLB_CONFIG_MAP_TIME, "watcher_interval", "2s",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, watcher_interval),
- },
- {
- FLB_CONFIG_MAP_TIME, "progress_check_interval", "2s",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, progress_check_interval),
- },
- {
- FLB_CONFIG_MAP_INT, "progress_check_interval_nsec", "0",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, progress_check_interval_nsec),
- },
- {
- FLB_CONFIG_MAP_TIME, "rotate_wait", FLB_TAIL_ROTATE_WAIT,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, rotate_wait),
- "specify the number of extra time in seconds to monitor a file once is "
- "rotated in case some pending data is flushed."
- },
- {
- FLB_CONFIG_MAP_BOOL, "docker_mode", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, docker_mode),
- "If enabled, the plugin will recombine split Docker log lines before "
- "passing them to any parser as configured above. This mode cannot be "
- "used at the same time as Multiline."
- },
- {
- FLB_CONFIG_MAP_INT, "docker_mode_flush", "4",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, docker_mode_flush),
- "wait period time in seconds to flush queued unfinished split lines."
-
- },
-#ifdef FLB_HAVE_REGEX
- {
- FLB_CONFIG_MAP_STR, "docker_mode_parser", NULL,
- 0, FLB_FALSE, 0,
- "specify the parser name to fetch log first line for muliline log"
- },
-#endif
- {
- FLB_CONFIG_MAP_STR, "path_key", NULL,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, path_key),
- "set the 'key' name where the name of monitored file will be appended."
- },
- {
- FLB_CONFIG_MAP_STR, "offset_key", NULL,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, offset_key),
- "set the 'key' name where the offset of monitored file will be appended."
- },
- {
- FLB_CONFIG_MAP_TIME, "ignore_older", "0",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, ignore_older),
- "ignore files older than 'ignore_older'. Supports m,h,d (minutes, "
- "hours, days) syntax. Default behavior is to read all the files."
- },
- {
- FLB_CONFIG_MAP_SIZE, "buffer_chunk_size", FLB_TAIL_CHUNK,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, buf_chunk_size),
- "set the initial buffer size to read data from files. This value is "
- "used too to increase buffer size."
- },
- {
- FLB_CONFIG_MAP_SIZE, "buffer_max_size", FLB_TAIL_CHUNK,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, buf_max_size),
- "set the limit of the buffer size per monitored file. When a buffer "
- "needs to be increased (e.g: very long lines), this value is used to "
- "restrict how much the memory buffer can grow. If reading a file exceed "
- "this limit, the file is removed from the monitored file list."
- },
- {
- FLB_CONFIG_MAP_SIZE, "static_batch_size", FLB_TAIL_STATIC_BATCH_SIZE,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, static_batch_size),
- "On start, Fluent Bit might process files which already contains data, "
- "these files are called 'static' files. The configuration property "
- "in question set's the maximum number of bytes to process per iteration "
- "for the static files monitored."
- },
- {
- FLB_CONFIG_MAP_SIZE, "event_batch_size", FLB_TAIL_EVENT_BATCH_SIZE,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, event_batch_size),
- "When Fluent Bit is processing files in event based mode the amount of"
- "data available for consumption could be too much and cause the input plugin "
- "to over extend and smother other plugins"
- "The configuration property sets the maximum number of bytes to process per iteration "
- "for the files monitored (in event mode)."
- },
- {
- FLB_CONFIG_MAP_BOOL, "skip_long_lines", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, skip_long_lines),
- "if a monitored file reach it buffer capacity due to a very long line "
- "(buffer_max_size), the default behavior is to stop monitoring that "
- "file. This option alter that behavior and instruct Fluent Bit to skip "
- "long lines and continue processing other lines that fits into the buffer."
- },
- {
- FLB_CONFIG_MAP_BOOL, "exit_on_eof", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, exit_on_eof),
- "exit Fluent Bit when reaching EOF on a monitored file."
- },
-
- {
- FLB_CONFIG_MAP_BOOL, "skip_empty_lines", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, skip_empty_lines),
- "Allows to skip empty lines."
- },
-
-#ifdef FLB_HAVE_INOTIFY
- {
- FLB_CONFIG_MAP_BOOL, "inotify_watcher", "true",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, inotify_watcher),
- "set to false to use file stat watcher instead of inotify."
- },
-#endif
-#ifdef FLB_HAVE_REGEX
- {
- FLB_CONFIG_MAP_STR, "parser", NULL,
- 0, FLB_FALSE, 0,
- "specify the parser name to process an unstructured message."
- },
- {
- FLB_CONFIG_MAP_STR, "tag_regex", NULL,
- 0, FLB_FALSE, 0,
- "set a regex to extract fields from the file name and use them later to "
- "compose the Tag."
- },
-#endif
-
-#ifdef FLB_HAVE_SQLDB
- {
- FLB_CONFIG_MAP_STR, "db", NULL,
- 0, FLB_FALSE, 0,
- "set a database file to keep track of monitored files and it offsets."
- },
- {
- FLB_CONFIG_MAP_STR, "db.sync", "normal",
- 0, FLB_FALSE, 0,
- "set a database sync method. values: extra, full, normal and off."
- },
- {
- FLB_CONFIG_MAP_BOOL, "db.locking", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, db_locking),
- "set exclusive locking mode, increase performance but don't allow "
- "external connections to the database file."
- },
- {
- FLB_CONFIG_MAP_STR, "db.journal_mode", "WAL",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, db_journal_mode),
- "Option to provide WAL configuration for Work Ahead Logging mechanism (WAL). Enabling WAL "
- "provides higher performance. Note that WAL is not compatible with "
- "shared network file systems."
- },
-#endif
-
- /* Multiline Options */
-#ifdef FLB_HAVE_PARSER
- {
- FLB_CONFIG_MAP_BOOL, "multiline", "false",
- 0, FLB_TRUE, offsetof(struct flb_tail_config, multiline),
- "if enabled, the plugin will try to discover multiline messages and use "
- "the proper parsers to compose the outgoing messages. Note that when this "
- "option is enabled the Parser option is not used."
- },
- {
- FLB_CONFIG_MAP_TIME, "multiline_flush", FLB_TAIL_MULT_FLUSH,
- 0, FLB_TRUE, offsetof(struct flb_tail_config, multiline_flush),
- "wait period time in seconds to process queued multiline messages."
- },
- {
- FLB_CONFIG_MAP_STR, "parser_firstline", NULL,
- 0, FLB_FALSE, 0,
- "name of the parser that matches the beginning of a multiline message. "
- "Note that the regular expression defined in the parser must include a "
- "group name (named capture)."
- },
- {
- FLB_CONFIG_MAP_STR_PREFIX, "parser_", NULL,
- 0, FLB_FALSE, 0,
- "optional extra parser to interpret and structure multiline entries. This "
- "option can be used to define multiple parsers, e.g: Parser_1 ab1, "
- "Parser_2 ab2, Parser_N abN."
- },
-
- /* Multiline Core Engine based API */
- {
- FLB_CONFIG_MAP_CLIST, "multiline.parser", NULL,
- FLB_CONFIG_MAP_MULT, FLB_TRUE, offsetof(struct flb_tail_config, multiline_parsers),
- "specify one or multiple multiline parsers: docker, cri, go, java, etc."
- },
-#endif
-
- /* EOF */
- {0}
-};
-
-struct flb_input_plugin in_tail_plugin = {
- .name = "tail",
- .description = "Tail files",
- .cb_init = in_tail_init,
- .cb_pre_run = in_tail_pre_run,
- .cb_collect = NULL,
- .cb_flush_buf = NULL,
- .cb_pause = in_tail_pause,
- .cb_resume = in_tail_resume,
- .cb_exit = in_tail_exit,
- .config_map = config_map,
- .flags = 0
-};
diff --git a/fluent-bit/plugins/in_tail/tail.h b/fluent-bit/plugins/in_tail/tail.h
deleted file mode 100644
index 074f7a49f..000000000
--- a/fluent-bit/plugins/in_tail/tail.h
+++ /dev/null
@@ -1,45 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_H
-#define FLB_TAIL_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-
-/* Internal return values */
-#define FLB_TAIL_ERROR -1
-#define FLB_TAIL_OK 0
-#define FLB_TAIL_WAIT 1
-#define FLB_TAIL_BUSY 2
-
-/* Consuming mode */
-#define FLB_TAIL_STATIC 0 /* Data is being consumed through read(2) */
-#define FLB_TAIL_EVENT 1 /* Data is being consumed through inotify */
-
-/* Config */
-#define FLB_TAIL_CHUNK "32768" /* buffer chunk = 32KB */
-#define FLB_TAIL_REFRESH 60 /* refresh every 60 seconds */
-#define FLB_TAIL_ROTATE_WAIT "5" /* time to monitor after rotation */
-#define FLB_TAIL_STATIC_BATCH_SIZE "50M" /* static batch size */
-#define FLB_TAIL_EVENT_BATCH_SIZE "50M" /* event batch size */
-
-int in_tail_collect_event(void *file, struct flb_config *config);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_config.c b/fluent-bit/plugins/in_tail/tail_config.c
deleted file mode 100644
index 360b3dbea..000000000
--- a/fluent-bit/plugins/in_tail/tail_config.c
+++ /dev/null
@@ -1,472 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/multiline/flb_ml.h>
-#include <fluent-bit/multiline/flb_ml_parser.h>
-
-#include <stdlib.h>
-#include <fcntl.h>
-
-#include "tail_fs.h"
-#include "tail_db.h"
-#include "tail_config.h"
-#include "tail_scan.h"
-#include "tail_sql.h"
-#include "tail_dockermode.h"
-
-#ifdef FLB_HAVE_PARSER
-#include "tail_multiline.h"
-#endif
-
-static int multiline_load_parsers(struct flb_tail_config *ctx)
-{
- struct mk_list *head;
- struct mk_list *head_p;
- struct flb_config_map_val *mv;
- struct flb_slist_entry *val = NULL;
- struct flb_ml_parser_ins *parser_i;
-
- if (!ctx->multiline_parsers) {
- return 0;
- }
-
- /* Create Multiline context using the plugin instance name */
- ctx->ml_ctx = flb_ml_create(ctx->config, ctx->ins->name);
- if (!ctx->ml_ctx) {
- return -1;
- }
-
- /*
- * Iterate all 'multiline.parser' entries. Every entry is considered
- * a group which can have multiple multiline parser instances.
- */
- flb_config_map_foreach(head, mv, ctx->multiline_parsers) {
- mk_list_foreach(head_p, mv->val.list) {
- val = mk_list_entry(head_p, struct flb_slist_entry, _head);
-
- /* Create an instance of the defined parser */
- parser_i = flb_ml_parser_instance_create(ctx->ml_ctx, val->str);
- if (!parser_i) {
- return -1;
- }
- }
- }
-
- return 0;
-}
-
-struct flb_tail_config *flb_tail_config_create(struct flb_input_instance *ins,
- struct flb_config *config)
-{
- int ret;
- int sec;
- int i;
- long nsec;
- const char *tmp;
- struct flb_tail_config *ctx;
-
- ctx = flb_calloc(1, sizeof(struct flb_tail_config));
- if (!ctx) {
- flb_errno();
- return NULL;
- }
- ctx->config = config;
- ctx->ins = ins;
- ctx->ignore_older = 0;
- ctx->skip_long_lines = FLB_FALSE;
-#ifdef FLB_HAVE_SQLDB
- ctx->db_sync = 1; /* sqlite sync 'normal' */
-#endif
-
- /* Load the config map */
- ret = flb_input_config_map_set(ins, (void *) ctx);
- if (ret == -1) {
- flb_free(ctx);
- return NULL;
- }
-
- /* Create the channel manager */
- ret = flb_pipe_create(ctx->ch_manager);
- if (ret == -1) {
- flb_errno();
- flb_free(ctx);
- return NULL;
- }
- ctx->ch_reads = 0;
- ctx->ch_writes = 0;
-
- /* Create the pending channel */
- ret = flb_pipe_create(ctx->ch_pending);
- if (ret == -1) {
- flb_errno();
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- /* Make pending channel non-blocking */
- for (i = 0; i <= 1; i++) {
- ret = flb_pipe_set_nonblocking(ctx->ch_pending[i]);
- if (ret == -1) {
- flb_errno();
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-
- /* Config: path/pattern to read files */
- if (!ctx->path_list || mk_list_size(ctx->path_list) == 0) {
- flb_plg_error(ctx->ins, "no input 'path' was given");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* Config: seconds interval before to re-scan the path */
- tmp = flb_input_get_property("refresh_interval", ins);
- if (!tmp) {
- ctx->refresh_interval_sec = FLB_TAIL_REFRESH;
- ctx->refresh_interval_nsec = 0;
- }
- else {
- ret = flb_utils_time_split(tmp, &sec, &nsec);
- if (ret == 0) {
- ctx->refresh_interval_sec = sec;
- ctx->refresh_interval_nsec = nsec;
-
- if (sec == 0 && nsec == 0) {
- flb_plg_error(ctx->ins, "invalid 'refresh_interval' config "
- "value (%s)", tmp);
- flb_free(ctx);
- return NULL;
- }
-
- if (sec == 0 && nsec <= 1000000) {
- flb_plg_warn(ctx->ins, "very low refresh_interval "
- "(%i.%lu nanoseconds) might cause high CPU usage",
- sec, nsec);
- }
- }
- else {
- flb_plg_error(ctx->ins,
- "invalid 'refresh_interval' config value (%s)",
- tmp);
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-
- /* Config: seconds interval to monitor file after rotation */
- if (ctx->rotate_wait <= 0) {
- flb_plg_error(ctx->ins, "invalid 'rotate_wait' config value");
- flb_free(ctx);
- return NULL;
- }
-
-#ifdef FLB_HAVE_PARSER
- /* Config: multi-line support */
- if (ctx->multiline == FLB_TRUE) {
- ret = flb_tail_mult_create(ctx, ins, config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-#endif
-
- /* Config: Docker mode */
- if(ctx->docker_mode == FLB_TRUE) {
- ret = flb_tail_dmode_create(ctx, ins, config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-
- /* Validate buffer limit */
- if (ctx->buf_chunk_size > ctx->buf_max_size) {
- flb_plg_error(ctx->ins, "buffer_max_size must be >= buffer_chunk");
- flb_free(ctx);
- return NULL;
- }
-
-#ifdef FLB_HAVE_REGEX
- /* Parser / Format */
- tmp = flb_input_get_property("parser", ins);
- if (tmp) {
- ctx->parser = flb_parser_get(tmp, config);
- if (!ctx->parser) {
- flb_plg_error(ctx->ins, "parser '%s' is not registered", tmp);
- }
- }
-#endif
-
- mk_list_init(&ctx->files_static);
- mk_list_init(&ctx->files_event);
- mk_list_init(&ctx->files_rotated);
-
- /* hash table for files lookups */
- ctx->static_hash = flb_hash_table_create(FLB_HASH_TABLE_EVICT_NONE, 1000, 0);
- if (!ctx->static_hash) {
- flb_plg_error(ctx->ins, "could not create static hash");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- ctx->event_hash = flb_hash_table_create(FLB_HASH_TABLE_EVICT_NONE, 1000, 0);
- if (!ctx->event_hash) {
- flb_plg_error(ctx->ins, "could not create event hash");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
-#ifdef FLB_HAVE_SQLDB
- ctx->db = NULL;
-#endif
-
-#ifdef FLB_HAVE_REGEX
- tmp = flb_input_get_property("tag_regex", ins);
- if (tmp) {
- ctx->tag_regex = flb_regex_create(tmp);
- if (ctx->tag_regex) {
- ctx->dynamic_tag = FLB_TRUE;
- }
- else {
- flb_plg_error(ctx->ins, "invalid 'tag_regex' config value");
- }
- }
- else {
- ctx->tag_regex = NULL;
- }
-#endif
-
- /* Check if it should use dynamic tags */
- tmp = strchr(ins->tag, '*');
- if (tmp) {
- ctx->dynamic_tag = FLB_TRUE;
- }
-
-#ifdef FLB_HAVE_SQLDB
- /* Database options (needs to be set before the context) */
- tmp = flb_input_get_property("db.sync", ins);
- if (tmp) {
- if (strcasecmp(tmp, "extra") == 0) {
- ctx->db_sync = 3;
- }
- else if (strcasecmp(tmp, "full") == 0) {
- ctx->db_sync = 2;
- }
- else if (strcasecmp(tmp, "normal") == 0) {
- ctx->db_sync = 1;
- }
- else if (strcasecmp(tmp, "off") == 0) {
- ctx->db_sync = 0;
- }
- else {
- flb_plg_error(ctx->ins, "invalid database 'db.sync' value");
- }
- }
-
- /* Initialize database */
- tmp = flb_input_get_property("db", ins);
- if (tmp) {
- ctx->db = flb_tail_db_open(tmp, ins, ctx, config);
- if (!ctx->db) {
- flb_plg_error(ctx->ins, "could not open/create database");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-
- /* Journal mode check */
- tmp = flb_input_get_property("db.journal_mode", ins);
- if (tmp) {
- if (strcasecmp(tmp, "DELETE") != 0 &&
- strcasecmp(tmp, "TRUNCATE") != 0 &&
- strcasecmp(tmp, "PERSIST") != 0 &&
- strcasecmp(tmp, "MEMORY") != 0 &&
- strcasecmp(tmp, "WAL") != 0 &&
- strcasecmp(tmp, "OFF") != 0) {
-
- flb_plg_error(ctx->ins, "invalid db.journal_mode=%s", tmp);
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- }
-
- /* Prepare Statement */
- if (ctx->db) {
- /* SQL_GET_FILE */
- ret = sqlite3_prepare_v2(ctx->db->handler,
- SQL_GET_FILE,
- -1,
- &ctx->stmt_get_file,
- 0);
- if (ret != SQLITE_OK) {
- flb_plg_error(ctx->ins, "error preparing database SQL statement");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* SQL_INSERT_FILE */
- ret = sqlite3_prepare_v2(ctx->db->handler,
- SQL_INSERT_FILE,
- -1,
- &ctx->stmt_insert_file,
- 0);
- if (ret != SQLITE_OK) {
- flb_plg_error(ctx->ins, "error preparing database SQL statement");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* SQL_ROTATE_FILE */
- ret = sqlite3_prepare_v2(ctx->db->handler,
- SQL_ROTATE_FILE,
- -1,
- &ctx->stmt_rotate_file,
- 0);
- if (ret != SQLITE_OK) {
- flb_plg_error(ctx->ins, "error preparing database SQL statement");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* SQL_UPDATE_OFFSET */
- ret = sqlite3_prepare_v2(ctx->db->handler,
- SQL_UPDATE_OFFSET,
- -1,
- &ctx->stmt_offset,
- 0);
- if (ret != SQLITE_OK) {
- flb_plg_error(ctx->ins, "error preparing database SQL statement");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* SQL_DELETE_FILE */
- ret = sqlite3_prepare_v2(ctx->db->handler,
- SQL_DELETE_FILE,
- -1,
- &ctx->stmt_delete_file,
- 0);
- if (ret != SQLITE_OK) {
- flb_plg_error(ctx->ins, "error preparing database SQL statement");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- }
-#endif
-
-#ifdef FLB_HAVE_PARSER
- /* Multiline core API */
- if (ctx->multiline_parsers && mk_list_size(ctx->multiline_parsers) > 0) {
- ret = multiline_load_parsers(ctx);
- if (ret != 0) {
- flb_plg_error(ctx->ins, "could not load multiline parsers");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
-
- /* Enable auto-flush routine */
- ret = flb_ml_auto_flush_init(ctx->ml_ctx);
- if (ret == -1) {
- flb_plg_error(ctx->ins, "could not start multiline auto-flush");
- flb_tail_config_destroy(ctx);
- return NULL;
- }
- flb_plg_info(ctx->ins, "multiline core started");
- }
-#endif
-
-#ifdef FLB_HAVE_METRICS
- ctx->cmt_files_opened = cmt_counter_create(ins->cmt,
- "fluentbit", "input",
- "files_opened_total",
- "Total number of opened files",
- 1, (char *[]) {"name"});
-
- ctx->cmt_files_closed = cmt_counter_create(ins->cmt,
- "fluentbit", "input",
- "files_closed_total",
- "Total number of closed files",
- 1, (char *[]) {"name"});
-
- ctx->cmt_files_rotated = cmt_counter_create(ins->cmt,
- "fluentbit", "input",
- "files_rotated_total",
- "Total number of rotated files",
- 1, (char *[]) {"name"});
-
- /* OLD metrics */
- flb_metrics_add(FLB_TAIL_METRIC_F_OPENED,
- "files_opened", ctx->ins->metrics);
- flb_metrics_add(FLB_TAIL_METRIC_F_CLOSED,
- "files_closed", ctx->ins->metrics);
- flb_metrics_add(FLB_TAIL_METRIC_F_ROTATED,
- "files_rotated", ctx->ins->metrics);
-#endif
-
- return ctx;
-}
-
-int flb_tail_config_destroy(struct flb_tail_config *config)
-{
-
-#ifdef FLB_HAVE_PARSER
- flb_tail_mult_destroy(config);
-
- if (config->ml_ctx) {
- flb_ml_destroy(config->ml_ctx);
- }
-#endif
-
- /* Close pipe ends */
- flb_pipe_close(config->ch_manager[0]);
- flb_pipe_close(config->ch_manager[1]);
- flb_pipe_close(config->ch_pending[0]);
- flb_pipe_close(config->ch_pending[1]);
-
-#ifdef FLB_HAVE_REGEX
- if (config->tag_regex) {
- flb_regex_destroy(config->tag_regex);
- }
-#endif
-
-#ifdef FLB_HAVE_SQLDB
- if (config->db != NULL) {
- sqlite3_finalize(config->stmt_get_file);
- sqlite3_finalize(config->stmt_insert_file);
- sqlite3_finalize(config->stmt_delete_file);
- sqlite3_finalize(config->stmt_rotate_file);
- sqlite3_finalize(config->stmt_offset);
- flb_tail_db_close(config->db);
- }
-#endif
-
- if (config->static_hash) {
- flb_hash_table_destroy(config->static_hash);
- }
- if (config->event_hash) {
- flb_hash_table_destroy(config->event_hash);
- }
-
- flb_free(config);
- return 0;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_config.h b/fluent-bit/plugins/in_tail/tail_config.h
deleted file mode 100644
index dcfa54e02..000000000
--- a/fluent-bit/plugins/in_tail/tail_config.h
+++ /dev/null
@@ -1,168 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_CONFIG_H
-#define FLB_TAIL_CONFIG_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-#include <fluent-bit/flb_parser.h>
-#include <fluent-bit/flb_macros.h>
-#include <fluent-bit/flb_sqldb.h>
-#include <fluent-bit/flb_metrics.h>
-#include <fluent-bit/flb_log_event.h>
-#ifdef FLB_HAVE_REGEX
-#include <fluent-bit/flb_regex.h>
-#endif
-#ifdef FLB_HAVE_PARSER
-#include <fluent-bit/multiline/flb_ml.h>
-#endif
-
-#include <xxhash.h>
-
-/* Metrics */
-#ifdef FLB_HAVE_METRICS
-#define FLB_TAIL_METRIC_F_OPENED 100 /* number of opened files */
-#define FLB_TAIL_METRIC_F_CLOSED 101 /* number of closed files */
-#define FLB_TAIL_METRIC_F_ROTATED 102 /* number of rotated files */
-#endif
-
-struct flb_tail_config {
- int fd_notify; /* inotify fd */
- flb_pipefd_t ch_manager[2]; /* pipe: channel manager */
- flb_pipefd_t ch_pending[2]; /* pipe: pending events */
- int ch_reads; /* count number if signal reads */
- int ch_writes; /* count number of signal writes */
-
- /* Buffer Config */
- size_t buf_chunk_size; /* allocation chunks */
- size_t buf_max_size; /* max size of a buffer */
-
- /* Static files processor */
- size_t static_batch_size;
-
- /* Event files processor */
- size_t event_batch_size;
-
- /* Collectors */
- int coll_fd_static;
- int coll_fd_scan;
- int coll_fd_watcher;
- int coll_fd_rotated;
- int coll_fd_pending;
- int coll_fd_inactive;
- int coll_fd_dmode_flush;
- int coll_fd_mult_flush;
- int coll_fd_progress_check;
-
- /* Backend collectors */
- int coll_fd_fs1; /* used by fs_inotify & fs_stat */
- int coll_fd_fs2; /* only used by fs_stat */
-
- /* Configuration */
- int dynamic_tag; /* dynamic tag ? e.g: abc.* */
-#ifdef FLB_HAVE_REGEX
- struct flb_regex *tag_regex;/* path to tag regex */
-#endif
- int refresh_interval_sec; /* seconds to re-scan */
- long refresh_interval_nsec;/* nanoseconds to re-scan */
- int read_from_head; /* read new files from head */
- int rotate_wait; /* sec to wait on rotated files */
- int watcher_interval; /* watcher interval */
- int ignore_older; /* ignore fields older than X seconds */
- time_t last_pending; /* last time a 'pending signal' was emitted' */
- struct mk_list *path_list; /* list of paths to scan (glob) */
- flb_sds_t path_key; /* key name of file path */
- flb_sds_t key; /* key for unstructured record */
- int skip_long_lines; /* skip long lines */
- int skip_empty_lines; /* skip empty lines (off) */
- int exit_on_eof; /* exit fluent-bit on EOF, test */
-
- int progress_check_interval; /* watcher interval */
- int progress_check_interval_nsec; /* watcher interval */
-
-#ifdef FLB_HAVE_INOTIFY
- int inotify_watcher; /* enable/disable inotify monitor */
-#endif
- flb_sds_t offset_key; /* key name of file offset */
-
- /* Database */
-#ifdef FLB_HAVE_SQLDB
- struct flb_sqldb *db;
- int db_sync;
- int db_locking;
- flb_sds_t db_journal_mode;
- sqlite3_stmt *stmt_get_file;
- sqlite3_stmt *stmt_insert_file;
- sqlite3_stmt *stmt_delete_file;
- sqlite3_stmt *stmt_rotate_file;
- sqlite3_stmt *stmt_offset;
-#endif
-
- /* Parser / Format */
- struct flb_parser *parser;
-
- /* Multiline */
- int multiline; /* multiline enabled ? */
- int multiline_flush; /* multiline flush/wait */
- struct flb_parser *mult_parser_firstline;
- struct mk_list mult_parsers;
-
- /* Docker mode */
- int docker_mode; /* Docker mode enabled ? */
- int docker_mode_flush; /* Docker mode flush/wait */
- struct flb_parser *docker_mode_parser; /* Parser for separate multiline logs */
-
- /* Multiline core engine */
- struct flb_ml *ml_ctx;
- struct mk_list *multiline_parsers;
-
- uint64_t files_static_count; /* number of items in the static file list */
- struct mk_list files_static;
- struct mk_list files_event;
-
- /* List of rotated files that needs to be removed after 'rotate_wait' */
- struct mk_list files_rotated;
-
- /* List of shell patterns used to exclude certain file names */
- struct mk_list *exclude_list;
-
- /* Plugin input instance */
- struct flb_input_instance *ins;
-
- struct flb_log_event_encoder log_event_encoder;
- struct flb_log_event_decoder log_event_decoder;
-
- /* Metrics */
- struct cmt_counter *cmt_files_opened;
- struct cmt_counter *cmt_files_closed;
- struct cmt_counter *cmt_files_rotated;
-
- /* Hash: hash tables for quick acess to registered files */
- struct flb_hash_table *static_hash;
- struct flb_hash_table *event_hash;
-
- struct flb_config *config;
-};
-
-struct flb_tail_config *flb_tail_config_create(struct flb_input_instance *ins,
- struct flb_config *config);
-int flb_tail_config_destroy(struct flb_tail_config *config);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_db.c b/fluent-bit/plugins/in_tail/tail_db.c
deleted file mode 100644
index 664963b6d..000000000
--- a/fluent-bit/plugins/in_tail/tail_db.c
+++ /dev/null
@@ -1,277 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_sqldb.h>
-
-#include "tail_db.h"
-#include "tail_sql.h"
-#include "tail_file.h"
-
-struct query_status {
- int id;
- int rows;
- int64_t offset;
-};
-
-/* Open or create database required by tail plugin */
-struct flb_sqldb *flb_tail_db_open(const char *path,
- struct flb_input_instance *in,
- struct flb_tail_config *ctx,
- struct flb_config *config)
-{
- int ret;
- char tmp[64];
- struct flb_sqldb *db;
-
- /* Open/create the database */
- db = flb_sqldb_open(path, in->name, config);
- if (!db) {
- return NULL;
- }
-
- /* Create table schema if it don't exists */
- ret = flb_sqldb_query(db, SQL_CREATE_FILES, NULL, NULL);
- if (ret != FLB_OK) {
- flb_plg_error(ctx->ins, "db: could not create 'in_tail_files' table");
- flb_sqldb_close(db);
- return NULL;
- }
-
- if (ctx->db_sync >= 0) {
- snprintf(tmp, sizeof(tmp) - 1, SQL_PRAGMA_SYNC,
- ctx->db_sync);
- ret = flb_sqldb_query(db, tmp, NULL, NULL);
- if (ret != FLB_OK) {
- flb_plg_error(ctx->ins, "db could not set pragma 'sync'");
- flb_sqldb_close(db);
- return NULL;
- }
- }
-
- if (ctx->db_locking == FLB_TRUE) {
- ret = flb_sqldb_query(db, SQL_PRAGMA_LOCKING_MODE, NULL, NULL);
- if (ret != FLB_OK) {
- flb_plg_error(ctx->ins, "db: could not set pragma 'locking_mode'");
- flb_sqldb_close(db);
- return NULL;
- }
- }
-
- if (ctx->db_journal_mode) {
- snprintf(tmp, sizeof(tmp) - 1, SQL_PRAGMA_JOURNAL_MODE,
- ctx->db_journal_mode);
- ret = flb_sqldb_query(db, tmp, NULL, NULL);
- if (ret != FLB_OK) {
- flb_plg_error(ctx->ins, "db could not set pragma 'journal_mode'");
- flb_sqldb_close(db);
- return NULL;
- }
- }
-
- return db;
-}
-
-int flb_tail_db_close(struct flb_sqldb *db)
-{
- flb_sqldb_close(db);
- return 0;
-}
-
-/*
- * Check if an file inode exists in the database. Return FLB_TRUE or
- * FLB_FALSE
- */
-static int db_file_exists(struct flb_tail_file *file,
- struct flb_tail_config *ctx,
- uint64_t *id, uint64_t *inode, off_t *offset)
-{
- int ret;
- int exists = FLB_FALSE;
-
- /* Bind parameters */
- sqlite3_bind_int64(ctx->stmt_get_file, 1, file->inode);
- ret = sqlite3_step(ctx->stmt_get_file);
-
- if (ret == SQLITE_ROW) {
- exists = FLB_TRUE;
-
- /* id: column 0 */
- *id = sqlite3_column_int64(ctx->stmt_get_file, 0);
-
- /* offset: column 2 */
- *offset = sqlite3_column_int64(ctx->stmt_get_file, 2);
-
- /* inode: column 3 */
- *inode = sqlite3_column_int64(ctx->stmt_get_file, 3);
- }
- else if (ret == SQLITE_DONE) {
- /* all good */
- }
- else {
- exists = -1;
- }
-
- sqlite3_clear_bindings(ctx->stmt_get_file);
- sqlite3_reset(ctx->stmt_get_file);
-
- return exists;
-
-}
-
-static int db_file_insert(struct flb_tail_file *file, struct flb_tail_config *ctx)
-
-{
- int ret;
- time_t created;
-
- /* Register the file */
- created = time(NULL);
-
- /* Bind parameters */
- sqlite3_bind_text(ctx->stmt_insert_file, 1, file->name, -1, 0);
- sqlite3_bind_int64(ctx->stmt_insert_file, 2, file->offset);
- sqlite3_bind_int64(ctx->stmt_insert_file, 3, file->inode);
- sqlite3_bind_int64(ctx->stmt_insert_file, 4, created);
-
- /* Run the insert */
- ret = sqlite3_step(ctx->stmt_insert_file);
- if (ret != SQLITE_DONE) {
- sqlite3_clear_bindings(ctx->stmt_insert_file);
- sqlite3_reset(ctx->stmt_insert_file);
- flb_plg_error(ctx->ins, "cannot execute insert file %s inode=%lu",
- file->name, file->inode);
- return -1;
- }
-
- sqlite3_clear_bindings(ctx->stmt_insert_file);
- sqlite3_reset(ctx->stmt_insert_file);
-
- /* Get the database ID for this file */
- return flb_sqldb_last_id(ctx->db);
-}
-
-int flb_tail_db_file_set(struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- int ret;
- uint64_t id = 0;
- off_t offset = 0;
- uint64_t inode = 0;
-
- /* Check if the file exists */
- ret = db_file_exists(file, ctx, &id, &inode, &offset);
- if (ret == -1) {
- flb_plg_error(ctx->ins, "cannot execute query to check inode: %lu",
- file->inode);
- return -1;
- }
-
- if (ret == FLB_FALSE) {
- /* Get the database ID for this file */
- file->db_id = db_file_insert(file, ctx);
- }
- else {
- file->db_id = id;
- file->offset = offset;
- }
-
- return 0;
-}
-
-/* Update Offset v2 */
-int flb_tail_db_file_offset(struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- int ret;
-
- /* Bind parameters */
- sqlite3_bind_int64(ctx->stmt_offset, 1, file->offset);
- sqlite3_bind_int64(ctx->stmt_offset, 2, file->db_id);
-
- ret = sqlite3_step(ctx->stmt_offset);
-
- if (ret != SQLITE_DONE) {
- sqlite3_clear_bindings(ctx->stmt_offset);
- sqlite3_reset(ctx->stmt_offset);
- return -1;
- }
-
- /* Verify number of updated rows */
- ret = sqlite3_changes(ctx->db->handler);
- if (ret == 0) {
- /*
- * 'someone' like you 'the reader' or another user has deleted the database
- * entry, just restore it.
- */
- file->db_id = db_file_insert(file, ctx);
- }
-
- sqlite3_clear_bindings(ctx->stmt_offset);
- sqlite3_reset(ctx->stmt_offset);
-
- return 0;
-}
-
-/* Mark a file as rotated v2 */
-int flb_tail_db_file_rotate(const char *new_name,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- int ret;
-
- /* Bind parameters */
- sqlite3_bind_text(ctx->stmt_rotate_file, 1, new_name, -1, 0);
- sqlite3_bind_int64(ctx->stmt_rotate_file, 2, file->db_id);
-
- ret = sqlite3_step(ctx->stmt_rotate_file);
-
- sqlite3_clear_bindings(ctx->stmt_rotate_file);
- sqlite3_reset(ctx->stmt_rotate_file);
-
- if (ret != SQLITE_DONE) {
- return -1;
- }
-
- return 0;
-}
-
-/* Delete file entry from the database */
-int flb_tail_db_file_delete(struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- int ret;
-
- /* Bind parameters */
- sqlite3_bind_int64(ctx->stmt_delete_file, 1, file->db_id);
- ret = sqlite3_step(ctx->stmt_delete_file);
-
- sqlite3_clear_bindings(ctx->stmt_delete_file);
- sqlite3_reset(ctx->stmt_delete_file);
-
- if (ret != SQLITE_DONE) {
- flb_plg_error(ctx->ins, "db: error deleting entry from database: %s",
- file->name);
- return -1;
- }
-
- flb_plg_debug(ctx->ins, "db: file deleted from database: %s", file->name);
- return 0;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_db.h b/fluent-bit/plugins/in_tail/tail_db.h
deleted file mode 100644
index 7b5355d22..000000000
--- a/fluent-bit/plugins/in_tail/tail_db.h
+++ /dev/null
@@ -1,43 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_DB_H
-#define FLB_TAIL_DB_H
-
-#include <fluent-bit/flb_input.h>
-#include <fluent-bit/flb_sqldb.h>
-
-#include "tail_file.h"
-
-struct flb_sqldb *flb_tail_db_open(const char *path,
- struct flb_input_instance *in,
- struct flb_tail_config *ctx,
- struct flb_config *config);
-
-int flb_tail_db_close(struct flb_sqldb *db);
-int flb_tail_db_file_set(struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-int flb_tail_db_file_offset(struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-int flb_tail_db_file_rotate(const char *new_name,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-int flb_tail_db_file_delete(struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_dockermode.c b/fluent-bit/plugins/in_tail/tail_dockermode.c
deleted file mode 100644
index a8f20f9cd..000000000
--- a/fluent-bit/plugins/in_tail/tail_dockermode.c
+++ /dev/null
@@ -1,459 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_pack.h>
-#include <fluent-bit/flb_unescape.h>
-
-#include "tail_config.h"
-#include "tail_dockermode.h"
-#include "tail_file_internal.h"
-
-int flb_tail_dmode_create(struct flb_tail_config *ctx,
- struct flb_input_instance *ins,
- struct flb_config *config)
-{
- const char *tmp;
-
- if (ctx->multiline == FLB_TRUE) {
- flb_plg_error(ctx->ins, "Docker mode cannot be enabled when multiline "
- "is enabled");
- return -1;
- }
-
-#ifdef FLB_HAVE_REGEX
- /* First line Parser */
- tmp = flb_input_get_property("docker_mode_parser", ins);
- if (tmp) {
- ctx->docker_mode_parser = flb_parser_get(tmp, config);
- if (!ctx->docker_mode_parser) {
- flb_plg_error(ctx->ins, "parser '%s' is not registered", tmp);
- }
- }
- else {
- ctx->docker_mode_parser = NULL;
- }
-#endif
-
- tmp = flb_input_get_property("docker_mode_flush", ins);
- if (!tmp) {
- ctx->docker_mode_flush = FLB_TAIL_DMODE_FLUSH;
- }
- else {
- ctx->docker_mode_flush = atoi(tmp);
- if (ctx->docker_mode_flush <= 0) {
- ctx->docker_mode_flush = 1;
- }
- }
-
- return 0;
-}
-
-static int modify_json_cond(char *js, size_t js_len,
- char **val, size_t *val_len,
- char **out, size_t *out_len,
- int cond(char*, size_t),
- int mod(char*, size_t, char**, size_t*, void*), void *data)
-{
- int ret;
- struct flb_pack_state state;
- jsmntok_t *t;
- jsmntok_t *t_val = NULL;
- int i;
- int i_root = -1;
- int i_key = -1;
- char *old_val;
- size_t old_val_len;
- char *new_val = NULL;
- size_t new_val_len = 0;
- size_t mod_len;
-
- ret = flb_pack_state_init(&state);
- if (ret != 0) {
- ret = -1;
- goto modify_json_cond_end;
- }
-
- ret = flb_json_tokenise(js, js_len, &state);
- if (ret != 0 || state.tokens_count == 0) {
- ret = -1;
- goto modify_json_cond_end;
- }
-
- for (i = 0; i < state.tokens_count; i++) {
- t = &state.tokens[i];
-
- if (i_key >= 0) {
- if (t->parent == i_key) {
- if (t->type == JSMN_STRING) {
- t_val = t;
- }
- break;
- }
- continue;
- }
-
- if (t->start == 0 && t->parent == -1 && t->type == JSMN_OBJECT) {
- i_root = i;
- continue;
- }
- if (i_root == -1) {
- continue;
- }
-
- if (t->parent == i_root && t->type == JSMN_STRING && t->end - t->start == 3 && strncmp(js + t->start, "log", 3) == 0) {
- i_key = i;
- }
- }
-
- if (!t_val) {
- ret = -1;
- goto modify_json_cond_end;
- }
-
- *out = js;
- *out_len = js_len;
-
- if (val) {
- *val = js + t_val->start;
- }
- if (val_len) {
- *val_len = t_val->end - t_val->start;
- }
-
- if (!cond || cond(js + t_val->start, t_val->end - t_val->start)) {
- old_val = js + t_val->start;
- old_val_len = t_val->end - t_val->start;
- ret = mod(old_val, old_val_len, &new_val, &new_val_len, data);
- if (ret != 0) {
- ret = -1;
- goto modify_json_cond_end;
- }
-
- ret = 1;
-
- if (new_val == old_val) {
- goto modify_json_cond_end;
- }
-
- mod_len = js_len + new_val_len - old_val_len;
- *out = flb_malloc(mod_len);
- if (!*out) {
- flb_errno();
- flb_free(new_val);
- ret = -1;
- goto modify_json_cond_end;
- }
- *out_len = mod_len;
-
- memcpy(*out, js, t_val->start);
- memcpy(*out + t_val->start, new_val, new_val_len);
- memcpy(*out + t_val->start + new_val_len, js + t_val->end, js_len - t_val->end);
-
- flb_free(new_val);
- }
-
- modify_json_cond_end:
- flb_pack_state_reset(&state);
- if (ret < 0) {
- *out = NULL;
- }
- return ret;
-}
-
-static int unesc_ends_with_nl(char *str, size_t len)
-{
- char* unesc;
- int unesc_len;
- int nl;
-
- unesc = flb_malloc(len + 1);
- if (!unesc) {
- flb_errno();
- return FLB_FALSE;
- }
- unesc_len = flb_unescape_string(str, len, &unesc);
- nl = unesc[unesc_len - 1] == '\n';
- flb_free(unesc);
- return nl;
-}
-
-static int prepend_sds_to_str(char *str, size_t len, char **out, size_t *out_len, void *data)
-{
- flb_sds_t sds = data;
-
- if (flb_sds_len(sds) == 0) {
- *out = str;
- *out_len = len;
- return 0;
- }
-
- size_t mod_len = flb_sds_len(sds) + len;
- *out = flb_malloc(mod_len);
- if (!*out) {
- flb_errno();
- return -1;
- }
- *out_len = mod_len;
-
- memcpy(*out, sds, flb_sds_len(sds));
- memcpy(*out + flb_sds_len(sds), str, len);
- return 0;
-}
-
-static int use_sds(char *str, size_t len, char **out, size_t *out_len, void *data)
-{
- flb_sds_t sds = data;
- size_t mod_len = flb_sds_len(sds);
- *out = flb_malloc(mod_len);
- if (!*out) {
- flb_errno();
- return -1;
- }
- *out_len = mod_len;
-
- memcpy(*out, sds, flb_sds_len(sds));
- return 0;
-}
-
-int flb_tail_dmode_process_content(time_t now,
- char* line, size_t line_len,
- char **repl_line, size_t *repl_line_len,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- char* val = NULL;
- size_t val_len;
- int ret;
- void *out_buf = NULL;
- size_t out_size;
- struct flb_time out_time = {0};
- *repl_line = NULL;
- *repl_line_len = 0;
- flb_sds_t tmp;
- flb_sds_t tmp_copy;
-
-#ifdef FLB_HAVE_REGEX
- if (ctx->docker_mode_parser) {
- ret = flb_parser_do(ctx->docker_mode_parser, line, line_len,
- &out_buf, &out_size, &out_time);
- flb_free(out_buf);
-
- /*
- * Set dmode_firstline if the line meets the first-line requirement
- */
- if(ret >= 0) {
- file->dmode_firstline = true;
- }
-
- /*
- * Process if buffer contains full log line
- */
- if (flb_sds_len(file->dmode_lastline) > 0 && file->dmode_complete) {
- /*
- * Buffered log should be flushed out
- * as current line meets first-line requirement
- */
- if(ret >= 0) {
- flb_tail_dmode_flush(file, ctx);
- }
-
- /*
- * Flush the buffer if multiline has not been detected yet
- */
- if (!file->dmode_firstline) {
- flb_tail_dmode_flush(file, ctx);
- }
- }
- }
-#endif
-
- ret = modify_json_cond(line, line_len,
- &val, &val_len,
- repl_line, repl_line_len,
- unesc_ends_with_nl,
- prepend_sds_to_str, file->dmode_buf);
- if (ret >= 0) {
- /* line is a valid json */
- flb_sds_len_set(file->dmode_lastline, 0);
-
- /* concatenate current log line with buffered one */
- tmp = flb_sds_cat(file->dmode_buf, val, val_len);
- if (!tmp) {
- flb_errno();
- return -1;
- }
- file->dmode_buf = tmp;
-
- tmp_copy = flb_sds_copy(file->dmode_lastline, line, line_len);
- if (!tmp_copy) {
- flb_errno();
- return -1;
- }
-
- file->dmode_lastline = tmp_copy;
- file->dmode_flush_timeout = now + (ctx->docker_mode_flush - 1);
-
- if (ret == 0) {
- /* Line not ended with newline */
- file->dmode_complete = false;
- }
- else {
- /* Line ended with newline */
- file->dmode_complete = true;
-#ifdef FLB_HAVE_REGEX
- if (!ctx->docker_mode_parser) {
- flb_tail_dmode_flush(file, ctx);
- }
-#else
- flb_tail_dmode_flush(file, ctx);
-#endif
- }
- }
- return ret;
-}
-
-void flb_tail_dmode_flush(struct flb_tail_file *file, struct flb_tail_config *ctx)
-{
- int ret;
- char *repl_line = NULL;
- size_t repl_line_len = 0;
- void *out_buf = NULL;
- size_t out_size;
- struct flb_time out_time = {0};
- time_t now = time(NULL);
-
- if (flb_sds_len(file->dmode_lastline) == 0) {
- return;
- }
-
- flb_time_zero(&out_time);
-
- ret = modify_json_cond(file->dmode_lastline,
- flb_sds_len(file->dmode_lastline),
- NULL, NULL,
- &repl_line, &repl_line_len,
- NULL,
- use_sds, file->dmode_buf);
- if (ret < 0) {
- return;
- }
-
- flb_sds_len_set(file->dmode_buf, 0);
- flb_sds_len_set(file->dmode_lastline, 0);
- file->dmode_flush_timeout = 0;
-
-#ifdef FLB_HAVE_REGEX
- if (ctx->parser) {
- ret = flb_parser_do(ctx->parser, repl_line, repl_line_len,
- &out_buf, &out_size, &out_time);
- if (ret >= 0) {
- if (flb_time_to_double(&out_time) == 0) {
- flb_time_get(&out_time);
- }
- if (ctx->ignore_older > 0 && (now - ctx->ignore_older) > out_time.tm.tv_sec) {
- goto dmode_flush_end;
- }
-
- flb_tail_pack_line_map(&out_time, (char**) &out_buf, &out_size, file, 0);
-
- goto dmode_flush_end;
- }
- }
-#endif
-
- flb_tail_file_pack_line(NULL, repl_line, repl_line_len, file, 0);
-
- dmode_flush_end:
- flb_free(repl_line);
- flb_free(out_buf);
-}
-
-static void file_pending_flush(struct flb_tail_config *ctx,
- struct flb_tail_file *file, time_t now)
-{
- if (file->dmode_flush_timeout > now) {
- return;
- }
-
- if (flb_sds_len(file->dmode_lastline) == 0) {
- return;
- }
-
- flb_tail_dmode_flush(file, ctx);
-
- if (file->sl_log_event_encoder->output_length > 0) {
- flb_input_log_append(ctx->ins,
- file->tag_buf,
- file->tag_len,
- file->sl_log_event_encoder->output_buffer,
- file->sl_log_event_encoder->output_length);
-
- flb_log_event_encoder_reset(file->sl_log_event_encoder);
- }
-}
-
-int flb_tail_dmode_pending_flush_all(struct flb_tail_config *ctx)
-{
- time_t expired;
- struct mk_list *head;
- struct flb_tail_file *file;
-
- expired = time(NULL) + 3600;
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, expired);
- }
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, expired);
- }
-
- return 0;
-}
-
-int flb_tail_dmode_pending_flush(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- time_t now;
- struct mk_list *head;
- struct flb_tail_file *file;
- struct flb_tail_config *ctx = context;
-
- now = time(NULL);
-
- /* Iterate static event files with pending bytes */
- mk_list_foreach(head, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, now);
- }
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, now);
- }
-
- return 0;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_dockermode.h b/fluent-bit/plugins/in_tail/tail_dockermode.h
deleted file mode 100644
index 50869ff62..000000000
--- a/fluent-bit/plugins/in_tail/tail_dockermode.h
+++ /dev/null
@@ -1,38 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_DOCKERMODE_H
-#define FLB_TAIL_DOCKERMODE_H
-
-#include "tail_file.h"
-#define FLB_TAIL_DMODE_FLUSH 4
-
-int flb_tail_dmode_create(struct flb_tail_config *ctx,
- struct flb_input_instance *ins, struct flb_config *config);
-int flb_tail_dmode_process_content(time_t now,
- char* line, size_t line_len,
- char **repl_line, size_t *repl_line_len,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-void flb_tail_dmode_flush(struct flb_tail_file *file, struct flb_tail_config *ctx);
-int flb_tail_dmode_pending_flush(struct flb_input_instance *ins,
- struct flb_config *config, void *context);
-int flb_tail_dmode_pending_flush_all(struct flb_tail_config *ctx);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_file.c b/fluent-bit/plugins/in_tail/tail_file.c
deleted file mode 100644
index 2385f0626..000000000
--- a/fluent-bit/plugins/in_tail/tail_file.c
+++ /dev/null
@@ -1,1860 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <fcntl.h>
-#include <time.h>
-#ifdef FLB_SYSTEM_FREEBSD
-#include <sys/user.h>
-#include <libutil.h>
-#endif
-
-#include <fluent-bit/flb_compat.h>
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_parser.h>
-#ifdef FLB_HAVE_REGEX
-#include <fluent-bit/flb_regex.h>
-#include <fluent-bit/flb_hash_table.h>
-#endif
-
-#include "tail.h"
-#include "tail_file.h"
-#include "tail_config.h"
-#include "tail_db.h"
-#include "tail_signal.h"
-#include "tail_dockermode.h"
-#include "tail_multiline.h"
-#include "tail_scan.h"
-
-#ifdef FLB_SYSTEM_WINDOWS
-#include "win32.h"
-#endif
-
-#include <cfl/cfl.h>
-
-static inline void consume_bytes(char *buf, int bytes, int length)
-{
- memmove(buf, buf + bytes, length - bytes);
-}
-
-static uint64_t stat_get_st_dev(struct stat *st)
-{
-#ifdef FLB_SYSTEM_WINDOWS
- /* do you want to contribute with a way to extract volume serial number ? */
- return 0;
-#else
- return st->st_dev;
-#endif
-}
-
-static int stat_to_hash_bits(struct flb_tail_config *ctx, struct stat *st,
- uint64_t *out_hash)
-{
- int len;
- uint64_t st_dev;
- char tmp[64];
-
- st_dev = stat_get_st_dev(st);
-
- len = snprintf(tmp, sizeof(tmp) - 1, "%" PRIu64 ":%" PRIu64,
- st_dev, (uint64_t)st->st_ino);
-
- *out_hash = cfl_hash_64bits(tmp, len);
- return 0;
-}
-
-static int stat_to_hash_key(struct flb_tail_config *ctx, struct stat *st,
- flb_sds_t *key)
-{
- uint64_t st_dev;
- flb_sds_t tmp;
- flb_sds_t buf;
-
- buf = flb_sds_create_size(64);
- if (!buf) {
- return -1;
- }
-
- st_dev = stat_get_st_dev(st);
- tmp = flb_sds_printf(&buf, "%" PRIu64 ":%" PRIu64,
- st_dev, (uint64_t)st->st_ino);
- if (!tmp) {
- flb_sds_destroy(buf);
- return -1;
- }
-
- *key = buf;
- return 0;
-}
-
-/* Append custom keys and report the number of records processed */
-static int record_append_custom_keys(struct flb_tail_file *file,
- char *in_data, size_t in_size,
- char **out_data, size_t *out_size)
-{
- int i;
- int ret;
- int records = 0;
- msgpack_object k;
- msgpack_object v;
- struct flb_log_event event;
- struct flb_tail_config *ctx;
- struct flb_log_event_encoder encoder;
- struct flb_log_event_decoder decoder;
-
- ctx = (struct flb_tail_config *) file->config;
-
- ret = flb_log_event_decoder_init(&decoder, in_data, in_size);
-
- if (ret != FLB_EVENT_DECODER_SUCCESS) {
- return -1;
- }
-
- ret = flb_log_event_encoder_init(&encoder, FLB_LOG_EVENT_FORMAT_DEFAULT);
-
- if (ret != FLB_EVENT_ENCODER_SUCCESS) {
- flb_log_event_decoder_destroy(&decoder);
-
- return -2;
- }
-
- while (flb_log_event_decoder_next(&decoder, &event) ==
- FLB_EVENT_DECODER_SUCCESS) {
-
- ret = flb_log_event_encoder_begin_record(&encoder);
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_set_timestamp(&encoder, &event.timestamp);
- }
-
- /* append previous map keys */
- for (i = 0; i < event.body->via.map.size; i++) {
- k = event.body->via.map.ptr[i].key;
- v = event.body->via.map.ptr[i].val;
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_msgpack_object(
- &encoder,
- &k);
- }
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_msgpack_object(
- &encoder,
- &v);
- }
- }
-
- /* path_key */
- if (ctx->path_key != NULL) {
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_cstring(
- &encoder,
- file->config->path_key);
- }
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_cstring(
- &encoder,
- file->orig_name);
- }
- }
-
- /* offset_key */
- if (ctx->offset_key != NULL) {
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_cstring(
- &encoder,
- file->config->offset_key);
- }
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_append_body_uint64(
- &encoder,
- file->offset +
- file->last_processed_bytes);
- }
- }
-
- if (ret == FLB_EVENT_ENCODER_SUCCESS) {
- ret = flb_log_event_encoder_commit_record(&encoder);
- }
- else {
- flb_plg_error(file->config->ins, "error packing event : %d", ret);
-
- flb_log_event_encoder_rollback_record(&encoder);
- }
-
- /* counter */
- records++;
- }
-
- *out_data = encoder.output_buffer;
- *out_size = encoder.output_length;
-
- /* This function transfers ownership of the internal memory allocated by
- * sbuffer using msgpack_sbuffer_release which means the caller is
- * responsible for releasing the memory.
- */
- flb_log_event_encoder_claim_internal_buffer_ownership(&encoder);
-
- flb_log_event_decoder_destroy(&decoder);
- flb_log_event_encoder_destroy(&encoder);
-
- return records;
-}
-
-static int flb_tail_repack_map(struct flb_log_event_encoder *encoder,
- char *data,
- size_t data_size)
-{
- msgpack_unpacked source_map;
- size_t offset;
- int result;
- size_t index;
- msgpack_object value;
- msgpack_object key;
-
- result = FLB_EVENT_ENCODER_SUCCESS;
-
- if (data_size > 0) {
- msgpack_unpacked_init(&source_map);
-
- offset = 0;
- result = msgpack_unpack_next(&source_map,
- data,
- data_size,
- &offset);
-
- if (result == MSGPACK_UNPACK_SUCCESS) {
- result = FLB_EVENT_ENCODER_SUCCESS;
- }
- else {
- result = FLB_EVENT_DECODER_ERROR_DESERIALIZATION_FAILURE;
- }
-
- for (index = 0;
- index < source_map.data.via.map.size &&
- result == FLB_EVENT_ENCODER_SUCCESS;
- index++) {
- key = source_map.data.via.map.ptr[index].key;
- value = source_map.data.via.map.ptr[index].val;
-
- result = flb_log_event_encoder_append_body_msgpack_object(
- encoder,
- &key);
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_msgpack_object(
- encoder,
- &value);
- }
- }
-
- msgpack_unpacked_destroy(&source_map);
- }
-
- return result;
-}
-
-int flb_tail_pack_line_map(struct flb_time *time, char **data,
- size_t *data_size, struct flb_tail_file *file,
- size_t processed_bytes)
-{
- int result;
-
- result = flb_log_event_encoder_begin_record(file->sl_log_event_encoder);
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_set_timestamp(
- file->sl_log_event_encoder, time);
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_tail_repack_map(file->sl_log_event_encoder,
- *data,
- *data_size);
- }
-
- /* path_key */
- if (file->config->path_key != NULL) {
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_values(
- file->sl_log_event_encoder,
- FLB_LOG_EVENT_CSTRING_VALUE(file->config->path_key),
- FLB_LOG_EVENT_STRING_VALUE(file->orig_name,
- file->orig_name_len));
- }
- }
-
- /* offset_key */
- if (file->config->offset_key != NULL) {
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_values(
- file->sl_log_event_encoder,
- FLB_LOG_EVENT_CSTRING_VALUE(file->config->offset_key),
- FLB_LOG_EVENT_UINT64_VALUE(file->offset + processed_bytes));
- }
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_commit_record(file->sl_log_event_encoder);
- }
- else {
- flb_log_event_encoder_rollback_record(file->sl_log_event_encoder);
- }
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- flb_plg_error(file->config->ins, "error packing event");
-
- return -1;
- }
-
- return 0;
-}
-
-int flb_tail_file_pack_line(struct flb_time *time, char *data, size_t data_size,
- struct flb_tail_file *file, size_t processed_bytes)
-{
- int result;
-
- result = flb_log_event_encoder_begin_record(file->sl_log_event_encoder);
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_set_timestamp(
- file->sl_log_event_encoder, time);
- }
-
- /* path_key */
- if (file->config->path_key != NULL) {
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_values(
- file->sl_log_event_encoder,
- FLB_LOG_EVENT_CSTRING_VALUE(file->config->path_key),
- FLB_LOG_EVENT_STRING_VALUE(file->orig_name,
- file->orig_name_len));
- }
- }
-
- /* offset_key */
- if (file->config->offset_key != NULL) {
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_values(
- file->sl_log_event_encoder,
- FLB_LOG_EVENT_CSTRING_VALUE(file->config->offset_key),
- FLB_LOG_EVENT_UINT64_VALUE(file->offset + processed_bytes));
- }
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_append_body_values(
- file->sl_log_event_encoder,
- FLB_LOG_EVENT_CSTRING_VALUE(file->config->key),
- FLB_LOG_EVENT_STRING_VALUE(data,
- data_size));
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_commit_record(file->sl_log_event_encoder);
- }
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- flb_plg_error(file->config->ins, "error packing event : %d", result);
-
- return -1;
- }
-
- return 0;
-}
-
-static int ml_stream_buffer_append(struct flb_tail_file *file, char *buf_data, size_t buf_size)
-{
- int result;
-
- result = flb_log_event_encoder_emit_raw_record(
- file->ml_log_event_encoder,
- buf_data, buf_size);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- flb_plg_error(file->config->ins,
- "log event raw append error : %d",
- result);
-
- return -1;
- }
-
- return 0;
-}
-
-static int ml_stream_buffer_flush(struct flb_tail_config *ctx, struct flb_tail_file *file)
-{
- if (file->ml_log_event_encoder->output_length > 0) {
- flb_input_log_append(ctx->ins,
- file->tag_buf,
- file->tag_len,
- file->ml_log_event_encoder->output_buffer,
- file->ml_log_event_encoder->output_length);
-
- flb_log_event_encoder_reset(file->ml_log_event_encoder);
- }
-
- return 0;
-}
-
-static int process_content(struct flb_tail_file *file, size_t *bytes)
-{
- size_t len;
- int lines = 0;
- int ret;
- size_t processed_bytes = 0;
- char *data;
- char *end;
- char *p;
- void *out_buf;
- size_t out_size;
- int crlf;
- char *line;
- size_t line_len;
- char *repl_line;
- size_t repl_line_len;
- time_t now = time(NULL);
- struct flb_time out_time = {0};
- struct flb_tail_config *ctx;
-
- ctx = (struct flb_tail_config *) file->config;
-
- /* Parse the data content */
- data = file->buf_data;
- end = data + file->buf_len;
-
- /* reset last processed bytes */
- file->last_processed_bytes = 0;
-
- /* Skip null characters from the head (sometimes introduced by copy-truncate log rotation) */
- while (data < end && *data == '\0') {
- data++;
- processed_bytes++;
- }
-
- while (data < end && (p = memchr(data, '\n', end - data))) {
- len = (p - data);
- crlf = 0;
- if (file->skip_next == FLB_TRUE) {
- data += len + 1;
- processed_bytes += len + 1;
- file->skip_next = FLB_FALSE;
- continue;
- }
-
- /*
- * Empty line (just breakline)
- * ---------------------------
- * [NOTE] with the new Multiline core feature and Multiline Filter on
- * Fluent Bit v1.8.2, there are a couple of cases where stack traces
- * or multi line patterns expects an empty line (meaning only the
- * breakline), skipping empty lines on this plugin will break that
- * functionality.
- *
- * We are introducing 'skip_empty_lines=off' configuration
- * property to revert this behavior if some user is affected by
- * this change.
- */
-
- if (len == 0 && ctx->skip_empty_lines) {
- data++;
- processed_bytes++;
- continue;
- }
-
- /* Process '\r\n' */
- if (len >= 2) {
- crlf = (data[len-1] == '\r');
- if (len == 1 && crlf) {
- data += 2;
- processed_bytes += 2;
- continue;
- }
- }
-
- /* Reset time for each line */
- flb_time_zero(&out_time);
-
- line = data;
- line_len = len - crlf;
- repl_line = NULL;
-
- if (ctx->ml_ctx) {
- ret = flb_ml_append_text(ctx->ml_ctx,
- file->ml_stream_id,
- &out_time,
- line,
- line_len);
- goto go_next;
- }
- else if (ctx->docker_mode) {
- ret = flb_tail_dmode_process_content(now, line, line_len,
- &repl_line, &repl_line_len,
- file, ctx);
- if (ret >= 0) {
- if (repl_line == line) {
- repl_line = NULL;
- }
- else {
- line = repl_line;
- line_len = repl_line_len;
- }
- /* Skip normal parsers flow */
- goto go_next;
- }
- else {
- flb_tail_dmode_flush(file, ctx);
- }
- }
-
-#ifdef FLB_HAVE_PARSER
- if (ctx->parser) {
- /* Common parser (non-multiline) */
- ret = flb_parser_do(ctx->parser, line, line_len,
- &out_buf, &out_size, &out_time);
- if (ret >= 0) {
- if (flb_time_to_nanosec(&out_time) == 0L) {
- flb_time_get(&out_time);
- }
-
- /* If multiline is enabled, flush any buffered data */
- if (ctx->multiline == FLB_TRUE) {
- flb_tail_mult_flush(file, ctx);
- }
-
- flb_tail_pack_line_map(&out_time,
- (char**) &out_buf, &out_size, file,
- processed_bytes);
-
- flb_free(out_buf);
- }
- else {
- /* Parser failed, pack raw text */
- flb_tail_file_pack_line(NULL, data, len, file, processed_bytes);
- }
- }
- else if (ctx->multiline == FLB_TRUE) {
- ret = flb_tail_mult_process_content(now,
- line, line_len,
- file, ctx, processed_bytes);
-
- /* No multiline */
- if (ret == FLB_TAIL_MULT_NA) {
- flb_tail_mult_flush(file, ctx);
-
- flb_tail_file_pack_line(NULL,
- line, line_len, file, processed_bytes);
- }
- else if (ret == FLB_TAIL_MULT_MORE) {
- /* we need more data, do nothing */
- goto go_next;
- }
- else if (ret == FLB_TAIL_MULT_DONE) {
- /* Finalized */
- }
- }
- else {
- flb_tail_file_pack_line(NULL,
- line, line_len, file, processed_bytes);
- }
-#else
- flb_tail_file_pack_line(NULL,
- line, line_len, file, processed_bytes);
-#endif
-
- go_next:
- flb_free(repl_line);
- repl_line = NULL;
- /* Adjust counters */
- data += len + 1;
- processed_bytes += len + 1;
- lines++;
- file->parsed = 0;
- file->last_processed_bytes += processed_bytes;
- }
- file->parsed = file->buf_len;
-
- if (lines > 0) {
- /* Append buffer content to a chunk */
- *bytes = processed_bytes;
-
- if (file->sl_log_event_encoder->output_length > 0) {
- flb_input_log_append_records(ctx->ins,
- lines,
- file->tag_buf,
- file->tag_len,
- file->sl_log_event_encoder->output_buffer,
- file->sl_log_event_encoder->output_length);
-
- flb_log_event_encoder_reset(file->sl_log_event_encoder);
- }
- }
- else if (file->skip_next) {
- *bytes = file->buf_len;
- }
- else {
- *bytes = processed_bytes;
- }
-
- if (ctx->ml_ctx) {
- ml_stream_buffer_flush(ctx, file);
- }
-
- return lines;
-}
-
-static inline void drop_bytes(char *buf, size_t len, int pos, int bytes)
-{
- memmove(buf + pos,
- buf + pos + bytes,
- len - pos - bytes);
-}
-
-#ifdef FLB_HAVE_REGEX
-static void cb_results(const char *name, const char *value,
- size_t vlen, void *data)
-{
- struct flb_hash_table *ht = data;
-
- if (vlen == 0) {
- return;
- }
-
- flb_hash_table_add(ht, name, strlen(name), (void *) value, vlen);
-}
-#endif
-
-#ifdef FLB_HAVE_REGEX
-static int tag_compose(char *tag, struct flb_regex *tag_regex, char *fname,
- char *out_buf, size_t *out_size,
- struct flb_tail_config *ctx)
-#else
-static int tag_compose(char *tag, char *fname, char *out_buf, size_t *out_size,
- struct flb_tail_config *ctx)
-#endif
-{
- int i;
- size_t len;
- char *p;
- size_t buf_s = 0;
-#ifdef FLB_HAVE_REGEX
- ssize_t n;
- struct flb_regex_search result;
- struct flb_hash_table *ht;
- char *beg;
- char *end;
- int ret;
- const char *tmp;
- size_t tmp_s;
-#endif
-
-#ifdef FLB_HAVE_REGEX
- if (tag_regex) {
- n = flb_regex_do(tag_regex, fname, strlen(fname), &result);
- if (n <= 0) {
- flb_plg_error(ctx->ins, "invalid tag_regex pattern for file %s",
- fname);
- return -1;
- }
- else {
- ht = flb_hash_table_create(FLB_HASH_TABLE_EVICT_NONE,
- FLB_HASH_TABLE_SIZE, FLB_HASH_TABLE_SIZE);
- flb_regex_parse(tag_regex, &result, cb_results, ht);
-
- for (p = tag, beg = p; (beg = strchr(p, '<')); p = end + 2) {
- if (beg != p) {
- len = (beg - p);
- memcpy(out_buf + buf_s, p, len);
- buf_s += len;
- }
-
- beg++;
-
- end = strchr(beg, '>');
- if (end && !memchr(beg, '<', end - beg)) {
- end--;
-
- len = end - beg + 1;
- ret = flb_hash_table_get(ht, beg, len, (void *) &tmp, &tmp_s);
- if (ret != -1) {
- memcpy(out_buf + buf_s, tmp, tmp_s);
- buf_s += tmp_s;
- }
- else {
- memcpy(out_buf + buf_s, "_", 1);
- buf_s++;
- }
- }
- else {
- flb_plg_error(ctx->ins,
- "missing closing angle bracket in tag %s "
- "at position %lu", tag, beg - tag);
- flb_hash_table_destroy(ht);
- return -1;
- }
- }
-
- flb_hash_table_destroy(ht);
- if (*p) {
- len = strlen(p);
- memcpy(out_buf + buf_s, p, len);
- buf_s += len;
- }
- }
- }
- else {
-#endif
- p = strchr(tag, '*');
- if (!p) {
- return -1;
- }
-
- /* Copy tag prefix if any */
- len = (p - tag);
- if (len > 0) {
- memcpy(out_buf, tag, len);
- buf_s += len;
- }
-
- /* Append file name */
- len = strlen(fname);
- memcpy(out_buf + buf_s, fname, len);
- buf_s += len;
-
- /* Tag suffix (if any) */
- p++;
- if (*p) {
- len = strlen(tag);
- memcpy(out_buf + buf_s, p, (len - (p - tag)));
- buf_s += (len - (p - tag));
- }
-
- /* Sanitize buffer */
- for (i = 0; i < buf_s; i++) {
- if (out_buf[i] == '/' || out_buf[i] == '\\' || out_buf[i] == ':') {
- if (i > 0) {
- out_buf[i] = '.';
- }
- else {
- drop_bytes(out_buf, buf_s, i, 1);
- buf_s--;
- i--;
- }
- }
-
- if (i > 0 && out_buf[i] == '.') {
- if (out_buf[i - 1] == '.') {
- drop_bytes(out_buf, buf_s, i, 1);
- buf_s--;
- i--;
- }
- }
- else if (out_buf[i] == '*') {
- drop_bytes(out_buf, buf_s, i, 1);
- buf_s--;
- i--;
- }
- }
-
- /* Check for an ending '.' */
- if (out_buf[buf_s - 1] == '.') {
- drop_bytes(out_buf, buf_s, buf_s - 1, 1);
- buf_s--;
- }
-#ifdef FLB_HAVE_REGEX
- }
-#endif
-
- out_buf[buf_s] = '\0';
- *out_size = buf_s;
-
- return 0;
-}
-
-static inline int flb_tail_file_exists(struct stat *st,
- struct flb_tail_config *ctx)
-{
- int ret;
- uint64_t hash;
-
- ret = stat_to_hash_bits(ctx, st, &hash);
- if (ret != 0) {
- return -1;
- }
-
- /* static hash */
- if (flb_hash_table_exists(ctx->static_hash, hash)) {
- return FLB_TRUE;
- }
-
- /* event hash */
- if (flb_hash_table_exists(ctx->event_hash, hash)) {
- return FLB_TRUE;
- }
-
- return FLB_FALSE;
-}
-
-/*
- * Based in the configuration or database offset, set the proper 'offset' for the
- * file in question.
- */
-static int set_file_position(struct flb_tail_config *ctx,
- struct flb_tail_file *file)
-{
- int64_t ret;
-
-#ifdef FLB_HAVE_SQLDB
- /*
- * If the database option is enabled, try to gather the file position. The
- * database function updates the file->offset entry.
- */
- if (ctx->db) {
- ret = flb_tail_db_file_set(file, ctx);
- if (ret == 0) {
- if (file->offset > 0) {
- ret = lseek(file->fd, file->offset, SEEK_SET);
- if (ret == -1) {
- flb_errno();
- return -1;
- }
- }
- else if (ctx->read_from_head == FLB_FALSE) {
- ret = lseek(file->fd, 0, SEEK_END);
- if (ret == -1) {
- flb_errno();
- return -1;
- }
- file->offset = ret;
- flb_tail_db_file_offset(file, ctx);
- }
- return 0;
- }
- }
-#endif
-
- if (ctx->read_from_head == FLB_TRUE) {
- /* no need to seek, offset position is already zero */
- return 0;
- }
-
- /* tail... */
- ret = lseek(file->fd, 0, SEEK_END);
- if (ret == -1) {
- flb_errno();
- return -1;
- }
- file->offset = ret;
-
- return 0;
-}
-
-/* Multiline flush callback: invoked every time some content is complete */
-static int ml_flush_callback(struct flb_ml_parser *parser,
- struct flb_ml_stream *mst,
- void *data, char *buf_data, size_t buf_size)
-{
- int result;
- size_t mult_size = 0;
- char *mult_buf = NULL;
- struct flb_tail_file *file = data;
- struct flb_tail_config *ctx = file->config;
-
- if (ctx->path_key == NULL && ctx->offset_key == NULL) {
- ml_stream_buffer_append(file, buf_data, buf_size);
- }
- else {
- /* adjust the records in a new buffer */
- result = record_append_custom_keys(file,
- buf_data,
- buf_size,
- &mult_buf,
- &mult_size);
-
- if (result < 0) {
- ml_stream_buffer_append(file, buf_data, buf_size);
- }
- else {
- ml_stream_buffer_append(file, mult_buf, mult_size);
-
- flb_free(mult_buf);
- }
- }
-
- if (mst->forced_flush) {
- ml_stream_buffer_flush(ctx, file);
- }
-
- return 0;
-}
-
-int flb_tail_file_append(char *path, struct stat *st, int mode,
- struct flb_tail_config *ctx)
-{
- int fd;
- int ret;
- uint64_t stream_id;
- uint64_t ts;
- uint64_t hash_bits;
- flb_sds_t hash_key;
- size_t len;
- char *tag;
- char *name;
- size_t tag_len;
- struct flb_tail_file *file;
- struct stat lst;
- flb_sds_t inode_str;
-
- if (!S_ISREG(st->st_mode)) {
- return -1;
- }
-
- if (flb_tail_file_exists(st, ctx) == FLB_TRUE) {
- return -1;
- }
-
- fd = open(path, O_RDONLY);
- if (fd == -1) {
- flb_errno();
- flb_plg_error(ctx->ins, "cannot open %s", path);
- return -1;
- }
-
- file = flb_calloc(1, sizeof(struct flb_tail_file));
- if (!file) {
- flb_errno();
- goto error;
- }
-
- /* Initialize */
- file->watch_fd = -1;
- file->fd = fd;
-
- /* On non-windows environments check if the original path is a link */
- ret = lstat(path, &lst);
- if (ret == 0) {
- if (S_ISLNK(lst.st_mode)) {
- file->is_link = FLB_TRUE;
- file->link_inode = lst.st_ino;
- }
- }
-
- /* get unique hash for this file */
- ret = stat_to_hash_bits(ctx, st, &hash_bits);
- if (ret != 0) {
- flb_plg_error(ctx->ins, "error procesisng hash bits for file %s", path);
- goto error;
- }
- file->hash_bits = hash_bits;
-
- /* store the hash key used for hash_bits */
- ret = stat_to_hash_key(ctx, st, &hash_key);
- if (ret != 0) {
- flb_plg_error(ctx->ins, "error procesisng hash key for file %s", path);
- goto error;
- }
- file->hash_key = hash_key;
-
- file->inode = st->st_ino;
- file->offset = 0;
- file->size = st->st_size;
- file->buf_len = 0;
- file->parsed = 0;
- file->config = ctx;
- file->tail_mode = mode;
- file->tag_len = 0;
- file->tag_buf = NULL;
- file->rotated = 0;
- file->pending_bytes = 0;
- file->mult_firstline = FLB_FALSE;
- file->mult_keys = 0;
- file->mult_flush_timeout = 0;
- file->mult_skipping = FLB_FALSE;
-
- /*
- * Duplicate string into 'file' structure, the called function
- * take cares to resolve real-name of the file in case we are
- * running in a non-Linux system.
- *
- * Depending of the operating system, the way to obtain the file
- * name associated to it file descriptor can have different behaviors
- * specifically if it root path it's under a symbolic link. On Linux
- * we can trust the file name but in others it's better to solve it
- * with some extra calls.
- */
- ret = flb_tail_file_name_dup(path, file);
- if (!file->name) {
- flb_errno();
- goto error;
- }
-
- /* We keep a copy of the initial filename in orig_name. This is required
- * for path_key to continue working after rotation. */
- file->orig_name = flb_strdup(file->name);
- if (!file->orig_name) {
- flb_free(file->name);
- flb_errno();
- goto error;
- }
- file->orig_name_len = file->name_len;
-
- /* multiline msgpack buffers */
- file->mult_records = 0;
- msgpack_sbuffer_init(&file->mult_sbuf);
- msgpack_packer_init(&file->mult_pck, &file->mult_sbuf,
- msgpack_sbuffer_write);
-
- /* docker mode */
- file->dmode_flush_timeout = 0;
- file->dmode_complete = true;
- file->dmode_buf = flb_sds_create_size(ctx->docker_mode == FLB_TRUE ? 65536 : 0);
- file->dmode_lastline = flb_sds_create_size(ctx->docker_mode == FLB_TRUE ? 20000 : 0);
- file->dmode_firstline = false;
-#ifdef FLB_HAVE_SQLDB
- file->db_id = 0;
-#endif
- file->skip_next = FLB_FALSE;
- file->skip_warn = FLB_FALSE;
-
- /* Multiline core mode */
- if (ctx->ml_ctx) {
- /*
- * Create inode str to get stream_id.
- *
- * If stream_id is created by filename,
- * it will be same after file rotation and it causes invalid destruction:
- *
- * - https://github.com/fluent/fluent-bit/issues/4190
- */
- inode_str = flb_sds_create_size(64);
- flb_sds_printf(&inode_str, "%"PRIu64, file->inode);
-
- /* Create a stream for this file */
- ret = flb_ml_stream_create(ctx->ml_ctx,
- inode_str, flb_sds_len(inode_str),
- ml_flush_callback, file,
- &stream_id);
- if (ret != 0) {
- flb_plg_error(ctx->ins,
- "could not create multiline stream for file: %s",
- inode_str);
- flb_sds_destroy(inode_str);
- goto error;
- }
- file->ml_stream_id = stream_id;
- flb_sds_destroy(inode_str);
-
- /*
- * Multiline core file buffer: the multiline core functionality invokes a callback everytime a message is ready
- * to be processed by the caller, this can be a multiline message or a message that is considered 'complete'. In
- * the previous version of Tail, when it received a message this message was automatically ingested into the pipeline
- * without any previous buffering which leads to performance degradation.
- *
- * The msgpack buffer 'ml_sbuf' keeps all ML provided records and it's flushed just when the file processor finish
- * processing the "read() bytes".
- */
- }
-
- /* Local buffer */
- file->buf_size = ctx->buf_chunk_size;
- file->buf_data = flb_malloc(file->buf_size);
- if (!file->buf_data) {
- flb_errno();
- goto error;
- }
-
- /* Initialize (optional) dynamic tag */
- if (ctx->dynamic_tag == FLB_TRUE) {
- len = ctx->ins->tag_len + strlen(path) + 1;
- tag = flb_malloc(len);
- if (!tag) {
- flb_errno();
- flb_plg_error(ctx->ins, "failed to allocate tag buffer");
- goto error;
- }
-#ifdef FLB_HAVE_REGEX
- ret = tag_compose(ctx->ins->tag, ctx->tag_regex, path, tag, &tag_len, ctx);
-#else
- ret = tag_compose(ctx->ins->tag, path, tag, &tag_len, ctx);
-#endif
- if (ret == 0) {
- file->tag_len = tag_len;
- file->tag_buf = flb_strdup(tag);
- }
- flb_free(tag);
- if (ret != 0) {
- flb_plg_error(ctx->ins, "failed to compose tag for file: %s", path);
- goto error;
- }
- }
- else {
- file->tag_len = strlen(ctx->ins->tag);
- file->tag_buf = flb_strdup(ctx->ins->tag);
- }
- if (!file->tag_buf) {
- flb_plg_error(ctx->ins, "failed to set tag for file: %s", path);
- flb_errno();
- goto error;
- }
-
- if (mode == FLB_TAIL_STATIC) {
- mk_list_add(&file->_head, &ctx->files_static);
- ctx->files_static_count++;
- flb_hash_table_add(ctx->static_hash, file->hash_key, flb_sds_len(file->hash_key),
- file, sizeof(file));
- tail_signal_manager(file->config);
- }
- else if (mode == FLB_TAIL_EVENT) {
- mk_list_add(&file->_head, &ctx->files_event);
- flb_hash_table_add(ctx->event_hash, file->hash_key, flb_sds_len(file->hash_key),
- file, sizeof(file));
-
- /* Register this file into the fs_event monitoring */
- ret = flb_tail_fs_add(ctx, file);
- if (ret == -1) {
- flb_plg_error(ctx->ins, "could not register file into fs_events");
- goto error;
- }
- }
-
- /* Set the file position (database offset, head or tail) */
- ret = set_file_position(ctx, file);
- if (ret == -1) {
- flb_tail_file_remove(file);
- goto error;
- }
-
- /* Remaining bytes to read */
- file->pending_bytes = file->size - file->offset;
-
-#ifdef FLB_HAVE_METRICS
- name = (char *) flb_input_name(ctx->ins);
- ts = cfl_time_now();
- cmt_counter_inc(ctx->cmt_files_opened, ts, 1, (char *[]) {name});
-
- /* Old api */
- flb_metrics_sum(FLB_TAIL_METRIC_F_OPENED, 1, ctx->ins->metrics);
-#endif
-
- file->sl_log_event_encoder = flb_log_event_encoder_create(
- FLB_LOG_EVENT_FORMAT_DEFAULT);
-
- if (file->sl_log_event_encoder == NULL) {
- flb_tail_file_remove(file);
-
- goto error;
- }
-
- file->ml_log_event_encoder = flb_log_event_encoder_create(
- FLB_LOG_EVENT_FORMAT_DEFAULT);
-
- if (file->ml_log_event_encoder == NULL) {
- flb_tail_file_remove(file);
-
- goto error;
- }
-
- flb_plg_debug(ctx->ins,
- "inode=%"PRIu64" with offset=%"PRId64" appended as %s",
- file->inode, file->offset, path);
- return 0;
-
-error:
- if (file) {
- if (file->buf_data) {
- flb_free(file->buf_data);
- }
- if (file->name) {
- flb_free(file->name);
- }
- flb_free(file);
- }
- close(fd);
-
- return -1;
-}
-
-void flb_tail_file_remove(struct flb_tail_file *file)
-{
- uint64_t ts;
- char *name;
- struct flb_tail_config *ctx;
-
- ctx = file->config;
-
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" removing file name %s",
- file->inode, file->name);
-
- /* remove the multiline.core stream */
- if (ctx->ml_ctx && file->ml_stream_id > 0) {
- /* destroy ml stream */
- flb_ml_stream_id_destroy_all(ctx->ml_ctx, file->ml_stream_id);
- }
-
- if (file->rotated > 0) {
-#ifdef FLB_HAVE_SQLDB
- /*
- * Make sure to remove a the file entry from the database if the file
- * was rotated and it's not longer being monitored.
- */
- if (ctx->db) {
- flb_tail_db_file_delete(file, file->config);
- }
-#endif
- mk_list_del(&file->_rotate_head);
- }
-
- msgpack_sbuffer_destroy(&file->mult_sbuf);
-
- if (file->sl_log_event_encoder != NULL) {
- flb_log_event_encoder_destroy(file->sl_log_event_encoder);
- }
-
- if (file->ml_log_event_encoder != NULL) {
- flb_log_event_encoder_destroy(file->ml_log_event_encoder);
- }
-
- flb_sds_destroy(file->dmode_buf);
- flb_sds_destroy(file->dmode_lastline);
- mk_list_del(&file->_head);
- flb_tail_fs_remove(ctx, file);
-
- /* avoid deleting file with -1 fd */
- if (file->fd != -1) {
- close(file->fd);
- }
- if (file->tag_buf) {
- flb_free(file->tag_buf);
- }
-
- /* remove any potential entry from the hash tables */
- flb_hash_table_del(ctx->static_hash, file->hash_key);
- flb_hash_table_del(ctx->event_hash, file->hash_key);
-
- flb_free(file->buf_data);
- flb_free(file->name);
- flb_free(file->orig_name);
- flb_free(file->real_name);
- flb_sds_destroy(file->hash_key);
-
-#ifdef FLB_HAVE_METRICS
- name = (char *) flb_input_name(ctx->ins);
- ts = cfl_time_now();
- cmt_counter_inc(ctx->cmt_files_closed, ts, 1, (char *[]) {name});
-
- /* old api */
- flb_metrics_sum(FLB_TAIL_METRIC_F_CLOSED, 1, ctx->ins->metrics);
-#endif
-
- flb_free(file);
-}
-
-int flb_tail_file_remove_all(struct flb_tail_config *ctx)
-{
- int count = 0;
- struct mk_list *head;
- struct mk_list *tmp;
- struct flb_tail_file *file;
-
- mk_list_foreach_safe(head, tmp, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- flb_tail_file_remove(file);
- count++;
- }
-
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- flb_tail_file_remove(file);
- count++;
- }
-
- return count;
-}
-
-static int adjust_counters(struct flb_tail_config *ctx, struct flb_tail_file *file)
-{
- int ret;
- int64_t offset;
- struct stat st;
-
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_errno();
- return FLB_TAIL_ERROR;
- }
-
- /* Check if the file was truncated */
- if (file->offset > st.st_size) {
- offset = lseek(file->fd, 0, SEEK_SET);
- if (offset == -1) {
- flb_errno();
- return FLB_TAIL_ERROR;
- }
-
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" file truncated %s",
- file->inode, file->name);
- file->offset = offset;
- file->buf_len = 0;
-
- /* Update offset in the database file */
-#ifdef FLB_HAVE_SQLDB
- if (ctx->db) {
- flb_tail_db_file_offset(file, ctx);
- }
-#endif
- }
- else {
- file->size = st.st_size;
- file->pending_bytes = (st.st_size - file->offset);
- }
-
- return FLB_TAIL_OK;
-}
-
-int flb_tail_file_chunk(struct flb_tail_file *file)
-{
- int ret;
- char *tmp;
- size_t size;
- size_t capacity;
- size_t processed_bytes;
- ssize_t bytes;
- struct flb_tail_config *ctx;
-
- /* Check if we the engine issued a pause */
- ctx = file->config;
- if (flb_input_buf_paused(ctx->ins) == FLB_TRUE) {
- return FLB_TAIL_BUSY;
- }
-
- capacity = (file->buf_size - file->buf_len) - 1;
- if (capacity < 1) {
- /*
- * If there is no more room for more data, try to increase the
- * buffer under the limit of buffer_max_size.
- */
- if (file->buf_size >= ctx->buf_max_size) {
- if (ctx->skip_long_lines == FLB_FALSE) {
- flb_plg_error(ctx->ins, "file=%s requires a larger buffer size, "
- "lines are too long. Skipping file.", file->name);
- return FLB_TAIL_ERROR;
- }
-
- /* Warn the user */
- if (file->skip_warn == FLB_FALSE) {
- flb_plg_warn(ctx->ins, "file=%s have long lines. "
- "Skipping long lines.", file->name);
- file->skip_warn = FLB_TRUE;
- }
-
- /* Do buffer adjustments */
- file->offset += file->buf_len;
- file->buf_len = 0;
- file->skip_next = FLB_TRUE;
- }
- else {
- size = file->buf_size + ctx->buf_chunk_size;
- if (size > ctx->buf_max_size) {
- size = ctx->buf_max_size;
- }
-
- /* Increase the buffer size */
- tmp = flb_realloc(file->buf_data, size);
- if (tmp) {
- flb_plg_trace(ctx->ins, "file=%s increase buffer size "
- "%lu => %lu bytes",
- file->name, file->buf_size, size);
- file->buf_data = tmp;
- file->buf_size = size;
- }
- else {
- flb_errno();
- flb_plg_error(ctx->ins, "cannot increase buffer size for %s, "
- "skipping file.", file->name);
- return FLB_TAIL_ERROR;
- }
- }
- capacity = (file->buf_size - file->buf_len) - 1;
- }
-
- bytes = read(file->fd, file->buf_data + file->buf_len, capacity);
- if (bytes > 0) {
- /* we read some data, let the content processor take care of it */
- file->buf_len += bytes;
- file->buf_data[file->buf_len] = '\0';
-
- /* Now that we have some data in the buffer, call the data processor
- * which aims to cut lines and register the entries into the engine.
- *
- * The returned value is the absolute offset the file must be seek
- * now. It may need to get back a few bytes at the beginning of a new
- * line.
- */
- ret = process_content(file, &processed_bytes);
- if (ret < 0) {
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" file=%s process content ERROR",
- file->inode, file->name);
- return FLB_TAIL_ERROR;
- }
-
- /* Adjust the file offset and buffer */
- file->offset += processed_bytes;
- consume_bytes(file->buf_data, processed_bytes, file->buf_len);
- file->buf_len -= processed_bytes;
- file->buf_data[file->buf_len] = '\0';
-
-#ifdef FLB_HAVE_SQLDB
- if (file->config->db) {
- flb_tail_db_file_offset(file, file->config);
- }
-#endif
-
- /* adjust file counters, returns FLB_TAIL_OK or FLB_TAIL_ERROR */
- ret = adjust_counters(ctx, file);
-
- /* Data was consumed but likely some bytes still remain */
- return ret;
- }
- else if (bytes == 0) {
- /* We reached the end of file, let's wait for some incoming data */
- ret = adjust_counters(ctx, file);
- if (ret == FLB_TAIL_OK) {
- return FLB_TAIL_WAIT;
- }
- else {
- return FLB_TAIL_ERROR;
- }
- }
- else {
- /* error */
- flb_errno();
- flb_plg_error(ctx->ins, "error reading %s", file->name);
- return FLB_TAIL_ERROR;
- }
-
- return FLB_TAIL_ERROR;
-}
-
-/* Returns FLB_TRUE if a file has been rotated, otherwise FLB_FALSE */
-int flb_tail_file_is_rotated(struct flb_tail_config *ctx,
- struct flb_tail_file *file)
-{
- int ret;
- char *name;
- struct stat st;
-
- /*
- * Do not double-check already rotated files since the caller of this
- * function will trigger a rotation.
- */
- if (file->rotated != 0) {
- return FLB_FALSE;
- }
-
- /* Check if the 'original monitored file' is a link and rotated */
- if (file->is_link == FLB_TRUE) {
- ret = lstat(file->name, &st);
- if (ret == -1) {
- /* Broken link or missing file */
- if (errno == ENOENT) {
- flb_plg_info(ctx->ins, "inode=%"PRIu64" link_rotated: %s",
- file->link_inode, file->name);
- return FLB_TRUE;
- }
- else {
- flb_errno();
- flb_plg_error(ctx->ins,
- "link_inode=%"PRIu64" cannot detect if file: %s",
- file->link_inode, file->name);
- return -1;
- }
- }
- else {
- /* The file name is there, check if the same that we have */
- if (st.st_ino != file->link_inode) {
- return FLB_TRUE;
- }
- }
- }
-
- /* Retrieve the real file name, operating system lookup */
- name = flb_tail_file_name(file);
- if (!name) {
- flb_plg_error(ctx->ins,
- "inode=%"PRIu64" cannot detect if file was rotated: %s",
- file->inode, file->name);
- return -1;
- }
-
-
- /* Get stats from the file name */
- ret = stat(name, &st);
- if (ret == -1) {
- flb_errno();
- flb_free(name);
- return -1;
- }
-
- /* Compare inodes and names */
- if (file->inode == st.st_ino &&
- flb_tail_target_file_name_cmp(name, file) == 0) {
- flb_free(name);
- return FLB_FALSE;
- }
-
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" rotated: %s => %s",
- file->inode, file->name, name);
-
- flb_free(name);
- return FLB_TRUE;
-}
-
-/* Promote a event in the static list to the dynamic 'events' interface */
-int flb_tail_file_to_event(struct flb_tail_file *file)
-{
- int ret;
- struct stat st;
- struct flb_tail_config *ctx = file->config;
-
- /* Check if the file promoted have pending bytes */
- ret = fstat(file->fd, &st);
- if (ret != 0) {
- flb_errno();
- return -1;
- }
-
- if (file->offset < st.st_size) {
- file->pending_bytes = (st.st_size - file->offset);
- tail_signal_pending(file->config);
- }
- else {
- file->pending_bytes = 0;
- }
-
- /* Check if the file has been rotated */
- ret = flb_tail_file_is_rotated(ctx, file);
- if (ret == FLB_TRUE) {
- flb_tail_file_rotated(file);
- }
-
- /* Notify the fs-event handler that we will start monitoring this 'file' */
- ret = flb_tail_fs_add(ctx, file);
- if (ret == -1) {
- return -1;
- }
-
- /* List swap: change from 'static' to 'event' list */
- mk_list_del(&file->_head);
- ctx->files_static_count--;
- flb_hash_table_del(ctx->static_hash, file->hash_key);
-
- mk_list_add(&file->_head, &file->config->files_event);
- flb_hash_table_add(ctx->event_hash, file->hash_key, flb_sds_len(file->hash_key),
- file, sizeof(file));
-
- file->tail_mode = FLB_TAIL_EVENT;
-
- return 0;
-}
-
-/*
- * Given an open file descriptor, return the filename. This function is a
- * bit slow and it aims to be used only when a file is rotated.
- */
-char *flb_tail_file_name(struct flb_tail_file *file)
-{
- int ret;
- char *buf;
-#ifdef __linux__
- ssize_t s;
- char tmp[128];
-#elif defined(__APPLE__)
- char path[PATH_MAX];
-#elif defined(FLB_SYSTEM_WINDOWS)
- HANDLE h;
-#elif defined(FLB_SYSTEM_FREEBSD)
- struct kinfo_file *file_entries;
- int file_count;
- int file_index;
-#endif
-
- buf = flb_malloc(PATH_MAX);
- if (!buf) {
- flb_errno();
- return NULL;
- }
-
-#ifdef __linux__
- ret = snprintf(tmp, sizeof(tmp) - 1, "/proc/%i/fd/%i", getpid(), file->fd);
- if (ret == -1) {
- flb_errno();
- flb_free(buf);
- return NULL;
- }
-
- s = readlink(tmp, buf, PATH_MAX);
- if (s == -1) {
- flb_free(buf);
- flb_errno();
- return NULL;
- }
- buf[s] = '\0';
-
-#elif __APPLE__
- int len;
-
- ret = fcntl(file->fd, F_GETPATH, path);
- if (ret == -1) {
- flb_errno();
- flb_free(buf);
- return NULL;
- }
-
- len = strlen(path);
- memcpy(buf, path, len);
- buf[len] = '\0';
-
-#elif defined(FLB_SYSTEM_WINDOWS)
- int len;
-
- h = (HANDLE) _get_osfhandle(file->fd);
- if (h == INVALID_HANDLE_VALUE) {
- flb_errno();
- flb_free(buf);
- return NULL;
- }
-
- /* This function returns the length of the string excluding "\0"
- * and the resulting path has a "\\?\" prefix.
- */
- len = GetFinalPathNameByHandleA(h, buf, PATH_MAX, FILE_NAME_NORMALIZED);
- if (len == 0 || len >= PATH_MAX) {
- flb_free(buf);
- return NULL;
- }
-
- if (strstr(buf, "\\\\?\\")) {
- memmove(buf, buf + 4, len + 1);
- }
-#elif defined(FLB_SYSTEM_FREEBSD)
- if ((file_entries = kinfo_getfile(getpid(), &file_count)) == NULL) {
- flb_free(buf);
- return NULL;
- }
-
- for (file_index=0; file_index < file_count; file_index++) {
- if (file_entries[file_index].kf_fd == file->fd) {
- strncpy(buf, file_entries[file_index].kf_path, PATH_MAX - 1);
- buf[PATH_MAX - 1] = 0;
- break;
- }
- }
- free(file_entries);
-#endif
- return buf;
-}
-
-int flb_tail_file_name_dup(char *path, struct flb_tail_file *file)
-{
- file->name = flb_strdup(path);
- if (!file->name) {
- flb_errno();
- return -1;
- }
- file->name_len = strlen(file->name);
-
- if (file->real_name) {
- flb_free(file->real_name);
- }
-
- file->real_name = flb_tail_file_name(file);
- if (!file->real_name) {
- flb_errno();
- flb_free(file->name);
- file->name = NULL;
- return -1;
- }
-
- return 0;
-}
-
-/* Invoked every time a file was rotated */
-int flb_tail_file_rotated(struct flb_tail_file *file)
-{
- int ret;
- uint64_t ts;
- char *name;
- char *i_name;
- char *tmp;
- struct stat st;
- struct flb_tail_config *ctx = file->config;
-
- /* Get the new file name */
- name = flb_tail_file_name(file);
- if (!name) {
- return -1;
- }
-
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" rotated %s -> %s",
- file->inode, file->name, name);
-
- /* Update local file entry */
- tmp = file->name;
- flb_tail_file_name_dup(name, file);
- flb_plg_info(ctx->ins, "inode=%"PRIu64" handle rotation(): %s => %s",
- file->inode, tmp, file->name);
- if (file->rotated == 0) {
- file->rotated = time(NULL);
- mk_list_add(&file->_rotate_head, &file->config->files_rotated);
-
- /* Rotate the file in the database */
-#ifdef FLB_HAVE_SQLDB
- if (file->config->db) {
- ret = flb_tail_db_file_rotate(name, file, file->config);
- if (ret == -1) {
- flb_plg_error(ctx->ins, "could not rotate file %s->%s in database",
- file->name, name);
- }
- }
-#endif
-
-#ifdef FLB_HAVE_METRICS
- i_name = (char *) flb_input_name(ctx->ins);
- ts = cfl_time_now();
- cmt_counter_inc(ctx->cmt_files_rotated, ts, 1, (char *[]) {i_name});
-
- /* OLD api */
- flb_metrics_sum(FLB_TAIL_METRIC_F_ROTATED,
- 1, file->config->ins->metrics);
-#endif
-
- /* Check if a new file has been created */
- ret = stat(tmp, &st);
- if (ret == 0 && st.st_ino != file->inode) {
- if (flb_tail_file_exists(&st, ctx) == FLB_FALSE) {
- ret = flb_tail_file_append(tmp, &st, FLB_TAIL_STATIC, ctx);
- if (ret == -1) {
- flb_tail_scan(ctx->path_list, ctx);
- }
- else {
- tail_signal_manager(file->config);
- }
- }
- }
- }
- flb_free(tmp);
- flb_free(name);
-
- return 0;
-}
-
-static int check_purge_deleted_file(struct flb_tail_config *ctx,
- struct flb_tail_file *file, time_t ts)
-{
- int ret;
- struct stat st;
-
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_plg_debug(ctx->ins, "error stat(2) %s, removing", file->name);
- flb_tail_file_remove(file);
- return FLB_TRUE;
- }
-
- if (st.st_nlink == 0) {
- flb_plg_debug(ctx->ins, "purge: monitored file has been deleted: %s",
- file->name);
-#ifdef FLB_HAVE_SQLDB
- if (ctx->db) {
- /* Remove file entry from the database */
- flb_tail_db_file_delete(file, file->config);
- }
-#endif
- /* Remove file from the monitored list */
- flb_tail_file_remove(file);
- return FLB_TRUE;
- }
-
- return FLB_FALSE;
-}
-
-/* Purge rotated and deleted files */
-int flb_tail_file_purge(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- int ret;
- int count = 0;
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_file *file;
- struct flb_tail_config *ctx = context;
- time_t now;
- struct stat st;
-
- /* Rotated files */
- now = time(NULL);
- mk_list_foreach_safe(head, tmp, &ctx->files_rotated) {
- file = mk_list_entry(head, struct flb_tail_file, _rotate_head);
- if ((file->rotated + ctx->rotate_wait) <= now) {
- ret = fstat(file->fd, &st);
- if (ret == 0) {
- flb_plg_debug(ctx->ins,
- "inode=%"PRIu64" purge rotated file %s " \
- "(offset=%"PRId64" / size = %"PRIu64")",
- file->inode, file->name, file->offset, (uint64_t)st.st_size);
- if (file->pending_bytes > 0 && flb_input_buf_paused(ins)) {
- flb_plg_warn(ctx->ins, "purged rotated file while data "
- "ingestion is paused, consider increasing "
- "rotate_wait");
- }
- }
- else {
- flb_plg_debug(ctx->ins,
- "inode=%"PRIu64" purge rotated file %s (offset=%"PRId64")",
- file->inode, file->name, file->offset);
- }
-
- flb_tail_file_remove(file);
- count++;
- }
- }
-
- /*
- * Deleted files: under high load scenarios, exists the chances that in
- * our event loop we miss some notifications about a file. In order to
- * sanitize our list of monitored files we will iterate all of them and check
- * if they have been deleted or not.
- */
- mk_list_foreach_safe(head, tmp, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- check_purge_deleted_file(ctx, file, now);
- }
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- check_purge_deleted_file(ctx, file, now);
- }
-
- return count;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_file.h b/fluent-bit/plugins/in_tail/tail_file.h
deleted file mode 100644
index 796224c37..000000000
--- a/fluent-bit/plugins/in_tail/tail_file.h
+++ /dev/null
@@ -1,137 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_FILE_H
-#define FLB_TAIL_FILE_H
-
-#include <sys/types.h>
-#include <sys/stat.h>
-
-#include <fluent-bit/flb_compat.h>
-#include <fluent-bit/flb_input.h>
-
-#include "tail.h"
-#include "tail_fs.h"
-#include "tail_config.h"
-#include "tail_file_internal.h"
-
-#ifdef FLB_SYSTEM_WINDOWS
-#include "win32.h"
-#endif
-
-#ifdef FLB_HAVE_REGEX
-#define FLB_HASH_TABLE_SIZE 50
-#endif
-
-/* return the file modification time in seconds since epoch */
-static inline int64_t flb_tail_stat_mtime(struct stat *st)
-{
-#if defined(FLB_HAVE_WINDOWS)
- return (int64_t) st->st_mtime;
-#elif defined(__APPLE__) && !defined(_POSIX_C_SOURCE)
- return (int64_t) st->st_mtimespec.tv_sec;
-#elif (_POSIX_C_SOURCE >= 200809L || \
- defined(_BSD_SOURCE) || defined(_SVID_SOURCE) || \
- defined(__BIONIC__) || (defined (__SVR4) && defined (__sun)) || \
- defined(__FreeBSD__) || defined (__linux__))
- return (int64_t) st->st_mtim.tv_sec;
-#elif defined(_AIX)
- return (int64_t) st->st_mtime;
-#else
- return (int64_t) st->st_mtime;
-#endif
-
- /* backend unsupported: submit a PR :) */
- return -1;
-}
-
-static inline int flb_tail_target_file_name_cmp(char *name,
- struct flb_tail_file *file)
-{
- int ret;
- char *name_a = NULL;
- char *name_b = NULL;
- char *base_a = NULL;
- char *base_b = NULL;
-
- name_a = flb_strdup(name);
- if (!name_a) {
- flb_errno();
- ret = -1;
- goto out;
- }
-
- base_a = flb_strdup(basename(name_a));
- if (!base_a) {
- flb_errno();
- ret = -1;
- goto out;
- }
-
-#if defined(FLB_SYSTEM_WINDOWS)
- name_b = flb_strdup(file->real_name);
- if (!name_b) {
- flb_errno();
- ret = -1;
- goto out;
- }
-
- base_b = basename(name_b);
- ret = _stricmp(base_a, base_b);
-#else
- name_b = flb_strdup(file->real_name);
- if (!name_b) {
- flb_errno();
- ret = -1;
- goto out;
- }
- base_b = basename(name_b);
- ret = strcmp(base_a, base_b);
-#endif
-
- out:
- flb_free(name_a);
- flb_free(name_b);
- flb_free(base_a);
-
- /* FYI: 'base_b' never points to a new allocation, no flb_free is needed */
-
- return ret;
-}
-
-int flb_tail_file_name_dup(char *path, struct flb_tail_file *file);
-int flb_tail_file_to_event(struct flb_tail_file *file);
-int flb_tail_file_chunk(struct flb_tail_file *file);
-int flb_tail_file_append(char *path, struct stat *st, int mode,
- struct flb_tail_config *ctx);
-void flb_tail_file_remove(struct flb_tail_file *file);
-int flb_tail_file_remove_all(struct flb_tail_config *ctx);
-char *flb_tail_file_name(struct flb_tail_file *file);
-int flb_tail_file_is_rotated(struct flb_tail_config *ctx,
- struct flb_tail_file *file);
-int flb_tail_file_rotated(struct flb_tail_file *file);
-int flb_tail_file_purge(struct flb_input_instance *ins,
- struct flb_config *config, void *context);
-int flb_tail_pack_line_map(struct flb_time *time, char **data,
- size_t *data_size, struct flb_tail_file *file,
- size_t processed_bytes);
-int flb_tail_file_pack_line(struct flb_time *time, char *data, size_t data_size,
- struct flb_tail_file *file, size_t processed_bytes);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_file_internal.h b/fluent-bit/plugins/in_tail/tail_file_internal.h
deleted file mode 100644
index 6d95c87c1..000000000
--- a/fluent-bit/plugins/in_tail/tail_file_internal.h
+++ /dev/null
@@ -1,130 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_INTERNAL_H
-#define FLB_TAIL_INTERNAL_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-#include <fluent-bit/flb_sds.h>
-#include <fluent-bit/flb_time.h>
-#include <fluent-bit/flb_log_event_encoder.h>
-
-#ifdef FLB_HAVE_PARSER
-#include <fluent-bit/multiline/flb_ml.h>
-#endif
-
-#include "tail.h"
-#include "tail_config.h"
-
-struct flb_tail_file {
- /* Inotify */
- int watch_fd;
- /* file lookup info */
- int fd;
- int64_t size;
- int64_t offset;
- int64_t last_line;
- uint64_t dev_id;
- uint64_t inode;
- uint64_t link_inode;
- int is_link;
- char *name; /* target file name given by scan routine */
- char *real_name; /* real file name in the file system */
- char *orig_name; /* original file name (before rotation) */
- size_t name_len;
- size_t orig_name_len;
- time_t rotated;
- int64_t pending_bytes;
-
- /* dynamic tag for this file */
- int tag_len;
- char *tag_buf;
-
- /* OLD multiline */
- time_t mult_flush_timeout; /* time when multiline started */
- int mult_firstline; /* bool: mult firstline found ? */
- int mult_firstline_append; /* bool: mult firstline appendable ? */
- int mult_skipping; /* skipping because ignode_older than ? */
- int mult_keys; /* total number of buffered keys */
-
- int mult_records; /* multiline records counter mult_sbuf */
- msgpack_sbuffer mult_sbuf; /* temporary msgpack buffer */
- msgpack_packer mult_pck; /* temporary msgpack packer */
- struct flb_time mult_time; /* multiline time parsed from first line */
-
- /* OLD docker mode */
- time_t dmode_flush_timeout; /* time when docker mode started */
- flb_sds_t dmode_buf; /* buffer for docker mode */
- flb_sds_t dmode_lastline; /* last incomplete line */
- bool dmode_complete; /* buffer contains completed log */
- bool dmode_firstline; /* dmode mult firstline found ? */
-
- /* multiline engine: file stream_id and local buffers */
- uint64_t ml_stream_id;
-
- /* content parsing, positions and buffer */
- size_t parsed;
- size_t buf_len;
- size_t buf_size;
- char *buf_data;
-
- /*
- * This value represent the number of bytes procesed by process_content()
- * in the last iteration.
- */
- size_t last_processed_bytes;
-
- /*
- * Long-lines handling: this flag is enabled when a previous line was
- * too long and the buffer did not contain a \n, so when reaching the
- * missing \n, skip that content and move forward.
- *
- * This flag is only set when Skip_Long_Lines is On.
- */
- int skip_next;
-
- /* Did the plugin already warn the user about long lines ? */
- int skip_warn;
-
- /* Opaque data type for specific fs-event backend data */
- void *fs_backend;
-
- /* database reference */
- uint64_t db_id;
-
- uint64_t hash_bits;
- flb_sds_t hash_key;
-
- /* There are dedicated log event encoders for
- * single and multi line events because I am respecting
- * the old behavior which resulted in grouping both types
- * of logs in tail_file.c but I don't know if this is
- * strictly necessary.
- */
- struct flb_log_event_encoder *ml_log_event_encoder;
- struct flb_log_event_encoder *sl_log_event_encoder;
-
- /* reference */
- int tail_mode;
- struct flb_tail_config *config;
- struct mk_list _head;
- struct mk_list _rotate_head;
-};
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_fs.h b/fluent-bit/plugins/in_tail/tail_fs.h
deleted file mode 100644
index 948954333..000000000
--- a/fluent-bit/plugins/in_tail/tail_fs.h
+++ /dev/null
@@ -1,96 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_FS_H
-#define FLB_TAIL_FS_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-
-#include "tail_config.h"
-#include "tail_file_internal.h"
-
-#include "tail_fs_stat.h"
-#ifdef FLB_HAVE_INOTIFY
-#include "tail_fs_inotify.h"
-#endif
-
-static inline int flb_tail_fs_init(struct flb_input_instance *in,
- struct flb_tail_config *ctx, struct flb_config *config)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_init(in, ctx, config);
- }
-#endif
- return flb_tail_fs_stat_init(in, ctx, config);
-}
-
-static inline void flb_tail_fs_pause(struct flb_tail_config *ctx)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_pause(ctx);
- }
-#endif
- return flb_tail_fs_stat_pause(ctx);
-}
-
-static inline void flb_tail_fs_resume(struct flb_tail_config *ctx)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_resume(ctx);
- }
-#endif
- return flb_tail_fs_stat_resume(ctx);
-}
-
-static inline int flb_tail_fs_add(struct flb_tail_config *ctx, struct flb_tail_file *file)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_add(file);
- }
-#endif
- return flb_tail_fs_stat_add(file);
-}
-
-static inline int flb_tail_fs_remove(struct flb_tail_config *ctx, struct flb_tail_file *file)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_remove(file);
- }
-#endif
- return flb_tail_fs_stat_remove(file);
-}
-
-static inline int flb_tail_fs_exit(struct flb_tail_config *ctx)
-{
-#ifdef FLB_HAVE_INOTIFY
- if (ctx->inotify_watcher) {
- return flb_tail_fs_inotify_exit(ctx);
- }
-#endif
- return flb_tail_fs_stat_exit(ctx);
-}
-
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_fs_inotify.c b/fluent-bit/plugins/in_tail/tail_fs_inotify.c
deleted file mode 100644
index 59d10ca08..000000000
--- a/fluent-bit/plugins/in_tail/tail_fs_inotify.c
+++ /dev/null
@@ -1,433 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#define _DEFAULT_SOURCE
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-#include <fluent-bit/flb_input_plugin.h>
-
-#include <stdio.h>
-#include <stdlib.h>
-#include <unistd.h>
-#include <sys/inotify.h>
-
-#include "tail_config.h"
-#include "tail_file.h"
-#include "tail_db.h"
-#include "tail_signal.h"
-
-#include <limits.h>
-#include <fcntl.h>
-
-#include <sys/ioctl.h>
-
-static int debug_event_mask(struct flb_tail_config *ctx,
- struct flb_tail_file *file,
- uint32_t mask)
-{
- flb_sds_t buf;
- int buf_size = 256;
-
- /* Only enter this function if debug mode is allowed */
- if (flb_log_check(FLB_LOG_DEBUG) == 0) {
- return 0;
- }
-
- if (file) {
- buf_size = file->name_len + 128;
- }
-
- if (buf_size < 256) {
- buf_size = 256;
- }
-
- /* Create buffer */
- buf = flb_sds_create_size(buf_size);
- if (!buf) {
- return -1;
- }
-
- /* Print info into sds */
- if (file) {
- flb_sds_printf(&buf, "inode=%"PRIu64", %s, events: ", file->inode, file->name);
- }
- else {
- flb_sds_printf(&buf, "events: ");
- }
-
- if (mask & IN_ATTRIB) {
- flb_sds_printf(&buf, "IN_ATTRIB ");
- }
- if (mask & IN_IGNORED) {
- flb_sds_printf(&buf, "IN_IGNORED ");
- }
- if (mask & IN_MODIFY) {
- flb_sds_printf(&buf, "IN_MODIFY ");
- }
- if (mask & IN_MOVE_SELF) {
- flb_sds_printf(&buf, "IN_MOVE_SELF ");
- }
- if (mask & IN_Q_OVERFLOW) {
- flb_sds_printf(&buf, "IN_Q_OVERFLOW ");
- }
-
- flb_plg_debug(ctx->ins, "%s", buf);
- flb_sds_destroy(buf);
-
- return 0;
-}
-
-static int tail_fs_add(struct flb_tail_file *file, int check_rotated)
-{
- int flags;
- int watch_fd;
- char *name;
- struct flb_tail_config *ctx = file->config;
-
- /*
- * If there is no watcher associated, we only want to monitor events if
- * this file is rotated to somewhere. Note at this point we are polling
- * lines from the file and once we reach EOF (and a watch_fd exists),
- * we update the flags to receive notifications.
- */
- flags = IN_ATTRIB | IN_IGNORED | IN_MODIFY | IN_Q_OVERFLOW;
-
- if (check_rotated == FLB_TRUE) {
- flags |= IN_MOVE_SELF;
- }
-
- /*
- * Double check real name of the file associated to the inode:
- *
- * Inotify interface in the Kernel uses the inode number as a real reference
- * for the file we have opened. If for some reason the file we are pointing
- * out in file->name has been rotated and not been updated, we might not add
- * the watch to the real file we aim to.
- *
- * A case like this can generate the issue:
- *
- * 1. inode=1 : file a.log is being watched
- * 2. inode=1 : file a.log is rotated to a.log.1, but notification not
- * delivered yet.
- * 3. inode=2 : new file 'a.log' is created
- * 4. inode=2 : the scan_path routine discover the new 'a.log' file
- * 5. inode=2 : add an inotify watcher for 'a.log'
- * 6. conflict: inotify_add_watch() receives the path 'a.log',
- */
-
- name = flb_tail_file_name(file);
- if (!name) {
- flb_plg_error(ctx->ins, "inode=%"PRIu64" cannot get real filename for inotify",
- file->inode);
- return -1;
- }
-
- /* Register or update the flags */
- watch_fd = inotify_add_watch(ctx->fd_notify, name, flags);
- flb_free(name);
-
- if (watch_fd == -1) {
- flb_errno();
- if (errno == ENOSPC) {
- flb_plg_error(ctx->ins, "inotify: The user limit on the total "
- "number of inotify watches was reached or the kernel "
- "failed to allocate a needed resource (ENOSPC)");
- }
- return -1;
- }
- file->watch_fd = watch_fd;
- flb_plg_info(ctx->ins, "inotify_fs_add(): inode=%"PRIu64" watch_fd=%i name=%s",
- file->inode, watch_fd, file->name);
- return 0;
-}
-
-static int flb_tail_fs_add_rotated(struct flb_tail_file *file)
-{
- return tail_fs_add(file, FLB_FALSE);
-}
-
-static int tail_fs_event(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- int ret;
- int64_t offset;
- struct mk_list *head;
- struct mk_list *tmp;
- struct flb_tail_config *ctx = in_context;
- struct flb_tail_file *file = NULL;
- struct inotify_event ev;
- struct stat st;
-
- /* Read the event */
- ret = read(ctx->fd_notify, &ev, sizeof(struct inotify_event));
- if (ret < 1) {
- return -1;
- }
-
- /* Lookup watched file */
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- if (file->watch_fd != ev.wd) {
- file = NULL;
- continue;
- }
- break;
- }
-
- if (!file) {
- return -1;
- }
-
- /* Debug event */
- debug_event_mask(ctx, file, ev.mask);
-
- if (ev.mask & IN_IGNORED) {
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" watch_fd=%i IN_IGNORED",
- file->inode, ev.wd);
- return -1;
- }
-
- /* Check file rotation (only if it has not been rotated before) */
- if (ev.mask & IN_MOVE_SELF && file->rotated == 0) {
- flb_plg_debug(ins, "inode=%"PRIu64" rotated IN_MOVE SELF '%s'",
- file->inode, file->name);
-
- /* A rotated file must be re-registered */
- flb_tail_file_rotated(file);
- flb_tail_fs_remove(ctx, file);
- flb_tail_fs_add_rotated(file);
- }
-
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_plg_debug(ins, "inode=%"PRIu64" error stat(2) %s, removing",
- file->inode, file->name);
- flb_tail_file_remove(file);
- return 0;
- }
- file->size = st.st_size;
- file->pending_bytes = (file->size - file->offset);
-
- /* File was removed ? */
- if (ev.mask & IN_ATTRIB) {
- /* Check if the file have been deleted */
- if (st.st_nlink == 0) {
- flb_plg_debug(ins, "inode=%"PRIu64" file has been deleted: %s",
- file->inode, file->name);
-
-#ifdef FLB_HAVE_SQLDB
- if (ctx->db) {
- /* Remove file entry from the database */
- flb_tail_db_file_delete(file, ctx);
- }
-#endif
- /* Remove file from the monitored list */
- flb_tail_file_remove(file);
- return 0;
- }
- }
-
- if (ev.mask & IN_MODIFY) {
- /*
- * The file was modified, check how many new bytes do
- * we have.
- */
-
- /* Check if the file was truncated */
- if (file->offset > st.st_size) {
- offset = lseek(file->fd, 0, SEEK_SET);
- if (offset == -1) {
- flb_errno();
- return -1;
- }
-
- flb_plg_debug(ctx->ins, "inode=%"PRIu64" file truncated %s",
- file->inode, file->name);
- file->offset = offset;
- file->buf_len = 0;
-
- /* Update offset in the database file */
-#ifdef FLB_HAVE_SQLDB
- if (ctx->db) {
- flb_tail_db_file_offset(file, ctx);
- }
-#endif
- }
- }
-
- /* Collect the data */
- ret = in_tail_collect_event(file, config);
- if (ret != FLB_TAIL_ERROR) {
- /*
- * Due to read buffer size capacity, there are some cases where the
- * read operation cannot consume all new data available on one
- * round; upon successfull read(2) some data can still remain.
- *
- * If that is the case, we set in the structure how
- * many bytes are available 'now', so then the further
- * routine that check pending bytes and then the inotified-file
- * can process them properly after an internal signal.
- *
- * The goal to defer this routine is to avoid a blocking
- * read(2) operation, that might kill performance. Just let's
- * wait a second and do a good job.
- */
- tail_signal_pending(ctx);
- }
- else {
- return ret;
- }
-
- return 0;
-}
-
-static int in_tail_progress_check_callback(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- int ret = 0;
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_config *ctx = context;
- struct flb_tail_file *file;
- int pending_data_detected;
- struct stat st;
-
- (void) config;
-
- pending_data_detected = FLB_FALSE;
-
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
-
- if (file->offset < file->size) {
- pending_data_detected = FLB_TRUE;
-
- continue;
- }
-
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_errno();
- flb_plg_error(ins, "fstat error");
-
- continue;
- }
-
- if (file->offset < st.st_size) {
- file->size = st.st_size;
- file->pending_bytes = (file->size - file->offset);
-
- pending_data_detected = FLB_TRUE;
- }
- }
-
- if (pending_data_detected) {
- tail_signal_pending(ctx);
- }
-
- return 0;
-}
-
-/* File System events based on Inotify(2). Linux >= 2.6.32 is suggested */
-int flb_tail_fs_inotify_init(struct flb_input_instance *in,
- struct flb_tail_config *ctx, struct flb_config *config)
-{
- int fd;
- int ret;
-
- flb_plg_debug(ctx->ins, "flb_tail_fs_inotify_init() initializing inotify tail input");
-
- /* Create inotify instance */
- fd = inotify_init1(IN_NONBLOCK | IN_CLOEXEC);
- if (fd == -1) {
- flb_errno();
- return -1;
- }
- flb_plg_debug(ctx->ins, "inotify watch fd=%i", fd);
- ctx->fd_notify = fd;
-
- /* This backend use Fluent Bit event-loop to trigger notifications */
- ret = flb_input_set_collector_event(in, tail_fs_event,
- ctx->fd_notify, config);
- if (ret < 0) {
- close(fd);
- return -1;
- }
- ctx->coll_fd_fs1 = ret;
-
- /* Register callback to check current tail offsets */
- ret = flb_input_set_collector_time(in, in_tail_progress_check_callback,
- ctx->progress_check_interval,
- ctx->progress_check_interval_nsec,
- config);
- if (ret == -1) {
- flb_tail_config_destroy(ctx);
- return -1;
- }
- ctx->coll_fd_progress_check = ret;
-
- return 0;
-}
-
-void flb_tail_fs_inotify_pause(struct flb_tail_config *ctx)
-{
- flb_input_collector_pause(ctx->coll_fd_fs1, ctx->ins);
-}
-
-void flb_tail_fs_inotify_resume(struct flb_tail_config *ctx)
-{
- flb_input_collector_resume(ctx->coll_fd_fs1, ctx->ins);
-}
-
-int flb_tail_fs_inotify_add(struct flb_tail_file *file)
-{
- int ret;
- struct flb_tail_config *ctx = file->config;
-
- ret = tail_fs_add(file, FLB_TRUE);
- if (ret == -1) {
- flb_plg_error(ctx->ins, "inode=%"PRIu64" cannot register file %s",
- file->inode, file->name);
- return -1;
- }
-
- return 0;
-}
-
-int flb_tail_fs_inotify_remove(struct flb_tail_file *file)
-{
- struct flb_tail_config *ctx = file->config;
-
- if (file->watch_fd == -1) {
- return 0;
- }
-
- flb_plg_info(ctx->ins, "inotify_fs_remove(): inode=%"PRIu64" watch_fd=%i",
- file->inode, file->watch_fd);
-
- inotify_rm_watch(file->config->fd_notify, file->watch_fd);
- file->watch_fd = -1;
- return 0;
-}
-
-int flb_tail_fs_inotify_exit(struct flb_tail_config *ctx)
-{
- return close(ctx->fd_notify);
-}
diff --git a/fluent-bit/plugins/in_tail/tail_fs_inotify.h b/fluent-bit/plugins/in_tail/tail_fs_inotify.h
deleted file mode 100644
index 128ab0624..000000000
--- a/fluent-bit/plugins/in_tail/tail_fs_inotify.h
+++ /dev/null
@@ -1,37 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_FS_INOTIFY_H
-#define FLB_TAIL_FS_INOTIFY_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-
-#include "tail_config.h"
-#include "tail_file_internal.h"
-
-int flb_tail_fs_inotify_init(struct flb_input_instance *in,
- struct flb_tail_config *ctx, struct flb_config *config);
-int flb_tail_fs_inotify_add(struct flb_tail_file *file);
-int flb_tail_fs_inotify_remove(struct flb_tail_file *file);
-int flb_tail_fs_inotify_exit(struct flb_tail_config *ctx);
-void flb_tail_fs_inotify_pause(struct flb_tail_config *ctx);
-void flb_tail_fs_inotify_resume(struct flb_tail_config *ctx);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_fs_stat.c b/fluent-bit/plugins/in_tail/tail_fs_stat.c
deleted file mode 100644
index 6b312c9bd..000000000
--- a/fluent-bit/plugins/in_tail/tail_fs_stat.c
+++ /dev/null
@@ -1,253 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#define _DEFAULT_SOURCE
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-
-#include <sys/types.h>
-#include <sys/stat.h>
-
-#include "tail_file.h"
-#include "tail_db.h"
-#include "tail_config.h"
-#include "tail_signal.h"
-
-#ifdef FLB_SYSTEM_WINDOWS
-#include "win32.h"
-#endif
-
-struct fs_stat {
- /* last time check */
- time_t checked;
-
- /* previous status */
- struct stat st;
-};
-
-static int tail_fs_event(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- int ret;
- struct mk_list *head;
- struct mk_list *tmp;
- struct flb_tail_config *ctx = in_context;
- struct flb_tail_file *file = NULL;
- struct fs_stat *fst;
- struct stat st;
- time_t t;
-
- t = time(NULL);
-
- /* Lookup watched file */
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- fst = file->fs_backend;
-
- /* Check current status of the file */
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_errno();
- continue;
- }
-
- /* Check if the file was modified */
- if ((fst->st.st_mtime != st.st_mtime) ||
- (fst->st.st_size != st.st_size)) {
- /* Update stat info and trigger the notification */
- memcpy(&fst->st, &st, sizeof(struct stat));
- fst->checked = t;
- in_tail_collect_event(file, config);
- }
- }
-
- return 0;
-}
-
-static int tail_fs_check(struct flb_input_instance *ins,
- struct flb_config *config, void *in_context)
-{
- int ret;
- int64_t offset;
- char *name;
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_config *ctx = in_context;
- struct flb_tail_file *file = NULL;
- struct fs_stat *fst;
- struct stat st;
-
- /* Lookup watched file */
- mk_list_foreach_safe(head, tmp, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- fst = file->fs_backend;
-
- ret = fstat(file->fd, &st);
- if (ret == -1) {
- flb_plg_debug(ctx->ins, "error stat(2) %s, removing", file->name);
- flb_tail_file_remove(file);
- continue;
- }
-
- /* Check if the file have been deleted */
- if (st.st_nlink == 0) {
- flb_plg_debug(ctx->ins, "file has been deleted: %s", file->name);
-#ifdef FLB_HAVE_SQLDB
- if (ctx->db) {
- /* Remove file entry from the database */
- flb_tail_db_file_delete(file, ctx);
- }
-#endif
- flb_tail_file_remove(file);
- continue;
- }
-
- /* Check if the file was truncated */
- if (file->offset > st.st_size) {
- offset = lseek(file->fd, 0, SEEK_SET);
- if (offset == -1) {
- flb_errno();
- return -1;
- }
-
- flb_plg_debug(ctx->ins, "file truncated %s", file->name);
- file->offset = offset;
- file->buf_len = 0;
- memcpy(&fst->st, &st, sizeof(struct stat));
-
-#ifdef FLB_HAVE_SQLDB
- /* Update offset in database file */
- if (ctx->db) {
- flb_tail_db_file_offset(file, ctx);
- }
-#endif
- }
-
- if (file->offset < st.st_size) {
- file->pending_bytes = (st.st_size - file->offset);
- tail_signal_pending(ctx);
- }
- else {
- file->pending_bytes = 0;
- }
-
-
- /* Discover the current file name for the open file descriptor */
- name = flb_tail_file_name(file);
- if (!name) {
- flb_plg_debug(ctx->ins, "could not resolve %s, removing", file->name);
- flb_tail_file_remove(file);
- continue;
- }
-
- /*
- * Check if file still exists. This method requires explicity that the
- * user is using an absolute path, otherwise we will be rotating the
- * wrong file.
- *
- * flb_tail_target_file_name_cmp is a deeper compare than
- * flb_tail_file_name_cmp. If applicable, it compares to the underlying
- * real_name of the file.
- */
- if (flb_tail_file_is_rotated(ctx, file) == FLB_TRUE) {
- flb_tail_file_rotated(file);
- }
- flb_free(name);
-
- }
-
- return 0;
-}
-
-/* File System events based on stat(2) */
-int flb_tail_fs_stat_init(struct flb_input_instance *in,
- struct flb_tail_config *ctx, struct flb_config *config)
-{
- int ret;
-
- flb_plg_debug(ctx->ins, "flb_tail_fs_stat_init() initializing stat tail input");
-
- /* Set a manual timer to collect events every 0.250 seconds */
- ret = flb_input_set_collector_time(in, tail_fs_event,
- 0, 250000000, config);
- if (ret < 0) {
- return -1;
- }
- ctx->coll_fd_fs1 = ret;
-
- /* Set a manual timer to check deleted/rotated files every 2.5 seconds */
- ret = flb_input_set_collector_time(in, tail_fs_check,
- 2, 500000000, config);
- if (ret < 0) {
- return -1;
- }
- ctx->coll_fd_fs2 = ret;
-
- return 0;
-}
-
-void flb_tail_fs_stat_pause(struct flb_tail_config *ctx)
-{
- flb_input_collector_pause(ctx->coll_fd_fs1, ctx->ins);
- flb_input_collector_pause(ctx->coll_fd_fs2, ctx->ins);
-}
-
-void flb_tail_fs_stat_resume(struct flb_tail_config *ctx)
-{
- flb_input_collector_resume(ctx->coll_fd_fs1, ctx->ins);
- flb_input_collector_resume(ctx->coll_fd_fs2, ctx->ins);
-}
-
-int flb_tail_fs_stat_add(struct flb_tail_file *file)
-{
- int ret;
- struct fs_stat *fst;
-
- fst = flb_malloc(sizeof(struct fs_stat));
- if (!fst) {
- flb_errno();
- return -1;
- }
-
- fst->checked = time(NULL);
- ret = stat(file->name, &fst->st);
- if (ret == -1) {
- flb_errno();
- flb_free(fst);
- return -1;
- }
- file->fs_backend = fst;
-
- return 0;
-}
-
-int flb_tail_fs_stat_remove(struct flb_tail_file *file)
-{
- if (file->tail_mode == FLB_TAIL_EVENT) {
- flb_free(file->fs_backend);
- }
- return 0;
-}
-
-int flb_tail_fs_stat_exit(struct flb_tail_config *ctx)
-{
- (void) ctx;
- return 0;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_fs_stat.h b/fluent-bit/plugins/in_tail/tail_fs_stat.h
deleted file mode 100644
index 21a0704cb..000000000
--- a/fluent-bit/plugins/in_tail/tail_fs_stat.h
+++ /dev/null
@@ -1,37 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_FS_STAT_H
-#define FLB_TAIL_FS_STAT_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-
-#include "tail_config.h"
-#include "tail_file_internal.h"
-
-int flb_tail_fs_stat_init(struct flb_input_instance *in,
- struct flb_tail_config *ctx, struct flb_config *config);
-int flb_tail_fs_stat_add(struct flb_tail_file *file);
-int flb_tail_fs_stat_remove(struct flb_tail_file *file);
-int flb_tail_fs_stat_exit(struct flb_tail_config *ctx);
-void flb_tail_fs_stat_pause(struct flb_tail_config *ctx);
-void flb_tail_fs_stat_resume(struct flb_tail_config *ctx);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_multiline.c b/fluent-bit/plugins/in_tail/tail_multiline.c
deleted file mode 100644
index 71c031014..000000000
--- a/fluent-bit/plugins/in_tail/tail_multiline.c
+++ /dev/null
@@ -1,606 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_config.h>
-#include <fluent-bit/flb_kv.h>
-
-#include "tail_config.h"
-#include "tail_multiline.h"
-
-static int tail_mult_append(struct flb_parser *parser,
- struct flb_tail_config *ctx)
-{
- struct flb_tail_mult *mp;
-
- mp = flb_malloc(sizeof(struct flb_tail_mult));
- if (!mp) {
- flb_errno();
- return -1;
- }
-
- mp->parser = parser;
- mk_list_add(&mp->_head, &ctx->mult_parsers);
-
- return 0;
-}
-
-int flb_tail_mult_create(struct flb_tail_config *ctx,
- struct flb_input_instance *ins,
- struct flb_config *config)
-{
- int ret;
- const char *tmp;
- struct mk_list *head;
- struct flb_parser *parser;
- struct flb_kv *kv;
-
- if (ctx->multiline_flush <= 0) {
- ctx->multiline_flush = 1;
- }
-
- mk_list_init(&ctx->mult_parsers);
-
- /* Get firstline parser */
- tmp = flb_input_get_property("parser_firstline", ins);
- if (!tmp) {
- flb_plg_error(ctx->ins, "multiline: no parser defined for firstline");
- return -1;
- }
- parser = flb_parser_get(tmp, config);
- if (!parser) {
- flb_plg_error(ctx->ins, "multiline: invalid parser '%s'", tmp);
- return -1;
- }
-
- ctx->mult_parser_firstline = parser;
-
- /* Read all multiline rules */
- mk_list_foreach(head, &ins->properties) {
- kv = mk_list_entry(head, struct flb_kv, _head);
- if (strcasecmp("parser_firstline", kv->key) == 0) {
- continue;
- }
-
- if (strncasecmp("parser_", kv->key, 7) == 0) {
- parser = flb_parser_get(kv->val, config);
- if (!parser) {
- flb_plg_error(ctx->ins, "multiline: invalid parser '%s'", kv->val);
- return -1;
- }
-
- ret = tail_mult_append(parser, ctx);
- if (ret == -1) {
- return -1;
- }
- }
- }
-
- return 0;
-}
-
-int flb_tail_mult_destroy(struct flb_tail_config *ctx)
-{
- struct mk_list *tmp;
- struct mk_list *head;
- struct flb_tail_mult *mp;
-
- if (ctx->multiline == FLB_FALSE) {
- return 0;
- }
-
- mk_list_foreach_safe(head, tmp, &ctx->mult_parsers) {
- mp = mk_list_entry(head, struct flb_tail_mult, _head);
- mk_list_del(&mp->_head);
- flb_free(mp);
- }
-
- return 0;
-}
-
-/* Process the result of a firstline match */
-int flb_tail_mult_process_first(time_t now,
- char *buf, size_t size,
- struct flb_time *out_time,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx)
-{
- int ret;
- size_t off;
- msgpack_object map;
- msgpack_unpacked result;
-
- /* If a previous multiline context already exists, flush first */
- if (file->mult_firstline && !file->mult_skipping) {
- flb_tail_mult_flush(file, ctx);
- }
-
- /* Remark as first multiline message */
- file->mult_firstline = FLB_TRUE;
-
- /* Validate obtained time, if not set, set the current time */
- if (flb_time_to_nanosec(out_time) == 0L) {
- flb_time_get(out_time);
- }
-
- /* Should we skip this multiline record ? */
- if (ctx->ignore_older > 0) {
- if ((now - ctx->ignore_older) > out_time->tm.tv_sec) {
- flb_free(buf);
- file->mult_skipping = FLB_TRUE;
- file->mult_firstline = FLB_TRUE;
-
- /* we expect more data to skip */
- return FLB_TAIL_MULT_MORE;
- }
- }
-
- /* Re-initiate buffers */
- msgpack_sbuffer_init(&file->mult_sbuf);
- msgpack_packer_init(&file->mult_pck, &file->mult_sbuf, msgpack_sbuffer_write);
-
- /*
- * flb_parser_do() always return a msgpack buffer, so we tweak our
- * local msgpack reference to avoid an extra allocation. The only
- * concern is that we don't know what's the real size of the memory
- * allocated, so we assume it's just 'out_size'.
- */
- file->mult_flush_timeout = now + (ctx->multiline_flush - 1);
- file->mult_sbuf.data = buf;
- file->mult_sbuf.size = size;
- file->mult_sbuf.alloc = size;
-
- /* Set multiline status */
- file->mult_firstline = FLB_TRUE;
- file->mult_skipping = FLB_FALSE;
- flb_time_copy(&file->mult_time, out_time);
-
- off = 0;
- msgpack_unpacked_init(&result);
- ret = msgpack_unpack_next(&result, buf, size, &off);
- if (ret != MSGPACK_UNPACK_SUCCESS) {
- msgpack_sbuffer_destroy(&file->mult_sbuf);
- msgpack_unpacked_destroy(&result);
- return FLB_TAIL_MULT_NA;
- }
-
- map = result.data;
- file->mult_keys = map.via.map.size;
- msgpack_unpacked_destroy(&result);
-
- /* We expect more data */
- return FLB_TAIL_MULT_MORE;
-}
-
-/* Append a raw log entry to the last structured field in the mult buffer */
-static inline void flb_tail_mult_append_raw(char *buf, int size,
- struct flb_tail_file *file,
- struct flb_tail_config *config)
-{
- /* Append the raw string */
- msgpack_pack_str(&file->mult_pck, size);
- msgpack_pack_str_body(&file->mult_pck, buf, size);
-}
-
-/* Check if the last key value type of a map is string or not */
-static inline int is_last_key_val_string(char *buf, size_t size)
-{
- int ret = FLB_FALSE;
- size_t off;
- msgpack_unpacked result;
- msgpack_object v;
- msgpack_object root;
-
- off = 0;
- msgpack_unpacked_init(&result);
- ret = msgpack_unpack_next(&result, buf, size, &off);
- if (ret != MSGPACK_UNPACK_SUCCESS) {
- return ret;
- }
-
- root = result.data;
- if (root.type != MSGPACK_OBJECT_MAP) {
- ret = FLB_FALSE;
- }
- else {
- if (root.via.map.size == 0) {
- ret = FLB_FALSE;
- }
- else {
- v = root.via.map.ptr[root.via.map.size - 1].val;
- if (v.type == MSGPACK_OBJECT_STR) {
- ret = FLB_TRUE;
- }
- }
- }
-
- msgpack_unpacked_destroy(&result);
- return ret;
-}
-
-int flb_tail_mult_process_content(time_t now,
- char *buf, size_t len,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx,
- size_t processed_bytes)
-{
- int ret;
- size_t off;
- void *out_buf;
- size_t out_size = 0;
- struct mk_list *head;
- struct flb_tail_mult *mult_parser = NULL;
- struct flb_time out_time = {0};
- msgpack_object map;
- msgpack_unpacked result;
-
- /* Always check if this line is the beginning of a new multiline message */
- ret = flb_parser_do(ctx->mult_parser_firstline,
- buf, len,
- &out_buf, &out_size, &out_time);
- if (ret >= 0) {
- /*
- * The content is a candidate for a firstline, but we need to perform
- * the extra-mandatory check where the last key value type must be
- * a string, otherwise no string concatenation with continuation lines
- * will be possible.
- */
- ret = is_last_key_val_string(out_buf, out_size);
- if (ret == FLB_TRUE)
- file->mult_firstline_append = FLB_TRUE;
- else
- file->mult_firstline_append = FLB_FALSE;
-
- flb_tail_mult_process_first(now, out_buf, out_size, &out_time,
- file, ctx);
- return FLB_TAIL_MULT_MORE;
- }
-
- if (file->mult_skipping == FLB_TRUE) {
- return FLB_TAIL_MULT_MORE;
- }
-
- /*
- * Once here means we have some data that is a continuation, iterate
- * parsers trying to find a match
- */
- out_buf = NULL;
- mk_list_foreach(head, &ctx->mult_parsers) {
- mult_parser = mk_list_entry(head, struct flb_tail_mult, _head);
-
- /* Process line text with current parser */
- out_buf = NULL;
- out_size = 0;
- ret = flb_parser_do(mult_parser->parser,
- buf, len,
- &out_buf, &out_size, &out_time);
- if (ret < 0) {
- mult_parser = NULL;
- continue;
- }
-
- /* The line was processed, break the loop and buffer the data */
- break;
- }
-
- if (!mult_parser) {
- /*
- * If no parser was found means the string log must be appended
- * to the last structured field.
- */
- if (file->mult_firstline && file->mult_firstline_append) {
- flb_tail_mult_append_raw(buf, len, file, ctx);
- }
- else {
- flb_tail_file_pack_line(NULL, buf, len, file, processed_bytes);
- }
-
- return FLB_TAIL_MULT_MORE;
- }
-
- off = 0;
- msgpack_unpacked_init(&result);
- msgpack_unpack_next(&result, out_buf, out_size, &off);
- map = result.data;
-
- /* Append new map to our local msgpack buffer */
- file->mult_keys += map.via.map.size;
- msgpack_unpacked_destroy(&result);
- msgpack_sbuffer_write(&file->mult_sbuf, out_buf, out_size);
- flb_free(out_buf);
-
- return FLB_TAIL_MULT_MORE;
-}
-
-static int flb_tail_mult_pack_line_body(
- struct flb_log_event_encoder *context,
- struct flb_tail_file *file)
-{
- size_t adjacent_object_offset;
- size_t continuation_length;
- msgpack_unpacked adjacent_object;
- msgpack_unpacked current_object;
- size_t entry_index;
- msgpack_object entry_value;
- msgpack_object entry_key;
- msgpack_object_map *data_map;
- int map_size;
- size_t offset;
- struct flb_tail_config *config;
- int result;
-
- result = FLB_EVENT_ENCODER_SUCCESS;
- config = (struct flb_tail_config *) file->config;
-
- /* New Map size */
- map_size = file->mult_keys;
-
- if (file->config->path_key != NULL) {
- map_size++;
-
- result = flb_log_event_encoder_append_body_values(
- context,
- FLB_LOG_EVENT_CSTRING_VALUE(config->path_key),
- FLB_LOG_EVENT_CSTRING_VALUE(file->name));
- }
-
-
- msgpack_unpacked_init(&current_object);
- msgpack_unpacked_init(&adjacent_object);
-
- offset = 0;
-
- while (result == FLB_EVENT_ENCODER_SUCCESS &&
- msgpack_unpack_next(&current_object,
- file->mult_sbuf.data,
- file->mult_sbuf.size,
- &offset) == MSGPACK_UNPACK_SUCCESS) {
- if (current_object.data.type != MSGPACK_OBJECT_MAP) {
- continue;
- }
-
- data_map = &current_object.data.via.map;
-
- continuation_length = 0;
-
- for (entry_index = 0; entry_index < data_map->size; entry_index++) {
- entry_key = data_map->ptr[entry_index].key;
- entry_value = data_map->ptr[entry_index].val;
-
- result = flb_log_event_encoder_append_body_msgpack_object(context,
- &entry_key);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- break;
- }
-
- /* Check if this is the last entry in the map and if that is
- * the case then add the lengths of all the trailing string
- * objects after the map in order to append them to the value
- * but only if the value object is a string
- */
- if (entry_index + 1 == data_map->size &&
- entry_value.type == MSGPACK_OBJECT_STR) {
- adjacent_object_offset = offset;
-
- while (msgpack_unpack_next(
- &adjacent_object,
- file->mult_sbuf.data,
- file->mult_sbuf.size,
- &adjacent_object_offset) == MSGPACK_UNPACK_SUCCESS) {
- if (adjacent_object.data.type != MSGPACK_OBJECT_STR) {
- break;
- }
-
- /* Sum total bytes to append */
- continuation_length += adjacent_object.data.via.str.size + 1;
- }
-
- result = flb_log_event_encoder_append_body_string_length(
- context,
- entry_value.via.str.size +
- continuation_length);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- break;
- }
-
- result = flb_log_event_encoder_append_body_string_body(
- context,
- (char *) entry_value.via.str.ptr,
- entry_value.via.str.size);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- break;
- }
-
- if (continuation_length > 0) {
- adjacent_object_offset = offset;
-
- while (msgpack_unpack_next(
- &adjacent_object,
- file->mult_sbuf.data,
- file->mult_sbuf.size,
- &adjacent_object_offset) == MSGPACK_UNPACK_SUCCESS) {
- if (adjacent_object.data.type != MSGPACK_OBJECT_STR) {
- break;
- }
-
- result = flb_log_event_encoder_append_body_string_body(
- context,
- "\n",
- 1);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- break;
- }
-
- result = flb_log_event_encoder_append_body_string_body(
- context,
- (char *) adjacent_object.data.via.str.ptr,
- adjacent_object.data.via.str.size);
-
- if (result != FLB_EVENT_ENCODER_SUCCESS) {
- break;
- }
- }
- }
- }
- else {
- result = flb_log_event_encoder_append_body_msgpack_object(context,
- &entry_value);
- }
- }
- }
-
- msgpack_unpacked_destroy(&current_object);
- msgpack_unpacked_destroy(&adjacent_object);
-
- /* Reset status */
- file->mult_firstline = FLB_FALSE;
- file->mult_skipping = FLB_FALSE;
- file->mult_keys = 0;
- file->mult_flush_timeout = 0;
-
- msgpack_sbuffer_destroy(&file->mult_sbuf);
-
- file->mult_sbuf.data = NULL;
-
- flb_time_zero(&file->mult_time);
-
- return result;
-}
-
-/* Flush any multiline context data into outgoing buffers */
-int flb_tail_mult_flush(struct flb_tail_file *file, struct flb_tail_config *ctx)
-{
- int result;
-
- /* nothing to flush */
- if (file->mult_firstline == FLB_FALSE) {
- return -1;
- }
-
- if (file->mult_keys == 0) {
- return -1;
- }
-
- result = flb_log_event_encoder_begin_record(file->ml_log_event_encoder);
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_set_timestamp(
- file->ml_log_event_encoder, &file->mult_time);
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_tail_mult_pack_line_body(
- file->ml_log_event_encoder,
- file);
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- result = flb_log_event_encoder_commit_record(
- file->ml_log_event_encoder);
- }
-
- if (result == FLB_EVENT_ENCODER_SUCCESS) {
- flb_input_log_append(ctx->ins,
- file->tag_buf,
- file->tag_len,
- file->ml_log_event_encoder->output_buffer,
- file->ml_log_event_encoder->output_length);
- result = 0;
- }
- else {
- flb_plg_error(file->config->ins, "error packing event : %d", result);
-
- result = -1;
- }
-
- flb_log_event_encoder_reset(file->ml_log_event_encoder);
-
- return result;
-}
-
-static void file_pending_flush(struct flb_tail_config *ctx,
- struct flb_tail_file *file, time_t now)
-{
- if (file->mult_flush_timeout > now) {
- return;
- }
-
- if (file->mult_firstline == FLB_FALSE) {
- if (file->mult_sbuf.data == NULL || file->mult_sbuf.size <= 0) {
- return;
- }
- }
-
- flb_tail_mult_flush(file, ctx);
-}
-
-int flb_tail_mult_pending_flush_all(struct flb_tail_config *ctx)
-{
- time_t expired;
- struct mk_list *head;
- struct flb_tail_file *file;
-
- expired = time(NULL) + 3600;
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, expired);
- }
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
- file_pending_flush(ctx, file, expired);
- }
-
- return 0;
-}
-
-int flb_tail_mult_pending_flush(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- time_t now;
- struct mk_list *head;
- struct flb_tail_file *file;
- struct flb_tail_config *ctx = context;
-
- now = time(NULL);
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_static) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
-
- file_pending_flush(ctx, file, now);
- }
-
- /* Iterate promoted event files with pending bytes */
- mk_list_foreach(head, &ctx->files_event) {
- file = mk_list_entry(head, struct flb_tail_file, _head);
-
- file_pending_flush(ctx, file, now);
- }
-
- return 0;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_multiline.h b/fluent-bit/plugins/in_tail/tail_multiline.h
deleted file mode 100644
index d7f7539b1..000000000
--- a/fluent-bit/plugins/in_tail/tail_multiline.h
+++ /dev/null
@@ -1,57 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_TAIL_MULT_H
-#define FLB_TAIL_TAIL_MULT_H
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_input.h>
-
-#include "tail_config.h"
-#include "tail_file.h"
-
-#define FLB_TAIL_MULT_NA -1 /* not applicable as a multiline stream */
-#define FLB_TAIL_MULT_DONE 0 /* finished a multiline stream */
-#define FLB_TAIL_MULT_MORE 1 /* expect more lines to come */
-#define FLB_TAIL_MULT_FLUSH "4" /* max flush time for multiline: 4 seconds */
-
-struct flb_tail_mult {
- struct flb_parser *parser;
- struct mk_list _head;
-};
-
-int flb_tail_mult_create(struct flb_tail_config *ctx,
- struct flb_input_instance *ins,
- struct flb_config *config);
-
-int flb_tail_mult_destroy(struct flb_tail_config *ctx);
-
-int flb_tail_mult_process_content(time_t now,
- char *buf, size_t len,
- struct flb_tail_file *file,
- struct flb_tail_config *ctx,
- size_t processed_bytes);
-int flb_tail_mult_flush(struct flb_tail_file *file,
- struct flb_tail_config *ctx);
-
-int flb_tail_mult_pending_flush(struct flb_input_instance *ins,
- struct flb_config *config, void *context);
-int flb_tail_mult_pending_flush_all(struct flb_tail_config *ctx);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_scan.c b/fluent-bit/plugins/in_tail/tail_scan.c
deleted file mode 100644
index ccb8e070a..000000000
--- a/fluent-bit/plugins/in_tail/tail_scan.c
+++ /dev/null
@@ -1,71 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <fluent-bit/flb_input_plugin.h>
-#include "tail.h"
-#include "tail_config.h"
-
-/*
- * Include proper scan backend
- */
-#ifdef FLB_SYSTEM_WINDOWS
-#include "tail_scan_win32.c"
-#else
-#include "tail_scan_glob.c"
-#endif
-
-int flb_tail_scan(struct mk_list *path_list, struct flb_tail_config *ctx)
-{
- int ret;
- struct mk_list *head;
- struct flb_slist_entry *pattern;
-
- mk_list_foreach(head, path_list) {
- pattern = mk_list_entry(head, struct flb_slist_entry, _head);
- ret = tail_scan_path(pattern->str, ctx);
- if (ret == -1) {
- flb_plg_warn(ctx->ins, "error scanning path: %s", pattern->str);
- }
- else {
- flb_plg_debug(ctx->ins, "%i new files found on path '%s'",
- ret, pattern->str);
- }
- }
-
- return 0;
-}
-
-/*
- * Triggered by refresh_interval, it re-scan the path looking for new files
- * that match the original path pattern.
- */
-int flb_tail_scan_callback(struct flb_input_instance *ins,
- struct flb_config *config, void *context)
-{
- int ret;
- struct flb_tail_config *ctx = context;
- (void) config;
-
- ret = flb_tail_scan(ctx->path_list, ctx);
- if (ret > 0) {
- flb_plg_debug(ins, "%i new files found", ret);
- }
-
- return ret;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_scan.h b/fluent-bit/plugins/in_tail/tail_scan.h
deleted file mode 100644
index ec3c96a2a..000000000
--- a/fluent-bit/plugins/in_tail/tail_scan.h
+++ /dev/null
@@ -1,29 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_SCAN_H
-#define FLB_TAIL_SCAN_H
-
-#include "tail_config.h"
-
-int flb_tail_scan(struct mk_list *path, struct flb_tail_config *ctx);
-int flb_tail_scan_callback(struct flb_input_instance *ins,
- struct flb_config *config, void *context);
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_scan_glob.c b/fluent-bit/plugins/in_tail/tail_scan_glob.c
deleted file mode 100644
index b330b7c3b..000000000
--- a/fluent-bit/plugins/in_tail/tail_scan_glob.c
+++ /dev/null
@@ -1,278 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <sys/types.h>
-#include <sys/stat.h>
-#include <glob.h>
-#include <fnmatch.h>
-
-#include <fluent-bit/flb_compat.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_utils.h>
-
-#include "tail.h"
-#include "tail_file.h"
-#include "tail_signal.h"
-#include "tail_scan.h"
-#include "tail_config.h"
-
-/* Define missing GLOB_TILDE if not exists */
-#ifndef GLOB_TILDE
-#define GLOB_TILDE 1<<2 /* use GNU Libc value */
-#define UNSUP_TILDE 1
-
-/* we need these extra headers for path resolution */
-#include <limits.h>
-#include <sys/types.h>
-#include <pwd.h>
-
-static char *expand_tilde(const char *path)
-{
- int len;
- char user[256];
- char *p = NULL;
- char *dir = NULL;
- char *tmp = NULL;
- struct passwd *uinfo = NULL;
-
- if (path[0] == '~') {
- p = strchr(path, '/');
-
- if (p) {
- /* check case '~/' */
- if ((p - path) == 1) {
- dir = getenv("HOME");
- if (!dir) {
- return path;
- }
- }
- else {
- /*
- * it refers to a different user: ~user/abc, first step grab
- * the user name.
- */
- len = (p - path) - 1;
- memcpy(user, path + 1, len);
- user[len] = '\0';
-
- /* use getpwnam() to resolve user information */
- uinfo = getpwnam(user);
- if (!uinfo) {
- return path;
- }
-
- dir = uinfo->pw_dir;
- }
- }
- else {
- dir = getenv("HOME");
- if (!dir) {
- return path;
- }
- }
-
- if (p) {
- tmp = flb_malloc(PATH_MAX);
- if (!tmp) {
- flb_errno();
- return NULL;
- }
- snprintf(tmp, PATH_MAX - 1, "%s%s", dir, p);
- }
- else {
- dir = getenv("HOME");
- if (!dir) {
- return path;
- }
-
- tmp = flb_strdup(dir);
- if (!tmp) {
- return path;
- }
- }
-
- return tmp;
- }
-
- return path;
-}
-#endif
-
-static int tail_is_excluded(char *path, struct flb_tail_config *ctx)
-{
- struct mk_list *head;
- struct flb_slist_entry *pattern;
-
- if (!ctx->exclude_list) {
- return FLB_FALSE;
- }
-
- mk_list_foreach(head, ctx->exclude_list) {
- pattern = mk_list_entry(head, struct flb_slist_entry, _head);
- if (fnmatch(pattern->str, path, 0) == 0) {
- return FLB_TRUE;
- }
- }
-
- return FLB_FALSE;
-}
-
-static inline int do_glob(const char *pattern, int flags,
- void *not_used, glob_t *pglob)
-{
- int ret;
- int new_flags;
- char *tmp = NULL;
- int tmp_needs_free = FLB_FALSE;
- (void) not_used;
-
- /* Save current values */
- new_flags = flags;
-
- if (flags & GLOB_TILDE) {
-#ifdef UNSUP_TILDE
- /*
- * Some libc libraries like Musl do not support GLOB_TILDE for tilde
- * expansion. A workaround is to use wordexp(3) but looking at it
- * implementation in Musl it looks quite expensive:
- *
- * http://git.musl-libc.org/cgit/musl/tree/src/misc/wordexp.c
- *
- * the workaround is to do our own tilde expansion in a temporary buffer.
- */
-
- /* Look for a tilde */
- tmp = expand_tilde(pattern);
- if (tmp != pattern) {
- /* the path was expanded */
- pattern = tmp;
- tmp_needs_free = FLB_TRUE;
- }
-
- /* remove unused flag */
- new_flags &= ~GLOB_TILDE;
-#endif
- }
-
- /* invoke glob with new parameters */
- ret = glob(pattern, new_flags, NULL, pglob);
-
- /* remove temporary buffer, if allocated by expand_tilde above.
- * Note that this buffer is only used for libc implementations
- * that do not support the GLOB_TILDE flag, like musl. */
- if ((tmp != NULL) && (tmp_needs_free == FLB_TRUE)) {
- flb_free(tmp);
- }
-
- return ret;
-}
-
-/* Scan a path, register the entries and return how many */
-static int tail_scan_path(const char *path, struct flb_tail_config *ctx)
-{
- int i;
- int ret;
- int count = 0;
- glob_t globbuf;
- time_t now;
- int64_t mtime;
- struct stat st;
-
- flb_plg_debug(ctx->ins, "scanning path %s", path);
-
- /* Safe reset for globfree() */
- globbuf.gl_pathv = NULL;
-
- /* Scan the given path */
- ret = do_glob(path, GLOB_TILDE | GLOB_ERR, NULL, &globbuf);
- if (ret != 0) {
- switch (ret) {
- case GLOB_NOSPACE:
- flb_plg_error(ctx->ins, "no memory space available");
- return -1;
- case GLOB_ABORTED:
- flb_plg_error(ctx->ins, "read error, check permissions: %s", path);
- return -1;
- case GLOB_NOMATCH:
- ret = stat(path, &st);
- if (ret == -1) {
- flb_plg_debug(ctx->ins, "cannot read info from: %s", path);
- }
- else {
- ret = access(path, R_OK);
- if (ret == -1 && errno == EACCES) {
- flb_plg_error(ctx->ins, "NO read access for path: %s", path);
- }
- else {
- flb_plg_debug(ctx->ins, "NO matches for path: %s", path);
- }
- }
- return 0;
- }
- }
-
-
- /* For every entry found, generate an output list */
- now = time(NULL);
- for (i = 0; i < globbuf.gl_pathc; i++) {
- ret = stat(globbuf.gl_pathv[i], &st);
- if (ret == 0 && S_ISREG(st.st_mode)) {
- /* Check if this file is blacklisted */
- if (tail_is_excluded(globbuf.gl_pathv[i], ctx) == FLB_TRUE) {
- flb_plg_debug(ctx->ins, "excluded=%s", globbuf.gl_pathv[i]);
- continue;
- }
-
- if (ctx->ignore_older > 0) {
- mtime = flb_tail_stat_mtime(&st);
- if (mtime > 0) {
- if ((now - ctx->ignore_older) > mtime) {
- flb_plg_debug(ctx->ins, "excluded=%s (ignore_older)",
- globbuf.gl_pathv[i]);
- continue;
- }
- }
- }
-
- /* Append file to list */
- ret = flb_tail_file_append(globbuf.gl_pathv[i], &st,
- FLB_TAIL_STATIC, ctx);
- if (ret == 0) {
- flb_plg_debug(ctx->ins, "scan_glob add(): %s, inode %li",
- globbuf.gl_pathv[i], st.st_ino);
- count++;
- }
- else {
- flb_plg_debug(ctx->ins, "scan_blog add(): dismissed: %s, inode %li",
- globbuf.gl_pathv[i], st.st_ino);
- }
- }
- else {
- flb_plg_debug(ctx->ins, "skip (invalid) entry=%s",
- globbuf.gl_pathv[i]);
- }
- }
-
- if (count > 0) {
- tail_signal_manager(ctx);
- }
-
- globfree(&globbuf);
- return count;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_scan_win32.c b/fluent-bit/plugins/in_tail/tail_scan_win32.c
deleted file mode 100644
index 94733f065..000000000
--- a/fluent-bit/plugins/in_tail/tail_scan_win32.c
+++ /dev/null
@@ -1,245 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-/*
- * This file implements glob-like patch matching feature for Windows
- * based on Win32 API.
- */
-
-#include <fluent-bit/flb_info.h>
-#include <fluent-bit/flb_compat.h>
-#include <fluent-bit/flb_input_plugin.h>
-#include <fluent-bit/flb_utils.h>
-
-#include <shlwapi.h>
-
-#include "tail.h"
-#include "tail_file.h"
-#include "tail_signal.h"
-#include "tail_config.h"
-
-#include "win32.h"
-
-static int tail_is_excluded(char *path, struct flb_tail_config *ctx)
-{
- struct mk_list *head;
- struct flb_slist_entry *pattern;
-
- if (!ctx->exclude_list) {
- return FLB_FALSE;
- }
-
- mk_list_foreach(head, ctx->exclude_list) {
- pattern = mk_list_entry(head, struct flb_slist_entry, _head);
- if (PathMatchSpecA(path, pattern->str)) {
- return FLB_TRUE;
- }
- }
-
- return FLB_FALSE;
-}
-
-/*
- * This function is a thin wrapper over flb_tail_file_append(),
- * adding normalization and sanity checks on top of it.
- */
-static int tail_register_file(const char *target, struct flb_tail_config *ctx,
- time_t ts)
-{
- int64_t mtime;
- struct stat st;
- char path[MAX_PATH];
-
- if (_fullpath(path, target, MAX_PATH) == NULL) {
- flb_plg_error(ctx->ins, "cannot get absolute path of %s", target);
- return -1;
- }
-
- if (stat(path, &st) != 0 || !S_ISREG(st.st_mode)) {
- return -1;
- }
-
- if (ctx->ignore_older > 0) {
- mtime = flb_tail_stat_mtime(&st);
- if (mtime > 0) {
- if ((ts - ctx->ignore_older) > mtime) {
- flb_plg_debug(ctx->ins, "excluded=%s (ignore_older)",
- target);
- return -1;
- }
- }
- }
-
- if (tail_is_excluded(path, ctx) == FLB_TRUE) {
- flb_plg_trace(ctx->ins, "skip '%s' (excluded)", path);
- return -1;
- }
-
- return flb_tail_file_append(path, &st, FLB_TAIL_STATIC, ctx);
-}
-
-/*
- * Perform patern match on the given path string. This function
- * supports patterns with "nested" wildcards like below.
- *
- * tail_scan_pattern("C:\fluent-bit\*\*.txt", ctx);
- *
- * On success, the number of files found is returned (zero indicates
- * "no file found"). On error, -1 is returned.
- */
-static int tail_scan_pattern(const char *path, struct flb_tail_config *ctx)
-{
- char *star, *p0, *p1;
- char pattern[MAX_PATH];
- char buf[MAX_PATH];
- int ret;
- int n_added = 0;
- time_t now;
- int64_t mtime;
- HANDLE h;
- WIN32_FIND_DATA data;
-
- if (strlen(path) > MAX_PATH - 1) {
- flb_plg_error(ctx->ins, "path too long '%s'");
- return -1;
- }
-
- star = strchr(path, '*');
- if (star == NULL) {
- return -1;
- }
-
- /*
- * C:\data\tmp\input_*.conf
- * 0<-----|
- */
- p0 = star;
- while (path <= p0 && *p0 != '\\') {
- p0--;
- }
-
- /*
- * C:\data\tmp\input_*.conf
- * |---->1
- */
- p1 = star;
- while (*p1 && *p1 != '\\') {
- p1++;
- }
-
- memcpy(pattern, path, (p1 - path));
- pattern[p1 - path] = '\0';
-
- h = FindFirstFileA(pattern, &data);
- if (h == INVALID_HANDLE_VALUE) {
- return 0; /* none matched */
- }
-
- now = time(NULL);
- do {
- /* Ignore the current and parent dirs */
- if (!strcmp(".", data.cFileName) || !strcmp("..", data.cFileName)) {
- continue;
- }
-
- /* Avoid an infinite loop */
- if (strchr(data.cFileName, '*')) {
- continue;
- }
-
- /* Create a path (prefix + filename + suffix) */
- memcpy(buf, path, p0 - path + 1);
- buf[p0 - path + 1] = '\0';
-
- if (strlen(buf) + strlen(data.cFileName) + strlen(p1) > MAX_PATH - 1) {
- flb_plg_warn(ctx->ins, "'%s%s%s' is too long", buf, data.cFileName, p1);
- continue;
- }
- strcat(buf, data.cFileName);
- strcat(buf, p1);
-
- if (strchr(p1, '*')) {
- ret = tail_scan_pattern(buf, ctx); /* recursive */
- if (ret >= 0) {
- n_added += ret;
- }
- continue;
- }
-
- /* Try to register the target file */
- ret = tail_register_file(buf, ctx, now);
- if (ret == 0) {
- n_added++;
- }
- } while (FindNextFileA(h, &data) != 0);
-
- FindClose(h);
- return n_added;
-}
-
-static int tail_filepath(char *buf, int len, const char *basedir, const char *filename)
-{
- char drive[_MAX_DRIVE];
- char dir[_MAX_DIR];
- char fname[_MAX_FNAME];
- char ext[_MAX_EXT];
- char tmp[MAX_PATH];
- int ret;
-
- ret = _splitpath_s(basedir, drive, _MAX_DRIVE, dir, _MAX_DIR, NULL, 0, NULL, 0);
- if (ret) {
- return -1;
- }
-
- ret = _splitpath_s(filename, NULL, 0, NULL, 0, fname, _MAX_FNAME, ext, _MAX_EXT);
- if (ret) {
- return -1;
- }
-
- ret = _makepath_s(tmp, MAX_PATH, drive, dir, fname, ext);
- if (ret) {
- return -1;
- }
-
- if (_fullpath(buf, tmp, len) == NULL) {
- return -1;
- }
-
- return 0;
-}
-
-static int tail_scan_path(const char *path, struct flb_tail_config *ctx)
-{
- int ret;
- int n_added = 0;
- time_t now;
-
- if (strchr(path, '*')) {
- return tail_scan_pattern(path, ctx);
- }
-
- /* No wildcard involved. Let's just handle the file... */
- now = time(NULL);
- ret = tail_register_file(path, ctx, now);
- if (ret == 0) {
- n_added++;
- }
-
- return n_added;
-}
diff --git a/fluent-bit/plugins/in_tail/tail_signal.h b/fluent-bit/plugins/in_tail/tail_signal.h
deleted file mode 100644
index 1a81fec64..000000000
--- a/fluent-bit/plugins/in_tail/tail_signal.h
+++ /dev/null
@@ -1,98 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_SIGNAL_H
-#define FLB_TAIL_SIGNAL_H
-
-#include "tail_config.h"
-
-static inline int tail_signal_manager(struct flb_tail_config *ctx)
-{
- int n;
- uint64_t val = 0xc001;
-
- /*
- * The number of signal reads might be less than the written signals, this
- * means that some event is still pending in the queue. On that case we
- * don't need to signal it again.
- */
- if (ctx->ch_reads < ctx->ch_writes) {
- return 1;
- }
-
- /* Reset counters: prevent an overflow, unlikely..but let's keep safe */
- if (ctx->ch_reads == ctx->ch_writes) {
- ctx->ch_reads = 0;
- ctx->ch_writes = 0;
- }
-
- /* Insert a dummy event into the channel manager */
- n = flb_pipe_w(ctx->ch_manager[1], (const char *) &val, sizeof(val));
- if (n == -1) {
- flb_errno();
- return -1;
- }
- else {
- ctx->ch_writes++;
- }
-
- return n;
-}
-
-static inline int tail_signal_pending(struct flb_tail_config *ctx)
-{
- int n;
- uint64_t val = 0xc002;
-
- /* Insert a dummy event into the 'pending' channel */
- n = flb_pipe_w(ctx->ch_pending[1], (const char *) &val, sizeof(val));
-
- /*
- * If we get EAGAIN, it simply means pending channel is full. As
- * notification is already pending, it's safe to ignore.
- */
- if (n == -1 && !FLB_PIPE_WOULDBLOCK()) {
- flb_errno();
- return -1;
- }
-
- return n;
-}
-
-static inline int tail_consume_pending(struct flb_tail_config *ctx)
-{
- int ret;
- uint64_t val;
-
- /*
- * We need to consume the pending bytes. Loop until we would have
- * blocked (pipe is empty).
- */
- do {
- ret = flb_pipe_r(ctx->ch_pending[0], (char *) &val, sizeof(val));
- if (ret <= 0 && !FLB_PIPE_WOULDBLOCK()) {
- flb_errno();
- return -1;
- }
- } while (!FLB_PIPE_WOULDBLOCK());
-
- return 0;
-}
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/tail_sql.h b/fluent-bit/plugins/in_tail/tail_sql.h
deleted file mode 100644
index 855933a01..000000000
--- a/fluent-bit/plugins/in_tail/tail_sql.h
+++ /dev/null
@@ -1,65 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_SQL_H
-#define FLB_TAIL_SQL_H
-
-/*
- * In Fluent Bit we try to have a common convention for table names,
- * if the table belong to an input/output plugin, use plugin name
- * plus to what it's about, e.g:
- *
- * in_tail plugin table to track files: in_tail_files
- */
-#define SQL_CREATE_FILES \
- "CREATE TABLE IF NOT EXISTS in_tail_files (" \
- " id INTEGER PRIMARY KEY," \
- " name TEXT NOT NULL," \
- " offset INTEGER," \
- " inode INTEGER," \
- " created INTEGER," \
- " rotated INTEGER DEFAULT 0" \
- ");"
-
-#define SQL_GET_FILE \
- "SELECT * from in_tail_files WHERE inode=@inode order by id desc;"
-
-#define SQL_INSERT_FILE \
- "INSERT INTO in_tail_files (name, offset, inode, created)" \
- " VALUES (@name, @offset, @inode, @created);"
-
-#define SQL_ROTATE_FILE \
- "UPDATE in_tail_files set name=@name,rotated=1 WHERE id=@id;"
-
-#define SQL_UPDATE_OFFSET \
- "UPDATE in_tail_files set offset=@offset WHERE id=@id;"
-
-#define SQL_DELETE_FILE \
- "DELETE FROM in_tail_files WHERE id=@id;"
-
-#define SQL_PRAGMA_SYNC \
- "PRAGMA synchronous=%i;"
-
-#define SQL_PRAGMA_JOURNAL_MODE \
- "PRAGMA journal_mode=%s;"
-
-#define SQL_PRAGMA_LOCKING_MODE \
- "PRAGMA locking_mode=EXCLUSIVE;"
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/win32.h b/fluent-bit/plugins/in_tail/win32.h
deleted file mode 100644
index a9414f892..000000000
--- a/fluent-bit/plugins/in_tail/win32.h
+++ /dev/null
@@ -1,67 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-/*
- * This is the interface file that replaces POSIX functions
- * with our own custom implementation.
- */
-
-#ifndef FLB_TAIL_WIN32_H
-#define FLB_TAIL_WIN32_H
-
-#include "win32/interface.h"
-
-#undef open
-#undef stat
-#undef lstat
-#undef fstat
-#undef lseek
-
-#undef S_IFDIR
-#undef S_IFCHR
-#undef S_IFIFO
-#undef S_IFREG
-#undef S_IFLNK
-#undef S_IFMT
-#undef S_ISDIR
-#undef S_ISCHR
-#undef S_ISFIFO
-#undef S_ISREG
-#undef S_ISLNK
-
-#define open win32_open
-#define stat win32_stat
-#define lstat win32_lstat
-#define fstat win32_fstat
-
-#define lseek _lseeki64
-
-#define S_IFDIR WIN32_S_IFDIR
-#define S_IFCHR WIN32_S_IFCHR
-#define S_IFIFO WIN32_S_IFIFO
-#define S_IFREG WIN32_S_IFREG
-#define S_IFLNK WIN32_S_IFLNK
-#define S_IFMT WIN32_S_IFMT
-
-#define S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR)
-#define S_ISCHR(m) (((m) & S_IFMT) == S_IFCHR)
-#define S_ISIFO(m) (((m) & S_IFMT) == S_IFIFO)
-#define S_ISREG(m) (((m) & S_IFMT) == S_IFREG)
-#define S_ISLNK(m) (((m) & S_IFMT) == S_IFLNK)
-#endif
diff --git a/fluent-bit/plugins/in_tail/win32/interface.h b/fluent-bit/plugins/in_tail/win32/interface.h
deleted file mode 100644
index 73b2ef233..000000000
--- a/fluent-bit/plugins/in_tail/win32/interface.h
+++ /dev/null
@@ -1,44 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#ifndef FLB_TAIL_WIN32_INTERFACE_H
-#define FLB_TAIL_WIN32_INTERFACE_H
-
-struct win32_stat {
- uint64_t st_ino;
- uint16_t st_mode;
- int64_t st_mtime;
- int16_t st_nlink;
- int64_t st_size;
-};
-
-int win32_stat(const char *path, struct win32_stat *wst);
-int win32_lstat(const char *path, struct win32_stat *wst);
-int win32_fstat(int fd, struct win32_stat *wst);
-
-int win32_open(const char *path, int flags);
-
-#define WIN32_S_IFDIR 0x1000
-#define WIN32_S_IFCHR 0x2000
-#define WIN32_S_IFIFO 0x4000
-#define WIN32_S_IFREG 0x8000
-#define WIN32_S_IFLNK 0xc000
-#define WIN32_S_IFMT 0xf000
-
-#endif
diff --git a/fluent-bit/plugins/in_tail/win32/io.c b/fluent-bit/plugins/in_tail/win32/io.c
deleted file mode 100644
index 45928b04a..000000000
--- a/fluent-bit/plugins/in_tail/win32/io.c
+++ /dev/null
@@ -1,47 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <Windows.h>
-#include <stdlib.h>
-#include <stdint.h>
-#include <fcntl.h>
-#include <io.h>
-#include "interface.h"
-
-/*
- * POSIX IO emulation tailored for in_tail's usage.
- *
- * open(2) that does not acquire an exclusive lock.
- */
-
-int win32_open(const char *path, int flags)
-{
- HANDLE h;
- h = CreateFileA(path,
- GENERIC_READ,
- FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
- NULL, /* lpSecurityAttributes */
- OPEN_EXISTING, /* dwCreationDisposition */
- 0, /* dwFlagsAndAttributes */
- NULL); /* hTemplateFile */
- if (h == INVALID_HANDLE_VALUE) {
- return -1;
- }
- return _open_osfhandle((intptr_t) h, _O_RDONLY);
-}
diff --git a/fluent-bit/plugins/in_tail/win32/stat.c b/fluent-bit/plugins/in_tail/win32/stat.c
deleted file mode 100644
index bce802749..000000000
--- a/fluent-bit/plugins/in_tail/win32/stat.c
+++ /dev/null
@@ -1,332 +0,0 @@
-/* -*- Mode: C; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-
-/* Fluent Bit
- * ==========
- * Copyright (C) 2015-2022 The Fluent Bit Authors
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-#include <Windows.h>
-#include <stdlib.h>
-#include <stdint.h>
-#include <io.h>
-#include "interface.h"
-
-/*
- * NTFS stat(2) emulation tailored for in_tail's usage.
- *
- * (1) Support st_ino (inode) for Windows NTFS.
- * (2) Support NTFS symlinks.
- * (3) Support large files >= 2GB.
- *
- * To use it, include "win32.h" and it will transparently
- * replace stat(), lstat() and fstat().
- */
-
-#define UINT64(high, low) ((uint64_t) (high) << 32 | (low))
-#define WINDOWS_TICKS_TO_SECONDS_RATIO 10000000
-#define WINDOWS_EPOCH_TO_UNIX_EPOCH_DELTA 11644473600
-
-/*
- * FILETIME timestamps are represented in 100-nanosecond intervals,
- * because of this, that's why we need to divide the number by 10000000
- * in order to convert it to seconds.
- *
- * While UNIX timestamps use January 1, 1970 as epoch Windows FILETIME
- * timestamps use January 1, 1601. Because of this we need to subtract
- * 11644473600 seconds to account for it.
- *
- * Note: Even though this does not account for leap seconds it should be
- * accurate enough.
- */
-
-static uint64_t filetime_to_epoch(FILETIME *ft)
-{
- ULARGE_INTEGER timestamp;
-
- if (ft == NULL) {
- return 0;
- }
-
- timestamp.HighPart = ft->dwHighDateTime;
- timestamp.LowPart = ft->dwLowDateTime;
-
- timestamp.QuadPart /= WINDOWS_TICKS_TO_SECONDS_RATIO;
- timestamp.QuadPart -= WINDOWS_EPOCH_TO_UNIX_EPOCH_DELTA;
-
- return timestamp.QuadPart;
-}
-
-static void reset_errno()
-{
- errno = 0;
-}
-
-static void propagate_last_error_to_errno()
-{
- DWORD error_code;
-
- error_code = GetLastError();
-
- switch (error_code) {
- case ERROR_INVALID_TARGET_HANDLE:
- case ERROR_INVALID_HANDLE:
- errno = EBADF;
- break;
-
- case ERROR_TOO_MANY_OPEN_FILES:
- errno = EMFILE;
- break;
-
- case ERROR_INVALID_FLAG_NUMBER:
- case ERROR_INVALID_PARAMETER:
- errno = EINVAL;
- break;
-
- case ERROR_NOT_ENOUGH_MEMORY:
- case ERROR_OUTOFMEMORY:
- errno = ENOMEM;
- break;
-
- case ERROR_SHARING_VIOLATION:
- case ERROR_LOCK_VIOLATION:
- case ERROR_PATH_BUSY:
- case ERROR_BUSY:
- errno = EBUSY;
- break;
-
- case ERROR_HANDLE_DISK_FULL:
- case ERROR_DISK_FULL:
- errno = ENOSPC;
- break;
-
- case ERROR_INVALID_ADDRESS:
- errno = EFAULT;
- break;
-
- case ERROR_FILE_TOO_LARGE:
- errno = EFBIG;
- break;
-
- case ERROR_ALREADY_EXISTS:
- case ERROR_FILE_EXISTS:
- errno = EEXIST;
- break;
-
- case ERROR_FILE_NOT_FOUND:
- case ERROR_PATH_NOT_FOUND:
- case ERROR_INVALID_DRIVE:
- case ERROR_BAD_PATHNAME:
- case ERROR_INVALID_NAME:
- case ERROR_BAD_UNIT:
- errno = ENOENT;
- break;
-
- case ERROR_SEEK_ON_DEVICE:
- case ERROR_NEGATIVE_SEEK:
- errno = ESPIPE;
- break;
-
- case ERROR_ACCESS_DENIED:
- errno = EACCES;
- break;
-
- case ERROR_DIR_NOT_EMPTY:
- errno = ENOTEMPTY;
- break;
-
- case ERROR_BROKEN_PIPE:
- errno = EPIPE;
- break;
-
- case ERROR_GEN_FAILURE:
- errno = EIO;
- break;
-
- case ERROR_OPEN_FAILED:
- errno = EIO;
- break;
-
- case ERROR_SUCCESS:
- errno = 0;
- break;
-
- default:
- /* This is just a canary, if you find this
- * error then it means we need to expand the
- * translation list.
- */
-
- errno = EOWNERDEAD;
- break;
- }
-}
-
-static int get_mode(unsigned int attr)
-{
- if (attr & FILE_ATTRIBUTE_DIRECTORY) {
- return WIN32_S_IFDIR;
- }
- return WIN32_S_IFREG;
-}
-
-
-
-static int is_symlink(const char *path)
-{
- WIN32_FIND_DATA data;
- HANDLE h;
-
- SetLastError(0);
- reset_errno();
-
- h = FindFirstFileA(path, &data);
-
- if (h == INVALID_HANDLE_VALUE) {
- propagate_last_error_to_errno();
-
- return 0;
- }
-
- FindClose(h);
-
- /*
- * A NTFS symlink is a file with a bit of metadata ("reparse point"),
- * So (1) check if the file has metadata and then (2) confirm that
- * it is indeed a symlink.
- */
- if (data.dwFileAttributes & FILE_ATTRIBUTE_REPARSE_POINT) {
- if (data.dwReserved0 == IO_REPARSE_TAG_SYMLINK) {
- return 1;
- }
- }
-
- return 0;
-}
-
-static int hstat(HANDLE h, struct win32_stat *wst)
-{
- BY_HANDLE_FILE_INFORMATION info;
- FILE_STANDARD_INFO std;
-
- SetLastError(0);
- reset_errno();
-
- if (!GetFileInformationByHandle(h, &info)) {
- propagate_last_error_to_errno();
-
- return -1;
- }
-
- if (!GetFileInformationByHandleEx(h, FileStandardInfo,
- &std, sizeof(std))) {
- propagate_last_error_to_errno();
-
- return -1;
- }
-
- wst->st_nlink = std.NumberOfLinks;
- if (std.DeletePending) {
- wst->st_nlink = 0;
- }
-
- wst->st_mode = get_mode(info.dwFileAttributes);
- wst->st_size = UINT64(info.nFileSizeHigh, info.nFileSizeLow);
- wst->st_ino = UINT64(info.nFileIndexHigh, info.nFileIndexLow);
- wst->st_mtime = filetime_to_epoch(&info.ftLastWriteTime);
-
- return 0;
-}
-
-int win32_stat(const char *path, struct win32_stat *wst)
-{
- HANDLE h;
-
- SetLastError(0);
- reset_errno();
-
- h = CreateFileA(path,
- GENERIC_READ,
- FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
- NULL, /* lpSecurityAttributes */
- OPEN_EXISTING, /* dwCreationDisposition */
- 0, /* dwFlagsAndAttributes */
- NULL); /* hTemplateFile */
-
- if (h == INVALID_HANDLE_VALUE) {
- propagate_last_error_to_errno();
-
- return -1;
- }
-
- if (hstat(h, wst)) {
- CloseHandle(h);
- return -1;
- }
-
- CloseHandle(h);
- return 0;
-}
-
-int win32_lstat(const char *path, struct win32_stat *wst)
-{
- HANDLE h;
-
- SetLastError(0);
- reset_errno();
-
- h = CreateFileA(path,
- GENERIC_READ,
- FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE,
- NULL, /* lpSecurityAttributes */
- OPEN_EXISTING, /* dwCreationDisposition */
- FILE_FLAG_OPEN_REPARSE_POINT,
- NULL); /* hTemplateFile */
-
- if (h == INVALID_HANDLE_VALUE) {
- propagate_last_error_to_errno();
-
- return -1;
- }
-
- if (hstat(h, wst)) {
- CloseHandle(h);
- return -1;
- }
-
- if (is_symlink(path)) {
- wst->st_mode = WIN32_S_IFLNK;
- }
-
- CloseHandle(h);
- return 0;
-}
-
-int win32_fstat(int fd, struct win32_stat *wst)
-{
- HANDLE h;
-
- SetLastError(0);
- reset_errno();
-
- h = (HANDLE) _get_osfhandle(fd);
-
- if (h == INVALID_HANDLE_VALUE) {
- propagate_last_error_to_errno();
-
- return -1;
- }
-
- return hstat(h, wst);
-}