Skip to content

Commit

Permalink
[feat] add port for client status
Browse files Browse the repository at this point in the history
Signed-off-by: montaguelhz <[email protected]>
  • Loading branch information
montaguelhz committed Oct 7, 2023
1 parent 061b6e4 commit d0d8e69
Show file tree
Hide file tree
Showing 5 changed files with 105 additions and 2 deletions.
1 change: 1 addition & 0 deletions internal/errno/errno.go
Original file line number Diff line number Diff line change
Expand Up @@ -572,6 +572,7 @@ var (
ERR_INSPECT_CONTAINER_FAILED = EC(630012, "get container low-level information failed")
ERR_GET_CONTAINER_LOGS_FAILED = EC(630013, "get container logs failed")
ERR_UPDATE_CONTAINER_FAILED = EC(630014, "update container failed")
ERR_TOP_CONTAINER_FAILED = EC(630015, "top container failed")

// 690: execuetr task (others)
ERR_START_CRONTAB_IN_CONTAINER_FAILED = EC(690000, "start crontab in container failed")
Expand Down
12 changes: 12 additions & 0 deletions internal/task/step/container.go
Original file line number Diff line number Diff line change
Expand Up @@ -162,6 +162,12 @@ type (
Success *bool
module.ExecOptions
}

TopContainer struct {
ContainerId string
Out *string
module.ExecOptions
}
)

