Files
I Hsin Cheng 83f9f18776 runsc: Fix newline for cmd usage display
Originally, some of the commands' usage display didn't achieved to show
newline, while some of them does, which are more user-friendly, and
won't confuse the message if there're more which should be in the
nextline.

Fix this issue by returning usage string with a newline.

Signed-off-by: I Hsin Cheng <richard120310@gmail.com>
2024-12-29 03:49:46 +08:00

95 lines
2.6 KiB
Go

// Copyright 2021 The gVisor Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package cmd
import (
"context"
"encoding/json"
"github.com/google/subcommands"
"gvisor.dev/gvisor/runsc/cmd/util"
"gvisor.dev/gvisor/runsc/config"
"gvisor.dev/gvisor/runsc/container"
"gvisor.dev/gvisor/runsc/flag"
)
// Usage implements subcommands.Command for the "usage" command.
type Usage struct {
full bool
fd bool
}
// Name implements subcommands.Command.Name.
func (*Usage) Name() string {
return "usage"
}
// Synopsis implements subcommands.Command.Synopsis.
func (*Usage) Synopsis() string {
return "Usage shows application memory usage across various categories in bytes."
}
// Usage implements subcommands.Command.Usage.
func (*Usage) Usage() string {
return "usage [flags] <container id> - print memory usages to standard output.\n"
}
// SetFlags implements subcommands.Command.SetFlags.
func (u *Usage) SetFlags(f *flag.FlagSet) {
f.BoolVar(&u.full, "full", false, "enumerate all usage by categories")
f.BoolVar(&u.fd, "fd", false, "retrieves a subset of usage through the established usage FD")
}
// Execute implements subcommands.Command.Execute.
func (u *Usage) Execute(_ context.Context, f *flag.FlagSet, args ...any) subcommands.ExitStatus {
if f.NArg() < 1 {
f.Usage()
return subcommands.ExitUsageError
}
id := f.Arg(0)
conf := args[0].(*config.Config)
cont, err := container.Load(conf.RootDir, container.FullID{ContainerID: id}, container.LoadOpts{SkipCheck: true})
if err != nil {
util.Fatalf("loading container: %v", err)
}
if u.fd {
m, err := cont.Sandbox.UsageFD()
if err != nil {
util.Fatalf("usagefd failed: %v", err)
}
mapped, unknown, total, err := m.Fetch()
if err != nil {
util.Fatalf("Fetch memory usage failed: %v", err)
}
util.Infof("Mapped %v, Unknown %v, Total %v\n", mapped, unknown, total)
} else {
m, err := cont.Sandbox.Usage(u.full)
if err != nil {
util.Fatalf("usage failed: %v", err)
}
encoder := json.NewEncoder(&util.Writer{})
encoder.SetIndent("", " ")
if err := encoder.Encode(m); err != nil {
util.Fatalf("Encode MemoryUsage failed: %v", err)
}
}
return subcommands.ExitSuccess
}