package run import ( "context" "fmt" "log" "os" "path" "strconv" "github.com/docker/docker/api/types" "github.com/docker/docker/api/types/filters" "github.com/docker/docker/client" "github.com/mitchellh/go-homedir" "github.com/olekukonko/tablewriter" ) type cluster struct { name string image string status string serverPorts []string server types.Container workers []types.Container } // createDirIfNotExists checks for the existence of a directory and creates it along with all required parents if not. // It returns an error if the directory (or parents) couldn't be created and nil if it worked fine or if the path already exists. func createDirIfNotExists(path string) error { if _, err := os.Stat(path); os.IsNotExist(err) { return os.MkdirAll(path, os.ModePerm) } return nil } // createClusterDir creates a directory with the cluster name under $HOME/.config/k3d/. // The cluster directory will be used e.g. to store the kubeconfig file. func createClusterDir(name string) { clusterPath, _ := getClusterDir(name) if err := createDirIfNotExists(clusterPath); err != nil { log.Fatalf("ERROR: couldn't create cluster directory [%s] -> %+v", clusterPath, err) } } // deleteClusterDir contrary to createClusterDir, this deletes the cluster directory under $HOME/.config/k3d/ func deleteClusterDir(name string) { clusterPath, _ := getClusterDir(name) if err := os.RemoveAll(clusterPath); err != nil { log.Printf("WARNING: couldn't delete cluster directory [%s]. You might want to delete it manually.", clusterPath) } } // getClusterDir returns the path to the cluster directory which is $HOME/.config/k3d/ func getClusterDir(name string) (string, error) { homeDir, err := homedir.Dir() if err != nil { log.Printf("ERROR: Couldn't get user's home directory") return "", err } return path.Join(homeDir, ".config", "k3d", name), nil } // printClusters prints the names of existing clusters func printClusters(all bool) { clusters, err := getClusters() if err != nil { log.Fatalf("ERROR: Couldn't list clusters\n%+v", err) } if len(clusters) == 0 { log.Printf("No clusters found!") return } table := tablewriter.NewWriter(os.Stdout) table.SetHeader([]string{"NAME", "IMAGE", "STATUS", "WORKERS"}) for _, cluster := range clusters { workersRunning := 0 for _, worker := range cluster.workers { if worker.State == "running" { workersRunning++ } } workerData := fmt.Sprintf("%d/%d", workersRunning, len(cluster.workers)) clusterData := []string{cluster.name, cluster.image, cluster.status, workerData} if cluster.status == "running" || all { table.Append(clusterData) } } table.Render() } // getClusters uses the docker API to get existing clusters and compares that with the list of cluster directories func getClusters() (map[string]cluster, error) { ctx := context.Background() docker, err := client.NewEnvClient() if err != nil { return nil, fmt.Errorf("ERROR: couldn't create docker client\n%+v", err) } // Prepare docker label filters filters := filters.NewArgs() filters.Add("label", "app=k3d") filters.Add("label", "component=server") // get all servers created by k3d k3dServers, err := docker.ContainerList(ctx, types.ContainerListOptions{ All: true, Filters: filters, }) if err != nil { return nil, fmt.Errorf("WARNING: couldn't list server containers\n%+v", err) } clusters := make(map[string]cluster) // don't filter for servers but for workers now filters.Del("label", "component=server") filters.Add("label", "component=worker") // for all servers created by k3d, get workers and cluster information for _, server := range k3dServers { filters.Add("label", fmt.Sprintf("cluster=%s", server.Labels["cluster"])) // get workers workers, err := docker.ContainerList(ctx, types.ContainerListOptions{ All: true, Filters: filters, }) if err != nil { log.Printf("WARNING: couldn't get worker containers for cluster %s\n%+v", server.Labels["cluster"], err) } // save cluster information serverPorts := []string{} for _, port := range server.Ports { serverPorts = append(serverPorts, strconv.Itoa(int(port.PublicPort))) } clusters[server.Labels["cluster"]] = cluster{ name: server.Labels["cluster"], image: server.Image, status: server.State, serverPorts: serverPorts, server: server, workers: workers, } // clear label filters before searching for next cluster filters.Del("label", fmt.Sprintf("cluster=%s", server.Labels["cluster"])) } return clusters, nil } // getCluster creates a cluster struct with populated information fields func getCluster(name string) (cluster, error) { clusters, err := getClusters() return clusters[name], err }