mirror of
https://github.com/containers/podman.git
synced 2025-12-05 04:40:47 +08:00
With the advent of Podman 2.0.0 we crossed the magical barrier of go modules. While we were able to continue importing all packages inside of the project, the project could not be vendored anymore from the outside. Move the go module to new major version and change all imports to `github.com/containers/libpod/v2`. The renaming of the imports was done via `gomove` [1]. [1] https://github.com/KSubedi/gomove Signed-off-by: Valentin Rothberg <rothberg@redhat.com>
286 lines
8.3 KiB
Go
286 lines
8.3 KiB
Go
package pods
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"net/http"
|
|
"net/url"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/containers/libpod/v2/pkg/api/handlers"
|
|
"github.com/containers/libpod/v2/pkg/bindings"
|
|
"github.com/containers/libpod/v2/pkg/domain/entities"
|
|
"github.com/containers/libpod/v2/pkg/specgen"
|
|
jsoniter "github.com/json-iterator/go"
|
|
)
|
|
|
|
func CreatePodFromSpec(ctx context.Context, s *specgen.PodSpecGenerator) (*entities.PodCreateReport, error) {
|
|
var (
|
|
pcr entities.PodCreateReport
|
|
)
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
specgenString, err := jsoniter.MarshalToString(s)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
stringReader := strings.NewReader(specgenString)
|
|
response, err := conn.DoRequest(stringReader, http.MethodPost, "/pods/create", nil, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &pcr, response.Process(&pcr)
|
|
}
|
|
|
|
// Exists is a lightweight method to determine if a pod exists in local storage
|
|
func Exists(ctx context.Context, nameOrID string) (bool, error) {
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodGet, "/pods/%s/exists", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return response.IsSuccess(), nil
|
|
}
|
|
|
|
// Inspect returns low-level information about the given pod.
|
|
func Inspect(ctx context.Context, nameOrID string) (*entities.PodInspectReport, error) {
|
|
var (
|
|
report entities.PodInspectReport
|
|
)
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodGet, "/pods/%s/json", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Kill sends a SIGTERM to all the containers in a pod. The optional signal parameter
|
|
// can be used to override SIGTERM.
|
|
func Kill(ctx context.Context, nameOrID string, signal *string) (*entities.PodKillReport, error) {
|
|
var (
|
|
report entities.PodKillReport
|
|
)
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
if signal != nil {
|
|
params.Set("signal", *signal)
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/kill", params, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Pause pauses all running containers in a given pod.
|
|
func Pause(ctx context.Context, nameOrID string) (*entities.PodPauseReport, error) {
|
|
var report entities.PodPauseReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/pause", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Prune by default removes all non-running pods in local storage.
|
|
// And with force set true removes all pods.
|
|
func Prune(ctx context.Context) ([]*entities.PodPruneReport, error) {
|
|
var reports []*entities.PodPruneReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/prune", nil, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return reports, response.Process(&reports)
|
|
}
|
|
|
|
// List returns all pods in local storage. The optional filters parameter can
|
|
// be used to refine which pods should be listed.
|
|
func List(ctx context.Context, filters map[string][]string) ([]*entities.ListPodsReport, error) {
|
|
var (
|
|
podsReports []*entities.ListPodsReport
|
|
)
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
if filters != nil {
|
|
stringFilter, err := bindings.FiltersToString(filters)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params.Set("filters", stringFilter)
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodGet, "/pods/json", params, nil)
|
|
if err != nil {
|
|
return podsReports, err
|
|
}
|
|
return podsReports, response.Process(&podsReports)
|
|
}
|
|
|
|
// Restart restarts all containers in a pod.
|
|
func Restart(ctx context.Context, nameOrID string) (*entities.PodRestartReport, error) {
|
|
var report entities.PodRestartReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/restart", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Remove deletes a Pod from from local storage. The optional force parameter denotes
|
|
// that the Pod can be removed even if in a running state.
|
|
func Remove(ctx context.Context, nameOrID string, force *bool) (*entities.PodRmReport, error) {
|
|
var report entities.PodRmReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
if force != nil {
|
|
params.Set("force", strconv.FormatBool(*force))
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodDelete, "/pods/%s", params, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Start starts all containers in a pod.
|
|
func Start(ctx context.Context, nameOrID string) (*entities.PodStartReport, error) {
|
|
var report entities.PodStartReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/start", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if response.StatusCode == http.StatusNotModified {
|
|
report.Id = nameOrID
|
|
return &report, nil
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Stop stops all containers in a Pod. The optional timeout parameter can be
|
|
// used to override the timeout before the container is killed.
|
|
func Stop(ctx context.Context, nameOrID string, timeout *int) (*entities.PodStopReport, error) {
|
|
var report entities.PodStopReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
if timeout != nil {
|
|
params.Set("t", strconv.Itoa(*timeout))
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/stop", params, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if response.StatusCode == http.StatusNotModified {
|
|
report.Id = nameOrID
|
|
return &report, nil
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Top gathers statistics about the running processes in a pod. The nameOrID can be a pod name
|
|
// or a partial/full ID. The descriptors allow for specifying which data to collect from each process.
|
|
func Top(ctx context.Context, nameOrID string, descriptors []string) ([]string, error) {
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
|
|
if len(descriptors) > 0 {
|
|
// flatten the slice into one string
|
|
params.Set("ps_args", strings.Join(descriptors, ","))
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodGet, "/pods/%s/top", params, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
body := handlers.PodTopOKBody{}
|
|
if err = response.Process(&body); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// handlers.PodTopOKBody{} returns a slice of slices where each cell in the top table is an item.
|
|
// In libpod land, we're just using a slice with cells being split by tabs, which allows for an idiomatic
|
|
// usage of the tabwriter.
|
|
topOutput := []string{strings.Join(body.Titles, "\t")}
|
|
for _, out := range body.Processes {
|
|
topOutput = append(topOutput, strings.Join(out, "\t"))
|
|
}
|
|
|
|
return topOutput, err
|
|
}
|
|
|
|
// Unpause unpauses all paused containers in a Pod.
|
|
func Unpause(ctx context.Context, nameOrID string) (*entities.PodUnpauseReport, error) {
|
|
var report entities.PodUnpauseReport
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
response, err := conn.DoRequest(nil, http.MethodPost, "/pods/%s/unpause", nil, nil, nameOrID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &report, response.Process(&report)
|
|
}
|
|
|
|
// Stats display resource-usage statistics of one or more pods.
|
|
func Stats(ctx context.Context, namesOrIDs []string, options entities.PodStatsOptions) ([]*entities.PodStatsReport, error) {
|
|
if options.Latest {
|
|
return nil, errors.New("latest is not supported")
|
|
}
|
|
conn, err := bindings.GetClient(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
params := url.Values{}
|
|
for _, i := range namesOrIDs {
|
|
params.Add("namesOrIDs", i)
|
|
}
|
|
params.Set("all", strconv.FormatBool(options.All))
|
|
|
|
var reports []*entities.PodStatsReport
|
|
response, err := conn.DoRequest(nil, http.MethodGet, "/pods/stats", params, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return reports, response.Process(&reports)
|
|
}
|