Skip to content

Commit 4b6ffda

Browse files
Merge pull request #11409 from cdoern/podVolumes
Pod Volumes Support
2 parents 4dd7bfd + 8400533 commit 4b6ffda

File tree

11 files changed

+265
-24
lines changed

11 files changed

+265
-24
lines changed

cmd/podman/common/create.go

Lines changed: 12 additions & 12 deletions
Original file line numberDiff line numberDiff line change
@@ -660,18 +660,6 @@ func DefineCreateFlags(cmd *cobra.Command, cf *entities.ContainerCreateOptions,
660660
)
661661
_ = cmd.RegisterFlagCompletionFunc(mountFlagName, AutocompleteMountFlag)
662662

663-
volumeDesciption := "Bind mount a volume into the container"
664-
if registry.IsRemote() {
665-
volumeDesciption = "Bind mount a volume into the container. Volume src will be on the server machine, not the client"
666-
}
667-
volumeFlagName := "volume"
668-
createFlags.StringArrayVarP(
669-
&cf.Volume,
670-
volumeFlagName, "v", volumes(),
671-
volumeDesciption,
672-
)
673-
_ = cmd.RegisterFlagCompletionFunc(volumeFlagName, AutocompleteVolumeFlag)
674-
675663
volumesFromFlagName := "volumes-from"
676664
createFlags.StringArrayVar(
677665
&cf.VolumesFrom,
@@ -865,4 +853,16 @@ func DefineCreateFlags(cmd *cobra.Command, cf *entities.ContainerCreateOptions,
865853
"PID namespace to use",
866854
)
867855
_ = cmd.RegisterFlagCompletionFunc(pidFlagName, AutocompleteNamespace)
856+
857+
volumeDesciption := "Bind mount a volume into the container"
858+
if registry.IsRemote() {
859+
volumeDesciption = "Bind mount a volume into the container. Volume source will be on the server machine, not the client"
860+
}
861+
volumeFlagName := "volume"
862+
createFlags.StringArrayVarP(
863+
&cf.Volume,
864+
volumeFlagName, "v", volumes(),
865+
volumeDesciption,
866+
)
867+
_ = cmd.RegisterFlagCompletionFunc(volumeFlagName, AutocompleteVolumeFlag)
868868
}

cmd/podman/pods/create.go

