mirror of https://github.com/containers/podman.git
Merge pull request #14967 from sstosh/pause-option
Add pause/unpause --latest, --cidfile, --filter
This commit is contained in:
commit
9b4df69830
|
@ -2,61 +2,88 @@ package containers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/containers/common/pkg/completion"
|
||||||
"github.com/containers/podman/v4/cmd/podman/common"
|
"github.com/containers/podman/v4/cmd/podman/common"
|
||||||
"github.com/containers/podman/v4/cmd/podman/registry"
|
"github.com/containers/podman/v4/cmd/podman/registry"
|
||||||
"github.com/containers/podman/v4/cmd/podman/utils"
|
"github.com/containers/podman/v4/cmd/podman/utils"
|
||||||
|
"github.com/containers/podman/v4/cmd/podman/validate"
|
||||||
"github.com/containers/podman/v4/pkg/domain/entities"
|
"github.com/containers/podman/v4/pkg/domain/entities"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"github.com/spf13/pflag"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
pauseDescription = `Pauses one or more running containers. The container name or ID can be used.`
|
pauseDescription = `Pauses one or more running containers. The container name or ID can be used.`
|
||||||
pauseCommand = &cobra.Command{
|
pauseCommand = &cobra.Command{
|
||||||
Use: "pause [options] CONTAINER [CONTAINER...]",
|
Use: "pause [options] CONTAINER [CONTAINER...]",
|
||||||
Short: "Pause all the processes in one or more containers",
|
Short: "Pause all the processes in one or more containers",
|
||||||
Long: pauseDescription,
|
Long: pauseDescription,
|
||||||
RunE: pause,
|
RunE: pause,
|
||||||
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
|
return validate.CheckAllLatestAndIDFile(cmd, args, false, "cidfile")
|
||||||
|
},
|
||||||
ValidArgsFunction: common.AutocompleteContainersRunning,
|
ValidArgsFunction: common.AutocompleteContainersRunning,
|
||||||
Example: `podman pause mywebserver
|
Example: `podman pause mywebserver
|
||||||
podman pause 860a4b23
|
podman pause 860a4b23
|
||||||
podman pause -a`,
|
podman pause --all`,
|
||||||
}
|
}
|
||||||
|
|
||||||
containerPauseCommand = &cobra.Command{
|
containerPauseCommand = &cobra.Command{
|
||||||
Use: pauseCommand.Use,
|
Use: pauseCommand.Use,
|
||||||
Short: pauseCommand.Short,
|
Short: pauseCommand.Short,
|
||||||
Long: pauseCommand.Long,
|
Long: pauseCommand.Long,
|
||||||
RunE: pauseCommand.RunE,
|
RunE: pauseCommand.RunE,
|
||||||
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
|
return validate.CheckAllLatestAndIDFile(cmd, args, false, "cidfile")
|
||||||
|
},
|
||||||
ValidArgsFunction: pauseCommand.ValidArgsFunction,
|
ValidArgsFunction: pauseCommand.ValidArgsFunction,
|
||||||
Example: `podman container pause mywebserver
|
Example: `podman container pause mywebserver
|
||||||
podman container pause 860a4b23
|
podman container pause 860a4b23
|
||||||
podman container pause -a`,
|
podman container pause --all`,
|
||||||
}
|
}
|
||||||
|
|
||||||
pauseOpts = entities.PauseUnPauseOptions{}
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func pauseFlags(flags *pflag.FlagSet) {
|
var (
|
||||||
|
pauseOpts = entities.PauseUnPauseOptions{
|
||||||
|
Filters: make(map[string][]string),
|
||||||
|
}
|
||||||
|
pauseCidFiles = []string{}
|
||||||
|
)
|
||||||
|
|
||||||
|
func pauseFlags(cmd *cobra.Command) {
|
||||||
|
flags := cmd.Flags()
|
||||||
|
|
||||||
flags.BoolVarP(&pauseOpts.All, "all", "a", false, "Pause all running containers")
|
flags.BoolVarP(&pauseOpts.All, "all", "a", false, "Pause all running containers")
|
||||||
|
|
||||||
|
cidfileFlagName := "cidfile"
|
||||||
|
flags.StringArrayVar(&pauseCidFiles, cidfileFlagName, nil, "Read the container ID from the file")
|
||||||
|
_ = cmd.RegisterFlagCompletionFunc(cidfileFlagName, completion.AutocompleteDefault)
|
||||||
|
|
||||||
|
filterFlagName := "filter"
|
||||||
|
flags.StringSliceVarP(&filters, filterFlagName, "f", []string{}, "Filter output based on conditions given")
|
||||||
|
_ = cmd.RegisterFlagCompletionFunc(filterFlagName, common.AutocompletePsFilters)
|
||||||
|
|
||||||
|
if registry.IsRemote() {
|
||||||
|
_ = flags.MarkHidden("cidfile")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
registry.Commands = append(registry.Commands, registry.CliCommand{
|
registry.Commands = append(registry.Commands, registry.CliCommand{
|
||||||
Command: pauseCommand,
|
Command: pauseCommand,
|
||||||
})
|
})
|
||||||
flags := pauseCommand.Flags()
|
pauseFlags(pauseCommand)
|
||||||
pauseFlags(flags)
|
validate.AddLatestFlag(pauseCommand, &pauseOpts.Latest)
|
||||||
|
|
||||||
registry.Commands = append(registry.Commands, registry.CliCommand{
|
registry.Commands = append(registry.Commands, registry.CliCommand{
|
||||||
Command: containerPauseCommand,
|
Command: containerPauseCommand,
|
||||||
Parent: containerCmd,
|
Parent: containerCmd,
|
||||||
})
|
})
|
||||||
containerPauseFlags := containerPauseCommand.Flags()
|
pauseFlags(containerPauseCommand)
|
||||||
pauseFlags(containerPauseFlags)
|
validate.AddLatestFlag(containerPauseCommand, &pauseOpts.Latest)
|
||||||
}
|
}
|
||||||
|
|
||||||
func pause(cmd *cobra.Command, args []string) error {
|
func pause(cmd *cobra.Command, args []string) error {
|
||||||
|
@ -64,16 +91,30 @@ func pause(cmd *cobra.Command, args []string) error {
|
||||||
errs utils.OutputErrors
|
errs utils.OutputErrors
|
||||||
)
|
)
|
||||||
|
|
||||||
if len(args) < 1 && !pauseOpts.All {
|
for _, cidFile := range pauseCidFiles {
|
||||||
return errors.New("you must provide at least one container name or id")
|
content, err := ioutil.ReadFile(cidFile)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error reading CIDFile: %w", err)
|
||||||
|
}
|
||||||
|
id := strings.Split(string(content), "\n")[0]
|
||||||
|
args = append(args, id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, f := range filters {
|
||||||
|
split := strings.SplitN(f, "=", 2)
|
||||||
|
if len(split) < 2 {
|
||||||
|
return fmt.Errorf("invalid filter %q", f)
|
||||||
|
}
|
||||||
|
pauseOpts.Filters[split[0]] = append(pauseOpts.Filters[split[0]], split[1])
|
||||||
|
}
|
||||||
|
|
||||||
responses, err := registry.ContainerEngine().ContainerPause(context.Background(), args, pauseOpts)
|
responses, err := registry.ContainerEngine().ContainerPause(context.Background(), args, pauseOpts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
for _, r := range responses {
|
for _, r := range responses {
|
||||||
if r.Err == nil {
|
if r.Err == nil {
|
||||||
fmt.Println(r.Id)
|
fmt.Println(r.RawInput)
|
||||||
} else {
|
} else {
|
||||||
errs = append(errs, r.Err)
|
errs = append(errs, r.Err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,59 +4,87 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/containers/common/pkg/cgroups"
|
"github.com/containers/common/pkg/cgroups"
|
||||||
|
"github.com/containers/common/pkg/completion"
|
||||||
"github.com/containers/podman/v4/cmd/podman/common"
|
"github.com/containers/podman/v4/cmd/podman/common"
|
||||||
"github.com/containers/podman/v4/cmd/podman/registry"
|
"github.com/containers/podman/v4/cmd/podman/registry"
|
||||||
"github.com/containers/podman/v4/cmd/podman/utils"
|
"github.com/containers/podman/v4/cmd/podman/utils"
|
||||||
|
"github.com/containers/podman/v4/cmd/podman/validate"
|
||||||
"github.com/containers/podman/v4/pkg/domain/entities"
|
"github.com/containers/podman/v4/pkg/domain/entities"
|
||||||
"github.com/containers/podman/v4/pkg/rootless"
|
"github.com/containers/podman/v4/pkg/rootless"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"github.com/spf13/pflag"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
unpauseDescription = `Unpauses one or more previously paused containers. The container name or ID can be used.`
|
unpauseDescription = `Unpauses one or more previously paused containers. The container name or ID can be used.`
|
||||||
unpauseCommand = &cobra.Command{
|
unpauseCommand = &cobra.Command{
|
||||||
Use: "unpause [options] CONTAINER [CONTAINER...]",
|
Use: "unpause [options] CONTAINER [CONTAINER...]",
|
||||||
Short: "Unpause the processes in one or more containers",
|
Short: "Unpause the processes in one or more containers",
|
||||||
Long: unpauseDescription,
|
Long: unpauseDescription,
|
||||||
RunE: unpause,
|
RunE: unpause,
|
||||||
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
|
return validate.CheckAllLatestAndIDFile(cmd, args, false, "cidfile")
|
||||||
|
},
|
||||||
ValidArgsFunction: common.AutocompleteContainersPaused,
|
ValidArgsFunction: common.AutocompleteContainersPaused,
|
||||||
Example: `podman unpause ctrID
|
Example: `podman unpause ctrID
|
||||||
podman unpause --all`,
|
podman unpause --all`,
|
||||||
}
|
}
|
||||||
unPauseOptions = entities.PauseUnPauseOptions{}
|
|
||||||
|
|
||||||
containerUnpauseCommand = &cobra.Command{
|
containerUnpauseCommand = &cobra.Command{
|
||||||
Use: unpauseCommand.Use,
|
Use: unpauseCommand.Use,
|
||||||
Short: unpauseCommand.Short,
|
Short: unpauseCommand.Short,
|
||||||
Long: unpauseCommand.Long,
|
Long: unpauseCommand.Long,
|
||||||
RunE: unpauseCommand.RunE,
|
RunE: unpauseCommand.RunE,
|
||||||
|
Args: func(cmd *cobra.Command, args []string) error {
|
||||||
|
return validate.CheckAllLatestAndIDFile(cmd, args, false, "cidfile")
|
||||||
|
},
|
||||||
ValidArgsFunction: unpauseCommand.ValidArgsFunction,
|
ValidArgsFunction: unpauseCommand.ValidArgsFunction,
|
||||||
Example: `podman container unpause ctrID
|
Example: `podman container unpause ctrID
|
||||||
podman container unpause --all`,
|
podman container unpause --all`,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
||||||
func unpauseFlags(flags *pflag.FlagSet) {
|
var (
|
||||||
flags.BoolVarP(&unPauseOptions.All, "all", "a", false, "Pause all running containers")
|
unpauseOpts = entities.PauseUnPauseOptions{
|
||||||
|
Filters: make(map[string][]string),
|
||||||
|
}
|
||||||
|
unpauseCidFiles = []string{}
|
||||||
|
)
|
||||||
|
|
||||||
|
func unpauseFlags(cmd *cobra.Command) {
|
||||||
|
flags := cmd.Flags()
|
||||||
|
|
||||||
|
flags.BoolVarP(&unpauseOpts.All, "all", "a", false, "Unpause all paused containers")
|
||||||
|
|
||||||
|
cidfileFlagName := "cidfile"
|
||||||
|
flags.StringArrayVar(&unpauseCidFiles, cidfileFlagName, nil, "Read the container ID from the file")
|
||||||
|
_ = cmd.RegisterFlagCompletionFunc(cidfileFlagName, completion.AutocompleteDefault)
|
||||||
|
|
||||||
|
filterFlagName := "filter"
|
||||||
|
flags.StringSliceVarP(&filters, filterFlagName, "f", []string{}, "Filter output based on conditions given")
|
||||||
|
_ = cmd.RegisterFlagCompletionFunc(filterFlagName, common.AutocompletePsFilters)
|
||||||
|
|
||||||
|
if registry.IsRemote() {
|
||||||
|
_ = flags.MarkHidden("cidfile")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
registry.Commands = append(registry.Commands, registry.CliCommand{
|
registry.Commands = append(registry.Commands, registry.CliCommand{
|
||||||
Command: unpauseCommand,
|
Command: unpauseCommand,
|
||||||
})
|
})
|
||||||
flags := unpauseCommand.Flags()
|
unpauseFlags(unpauseCommand)
|
||||||
unpauseFlags(flags)
|
validate.AddLatestFlag(unpauseCommand, &unpauseOpts.Latest)
|
||||||
|
|
||||||
registry.Commands = append(registry.Commands, registry.CliCommand{
|
registry.Commands = append(registry.Commands, registry.CliCommand{
|
||||||
Command: containerUnpauseCommand,
|
Command: containerUnpauseCommand,
|
||||||
Parent: containerCmd,
|
Parent: containerCmd,
|
||||||
})
|
})
|
||||||
|
unpauseFlags(containerUnpauseCommand)
|
||||||
unpauseCommandFlags := containerUnpauseCommand.Flags()
|
validate.AddLatestFlag(containerUnpauseCommand, &unpauseOpts.Latest)
|
||||||
unpauseFlags(unpauseCommandFlags)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func unpause(cmd *cobra.Command, args []string) error {
|
func unpause(cmd *cobra.Command, args []string) error {
|
||||||
|
@ -69,16 +97,32 @@ func unpause(cmd *cobra.Command, args []string) error {
|
||||||
return errors.New("unpause is not supported for cgroupv1 rootless containers")
|
return errors.New("unpause is not supported for cgroupv1 rootless containers")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if len(args) < 1 && !unPauseOptions.All {
|
|
||||||
return errors.New("you must provide at least one container name or id")
|
for _, cidFile := range unpauseCidFiles {
|
||||||
|
content, err := ioutil.ReadFile(cidFile)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("error reading CIDFile: %w", err)
|
||||||
|
}
|
||||||
|
id := strings.Split(string(content), "\n")[0]
|
||||||
|
args = append(args, id)
|
||||||
}
|
}
|
||||||
responses, err := registry.ContainerEngine().ContainerUnpause(context.Background(), args, unPauseOptions)
|
|
||||||
|
for _, f := range filters {
|
||||||
|
split := strings.SplitN(f, "=", 2)
|
||||||
|
if len(split) < 2 {
|
||||||
|
return fmt.Errorf("invalid filter %q", f)
|
||||||
|
}
|
||||||
|
unpauseOpts.Filters[split[0]] = append(unpauseOpts.Filters[split[0]], split[1])
|
||||||
|
}
|
||||||
|
|
||||||
|
responses, err := registry.ContainerEngine().ContainerUnpause(context.Background(), args, unpauseOpts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, r := range responses {
|
for _, r := range responses {
|
||||||
if r.Err == nil {
|
if r.Err == nil {
|
||||||
fmt.Println(r.Id)
|
fmt.Println(r.RawInput)
|
||||||
} else {
|
} else {
|
||||||
errs = append(errs, r.Err)
|
errs = append(errs, r.Err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,21 +17,65 @@ Pauses all the processes in one or more containers. You may use container IDs o
|
||||||
|
|
||||||
Pause all running containers.
|
Pause all running containers.
|
||||||
|
|
||||||
|
#### **--cidfile**
|
||||||
|
|
||||||
|
Read container ID from the specified file and pause the container. Can be specified multiple times.
|
||||||
|
|
||||||
|
#### **--filter**, **-f**=*filter*
|
||||||
|
|
||||||
|
Filter what containers pause.
|
||||||
|
Multiple filters can be given with multiple uses of the --filter flag.
|
||||||
|
Filters with the same key work inclusive with the only exception being
|
||||||
|
`label` which is exclusive. Filters with different keys always work exclusive.
|
||||||
|
|
||||||
|
Valid filters are listed below:
|
||||||
|
|
||||||
|
| **Filter** | **Description** |
|
||||||
|
| --------------- | -------------------------------------------------------------------------------- |
|
||||||
|
| id | [ID] Container's ID (accepts regex) |
|
||||||
|
| name | [Name] Container's name (accepts regex) |
|
||||||
|
| label | [Key] or [Key=Value] Label assigned to a container |
|
||||||
|
| exited | [Int] Container's exit code |
|
||||||
|
| status | [Status] Container's status: 'created', 'exited', 'paused', 'running', 'unknown' |
|
||||||
|
| ancestor | [ImageName] Image or descendant used to create container |
|
||||||
|
| before | [ID] or [Name] Containers created before this container |
|
||||||
|
| since | [ID] or [Name] Containers created since this container |
|
||||||
|
| volume | [VolumeName] or [MountpointDestination] Volume mounted in container |
|
||||||
|
| health | [Status] healthy or unhealthy |
|
||||||
|
| pod | [Pod] name or full or partial ID of pod |
|
||||||
|
| network | [Network] name or full ID of network |
|
||||||
|
|
||||||
|
#### **--latest**, **-l**
|
||||||
|
|
||||||
|
Instead of providing the container name or ID, use the last created container. If you use methods other than Podman
|
||||||
|
to run containers such as CRI-O, the last started container could be from either of those methods. (This option is not available with the remote Podman client, including Mac and Windows (excluding WSL2) machines)
|
||||||
|
|
||||||
## EXAMPLE
|
## EXAMPLE
|
||||||
|
|
||||||
Pause a container named 'mywebserver'
|
Pause container named 'mywebserver'
|
||||||
```
|
```
|
||||||
podman pause mywebserver
|
podman pause mywebserver
|
||||||
```
|
```
|
||||||
|
|
||||||
Pause a container by partial container ID.
|
Pause container by partial container ID.
|
||||||
```
|
```
|
||||||
podman pause 860a4b23
|
podman pause 860a4b23
|
||||||
```
|
```
|
||||||
|
|
||||||
Pause all **running** containers.
|
Pause all **running** containers.
|
||||||
```
|
```
|
||||||
podman pause -a
|
podman pause --all
|
||||||
|
```
|
||||||
|
|
||||||
|
Pause container using ID specified in a given files.
|
||||||
|
```
|
||||||
|
podman pause --cidfile /home/user/cidfile-1
|
||||||
|
podman pause --cidfile /home/user/cidfile-1 --cidfile ./cidfile-2
|
||||||
|
```
|
||||||
|
|
||||||
|
Pause the latest container created by Podman.
|
||||||
|
```
|
||||||
|
podman pause --latest
|
||||||
```
|
```
|
||||||
|
|
||||||
## SEE ALSO
|
## SEE ALSO
|
||||||
|
|
|
@ -17,21 +17,65 @@ Unpauses the processes in one or more containers. You may use container IDs or
|
||||||
|
|
||||||
Unpause all paused containers.
|
Unpause all paused containers.
|
||||||
|
|
||||||
|
#### **--cidfile**
|
||||||
|
|
||||||
|
Read container ID from the specified file and unpause the container. Can be specified multiple times.
|
||||||
|
|
||||||
|
#### **--filter**, **-f**=*filter*
|
||||||
|
|
||||||
|
Filter what containers unpause.
|
||||||
|
Multiple filters can be given with multiple uses of the --filter flag.
|
||||||
|
Filters with the same key work inclusive with the only exception being
|
||||||
|
`label` which is exclusive. Filters with different keys always work exclusive.
|
||||||
|
|
||||||
|
Valid filters are listed below:
|
||||||
|
|
||||||
|
| **Filter** | **Description** |
|
||||||
|
| --------------- | -------------------------------------------------------------------------------- |
|
||||||
|
| id | [ID] Container's ID (accepts regex) |
|
||||||
|
| name | [Name] Container's name (accepts regex) |
|
||||||
|
| label | [Key] or [Key=Value] Label assigned to a container |
|
||||||
|
| exited | [Int] Container's exit code |
|
||||||
|
| status | [Status] Container's status: 'created', 'exited', 'paused', 'running', 'unknown' |
|
||||||
|
| ancestor | [ImageName] Image or descendant used to create container |
|
||||||
|
| before | [ID] or [Name] Containers created before this container |
|
||||||
|
| since | [ID] or [Name] Containers created since this container |
|
||||||
|
| volume | [VolumeName] or [MountpointDestination] Volume mounted in container |
|
||||||
|
| health | [Status] healthy or unhealthy |
|
||||||
|
| pod | [Pod] name or full or partial ID of pod |
|
||||||
|
| network | [Network] name or full ID of network |
|
||||||
|
|
||||||
|
#### **--latest**, **-l**
|
||||||
|
|
||||||
|
Instead of providing the container name or ID, use the last created container. If you use methods other than Podman
|
||||||
|
to run containers such as CRI-O, the last started container could be from either of those methods. (This option is not available with the remote Podman client, including Mac and Windows (excluding WSL2) machines)
|
||||||
|
|
||||||
## EXAMPLE
|
## EXAMPLE
|
||||||
|
|
||||||
Unpause a container called 'mywebserver'
|
Unpause container called 'mywebserver'
|
||||||
```
|
```
|
||||||
podman unpause mywebserver
|
podman unpause mywebserver
|
||||||
```
|
```
|
||||||
|
|
||||||
Unpause a container by a partial container ID.
|
Unpause container by a partial container ID.
|
||||||
```
|
```
|
||||||
podman unpause 860a4b23
|
podman unpause 860a4b23
|
||||||
```
|
```
|
||||||
|
|
||||||
Unpause all **paused** containers.
|
Unpause all **paused** containers.
|
||||||
```
|
```
|
||||||
podman unpause -a
|
podman unpause --all
|
||||||
|
```
|
||||||
|
|
||||||
|
Unpause container using ID specified in a given files.
|
||||||
|
```
|
||||||
|
podman unpause --cidfile /home/user/cidfile-1
|
||||||
|
podman unpause --cidfile /home/user/cidfile-1 --cidfile ./cidfile-2
|
||||||
|
```
|
||||||
|
|
||||||
|
Unpause the latest container created by Podman.
|
||||||
|
```
|
||||||
|
podman unpause --latest
|
||||||
```
|
```
|
||||||
|
|
||||||
## SEE ALSO
|
## SEE ALSO
|
||||||
|
|
|
@ -71,12 +71,15 @@ type StringSliceReport struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type PauseUnPauseOptions struct {
|
type PauseUnPauseOptions struct {
|
||||||
All bool
|
Filters map[string][]string
|
||||||
|
All bool
|
||||||
|
Latest bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type PauseUnpauseReport struct {
|
type PauseUnpauseReport struct {
|
||||||
Err error
|
Err error
|
||||||
Id string //nolint:revive,stylecheck
|
Id string //nolint:revive,stylecheck
|
||||||
|
RawInput string
|
||||||
}
|
}
|
||||||
|
|
||||||
type StopOptions struct {
|
type StopOptions struct {
|
||||||
|
|
|
@ -62,6 +62,11 @@ func getContainersAndInputByContext(all, latest bool, names []string, filters ma
|
||||||
}
|
}
|
||||||
case all:
|
case all:
|
||||||
ctrs, err = runtime.GetAllContainers()
|
ctrs, err = runtime.GetAllContainers()
|
||||||
|
if err == nil {
|
||||||
|
for _, ctr := range ctrs {
|
||||||
|
rawInput = append(rawInput, ctr.ID())
|
||||||
|
}
|
||||||
|
}
|
||||||
case latest:
|
case latest:
|
||||||
ctr, err = runtime.GetLatestContainer()
|
ctr, err = runtime.GetLatestContainer()
|
||||||
if err == nil {
|
if err == nil {
|
||||||
|
@ -133,37 +138,57 @@ func (ic *ContainerEngine) ContainerWait(ctx context.Context, namesOrIds []strin
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ic *ContainerEngine) ContainerPause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
func (ic *ContainerEngine) ContainerPause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
||||||
ctrs, err := getContainersByContext(options.All, false, namesOrIds, ic.Libpod)
|
ctrs, rawInputs, err := getContainersAndInputByContext(options.All, options.Latest, namesOrIds, options.Filters, ic.Libpod)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
report := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
ctrMap := map[string]string{}
|
||||||
|
if len(rawInputs) == len(ctrs) {
|
||||||
|
for i := range ctrs {
|
||||||
|
ctrMap[ctrs[i].ID()] = rawInputs[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
reports := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
||||||
for _, c := range ctrs {
|
for _, c := range ctrs {
|
||||||
err := c.Pause()
|
err := c.Pause()
|
||||||
if err != nil && options.All && errors.Is(err, define.ErrCtrStateInvalid) {
|
if err != nil && options.All && errors.Is(err, define.ErrCtrStateInvalid) {
|
||||||
logrus.Debugf("Container %s is not running", c.ID())
|
logrus.Debugf("Container %s is not running", c.ID())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
report = append(report, &entities.PauseUnpauseReport{Id: c.ID(), Err: err})
|
reports = append(reports, &entities.PauseUnpauseReport{
|
||||||
|
Id: c.ID(),
|
||||||
|
Err: err,
|
||||||
|
RawInput: ctrMap[c.ID()],
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return report, nil
|
return reports, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ic *ContainerEngine) ContainerUnpause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
func (ic *ContainerEngine) ContainerUnpause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
||||||
ctrs, err := getContainersByContext(options.All, false, namesOrIds, ic.Libpod)
|
ctrs, rawInputs, err := getContainersAndInputByContext(options.All, options.Latest, namesOrIds, options.Filters, ic.Libpod)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
report := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
ctrMap := map[string]string{}
|
||||||
|
if len(rawInputs) == len(ctrs) {
|
||||||
|
for i := range ctrs {
|
||||||
|
ctrMap[ctrs[i].ID()] = rawInputs[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
reports := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
||||||
for _, c := range ctrs {
|
for _, c := range ctrs {
|
||||||
err := c.Unpause()
|
err := c.Unpause()
|
||||||
if err != nil && options.All && errors.Is(err, define.ErrCtrStateInvalid) {
|
if err != nil && options.All && errors.Is(err, define.ErrCtrStateInvalid) {
|
||||||
logrus.Debugf("Container %s is not paused", c.ID())
|
logrus.Debugf("Container %s is not paused", c.ID())
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
report = append(report, &entities.PauseUnpauseReport{Id: c.ID(), Err: err})
|
reports = append(reports, &entities.PauseUnpauseReport{
|
||||||
|
Id: c.ID(),
|
||||||
|
Err: err,
|
||||||
|
RawInput: ctrMap[c.ID()],
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return report, nil
|
return reports, nil
|
||||||
}
|
}
|
||||||
func (ic *ContainerEngine) ContainerStop(ctx context.Context, namesOrIds []string, options entities.StopOptions) ([]*entities.StopReport, error) {
|
func (ic *ContainerEngine) ContainerStop(ctx context.Context, namesOrIds []string, options entities.StopOptions) ([]*entities.StopReport, error) {
|
||||||
names := namesOrIds
|
names := namesOrIds
|
||||||
|
|
|
@ -57,10 +57,14 @@ func (ic *ContainerEngine) ContainerWait(ctx context.Context, namesOrIds []strin
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ic *ContainerEngine) ContainerPause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
func (ic *ContainerEngine) ContainerPause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
||||||
ctrs, err := getContainersByContext(ic.ClientCtx, options.All, false, namesOrIds)
|
ctrs, rawInputs, err := getContainersAndInputByContext(ic.ClientCtx, options.All, false, namesOrIds, options.Filters)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
ctrMap := map[string]string{}
|
||||||
|
for i := range ctrs {
|
||||||
|
ctrMap[ctrs[i].ID] = rawInputs[i]
|
||||||
|
}
|
||||||
reports := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
reports := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
||||||
for _, c := range ctrs {
|
for _, c := range ctrs {
|
||||||
err := containers.Pause(ic.ClientCtx, c.ID, nil)
|
err := containers.Pause(ic.ClientCtx, c.ID, nil)
|
||||||
|
@ -68,24 +72,36 @@ func (ic *ContainerEngine) ContainerPause(ctx context.Context, namesOrIds []stri
|
||||||
logrus.Debugf("Container %s is not running", c.ID)
|
logrus.Debugf("Container %s is not running", c.ID)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
reports = append(reports, &entities.PauseUnpauseReport{Id: c.ID, Err: err})
|
reports = append(reports, &entities.PauseUnpauseReport{
|
||||||
|
Id: c.ID,
|
||||||
|
Err: err,
|
||||||
|
RawInput: ctrMap[c.ID],
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return reports, nil
|
return reports, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ic *ContainerEngine) ContainerUnpause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
func (ic *ContainerEngine) ContainerUnpause(ctx context.Context, namesOrIds []string, options entities.PauseUnPauseOptions) ([]*entities.PauseUnpauseReport, error) {
|
||||||
reports := []*entities.PauseUnpauseReport{}
|
ctrs, rawInputs, err := getContainersAndInputByContext(ic.ClientCtx, options.All, false, namesOrIds, options.Filters)
|
||||||
ctrs, err := getContainersByContext(ic.ClientCtx, options.All, false, namesOrIds)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
ctrMap := map[string]string{}
|
||||||
|
for i := range ctrs {
|
||||||
|
ctrMap[ctrs[i].ID] = rawInputs[i]
|
||||||
|
}
|
||||||
|
reports := make([]*entities.PauseUnpauseReport, 0, len(ctrs))
|
||||||
for _, c := range ctrs {
|
for _, c := range ctrs {
|
||||||
err := containers.Unpause(ic.ClientCtx, c.ID, nil)
|
err := containers.Unpause(ic.ClientCtx, c.ID, nil)
|
||||||
if err != nil && options.All && strings.Contains(err.Error(), define.ErrCtrStateInvalid.Error()) {
|
if err != nil && options.All && strings.Contains(err.Error(), define.ErrCtrStateInvalid.Error()) {
|
||||||
logrus.Debugf("Container %s is not paused", c.ID)
|
logrus.Debugf("Container %s is not paused", c.ID)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
reports = append(reports, &entities.PauseUnpauseReport{Id: c.ID, Err: err})
|
reports = append(reports, &entities.PauseUnpauseReport{
|
||||||
|
Id: c.ID,
|
||||||
|
Err: err,
|
||||||
|
RawInput: ctrMap[c.ID],
|
||||||
|
})
|
||||||
}
|
}
|
||||||
return reports, nil
|
return reports, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -319,4 +319,181 @@ var _ = Describe("Podman pause", func() {
|
||||||
Expect(running.OutputToStringArray()).To(HaveLen(3))
|
Expect(running.OutputToStringArray()).To(HaveLen(3))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
It("podman pause --latest", func() {
|
||||||
|
SkipIfRemote("--latest flag n/a")
|
||||||
|
session := podmanTest.RunTopContainer("")
|
||||||
|
session.WaitWithDefaultTimeout()
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
result := podmanTest.Podman([]string{"pause", "-l"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
Expect(podmanTest.NumberOfContainersRunning()).To(Equal(0))
|
||||||
|
Expect(strings.ToLower(podmanTest.GetContainerStatus())).To(ContainSubstring(pausedState))
|
||||||
|
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "-l"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
})
|
||||||
|
|
||||||
|
It("podman pause --cidfile", func() {
|
||||||
|
tmpDir, err := ioutil.TempDir("", "")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
tmpFile := tmpDir + "cid"
|
||||||
|
|
||||||
|
defer os.RemoveAll(tmpDir)
|
||||||
|
|
||||||
|
session := podmanTest.Podman([]string{"create", "--cidfile", tmpFile, ALPINE, "top"})
|
||||||
|
session.WaitWithDefaultTimeout()
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
cid := session.OutputToStringArray()[0]
|
||||||
|
|
||||||
|
session = podmanTest.Podman([]string{"start", cid})
|
||||||
|
session.WaitWithDefaultTimeout()
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
|
||||||
|
result := podmanTest.Podman([]string{"pause", "--cidfile", tmpFile})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(0))
|
||||||
|
output := result.OutputToString()
|
||||||
|
Expect(output).To(ContainSubstring(cid))
|
||||||
|
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "--cidfile", tmpFile})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(0))
|
||||||
|
output = result.OutputToString()
|
||||||
|
Expect(output).To(ContainSubstring(cid))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("podman pause multiple --cidfile", func() {
|
||||||
|
tmpDir, err := ioutil.TempDir("", "")
|
||||||
|
Expect(err).To(BeNil())
|
||||||
|
tmpFile1 := tmpDir + "cid-1"
|
||||||
|
tmpFile2 := tmpDir + "cid-2"
|
||||||
|
|
||||||
|
defer os.RemoveAll(tmpDir)
|
||||||
|
|
||||||
|
session := podmanTest.Podman([]string{"run", "--cidfile", tmpFile1, "-d", ALPINE, "top"})
|
||||||
|
session.WaitWithDefaultTimeout()
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
cid1 := session.OutputToStringArray()[0]
|
||||||
|
Expect(podmanTest.NumberOfContainers()).To(Equal(1))
|
||||||
|
|
||||||
|
session = podmanTest.Podman([]string{"run", "--cidfile", tmpFile2, "-d", ALPINE, "top"})
|
||||||
|
session.WaitWithDefaultTimeout()
|
||||||
|
Expect(session).Should(Exit(0))
|
||||||
|
cid2 := session.OutputToStringArray()[0]
|
||||||
|
Expect(podmanTest.NumberOfContainers()).To(Equal(2))
|
||||||
|
|
||||||
|
result := podmanTest.Podman([]string{"pause", "--cidfile", tmpFile1, "--cidfile", tmpFile2})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(0))
|
||||||
|
output := result.OutputToString()
|
||||||
|
Expect(output).To(ContainSubstring(cid1))
|
||||||
|
Expect(output).To(ContainSubstring(cid2))
|
||||||
|
Expect(podmanTest.NumberOfContainers()).To(Equal(2))
|
||||||
|
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "--cidfile", tmpFile1, "--cidfile", tmpFile2})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(0))
|
||||||
|
output = result.OutputToString()
|
||||||
|
Expect(output).To(ContainSubstring(cid1))
|
||||||
|
Expect(output).To(ContainSubstring(cid2))
|
||||||
|
Expect(podmanTest.NumberOfContainers()).To(Equal(2))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("podman pause invalid --latest and --cidfile and --all", func() {
|
||||||
|
SkipIfRemote("--latest flag n/a")
|
||||||
|
result := podmanTest.Podman([]string{"pause", "--cidfile", "foobar", "--latest"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"pause", "--cidfile", "foobar", "--all"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"pause", "--cidfile", "foobar", "--all", "--latest"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"pause", "--latest", "--all"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("podman unpause invalid --latest and --cidfile and --all", func() {
|
||||||
|
SkipIfRemote("--latest flag n/a")
|
||||||
|
result := podmanTest.Podman([]string{"unpause", "--cidfile", "foobar", "--latest"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "--cidfile", "foobar", "--all"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "--cidfile", "foobar", "--all", "--latest"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
result = podmanTest.Podman([]string{"unpause", "--latest", "--all"})
|
||||||
|
result.WaitWithDefaultTimeout()
|
||||||
|
Expect(result).Should(Exit(125))
|
||||||
|
Expect(result.ErrorToString()).To(ContainSubstring("cannot be used together"))
|
||||||
|
})
|
||||||
|
|
||||||
|
It("podman pause --filter", func() {
|
||||||
|
session1 := podmanTest.RunTopContainer("")
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
cid1 := session1.OutputToString()
|
||||||
|
|
||||||
|
session1 = podmanTest.RunTopContainer("")
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
cid2 := session1.OutputToString()
|
||||||
|
|
||||||
|
session1 = podmanTest.RunTopContainer("")
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
cid3 := session1.OutputToString()
|
||||||
|
shortCid3 := cid3[0:5]
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"pause", cid1, "-f", "status=test"})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(125))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"unpause", cid1, "-f", "status=paused"})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(125))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"pause", "-a", "--filter", fmt.Sprintf("id=%swrongid", shortCid3)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(HaveLen(0))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"pause", "-a", "--filter", fmt.Sprintf("id=%s", shortCid3)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(BeEquivalentTo(cid3))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"unpause", "-a", "--filter", fmt.Sprintf("id=%swrongid", shortCid3)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(HaveLen(0))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"unpause", "-a", "--filter", fmt.Sprintf("id=%s", shortCid3)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(BeEquivalentTo(cid3))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"pause", "-f", fmt.Sprintf("id=%s", cid2)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(BeEquivalentTo(cid2))
|
||||||
|
|
||||||
|
session1 = podmanTest.Podman([]string{"unpause", "-f", fmt.Sprintf("id=%s", cid2)})
|
||||||
|
session1.WaitWithDefaultTimeout()
|
||||||
|
Expect(session1).Should(Exit(0))
|
||||||
|
Expect(session1.OutputToString()).To(BeEquivalentTo(cid2))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Reference in New Issue