metricbeat/magefile.go (167 lines of code) (raw):
// Licensed to Elasticsearch B.V. under one or more contributor
// license agreements. See the NOTICE file distributed with
// this work for additional information regarding copyright
// ownership. Elasticsearch B.V. licenses this file to you 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.
//go:build mage
package main
import (
"context"
"fmt"
"os"
"strconv"
"strings"
"time"
"github.com/magefile/mage/mg"
devtools "github.com/elastic/beats/v7/dev-tools/mage"
"github.com/elastic/beats/v7/dev-tools/mage/gotool"
metricbeat "github.com/elastic/beats/v7/metricbeat/scripts/mage"
// register kubernetes runner
_ "github.com/elastic/beats/v7/dev-tools/mage/kubernetes"
//mage:import
"github.com/elastic/beats/v7/dev-tools/mage/target/build"
//mage:import
"github.com/elastic/beats/v7/dev-tools/mage/target/common"
//mage:import
_ "github.com/elastic/beats/v7/dev-tools/mage/target/dashboards"
//mage:import
_ "github.com/elastic/beats/v7/dev-tools/mage/target/docs"
//mage:import
_ "github.com/elastic/beats/v7/dev-tools/mage/target/pkg"
//mage:import
"github.com/elastic/beats/v7/dev-tools/mage/target/test"
//mage:import
"github.com/elastic/beats/v7/dev-tools/mage/target/unittest"
//mage:import
_ "github.com/elastic/beats/v7/dev-tools/mage/target/integtest/docker"
//mage:import
_ "github.com/elastic/beats/v7/metricbeat/scripts/mage/target/metricset"
)
func init() {
common.RegisterCheckDeps(Update)
test.RegisterDeps(IntegTest)
unittest.RegisterGoTestDeps(Fields)
unittest.RegisterPythonTestDeps(Fields)
devtools.BeatDescription = "Metricbeat is a lightweight shipper for metrics."
}
// BuildSystemTestBinary builds a binary instrumented for use with Python system tests.
func BuildSystemTestBinary() error {
return devtools.BuildSystemTestBinary()
}
// CollectAll generates the docs and the fields.
func CollectAll() {
mg.Deps(CollectDocs, FieldsDocs)
}
// Package packages the Beat for distribution.
// Use SNAPSHOT=true to build snapshots.
// Use PLATFORMS to control the target platforms.
// Use VERSION_QUALIFIER to control the version qualifier.
func Package() {
start := time.Now()
defer func() { fmt.Println("package ran for", time.Since(start)) }()
devtools.UseElasticBeatOSSPackaging()
metricbeat.CustomizePackaging()
devtools.PackageKibanaDashboardsFromBuildDir()
mg.Deps(Update)
mg.Deps(build.CrossBuild)
mg.SerialDeps(devtools.Package, TestPackages)
}
// Package packages the Beat for IronBank distribution.
//
// Use SNAPSHOT=true to build snapshots.
func Ironbank() error {
fmt.Println(">> Ironbank: this module is not subscribed to the IronBank releases.")
return nil
}
// TestPackages tests the generated packages (i.e. file modes, owners, groups).
func TestPackages() error {
return devtools.TestPackages(
devtools.WithModulesD(),
devtools.WithModules(),
// To be increased or removed when more light modules are added
devtools.MinModules(1),
)
}
// Dashboards collects all the dashboards and generates index patterns.
func Dashboards() error {
return devtools.KibanaDashboards("module")
}
// Config generates both the short and reference configs.
func Config() {
mg.Deps(configYML, metricbeat.GenerateDirModulesD)
}
func configYML() error {
return devtools.Config(devtools.AllConfigTypes, metricbeat.OSSConfigFileParams(), ".")
}
// MockedTests runs the HTTP tests using the mocked data inside each {module}/{metricset}/testdata folder.
// Use MODULE={module_name} to run only mocked tests with a single module.
// Use GENERATE=true or GENERATE=1 to regenerate JSON files.
func MockedTests(ctx context.Context) error {
params := devtools.DefaultGoTestUnitArgs()
params.ExtraFlags = []string{"github.com/elastic/beats/v7/metricbeat/mb/testing/data/."}
if module := os.Getenv("MODULE"); module != "" {
params.ExtraFlags = append(params.ExtraFlags, "-module="+module)
}
if generate, _ := strconv.ParseBool(os.Getenv("GENERATE")); generate {
params.ExtraFlags = append(params.ExtraFlags, "-data")
}
params.Packages = nil
return devtools.GoTest(ctx, params)
}
// Fields generates a fields.yml and fields.go for each module.
func Fields() {
mg.Deps(libbeatAndMetricbeatCommonFieldsGo, moduleFieldsGo)
mg.Deps(fieldsYML)
}
// libbeatAndMetricbeatCommonFieldsGo generates a fields.go containing both
// libbeat and metricbeat's common fields.
func libbeatAndMetricbeatCommonFieldsGo() error {
if err := devtools.GenerateFieldsYAML(); err != nil {
return err
}
return devtools.GenerateMetricbeatAllInOneFieldsGo()
}
func fieldsYML() error {
return devtools.GenerateFieldsYAML("module")
}
func moduleFieldsGo() error {
return devtools.GenerateModuleFieldsGo("module")
}
// Update is an alias for running fields, dashboards, config.
func Update() {
mg.SerialDeps(
Fields, Dashboards, Config, CollectAll,
metricbeat.PrepareModulePackagingOSS,
metricbeat.GenerateOSSMetricbeatModuleIncludeListGo)
}
// FieldsDocs generates docs/fields.asciidoc containing all fields
// (including x-pack).
func FieldsDocs() error {
inputs := []string{
devtools.OSSBeatDir("module"),
devtools.XPackBeatDir("module"),
}
output := devtools.CreateDir("build/fields/fields.all.yml")
if err := devtools.GenerateFieldsYAMLTo(output, inputs...); err != nil {
return err
}
return devtools.Docs.FieldDocs(output)
}
// CollectDocs creates the documentation under docs/
func CollectDocs() error {
return metricbeat.CollectDocs()
}
// ExportDashboard exports a dashboard and writes it into the correct directory.
//
// Required environment variables:
// - MODULE: Name of the module
// - ID: Dashboard id
func ExportDashboard() error {
return devtools.ExportDashboard()
}
// IntegTest executes integration tests (it uses Docker to run the tests).
func IntegTest() {
mg.SerialDeps(GoIntegTest, PythonIntegTest)
}
// GoIntegTest executes the Go integration tests.
// Use TEST_COVERAGE=true to enable code coverage profiling.
// Use RACE_DETECTOR=true to enable the race detector.
// Use TEST_TAGS=tag1,tag2 to add additional build tags.
// Use MODULE=module to run only tests for `module`.
func GoIntegTest(ctx context.Context) error {
if os.Getenv("CI") == "true" {
mg.Deps(devtools.DefineModules)
}
if !devtools.IsInIntegTestEnv() {
mg.SerialDeps(Fields, Dashboards)
}
return devtools.GoTestIntegrationForModule(ctx)
}
// PythonIntegTest executes the python system tests in the integration
// environment (Docker).
// Use MODULE=module to run only tests for `module`.
// Use PYTEST_ADDOPTS="-k pattern" to only run tests matching the specified pattern.
// Use any other PYTEST_* environment variable to influence the behavior of pytest.
func PythonIntegTest(ctx context.Context) error {
if os.Getenv("CI") == "true" {
mg.Deps(devtools.DefineModules)
}
if !devtools.IsInIntegTestEnv() {
mg.SerialDeps(Fields, Dashboards)
}
passThroughEnvVars := append(
[]string{"ELASTICSEARCH_VERSION", "KIBANA_VERSION", "BEAT_VERSION"},
devtools.ListMatchingEnvVars("PYTEST_")...,
)
runner, err := devtools.NewDockerIntegrationRunner(passThroughEnvVars...)
if err != nil {
return err
}
return runner.Test("pythonIntegTest", func() error {
mg.Deps(devtools.BuildSystemTestBinary)
args := devtools.DefaultPythonTestIntegrationArgs()
// Always create a fresh virtual environment when running tests in a container, until we get
// get the requirements installed as part of the container build.
args.ForceCreateVenv = true
return devtools.PythonTestForModule(args)
})
}
// FIPSOnlyUnitTest sets GODEBUG=fips140=only when running unit tests
// Will also filter out packages that fail to run with the GODEBUG=fips140=only var set
func FIPSOnlyUnitTest() error {
ctx := context.Background()
fipsArgs := devtools.DefaultGoFIPSOnlyTestArgs()
packages, err := gotool.ListProjectPackages()
if err != nil {
return err
}
filteredPackages := make([]string, 0, len(packages))
for _, pkg := range packages {
// Filter out tests from metricbeat/module/vsphere as the github.com/vmware/govmomi simulator uses SHA-1.
// This causes tests to panic on load before TestMain is ran.
if !strings.Contains(pkg, "github.com/elastic/beats/v7/metricbeat/module/vsphere") {
filteredPackages = append(filteredPackages, pkg)
}
}
fipsArgs.Packages = filteredPackages
return devtools.GoTest(ctx, fipsArgs)
}