2019-04-29 16:27:57 +00:00
|
|
|
package xds
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
2020-07-09 22:04:51 +00:00
|
|
|
"fmt"
|
2019-04-29 16:27:57 +00:00
|
|
|
"io/ioutil"
|
2021-02-24 22:04:10 +00:00
|
|
|
"os"
|
2019-04-29 16:27:57 +00:00
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
envoy "github.com/envoyproxy/go-control-plane/envoy/api/v2"
|
2020-06-23 20:19:56 +00:00
|
|
|
"github.com/golang/protobuf/jsonpb"
|
2020-08-27 17:20:58 +00:00
|
|
|
"github.com/golang/protobuf/proto"
|
2020-07-09 22:04:51 +00:00
|
|
|
"github.com/hashicorp/go-version"
|
2019-04-29 16:27:57 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
// update allows golden files to be updated based on the current output.
|
|
|
|
var update = flag.Bool("update", false, "update golden files")
|
|
|
|
|
2021-02-24 22:04:10 +00:00
|
|
|
// goldenSimple is just for read/write access to a golden file that is not
|
|
|
|
// envoy specific.
|
|
|
|
func goldenSimple(t *testing.T, name, got string) string {
|
|
|
|
return golden(t, name, "", "", got)
|
|
|
|
}
|
|
|
|
|
2020-07-09 22:04:51 +00:00
|
|
|
// goldenEnvoy is a special variant of golden() that silos each named test by
|
|
|
|
// each supported envoy version
|
2021-02-24 22:04:10 +00:00
|
|
|
func goldenEnvoy(t *testing.T, name, envoyVersion, latestEnvoyVersion, got string) string {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
require.NotEmpty(t, envoyVersion)
|
|
|
|
|
|
|
|
// We'll need both the name of this golden file for the requested version
|
|
|
|
// of envoy AND the latest version of envoy due to how the golden file
|
|
|
|
// coalescing works below when there is no xDS generated skew across envoy
|
|
|
|
// versions.
|
|
|
|
subname := goldenEnvoyVersionName(t, envoyVersion)
|
|
|
|
latestSubname := goldenEnvoyVersionName(t, latestEnvoyVersion)
|
|
|
|
|
|
|
|
return golden(t, name, subname, latestSubname, got)
|
|
|
|
}
|
|
|
|
|
|
|
|
func goldenEnvoyVersionName(t *testing.T, envoyVersion string) string {
|
|
|
|
t.Helper()
|
|
|
|
|
2020-07-09 22:04:51 +00:00
|
|
|
require.NotEmpty(t, envoyVersion)
|
|
|
|
|
|
|
|
// We do version sniffing on the complete version, but only generate
|
|
|
|
// golden files ignoring the patch portion
|
|
|
|
version := version.Must(version.NewVersion(envoyVersion))
|
|
|
|
segments := version.Segments()
|
|
|
|
require.Len(t, segments, 3)
|
|
|
|
|
2021-02-24 22:04:10 +00:00
|
|
|
return fmt.Sprintf("envoy-%d-%d-x", segments[0], segments[1])
|
2020-07-09 22:04:51 +00:00
|
|
|
}
|
|
|
|
|
2019-04-29 16:27:57 +00:00
|
|
|
// golden reads and optionally writes the expected data to the golden file,
|
|
|
|
// returning the contents as a string.
|
2021-02-24 22:04:10 +00:00
|
|
|
//
|
|
|
|
// The golden file is named with two components the "name" and the "subname".
|
|
|
|
// In the common case of xDS tests the "name" component is the logical name of
|
|
|
|
// the test itself, and the "subname" is derived from the envoy major version.
|
|
|
|
//
|
|
|
|
// If latestSubname is specified we use that as a fallback source of comparison
|
|
|
|
// if the specific golden file referred to by subname is absent.
|
|
|
|
//
|
|
|
|
// If the -update flag is passed when executing the tests then the contents of
|
|
|
|
// the "got" argument are written to the golden file on disk. If the
|
|
|
|
// latestSubname argument is specified in this mode and the generated content
|
|
|
|
// matches that of the latest generated content then the specific golden file
|
|
|
|
// referred to by 'subname' is deleted to avoid unnecessary duplication in the
|
|
|
|
// testdata directory.
|
|
|
|
func golden(t *testing.T, name, subname, latestSubname, got string) string {
|
2019-04-29 16:27:57 +00:00
|
|
|
t.Helper()
|
|
|
|
|
2020-07-09 22:04:51 +00:00
|
|
|
suffix := ".golden"
|
|
|
|
if subname != "" {
|
|
|
|
suffix = fmt.Sprintf(".%s.golden", subname)
|
|
|
|
}
|
|
|
|
|
|
|
|
golden := filepath.Join("testdata", name+suffix)
|
2021-02-24 22:04:10 +00:00
|
|
|
|
|
|
|
// Always load the latest golden file if configured to do so.
|
|
|
|
latestExpected := ""
|
|
|
|
if latestSubname != "" && subname != latestSubname {
|
|
|
|
latestGolden := filepath.Join("testdata", fmt.Sprintf("%s.%s.golden", name, latestSubname))
|
|
|
|
raw, err := ioutil.ReadFile(latestGolden)
|
2019-04-29 16:27:57 +00:00
|
|
|
require.NoError(t, err)
|
2021-02-24 22:04:10 +00:00
|
|
|
latestExpected = string(raw)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle easy updates to the golden files in the agent/xds/testdata
|
|
|
|
// directory.
|
|
|
|
//
|
|
|
|
// To trim down PRs, we only create per-version golden files if they differ
|
|
|
|
// from the latest version.
|
|
|
|
if *update && got != "" {
|
|
|
|
if latestExpected == got {
|
|
|
|
// In update mode we erase a golden file if it is identical to
|
|
|
|
// the golden file corresponding to the latest version of
|
|
|
|
// envoy.
|
|
|
|
err := os.Remove(golden)
|
|
|
|
if err != nil && !os.IsNotExist(err) {
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
return got
|
|
|
|
}
|
|
|
|
|
|
|
|
require.NoError(t, ioutil.WriteFile(golden, []byte(got), 0644))
|
|
|
|
return got
|
2019-04-29 16:27:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
expected, err := ioutil.ReadFile(golden)
|
2021-02-24 22:04:10 +00:00
|
|
|
if latestExpected != "" && os.IsNotExist(err) {
|
|
|
|
// In readonly mode if a specific golden file isn't found, we fallback
|
|
|
|
// on the latest one.
|
|
|
|
return latestExpected
|
|
|
|
}
|
2019-04-29 16:27:57 +00:00
|
|
|
require.NoError(t, err)
|
2021-02-24 22:04:10 +00:00
|
|
|
return string(expected)
|
|
|
|
}
|
2019-04-29 16:27:57 +00:00
|
|
|
|
2021-02-24 22:04:10 +00:00
|
|
|
func loadTestResource(t *testing.T, name string) string {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
expected, err := ioutil.ReadFile(filepath.Join("testdata", name+".golden"))
|
|
|
|
require.NoError(t, err)
|
2019-04-29 16:27:57 +00:00
|
|
|
return string(expected)
|
|
|
|
}
|
|
|
|
|
|
|
|
func responseToJSON(t *testing.T, r *envoy.DiscoveryResponse) string {
|
2020-08-27 17:20:58 +00:00
|
|
|
return protoToJSON(t, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
func protoToJSON(t *testing.T, pb proto.Message) string {
|
2019-04-29 16:27:57 +00:00
|
|
|
t.Helper()
|
|
|
|
m := jsonpb.Marshaler{
|
|
|
|
Indent: " ",
|
|
|
|
}
|
2020-08-27 17:20:58 +00:00
|
|
|
gotJSON, err := m.MarshalToString(pb)
|
2019-04-29 16:27:57 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
return gotJSON
|
|
|
|
}
|