Skip to content

Commit 70fc46c

Browse files
author
weipeng
committed
Add image squash ut & Update docs/command-reference.md
Signed-off-by: weipeng <[email protected]>
1 parent baf13d2 commit 70fc46c

File tree

6 files changed

+177
-60
lines changed

6 files changed

+177
-60
lines changed

cmd/nerdctl/image/image_squash.go

+7-12
Original file line numberDiff line numberDiff line change
@@ -28,15 +28,15 @@ import (
2828
)
2929

3030
func addSquashFlags(cmd *cobra.Command) {
31-
cmd.Flags().IntP("layer-count", "c", 0, "The number of layers that can be compressed")
32-
cmd.Flags().StringP("layer-digest", "d", "", "The digest of the layer to be compressed")
33-
cmd.Flags().StringP("author", "a", "", `Author (e.g., "nerdctl contributor <[email protected]>")`)
34-
cmd.Flags().StringP("message", "m", "", "Commit message")
31+
cmd.Flags().IntP("last-n-layer", "n", 0, "The number of specify squashing the last N (N=layer-count) layers")
32+
cmd.Flags().StringP("author", "a", "nerdctl", `Author (e.g., "nerdctl contributor <[email protected]>")`)
33+
cmd.Flags().StringP("message", "m", "generated by nerdctl squash", "Commit message")
3534
}
3635

