341 lines
12 KiB
Go
341 lines
12 KiB
Go
package main
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
|
|
"github.com/containers/storage"
|
|
graphdriver "github.com/containers/storage/drivers"
|
|
"github.com/containers/storage/internal/opts"
|
|
"github.com/containers/storage/pkg/idtools"
|
|
"github.com/containers/storage/pkg/mflag"
|
|
"github.com/containers/storage/pkg/stringid"
|
|
"github.com/containers/storage/types"
|
|
digest "github.com/opencontainers/go-digest"
|
|
)
|
|
|
|
var (
|
|
paramMountOptions = ""
|
|
paramMountLabel = ""
|
|
paramNames = []string{}
|
|
paramID = ""
|
|
paramDigest = ""
|
|
paramLayer = ""
|
|
paramMetadata = ""
|
|
paramMetadataFile = ""
|
|
paramCreateRO = false
|
|
paramHostUIDMap = false
|
|
paramHostGIDMap = false
|
|
paramUIDMap = ""
|
|
paramGIDMap = ""
|
|
paramSubUIDMap = ""
|
|
paramSubGIDMap = ""
|
|
paramReadOnly = false
|
|
paramVolatile = false
|
|
)
|
|
|
|
func paramIDMapping() (*types.IDMappingOptions, error) {
|
|
options := types.IDMappingOptions{
|
|
HostUIDMapping: paramHostUIDMap,
|
|
HostGIDMapping: paramHostGIDMap,
|
|
}
|
|
if paramHostUIDMap && (len(paramUIDMap) > 0 || paramSubUIDMap != "") {
|
|
return nil, fmt.Errorf("host uid map specified along with UID mapping")
|
|
}
|
|
if paramHostGIDMap && (len(paramGIDMap) > 0 || paramSubGIDMap != "") {
|
|
return nil, fmt.Errorf("host gid map specified along with GID mapping")
|
|
}
|
|
if paramSubGIDMap == "" && paramSubUIDMap != "" {
|
|
paramSubGIDMap = paramSubUIDMap
|
|
}
|
|
if paramSubUIDMap == "" && paramSubGIDMap != "" {
|
|
paramSubUIDMap = paramSubGIDMap
|
|
}
|
|
if paramSubUIDMap != "" && paramSubGIDMap != "" {
|
|
mappings, err := idtools.NewIDMappings(paramSubUIDMap, paramSubGIDMap)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
options.UIDMap = mappings.UIDs()
|
|
options.GIDMap = mappings.GIDs()
|
|
}
|
|
parsedUIDMap, err := idtools.ParseIDMap([]string{paramUIDMap}, "uid")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
parsedGIDMap, err := idtools.ParseIDMap([]string{paramGIDMap}, "gid")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
options.UIDMap = append(options.UIDMap, parsedUIDMap...)
|
|
options.GIDMap = append(options.GIDMap, parsedGIDMap...)
|
|
return &options, nil
|
|
}
|
|
|
|
func createStorageLayer(flags *mflag.FlagSet, action string, m storage.Store, args []string) (int, error) {
|
|
parent := ""
|
|
if len(args) > 0 {
|
|
parent = args[0]
|
|
}
|
|
mappings, err := paramIDMapping()
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
driver, err := m.GraphDriver()
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
opts := graphdriver.CreateOpts{
|
|
MountLabel: paramMountLabel,
|
|
IDMappings: idtools.NewIDMappingsFromMaps(mappings.UIDMap, mappings.GIDMap),
|
|
}
|
|
if paramID == "" {
|
|
paramID = stringid.GenerateNonCryptoID()
|
|
}
|
|
if paramCreateRO {
|
|
if err := driver.Create(paramID, parent, &opts); err != nil {
|
|
return 1, err
|
|
}
|
|
} else {
|
|
if err := driver.CreateReadWrite(paramID, parent, &opts); err != nil {
|
|
return 1, err
|
|
}
|
|
}
|
|
if jsonOutput {
|
|
if err := json.NewEncoder(os.Stdout).Encode(paramID); err != nil {
|
|
return 1, err
|
|
}
|
|
} else {
|
|
fmt.Printf("%s\n", paramID)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func createLayer(flags *mflag.FlagSet, action string, m storage.Store, args []string) (int, error) {
|
|
parent := ""
|
|
if len(args) > 0 {
|
|
parent = args[0]
|
|
}
|
|
mappings, err := paramIDMapping()
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
options := &storage.LayerOptions{IDMappingOptions: *mappings}
|
|
layer, err := m.CreateLayer(paramID, parent, paramNames, paramMountLabel, !paramCreateRO, options)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
if jsonOutput {
|
|
return outputJSON(layer)
|
|
}
|
|
fmt.Printf("%s\n", layer.ID)
|
|
for _, name := range layer.Names {
|
|
fmt.Printf("\t%s\n", name)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func importLayer(flags *mflag.FlagSet, action string, m storage.Store, args []string) (int, error) {
|
|
parent := ""
|
|
if len(args) > 0 {
|
|
parent = args[0]
|
|
}
|
|
diffStream := io.Reader(os.Stdin)
|
|
if applyDiffFile != "" {
|
|
f, err := os.Open(applyDiffFile)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
diffStream = f
|
|
defer f.Close()
|
|
}
|
|
mappings, err := paramIDMapping()
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
options := &storage.LayerOptions{IDMappingOptions: *mappings}
|
|
layer, _, err := m.PutLayer(paramID, parent, paramNames, paramMountLabel, !paramCreateRO, options, diffStream)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
if jsonOutput {
|
|
return outputJSON(layer)
|
|
}
|
|
fmt.Printf("%s\n", layer.ID)
|
|
for _, name := range layer.Names {
|
|
fmt.Printf("\t%s\n", name)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func createImage(flags *mflag.FlagSet, action string, m storage.Store, args []string) (int, error) {
|
|
if paramMetadataFile != "" {
|
|
f, err := os.Open(paramMetadataFile)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
b, err := io.ReadAll(f)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
paramMetadata = string(b)
|
|
}
|
|
layer := ""
|
|
if len(args) > 0 {
|
|
layer = args[0]
|
|
}
|
|
imageOptions := &storage.ImageOptions{
|
|
Digest: digest.Digest(paramDigest),
|
|
}
|
|
image, err := m.CreateImage(paramID, paramNames, layer, paramMetadata, imageOptions)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
if jsonOutput {
|
|
return outputJSON(image)
|
|
}
|
|
fmt.Printf("%s\n", image.ID)
|
|
for _, name := range image.Names {
|
|
fmt.Printf("\t%s\n", name)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func createContainer(flags *mflag.FlagSet, action string, m storage.Store, args []string) (int, error) {
|
|
if paramMetadataFile != "" {
|
|
f, err := os.Open(paramMetadataFile)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
b, err := io.ReadAll(f)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
paramMetadata = string(b)
|
|
}
|
|
mappings, err := paramIDMapping()
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
options := &storage.ContainerOptions{IDMappingOptions: *mappings, Volatile: paramVolatile}
|
|
image := args[0]
|
|
container, err := m.CreateContainer(paramID, paramNames, image, paramLayer, paramMetadata, options)
|
|
if err != nil {
|
|
return 1, err
|
|
}
|
|
if jsonOutput {
|
|
return outputJSON(container)
|
|
}
|
|
fmt.Printf("%s\n", container.ID)
|
|
for _, name := range container.Names {
|
|
fmt.Printf("\t%s\n", name)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func init() {
|
|
commands = append(commands, command{
|
|
names: []string{"create-storage-layer"},
|
|
optionsHelp: "[options [...]] [parentLayerNameOrID]",
|
|
usage: "Create a new layer only in the storage driver",
|
|
minArgs: 0,
|
|
maxArgs: 1,
|
|
action: createStorageLayer,
|
|
addFlags: func(flags *mflag.FlagSet, cmd *command) {
|
|
flags.StringVar(¶mMountLabel, []string{"-label", "l"}, "", "Mount Label")
|
|
flags.StringVar(¶mID, []string{"-id", "i"}, "", "Layer ID")
|
|
flags.BoolVar(¶mCreateRO, []string{"-readonly", "r"}, false, "Mark as read-only")
|
|
flags.BoolVar(&jsonOutput, []string{"-json", "j"}, jsonOutput, "Prefer JSON output")
|
|
flags.BoolVar(¶mHostUIDMap, []string{"-hostuidmap"}, paramHostUIDMap, "Force host UID map")
|
|
flags.BoolVar(¶mHostGIDMap, []string{"-hostgidmap"}, paramHostGIDMap, "Force host GID map")
|
|
flags.StringVar(¶mUIDMap, []string{"-uidmap"}, "", "UID map")
|
|
flags.StringVar(¶mGIDMap, []string{"-gidmap"}, "", "GID map")
|
|
flags.StringVar(¶mSubUIDMap, []string{"-subuidmap"}, "", "subuid UID map for a user")
|
|
flags.StringVar(¶mSubGIDMap, []string{"-subgidmap"}, "", "subgid GID map for a group")
|
|
},
|
|
})
|
|
commands = append(commands, command{
|
|
names: []string{"create-layer", "createlayer"},
|
|
optionsHelp: "[options [...]] [parentLayerNameOrID]",
|
|
usage: "Create a new layer",
|
|
minArgs: 0,
|
|
maxArgs: 1,
|
|
action: createLayer,
|
|
addFlags: func(flags *mflag.FlagSet, cmd *command) {
|
|
flags.StringVar(¶mMountLabel, []string{"-label", "l"}, "", "Mount Label")
|
|
flags.Var(opts.NewListOptsRef(¶mNames, nil), []string{"-name", "n"}, "Layer name")
|
|
flags.StringVar(¶mID, []string{"-id", "i"}, "", "Layer ID")
|
|
flags.BoolVar(¶mCreateRO, []string{"-readonly", "r"}, false, "Mark as read-only")
|
|
flags.BoolVar(&jsonOutput, []string{"-json", "j"}, jsonOutput, "Prefer JSON output")
|
|
flags.BoolVar(¶mHostUIDMap, []string{"-hostuidmap"}, paramHostUIDMap, "Force host UID map")
|
|
flags.BoolVar(¶mHostGIDMap, []string{"-hostgidmap"}, paramHostGIDMap, "Force host GID map")
|
|
flags.StringVar(¶mUIDMap, []string{"-uidmap"}, "", "UID map")
|
|
flags.StringVar(¶mGIDMap, []string{"-gidmap"}, "", "GID map")
|
|
flags.StringVar(¶mSubUIDMap, []string{"-subuidmap"}, "", "subuid UID map for a user")
|
|
flags.StringVar(¶mSubGIDMap, []string{"-subgidmap"}, "", "subgid GID map for a group")
|
|
},
|
|
})
|
|
commands = append(commands, command{
|
|
names: []string{"import-layer", "importlayer"},
|
|
optionsHelp: "[options [...]] [parentLayerNameOrID]",
|
|
usage: "Import a new layer",
|
|
minArgs: 0,
|
|
maxArgs: 1,
|
|
action: importLayer,
|
|
addFlags: func(flags *mflag.FlagSet, cmd *command) {
|
|
flags.StringVar(¶mMountLabel, []string{"-label", "l"}, "", "Mount Label")
|
|
flags.Var(opts.NewListOptsRef(¶mNames, nil), []string{"-name", "n"}, "Layer name")
|
|
flags.StringVar(¶mID, []string{"-id", "i"}, "", "Layer ID")
|
|
flags.BoolVar(¶mCreateRO, []string{"-readonly", "r"}, false, "Mark as read-only")
|
|
flags.BoolVar(&jsonOutput, []string{"-json", "j"}, jsonOutput, "Prefer JSON output")
|
|
flags.StringVar(&applyDiffFile, []string{"-file", "f"}, "", "Read from file instead of stdin")
|
|
flags.BoolVar(¶mHostUIDMap, []string{"-hostuidmap"}, paramHostUIDMap, "Force host UID map")
|
|
flags.BoolVar(¶mHostGIDMap, []string{"-hostgidmap"}, paramHostGIDMap, "Force host GID map")
|
|
flags.StringVar(¶mUIDMap, []string{"-uidmap"}, "", "UID map")
|
|
flags.StringVar(¶mGIDMap, []string{"-gidmap"}, "", "GID map")
|
|
flags.StringVar(¶mSubUIDMap, []string{"-subuidmap"}, "", "subuid UID map for a user")
|
|
flags.StringVar(¶mSubGIDMap, []string{"-subgidmap"}, "", "subgid GID map for a group")
|
|
},
|
|
})
|
|
commands = append(commands, command{
|
|
names: []string{"create-image", "createimage"},
|
|
optionsHelp: "[options [...]] topLayerNameOrID",
|
|
usage: "Create a new image using layers",
|
|
minArgs: 0,
|
|
maxArgs: 1,
|
|
action: createImage,
|
|
addFlags: func(flags *mflag.FlagSet, cmd *command) {
|
|
flags.Var(opts.NewListOptsRef(¶mNames, nil), []string{"-name", "n"}, "Image name")
|
|
flags.StringVar(¶mID, []string{"-id", "i"}, "", "Image ID")
|
|
flags.StringVar(¶mDigest, []string{"-digest", "d"}, "", "Image Digest")
|
|
flags.StringVar(¶mMetadata, []string{"-metadata", "m"}, "", "Metadata")
|
|
flags.StringVar(¶mMetadataFile, []string{"-metadata-file", "f"}, "", "Metadata File")
|
|
flags.BoolVar(&jsonOutput, []string{"-json", "j"}, jsonOutput, "Prefer JSON output")
|
|
},
|
|
})
|
|
commands = append(commands, command{
|
|
names: []string{"create-container", "createcontainer"},
|
|
optionsHelp: "[options [...]] parentImageNameOrID",
|
|
usage: "Create a new container from an image",
|
|
minArgs: 1,
|
|
maxArgs: 1,
|
|
action: createContainer,
|
|
addFlags: func(flags *mflag.FlagSet, cmd *command) {
|
|
flags.Var(opts.NewListOptsRef(¶mNames, nil), []string{"-name", "n"}, "Container name")
|
|
flags.BoolVar(¶mVolatile, []string{"-volatile"}, false, "Mark as volatile")
|
|
flags.StringVar(¶mID, []string{"-id", "i"}, "", "Container ID")
|
|
flags.StringVar(¶mMetadata, []string{"-metadata", "m"}, "", "Metadata")
|
|
flags.StringVar(¶mMetadataFile, []string{"-metadata-file", "f"}, "", "Metadata File")
|
|
flags.BoolVar(&jsonOutput, []string{"-json", "j"}, jsonOutput, "Prefer JSON output")
|
|
flags.BoolVar(¶mHostUIDMap, []string{"-hostuidmap"}, paramHostUIDMap, "Force host UID map")
|
|
flags.BoolVar(¶mHostGIDMap, []string{"-hostgidmap"}, paramHostGIDMap, "Force host GID map")
|
|
flags.StringVar(¶mUIDMap, []string{"-uidmap"}, "", "UID map")
|
|
flags.StringVar(¶mGIDMap, []string{"-gidmap"}, "", "GID map")
|
|
flags.StringVar(¶mSubUIDMap, []string{"-subuidmap"}, "", "subuid UID map for a user")
|
|
flags.StringVar(¶mSubGIDMap, []string{"-subgidmap"}, "", "subgid GID map for a group")
|
|
},
|
|
})
|
|
}
|