Skip to content

Commit

Permalink
Implement antctl agent-info
Browse files Browse the repository at this point in the history
  • Loading branch information
lzhecheng committed Feb 5, 2020
1 parent 7600b54 commit bb8aae1
Show file tree
Hide file tree
Showing 6 changed files with 167 additions and 3 deletions.
12 changes: 12 additions & 0 deletions pkg/antctl/antctl.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ import (
"k8s.io/klog"

"github.com/vmware-tanzu/antrea/pkg/antctl/handlers"
"github.com/vmware-tanzu/antrea/pkg/apis/clusterinformation/v1beta1"
"github.com/vmware-tanzu/antrea/pkg/client/clientset/versioned/scheme"
"github.com/vmware-tanzu/antrea/pkg/version"
)
Expand Down Expand Up @@ -76,6 +77,17 @@ var CommandList = &commandList{
CommandGroup: flat,
AddonTransform: versionTransform,
},
{
Use: "agent-info",
Short: "Print agent's basic information",
Long: "Print agent's basic information including version, node subnet, OVS info, AgentConditions, etc.",
HandlerFactory: new(handlers.AgentInfo),
GroupVersion: &systemGroup,
TransformedResponse: reflect.TypeOf(v1beta1.AntreaAgentInfo{}),
Agent: true,
SingleObject: true,
CommandGroup: flat,
},
},
codec: scheme.Codecs,
}
46 changes: 46 additions & 0 deletions pkg/antctl/handlers/agentinfo.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
// Copyright 2020 Antrea 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 handlers

import (
"encoding/json"
"net/http"

"k8s.io/klog"

"github.com/vmware-tanzu/antrea/pkg/apis/clusterinformation/v1beta1"
"github.com/vmware-tanzu/antrea/pkg/monitor"
)

var _ Factory = new(AgentInfo)

// AgentInfo is the implementation of the Factory interface for the agent-info command.
type AgentInfo struct{}

// Handler returns the function which can handle queries issued by agent-info commands,
// the handler function populate component's agent-info to the response.
func (v *AgentInfo) Handler(aq monitor.AgentQuerier, cq monitor.ControllerQuerier) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
var m *v1beta1.AntreaAgentInfo
if aq != nil {
m = aq.GetAgentInfo()
}
err := json.NewEncoder(w).Encode(m)
if err != nil {
w.WriteHeader(http.StatusInternalServerError)
klog.Errorf("Error when encoding AntreaAgentInfo to json: %v", err)
}
}
}
82 changes: 82 additions & 0 deletions pkg/antctl/handlers/agentinfo_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
// Copyright 2020 Antrea 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 handlers

import (
"net/http"
"net/http/httptest"
"testing"

"github.com/vmware-tanzu/antrea/pkg/apis/clusterinformation/v1beta1"
mockmonitor "github.com/vmware-tanzu/antrea/pkg/monitor/testing"

"github.com/golang/mock/gomock"
"github.com/stretchr/testify/assert"
corev1 "k8s.io/api/core/v1"
)

var antreaAgentInfo0 v1beta1.AntreaAgentInfo = v1beta1.AntreaAgentInfo{
Version: "1.0",
PodRef: corev1.ObjectReference{
Kind: "Pod",
Namespace: "kube-system",
Name: "antrea-agent-flx99",
},
NodeRef: corev1.ObjectReference{
Kind: "Node",
Name: "node0-k8s",
},
NodeSubnet: []string{
"192.168.0.0/24",
},
OVSInfo: v1beta1.OVSInfo{},
NetworkPolicyControllerInfo: v1beta1.NetworkPolicyControllerInfo{},
LocalPodNum: 1,
AgentConditions: []v1beta1.AgentCondition{
{
Type: v1beta1.AgentHealthy,
Status: corev1.ConditionTrue,
},
},
}

