Este é um tutorial passo a passo sobre como criar um plugin simples carregável para o Zabbix agent 2.
Você também pode usar o repositório de exemplo como um modelo ou guia para criar seus próprios plugins.
Este tutorial demonstra como criar um novo plugin carregável MyIP. O plugin irá implementar uma única chave de item, myip, que retorna o endereço IP externo do host onde o Zabbix agent 2 está em execução.
go.mod no diretório do plugin para rastrear as dependências do plugin:golang.zabbix.com/sdk):Substitua $LATEST_COMMIT_HASH pelo hash do commit HEAD mais recente do repositório golang.zabbix.com/sdk no branch de release apropriado. Por exemplo:
Observe que a versão do golang.zabbix.com/sdk atualmente não é suportada, mas isso pode mudar no futuro.
Dependências adicionais podem ser instaladas conforme necessário usando go get.
main.go para o código-fonte do plugin:Agora a configuração inicial está concluída e o plugin está pronto para desenvolvimento.
O módulo golang.zabbix.com/sdk, instalado na etapa anterior, fornece a estrutura necessária para o desenvolvimento de plugins e garante que todos os plugins tenham uma estrutura consistente.
Comece definindo o fluxo principal de execução do plugin. Adicione o seguinte código ao main.go:
package main
func main() {
err := run()
if err != nil {
panic(err)
}
}
func run() error {
return nil
}Isso estabelece o fluxo básico de execução para o plugin. A função run conterá posteriormente a lógica principal do plugin.
Um plugin do Zabbix agent 2 deve ser representado por uma struct que implementa interfaces do pacote golang.zabbix.com/sdk/plugin:
Você pode implementar essas interfaces por conta própria ou usar as implementações padrão fornecidas pelo Zabbix Go SDK, modificando-as conforme necessário. Este tutorial utiliza as implementações padrão.
Agora, importe o pacote plugin e crie uma struct myIP que incorpora a struct plugin.Base:
A struct myIP atualmente satisfaz a interface Accessor. Um método para implementar uma das interfaces funcionais do plugin, o Exporter, será adicionado posteriormente no tutorial.
Seu plugin precisa das chaves de item para coletar dados e fornecê-los ao Zabbix server ou proxy.
plugin.RegisterMetrics() dentro da função run():func run() error {
p := &myIP{}
// Registra a chave de item `myip`.
err := plugin.RegisterMetrics(
p,
"MyIP", // Nome do plugin
"myip", // Nome da chave de item
"Retorna o endereço IP do host.", // Descrição da chave de item
)
if err != nil {
return errs.Wrap(err, "falha ao registrar métricas")
}
return nil
}Para registrar várias chaves de item, repita os parâmetros nome da métrica e descrição para cada métrica. Por exemplo:
plugin.RegisterMetrics(&impl, "Myip", "metric.one", "Descrição da métrica um.", "metric.two", "Descrição da métrica dois.")O handler facilita a comunicação entre o agent e o plugin.
run(), adicione o código para criar e configurar um handler:func run() error {
p := &myIP{}
// Registre a chave de item `myip`.
err := plugin.RegisterMetrics(
p,
"MyIP", // Nome do plugin
"myip", // Nome da chave do item
"Returns the host's IP address.", // Descrição da chave do item
)
if err != nil {
return errs.Wrap(err, "failed to register metrics")
}
// Crie um novo handler.
h, err := container.NewHandler("MyIP") // Nome do plugin
if err != nil {
return errs.Wrap(err, "failed to create new handler")
}
// Configure o log para encaminhar logs do plugin para o agent.
// Disponível via p.Logger.Infof, p.Logger.Debugf, etc.
p.Logger = h
// Inicie a execução do plugin.
// Bloqueia até que uma solicitação de término seja recebida do agent.
err = h.Execute()
if err != nil {
return errs.Wrap(err, "failed to execute plugin handler")
}
return nil
}A coleta de dados é feita por meio da interface Exporter, que descreve o método Export:
func Export(
key string, // O item key a ser coletado.
params []string, // Argumentos passados para o item key (`myip[arg1, arg2]`).
context ContextProvider // Metadados sobre a coleta de dados do item key.
) (any, error)import (
"io"
"net/http"
)
Export para a struct myIP:func (p *myIP) Export(
key string, params []string, context plugin.ContextProvider,
) (any, error) {
// O plugin pode usar diferentes lógicas de coleta de dados com base no parâmetro `key`.
// Esta implementação apenas verifica se o `key` fornecido é suportado.
if key != "myip" {
return nil, errs.Errorf("unknown item key %q", key)
}
// O log será encaminhado para o log do agent 2.
p.Infof(
"received request to handle %q key with %d parameters",
key,
len(params),
)
// Colete os dados e retorne-os.
resp, err := http.Get("https://api.ipify.org")
if err != nil {
return nil, errs.Wrap(err, "failed to get IP address")
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, errs.Wrap(err, "failed to read response body")
}
return string(body), nil
}Isso deve criar um executável myip no diretório atual.
echo "Plugins.MyIP.System.Path=$PATH_TO_THE_MYIP_PLUGIN_EXECUTABLE" > /etc/zabbix_agent2.d/plugins.d/myip.confSubstitua $PATH_TO_THE_MYIP_PLUGIN_EXECUTABLE pelo caminho para o myip criado na etapa 5.
O nome do plugin no parâmetro de configuração (MyIP neste tutorial) deve corresponder ao nome do plugin definido na função plugin.RegisterMetrics().
myip, execute:A saída deve conter um endereço IP externo do seu host e ser semelhante a esta:
myip [s|192.0.2.0]
Com isso, você criou um plugin simples carregável para o Zabbix agent 2. Parabéns!
package main
import (
"io"
"net/http"
"golang.zabbix.com/sdk/errs"
"golang.zabbix.com/sdk/plugin"
"golang.zabbix.com/sdk/plugin/container"
)
var _ plugin.Exporter = (*myIP)(nil)
type myIP struct {
plugin.Base
}
func main() {
err := run()
if err != nil {
panic(err)
}
}
func run() error {
p := &myIP{}
// Registra a chave do item `myip`.
err := plugin.RegisterMetrics(
p,
"MyIP", // Nome do plugin
"myip", // Nome da chave do item
"Returns the host's IP address.", // Descrição da chave do item
)
if err != nil {
return errs.Wrap(err, "failed to register metrics")
}
// Cria um novo handler.
h, err := container.NewHandler("MyIP") // Nome do plugin
if err != nil {
return errs.Wrap(err, "failed to create new handler")
}
// Configura o log para encaminhar logs do plugin para o agent.
// Disponível via p.Logger.Infof, p.Logger.Debugf, etc.
p.Logger = h
// Inicia a execução do plugin.
// Bloqueia até receber uma solicitação de término do agent.
err = h.Execute()
if err != nil {
return errs.Wrap(err, "failed to execute plugin handler")
}
return nil
}
func (p *myIP) Export(
key string, params []string, context plugin.ContextProvider,
) (any, error) {
// O plugin pode usar diferentes lógicas de coleta de dados com base no parâmetro `key`.
// Esta implementação apenas verifica se a `key` fornecida é suportada.
if key != "myip" {
return nil, errs.Errorf("unknown item key %q", key)
}
// O log será encaminhado para o log do agent 2.
p.Infof(
"received request to handle %q key with %d parameters",
key,
len(params),
)
// Coleta os dados e os retorna.
resp, err := http.Get("https://api.ipify.org")
if err != nil {
return nil, errs.Wrap(err, "failed to get IP address")
}
defer resp.Body.Close()
body, err := io.ReadAll(resp.Body)
if err != nil {
return nil, errs.Wrap(err, "failed to read response body")
}
return string(body), nil
}