2016-09-22 17:04:34 -04:00
|
|
|
package volume
|
|
|
|
|
|
|
|
import (
|
2018-05-03 21:02:44 -04:00
|
|
|
"context"
|
2016-09-22 17:04:34 -04:00
|
|
|
"fmt"
|
|
|
|
|
2017-04-17 18:07:56 -04:00
|
|
|
"github.com/docker/cli/cli"
|
|
|
|
"github.com/docker/cli/cli/command"
|
2022-03-30 09:27:25 -04:00
|
|
|
"github.com/docker/cli/cli/command/completion"
|
2017-05-15 08:45:19 -04:00
|
|
|
"github.com/docker/cli/opts"
|
2023-04-20 06:54:56 -04:00
|
|
|
"github.com/docker/docker/api/types/versions"
|
|
|
|
"github.com/docker/docker/errdefs"
|
2016-09-22 17:04:34 -04:00
|
|
|
units "github.com/docker/go-units"
|
2023-04-20 06:54:56 -04:00
|
|
|
"github.com/pkg/errors"
|
2016-09-22 17:04:34 -04:00
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
|
|
|
type pruneOptions struct {
|
2023-04-20 06:54:56 -04:00
|
|
|
all bool
|
2017-02-04 12:10:05 -05:00
|
|
|
force bool
|
|
|
|
filter opts.FilterOpt
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewPruneCommand returns a new cobra prune command for volumes
|
2017-02-27 12:39:35 -05:00
|
|
|
func NewPruneCommand(dockerCli command.Cli) *cobra.Command {
|
2017-05-15 08:45:19 -04:00
|
|
|
options := pruneOptions{filter: opts.NewFilterOpt()}
|
2016-09-22 17:04:34 -04:00
|
|
|
|
|
|
|
cmd := &cobra.Command{
|
2016-10-10 11:07:32 -04:00
|
|
|
Use: "prune [OPTIONS]",
|
2023-08-12 09:19:17 -04:00
|
|
|
Short: "Remove unused local volumes",
|
2016-09-22 17:04:34 -04:00
|
|
|
Args: cli.NoArgs,
|
|
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
2017-05-15 08:45:19 -04:00
|
|
|
spaceReclaimed, output, err := runPrune(dockerCli, options)
|
2016-09-22 17:04:34 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if output != "" {
|
|
|
|
fmt.Fprintln(dockerCli.Out(), output)
|
|
|
|
}
|
|
|
|
fmt.Fprintln(dockerCli.Out(), "Total reclaimed space:", units.HumanSize(float64(spaceReclaimed)))
|
|
|
|
return nil
|
|
|
|
},
|
2022-03-30 09:27:25 -04:00
|
|
|
Annotations: map[string]string{"version": "1.25"},
|
|
|
|
ValidArgsFunction: completion.NoComplete,
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
flags := cmd.Flags()
|
2023-04-20 06:54:56 -04:00
|
|
|
flags.BoolVarP(&options.all, "all", "a", false, "Remove all unused volumes, not just anonymous ones")
|
|
|
|
flags.SetAnnotation("all", "version", []string{"1.42"})
|
2017-05-15 08:45:19 -04:00
|
|
|
flags.BoolVarP(&options.force, "force", "f", false, "Do not prompt for confirmation")
|
2023-01-03 05:12:24 -05:00
|
|
|
flags.Var(&options.filter, "filter", `Provide filter values (e.g. "label=<label>")`)
|
2016-09-22 17:04:34 -04:00
|
|
|
|
|
|
|
return cmd
|
|
|
|
}
|
|
|
|
|
2023-04-20 06:54:56 -04:00
|
|
|
const (
|
|
|
|
unusedVolumesWarning = `WARNING! This will remove anonymous local volumes not used by at least one container.
|
2016-09-22 17:04:34 -04:00
|
|
|
Are you sure you want to continue?`
|
2023-04-20 06:54:56 -04:00
|
|
|
allVolumesWarning = `WARNING! This will remove all local volumes not used by at least one container.
|
|
|
|
Are you sure you want to continue?`
|
|
|
|
)
|
2016-09-22 17:04:34 -04:00
|
|
|
|
2017-05-15 08:45:19 -04:00
|
|
|
func runPrune(dockerCli command.Cli, options pruneOptions) (spaceReclaimed uint64, output string, err error) {
|
|
|
|
pruneFilters := command.PruneFilters(dockerCli, options.filter.Value())
|
2017-02-04 12:10:05 -05:00
|
|
|
|
2023-04-20 06:54:56 -04:00
|
|
|
warning := unusedVolumesWarning
|
|
|
|
if versions.GreaterThanOrEqualTo(dockerCli.CurrentVersion(), "1.42") {
|
|
|
|
if options.all {
|
|
|
|
if pruneFilters.Contains("all") {
|
|
|
|
return 0, "", errdefs.InvalidParameter(errors.New("conflicting options: cannot specify both --all and --filter all=1"))
|
|
|
|
}
|
|
|
|
pruneFilters.Add("all", "true")
|
|
|
|
warning = allVolumesWarning
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// API < v1.42 removes all volumes (anonymous and named) by default.
|
|
|
|
warning = allVolumesWarning
|
|
|
|
}
|
2017-05-15 08:45:19 -04:00
|
|
|
if !options.force && !command.PromptForConfirmation(dockerCli.In(), dockerCli.Out(), warning) {
|
2017-10-12 11:44:03 -04:00
|
|
|
return 0, "", nil
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|
|
|
|
|
2017-02-04 12:10:05 -05:00
|
|
|
report, err := dockerCli.Client().VolumesPrune(context.Background(), pruneFilters)
|
2016-09-22 17:04:34 -04:00
|
|
|
if err != nil {
|
2017-10-12 11:44:03 -04:00
|
|
|
return 0, "", err
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(report.VolumesDeleted) > 0 {
|
|
|
|
output = "Deleted Volumes:\n"
|
|
|
|
for _, id := range report.VolumesDeleted {
|
|
|
|
output += id + "\n"
|
|
|
|
}
|
|
|
|
spaceReclaimed = report.SpaceReclaimed
|
|
|
|
}
|
|
|
|
|
2017-10-12 11:44:03 -04:00
|
|
|
return spaceReclaimed, output, nil
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|
|
|
|
|
2016-10-08 06:38:25 -04:00
|
|
|
// RunPrune calls the Volume Prune API
|
2016-09-22 17:04:34 -04:00
|
|
|
// This returns the amount of space reclaimed and a detailed output string
|
cli/command: RunPrune(): remove name for unused "all" parameter (revive)
These functions must have the same signature, but only some of them accept
an "all" boolean argument;
https://github.com/docker/cli/blob/88924b180210be1b6b82a7ad1ac6732606ff270f/cli/command/system/prune.go#L79
cli/command/container/prune.go:78:38: unused-parameter: parameter 'all' seems to be unused, consider removing or renaming it as _ (revive)
func RunPrune(dockerCli command.Cli, all bool, filter opts.FilterOpt) (uint64, string, error) {
^
cli/command/network/prune.go:73:38: unused-parameter: parameter 'all' seems to be unused, consider removing or renaming it as _ (revive)
func RunPrune(dockerCli command.Cli, all bool, filter opts.FilterOpt) (uint64, string, error) {
^
cli/command/volume/prune.go:78:38: unused-parameter: parameter 'all' seems to be unused, consider removing or renaming it as _ (revive)
func RunPrune(dockerCli command.Cli, all bool, filter opts.FilterOpt) (uint64, string, error) {
^
Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
2023-03-29 09:54:33 -04:00
|
|
|
func RunPrune(dockerCli command.Cli, _ bool, filter opts.FilterOpt) (uint64, string, error) {
|
2017-02-04 12:10:05 -05:00
|
|
|
return runPrune(dockerCli, pruneOptions{force: true, filter: filter})
|
2016-09-22 17:04:34 -04:00
|
|
|
}
|