func TestAgentInfo(t *testing.T) {
ctrl := gomock.NewController(t)
defer ctrl.Finish()

testcases := map[string]struct {
agentInfo *v1beta1.AntreaAgentInfo
expectedOutput string
expectedStatusCode int
}{
"AgentInfo": {
agentInfo: &antreaAgentInfo0,
expectedOutput: "{\"metadata\":{\"creationTimestamp\":null},\"version\":\"1.0\",\"podRef\":{\"kind\":\"Pod\",\"namespace\":\"kube-system\",\"name\":\"antrea-agent-flx99\"},\"nodeRef\":{\"kind\":\"Node\",\"name\":\"node0-k8s\"},\"nodeSubnet\":[\"192.168.0.0/24\"],\"ovsInfo\":{},\"networkPolicyControllerInfo\":{},\"localPodNum\":1,\"agentConditions\":[{\"type\":\"AgentHealthy\",\"status\":\"True\",\"lastHeartbeatTime\":null}]}\n",
expectedStatusCode: http.StatusOK,
},
}
for k, tc := range testcases {
t.Run(k, func(t *testing.T) {
req, err := http.NewRequest("GET", "/", nil)
assert.Nil(t, err)
recorder := httptest.NewRecorder()
aq := mockmonitor.NewMockAgentQuerier(ctrl)
aq.EXPECT().GetAgentInfo().Return(tc.agentInfo)
new(AgentInfo).Handler(aq, nil).ServeHTTP(recorder, req)
assert.Equal(t, tc.expectedStatusCode, recorder.Code, k)
assert.Equal(t, tc.expectedOutput, recorder.Body.String(), k)
})
}
}
10 changes: 7 additions & 3 deletions pkg/monitor/monitor.go
Original file line number Diff line number Diff line change
Expand Up @@ -264,12 +264,12 @@ func (monitor *agentMonitor) getAgentCRD(crdName string) *v1beta1.AntreaAgentInf
return agentCRD
}

func (monitor *agentMonitor) createAgentCRD(crdName string) (*v1beta1.AntreaAgentInfo, error) {
func (monitor *agentMonitor) getAntreaAgentInfo(name string) *v1beta1.AntreaAgentInfo {
ovsVersion := monitor.GetOVSVersion()
ovsConnected := ovsVersion != ""
agentCRD := &v1beta1.AntreaAgentInfo{
return &v1beta1.AntreaAgentInfo{
ObjectMeta: metav1.ObjectMeta{
Name: crdName,
Name: name,
},
Version: version.GetFullVersion(),
PodRef: monitor.GetSelfPod(),
Expand All @@ -280,6 +280,10 @@ func (monitor *agentMonitor) createAgentCRD(crdName string) (*v1beta1.AntreaAgen
LocalPodNum: monitor.GetLocalPodNum(),
AgentConditions: monitor.GetAgentConditions(ovsConnected),
}
}

func (monitor *agentMonitor) createAgentCRD(crdName string) (*v1beta1.AntreaAgentInfo, error) {
agentCRD := monitor.getAntreaAgentInfo(crdName)
klog.V(2).Infof("Creating agent monitoring CRD %v", agentCRD)
return monitor.client.ClusterinformationV1beta1().AntreaAgentInfos().Create(agentCRD)
}
Expand Down
6 changes: 6 additions & 0 deletions pkg/monitor/querier.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ type AgentQuerier interface {
Querier
GetOVSFlowTable() map[string]int32
GetLocalPodNum() int32
GetAgentInfo() *v1beta1.AntreaAgentInfo
}

type ControllerQuerier interface {
Expand Down Expand Up @@ -151,6 +152,11 @@ func (monitor *agentMonitor) GetAgentConditions(ovsConnected bool) []v1beta1.Age
}
}

// GetAgentInfo gets the antrea agent information.
func (monitor *agentMonitor) GetAgentInfo() *v1beta1.AntreaAgentInfo {
return monitor.getAntreaAgentInfo(monitor.GetSelfNode().Name)
}

func (monitor *agentMonitor) GetVersion() string {
return version.GetFullVersion()
}
Expand Down
14 changes: 14 additions & 0 deletions pkg/monitor/testing/mock_monitor.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

0 comments on commit bb8aae1

Please sign in to comment.