Skip to content

Commit

Permalink
Add validator interface. Add resource interface and change pod.go to …
Browse files Browse the repository at this point in the history
…implement to Validator interface
  • Loading branch information
tejal29 committed Feb 27, 2020
1 parent 3d55c47 commit b0d31ab
Show file tree
Hide file tree
Showing 4 changed files with 182 additions and 82 deletions.
64 changes: 44 additions & 20 deletions pkg/diag/validator/pod.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ limitations under the License.
package validator

import (
"context"
"fmt"

v1 "k8s.io/api/core/v1"
Expand All @@ -36,36 +37,59 @@ var (
waitingContainerStatus = getWaitingContainerStatus
)

type PodStatus struct {
phase string
err *PodErr
// PodValidator implements the Validator interface for Pods
type PodValidator struct {
k kubernetes.Interface
}

type PodErr struct {
// NewPodValidator initializes a PodValidator
func NewPodValidator(k kubernetes.Interface) *PodValidator {
return &PodValidator{k: k}
}

// Run implements the Run method for Validator interface
func (p *PodValidator) Run(ctx context.Context, ns string, opts meta_v1.ListOptions) ([]Resource, error) {
pods, err := p.k.CoreV1().Pods(ns).List(opts)
if err != nil {
return nil, err
}
rs := []Resource{}
for _, po := range pods.Items {
ps := p.getPodStatus(&po)
rs = append(rs, NewResourceFromObject(&po, Status(ps.phase), ps.reason.String()))
}
return rs, nil
}

type podStatus struct {
phase string
reason *podReason
}

type podReason struct {
reason string
message string
}

func (e *PodErr) Error() string {
return fmt.Sprintf("pod in error due to reason: %s, message: %s", e.reason, e.message)
func (r *podReason) String() string {
if r == nil {
return ""
}
return fmt.Sprintf("pod unstable due to reason: %s, message: %s", r.reason, r.message)
}

func GetPodDetails(client kubernetes.Interface, ns string, podName string) PodStatus {
pod, err := client.CoreV1().Pods(ns).Get(podName, meta_v1.GetOptions{})
if err != nil {
return PodStatus{err: &PodErr{message: err.Error()}}
}
func (p *PodValidator) getPodStatus(pod *v1.Pod) podStatus {
switch pod.Status.Phase {
case v1.PodSucceeded:
return PodStatus{phase: success}
return podStatus{phase: success}
case v1.PodRunning:
return PodStatus{phase: running}
return podStatus{phase: running}
default:
return getPendingDetails(pod)
}
}

func getPendingDetails(pod *v1.Pod) PodStatus {
func getPendingDetails(pod *v1.Pod) podStatus {
// See https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#pod-conditions
for _, c := range pod.Status.Conditions {
switch c.Status {
Expand All @@ -91,20 +115,20 @@ func getWaitingContainerStatus(cs []v1.ContainerStatus) (string, string) {
return success, success
}

func newPendingStatus(r string, d string) PodStatus {
return PodStatus{
func newPendingStatus(r string, d string) podStatus {
return podStatus{
phase: pending,
err: &PodErr{
reason: &podReason{
reason: r,
message: d,
},
}
}

func newUnknownStatus() PodStatus {
return PodStatus{
func newUnknownStatus() podStatus {
return podStatus{
phase: unknown,
err: &PodErr{
reason: &podReason{
reason: unknown,
message: unknown,
},
Expand Down
96 changes: 34 additions & 62 deletions pkg/diag/validator/pod_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,44 +17,37 @@ limitations under the License.
package validator

import (
"context"
"testing"

"github.com/google/go-cmp/cmp"
v1 "k8s.io/api/core/v1"
meta_v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
fakekubeclientset "k8s.io/client-go/kubernetes/fake"

"github.com/GoogleContainerTools/skaffold/testutil"
)

func TestGetPodDetails(t *testing.T) {
func TestRun(t *testing.T) {
tests := []struct {
description string
pod v1.Pod
name string
expected PodStatus
shouldErr bool
pods []*v1.Pod
expected []Resource
}{
{
description: "pod does not exist",
pod: v1.Pod{
description: "pod don't exist in test namespace",
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
},
Namespace: "foo-ns",
}},
},
name: "not-there",
expected: PodStatus{
err: &PodErr{
message: "pods \"not-there\" not found",
},
},
shouldErr: true,
expected: []Resource{},
},
{
description: "pod is Waiting conditions with reason and message",
name: "foo",
pod: v1.Pod{
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
Expand All @@ -74,20 +67,13 @@ func TestGetPodDetails(t *testing.T) {
},
},
},
},
shouldErr: true,
expected: PodStatus{
phase: pending,
err: &PodErr{
reason: "ErrImgPull",
message: "could not pull the container image",
},
},
}},
expected: []Resource{NewResource("test", "", "foo", "Pending",
"pod unstable due to reason: ErrImgPull, message: could not pull the container image")},
},
{
description: "pod is Waiting conditions with reason but no message",
name: "foo",
pod: v1.Pod{
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
Expand All @@ -106,19 +92,12 @@ func TestGetPodDetails(t *testing.T) {
},
},
},
},
shouldErr: true,
expected: PodStatus{
phase: pending,
err: &PodErr{
reason: "Unschedulable",
},
},
}},
expected: []Resource{NewResource("test", "", "foo", "Pending", "pod unstable due to reason: Unschedulable, message: ")},
},
{
description: "pod is in Terminated State",
name: "foo",
pod: v1.Pod{
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
Expand All @@ -127,15 +106,12 @@ func TestGetPodDetails(t *testing.T) {
Phase: v1.PodSucceeded,
Conditions: []v1.PodCondition{{Status: v1.ConditionTrue}},
},
},
expected: PodStatus{
phase: "Succeeded",
},
}},
expected: []Resource{NewResource("test", "", "foo", "Succeeded", "")},
},
{
description: "pod is in Stable State",
name: "foo",
pod: v1.Pod{
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
Expand All @@ -150,15 +126,12 @@ func TestGetPodDetails(t *testing.T) {
},
},
},
},
expected: PodStatus{
phase: running,
},
}},
expected: []Resource{NewResource("test", "", "foo", "Running", "")},
},
{
description: "pod condition unknown",
name: "foo",
pod: v1.Pod{
pods: []*v1.Pod{{
ObjectMeta: meta_v1.ObjectMeta{
Name: "foo",
Namespace: "test",
Expand All @@ -170,22 +143,21 @@ func TestGetPodDetails(t *testing.T) {
Message: "could not determine",
}},
},
},
expected: PodStatus{
phase: pending,
err: &PodErr{
reason: "Unknown",
message: "could not determine",
},
},
}},
expected: []Resource{NewResource("test", "", "foo", "Pending", "pod unstable due to reason: Unknown, message: could not determine")},
},
}

for _, test := range tests {
testutil.Run(t, test.description, func(t *testutil.T) {
client := fakekubeclientset.NewSimpleClientset(&test.pod)
actual := GetPodDetails(client, "test", test.name)
t.CheckDeepEqual(test.expected, actual, cmp.AllowUnexported(PodStatus{}, PodErr{}))
rs := make([]runtime.Object, len(test.pods))
for i, p := range test.pods {
rs[i] = p
}
f := fakekubeclientset.NewSimpleClientset(rs...)
actual, err := NewPodValidator(f).Run(context.Background(), "test", meta_v1.ListOptions{})
t.CheckNoError(err)
t.CheckDeepEqual(test.expected, actual, cmp.AllowUnexported(resource{}))
})
}
}
Expand Down
73 changes: 73 additions & 0 deletions pkg/diag/validator/resource.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
/*
Copyright 2020 The Skaffold 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 validator

import (
"fmt"

meta_v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
)

type Resource interface {
// Namespace of the resource.
Namespace() string

// Kind of resource, e.g., service, pod, deployment
Kind() string
// Name of the resource, e.g., cluster name, node name, persistent volume name, etc.
Name() string

// Status if resource is healthy or not
Status() Status

// Reason if resource is not stable.
Reason() string
}

type resource struct {
namespace string
kind string
name string
reason string
status Status
}

func (r *resource) Kind() string { return r.kind }
func (r *resource) Name() string { return r.name }
func (r *resource) Reason() string { return r.reason }
func (r *resource) Namespace() string { return r.namespace }
func (r *resource) Status() Status { return r.status }
func (r *resource) String() string {
return fmt.Sprintf("{%s:%s/%s}", r.kind, r.namespace, r.name)
}

// NewResource creates new resource of kind
func NewResource(namespace, kind, name string, status Status, reason string) Resource {
return &resource{namespace: namespace, kind: kind, name: name, status: status, reason: reason}
}

// objectWithMetadata is any k8s object that has kind and object metadata.
type objectWithMetadata interface {
runtime.Object
meta_v1.Object
}

// NewResourceFromObject creates new resource with fields populated from object metadata.
func NewResourceFromObject(object objectWithMetadata, status Status, reason string) Resource {
return NewResource(object.GetNamespace(), object.GetObjectKind().GroupVersionKind().Kind, object.GetName(), status, reason)
}
31 changes: 31 additions & 0 deletions pkg/diag/validator/validator.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
/*
Copyright 2020 The Skaffold 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 validator

import (
"context"

metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"
)

type Status string

type Validator interface {
// Run runs the validator and returns the list of resources with status.
Run(ctx context.Context, client kubernetes.Interface, ns string, opts metav1.ListOptions) ([]Resource, error)
}

0 comments on commit b0d31ab

Please sign in to comment.