linkerd2/controller/api/util/api_utils_test.go

249 lines
6.6 KiB
Go

package util
import (
"errors"
"fmt"
"reflect"
"testing"
"github.com/linkerd/linkerd2/pkg/k8s"
pb "github.com/linkerd/linkerd2/viz/metrics-api/gen/viz"
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
k8sError "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/runtime/schema"
)
func TestGRPCError(t *testing.T) {
t.Run("Maps errors to gRPC errors", func(t *testing.T) {
expectations := map[error]error{
nil: nil,
errors.New("normal error"): errors.New("rpc error: code = Unknown desc = normal error"),
status.Error(codes.NotFound, "grpc not found"): errors.New("rpc error: code = NotFound desc = grpc not found"),
k8sError.NewNotFound(schema.GroupResource{Group: "foo", Resource: "bar"}, "http not found"): errors.New("rpc error: code = NotFound desc = bar.foo \"http not found\" not found"),
k8sError.NewServiceUnavailable("unavailable"): errors.New("rpc error: code = Unavailable desc = unavailable"),
k8sError.NewGone("gone"): errors.New("rpc error: code = Internal desc = gone"),
}
for in, out := range expectations {
err := GRPCError(in)
if err != nil || out != nil {
if (err == nil && out != nil) ||
(err != nil && out == nil) ||
(err.Error() != out.Error()) {
t.Fatalf("Expected GRPCError to return [%s], got: [%s]", out, GRPCError(in))
}
}
}
})
}
type resourceExp struct {
namespace string
args []string
resource *pb.Resource
}
func (r *resourceExp) String() string {
return fmt.Sprintf("namespace: %s, args: %s, resource: %s", r.namespace, r.args, r.resource.String())
}
func TestBuildResource(t *testing.T) {
t.Run("Returns expected errors on invalid input", func(t *testing.T) {
msg := "cannot find Kubernetes canonical name from friendly name [invalid]"
expectations := []resourceExp{
{
namespace: "",
args: []string{"invalid"},
},
}
for _, exp := range expectations {
_, err := BuildResource(exp.namespace, exp.args[0])
if err == nil {
t.Fatalf("BuildResource called with invalid resources unexpectedly succeeded, should have returned %s", msg)
}
if err.Error() != msg {
t.Fatalf("BuildResource called with invalid resources should have returned: %s but got unexpected message: %s", msg, err)
}
}
})
t.Run("Correctly parses Kubernetes resources from the command line", func(t *testing.T) {
expectations := []resourceExp{
{
namespace: "test-ns",
args: []string{"deployments"},
resource: &pb.Resource{
Namespace: "test-ns",
Type: k8s.Deployment,
Name: "",
},
},
{
namespace: "",
args: []string{"deploy/foo"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Deployment,
Name: "foo",
},
},
{
namespace: "foo-ns",
args: []string{"po"},
resource: &pb.Resource{
Namespace: "foo-ns",
Type: k8s.Pod,
Name: "",
},
},
{
namespace: "foo-ns",
args: []string{"ns"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Namespace,
Name: "",
},
},
{
namespace: "foo-ns",
args: []string{"ns/foo-ns2"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Namespace,
Name: "foo-ns2",
},
},
}
for _, exp := range expectations {
res, err := BuildResource(exp.namespace, exp.args[0])
if err != nil {
t.Fatalf("Unexpected error from BuildResource(%s) => %s", exp.String(), err)
}
if !reflect.DeepEqual(exp.resource, res) {
t.Fatalf("Expected resource to be [%+v] but was [%+v]", exp.resource, res)
}
}
})
}
func TestBuildResources(t *testing.T) {
t.Run("Rejects duped resources", func(t *testing.T) {
msg := "cannot supply duplicate resources"
expectations := []resourceExp{
{
namespace: "test-ns",
args: []string{"foo", "foo"},
},
{
namespace: "test-ns",
args: []string{"all", "all"},
},
}
for _, exp := range expectations {
_, err := BuildResources(exp.namespace, exp.args)
if err == nil {
t.Fatalf("BuildResources called with duped resources unexpectedly succeeded, should have returned %s", msg)
}
if err.Error() != msg {
t.Fatalf("BuildResources called with duped resources should have returned: %s but got unexpected message: %s", msg, err)
}
}
})
t.Run("Ensures 'all' can't be supplied alongside other resources", func(t *testing.T) {
msg := "'all' can't be supplied alongside other resources"
expectations := []resourceExp{
{
namespace: "test-ns",
args: []string{"po", "foo", "all"},
},
{
namespace: "test-ns",
args: []string{"foo", "all"},
},
{
namespace: "test-ns",
args: []string{"all", "foo"},
},
}
for _, exp := range expectations {
_, err := BuildResources(exp.namespace, exp.args)
if err == nil {
t.Fatalf("BuildResources called with 'all' and another resource unexpectedly succeeded, should have returned %s", msg)
}
if err.Error() != msg {
t.Fatalf("BuildResources called with 'all' and another resource should have returned: %s but got unexpected message: %s", msg, err)
}
}
})
t.Run("Correctly parses Kubernetes resources from the command line", func(t *testing.T) {
expectations := []resourceExp{
{
namespace: "test-ns",
args: []string{"deployments"},
resource: &pb.Resource{
Namespace: "test-ns",
Type: k8s.Deployment,
Name: "",
},
},
{
namespace: "",
args: []string{"deploy/foo"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Deployment,
Name: "foo",
},
},
{
namespace: "foo-ns",
args: []string{"po", "foo"},
resource: &pb.Resource{
Namespace: "foo-ns",
Type: k8s.Pod,
Name: "foo",
},
},
{
namespace: "foo-ns",
args: []string{"ns", "foo-ns2"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Namespace,
Name: "foo-ns2",
},
},
{
namespace: "foo-ns",
args: []string{"ns/foo-ns2"},
resource: &pb.Resource{
Namespace: "",
Type: k8s.Namespace,
Name: "foo-ns2",
},
},
}
for _, exp := range expectations {
res, err := BuildResources(exp.namespace, exp.args)
if err != nil {
t.Fatalf("Unexpected error from BuildResources(%+v) => %s", exp, err)
}
if !reflect.DeepEqual(exp.resource, res[0]) {
t.Fatalf("Expected resource to be [%+v] but was [%+v]", exp.resource, res[0])
}
}
})
}