//go:build linux // +build linux package libpod import ( "bufio" "bytes" "errors" "fmt" "io/ioutil" "math" "os" "runtime" "strconv" "strings" "syscall" "time" "github.com/containers/buildah" "github.com/containers/image/v5/pkg/sysregistriesv2" "github.com/containers/podman/v4/libpod/define" "github.com/containers/podman/v4/libpod/linkmode" "github.com/containers/podman/v4/pkg/rootless" "github.com/containers/storage" "github.com/containers/storage/pkg/system" "github.com/sirupsen/logrus" ) // Info returns the store and host information func (r *Runtime) info() (*define.Info, error) { info := define.Info{} versionInfo, err := define.GetVersion() if err != nil { return nil, fmt.Errorf("error getting version info: %w", err) } info.Version = versionInfo // get host information hostInfo, err := r.hostInfo() if err != nil { return nil, fmt.Errorf("error getting host info: %w", err) } info.Host = hostInfo // get store information storeInfo, err := r.storeInfo() if err != nil { return nil, fmt.Errorf("error getting store info: %w", err) } info.Store = storeInfo registries := make(map[string]interface{}) sys := r.SystemContext() data, err := sysregistriesv2.GetRegistries(sys) if err != nil { return nil, fmt.Errorf("error getting registries: %w", err) } for _, reg := range data { registries[reg.Prefix] = reg } regs, err := sysregistriesv2.UnqualifiedSearchRegistries(sys) if err != nil { return nil, fmt.Errorf("error getting registries: %w", err) } if len(regs) > 0 { registries["search"] = regs } volumePlugins := make([]string, 0, len(r.config.Engine.VolumePlugins)+1) // the local driver always exists volumePlugins = append(volumePlugins, "local") for plugin := range r.config.Engine.VolumePlugins { volumePlugins = append(volumePlugins, plugin) } info.Plugins.Volume = volumePlugins info.Plugins.Network = r.network.Drivers() info.Plugins.Log = logDrivers info.Registries = registries return &info, nil } // top-level "host" info func (r *Runtime) hostInfo() (*define.HostInfo, error) { // lets say OS, arch, number of cpus, amount of memory, maybe os distribution/version, hostname, kernel version, uptime mi, err := system.ReadMemInfo() if err != nil { return nil, fmt.Errorf("error reading memory info: %w", err) } hostDistributionInfo := r.GetHostDistributionInfo() kv, err := readKernelVersion() if err != nil { return nil, fmt.Errorf("error reading kernel version: %w", err) } host, err := os.Hostname() if err != nil { return nil, fmt.Errorf("error getting hostname: %w", err) } cpuUtil, err := getCPUUtilization() if err != nil { return nil, err } info := define.HostInfo{ Arch: runtime.GOARCH, BuildahVersion: buildah.Version, Linkmode: linkmode.Linkmode(), CPUs: runtime.NumCPU(), CPUUtilization: cpuUtil, Distribution: hostDistributionInfo, LogDriver: r.config.Containers.LogDriver, EventLogger: r.eventer.String(), Hostname: host, Kernel: kv, MemFree: mi.MemFree, MemTotal: mi.MemTotal, NetworkBackend: r.config.Network.NetworkBackend, OS: runtime.GOOS, SwapFree: mi.SwapFree, SwapTotal: mi.SwapTotal, } if err := r.setPlatformHostInfo(&info); err != nil { return nil, err } conmonInfo, ociruntimeInfo, err := r.defaultOCIRuntime.RuntimeInfo() if err != nil { logrus.Errorf("Getting info on OCI runtime %s: %v", r.defaultOCIRuntime.Name(), err) } else { info.Conmon = conmonInfo info.OCIRuntime = ociruntimeInfo } duration, err := procUptime() if err != nil { return nil, fmt.Errorf("error reading up time: %w", err) } uptime := struct { hours float64 minutes float64 seconds float64 }{ hours: duration.Truncate(time.Hour).Hours(), minutes: duration.Truncate(time.Minute).Minutes(), seconds: duration.Truncate(time.Second).Seconds(), } // Could not find a humanize-formatter for time.Duration var buffer bytes.Buffer buffer.WriteString(fmt.Sprintf("%.0fh %.0fm %.2fs", uptime.hours, math.Mod(uptime.seconds, 3600)/60, math.Mod(uptime.seconds, 60), )) if int64(uptime.hours) > 0 { buffer.WriteString(fmt.Sprintf(" (Approximately %.2f days)", uptime.hours/24)) } info.Uptime = buffer.String() return &info, nil } func (r *Runtime) getContainerStoreInfo() (define.ContainerStore, error) { var paused, running, stopped int cs := define.ContainerStore{} cons, err := r.GetAllContainers() if err != nil { return cs, err } cs.Number = len(cons) for _, con := range cons { state, err := con.State() if err != nil { if errors.Is(err, define.ErrNoSuchCtr) { // container was probably removed cs.Number-- continue } return cs, err } switch state { case define.ContainerStateRunning: running++ case define.ContainerStatePaused: paused++ default: stopped++ } } cs.Paused = paused cs.Stopped = stopped cs.Running = running return cs, nil } // top-level "store" info func (r *Runtime) storeInfo() (*define.StoreInfo, error) { // lets say storage driver in use, number of images, number of containers configFile, err := storage.DefaultConfigFile(rootless.IsRootless()) if err != nil { return nil, err } images, err := r.store.Images() if err != nil { return nil, fmt.Errorf("error getting number of images: %w", err) } conInfo, err := r.getContainerStoreInfo() if err != nil { return nil, err } imageInfo := define.ImageStore{Number: len(images)} var grStats syscall.Statfs_t if err := syscall.Statfs(r.store.GraphRoot(), &grStats); err != nil { return nil, fmt.Errorf("unable to collect graph root usasge for %q: %w", r.store.GraphRoot(), err) } allocated := uint64(grStats.Bsize) * grStats.Blocks info := define.StoreInfo{ ImageStore: imageInfo, ImageCopyTmpDir: os.Getenv("TMPDIR"), ContainerStore: conInfo, GraphRoot: r.store.GraphRoot(), GraphRootAllocated: allocated, GraphRootUsed: allocated - (uint64(grStats.Bsize) * grStats.Bfree), RunRoot: r.store.RunRoot(), GraphDriverName: r.store.GraphDriverName(), GraphOptions: nil, VolumePath: r.config.Engine.VolumePath, ConfigFile: configFile, } graphOptions := map[string]interface{}{} for _, o := range r.store.GraphOptions() { split := strings.SplitN(o, "=", 2) if strings.HasSuffix(split[0], "mount_program") { version, err := programVersion(split[1]) if err != nil { logrus.Warnf("Failed to retrieve program version for %s: %v", split[1], err) } program := map[string]interface{}{} program["Executable"] = split[1] program["Version"] = version program["Package"] = packageVersion(split[1]) graphOptions[split[0]] = program } else { graphOptions[split[0]] = split[1] } } info.GraphOptions = graphOptions statusPairs, err := r.store.Status() if err != nil { return nil, err } status := map[string]string{} for _, pair := range statusPairs { status[pair[0]] = pair[1] } info.GraphStatus = status return &info, nil } func readKernelVersion() (string, error) { buf, err := ioutil.ReadFile("/proc/version") if err != nil { return "", err } f := bytes.Fields(buf) if len(f) < 3 { return string(bytes.TrimSpace(buf)), nil } return string(f[2]), nil } func procUptime() (time.Duration, error) { var zero time.Duration buf, err := ioutil.ReadFile("/proc/uptime") if err != nil { return zero, err } f := bytes.Fields(buf) if len(f) < 1 { return zero, errors.New("unable to parse uptime from /proc/uptime") } return time.ParseDuration(string(f[0]) + "s") } // GetHostDistributionInfo returns a map containing the host's distribution and version func (r *Runtime) GetHostDistributionInfo() define.DistributionInfo { // Populate values in case we cannot find the values // or the file dist := define.DistributionInfo{ Distribution: "unknown", Version: "unknown", } f, err := os.Open("/etc/os-release") if err != nil { return dist } defer f.Close() l := bufio.NewScanner(f) for l.Scan() { if strings.HasPrefix(l.Text(), "ID=") { dist.Distribution = strings.TrimPrefix(l.Text(), "ID=") } if strings.HasPrefix(l.Text(), "VARIANT_ID=") { dist.Variant = strings.Trim(strings.TrimPrefix(l.Text(), "VARIANT_ID="), "\"") } if strings.HasPrefix(l.Text(), "VERSION_ID=") { dist.Version = strings.Trim(strings.TrimPrefix(l.Text(), "VERSION_ID="), "\"") } if strings.HasPrefix(l.Text(), "VERSION_CODENAME=") { dist.Codename = strings.Trim(strings.TrimPrefix(l.Text(), "VERSION_CODENAME="), "\"") } } return dist } // getCPUUtilization Returns a CPUUsage object that summarizes CPU // usage for userspace, system, and idle time. func getCPUUtilization() (*define.CPUUsage, error) { f, err := os.Open("/proc/stat") if err != nil { return nil, err } defer f.Close() scanner := bufio.NewScanner(f) // Read first line of /proc/stat that has entries for system ("cpu" line) for scanner.Scan() { break } // column 1 is user, column 3 is system, column 4 is idle stats := strings.Fields(scanner.Text()) return statToPercent(stats) } func statToPercent(stats []string) (*define.CPUUsage, error) { userTotal, err := strconv.ParseFloat(stats[1], 64) if err != nil { return nil, fmt.Errorf("unable to parse user value %q: %w", stats[1], err) } systemTotal, err := strconv.ParseFloat(stats[3], 64) if err != nil { return nil, fmt.Errorf("unable to parse system value %q: %w", stats[3], err) } idleTotal, err := strconv.ParseFloat(stats[4], 64) if err != nil { return nil, fmt.Errorf("unable to parse idle value %q: %w", stats[4], err) } total := userTotal + systemTotal + idleTotal s := define.CPUUsage{ UserPercent: math.Round((userTotal/total*100)*100) / 100, SystemPercent: math.Round((systemTotal/total*100)*100) / 100, IdlePercent: math.Round((idleTotal/total*100)*100) / 100, } return &s, nil }