Lines changed: 15 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -132,12 +132,7 @@ func create(cmd *cobra.Command, args []string) error {
132132
createOptions.Share = nil
133133
} else {
134134
// reassign certain optios for lbpod api, these need to be populated in spec
135-
createOptions.InfraConmonPidFile = infraOptions.ConmonPIDFile
136-
createOptions.InfraName = infraOptions.Name
137-
createOptions.Hostname = infraOptions.Hostname
138-
createOptions.Cpus = infraOptions.CPUS
139-
createOptions.CpusetCpus = infraOptions.CPUSetCPUs
140-
createOptions.Pid = infraOptions.PID
135+
MapOptions()
141136
flags := cmd.Flags()
142137
infraOptions.Net, err = common.NetFlagsToNetOptions(nil, *flags, false)
143138
if err != nil {
@@ -265,6 +260,10 @@ func create(cmd *cobra.Command, args []string) error {
265260
if err != nil {
266261
return err
267262
}
263+
podSpec.Volumes = podSpec.InfraContainerSpec.Volumes
264+
podSpec.ImageVolumes = podSpec.InfraContainerSpec.ImageVolumes
265+
podSpec.OverlayVolumes = podSpec.InfraContainerSpec.OverlayVolumes
266+
podSpec.Mounts = podSpec.InfraContainerSpec.Mounts
268267
}
269268
PodSpec := entities.PodSpec{PodSpecGen: *podSpec}
270269
response, err := registry.ContainerEngine().PodCreate(context.Background(), PodSpec)
@@ -291,3 +290,13 @@ func replacePod(name string) error {
291290
}
292291
return removePods([]string{name}, rmOptions, false)
293292
}
293+
294+
func MapOptions() {
295+
createOptions.Cpus = infraOptions.CPUS
296+
createOptions.CpusetCpus = infraOptions.CPUSetCPUs
297+
createOptions.Hostname = infraOptions.Hostname
298+
createOptions.InfraConmonPidFile = infraOptions.ConmonPIDFile
299+
createOptions.InfraName = infraOptions.Name
300+
createOptions.Pid = infraOptions.PID
301+
createOptions.Volume = infraOptions.Volume
302+
}

docs/source/markdown/podman-pod-create.1.md

Lines changed: 162 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -200,6 +200,168 @@ Valid _mode_ values are:
200200
- *host*: run in the user namespace of the caller. The processes running in the container will have the same privileges on the host as any other process launched by the calling user (default).
201201
- *keep-id*: creates a user namespace where the current rootless user's UID:GID are mapped to the same values in the container. This option is ignored for containers created by the root user.
202202

203+
#### **--volume**, **-v**[=*[[SOURCE-VOLUME|HOST-DIR:]CONTAINER-DIR[:OPTIONS]]*]
204+
205+
Create a bind mount. If you specify, ` -v /HOST-DIR:/CONTAINER-DIR`, Podman
206+
bind mounts `/HOST-DIR` in the host to `/CONTAINER-DIR` in the Podman
207+
container. Similarly, `-v SOURCE-VOLUME:/CONTAINER-DIR` will mount the volume
208+
in the host to the container. If no such named volume exists, Podman will
209+
create one. The `OPTIONS` are a comma-separated list and can be: <sup>[[1]](#Footnote1)</sup> (Note when using the remote client, the volumes will be mounted from the remote server, not necessarily the client machine.)
210+
211+
The _options_ is a comma-separated list and can be:
212+
213+
* **rw**|**ro**
214+
* **z**|**Z**
215+
* [**r**]**shared**|[**r**]**slave**|[**r**]**private**[**r**]**unbindable**
216+
* [**r**]**bind**
217+
* [**no**]**exec**
218+
* [**no**]**dev**
219+
* [**no**]**suid**
220+
* [**O**]
221+
* [**U**]
222+
223+
The `CONTAINER-DIR` must be an absolute path such as `/src/docs`. The volume
224+
will be mounted into the container at this directory.
225+
226+
Volumes may specify a source as well, as either a directory on the host
227+
or the name of a named volume. If no source is given, the volume will be created as an
228+
anonymously named volume with a randomly generated name, and will be removed when
229+
the pod is removed via the `--rm` flag or `podman rm --volumes` commands.
230+
231+
If a volume source is specified, it must be a path on the host or the name of a
232+
named volume. Host paths are allowed to be absolute or relative; relative paths
233+
are resolved relative to the directory Podman is run in. If the source does not
234+
exist, Podman will return an error. Users must pre-create the source files or
235+
directories.
236+
237+
Any source that does not begin with a `.` or `/` will be treated as the name of
238+
a named volume. If a volume with that name does not exist, it will be created.
239+
Volumes created with names are not anonymous, and they are not removed by the `--rm`
240+
option and the `podman rm --volumes` command.
241+
242+
You can specify multiple **-v** options to mount one or more volumes into a
243+
pod.
244+
245+
`Write Protected Volume Mounts`
246+
247+
You can add `:ro` or `:rw` suffix to a volume to mount it read-only or
248+
read-write mode, respectively. By default, the volumes are mounted read-write.
249+
See examples.
250+
251+
`Chowning Volume Mounts`
252+
253+
By default, Podman does not change the owner and group of source volume
254+
directories mounted into containers. If a pod is created in a new user
255+
namespace, the UID and GID in the container may correspond to another UID and
256+
GID on the host.
257+
258+
The `:U` suffix tells Podman to use the correct host UID and GID based on the
259+
UID and GID within the pod, to change recursively the owner and group of
260+
the source volume.
261+
262+
**Warning** use with caution since this will modify the host filesystem.
263+
264+
`Labeling Volume Mounts`
265+
266+
Labeling systems like SELinux require that proper labels are placed on volume
267+
content mounted into a pod. Without a label, the security system might
268+
prevent the processes running inside the pod from using the content. By
269+
default, Podman does not change the labels set by the OS.
270+
271+
To change a label in the pod context, you can add either of two suffixes
272+
`:z` or `:Z` to the volume mount. These suffixes tell Podman to relabel file
273+
objects on the shared volumes. The `z` option tells Podman that two pods
274+
share the volume content. As a result, Podman labels the content with a shared
275+
content label. Shared volume labels allow all containers to read/write content.
276+
The `Z` option tells Podman to label the content with a private unshared label.
277+
Only the current pod can use a private volume.
278+
279+
`Overlay Volume Mounts`
280+
281+
The `:O` flag tells Podman to mount the directory from the host as a
282+
temporary storage using the `overlay file system`. The pod processes
283+
can modify content within the mountpoint which is stored in the
284+
container storage in a separate directory. In overlay terms, the source
285+
directory will be the lower, and the container storage directory will be the
286+
upper. Modifications to the mount point are destroyed when the pod
287+
finishes executing, similar to a tmpfs mount point being unmounted.
288+
289+
Subsequent executions of the container will see the original source directory
290+
content, any changes from previous pod executions no longer exist.
291+
292+
One use case of the overlay mount is sharing the package cache from the
293+
host into the container to allow speeding up builds.
294+
295+
Note:
296+
297+
- The `O` flag conflicts with other options listed above.
298+
Content mounted into the container is labeled with the private label.
299+
On SELinux systems, labels in the source directory must be readable
300+
by the infra container label. Usually containers can read/execute `container_share_t`
301+
and can read/write `container_file_t`. If you cannot change the labels on a
302+
source volume, SELinux container separation must be disabled for the infra container/pod
303+
to work.
304+
- The source directory mounted into the pod with an overlay mount
305+
should not be modified, it can cause unexpected failures. It is recommended
306+
that you do not modify the directory until the container finishes running.
307+
308+
`Mounts propagation`
309+
310+
By default bind mounted volumes are `private`. That means any mounts done
311+
inside pod will not be visible on host and vice versa. One can change
312+
this behavior by specifying a volume mount propagation property. Making a
313+
volume `shared` mounts done under that volume inside pod will be
314+
visible on host and vice versa. Making a volume `slave` enables only one
315+
way mount propagation and that is mounts done on host under that volume
316+
will be visible inside container but not the other way around. <sup>[[1]](#Footnote1)</sup>
317+
318+
To control mount propagation property of a volume one can use the [**r**]**shared**,
319+
[**r**]**slave**, [**r**]**private** or the [**r**]**unbindable** propagation flag.
320+
Propagation property can be specified only for bind mounted volumes and not for
321+
internal volumes or named volumes. For mount propagation to work the source mount
322+
point (the mount point where source dir is mounted on) has to have the right propagation
323+
properties. For shared volumes, the source mount point has to be shared. And for
324+
slave volumes, the source mount point has to be either shared or slave.
325+
<sup>[[1]](#Footnote1)</sup>
326+
327+
If you want to recursively mount a volume and all of its submounts into a
328+
pod, then you can use the `rbind` option. By default the bind option is
329+
used, and submounts of the source directory will not be mounted into the
330+
pod.
331+
332+
Mounting the volume with the `nosuid` options means that SUID applications on
333+
the volume will not be able to change their privilege. By default volumes
334+
are mounted with `nosuid`.
335+
336+
Mounting the volume with the noexec option means that no executables on the
337+
volume will be able to executed within the pod.
338+
339+
Mounting the volume with the nodev option means that no devices on the volume
340+
will be able to be used by processes within the pod. By default volumes
341+
are mounted with `nodev`.
342+
343+
If the `<source-dir>` is a mount point, then "dev", "suid", and "exec" options are
344+
ignored by the kernel.
345+
346+
Use `df <source-dir>` to figure out the source mount and then use
347+
`findmnt -o TARGET,PROPAGATION <source-mount-dir>` to figure out propagation
348+
properties of source mount. If `findmnt` utility is not available, then one
349+
can look at the mount entry for the source mount point in `/proc/self/mountinfo`. Look
350+
at `optional fields` and see if any propagation properties are specified.
351+
`shared:X` means mount is `shared`, `master:X` means mount is `slave` and if
352+
nothing is there that means mount is `private`. <sup>[[1]](#Footnote1)</sup>
353+
354+
To change propagation properties of a mount point use `mount` command. For
355+
example, if one wants to bind mount source directory `/foo` one can do
356+
`mount --bind /foo /foo` and `mount --make-private --make-shared /foo`. This
357+
will convert /foo into a `shared` mount point. Alternatively one can directly
358+
change propagation properties of source mount. Say `/` is source mount for
359+
`/foo`, then use `mount --make-shared /` to convert `/` into a `shared` mount.
360+
361+
Note: if the user only has access rights via a group, accessing the volume
362+
from inside a rootless pod will fail.
363+
364+
203365
## EXAMPLES
204366

205367
```

libpod/container_inspect.go

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ func (c *Container) getContainerInspectData(size bool, driverData *define.Driver
9292
}
9393

9494
namedVolumes, mounts := c.sortUserVolumes(ctrSpec)
95-
inspectMounts, err := c.getInspectMounts(namedVolumes, c.config.ImageVolumes, mounts)
95+
inspectMounts, err := c.GetInspectMounts(namedVolumes, c.config.ImageVolumes, mounts)
9696
if err != nil {
9797
return nil, err
9898
}
@@ -194,7 +194,7 @@ func (c *Container) getContainerInspectData(size bool, driverData *define.Driver
194194
// Get inspect-formatted mounts list.
195195
// Only includes user-specified mounts. Only includes bind mounts and named
196196
// volumes, not tmpfs volumes.
197-
func (c *Container) getInspectMounts(namedVolumes []*ContainerNamedVolume, imageVolumes []*ContainerImageVolume, mounts []spec.Mount) ([]define.InspectMount, error) {
197+
func (c *Container) GetInspectMounts(namedVolumes []*ContainerNamedVolume, imageVolumes []*ContainerImageVolume, mounts []spec.Mount) ([]define.InspectMount, error) {
198198
inspectMounts := []define.InspectMount{}
199199

200200
// No mounts, return early

libpod/define/pod_inspect.go

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,8 @@ type InspectPodData struct {
5757
CPUQuota int64 `json:"cpu_quota,omitempty"`
5858
// CPUSetCPUs contains linux specific CPU data for the pod
5959
CPUSetCPUs string `json:"cpuset_cpus,omitempty"`
60+
// Mounts contains volume related information for the pod
61+
Mounts []InspectMount `json:"mounts,omitempty"`
6062
}
6163

6264
// InspectPodInfraConfig contains the configuration of the pod's infra

libpod/pod_api.go

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -582,6 +582,7 @@ func (p *Pod) Inspect() (*define.InspectPodData, error) {
582582
// Infra config contains detailed information on the pod's infra
583583
// container.
584584
var infraConfig *define.InspectPodInfraConfig
585+
var inspectMounts []define.InspectMount
585586
if p.state.InfraContainerID != "" {
586587
infra, err := p.runtime.GetContainer(p.state.InfraContainerID)
587588
if err != nil {
@@ -597,6 +598,11 @@ func (p *Pod) Inspect() (*define.InspectPodData, error) {
597598
infraConfig.CPUSetCPUs = p.ResourceLim().CPU.Cpus
598599
infraConfig.PidNS = p.PidMode()
599600
infraConfig.UserNS = p.UserNSMode()
601+
namedVolumes, mounts := infra.sortUserVolumes(infra.Config().Spec)
602+
inspectMounts, err = infra.GetInspectMounts(namedVolumes, infra.config.ImageVolumes, mounts)
603+
if err != nil {
604+
return nil, err
605+
}
600606

601607
if len(infra.Config().ContainerNetworkConfig.DNSServer) > 0 {
602608
infraConfig.DNSServer = make([]string, 0, len(infra.Config().ContainerNetworkConfig.DNSServer))
@@ -645,6 +651,7 @@ func (p *Pod) Inspect() (*define.InspectPodData, error) {
645651
CPUSetCPUs: p.ResourceLim().CPU.Cpus,
646652
CPUPeriod: p.CPUPeriod(),
647653
CPUQuota: p.CPUQuota(),
654+
Mounts: inspectMounts,
648655
}
649656

650657
return &inspectData, nil

pkg/api/handlers/libpod/pods.go

Lines changed: 1 addition & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -52,13 +52,11 @@ func PodCreate(w http.ResponseWriter, r *http.Request) {
5252
utils.Error(w, "Something went wrong.", http.StatusInternalServerError, errors.Wrap(err, "failed to decode specgen"))
5353
return
5454
}
55-
tempSpec := &specgen.SpecGenerator{} // temporary spec since infra cannot be decoded into
56-
err = json.Unmarshal(out, tempSpec) // unmarhal matching options
55+
err = json.Unmarshal(out, psg.InfraContainerSpec) // unmarhal matching options
5756
if err != nil {
5857
utils.Error(w, "Something went wrong.", http.StatusInternalServerError, errors.Wrap(err, "failed to decode specgen"))
5958
return
6059
}
61-
psg.InfraContainerSpec = tempSpec // set infra spec equal to temp
6260
// a few extra that do not have the same json tags
6361
psg.InfraContainerSpec.Name = psg.InfraName
6462
psg.InfraContainerSpec.ConmonPidFile = psg.InfraConmonPidFile

pkg/domain/entities/pods.go

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -131,6 +131,7 @@ type PodCreateOptions struct {
131131
Cpus float64
132132
CpusetCpus string
133133
Userns specgen.Namespace
134+
Volume []string
134135
}
135136

136137
// PodLogsOptions describes the options to extract pod logs.

pkg/specgen/generate/container_create.go

Lines changed: 13 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -28,15 +28,27 @@ func MakeContainer(ctx context.Context, rt *libpod.Runtime, s *specgen.SpecGener
2828
return nil, nil, nil, err
2929
}
3030

31-
// If joining a pod, retrieve the pod for use.
31+
// If joining a pod, retrieve the pod for use, and its infra container
3232
var pod *libpod.Pod
33+
var cont *libpod.Container
34+
var config *libpod.ContainerConfig
3335
if s.Pod != "" {
3436
pod, err = rt.LookupPod(s.Pod)
3537
if err != nil {
3638
return nil, nil, nil, errors.Wrapf(err, "error retrieving pod %s", s.Pod)
3739
}
40+
if pod.HasInfraContainer() {
41+
cont, err = pod.InfraContainer()
42+
if err != nil {
43+
return nil, nil, nil, err
44+
}
45+
config = cont.Config()
46+
}
3847
}
3948

49+
if config != nil && (len(config.NamedVolumes) > 0 || len(config.UserVolumes) > 0 || len(config.ImageVolumes) > 0 || len(config.OverlayVolumes) > 0) {
50+
s.VolumesFrom = append(s.VolumesFrom, config.ID)
51+
}
4052
// Set defaults for unset namespaces
4153
if s.PidNS.IsDefault() {
4254
defaultNS, err := GetDefaultNamespaceMode("pid", rtc, pod)

pkg/specgen/podspecgen.go

Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,22 @@ type PodBasicConfig struct {
7272
// Any containers created within the pod will inherit the pod's userns settings.
7373
// Optional
7474
Userns Namespace `json:"userns,omitempty"`
75+
// Mounts are mounts that will be added to the pod.
76+
// These will supersede Image Volumes and VolumesFrom (WIP) volumes where
77+
// there are conflicts.
78+
// Optional.
79+
Mounts []spec.Mount `json:"mounts,omitempty"`
80+
// Volumes are named volumes that will be added to the pod.
81+
// These will supersede Image Volumes and VolumesFrom (WIP) volumes where
82+
// there are conflicts.
83+
// Optional.
84+
Volumes []*NamedVolume `json:"volumes,omitempty"`
85+
// Overlay volumes are named volumes that will be added to the pod.
86+
// Optional.
87+
OverlayVolumes []*OverlayVolume `json:"overlay_volumes,omitempty"`
88+
// Image volumes bind-mount a container-image mount into the pod's infra container.
89+
// Optional.
90+
ImageVolumes []*ImageVolume `json:"image_volumes,omitempty"`
7591
}
7692

7793
// PodNetworkConfig contains networking configuration for a pod.

0 commit comments

Comments
 (0)