mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Add --format
flag for docker plugin ls
This fix tries to address the enhancement discussed in 28735 to add `--format` for the output of `docker plugin ls`. This fix 1. Add `--format` and `--quiet` flags to `docker plugin ls` 2. Convert the current implementation to use `formatter`, consistent with other docker list commands. 3. Add `pluginsFormat` for config.json. Related docs has been updated. Several unit tests have been added to cover the changes. This fix is related to 28708 and 28735. Signed-off-by: Yong Tang <yong.tang.github@outlook.com>
This commit is contained in:
parent
7a97705b52
commit
1c0d37fa7f
7 changed files with 357 additions and 23 deletions
87
cli/command/formatter/plugin.go
Normal file
87
cli/command/formatter/plugin.go
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
package formatter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/docker/docker/pkg/stringid"
|
||||||
|
"github.com/docker/docker/pkg/stringutils"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultPluginTableFormat = "table {{.ID}}\t{{.Name}}\t{{.Description}}\t{{.Enabled}}"
|
||||||
|
|
||||||
|
pluginIDHeader = "ID"
|
||||||
|
descriptionHeader = "DESCRIPTION"
|
||||||
|
enabledHeader = "ENABLED"
|
||||||
|
)
|
||||||
|
|
||||||
|
// NewPluginFormat returns a Format for rendering using a plugin Context
|
||||||
|
func NewPluginFormat(source string, quiet bool) Format {
|
||||||
|
switch source {
|
||||||
|
case TableFormatKey:
|
||||||
|
if quiet {
|
||||||
|
return defaultQuietFormat
|
||||||
|
}
|
||||||
|
return defaultPluginTableFormat
|
||||||
|
case RawFormatKey:
|
||||||
|
if quiet {
|
||||||
|
return `plugin_id: {{.ID}}`
|
||||||
|
}
|
||||||
|
return `plugin_id: {{.ID}}\nname: {{.Name}}\ndescription: {{.Description}}\nenabled: {{.Enabled}}\n`
|
||||||
|
}
|
||||||
|
return Format(source)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PluginWrite writes the context
|
||||||
|
func PluginWrite(ctx Context, plugins []*types.Plugin) error {
|
||||||
|
render := func(format func(subContext subContext) error) error {
|
||||||
|
for _, plugin := range plugins {
|
||||||
|
pluginCtx := &pluginContext{trunc: ctx.Trunc, p: *plugin}
|
||||||
|
if err := format(pluginCtx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return ctx.Write(&pluginContext{}, render)
|
||||||
|
}
|
||||||
|
|
||||||
|
type pluginContext struct {
|
||||||
|
HeaderContext
|
||||||
|
trunc bool
|
||||||
|
p types.Plugin
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *pluginContext) MarshalJSON() ([]byte, error) {
|
||||||
|
return marshalJSON(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *pluginContext) ID() string {
|
||||||
|
c.AddHeader(pluginIDHeader)
|
||||||
|
if c.trunc {
|
||||||
|
return stringid.TruncateID(c.p.ID)
|
||||||
|
}
|
||||||
|
return c.p.ID
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *pluginContext) Name() string {
|
||||||
|
c.AddHeader(nameHeader)
|
||||||
|
return c.p.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *pluginContext) Description() string {
|
||||||
|
c.AddHeader(descriptionHeader)
|
||||||
|
desc := strings.Replace(c.p.Config.Description, "\n", "", -1)
|
||||||
|
desc = strings.Replace(desc, "\r", "", -1)
|
||||||
|
if c.trunc {
|
||||||
|
desc = stringutils.Ellipsis(desc, 45)
|
||||||
|
}
|
||||||
|
|
||||||
|
return desc
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *pluginContext) Enabled() bool {
|
||||||
|
c.AddHeader(enabledHeader)
|
||||||
|
return c.p.Enabled
|
||||||
|
}
|
188
cli/command/formatter/plugin_test.go
Normal file
188
cli/command/formatter/plugin_test.go
Normal file
|
@ -0,0 +1,188 @@
|
||||||
|
package formatter
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/docker/docker/api/types"
|
||||||
|
"github.com/docker/docker/pkg/stringid"
|
||||||
|
"github.com/docker/docker/pkg/testutil/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestPluginContext(t *testing.T) {
|
||||||
|
pluginID := stringid.GenerateRandomID()
|
||||||
|
|
||||||
|
var ctx pluginContext
|
||||||
|
cases := []struct {
|
||||||
|
pluginCtx pluginContext
|
||||||
|
expValue string
|
||||||
|
expHeader string
|
||||||
|
call func() string
|
||||||
|
}{
|
||||||
|
{pluginContext{
|
||||||
|
p: types.Plugin{ID: pluginID},
|
||||||
|
trunc: false,
|
||||||
|
}, pluginID, pluginIDHeader, ctx.ID},
|
||||||
|
{pluginContext{
|
||||||
|
p: types.Plugin{ID: pluginID},
|
||||||
|
trunc: true,
|
||||||
|
}, stringid.TruncateID(pluginID), pluginIDHeader, ctx.ID},
|
||||||
|
{pluginContext{
|
||||||
|
p: types.Plugin{Name: "plugin_name"},
|
||||||
|
}, "plugin_name", nameHeader, ctx.Name},
|
||||||
|
{pluginContext{
|
||||||
|
p: types.Plugin{Config: types.PluginConfig{Description: "plugin_description"}},
|
||||||
|
}, "plugin_description", descriptionHeader, ctx.Description},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, c := range cases {
|
||||||
|
ctx = c.pluginCtx
|
||||||
|
v := c.call()
|
||||||
|
if strings.Contains(v, ",") {
|
||||||
|
compareMultipleValues(t, v, c.expValue)
|
||||||
|
} else if v != c.expValue {
|
||||||
|
t.Fatalf("Expected %s, was %s\n", c.expValue, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
h := ctx.FullHeader()
|
||||||
|
if h != c.expHeader {
|
||||||
|
t.Fatalf("Expected %s, was %s\n", c.expHeader, h)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPluginContextWrite(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
context Context
|
||||||
|
expected string
|
||||||
|
}{
|
||||||
|
|
||||||
|
// Errors
|
||||||
|
{
|
||||||
|
Context{Format: "{{InvalidFunction}}"},
|
||||||
|
`Template parsing error: template: :1: function "InvalidFunction" not defined
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Context{Format: "{{nil}}"},
|
||||||
|
`Template parsing error: template: :1:2: executing "" at <nil>: nil is not a command
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
// Table format
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("table", false)},
|
||||||
|
`ID NAME DESCRIPTION ENABLED
|
||||||
|
pluginID1 foobar_baz description 1 true
|
||||||
|
pluginID2 foobar_bar description 2 false
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("table", true)},
|
||||||
|
`pluginID1
|
||||||
|
pluginID2
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("table {{.Name}}", false)},
|
||||||
|
`NAME
|
||||||
|
foobar_baz
|
||||||
|
foobar_bar
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("table {{.Name}}", true)},
|
||||||
|
`NAME
|
||||||
|
foobar_baz
|
||||||
|
foobar_bar
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
// Raw Format
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("raw", false)},
|
||||||
|
`plugin_id: pluginID1
|
||||||
|
name: foobar_baz
|
||||||
|
description: description 1
|
||||||
|
enabled: true
|
||||||
|
|
||||||
|
plugin_id: pluginID2
|
||||||
|
name: foobar_bar
|
||||||
|
description: description 2
|
||||||
|
enabled: false
|
||||||
|
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("raw", true)},
|
||||||
|
`plugin_id: pluginID1
|
||||||
|
plugin_id: pluginID2
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
// Custom Format
|
||||||
|
{
|
||||||
|
Context{Format: NewPluginFormat("{{.Name}}", false)},
|
||||||
|
`foobar_baz
|
||||||
|
foobar_bar
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, testcase := range cases {
|
||||||
|
plugins := []*types.Plugin{
|
||||||
|
{ID: "pluginID1", Name: "foobar_baz", Config: types.PluginConfig{Description: "description 1"}, Enabled: true},
|
||||||
|
{ID: "pluginID2", Name: "foobar_bar", Config: types.PluginConfig{Description: "description 2"}, Enabled: false},
|
||||||
|
}
|
||||||
|
out := bytes.NewBufferString("")
|
||||||
|
testcase.context.Output = out
|
||||||
|
err := PluginWrite(testcase.context, plugins)
|
||||||
|
if err != nil {
|
||||||
|
assert.Error(t, err, testcase.expected)
|
||||||
|
} else {
|
||||||
|
assert.Equal(t, out.String(), testcase.expected)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPluginContextWriteJSON(t *testing.T) {
|
||||||
|
plugins := []*types.Plugin{
|
||||||
|
{ID: "pluginID1", Name: "foobar_baz"},
|
||||||
|
{ID: "pluginID2", Name: "foobar_bar"},
|
||||||
|
}
|
||||||
|
expectedJSONs := []map[string]interface{}{
|
||||||
|
{"Description": "", "Enabled": false, "ID": "pluginID1", "Name": "foobar_baz"},
|
||||||
|
{"Description": "", "Enabled": false, "ID": "pluginID2", "Name": "foobar_bar"},
|
||||||
|
}
|
||||||
|
|
||||||
|
out := bytes.NewBufferString("")
|
||||||
|
err := PluginWrite(Context{Format: "{{json .}}", Output: out}, plugins)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
for i, line := range strings.Split(strings.TrimSpace(out.String()), "\n") {
|
||||||
|
var m map[string]interface{}
|
||||||
|
if err := json.Unmarshal([]byte(line), &m); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
assert.DeepEqual(t, m, expectedJSONs[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPluginContextWriteJSONField(t *testing.T) {
|
||||||
|
plugins := []*types.Plugin{
|
||||||
|
{ID: "pluginID1", Name: "foobar_baz"},
|
||||||
|
{ID: "pluginID2", Name: "foobar_bar"},
|
||||||
|
}
|
||||||
|
out := bytes.NewBufferString("")
|
||||||
|
err := PluginWrite(Context{Format: "{{json .ID}}", Output: out}, plugins)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
for i, line := range strings.Split(strings.TrimSpace(out.String()), "\n") {
|
||||||
|
var s string
|
||||||
|
if err := json.Unmarshal([]byte(line), &s); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
assert.Equal(t, s, plugins[i].ID)
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,20 +1,17 @@
|
||||||
package plugin
|
package plugin
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"strings"
|
|
||||||
"text/tabwriter"
|
|
||||||
|
|
||||||
"github.com/docker/docker/cli"
|
"github.com/docker/docker/cli"
|
||||||
"github.com/docker/docker/cli/command"
|
"github.com/docker/docker/cli/command"
|
||||||
"github.com/docker/docker/pkg/stringid"
|
"github.com/docker/docker/cli/command/formatter"
|
||||||
"github.com/docker/docker/pkg/stringutils"
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
"golang.org/x/net/context"
|
"golang.org/x/net/context"
|
||||||
)
|
)
|
||||||
|
|
||||||
type listOptions struct {
|
type listOptions struct {
|
||||||
|
quiet bool
|
||||||
noTrunc bool
|
noTrunc bool
|
||||||
|
format string
|
||||||
}
|
}
|
||||||
|
|
||||||
func newListCommand(dockerCli *command.DockerCli) *cobra.Command {
|
func newListCommand(dockerCli *command.DockerCli) *cobra.Command {
|
||||||
|
@ -32,7 +29,9 @@ func newListCommand(dockerCli *command.DockerCli) *cobra.Command {
|
||||||
|
|
||||||
flags := cmd.Flags()
|
flags := cmd.Flags()
|
||||||
|
|
||||||
|
flags.BoolVarP(&opts.quiet, "quiet", "q", false, "Only display plugin IDs")
|
||||||
flags.BoolVar(&opts.noTrunc, "no-trunc", false, "Don't truncate output")
|
flags.BoolVar(&opts.noTrunc, "no-trunc", false, "Don't truncate output")
|
||||||
|
flags.StringVar(&opts.format, "format", "", "Pretty-print plugins using a Go template")
|
||||||
|
|
||||||
return cmd
|
return cmd
|
||||||
}
|
}
|
||||||
|
@ -43,21 +42,19 @@ func runList(dockerCli *command.DockerCli, opts listOptions) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
w := tabwriter.NewWriter(dockerCli.Out(), 20, 1, 3, ' ', 0)
|
format := opts.format
|
||||||
fmt.Fprintf(w, "ID \tNAME \tDESCRIPTION\tENABLED")
|
if len(format) == 0 {
|
||||||
fmt.Fprintf(w, "\n")
|
if len(dockerCli.ConfigFile().PluginsFormat) > 0 && !opts.quiet {
|
||||||
|
format = dockerCli.ConfigFile().PluginsFormat
|
||||||
for _, p := range plugins {
|
} else {
|
||||||
id := p.ID
|
format = formatter.TableFormatKey
|
||||||
desc := strings.Replace(p.Config.Description, "\n", " ", -1)
|
|
||||||
desc = strings.Replace(desc, "\r", " ", -1)
|
|
||||||
if !opts.noTrunc {
|
|
||||||
id = stringid.TruncateID(p.ID)
|
|
||||||
desc = stringutils.Ellipsis(desc, 45)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Fprintf(w, "%s\t%s\t%s\t%v\n", id, p.Name, desc, p.Enabled)
|
|
||||||
}
|
}
|
||||||
w.Flush()
|
|
||||||
return nil
|
pluginsCtx := formatter.Context{
|
||||||
|
Output: dockerCli.Out(),
|
||||||
|
Format: formatter.NewPluginFormat(format, opts.quiet),
|
||||||
|
Trunc: !opts.noTrunc,
|
||||||
|
}
|
||||||
|
return formatter.PluginWrite(pluginsCtx, plugins)
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,6 +27,7 @@ type ConfigFile struct {
|
||||||
PsFormat string `json:"psFormat,omitempty"`
|
PsFormat string `json:"psFormat,omitempty"`
|
||||||
ImagesFormat string `json:"imagesFormat,omitempty"`
|
ImagesFormat string `json:"imagesFormat,omitempty"`
|
||||||
NetworksFormat string `json:"networksFormat,omitempty"`
|
NetworksFormat string `json:"networksFormat,omitempty"`
|
||||||
|
PluginsFormat string `json:"pluginsFormat,omitempty"`
|
||||||
VolumesFormat string `json:"volumesFormat,omitempty"`
|
VolumesFormat string `json:"volumesFormat,omitempty"`
|
||||||
StatsFormat string `json:"statsFormat,omitempty"`
|
StatsFormat string `json:"statsFormat,omitempty"`
|
||||||
DetachKeys string `json:"detachKeys,omitempty"`
|
DetachKeys string `json:"detachKeys,omitempty"`
|
||||||
|
|
|
@ -131,6 +131,12 @@ Docker's client uses this property. If this property is not set, the client
|
||||||
falls back to the default table format. For a list of supported formatting
|
falls back to the default table format. For a list of supported formatting
|
||||||
directives, see the [**Formatting** section in the `docker images` documentation](images.md)
|
directives, see the [**Formatting** section in the `docker images` documentation](images.md)
|
||||||
|
|
||||||
|
The property `pluginsFormat` specifies the default format for `docker plugin ls` output.
|
||||||
|
When the `--format` flag is not provided with the `docker plugin ls` command,
|
||||||
|
Docker's client uses this property. If this property is not set, the client
|
||||||
|
falls back to the default table format. For a list of supported formatting
|
||||||
|
directives, see the [**Formatting** section in the `docker plugin ls` documentation](plugin_ls.md)
|
||||||
|
|
||||||
The property `serviceInspectFormat` specifies the default format for `docker
|
The property `serviceInspectFormat` specifies the default format for `docker
|
||||||
service inspect` output. When the `--format` flag is not provided with the
|
service inspect` output. When the `--format` flag is not provided with the
|
||||||
`docker service inspect` command, Docker's client uses this property. If this
|
`docker service inspect` command, Docker's client uses this property. If this
|
||||||
|
@ -186,6 +192,7 @@ Following is a sample `config.json` file:
|
||||||
},
|
},
|
||||||
"psFormat": "table {{.ID}}\\t{{.Image}}\\t{{.Command}}\\t{{.Labels}}",
|
"psFormat": "table {{.ID}}\\t{{.Image}}\\t{{.Command}}\\t{{.Labels}}",
|
||||||
"imagesFormat": "table {{.ID}}\\t{{.Repository}}\\t{{.Tag}}\\t{{.CreatedAt}}",
|
"imagesFormat": "table {{.ID}}\\t{{.Repository}}\\t{{.Tag}}\\t{{.CreatedAt}}",
|
||||||
|
"pluginsFormat": "table {{.ID}}\t{{.Name}}\t{{.Enabled}}",
|
||||||
"statsFormat": "table {{.Container}}\t{{.CPUPerc}}\t{{.MemUsage}}",
|
"statsFormat": "table {{.Container}}\t{{.CPUPerc}}\t{{.MemUsage}}",
|
||||||
"serviceInspectFormat": "pretty",
|
"serviceInspectFormat": "pretty",
|
||||||
"detachKeys": "ctrl-e,e",
|
"detachKeys": "ctrl-e,e",
|
||||||
|
|
|
@ -24,8 +24,10 @@ Aliases:
|
||||||
ls, list
|
ls, list
|
||||||
|
|
||||||
Options:
|
Options:
|
||||||
--help Print usage
|
--format string Pretty-print plugins using a Go template
|
||||||
--no-trunc Don't truncate output
|
--help Print usage
|
||||||
|
--no-trunc Don't truncate output
|
||||||
|
-q, --quiet Only display plugin IDs
|
||||||
```
|
```
|
||||||
|
|
||||||
Lists all the plugins that are currently installed. You can install plugins
|
Lists all the plugins that are currently installed. You can install plugins
|
||||||
|
@ -40,6 +42,32 @@ ID NAME TAG DESCRIP
|
||||||
69553ca1d123 tiborvass/sample-volume-plugin latest A test plugin for Docker true
|
69553ca1d123 tiborvass/sample-volume-plugin latest A test plugin for Docker true
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Formatting
|
||||||
|
|
||||||
|
The formatting options (`--format`) pretty-prints plugins output
|
||||||
|
using a Go template.
|
||||||
|
|
||||||
|
Valid placeholders for the Go template are listed below:
|
||||||
|
|
||||||
|
Placeholder | Description
|
||||||
|
---------------|------------------------------------------------------------------------------------------
|
||||||
|
`.ID` | Plugin ID
|
||||||
|
`.Name` | Plugin name
|
||||||
|
`.Description` | Plugin description
|
||||||
|
`.Enabled` | Whether plugin is enabled or not
|
||||||
|
|
||||||
|
When using the `--format` option, the `plugin ls` command will either
|
||||||
|
output the data exactly as the template declares or, when using the
|
||||||
|
`table` directive, includes column headers as well.
|
||||||
|
|
||||||
|
The following example uses a template without headers and outputs the
|
||||||
|
`ID` and `Name` entries separated by a colon for all plugins:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ docker plugin ls --format "{{.ID}}: {{.Name}}"
|
||||||
|
4be01827a72e: tiborvass/no-remove
|
||||||
|
```
|
||||||
|
|
||||||
## Related information
|
## Related information
|
||||||
|
|
||||||
* [plugin create](plugin_create.md)
|
* [plugin create](plugin_create.md)
|
||||||
|
|
|
@ -401,3 +401,29 @@ func (s *DockerSuite) TestPluginIDPrefix(c *check.C) {
|
||||||
c.Assert(out, checker.Not(checker.Contains), pName)
|
c.Assert(out, checker.Not(checker.Contains), pName)
|
||||||
c.Assert(out, checker.Not(checker.Contains), pTag)
|
c.Assert(out, checker.Not(checker.Contains), pTag)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *DockerSuite) TestPluginListDefaultFormat(c *check.C) {
|
||||||
|
testRequires(c, DaemonIsLinux, Network, IsAmd64)
|
||||||
|
|
||||||
|
config, err := ioutil.TempDir("", "config-file-")
|
||||||
|
c.Assert(err, check.IsNil)
|
||||||
|
defer os.RemoveAll(config)
|
||||||
|
|
||||||
|
err = ioutil.WriteFile(filepath.Join(config, "config.json"), []byte(`{"pluginsFormat": "raw"}`), 0644)
|
||||||
|
c.Assert(err, check.IsNil)
|
||||||
|
|
||||||
|
out, _ := dockerCmd(c, "plugin", "install", "--grant-all-permissions", pName)
|
||||||
|
c.Assert(strings.TrimSpace(out), checker.Contains, pName)
|
||||||
|
|
||||||
|
out, _ = dockerCmd(c, "plugin", "inspect", "--format", "{{.ID}}", pNameWithTag)
|
||||||
|
id := strings.TrimSpace(out)
|
||||||
|
|
||||||
|
// We expect the format to be in `raw + --no-trunc`
|
||||||
|
expectedOutput := fmt.Sprintf(`plugin_id: %s
|
||||||
|
name: %s
|
||||||
|
description: A sample volume plugin for Docker
|
||||||
|
enabled: true`, id, pNameWithTag)
|
||||||
|
|
||||||
|
out, _ = dockerCmd(c, "--config", config, "plugin", "ls", "--no-trunc")
|
||||||
|
c.Assert(strings.TrimSpace(out), checker.Contains, expectedOutput)
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue