mirror of
https://gitee.com/XM-GO/PandaX.git
synced 2026-04-24 03:18:35 +08:00
148 lines
3.4 KiB
Go
148 lines
3.4 KiB
Go
package event
|
|
|
|
import (
|
|
"context"
|
|
v1 "k8s.io/api/core/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/client-go/kubernetes"
|
|
"pandax/apps/devops/services/k8s/common"
|
|
"strings"
|
|
)
|
|
|
|
func GetClusterNodeEvent(client *kubernetes.Clientset, namespace string, field string) (*v1.EventList, error) {
|
|
|
|
events, err := client.CoreV1().Events(namespace).List(context.TODO(),
|
|
metav1.ListOptions{
|
|
FieldSelector: field,
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return events, nil
|
|
}
|
|
|
|
// GetPodsEventWarnings returns warning pod events by filtering out events targeting only given pods
|
|
func GetPodsEventWarnings(events []v1.Event, pods []v1.Pod) []common.Event {
|
|
result := make([]common.Event, 0)
|
|
|
|
// Filter out only warning events
|
|
events = getWarningEvents(events)
|
|
failedPods := make([]v1.Pod, 0)
|
|
|
|
// Filter out ready and successful pods
|
|
for _, pod := range pods {
|
|
if !isReadyOrSucceeded(pod) {
|
|
failedPods = append(failedPods, pod)
|
|
}
|
|
}
|
|
|
|
// Filter events by failed pods UID
|
|
events = filterEventsByPodsUID(events, failedPods)
|
|
events = removeDuplicates(events)
|
|
|
|
for _, event := range events {
|
|
result = append(result, common.Event{
|
|
Message: event.Message,
|
|
Reason: event.Reason,
|
|
Type: event.Type,
|
|
})
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// Returns filtered list of event objects.
|
|
// Event list object is filtered to get only warning events.
|
|
func getWarningEvents(events []v1.Event) []v1.Event {
|
|
return filterEventsByType(FillEventsType(events), v1.EventTypeWarning)
|
|
}
|
|
|
|
// Returns true if given pod is in state ready or succeeded, false otherwise
|
|
func isReadyOrSucceeded(pod v1.Pod) bool {
|
|
if pod.Status.Phase == v1.PodSucceeded {
|
|
return true
|
|
}
|
|
if pod.Status.Phase == v1.PodRunning {
|
|
for _, c := range pod.Status.Conditions {
|
|
if c.Type == v1.PodReady {
|
|
if c.Status == v1.ConditionFalse {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// Returns filtered list of event objects. Events list is filtered to get only events targeting
|
|
// pods on the list.
|
|
func filterEventsByPodsUID(events []v1.Event, pods []v1.Pod) []v1.Event {
|
|
result := make([]v1.Event, 0)
|
|
podEventMap := make(map[types.UID]bool, 0)
|
|
|
|
if len(pods) == 0 || len(events) == 0 {
|
|
return result
|
|
}
|
|
|
|
for _, pod := range pods {
|
|
podEventMap[pod.UID] = true
|
|
}
|
|
|
|
for _, event := range events {
|
|
if _, exists := podEventMap[event.InvolvedObject.UID]; exists {
|
|
result = append(result, event)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// Removes duplicate strings from the slice
|
|
func removeDuplicates(slice []v1.Event) []v1.Event {
|
|
visited := make(map[string]bool, 0)
|
|
result := make([]v1.Event, 0)
|
|
|
|
for _, elem := range slice {
|
|
if !visited[elem.Reason] {
|
|
visited[elem.Reason] = true
|
|
result = append(result, elem)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// Filters kubernetes API event objects based on event type.
|
|
// Empty string will return all events.
|
|
func filterEventsByType(events []v1.Event, eventType string) []v1.Event {
|
|
if len(eventType) == 0 || len(events) == 0 {
|
|
return events
|
|
}
|
|
|
|
result := make([]v1.Event, 0)
|
|
for _, event := range events {
|
|
if event.Type == eventType {
|
|
result = append(result, event)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
// Returns true if reason string contains any partial string indicating that this may be a
|
|
// warning, false otherwise
|
|
func isFailedReason(reason string, partials ...string) bool {
|
|
for _, partial := range partials {
|
|
if strings.Contains(strings.ToLower(reason), partial) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|