如何编写 Logstash 过滤器插件

如何编写 Logstash 过滤器插件

要为 Logstash 开发一个新的过滤器,请构建一个自包含的 Ruby gem,其源代码位于其自己的 GitHub 存储库中。然后,可以将 Ruby gem 托管并在 RubyGems.org 上共享。您可以使用示例过滤器实现作为起点。(如果您不熟悉 Ruby,可以在 https://www.ruby-lang.org.cn/en/documentation/quickstart/ 找到优秀的快速入门指南。)

开始

让我们逐步了解如何使用 示例过滤器插件 创建过滤器插件。

为您的新插件创建 GitHub 存储库

每个 Logstash 插件都位于其自己的 GitHub 存储库中。要为您的插件创建一个新的存储库

  1. 登录 GitHub。
  2. 点击右上角的 存储库 选项卡。您将看到您已分叉或为其贡献的其他存储库的列表。
  3. 点击右上角的绿色 新建 按钮。
  4. 为您的新存储库指定以下设置

    • 存储库名称 — 以 logstash-filter-pluginname 格式的唯一名称。
    • 公开或私有 — 您的选择,但如果您想将其提交为官方插件,则存储库必须是公开的。
    • 使用自述文件初始化此存储库 — 使您能够立即将存储库克隆到您的计算机。
  5. 点击 创建存储库

使用插件生成器工具

您可以在几秒钟内创建自己的 Logstash 插件!bin/logstash-plugingenerate 子命令使用模板化文件创建 Logstash 插件的基础。它创建正确的目录结构、gemspec 文件和依赖项,以便您可以开始添加自定义代码以使用 Logstash 处理数据。

有关更多信息,请参阅 生成插件

复制过滤器代码

或者,您可以使用我们在 github.com 上托管的示例存储库

  1. 克隆您的插件。GITUSERNAME 替换为您的 github 用户名,将 MYPLUGINNAME 替换为您的插件名称。

    • git clone https://github.com/GITUSERNAME/logstash-filter-MYPLUGINNAME.git

      • 或者,通过 ssh:git clone [email protected]:GITUSERNAME/logstash-filter-MYPLUGINNAME.git
    • cd logstash-filter-MYPLUGINNAME
  2. 克隆过滤器插件示例并将其复制到您的插件分支。

    您不希望包含示例 .git 目录或其内容,因此在复制示例之前将其删除。

    • cd /tmp
    • git clone https://github.com/logstash-plugins/logstash-filter-example.git
    • cd logstash-filter-example
    • rm -rf .git
    • cp -R * /path/to/logstash-filter-mypluginname/
  3. 将以下文件重命名为与您的插件名称匹配。

    • logstash-filter-example.gemspec
    • example.rb
    • example_spec.rb

      cd /path/to/logstash-filter-mypluginname
      mv logstash-filter-example.gemspec logstash-filter-mypluginname.gemspec
      mv lib/logstash/filters/example.rb lib/logstash/filters/mypluginname.rb
      mv spec/filters/example_spec.rb spec/filters/mypluginname_spec.rb

您的文件结构应如下所示

$ tree logstash-filter-mypluginname
├── Gemfile
├── LICENSE
├── README.md
├── Rakefile
├── lib
│   └── logstash
│       └── filters
│           └── mypluginname.rb
├── logstash-filter-mypluginname.gemspec
└── spec
    └── filters
        └── mypluginname_spec.rb

有关 Ruby gem 文件结构的更多信息以及 Ruby gem 创建过程的优秀演练,请参阅 http://timelessrepo.com/making-ruby-gems

查看您的插件的外观

在深入了解细节之前,请在您最喜欢的文本编辑器中打开插件文件并查看。

require "logstash/filters/base"
require "logstash/namespace"

# Add any asciidoc formatted documentation here
# This example filter will replace the contents of the default
# message field with whatever you specify in the configuration.
#
# It is only intended to be used as an example.
class LogStash::Filters::Example < LogStash::Filters::Base

  # Setting the config_name here is required. This is how you
  # configure this filter from your Logstash config.
  #
  # filter {
  #   example { message => "My message..." }
  # }
  config_name "example"

  # Replace the message with this value.
  config :message, :validate => :string, :default => "Hello World!"


  public
  def register
    # Add instance variables
  end # def register

  public
  def filter(event)

    if @message
      # Replace the event message with our message as configured in the
      # config file.
      event.set("message", @message)
    end

    # filter_matched should go in the last line of our successful code
    filter_matched(event)
  end # def filter

end # class LogStash::Filters::Example

编写过滤器插件

现在让我们逐行查看示例插件。

require 语句

Logstash 过滤器插件需要在 logstash/filters/base 和 logstash/namespace 中定义的父类。

require "logstash/filters/base"
require "logstash/namespace"

当然,您构建的插件可能依赖于其他代码,甚至 gem。只需将它们与这些 Logstash 依赖项一起放在这里即可。

插件主体

让我们遍历插件本身的各个元素。

class 声明

过滤器插件类应该是 LogStash::Filters::Base 的子类。

class LogStash::Filters::Example < LogStash::Filters::Base

类名应与插件名称密切对应,例如

LogStash::Filters::Example

config_name

  config_name "example"

这是插件在过滤器配置块内部调用的名称。

如果在插件代码中设置了 config_name "example",则相应的 Logstash 配置块需要如下所示

配置参数

  config :variable_name, :validate => :variable_type, :default => "Default value", :required => boolean, :deprecated => boolean, :obsolete => string

配置或 config 部分允许您定义任意数量(或很少)的参数,以使 Logstash 能够处理事件。

有几个配置属性

  • :validate - 允许您强制为 Logstash 的此配置选项传递特定数据类型,例如 :string:password:boolean:number:array:hash:path(文件系统路径)、uri:codec(自 1.2.0 起)、:bytes。请注意,这也可以作为强制类型转换,因为如果我为布尔值指定“true”(即使在技术上是字符串),它也会在配置中变成有效的布尔值。此强制类型转换也适用于 :number 类型,其中“1.2”变为浮点数,“22”变为整数。
  • :default - 允许您为参数指定默认值
  • :required - 此参数是否为必填项(布尔值 truefalse
  • :list - 此值是否应为值列表。将对列表成员进行类型检查,并将标量转换为单元素列表。请注意,这主要取代了数组类型,但如果您需要复杂对象的列表,则更适合。
  • :deprecated - 信息性(也是布尔值 truefalse
  • :obsolete - 用于声明给定设置已被删除且不再起作用。其目的是为仍在使用现已删除设置的用户提供知情的升级路径。

插件方法

Logstash 过滤器必须实现 registerfilter 方法。

register 方法

  public
  def register
  end # def register

Logstash register 方法类似于 initialize 方法。它最初是为了强制调用 super 而创建的,从而避免了新手出现头痛的问题。(注意:它可能会被 initialize 取代,并结合一些强制测试以确保调用了 super。)

public 表示该方法可以在任何地方调用,而不仅仅是在类内部。这是 Ruby 中方法的默认行为,但这里无论如何都明确指定了它。

您还可以在此处分配实例变量(以 @ 开头的变量)。配置变量现在在作用域内作为实例变量,例如 @message

filter 方法

  public
  def filter(event)

    if @message
      # Replace the event message with our message as configured in the
      # config file.
      event.set("message", @message)
    end

  # filter_matched should go in the last line of our successful code
  filter_matched(event)
end # def filter

插件的 filter 方法是实际过滤工作发生的地方!在 filter 方法内部,您可以使用 Event 对象引用事件数据。Event 是封装 Logstash 内部数据流的主要对象,并为插件开发人员提供了一个 API 来与事件的内容进行交互。

filter 方法还应通过显式调用 Event 类中可用的 sprintf 方法来处理任何 依赖于事件的配置。例如

field_foo = event.sprintf(field)

请注意,配置变量现在在作用域内作为实例变量,例如 @message

  filter_matched(event)

在插件成功执行后调用 filter_matched 方法将确保通过 Logstash 配置为该过滤器添加的任何字段或标签都将得到正确处理。例如,任何 add_fieldremove_fieldadd_tag 和/或 remove_tag 操作都将在此时执行。

现在可以使用 event.cancel 等事件方法来控制正在处理的事件的工作流。

构建插件

在流程的这一步,您已编写了插件,并准备从中构建一个 Ruby Gem。以下信息将帮助您完成此过程。

外部依赖项

Ruby 中的 require 语句用于包含必要的代码。在某些情况下,您的插件可能需要其他文件。例如,collectd 插件 使用 collectd 提供的 types.db 文件。在插件的主目录中,名为 vendor.json 的文件用于描述这些文件。

vendor.json 文件包含一个 JSON 对象数组,每个对象描述一个文件依赖项。此示例来自 collectd 编解码器插件

[{
        "sha1": "a90fe6cc53b76b7bdd56dc57950d90787cb9c96e",
        "url": "http://collectd.org/files/collectd-5.4.0.tar.gz",
        "files": [ "/src/types.db" ]
}]
  • sha1 是用于验证 url 引用的文件完整性的 sha1 签名。
  • url 是 Logstash 将从中下载文件的地址。
  • files 是一个可选的文件数组,用于从下载的文件中提取。请注意,虽然 tar 归档可以使用绝对或相对路径,但在此数组中将它们视为绝对路径。如果不存在 files,则所有文件都将解压缩并提取到 vendor 目录中。

另一个 vendor.json 文件的示例是 geoip 过滤器

用于下载这些依赖项的过程是调用 rake vendor。这将在本文档的测试部分中进一步讨论。

另一种外部依赖项是对 jar 文件的依赖项。这将在“添加 gemspec 文件”部分中描述。

已弃用的功能

随着插件的发展,选项或功能可能不再服务于预期的目的,开发人员可能希望弃用其使用。弃用会警告用户有关选项的状态,因此当它在以后的版本中被删除时,他们不会措手不及。

Logstash 7.6 引入了弃用日志记录器,使处理这些情况变得更容易。您可以使用 适配器 来确保您的插件可以使用弃用日志记录器,同时仍支持旧版本的 Logstash。有关更多信息以及有关使用适配器的说明,请参阅 自述文件

弃用情况记录在 log 目录下的 logstash-deprecation.log 文件中。

添加 Gemfile

Gemfile 允许 Ruby 的 Bundler 维护插件的依赖项。目前,我们只需要 Logstash gem 用于测试,但如果您需要其他 gem,则应在此处添加它们。

有关更多详细信息,请参阅 Bundler 的 Gemfile 页面

source 'https://rubygems.org.cn'
gemspec
gem "logstash", :github => "elastic/logstash", :branch => "8.16"

添加 gemspec 文件

Gemspec 定义将要构建的 Ruby gem 并包含您的插件。

有关更多信息,请参阅 Rubygems 规范页面

Gem::Specification.new do |s|
  s.name = 'logstash-filter-example'
  s.version = '0.1.0'
  s.licenses = ['Apache License (2.0)']
  s.summary = "This filter does x, y, z in Logstash"
  s.description = "This gem is a logstash plugin required to be installed on top of the Logstash core pipeline using $LS_HOME/bin/logstash-plugin install gemname. This gem is not a stand-alone program"
  s.authors = ["Elastic"]
  s.email = '[email protected]'
  s.homepage = "https://elastic.ac.cn/guide/en/logstash/current/index.html"
  s.require_paths = ["lib"]

  # Files
  s.files = Dir['lib/**/*','spec/**/*','vendor/**/*','*.gemspec','*.md','CONTRIBUTORS','Gemfile','LICENSE','NOTICE.TXT']
   # Tests
  s.test_files = s.files.grep(%r{^(test|spec|features)/})

  # Special flag to let us know this is actually a logstash plugin
  s.metadata = { "logstash_plugin" => "true", "logstash_group" => "filter" }

  # Gem dependencies
  s.add_runtime_dependency "logstash-core-plugin-api", ">= 1.60", "<= 2.99"
  s.add_development_dependency 'logstash-devutils'
end

可以更改这些值以适合您的插件。特别是,s.names.summary 应反映您的插件的名称和行为。

s.licensess.version 也很重要,当您准备发布插件时将发挥作用。

Logstash 及其所有插件均在 Apache 许可证 2.0 版 ("ALv2") 下获得许可。如果您通过 RubyGems.org 公开发布您的插件,请确保在您的 gemspec 中包含此行

  • s.licenses = ['Apache License (2.0)']

s.version 指定的 gem 版本有助于跟踪插件随时间的变化。您应该对版本号使用 语义化版本控制 策略。

运行时和开发依赖项

gemspec 文件的底部有一个带有注释的部分:Gem dependencies。这是必须提及任何其他所需 gem 的地方。如果一个 gem 对于您的插件的功能是必要的,那么它就是一个运行时依赖项。如果一个 gem 仅用于测试,那么它就是一个开发依赖项。

您还可以为您的依赖项(包括其他 Logstash 插件)设置版本要求

  # Gem dependencies
  s.add_runtime_dependency "logstash-core-plugin-api", ">= 1.60", "<= 2.99"
  s.add_development_dependency 'logstash-devutils'

此 gemspec 对 logstash-core-plugin-api 具有运行时依赖性,并要求其版本号大于或等于 1.60 且小于或等于 2.99。

所有插件都对 logstash-core-plugin-api gem 具有运行时依赖性,并且对 logstash-devutils 具有开发依赖性。

Jar 依赖项

在某些情况下,例如 Elasticsearch 输出插件,您的代码可能依赖于 jar 文件。在这种情况下,依赖项以这种方式添加到 gemspec 文件中

  # Jar dependencies
  s.requirements << "jar 'org.elasticsearch:elasticsearch', '5.0.0'"
  s.add_runtime_dependency 'jar-dependencies'

通过定义这两者,安装过程将在 http://mvnrepository.com 上搜索所需的 jar 文件并下载指定的版本。

记录您的插件

文档是插件的重要组成部分。所有插件文档都将被渲染并放置在 Logstash 参考版本化插件文档 中。

有关提示和指南,请参阅 记录您的插件

添加测试

Logstash 喜欢测试。大量的测试。如果您在生产环境中使用新的过滤器插件,则需要进行一些测试以确保您没有破坏任何现有功能。

本文档不包含关于 RSpec 的完整阐述。在 https://rspec.ruby-lang.org.cn 上了解更多关于 RSpec 的信息

有关学习测试和测试的帮助,请查看其他几个类似插件的 spec/filters/ 目录。

克隆和测试!

现在让我们从插件的新克隆开始,构建它并运行测试。

  • 将您的插件克隆到临时位置GITUSERNAME 替换为您的 github 用户名,将 MYPLUGINNAME 替换为您的插件名称。

    • git clone https://github.com/GITUSERNAME/logstash-filter-MYPLUGINNAME.git

      • 或者,通过 ssh:git clone [email protected]:GITUSERNAME/logstash-filter-MYPLUGINNAME.git
    • cd logstash-filter-MYPLUGINNAME

然后,您需要使用 bundler 安装插件的依赖项

bundle install

如果您的插件在 vendor.json 中描述了外部文件依赖项,则必须在运行或测试之前下载该依赖项。您可以通过运行以下命令来实现

rake vendor

最后,运行测试

bundle exec rspec

您应该会看到一条成功消息,如下所示

Finished in 0.034 seconds
1 example, 0 failures

万岁!你快到了!(除非你看到了失败……你应该先修复它们)。

构建和测试

现在,您可以将您的(经过良好测试的)插件构建成一个 Ruby gem。

构建

您已经拥有了所有必要的成分,所以让我们继续运行构建命令

gem build logstash-filter-example.gemspec

就是这样!您的 gem 应该已经构建完成,并且位于同一路径下,名称为

logstash-filter-mypluginname-0.1.0.gem

来自您的 gemspec 文件的 s.version 编号将提供 gem 版本,在本例中为 0.1.0

测试安装

您应该测试将插件安装到 Logstash 的干净安装中。从 Logstash 下载页面 下载最新版本。

  1. 解压缩并进入目录

    curl -O https://download.elastic.co/logstash/logstash/logstash-8.16.0.tar.gz
    tar xzvf logstash-8.16.0.tar.gz
    cd logstash-8.16.0
  2. 使用插件工具,我们可以安装我们刚刚构建的 gem。

    • /my/logstash/plugins 替换为您环境中 gem 的正确路径,并将 0.1.0 替换为 gemspec 文件中正确的版本号。

      bin/logstash-plugin install /my/logstash/plugins/logstash-filter-example/logstash-filter-example-0.1.0.gem
    • 运行此命令后,您应该会看到 Logstash 的反馈,表明它已成功安装

      validating /my/logstash/plugins/logstash-filter-example/logstash-filter-example-0.1.0.gem >= 0
      Valid logstash plugin. Continuing...
      Successfully installed 'logstash-filter-example' with version '0.1.0'

      您还可以使用 Logstash 插件工具来确定当前可用的插件。

      bin/logstash-plugin list

      根据您安装的内容,您可能会看到一个简短或较长的插件列表:输入、编解码器、过滤器和输出。

  3. 现在尝试使用通过命令行传递的简单配置运行 Logstash,使用 -e 标志。

    您的结果将取决于您的过滤器插件的设计目的。

bin/logstash -e 'input { stdin{} } filter { example {} } output {stdout { codec => rubydebug }}'

通过发送输入到 stdin 和输出(过滤后)到 stdout 来测试您的过滤器,使用 rubydebug 编解码器,这可以提高可读性。

在示例过滤器插件的情况下,您发送的任何文本都将被 message 配置参数的内容替换,默认值为 "Hello World!"。

Testing 1, 2, 3
{
       "message" => "Hello World!",
      "@version" => "1",
    "@timestamp" => "2015-01-27T19:17:18.932Z",
          "host" => "cadenza"
}

随意通过更改 message 参数进行实验和测试

bin/logstash -e 'input { stdin{} } filter { example { message => "This is a new message!"} } output {stdout { codec => rubydebug }}'

恭喜!您已构建、部署并成功运行了一个 Logstash 过滤器。

将您的插件提交到 RubyGems.orglogstash-plugins

Logstash 使用 RubyGems.org 作为其所有插件工件的存储库。开发完新插件后,只需将其发布到 RubyGems.org 即可使其可供 Logstash 用户使用。

许可证

Logstash 及其所有插件均在 Apache 许可证 2.0 版 ("ALv2") 下获得许可。如果您通过 RubyGems.org 公开发布您的插件,请确保在您的 gemspec 中包含此行

  • s.licenses = ['Apache License (2.0)']

发布到 RubyGems.org

首先,您需要在 RubyGems.org 上创建一个帐户

创建帐户后,从 RubyGems.org 获取 API 密钥。默认情况下,RubyGems 使用 ~/.gem/credentials 文件存储您的 API 密钥。这些凭据将用于发布 gem。将 usernamepassword 替换为您在 RubyGems.org 上创建的凭据

curl -u username:password https://rubygems.org.cn/api/v1/api_key.yaml > ~/.gem/credentials
chmod 0600 ~/.gem/credentials

在继续之前,请确保您的 gemspec 文件中具有正确的版本并提交您的更改。

  • s.version = '0.1.0'

要发布新 logstash gem 的 0.1.0 版本

bundle install
bundle exec rake vendor
bundle exec rspec
bundle exec rake publish_gem

执行 rake publish_gem

  1. 从 gemspec 文件读取版本 (s.version = '0.1.0')
  2. 检查本地存储库中是否存在该版本的标签。如果标签已存在,则会中止该过程。否则,它将在您的本地存储库中创建一个新的版本标签。
  3. 构建 gem
  4. 将 gem 发布到 RubyGems.org

就是这样!您的插件已发布!Logstash 用户现在可以通过运行以下命令安装您的插件

bin/logstash-plugin install logstash-filter-mypluginname

将您的源代码贡献到 logstash-plugins

不需要将您的源代码贡献到 logstash-plugins github 组织,但我们始终欢迎新的插件!

优势

将您的插件包含在 logstash-plugins 存储库中有很多好处,其中一些包括

  • 发现。 您的插件将出现在 Logstash 参考 中,Logstash 用户首先会在那里查找插件和文档。
  • 文档。 您的插件文档将自动添加到 Logstash 参考 中。
  • 测试。 借助我们的测试基础设施,您的插件将针对 Logstash 的当前和未来版本进行持续测试。因此,用户可以确信,如果出现不兼容性,将会很快被发现并纠正。

验收指南

  • 代码审查。 您的插件必须由社区成员审查其一致性、质量、可读性、稳定性和安全性。
  • 测试。 您的插件必须包含测试才能被接受。这些测试也需要经过代码审查以确保范围和完整性。如果您不知道如何编写测试,也没关系——我们会指导您。我们正在努力发布一份关于为 Logstash 创建测试的指南,这将使它更容易。同时,您可以参考 http://betterspecs.org/ 获取示例。

要开始将您的插件迁移到 logstash-plugins,只需在 Logstash 存储库中创建一个新的 问题。完成验收指南后,我们将使用推荐的 github 流程 促进迁移到 logstash-plugins 组织。