mirror of
https://github.com/prometheus/prometheus.git
synced 2025-08-07 06:37:17 +02:00
Add script for converting PromQL tests to new syntax format (#16562)
Signed-off-by: Kapil Lamba <kapillamba4@gmail.com> Co-authored-by: Neeraj Gartia <80708727+NeerajGartia21@users.noreply.github.com>
This commit is contained in:
parent
8d9dfa075d
commit
69906bb4f5
@ -164,3 +164,30 @@ There can be multiple `<expect>` lines for a given `<type>`. Each `<type>` valid
|
|||||||
Every `<expect>` line must match at least one corresponding annotation or error.
|
Every `<expect>` line must match at least one corresponding annotation or error.
|
||||||
|
|
||||||
If at least one `<expect>` line of type `warn` or `info` is present, then all corresponding annotations must have a matching `expect` line.
|
If at least one `<expect>` line of type `warn` or `info` is present, then all corresponding annotations must have a matching `expect` line.
|
||||||
|
|
||||||
|
#### Migrating Test Files to the New Syntax
|
||||||
|
|
||||||
|
- All `.test` files in the directory specified by the --dir flag will be updated in place.
|
||||||
|
- Deprecated syntax will be replaced with the recommended `expect` line statements.
|
||||||
|
|
||||||
|
Usage:
|
||||||
|
```sh
|
||||||
|
go run ./promql/promqltest/cmd/migrate/main.go --mode=strict [--dir=<directory>]
|
||||||
|
```
|
||||||
|
|
||||||
|
The `--mode` flag controls how expectations are migrated:
|
||||||
|
- `strict`: Strictly migrates all expectations to the new syntax.
|
||||||
|
This is probably more verbose than intended because the old syntax
|
||||||
|
implied many constraints that are often not needed.
|
||||||
|
- `basic`: Like `strict` but never creates `no_info` and `no_warn`
|
||||||
|
expectations. This can be a good starting point to manually add
|
||||||
|
`no_info` and `no_warn` expectations and/or remove `info` and
|
||||||
|
`warn` expectations as needed.
|
||||||
|
- `tolerant`: Only creates `expect fail` and `expect ordered` where
|
||||||
|
appropriate. All desired expectations about presence or absence
|
||||||
|
of `info` and `warn` have to be added manually.
|
||||||
|
|
||||||
|
All three modes create valid passing tests from previously passing tests.
|
||||||
|
`basic` and `tolerant` just test fewer expectations than the previous tests.
|
||||||
|
|
||||||
|
The --dir flag specifies the directory containing test files to migrate.
|
||||||
|
33
promql/promqltest/cmd/migrate/main.go
Normal file
33
promql/promqltest/cmd/migrate/main.go
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
// Copyright 2025 The Prometheus 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.
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/prometheus/prometheus/promql/promqltest"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
mode := flag.String("mode", "strict", "Migration mode: strict, basic, or tolerant")
|
||||||
|
dir := flag.String("dir", "", "Directory to migrate")
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
if err := promqltest.MigrateTestData(*mode, *dir); err != nil {
|
||||||
|
fmt.Printf("Error migrating test files: %v\n", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
}
|
200
promql/promqltest/test_migrate.go
Normal file
200
promql/promqltest/test_migrate.go
Normal file
@ -0,0 +1,200 @@
|
|||||||
|
// Copyright 2025 The Prometheus 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.
|
||||||
|
|
||||||
|
package promqltest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/grafana/regexp"
|
||||||
|
)
|
||||||
|
|
||||||
|
const defaultTestDataDir = "promql/promqltest/testdata"
|
||||||
|
|
||||||
|
var (
|
||||||
|
evalRegex = regexp.MustCompile(`^(eval |eval_fail |eval_warn |eval_info |eval_ordered )(.*)$`)
|
||||||
|
indentRegex = regexp.MustCompile(`^([ \t]+)\S`)
|
||||||
|
)
|
||||||
|
|
||||||
|
type MigrateMode int
|
||||||
|
|
||||||
|
const (
|
||||||
|
MigrateStrict MigrateMode = iota
|
||||||
|
MigrateBasic
|
||||||
|
MigrateTolerant
|
||||||
|
)
|
||||||
|
|
||||||
|
func ParseMigrateMode(s string) (MigrateMode, error) {
|
||||||
|
switch s {
|
||||||
|
case "strict":
|
||||||
|
return MigrateStrict, nil
|
||||||
|
case "basic":
|
||||||
|
return MigrateBasic, nil
|
||||||
|
case "tolerant":
|
||||||
|
return MigrateTolerant, nil
|
||||||
|
default:
|
||||||
|
return MigrateStrict, fmt.Errorf("invalid mode: %s", s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MigrateTestData migrates all PromQL test files to the new syntax format.
|
||||||
|
// It applies annotation rules based on the provided migration mode ("strict", "basic", or "tolerant").
|
||||||
|
// The function parses each .test file, converts it to the new syntax and overwrites the file.
|
||||||
|
func MigrateTestData(mode, dir string) error {
|
||||||
|
if dir == "" {
|
||||||
|
dir = defaultTestDataDir
|
||||||
|
}
|
||||||
|
|
||||||
|
migrationMode, err := ParseMigrateMode(mode)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to parse mode: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
files, err := os.ReadDir(dir)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to read testdata directory: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
annotationMap := map[MigrateMode]map[string][]string{
|
||||||
|
MigrateStrict: {
|
||||||
|
"eval_fail": {"expect fail", "expect no_warn", "expect no_info"},
|
||||||
|
"eval_warn": {"expect warn", "expect no_info"},
|
||||||
|
"eval_info": {"expect info", "expect no_warn"},
|
||||||
|
"eval_ordered": {"expect ordered", "expect no_warn", "expect no_info"},
|
||||||
|
"eval": {"expect no_warn", "expect no_info"},
|
||||||
|
},
|
||||||
|
MigrateBasic: {
|
||||||
|
"eval_fail": {"expect fail"},
|
||||||
|
"eval_warn": {"expect warn"},
|
||||||
|
"eval_info": {"expect info"},
|
||||||
|
"eval_ordered": {"expect ordered"},
|
||||||
|
},
|
||||||
|
MigrateTolerant: {
|
||||||
|
"eval_fail": {"expect fail"},
|
||||||
|
"eval_ordered": {"expect ordered"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, file := range files {
|
||||||
|
if file.IsDir() || !strings.HasSuffix(file.Name(), ".test") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
path := filepath.Join(dir, file.Name())
|
||||||
|
content, err := os.ReadFile(path)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to read file %s: %w", path, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
lines := strings.Split(string(content), "\n")
|
||||||
|
processedLines, err := processTestFileLines(lines, annotationMap[migrationMode], evalRegex)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error processing file %s: %w", path, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := os.WriteFile(path, []byte(strings.Join(processedLines, "\n")), 0o644); err != nil {
|
||||||
|
return fmt.Errorf("failed to write file %s: %w", path, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func processTestFileLines(
|
||||||
|
lines []string,
|
||||||
|
annotationMap map[string][]string,
|
||||||
|
evalRegex *regexp.Regexp,
|
||||||
|
) (result []string, err error) {
|
||||||
|
for i := 0; i < len(lines); i++ {
|
||||||
|
startLine := lines[i]
|
||||||
|
matches := evalRegex.FindStringSubmatch(strings.TrimSpace(startLine))
|
||||||
|
if matches == nil {
|
||||||
|
result = append(result, startLine)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
var inputBlock []string
|
||||||
|
var outputBlock []string
|
||||||
|
skipBlock := false
|
||||||
|
i++
|
||||||
|
for i < len(lines) {
|
||||||
|
inputBlock = append(inputBlock, lines[i])
|
||||||
|
if strings.HasPrefix(strings.TrimSpace(lines[i]), "expect ") {
|
||||||
|
skipBlock = true
|
||||||
|
}
|
||||||
|
if i+1 < len(lines) && evalRegex.MatchString(strings.TrimSpace(lines[i+1])) {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
i++
|
||||||
|
}
|
||||||
|
|
||||||
|
if skipBlock {
|
||||||
|
result = append(result, startLine)
|
||||||
|
result = append(result, inputBlock...)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get leading whitespace from startLine using indentRegex.
|
||||||
|
leadingWS := ""
|
||||||
|
if indentMatch := indentRegex.FindStringSubmatch(startLine); indentMatch != nil {
|
||||||
|
leadingWS = indentMatch[1]
|
||||||
|
}
|
||||||
|
|
||||||
|
command := strings.TrimSpace(matches[1])
|
||||||
|
expression := matches[2]
|
||||||
|
var annotations []string
|
||||||
|
result = append(result, leadingWS+fmt.Sprintf("eval %s", expression))
|
||||||
|
|
||||||
|
// Detecting indentation style (tab or space) from the first non-empty, indented line.
|
||||||
|
indent := " "
|
||||||
|
for _, line := range inputBlock {
|
||||||
|
if indentMatch := indentRegex.FindStringSubmatch(line); indentMatch != nil {
|
||||||
|
indent = indentMatch[1]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, annotation := range annotationMap[command] {
|
||||||
|
annotations = append(annotations, indent+annotation)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, line := range inputBlock {
|
||||||
|
trimmedLine := strings.TrimSpace(line)
|
||||||
|
switch {
|
||||||
|
case strings.HasPrefix(trimmedLine, "expected_fail_message"):
|
||||||
|
msg := strings.TrimPrefix(trimmedLine, "expected_fail_message ")
|
||||||
|
for j, s := range annotations {
|
||||||
|
if strings.Contains(s, "expect fail") {
|
||||||
|
annotations[j] = indent + fmt.Sprintf("expect fail msg:%s", msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case strings.HasPrefix(trimmedLine, "expected_fail_regexp"):
|
||||||
|
regex := strings.TrimPrefix(trimmedLine, "expected_fail_regexp ")
|
||||||
|
for j, s := range annotations {
|
||||||
|
if strings.Contains(s, "expect fail") {
|
||||||
|
annotations[j] = indent + fmt.Sprintf("expect fail regex:%s", regex)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
outputBlock = append(outputBlock, line)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result = append(result, annotations...)
|
||||||
|
result = append(result, outputBlock...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return result, nil
|
||||||
|
}
|
381
promql/promqltest/test_migrate_test.go
Normal file
381
promql/promqltest/test_migrate_test.go
Normal file
@ -0,0 +1,381 @@
|
|||||||
|
// Copyright 2025 The Prometheus 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.
|
||||||
|
|
||||||
|
package promqltest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func writeTestFile(t *testing.T, dir, content string) string {
|
||||||
|
t.Helper()
|
||||||
|
testFile := filepath.Join(dir, "testcase.test")
|
||||||
|
require.NoError(t, os.WriteFile(testFile, []byte(content), 0o644))
|
||||||
|
return testFile
|
||||||
|
}
|
||||||
|
|
||||||
|
func readTestFile(t *testing.T, path string) string {
|
||||||
|
t.Helper()
|
||||||
|
output, err := os.ReadFile(path)
|
||||||
|
require.NoError(t, err)
|
||||||
|
return string(output)
|
||||||
|
}
|
||||||
|
|
||||||
|
func assertMigration(t *testing.T, mode, input, expected string) {
|
||||||
|
dir := t.TempDir()
|
||||||
|
testFile := writeTestFile(t, dir, input)
|
||||||
|
|
||||||
|
err := MigrateTestData(mode, dir)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
output := readTestFile(t, testFile)
|
||||||
|
require.Equal(t, expected, output)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMigrateTestData_BasicMode(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Basic mode with fail",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with warn",
|
||||||
|
input: `
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
expect warn
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with info",
|
||||||
|
input: `
|
||||||
|
eval_info instant at 3m min(baz)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 3m min(baz)
|
||||||
|
expect info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with ordered",
|
||||||
|
input: `
|
||||||
|
eval_ordered instant at 4m max(qux)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 4m max(qux)
|
||||||
|
expect ordered
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with multiple eval blocks",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something else went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something else went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
expect warn
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with already migrated syntax (no changes)",
|
||||||
|
input: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Basic mode with only comments and whitespace",
|
||||||
|
input: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
assertMigration(t, "basic", tc.input, tc.expected)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMigrateTestData_StrictMode(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Strict mode with fail",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
expect no_warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with warn",
|
||||||
|
input: `
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
expect warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with info",
|
||||||
|
input: `
|
||||||
|
eval_info instant at 3m min(baz)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 3m min(baz)
|
||||||
|
expect info
|
||||||
|
expect no_warn
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with ordered",
|
||||||
|
input: `
|
||||||
|
eval_ordered instant at 4m max(qux)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 4m max(qux)
|
||||||
|
expect ordered
|
||||||
|
expect no_warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with multiple eval blocks",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something else went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something else went wrong
|
||||||
|
expect no_warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
expect warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with already migrated syntax (no changes)",
|
||||||
|
input: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
expect no_warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
expect no_warn
|
||||||
|
expect no_info
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Strict mode with only comments and whitespace",
|
||||||
|
input: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
assertMigration(t, "strict", tc.input, tc.expected)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMigrateTestData_TolerantMode(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
name string
|
||||||
|
input string
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with fail",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with warn",
|
||||||
|
input: `
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with info",
|
||||||
|
input: `
|
||||||
|
eval_info instant at 3m min(baz)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 3m min(baz)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with ordered",
|
||||||
|
input: `
|
||||||
|
eval_ordered instant at 4m max(qux)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 4m max(qux)
|
||||||
|
expect ordered
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with multiple eval blocks",
|
||||||
|
input: `
|
||||||
|
eval_fail instant at 1m sum(foo)
|
||||||
|
expected_fail_message something else went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval_warn instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something else went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
|
||||||
|
eval instant at 2m avg(bar)
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with already migrated syntax (no changes)",
|
||||||
|
input: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
eval instant at 1m sum(foo)
|
||||||
|
expect fail msg:something went wrong
|
||||||
|
{src="a"} 1
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: "Tolerant mode with only comments and whitespace",
|
||||||
|
input: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
expected: `
|
||||||
|
# This is a comment
|
||||||
|
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range tests {
|
||||||
|
t.Run(tc.name, func(t *testing.T) {
|
||||||
|
assertMigration(t, "tolerant", tc.input, tc.expected)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user