2024-01-18 16:14:33 -05:00
|
|
|
package scheduler_k3s
|
|
|
|
|
|
|
|
|
|
import (
|
2024-10-17 00:12:48 -04:00
|
|
|
"bufio"
|
2024-01-18 16:14:33 -05:00
|
|
|
"context"
|
|
|
|
|
"errors"
|
2024-01-20 04:58:02 -05:00
|
|
|
"fmt"
|
2024-10-17 00:12:48 -04:00
|
|
|
"io"
|
|
|
|
|
"os"
|
|
|
|
|
"os/signal"
|
|
|
|
|
"strconv"
|
|
|
|
|
"strings"
|
|
|
|
|
"syscall"
|
2024-01-18 16:14:33 -05:00
|
|
|
|
2024-01-20 04:58:02 -05:00
|
|
|
"github.com/dokku/dokku/plugins/common"
|
2024-10-17 00:12:48 -04:00
|
|
|
"github.com/fatih/color"
|
2024-01-20 04:58:02 -05:00
|
|
|
"github.com/go-openapi/jsonpointer"
|
2024-02-28 17:05:08 -05:00
|
|
|
kedav1alpha1 "github.com/kedacore/keda/v2/apis/keda/v1alpha1"
|
2024-01-18 16:14:33 -05:00
|
|
|
appsv1 "k8s.io/api/apps/v1"
|
|
|
|
|
autoscalingv1 "k8s.io/api/autoscaling/v1"
|
|
|
|
|
batchv1 "k8s.io/api/batch/v1"
|
|
|
|
|
v1 "k8s.io/api/core/v1"
|
2024-03-12 03:55:40 -04:00
|
|
|
networkingv1 "k8s.io/api/networking/v1"
|
2024-01-18 16:14:33 -05:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
2024-01-20 04:58:02 -05:00
|
|
|
"k8s.io/apimachinery/pkg/types"
|
2024-02-28 17:05:08 -05:00
|
|
|
"k8s.io/client-go/dynamic"
|
2024-01-18 16:14:33 -05:00
|
|
|
"k8s.io/client-go/kubernetes"
|
2024-10-17 00:12:48 -04:00
|
|
|
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
|
2024-01-18 16:14:33 -05:00
|
|
|
"k8s.io/client-go/rest"
|
|
|
|
|
"k8s.io/client-go/tools/clientcmd"
|
|
|
|
|
clientcmdapi "k8s.io/client-go/tools/clientcmd/api"
|
2024-10-17 00:12:48 -04:00
|
|
|
"k8s.io/client-go/tools/remotecommand"
|
|
|
|
|
"k8s.io/kubectl/pkg/util/term"
|
2024-01-18 16:14:33 -05:00
|
|
|
"k8s.io/utils/ptr"
|
|
|
|
|
)
|
|
|
|
|
|
2024-02-14 03:50:15 -05:00
|
|
|
func getKubeconfigPath() string {
|
|
|
|
|
return common.PropertyGetDefault("scheduler-k3s", "--global", "kubeconfig-path", KubeConfigPath)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func getKubeContext() string {
|
|
|
|
|
return common.PropertyGetDefault("scheduler-k3s", "--global", "kube-context", DefaultKubeContext)
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// KubernetesClient is a wrapper around the Kubernetes client
|
|
|
|
|
type KubernetesClient struct {
|
|
|
|
|
// Client is the Kubernetes client
|
|
|
|
|
Client kubernetes.Clientset
|
|
|
|
|
|
2024-02-28 17:05:08 -05:00
|
|
|
// DynamicClient is the Kubernetes dynamic client
|
|
|
|
|
DynamicClient dynamic.Interface
|
|
|
|
|
|
2024-02-14 03:50:15 -05:00
|
|
|
// KubeConfigPath is the path to the Kubernetes config
|
|
|
|
|
KubeConfigPath string
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// RestClient is the Kubernetes REST client
|
|
|
|
|
RestClient rest.Interface
|
|
|
|
|
|
|
|
|
|
// RestConfig is the Kubernetes REST config
|
|
|
|
|
RestConfig rest.Config
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewKubernetesClient creates a new Kubernetes client
|
|
|
|
|
func NewKubernetesClient() (KubernetesClient, error) {
|
2024-02-14 03:50:15 -05:00
|
|
|
kubeconfigPath := getKubeconfigPath()
|
|
|
|
|
kubeContext := getKubeContext()
|
|
|
|
|
clientConfig := KubernetesClientConfig(kubeconfigPath, kubeContext)
|
2024-01-18 16:14:33 -05:00
|
|
|
restConf, err := clientConfig.ClientConfig()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return KubernetesClient{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
restConf.GroupVersion = &schema.GroupVersion{
|
|
|
|
|
Group: "api",
|
|
|
|
|
Version: "v1",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
client, err := kubernetes.NewForConfig(restConf)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return KubernetesClient{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
restConf.NegotiatedSerializer = runtime.NewSimpleNegotiatedSerializer(runtime.SerializerInfo{})
|
|
|
|
|
|
|
|
|
|
restClient, err := rest.RESTClientFor(restConf)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return KubernetesClient{}, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-28 17:05:08 -05:00
|
|
|
dynamicClient, err := dynamic.NewForConfig(restConf)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return KubernetesClient{}, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
return KubernetesClient{
|
2024-02-14 03:50:15 -05:00
|
|
|
Client: *client,
|
2024-02-28 17:05:08 -05:00
|
|
|
DynamicClient: dynamicClient,
|
2024-02-14 03:50:15 -05:00
|
|
|
KubeConfigPath: kubeconfigPath,
|
|
|
|
|
RestConfig: *restConf,
|
|
|
|
|
RestClient: restClient,
|
2024-01-18 16:14:33 -05:00
|
|
|
}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// KubernetesClientConfig returns a Kubernetes client config
|
2024-02-14 03:50:15 -05:00
|
|
|
func KubernetesClientConfig(kubeconfigPath string, kubecontext string) clientcmd.ClientConfig {
|
|
|
|
|
configOverrides := clientcmd.ConfigOverrides{ClusterInfo: clientcmdapi.Cluster{Server: ""}}
|
|
|
|
|
if kubecontext != "" {
|
|
|
|
|
configOverrides.CurrentContext = kubecontext
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
return clientcmd.NewNonInteractiveDeferredLoadingClientConfig(
|
2024-02-14 03:50:15 -05:00
|
|
|
&clientcmd.ClientConfigLoadingRules{ExplicitPath: kubeconfigPath},
|
|
|
|
|
&configOverrides,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k KubernetesClient) Ping() error {
|
|
|
|
|
_, err := k.Client.Discovery().ServerVersion()
|
|
|
|
|
return err
|
2024-01-18 16:14:33 -05:00
|
|
|
}
|
|
|
|
|
|
2024-01-20 06:57:20 -05:00
|
|
|
// AnnotateNodeInput contains all the information needed to annotates a Kubernetes node
|
|
|
|
|
type AnnotateNodeInput struct {
|
|
|
|
|
// Name is the Kubernetes node name
|
|
|
|
|
Name string
|
|
|
|
|
// Key is the annotation key
|
|
|
|
|
Key string
|
|
|
|
|
// Value is the annotation value
|
|
|
|
|
Value string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// AnnotateNode annotates a Kubernetes node
|
|
|
|
|
func (k KubernetesClient) AnnotateNode(ctx context.Context, input AnnotateNodeInput) error {
|
|
|
|
|
node, err := k.Client.CoreV1().Nodes().Get(ctx, input.Name, metav1.GetOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if node == nil {
|
|
|
|
|
return errors.New("node is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyPath := fmt.Sprintf("/metadata/annotations/%s", jsonpointer.Escape(input.Key))
|
|
|
|
|
patch := fmt.Sprintf(`[{"op":"add", "path":"%s", "value":"%s" }]`, keyPath, input.Value)
|
|
|
|
|
_, err = k.Client.CoreV1().Nodes().Patch(ctx, node.Name, types.JSONPatchType, []byte(patch), metav1.PatchOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("failed to annotate node: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-22 01:25:01 -05:00
|
|
|
type ApplyKubernetesManifestInput struct {
|
|
|
|
|
// Manifest is the path to the Kubernetes manifest
|
|
|
|
|
Manifest string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k KubernetesClient) ApplyKubernetesManifest(ctx context.Context, input ApplyKubernetesManifestInput) error {
|
2024-02-14 03:50:15 -05:00
|
|
|
args := []string{
|
|
|
|
|
"apply",
|
|
|
|
|
"-f",
|
|
|
|
|
input.Manifest,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if kubeContext := getKubeContext(); kubeContext != "" {
|
|
|
|
|
args = append([]string{"--context", kubeContext}, args...)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if kubeconfigPath := getKubeconfigPath(); kubeconfigPath != "" {
|
|
|
|
|
args = append([]string{"--kubeconfig", kubeconfigPath}, args...)
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-22 01:25:01 -05:00
|
|
|
upgradeCmd, err := common.CallExecCommand(common.ExecCommandInput{
|
2024-02-14 03:50:15 -05:00
|
|
|
Command: "kubectl",
|
|
|
|
|
Args: args,
|
2024-01-22 01:25:01 -05:00
|
|
|
StreamStdio: true,
|
|
|
|
|
})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("Unable to call kubectl command: %w", err)
|
|
|
|
|
}
|
|
|
|
|
if upgradeCmd.ExitCode != 0 {
|
|
|
|
|
return fmt.Errorf("Invalid exit code from kubectl command: %d", upgradeCmd.ExitCode)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// CreateJobInput contains all the information needed to create a Kubernetes job
|
|
|
|
|
type CreateJobInput struct {
|
|
|
|
|
// Job is the Kubernetes job
|
|
|
|
|
Job batchv1.Job
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CreateJob creates a Kubernetes job
|
|
|
|
|
func (k KubernetesClient) CreateJob(ctx context.Context, input CreateJobInput) (batchv1.Job, error) {
|
|
|
|
|
job, err := k.Client.BatchV1().Jobs(input.Namespace).Create(ctx, &input.Job, metav1.CreateOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return batchv1.Job{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if job == nil {
|
|
|
|
|
return batchv1.Job{}, errors.New("job is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return *job, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CreateNamespaceInput contains all the information needed to create a Kubernetes namespace
|
|
|
|
|
type CreateNamespaceInput struct {
|
2024-01-22 05:12:49 -05:00
|
|
|
// Name is the name of the Kubernetes namespace
|
|
|
|
|
Name v1.Namespace
|
2024-01-18 16:14:33 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// CreateNamespace creates a Kubernetes namespace
|
|
|
|
|
func (k KubernetesClient) CreateNamespace(ctx context.Context, input CreateNamespaceInput) (v1.Namespace, error) {
|
2024-01-22 05:12:49 -05:00
|
|
|
namespaces, err := k.ListNamespaces(ctx)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return v1.Namespace{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, namespace := range namespaces {
|
|
|
|
|
if namespace.Name == input.Name.Name {
|
|
|
|
|
return namespace, nil
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace, err := k.Client.CoreV1().Namespaces().Create(ctx, &input.Name, metav1.CreateOptions{})
|
2024-01-18 16:14:33 -05:00
|
|
|
if err != nil {
|
|
|
|
|
return v1.Namespace{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if namespace == nil {
|
|
|
|
|
return v1.Namespace{}, errors.New("namespace is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return *namespace, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 03:55:40 -04:00
|
|
|
// DeleteIngressInput contains all the information needed to delete a Kubernetes ingress
|
|
|
|
|
type DeleteIngressInput struct {
|
|
|
|
|
// Name is the Kubernetes ingress name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DeleteIngress deletes a Kubernetes ingress
|
|
|
|
|
func (k KubernetesClient) DeleteIngress(ctx context.Context, input DeleteIngressInput) error {
|
|
|
|
|
return k.Client.NetworkingV1().Ingresses(input.Namespace).Delete(ctx, input.Name, metav1.DeleteOptions{})
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// DeleteJobInput contains all the information needed to delete a Kubernetes job
|
|
|
|
|
type DeleteJobInput struct {
|
|
|
|
|
// Name is the Kubernetes job name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DeleteJob deletes a Kubernetes job
|
|
|
|
|
func (k KubernetesClient) DeleteJob(ctx context.Context, input DeleteJobInput) error {
|
|
|
|
|
return k.Client.BatchV1().Jobs(input.Namespace).Delete(ctx, input.Name, metav1.DeleteOptions{
|
|
|
|
|
PropagationPolicy: ptr.To(metav1.DeletePropagationForeground),
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-21 23:48:38 -05:00
|
|
|
// DeleteNodeInput contains all the information needed to delete a Kubernetes node
|
|
|
|
|
type DeleteNodeInput struct {
|
|
|
|
|
// Name is the Kubernetes node name
|
|
|
|
|
Name string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DeleteNode deletes a Kubernetes node
|
|
|
|
|
func (k KubernetesClient) DeleteNode(ctx context.Context, input DeleteNodeInput) error {
|
|
|
|
|
return k.Client.CoreV1().Nodes().Delete(ctx, input.Name, metav1.DeleteOptions{})
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// DeleteSecretInput contains all the information needed to delete a Kubernetes secret
|
|
|
|
|
type DeleteSecretInput struct {
|
|
|
|
|
// Name is the Kubernetes secret name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// DeleteSecret deletes a Kubernetes secret
|
|
|
|
|
func (k KubernetesClient) DeleteSecret(ctx context.Context, input DeleteSecretInput) error {
|
|
|
|
|
return k.Client.CoreV1().Secrets(input.Namespace).Delete(ctx, input.Name, metav1.DeleteOptions{})
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-17 00:12:48 -04:00
|
|
|
// ExecCommandInput contains all the information needed to execute a command in a Kubernetes pod
|
|
|
|
|
type ExecCommandInput struct {
|
|
|
|
|
// Command is the command to execute
|
|
|
|
|
Command []string
|
|
|
|
|
|
|
|
|
|
// ContainerName is the Kubernetes container name
|
|
|
|
|
ContainerName string
|
|
|
|
|
|
|
|
|
|
// Entrypoint is the command entrypoint
|
|
|
|
|
Entrypoint string
|
|
|
|
|
|
|
|
|
|
// Name is the Kubernetes pod name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
2024-10-21 22:08:53 -04:00
|
|
|
|
|
|
|
|
// Stderr is the error writer
|
|
|
|
|
Stderr io.Writer
|
|
|
|
|
|
|
|
|
|
// Stdout is the output writer
|
|
|
|
|
Stdout io.Writer
|
2024-10-17 00:12:48 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ExecCommand executes a command in a Kubernetes pod
|
|
|
|
|
func (k KubernetesClient) ExecCommand(ctx context.Context, input ExecCommandInput) error {
|
|
|
|
|
coreclient, err := corev1client.NewForConfig(&k.RestConfig)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("Error creating corev1 client: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
req := coreclient.RESTClient().Post().
|
|
|
|
|
Resource("pods").
|
|
|
|
|
Namespace(input.Namespace).
|
|
|
|
|
Name(input.Name).
|
|
|
|
|
SubResource("exec")
|
|
|
|
|
|
|
|
|
|
req.Param("container", input.ContainerName)
|
|
|
|
|
req.Param("stdin", "true")
|
|
|
|
|
req.Param("stdout", "true")
|
|
|
|
|
req.Param("stderr", "true")
|
|
|
|
|
req.Param("tty", "true")
|
|
|
|
|
|
|
|
|
|
if input.Entrypoint != "" {
|
|
|
|
|
req.Param("command", input.Entrypoint)
|
|
|
|
|
}
|
|
|
|
|
for _, cmd := range input.Command {
|
|
|
|
|
req.Param("command", cmd)
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-21 22:08:53 -04:00
|
|
|
var stdout io.Writer
|
|
|
|
|
if input.Stdout == nil {
|
|
|
|
|
stdout = os.Stdout
|
|
|
|
|
} else {
|
|
|
|
|
stdout = input.Stdout
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var stderr io.Writer
|
|
|
|
|
if input.Stderr == nil {
|
|
|
|
|
stderr = os.Stderr
|
|
|
|
|
} else {
|
|
|
|
|
stderr = input.Stderr
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-17 00:12:48 -04:00
|
|
|
t := term.TTY{
|
|
|
|
|
In: os.Stdin,
|
2024-10-21 22:08:53 -04:00
|
|
|
Out: stdout,
|
2024-10-17 00:12:48 -04:00
|
|
|
Raw: true,
|
|
|
|
|
}
|
2024-10-21 22:08:53 -04:00
|
|
|
|
2024-10-17 00:12:48 -04:00
|
|
|
size := t.GetSize()
|
|
|
|
|
sizeQueue := t.MonitorSize(size)
|
|
|
|
|
|
|
|
|
|
return t.Safe(func() error {
|
|
|
|
|
exec, err := remotecommand.NewSPDYExecutor(&k.RestConfig, "POST", req.URL())
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("Error creating executor: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return exec.StreamWithContext(ctx, remotecommand.StreamOptions{
|
|
|
|
|
Stdin: os.Stdin,
|
2024-10-21 22:08:53 -04:00
|
|
|
Stdout: stdout,
|
|
|
|
|
Stderr: stderr,
|
2024-10-17 00:12:48 -04:00
|
|
|
Tty: true,
|
|
|
|
|
TerminalSizeQueue: sizeQueue,
|
|
|
|
|
})
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-21 23:48:38 -05:00
|
|
|
// GetNodeInput contains all the information needed to get a Kubernetes node
|
|
|
|
|
type GetNodeInput struct {
|
|
|
|
|
// Name is the Kubernetes node name
|
|
|
|
|
Name string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetNode gets a Kubernetes node
|
|
|
|
|
func (k KubernetesClient) GetNode(ctx context.Context, input GetNodeInput) (Node, error) {
|
|
|
|
|
if input.Name == "" {
|
|
|
|
|
return Node{}, errors.New("node name is required")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
node, err := k.Client.CoreV1().Nodes().Get(ctx, input.Name, metav1.GetOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return Node{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if node == nil {
|
|
|
|
|
return Node{}, errors.New("node is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return kubernetesNodeToNode(*node), err
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// GetJobInput contains all the information needed to get a Kubernetes job
|
|
|
|
|
type GetPodInput struct {
|
|
|
|
|
// Name is the Kubernetes pod name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetJob gets a Kubernetes job
|
|
|
|
|
func (k KubernetesClient) GetPod(ctx context.Context, input GetPodInput) (v1.Pod, error) {
|
|
|
|
|
pod, err := k.Client.CoreV1().Pods(input.Namespace).Get(ctx, input.Name, metav1.GetOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return v1.Pod{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if pod == nil {
|
|
|
|
|
return v1.Pod{}, errors.New("pod is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return *pod, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-20 04:58:02 -05:00
|
|
|
// LabelNodeInput contains all the information needed to label a Kubernetes node
|
|
|
|
|
type LabelNodeInput struct {
|
|
|
|
|
// Name is the Kubernetes node name
|
|
|
|
|
Name string
|
|
|
|
|
// Key is the label key
|
|
|
|
|
Key string
|
|
|
|
|
// Value is the label value
|
|
|
|
|
Value string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// LabelNode labels a Kubernetes node
|
|
|
|
|
func (k KubernetesClient) LabelNode(ctx context.Context, input LabelNodeInput) error {
|
|
|
|
|
node, err := k.Client.CoreV1().Nodes().Get(ctx, input.Name, metav1.GetOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if node == nil {
|
|
|
|
|
return errors.New("node is nil")
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-20 06:57:20 -05:00
|
|
|
keyPath := fmt.Sprintf("/metadata/labels/%s", jsonpointer.Escape(input.Key))
|
|
|
|
|
patch := fmt.Sprintf(`[{"op":"add", "path":"%s", "value":"%s" }]`, keyPath, input.Value)
|
|
|
|
|
_, err = k.Client.CoreV1().Nodes().Patch(ctx, node.Name, types.JSONPatchType, []byte(patch), metav1.PatchOptions{})
|
2024-01-20 04:58:02 -05:00
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("failed to label node: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-28 17:05:08 -05:00
|
|
|
// ListClusterTriggerAuthenticationsInput contains all the information needed to list Kubernetes trigger authentications
|
|
|
|
|
type ListClusterTriggerAuthenticationsInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListClusterTriggerAuthentications lists Kubernetes trigger authentications
|
|
|
|
|
func (k KubernetesClient) ListClusterTriggerAuthentications(ctx context.Context, input ListClusterTriggerAuthenticationsInput) ([]kedav1alpha1.ClusterTriggerAuthentication, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{LabelSelector: input.LabelSelector}
|
|
|
|
|
|
|
|
|
|
gvr := schema.GroupVersionResource{
|
|
|
|
|
Group: "keda.sh",
|
|
|
|
|
Version: "v1alpha1",
|
|
|
|
|
Resource: "clustertriggerauthentications",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
response, err := k.DynamicClient.Resource(gvr).Namespace(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []kedav1alpha1.ClusterTriggerAuthentication{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
triggerAuthentications := []kedav1alpha1.ClusterTriggerAuthentication{}
|
|
|
|
|
for _, triggerAuthentication := range response.Items {
|
|
|
|
|
var ta kedav1alpha1.ClusterTriggerAuthentication
|
|
|
|
|
err := runtime.DefaultUnstructuredConverter.FromUnstructured(triggerAuthentication.Object, &ta)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []kedav1alpha1.ClusterTriggerAuthentication{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
triggerAuthentications = append(triggerAuthentications, ta)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return triggerAuthentications, nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:51:33 -05:00
|
|
|
// ListCronJobsInput contains all the information needed to list Kubernetes cron jobs
|
|
|
|
|
type ListCronJobsInput struct {
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListCronJobs lists Kubernetes cron jobs
|
|
|
|
|
func (k KubernetesClient) ListCronJobs(ctx context.Context, input ListCronJobsInput) ([]batchv1.CronJob, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{}
|
|
|
|
|
if input.LabelSelector != "" {
|
|
|
|
|
listOptions.LabelSelector = input.LabelSelector
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cronJobs, err := k.Client.BatchV1().CronJobs(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []batchv1.CronJob{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if cronJobs == nil {
|
|
|
|
|
return []batchv1.CronJob{}, errors.New("cron jobs is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return cronJobs.Items, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// ListDeploymentsInput contains all the information needed to list Kubernetes deployments
|
|
|
|
|
type ListDeploymentsInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListDeployments lists Kubernetes deployments
|
|
|
|
|
func (k KubernetesClient) ListDeployments(ctx context.Context, input ListDeploymentsInput) ([]appsv1.Deployment, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{LabelSelector: input.LabelSelector}
|
|
|
|
|
deployments, err := k.Client.AppsV1().Deployments(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []appsv1.Deployment{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if deployments == nil {
|
|
|
|
|
return []appsv1.Deployment{}, errors.New("deployments is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return deployments.Items, nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-12 03:55:40 -04:00
|
|
|
// ListIngressesInput contains all the information needed to list Kubernetes ingresses
|
|
|
|
|
type ListIngressesInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListIngresses lists Kubernetes ingresses
|
|
|
|
|
func (k KubernetesClient) ListIngresses(ctx context.Context, input ListIngressesInput) ([]networkingv1.Ingress, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{LabelSelector: input.LabelSelector}
|
|
|
|
|
ingresses, err := k.Client.NetworkingV1().Ingresses(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []networkingv1.Ingress{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ingresses == nil {
|
|
|
|
|
return []networkingv1.Ingress{}, errors.New("ingresses is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ingresses.Items, nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// ListNamespaces lists Kubernetes namespaces
|
|
|
|
|
func (k KubernetesClient) ListNamespaces(ctx context.Context) ([]v1.Namespace, error) {
|
|
|
|
|
namespaces, err := k.Client.CoreV1().Namespaces().List(ctx, metav1.ListOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []v1.Namespace{}, err
|
|
|
|
|
}
|
|
|
|
|
if namespaces == nil {
|
|
|
|
|
return []v1.Namespace{}, errors.New("namespaces is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return namespaces.Items, nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-20 04:58:02 -05:00
|
|
|
// ListNodesInput contains all the information needed to list Kubernetes nodes
|
|
|
|
|
type ListNodesInput struct {
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListNodes lists Kubernetes nodes
|
|
|
|
|
func (k KubernetesClient) ListNodes(ctx context.Context, input ListNodesInput) ([]v1.Node, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{}
|
|
|
|
|
if input.LabelSelector != "" {
|
|
|
|
|
common.LogDebug(fmt.Sprintf("Using label selector: %s", input.LabelSelector))
|
|
|
|
|
listOptions.LabelSelector = input.LabelSelector
|
|
|
|
|
}
|
|
|
|
|
nodeList, err := k.Client.CoreV1().Nodes().List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []v1.Node{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if nodeList == nil {
|
|
|
|
|
return []v1.Node{}, errors.New("pod list is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nodeList.Items, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// ListPodsInput contains all the information needed to list Kubernetes pods
|
|
|
|
|
type ListPodsInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListPods lists Kubernetes pods
|
|
|
|
|
func (k KubernetesClient) ListPods(ctx context.Context, input ListPodsInput) ([]v1.Pod, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{LabelSelector: input.LabelSelector}
|
|
|
|
|
podList, err := k.Client.CoreV1().Pods(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []v1.Pod{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if podList == nil {
|
|
|
|
|
return []v1.Pod{}, errors.New("pod list is nil")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return podList.Items, err
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-28 17:05:08 -05:00
|
|
|
// ListTriggerAuthenticationsInput contains all the information needed to list Kubernetes trigger authentications
|
|
|
|
|
type ListTriggerAuthenticationsInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// LabelSelector is the Kubernetes label selector
|
|
|
|
|
LabelSelector string
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ListTriggerAuthentications lists Kubernetes trigger authentications
|
|
|
|
|
func (k KubernetesClient) ListTriggerAuthentications(ctx context.Context, input ListTriggerAuthenticationsInput) ([]kedav1alpha1.TriggerAuthentication, error) {
|
|
|
|
|
listOptions := metav1.ListOptions{LabelSelector: input.LabelSelector}
|
|
|
|
|
|
|
|
|
|
gvr := schema.GroupVersionResource{
|
|
|
|
|
Group: "keda.sh",
|
|
|
|
|
Version: "v1alpha1",
|
|
|
|
|
Resource: "triggerauthentications",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
response, err := k.DynamicClient.Resource(gvr).Namespace(input.Namespace).List(ctx, listOptions)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []kedav1alpha1.TriggerAuthentication{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
triggerAuthentications := []kedav1alpha1.TriggerAuthentication{}
|
|
|
|
|
for _, triggerAuthentication := range response.Items {
|
|
|
|
|
var ta kedav1alpha1.TriggerAuthentication
|
|
|
|
|
err := runtime.DefaultUnstructuredConverter.FromUnstructured(triggerAuthentication.Object, &ta)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return []kedav1alpha1.TriggerAuthentication{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
triggerAuthentications = append(triggerAuthentications, ta)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return triggerAuthentications, nil
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-18 16:14:33 -05:00
|
|
|
// ScaleDeploymentInput contains all the information needed to scale a Kubernetes deployment
|
|
|
|
|
type ScaleDeploymentInput struct {
|
|
|
|
|
// Name is the Kubernetes deployment name
|
|
|
|
|
Name string
|
|
|
|
|
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// Replicas is the number of replicas to scale to
|
|
|
|
|
Replicas int32
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ScaleDeployment scales a Kubernetes deployment
|
|
|
|
|
func (k KubernetesClient) ScaleDeployment(ctx context.Context, input ScaleDeploymentInput) error {
|
|
|
|
|
_, err := k.Client.AppsV1().Deployments(input.Namespace).UpdateScale(ctx, input.Name, &autoscalingv1.Scale{
|
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
|
Name: input.Name,
|
|
|
|
|
Namespace: input.Namespace,
|
|
|
|
|
},
|
|
|
|
|
Spec: autoscalingv1.ScaleSpec{
|
|
|
|
|
Replicas: input.Replicas,
|
|
|
|
|
},
|
|
|
|
|
}, metav1.UpdateOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|
2024-10-17 00:12:48 -04:00
|
|
|
|
|
|
|
|
type StreamLogsInput struct {
|
|
|
|
|
// Namespace is the Kubernetes namespace
|
|
|
|
|
Namespace string
|
|
|
|
|
|
|
|
|
|
// DeploymentName is the Kubernetes deployment name
|
|
|
|
|
DeploymentName string
|
|
|
|
|
|
|
|
|
|
// ContainerName is the Kubernetes container name
|
|
|
|
|
ContainerName string
|
|
|
|
|
|
|
|
|
|
// Follow is whether to follow the logs
|
|
|
|
|
Follow bool
|
|
|
|
|
|
|
|
|
|
// TailLines is the number of lines to tail
|
|
|
|
|
TailLines int64
|
|
|
|
|
|
|
|
|
|
// Quiet is whether to suppress output
|
|
|
|
|
Quiet bool
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (k KubernetesClient) StreamLogs(ctx context.Context, input StreamLogsInput) error {
|
|
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
|
|
|
signals := make(chan os.Signal, 1)
|
|
|
|
|
signal.Notify(signals, os.Interrupt, syscall.SIGHUP,
|
|
|
|
|
syscall.SIGINT,
|
|
|
|
|
syscall.SIGQUIT,
|
|
|
|
|
syscall.SIGTERM)
|
|
|
|
|
go func() {
|
|
|
|
|
<-signals
|
|
|
|
|
cancel()
|
|
|
|
|
}()
|
|
|
|
|
|
|
|
|
|
if err := k.Ping(); err != nil {
|
|
|
|
|
return fmt.Errorf("kubernetes api not available: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
labelSelector := []string{fmt.Sprintf("app.kubernetes.io/part-of=%s", input.DeploymentName)}
|
|
|
|
|
processIndex := 0
|
|
|
|
|
if input.ContainerName != "" {
|
|
|
|
|
parts := strings.SplitN(input.ContainerName, ".", 2)
|
|
|
|
|
if len(parts) == 2 {
|
|
|
|
|
var err error
|
|
|
|
|
input.ContainerName = parts[0]
|
|
|
|
|
processIndex, err = strconv.Atoi(parts[1])
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("Error parsing process index: %w", err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
labelSelector = append(labelSelector, fmt.Sprintf("app.kubernetes.io/name=%s", input.ContainerName))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pods, err := k.ListPods(ctx, ListPodsInput{
|
|
|
|
|
Namespace: input.Namespace,
|
|
|
|
|
LabelSelector: strings.Join(labelSelector, ","),
|
|
|
|
|
})
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("Error listing pods: %w", err)
|
|
|
|
|
}
|
|
|
|
|
if len(pods) == 0 {
|
|
|
|
|
return fmt.Errorf("No pods found for app %s", input.DeploymentName)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ch := make(chan bool)
|
|
|
|
|
|
|
|
|
|
if os.Getenv("FORCE_TTY") == "1" {
|
|
|
|
|
color.NoColor = false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
colors := []color.Attribute{
|
|
|
|
|
color.FgRed,
|
|
|
|
|
color.FgYellow,
|
|
|
|
|
color.FgGreen,
|
|
|
|
|
color.FgCyan,
|
|
|
|
|
color.FgBlue,
|
|
|
|
|
color.FgMagenta,
|
|
|
|
|
}
|
|
|
|
|
// colorIndex := 0
|
|
|
|
|
for i := 0; i < len(pods); i++ {
|
|
|
|
|
if processIndex > 0 && i != (processIndex-1) {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logOptions := v1.PodLogOptions{
|
|
|
|
|
Follow: input.Follow,
|
|
|
|
|
}
|
|
|
|
|
if input.TailLines > 0 {
|
|
|
|
|
logOptions.TailLines = ptr.To(input.TailLines)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
podColor := colors[i%len(colors)]
|
|
|
|
|
dynoText := color.New(podColor).SprintFunc()
|
|
|
|
|
podName := pods[i].Name
|
|
|
|
|
podLogs, err := k.Client.CoreV1().Pods(input.Namespace).GetLogs(podName, &logOptions).Stream(ctx)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
buffer := bufio.NewReader(podLogs)
|
|
|
|
|
go func(ctx context.Context, buffer *bufio.Reader, prettyText func(a ...interface{}) string, ch chan bool) {
|
|
|
|
|
defer func() {
|
|
|
|
|
ch <- true
|
|
|
|
|
}()
|
|
|
|
|
for {
|
|
|
|
|
select {
|
|
|
|
|
case <-ctx.Done(): // if cancel() execute
|
|
|
|
|
ch <- true
|
|
|
|
|
return
|
|
|
|
|
default:
|
|
|
|
|
str, readErr := buffer.ReadString('\n')
|
|
|
|
|
if readErr == io.EOF {
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if str == "" {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if !input.Quiet {
|
|
|
|
|
str = fmt.Sprintf("%s %s", dynoText(fmt.Sprintf("app[%s]:", podName)), str)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_, err := fmt.Print(str)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}(ctx, buffer, dynoText, ch)
|
|
|
|
|
}
|
|
|
|
|
<-ch
|
|
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
}
|