func (s *EngineInfo) Execute(ctx *context.Context) error {
Expand Down Expand Up @@ -337,3 +343,9 @@ func (s *UpdateContainer) Execute(ctx *context.Context) error {
out, err := cli.Execute(s.ExecOptions)
return PostHandle(s.Success, s.Out, out, err, errno.ERR_UPDATE_CONTAINER_FAILED.FD("(%s update ID)", s.ExecWithEngine))
}

func (s *TopContainer) Execute(ctx *context.Context) error {
cli := ctx.Module().DockerCli().TopContainer(s.ContainerId)
out, err := cli.Execute(s.ExecOptions)
return PostHandle(nil, s.Out, out, err, errno.ERR_TOP_CONTAINER_FAILED.FD("(%s top ID)", s.ExecWithEngine))
}
81 changes: 81 additions & 0 deletions internal/task/task/common/client_status.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,8 @@ package common

import (
"fmt"
"regexp"
"strings"

"github.com/opencurve/curveadm/cli/cli"
comm "github.com/opencurve/curveadm/internal/common"
Expand All @@ -33,6 +35,7 @@ import (
"github.com/opencurve/curveadm/internal/task/task"
tui "github.com/opencurve/curveadm/internal/tui/common"
"github.com/opencurve/curveadm/internal/utils"
"github.com/opencurve/curveadm/pkg/module"
)

type (
Expand All @@ -41,6 +44,13 @@ type (
containerId string
status *string
memStorage *utils.SafeMap
port *string
}

step2GetPort struct {
containerId string
port *string
execOptions module.ExecOptions
}

ClientStatus struct {
Expand All @@ -49,6 +59,7 @@ type (
Kind string
ContainerId string
Status string
Port string
AuxInfo string
}
)
Expand All @@ -69,6 +80,7 @@ func setClientStatus(memStorage *utils.SafeMap, id string, status ClientStatus)

func (s *step2FormatClientStatus) Execute(ctx *context.Context) error {
status := *s.status
port := *s.port
if len(status) == 0 { // container losed
status = comm.CLIENT_STATUS_LOSED
}
Expand All @@ -81,11 +93,73 @@ func (s *step2FormatClientStatus) Execute(ctx *context.Context) error {
Kind: client.Kind,
ContainerId: s.containerId,
Status: status,
Port: port,
AuxInfo: client.AuxInfo,
})
return nil
}

func (s *step2GetPort) Execute(ctx *context.Context) error {
cmd := ctx.Module().DockerCli().TopContainer(s.containerId)
out, err := cmd.Execute(s.execOptions)
if err != nil {
return nil
}
lines := strings.Split(out, "\n")
var pid string
if len(lines) > 1 {
reg := regexp.MustCompile(`\s+`)
res := reg.Split(lines[1], -1)
if len(res) > 1 {
pid = res[1]
}
}

if len(pid) == 0 {
return nil
}

// execute "ss" command in container
cli := ctx.Module().Shell().SocketStatistics("")
cli.AddOption("--no-header")
cli.AddOption("--processes")
cli.AddOption("--listening")
command, err := cli.String()
if err != nil {
return nil
}
cmd = ctx.Module().DockerCli().ContainerExec(s.containerId, command)
out, err = cmd.Execute(s.execOptions)
if err != nil {
return nil
}

// handle output
ports := []string{}
lines = strings.Split(out, "\n")
for _, line := range lines {
port := s.extractPort(line, pid)
if len(port) > 0 {
ports = append(ports, port)
}
}
*s.port = strings.Join(ports, ",")

return nil
}

func (s *step2GetPort) extractPort(line, pid string) string {
// e.g: tcp LISTEN 0 128 10.246.159.123:2379 *:* users:(("etcd",pid=7,fd=5))
regex, err := regexp.Compile("^.*:([0-9]+).*pid=" + pid + ".*$")
if err == nil {
mu := regex.FindStringSubmatch(line)
if len(mu) > 1 {
return mu[1]
}
}
return ""
}

func NewGetClientStatusTask(curveadm *cli.CurveAdm, v interface{}) (*task.Task, error) {
client := v.(storage.Client)
hc, err := curveadm.GetHost(client.Host)
Expand All @@ -100,18 +174,25 @@ func NewGetClientStatusTask(curveadm *cli.CurveAdm, v interface{}) (*task.Task,

// add step
var status string
var port string
t.AddStep(&step.ListContainers{
ShowAll: true,
Format: `"{{.Status}}"`,
Filter: fmt.Sprintf("id=%s", containerId),
Out: &status,
ExecOptions: curveadm.ExecOptions(),
})
t.AddStep(&step2GetPort{
containerId: containerId,
port: &port,
execOptions: curveadm.ExecOptions(),
})
t.AddStep(&step2FormatClientStatus{
client: client,
containerId: containerId,
status: &status,
memStorage: curveadm.MemStorage(),
port: &port,
})

return t, nil
Expand Down
6 changes: 4 additions & 2 deletions internal/tui/client/status.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ func statusDecorate(status string) string {
return status
}

func sortStatues(statuses []task.ClientStatus) {
func sortStatuses(statuses []task.ClientStatus) {
sort.Slice(statuses, func(i, j int) bool {
s1, s2 := statuses[i], statuses[j]
if s1.Kind == s2.Kind {
Expand All @@ -59,21 +59,23 @@ func FormatStatus(statuses []task.ClientStatus, verbose bool) string {
"Host",
"Container Id",
"Status",
"Port",
"Aux Info",
}
first, second := tui.FormatTitle(title)
lines = append(lines, first)
lines = append(lines, second)

// status
sortStatues(statuses)
sortStatuses(statuses)
for _, status := range statuses {
lines = append(lines, []interface{}{
status.Id,
status.Kind,
status.Host,
tui.TrimContainerId(status.ContainerId),
tui.DecorateMessage{Message: status.Status, Decorate: statusDecorate},
status.Port,
status.AuxInfo,
})
}
Expand Down
7 changes: 7 additions & 0 deletions pkg/module/docker_cli.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ const (
TEMPLATE_INSPECT_CONTAINER = "{{.engine}} inspect {{.options}} {{.container}}"
TEMPLATE_CONTAINER_LOGS = "{{.engine}} logs {{.options}} {{.container}}"
TEMPLATE_UPDATE_CONTAINER = "{{.engine}} update {{.options}} {{.container}}"
TEMPLATE_TOP_CONTAINER = "{{.engine}} top {{.container}}"
)

type DockerCli struct {
Expand Down Expand Up @@ -168,3 +169,9 @@ func (cli *DockerCli) UpdateContainer(containerId string) *DockerCli {
cli.data["container"] = containerId
return cli
}

func (cli *DockerCli) TopContainer(containerId string) *DockerCli {
cli.tmpl = template.Must(template.New("TopContainer").Parse(TEMPLATE_TOP_CONTAINER))
cli.data["container"] = containerId
return cli
}

0 comments on commit d0d8e69

Please sign in to comment.