2015-06-30 07:49:13 +08:00
|
|
|
// +build linux
|
|
|
|
|
2015-06-22 10:31:12 +08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2015-08-19 17:28:51 +08:00
|
|
|
"fmt"
|
2015-06-22 10:31:12 +08:00
|
|
|
"io"
|
|
|
|
"os"
|
2015-09-19 02:35:23 +08:00
|
|
|
"syscall"
|
2015-06-22 10:31:12 +08:00
|
|
|
|
|
|
|
"github.com/docker/docker/pkg/term"
|
|
|
|
"github.com/opencontainers/runc/libcontainer"
|
|
|
|
)
|
|
|
|
|
2015-07-13 09:08:16 +08:00
|
|
|
// newTty creates a new tty for use with the container. If a tty is not to be
|
2015-06-22 10:31:12 +08:00
|
|
|
// created for the process, pipes are created so that the TTY of the parent
|
|
|
|
// process are not inherited by the container.
|
|
|
|
func newTty(create bool, p *libcontainer.Process, rootuid int) (*tty, error) {
|
|
|
|
if create {
|
|
|
|
return createTty(p, rootuid)
|
|
|
|
}
|
2015-09-19 02:35:23 +08:00
|
|
|
return createStdioPipes(p, rootuid)
|
2015-06-22 10:31:12 +08:00
|
|
|
}
|
|
|
|
|
2015-06-24 11:29:50 +08:00
|
|
|
// setup standard pipes so that the TTY of the calling runc process
|
2015-06-22 10:31:12 +08:00
|
|
|
// is not inherited by the container.
|
2015-09-19 02:35:23 +08:00
|
|
|
func createStdioPipes(p *libcontainer.Process, rootuid int) (*tty, error) {
|
|
|
|
var (
|
|
|
|
t = &tty{}
|
|
|
|
fds []int
|
|
|
|
)
|
2015-06-22 10:31:12 +08:00
|
|
|
r, w, err := os.Pipe()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-09-19 02:35:23 +08:00
|
|
|
fds = append(fds, int(r.Fd()), int(w.Fd()))
|
2015-06-22 10:31:12 +08:00
|
|
|
go io.Copy(w, os.Stdin)
|
|
|
|
t.closers = append(t.closers, w)
|
|
|
|
p.Stdin = r
|
|
|
|
if r, w, err = os.Pipe(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-09-19 02:35:23 +08:00
|
|
|
fds = append(fds, int(r.Fd()), int(w.Fd()))
|
2015-06-22 10:31:12 +08:00
|
|
|
go io.Copy(os.Stdout, r)
|
|
|
|
p.Stdout = w
|
|
|
|
t.closers = append(t.closers, r)
|
|
|
|
if r, w, err = os.Pipe(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-09-19 02:35:23 +08:00
|
|
|
fds = append(fds, int(r.Fd()), int(w.Fd()))
|
2015-06-22 10:31:12 +08:00
|
|
|
go io.Copy(os.Stderr, r)
|
|
|
|
p.Stderr = w
|
|
|
|
t.closers = append(t.closers, r)
|
2015-09-19 02:35:23 +08:00
|
|
|
// change the ownership of the pipe fds incase we are in a user namespace.
|
|
|
|
for _, fd := range fds {
|
|
|
|
if err := syscall.Fchown(fd, rootuid, rootuid); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2015-06-22 10:31:12 +08:00
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func createTty(p *libcontainer.Process, rootuid int) (*tty, error) {
|
|
|
|
console, err := p.NewConsole(rootuid)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
go io.Copy(console, os.Stdin)
|
|
|
|
go io.Copy(os.Stdout, console)
|
|
|
|
state, err := term.SetRawTerminal(os.Stdin.Fd())
|
|
|
|
if err != nil {
|
2015-08-19 17:28:51 +08:00
|
|
|
return nil, fmt.Errorf("failed to set the terminal from the stdin: %v", err)
|
2015-06-22 10:31:12 +08:00
|
|
|
}
|
|
|
|
t := &tty{
|
|
|
|
console: console,
|
|
|
|
state: state,
|
|
|
|
closers: []io.Closer{
|
|
|
|
console,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
p.Stderr = nil
|
|
|
|
p.Stdout = nil
|
|
|
|
p.Stdin = nil
|
|
|
|
return t, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type tty struct {
|
|
|
|
console libcontainer.Console
|
|
|
|
state *term.State
|
|
|
|
closers []io.Closer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *tty) Close() error {
|
|
|
|
for _, c := range t.closers {
|
|
|
|
c.Close()
|
|
|
|
}
|
|
|
|
if t.state != nil {
|
|
|
|
term.RestoreTerminal(os.Stdin.Fd(), t.state)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *tty) resize() error {
|
|
|
|
if t.console == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ws, err := term.GetWinsize(os.Stdin.Fd())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return term.SetWinsize(t.console.Fd(), ws)
|
|
|
|
}
|