package cli import ( "fmt" "strings" "time" "github.com/google/uuid" "github.com/jedib0t/go-pretty/v6/table" "github.com/spf13/cobra" "github.com/coder/coder/cli/cliui" "github.com/coder/coder/coderd/database" "github.com/coder/coder/codersdk" ) func list() *cobra.Command { var ( columns []string ) cmd := &cobra.Command{ Annotations: workspaceCommand, Use: "list", Short: "List all workspaces", Aliases: []string{"ls"}, RunE: func(cmd *cobra.Command, args []string) error { client, err := createClient(cmd) if err != nil { return err } workspaces, err := client.WorkspacesByUser(cmd.Context(), codersdk.Me) if err != nil { return err } if len(workspaces) == 0 { _, _ = fmt.Fprintln(cmd.OutOrStdout(), cliui.Styles.Prompt.String()+"No workspaces found! Create one:") _, _ = fmt.Fprintln(cmd.OutOrStdout()) _, _ = fmt.Fprintln(cmd.OutOrStdout(), " "+cliui.Styles.Code.Render("coder create ")) _, _ = fmt.Fprintln(cmd.OutOrStdout()) return nil } users, err := client.Users(cmd.Context(), codersdk.UsersRequest{}) if err != nil { return err } usersByID := map[uuid.UUID]codersdk.User{} for _, user := range users { usersByID[user.ID] = user } tableWriter := cliui.Table() header := table.Row{"workspace", "template", "status", "last built", "outdated"} tableWriter.AppendHeader(header) tableWriter.SortBy([]table.SortBy{{ Name: "workspace", }}) tableWriter.SetColumnConfigs(cliui.FilterTableColumns(header, columns)) for _, workspace := range workspaces { status := "" inProgress := false if workspace.LatestBuild.Job.Status == codersdk.ProvisionerJobRunning || workspace.LatestBuild.Job.Status == codersdk.ProvisionerJobCanceling { inProgress = true } switch workspace.LatestBuild.Transition { case database.WorkspaceTransitionStart: status = "Running" if inProgress { status = "Starting" } case database.WorkspaceTransitionStop: status = "Stopped" if inProgress { status = "Stopping" } case database.WorkspaceTransitionDelete: status = "Deleted" if inProgress { status = "Deleting" } } if workspace.LatestBuild.Job.Status == codersdk.ProvisionerJobFailed { status = "Failed" } duration := time.Now().UTC().Sub(workspace.LatestBuild.Job.CreatedAt).Truncate(time.Second) if duration > time.Hour { duration = duration.Truncate(time.Hour) } if duration > time.Minute { duration = duration.Truncate(time.Minute) } days := 0 for duration.Hours() > 24 { days++ duration -= 24 * time.Hour } durationDisplay := duration.String() if days > 0 { durationDisplay = fmt.Sprintf("%dd%s", days, durationDisplay) } if strings.HasSuffix(durationDisplay, "m0s") { durationDisplay = durationDisplay[:len(durationDisplay)-2] } if strings.HasSuffix(durationDisplay, "h0m") { durationDisplay = durationDisplay[:len(durationDisplay)-2] } user := usersByID[workspace.OwnerID] tableWriter.AppendRow(table.Row{ user.Username + "/" + workspace.Name, workspace.TemplateName, status, durationDisplay, workspace.Outdated, }) } _, err = fmt.Fprintln(cmd.OutOrStdout(), tableWriter.Render()) return err }, } cmd.Flags().StringArrayVarP(&columns, "column", "c", nil, "Specify a column to filter in the table.") return cmd }