36+
// NewSquashCommand returns a new `squash` command to compress the number of layers of the image
3737
func NewSquashCommand() *cobra.Command {
3838
var squashCommand = &cobra.Command{
39-
Use: "squash [flags] SOURCE_IMAGE TAG_IMAGE",
39+
Use: "squash [flags] SOURCE_IMAGE TARGET_IMAGE",
4040
Short: "Compress the number of layers of the image",
4141
Args: helpers.IsExactArgs(2),
4242
RunE: squashAction,
@@ -52,11 +52,7 @@ func processSquashCommandFlags(cmd *cobra.Command, args []string) (options types
5252
if err != nil {
5353
return options, err
5454
}
55-
layerCount, err := cmd.Flags().GetInt("layer-count")
56-
if err != nil {
57-
return options, err
58-
}
59-
layerDigest, err := cmd.Flags().GetString("layer-digest")
55+
layerN, err := cmd.Flags().GetInt("last-n-layer")
6056
if err != nil {
6157
return options, err
6258
}
@@ -78,8 +74,7 @@ func processSquashCommandFlags(cmd *cobra.Command, args []string) (options types
7874
SourceImageRef: args[0],
7975
TargetImageName: args[1],
8076

81-
SquashLayerCount: layerCount,
82-
SquashLayerDigest: layerDigest,
77+
SquashLayerLastN: layerN,
8378
}
8479
return options, nil
8580
}
+100
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
/*
2+
Copyright The containerd Authors.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package image
18+
19+
import (
20+
"fmt"
21+
"testing"
22+
23+
"gotest.tools/v3/assert"
24+
25+
"github.com/containerd/nerdctl/v2/pkg/testutil"
26+
"github.com/containerd/nerdctl/v2/pkg/testutil/nerdtest"
27+
"github.com/containerd/nerdctl/v2/pkg/testutil/test"
28+
)
29+
30+
func squashIdentifierName(identifier string) string {
31+
return fmt.Sprintf("%s-squash", identifier)
32+
}
33+
34+
func secondCommitedIdentifierName(identifier string) string {
35+
return fmt.Sprintf("%s-second", identifier)
36+
}
37+
38+
func TestSquash(t *testing.T) {
39+
testCase := nerdtest.Setup()
40+
41+
require := test.Require(
42+
test.Not(nerdtest.Docker),
43+
nerdtest.CGroup,
44+
)
45+
46+
testCase.SubTests = []*test.Case{
47+
{
48+
Description: "by last-n-layer",
49+
Require: require,
50+
NoParallel: true,
51+
Cleanup: func(data test.Data, helpers test.Helpers) {
52+
identifier := data.Identifier()
53+
secondIdentifier := secondCommitedIdentifierName(identifier)
54+
squashIdentifier := squashIdentifierName(identifier)
55+
helpers.Anyhow("rm", "-f", identifier)
56+
helpers.Anyhow("rm", "-f", secondIdentifier)
57+
helpers.Anyhow("rm", "-f", squashIdentifier)
58+
59+
helpers.Anyhow("rmi", "-f", secondIdentifier)
60+
helpers.Anyhow("rmi", "-f", identifier)
61+
helpers.Anyhow("rmi", "-f", squashIdentifier)
62+
helpers.Anyhow("image", "prune", "-f")
63+
},
64+
Setup: func(data test.Data, helpers test.Helpers) {
65+
identifier := data.Identifier()
66+
helpers.Ensure("run", "-d", "--name", identifier, testutil.CommonImage, "sleep", nerdtest.Infinity)
67+
helpers.Ensure("exec", identifier, "sh", "-euxc", `echo hello-first-commit > /foo`)
68+
helpers.Ensure("commit", "-c", `CMD ["cat", "/foo"]`, "-m", `first commit`, "--pause=true", identifier, identifier)
69+
out := helpers.Capture("run", "--rm", identifier)
70+
assert.Equal(t, out, "hello-first-commit\n")
71+
72+
secondIdentifier := secondCommitedIdentifierName(identifier)
73+
helpers.Ensure("run", "-d", "--name", secondIdentifier, identifier, "sleep", nerdtest.Infinity)
74+
helpers.Ensure("exec", secondIdentifier, "sh", "-euxc", `echo hello-second-commit > /bar && echo hello-squash-commit > /foo`)
75+
helpers.Ensure("commit", "-c", `CMD ["cat", "/foo", "/bar"]`, "-m", `second commit`, "--pause=true", secondIdentifier, secondIdentifier)
76+
out = helpers.Capture("run", "--rm", secondIdentifier)
77+
assert.Equal(t, out, "hello-squash-commit\nhello-second-commit\n")
78+
79+
squashIdentifier := squashIdentifierName(identifier)
80+
helpers.Ensure("image", "squash", "-n", "2", "-m", "squash commit", secondIdentifier, squashIdentifier)
81+
out = helpers.Capture("run", "--rm", squashIdentifier)
82+
assert.Equal(t, out, "hello-squash-commit\nhello-second-commit\n")
83+
},
84+
Command: func(data test.Data, helpers test.Helpers) test.TestableCommand {
85+
identifier := data.Identifier()
86+
87+
squashIdentifier := squashIdentifierName(identifier)
88+
return helpers.Command("image", "history", "--human=true", "--format=json", squashIdentifier)
89+
},
90+
Expected: test.Expects(0, nil, func(stdout string, info string, t *testing.T) {
91+
history, err := decode(stdout)
92+
assert.NilError(t, err, info)
93+
assert.Equal(t, len(history), 3, info)
94+
assert.Equal(t, history[0].Comment, "squash commit", info)
95+
}),
96+
},
97+
}
98+
99+
testCase.Run(t)
100+
}

cmd/nerdctl/main.go

-1
Original file line numberDiff line numberDiff line change
@@ -295,7 +295,6 @@ Config file ($NERDCTL_TOML): %s
295295
image.NewTagCommand(),
296296
image.NewRmiCommand(),
297297
image.NewHistoryCommand(),
298-
image.NewSquashCommand(),
299298
// #endregion
300299

301300
// #region System

docs/command-reference.md

+18
Original file line numberDiff line numberDiff line change
@@ -994,6 +994,24 @@ Flags:
994994
- `--platform=<PLATFORM>` : Convert content for a specific platform
995995
- `--all-platforms` : Convert content for all platforms (default: false)
996996

997+
### :nerd_face: nerdctl image squash
998+
999+
Squash last-n-layer into a single layer.
1000+
1001+
Usage: `nerdctl image squash [OPTIONS] SOURCE_IMAGE[:TAG] TARGET_IMAGE[:TAG]`
1002+
1003+
Example:
1004+
1005+
```bash
1006+
nerdctl image pull example.com/foo:latest
1007+
nerdctl image squash ----last-n-layer=2 --message="generated by nerdctl squash" example.com/foo:latest example.com/foo:squashed
1008+
```
1009+
1010+
Flags:
1011+
- `-n --last-n-layer=<NUMBER>`: The number of specify squashing the last N (N=layer-count) layers
1012+
- `-m --message=<MESSAGE>`: Commit message for the squashed image
1013+
- `-a --author=<AUTHOR>`: Author of the squashed image
1014+
9971015
## Registry
9981016

9991017
### :whale: nerdctl login

pkg/api/types/image_types.go

+2-4
Original file line numberDiff line numberDiff line change
@@ -305,8 +305,6 @@ type ImageSquashOptions struct {
305305
// TargetImageName is the name of the squashed image
306306
TargetImageName string
307307

308-
// SquashLayerCount is the number of layers to squash
309-
SquashLayerCount int
310-
// SquashLayerDigest is the digest of the layer to squash
311-
SquashLayerDigest string
308+
// SquashLayerLastN is the number of layers to squash
309+
SquashLayerLastN int
312310
}

0 commit comments

Comments
 (0)