You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
boundary/internal/cmd/commands/sessionscmd/funcs.go

302 lines
7.5 KiB

// Copyright (c) HashiCorp, Inc.
// SPDX-License-Identifier: BUSL-1.1
package sessionscmd
import (
"fmt"
"net"
"strconv"
"strings"
"time"
"github.com/hashicorp/boundary/api"
"github.com/hashicorp/boundary/api/sessions"
"github.com/hashicorp/boundary/internal/cmd/base"
)
const (
flagIncludeTerminated = "include-terminated"
)
func init() {
extraActionsFlagsMapFunc = extraActionsFlagsMapFuncImpl
extraFlagsFunc = extraFlagsFuncImpl
extraFlagsHandlingFunc = extraFlagsHandlingFuncImpl
executeExtraActions = executeExtraActionsImpl
}
func extraActionsFlagsMapFuncImpl() map[string][]string {
return map[string][]string{
"cancel": {"id"},
"list": {flagIncludeTerminated},
}
}
type extraCmdVars struct {
flagIncludeTerminated bool
}
func extraFlagsFuncImpl(c *Command, set *base.FlagSets, f *base.FlagSet) {
for _, name := range flagsMap[c.Func] {
switch name {
case flagIncludeTerminated:
f.BoolVar(&base.BoolVar{
Name: flagIncludeTerminated,
Target: &c.flagIncludeTerminated,
Usage: "If set, terminated sessions will be included in the results.",
})
}
}
}
func extraFlagsHandlingFuncImpl(c *Command, _ *base.FlagSets, opts *[]sessions.Option) bool {
if c.flagIncludeTerminated {
*opts = append(*opts, sessions.WithIncludeTerminated(c.flagIncludeTerminated))
}
return true
}
func (c *Command) extraHelpFunc(helpMap map[string]func() string) string {
var helpStr string
switch c.Func {
case "":
return base.WrapForHelpText([]string{
"Usage: boundary sessions [sub command] [options] [args]",
"",
" This command allows operations on Boundary sessions.",
"",
" Read a session:",
"",
` $ boundary sessions read -id s_1234567890`,
"",
" Please see the sessions subcommand help for detailed usage information.",
})
case "cancel":
helpStr = base.WrapForHelpText([]string{
"Usage: boundary sessions cancel [options] [args]",
"",
" Cancel the session specified by ID. If the session is already canceled, this command succeeds with no effect. Example:",
"",
` $ boundary sessions cancel -id s_1234567890`,
"",
"",
})
default:
helpStr = helpMap["base"]()
}
return helpStr + c.Flags().Help()
}
func executeExtraActionsImpl(c *Command, origResp *api.Response, origItem *sessions.Session, origItems []*sessions.Session, origError error, sessionClient *sessions.Client, version uint32, opts []sessions.Option) (*api.Response, *sessions.Session, []*sessions.Session, error) {
switch c.Func {
case "cancel":
result, err := sessionClient.Cancel(c.Context, c.FlagId, version, opts...)
if err != nil {
return nil, nil, nil, err
}
return result.GetResponse(), result.GetItem(), nil, err
}
return origResp, origItem, origItems, origError
}
func (c *Command) printListTable(items []*sessions.Session) string {
if len(items) == 0 {
return "No sessions found"
}
var output []string
output = []string{
"",
"Session information:",
}
for i, item := range items {
if i > 0 {
output = append(output, "")
}
if item.Id != "" {
output = append(output,
fmt.Sprintf(" ID: %s", item.Id),
)
} else {
output = append(output,
fmt.Sprintf(" ID: %s", "(not available)"),
)
}
if c.FlagRecursive && item.ScopeId != "" {
output = append(output,
fmt.Sprintf(" Scope ID: %s", item.ScopeId),
)
}
if item.Status != "" {
output = append(output,
fmt.Sprintf(" Status: %s", item.Status),
)
}
if !item.CreatedTime.IsZero() {
output = append(output,
fmt.Sprintf(" Created Time: %s", item.CreatedTime.Local().Format(time.RFC1123)),
)
}
if !item.ExpirationTime.IsZero() {
output = append(output,
fmt.Sprintf(" Expiration Time: %s", item.ExpirationTime.Local().Format(time.RFC1123)),
)
}
if !item.UpdatedTime.IsZero() {
output = append(output,
fmt.Sprintf(" Updated Time: %s", item.UpdatedTime.Local().Format(time.RFC1123)),
)
}
if item.UserId != "" {
output = append(output,
fmt.Sprintf(" User ID: %s", item.UserId),
)
}
if item.TargetId != "" {
output = append(output,
fmt.Sprintf(" Target ID: %s", item.TargetId),
)
}
if len(item.AuthorizedActions) > 0 {
output = append(output,
" Authorized Actions:",
base.WrapSlice(6, item.AuthorizedActions),
)
}
}
return base.WrapForHelpText(output)
}
func printItemTable(item *sessions.Session, resp *api.Response) string {
nonAttributeMap := map[string]any{}
if item.Id != "" {
nonAttributeMap["ID"] = item.Id
}
if item.Version != 0 {
nonAttributeMap["Version"] = item.Version
}
if item.Type != "" {
nonAttributeMap["Type"] = item.Type
}
if !item.CreatedTime.IsZero() {
nonAttributeMap["Created Time"] = item.CreatedTime.Local().Format(time.RFC1123)
}
if !item.UpdatedTime.IsZero() {
nonAttributeMap["Updated Time"] = item.UpdatedTime.Local().Format(time.RFC1123)
}
if !item.ExpirationTime.IsZero() {
nonAttributeMap["Expiration Time"] = item.ExpirationTime.Local().Format(time.RFC1123)
}
if item.TargetId != "" {
nonAttributeMap["Target ID"] = item.TargetId
}
if item.AuthTokenId != "" {
nonAttributeMap["Auth Token ID"] = item.AuthTokenId
}
if item.UserId != "" {
nonAttributeMap["User ID"] = item.UserId
}
if item.HostSetId != "" {
nonAttributeMap["Host Set ID"] = item.HostSetId
}
if item.HostId != "" {
nonAttributeMap["Host ID"] = item.HostId
}
if item.Endpoint != "" {
nonAttributeMap["Endpoint"] = item.Endpoint
}
if item.Status != "" {
nonAttributeMap["Status"] = item.Status
}
if len(strings.TrimSpace(item.TerminationReason)) > 0 {
nonAttributeMap["Termination Reason"] = item.TerminationReason
}
maxLength := base.MaxAttributesLength(nonAttributeMap, nil, nil)
var statesMaps []map[string]any
if len(item.States) > 0 {
for _, state := range item.States {
m := map[string]any{
"Status": state.Status,
"Start Time": state.StartTime.Local().Format(time.RFC1123),
}
if !state.EndTime.IsZero() {
m["End Time"] = state.EndTime.Local().Format(time.RFC1123)
}
statesMaps = append(statesMaps, m)
}
if l := len("Start Time"); l > maxLength {
maxLength = l
}
}
var connectionsMaps []map[string]any
for _, sc := range item.Connections {
cm := map[string]any{
"Client Address": net.JoinHostPort(sc.ClientTcpAddress, strconv.FormatUint(uint64(sc.ClientTcpPort), 10)),
"Endpoint Address": net.JoinHostPort(sc.EndpointTcpAddress, strconv.FormatUint(uint64(sc.EndpointTcpPort), 10)),
"Bytes Up": sc.BytesUp,
"Bytes Down": sc.BytesDown,
}
if len(sc.ClosedReason) != 0 {
cm["Closed Reason"] = sc.ClosedReason
}
connectionsMaps = append(connectionsMaps, cm)
}
ret := []string{
"",
"Session information:",
base.WrapMap(2, maxLength+2, nonAttributeMap),
}
if item.Scope != nil {
ret = append(ret,
"",
" Scope:",
base.ScopeInfoForOutput(item.Scope, maxLength),
)
}
if len(item.AuthorizedActions) > 0 {
ret = append(ret,
"",
" Authorized Actions:",
base.WrapSlice(4, item.AuthorizedActions),
)
}
if len(item.States) > 0 {
ret = append(ret,
"",
" States:",
)
for _, m := range statesMaps {
ret = append(ret,
base.WrapMap(4, maxLength, m),
"",
)
}
}
if len(item.Connections) > 0 {
ret = append(ret,
"",
" Connections:",
)
for _, c := range connectionsMaps {
ret = append(ret,
base.WrapMap(4, maxLength, c),
"",
)
}
}
return base.WrapForHelpText(ret)
}