All checks were successful
continuous-integration/drone/push Build is passing
See #478
211 lines
5.0 KiB
Go
211 lines
5.0 KiB
Go
package app
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"sort"
|
|
"strings"
|
|
|
|
"coopcloud.tech/abra/cli/internal"
|
|
appPkg "coopcloud.tech/abra/pkg/app"
|
|
"coopcloud.tech/abra/pkg/autocomplete"
|
|
"coopcloud.tech/abra/pkg/client"
|
|
"coopcloud.tech/abra/pkg/config"
|
|
"coopcloud.tech/abra/pkg/formatter"
|
|
"coopcloud.tech/abra/pkg/log"
|
|
abraService "coopcloud.tech/abra/pkg/service"
|
|
stack "coopcloud.tech/abra/pkg/upstream/stack"
|
|
dockerFormatter "github.com/docker/cli/cli/command/formatter"
|
|
containerTypes "github.com/docker/docker/api/types/container"
|
|
"github.com/docker/docker/api/types/filters"
|
|
dockerClient "github.com/docker/docker/client"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var AppPsCommand = &cobra.Command{
|
|
Use: "ps <domain> [flags]",
|
|
Aliases: []string{"p"},
|
|
Short: "Check app deployment status",
|
|
Args: cobra.ExactArgs(1),
|
|
ValidArgsFunction: func(
|
|
cmd *cobra.Command,
|
|
args []string,
|
|
toComplete string) ([]string, cobra.ShellCompDirective) {
|
|
return autocomplete.AppNameComplete()
|
|
},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
app := internal.ValidateApp(args)
|
|
|
|
if err := app.Recipe.Ensure(internal.GetEnsureContext()); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
cl, err := client.New(app.Server)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
deployMeta, err := stack.IsDeployed(context.Background(), cl, app.StackName())
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
if !deployMeta.IsDeployed {
|
|
log.Fatalf("%s is not deployed?", app.Name)
|
|
}
|
|
|
|
chaosVersion := config.CHAOS_DEFAULT
|
|
statuses, err := appPkg.GetAppStatuses([]appPkg.App{app}, true)
|
|
if statusMeta, ok := statuses[app.StackName()]; ok {
|
|
if isChaos, exists := statusMeta["chaos"]; exists && isChaos == "true" {
|
|
if cVersion, exists := statusMeta["chaosVersion"]; exists {
|
|
chaosVersion = cVersion
|
|
if strings.HasSuffix(chaosVersion, config.DIRTY_DEFAULT) {
|
|
chaosVersion = formatter.BoldDirtyDefault(chaosVersion)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
showPSOutput(app, cl, deployMeta.Version, chaosVersion)
|
|
},
|
|
}
|
|
|
|
// showPSOutput renders ps output.
|
|
func showPSOutput(app appPkg.App, cl *dockerClient.Client, deployedVersion, chaosVersion string) {
|
|
composeFiles, err := app.Recipe.GetComposeFiles(app.Env)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
return
|
|
}
|
|
|
|
deployOpts := stack.Deploy{
|
|
Composefiles: composeFiles,
|
|
Namespace: app.StackName(),
|
|
Prune: false,
|
|
ResolveImage: stack.ResolveImageAlways,
|
|
}
|
|
compose, err := appPkg.GetAppComposeConfig(app.Name, deployOpts, app.Env)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
return
|
|
}
|
|
|
|
services := compose.Services
|
|
sort.Slice(services, func(i, j int) bool {
|
|
return services[i].Name < services[j].Name
|
|
})
|
|
|
|
var rows [][]string
|
|
allContainerStats := make(map[string]map[string]string)
|
|
for _, service := range services {
|
|
filters := filters.NewArgs()
|
|
filters.Add("name", fmt.Sprintf("^%s_%s", app.StackName(), service.Name))
|
|
|
|
containers, err := cl.ContainerList(context.Background(), containerTypes.ListOptions{Filters: filters})
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
return
|
|
}
|
|
|
|
var containerStats map[string]string
|
|
if len(containers) == 0 {
|
|
containerStats = map[string]string{
|
|
"version": deployedVersion,
|
|
"chaos": chaosVersion,
|
|
"service": service.Name,
|
|
"image": "unknown",
|
|
"created": "unknown",
|
|
"status": "unknown",
|
|
"state": "unknown",
|
|
"ports": "unknown",
|
|
}
|
|
} else {
|
|
container := containers[0]
|
|
containerStats = map[string]string{
|
|
"version": deployedVersion,
|
|
"chaos": chaosVersion,
|
|
"service": abraService.ContainerToServiceName(container.Names, app.StackName()),
|
|
"image": formatter.RemoveSha(container.Image),
|
|
"created": formatter.HumanDuration(container.Created),
|
|
"status": container.Status,
|
|
"state": container.State,
|
|
"ports": dockerFormatter.DisplayablePorts(container.Ports),
|
|
}
|
|
}
|
|
|
|
allContainerStats[containerStats["service"]] = containerStats
|
|
|
|
// NOTE(d1): don't clobber these variables for --machine output
|
|
dVersion := deployedVersion
|
|
cVersion := chaosVersion
|
|
|
|
if containerStats["service"] != "app" {
|
|
// NOTE(d1): don't repeat info which only relevant for the "app" service
|
|
dVersion = ""
|
|
cVersion = ""
|
|
}
|
|
|
|
row := []string{
|
|
containerStats["service"],
|
|
containerStats["status"],
|
|
containerStats["image"],
|
|
dVersion,
|
|
cVersion,
|
|
}
|
|
|
|
rows = append(rows, row)
|
|
}
|
|
|
|
if internal.MachineReadable {
|
|
rendered, err := json.Marshal(allContainerStats)
|
|
if err != nil {
|
|
log.Fatal("unable to convert to JSON: %s", err)
|
|
}
|
|
|
|
fmt.Println(string(rendered))
|
|
|
|
return
|
|
}
|
|
|
|
table, err := formatter.CreateTable()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
headers := []string{
|
|
"SERVICE",
|
|
"STATUS",
|
|
"IMAGE",
|
|
"VERSION",
|
|
"CHAOS",
|
|
}
|
|
|
|
table.
|
|
Headers(headers...).
|
|
Rows(rows...)
|
|
|
|
if err := formatter.PrintTable(table); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func init() {
|
|
AppPsCommand.Flags().BoolVarP(
|
|
&internal.MachineReadable,
|
|
"machine",
|
|
"m",
|
|
false,
|
|
"print machine-readable output",
|
|
)
|
|
|
|
AppPsCommand.Flags().BoolVarP(
|
|
&internal.Chaos,
|
|
"chaos",
|
|
"C",
|
|
false,
|
|
"ignore uncommitted recipes changes",
|
|
)
|
|
}
|