181 lines
4.8 KiB
Go
181 lines
4.8 KiB
Go
|
package openweathermap
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"net/http"
|
||
|
"net/http/httptest"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"strings"
|
||
|
"testing"
|
||
|
"time"
|
||
|
|
||
|
"github.com/stretchr/testify/require"
|
||
|
|
||
|
"github.com/influxdata/telegraf"
|
||
|
"github.com/influxdata/telegraf/config"
|
||
|
"github.com/influxdata/telegraf/plugins/inputs"
|
||
|
"github.com/influxdata/telegraf/plugins/parsers/influx"
|
||
|
"github.com/influxdata/telegraf/testutil"
|
||
|
)
|
||
|
|
||
|
func TestFormatURL(t *testing.T) {
|
||
|
n := &OpenWeatherMap{
|
||
|
AppID: "appid",
|
||
|
Units: "metric",
|
||
|
Lang: "de",
|
||
|
BaseURL: "http://foo.com",
|
||
|
}
|
||
|
require.NoError(t, n.Init())
|
||
|
|
||
|
require.Equal(t,
|
||
|
"http://foo.com/data/2.5/forecast?APPID=appid&id=12345&lang=de&units=metric",
|
||
|
n.formatURL("/data/2.5/forecast", "12345"))
|
||
|
}
|
||
|
|
||
|
func TestDefaultUnits(t *testing.T) {
|
||
|
n := &OpenWeatherMap{}
|
||
|
require.NoError(t, n.Init())
|
||
|
|
||
|
require.Equal(t, "metric", n.Units)
|
||
|
}
|
||
|
|
||
|
func TestDefaultLang(t *testing.T) {
|
||
|
n := &OpenWeatherMap{}
|
||
|
require.NoError(t, n.Init())
|
||
|
|
||
|
require.Equal(t, "en", n.Lang)
|
||
|
}
|
||
|
|
||
|
func TestCases(t *testing.T) {
|
||
|
// Get all directories in testdata
|
||
|
folders, err := os.ReadDir("testcases")
|
||
|
require.NoError(t, err)
|
||
|
|
||
|
// Register the plugin
|
||
|
inputs.Add("openweathermap", func() telegraf.Input {
|
||
|
return &OpenWeatherMap{
|
||
|
ResponseTimeout: config.Duration(5 * time.Second),
|
||
|
}
|
||
|
})
|
||
|
|
||
|
// Prepare the influx parser for expectations
|
||
|
parser := &influx.Parser{}
|
||
|
require.NoError(t, parser.Init())
|
||
|
|
||
|
for _, f := range folders {
|
||
|
// Only handle folders
|
||
|
if !f.IsDir() {
|
||
|
continue
|
||
|
}
|
||
|
testcasePath := filepath.Join("testcases", f.Name())
|
||
|
configFilename := filepath.Join(testcasePath, "telegraf.conf")
|
||
|
expectedFilename := filepath.Join(testcasePath, "expected.out")
|
||
|
expectedErrorFilename := filepath.Join(testcasePath, "expected.err")
|
||
|
|
||
|
t.Run(f.Name(), func(t *testing.T) {
|
||
|
// Read the input data
|
||
|
input, err := readInputData(testcasePath)
|
||
|
require.NoError(t, err)
|
||
|
|
||
|
// Read the expected output if any
|
||
|
var expected []telegraf.Metric
|
||
|
if _, err := os.Stat(expectedFilename); err == nil {
|
||
|
var err error
|
||
|
expected, err = testutil.ParseMetricsFromFile(expectedFilename, parser)
|
||
|
require.NoError(t, err)
|
||
|
}
|
||
|
|
||
|
// Read the expected output if any
|
||
|
var expectedErrors []string
|
||
|
if _, err := os.Stat(expectedErrorFilename); err == nil {
|
||
|
var err error
|
||
|
expectedErrors, err = testutil.ParseLinesFromFile(expectedErrorFilename)
|
||
|
require.NoError(t, err)
|
||
|
require.NotEmpty(t, expectedErrors)
|
||
|
}
|
||
|
|
||
|
// Start the test-server
|
||
|
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||
|
// Lookup the response
|
||
|
key := strings.TrimPrefix(r.URL.Path, "/data/2.5/")
|
||
|
if resp, found := input[key]; found {
|
||
|
w.Header()["Content-Type"] = []string{"application/json"}
|
||
|
if _, err := w.Write(resp); err != nil {
|
||
|
w.WriteHeader(http.StatusInternalServerError)
|
||
|
t.Error(err)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Try to append the key and find the response
|
||
|
ids := strings.Split(r.URL.Query().Get("id"), ",")
|
||
|
if len(ids) > 1 {
|
||
|
w.WriteHeader(http.StatusBadRequest)
|
||
|
return
|
||
|
}
|
||
|
if len(ids) == 1 {
|
||
|
key += "_" + ids[0]
|
||
|
if resp, found := input[key]; found {
|
||
|
w.Header()["Content-Type"] = []string{"application/json"}
|
||
|
if _, err := w.Write(resp); err != nil {
|
||
|
w.WriteHeader(http.StatusInternalServerError)
|
||
|
t.Error(err)
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
w.WriteHeader(http.StatusNotFound)
|
||
|
}))
|
||
|
defer server.Close()
|
||
|
|
||
|
// Configure the plugin
|
||
|
cfg := config.NewConfig()
|
||
|
require.NoError(t, cfg.LoadConfig(configFilename))
|
||
|
require.Len(t, cfg.Inputs, 1)
|
||
|
|
||
|
// Fake the reading
|
||
|
plugin := cfg.Inputs[0].Input.(*OpenWeatherMap)
|
||
|
plugin.BaseURL = server.URL
|
||
|
require.NoError(t, plugin.Init())
|
||
|
|
||
|
var acc testutil.Accumulator
|
||
|
require.NoError(t, plugin.Gather(&acc))
|
||
|
if len(acc.Errors) > 0 {
|
||
|
var actualErrorMsgs []string
|
||
|
for _, err := range acc.Errors {
|
||
|
actualErrorMsgs = append(actualErrorMsgs, err.Error())
|
||
|
}
|
||
|
require.ElementsMatch(t, actualErrorMsgs, expectedErrors)
|
||
|
}
|
||
|
|
||
|
// Check the metric nevertheless as we might get some metrics despite errors.
|
||
|
actual := acc.GetTelegrafMetrics()
|
||
|
testutil.RequireMetricsEqual(t, expected, actual, testutil.SortMetrics())
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func readInputData(path string) (map[string][]byte, error) {
|
||
|
pattern := filepath.Join(path, "response_*.json")
|
||
|
matches, err := filepath.Glob(pattern)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// Iterate over the response_*.json files and read the into the data map
|
||
|
data := make(map[string][]byte, len(matches))
|
||
|
for _, filename := range matches {
|
||
|
resp, err := os.ReadFile(filename)
|
||
|
if err != nil {
|
||
|
return nil, fmt.Errorf("reading response %q failed: %w", filename, err)
|
||
|
}
|
||
|
key := filepath.Base(filename)
|
||
|
key = strings.TrimPrefix(key, "response_")
|
||
|
key = strings.TrimSuffix(key, ".json")
|
||
|
data[key] = resp
|
||
|
}
|
||
|
return data, nil
|
||
|
}
|