Ignore changing /dev/null permissions if used in STDIO

Whenever dev/null is used as one of the main processes STDIO, do not try
to change the permissions on it via fchown because we should not do it
in the first place and also this will fail if the container is supposed
to be readonly.

Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
This commit is contained in:
Michael Crosby 2015-09-22 11:12:55 -07:00
parent ff122b864a
commit 219b6c99e0
1 changed files with 32 additions and 10 deletions

View File

@ -187,16 +187,10 @@ func setupUser(config *initConfig) error {
return err return err
} }
} }
// change the permissions on the STDIO of the current process so that when the user // before we change to the container's user make sure that the processes STDIO
// is changed for the container, it's STDIO of the process matches the user. // is correctly owned by the user that we are switching to.
for _, fd := range []uintptr{ if err := fixStdioPermissions(execUser); err != nil {
os.Stdin.Fd(), return err
os.Stderr.Fd(),
os.Stdout.Fd(),
} {
if err := syscall.Fchown(int(fd), execUser.Uid, execUser.Gid); err != nil {
return err
}
} }
suppGroups := append(execUser.Sgids, addGroups...) suppGroups := append(execUser.Sgids, addGroups...)
if err := syscall.Setgroups(suppGroups); err != nil { if err := syscall.Setgroups(suppGroups); err != nil {
@ -218,6 +212,34 @@ func setupUser(config *initConfig) error {
return nil return nil
} }
// fixStdioPermissions fixes the permissions of PID 1's STDIO within the container to the specified user.
// The ownership needs to match because it is created outside of the container and needs to be
// localized.
func fixStdioPermissions(u *user.ExecUser) error {
var null syscall.Stat_t
if err := syscall.Stat("/dev/null", &null); err != nil {
return err
}
for _, fd := range []uintptr{
os.Stdin.Fd(),
os.Stderr.Fd(),
os.Stdout.Fd(),
} {
var s syscall.Stat_t
if err := syscall.Fstat(int(fd), &s); err != nil {
return err
}
// skip chown of /dev/null if it was used as one of the STDIO fds.
if s.Rdev == null.Rdev {
continue
}
if err := syscall.Fchown(int(fd), u.Uid, u.Gid); err != nil {
return err
}
}
return nil
}
// setupNetwork sets up and initializes any network interface inside the container. // setupNetwork sets up and initializes any network interface inside the container.
func setupNetwork(config *initConfig) error { func setupNetwork(config *initConfig) error {
for _, config := range config.Networks { for _, config := range config.Networks {