Merge pull request #333 from avagin/api-console
new-api: add Console to ProcessConfig
This commit is contained in:
commit
cc411c37bd
|
@ -100,7 +100,7 @@ func (c *linuxContainer) StartProcess(config *ProcessConfig) (int, error) {
|
||||||
|
|
||||||
if state != configs.Destroyed {
|
if state != configs.Destroyed {
|
||||||
glog.Info("start new container process")
|
glog.Info("start new container process")
|
||||||
return namespaces.ExecIn(config.Args, config.Env, cmd, c.config, c.state)
|
return namespaces.ExecIn(config.Args, config.Env, config.Console, cmd, c.config, c.state)
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := c.startInitProcess(cmd, config); err != nil {
|
if err := c.startInitProcess(cmd, config); err != nil {
|
||||||
|
@ -134,7 +134,7 @@ func (c *linuxContainer) updateStateFile() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *linuxContainer) startInitProcess(cmd *exec.Cmd, config *ProcessConfig) error {
|
func (c *linuxContainer) startInitProcess(cmd *exec.Cmd, config *ProcessConfig) error {
|
||||||
err := namespaces.Exec(config.Args, config.Env, cmd, c.config, c.cgroupManager, c.state)
|
err := namespaces.Exec(config.Args, config.Env, config.Console, cmd, c.config, c.cgroupManager, c.state)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,7 +19,7 @@ import (
|
||||||
// Move this to libcontainer package.
|
// Move this to libcontainer package.
|
||||||
// Exec performs setup outside of a namespace so that a container can be
|
// Exec performs setup outside of a namespace so that a container can be
|
||||||
// executed. Exec is a high level function for working with container namespaces.
|
// executed. Exec is a high level function for working with container namespaces.
|
||||||
func Exec(args []string, env []string, command *exec.Cmd, container *configs.Config, cgroupManager cgroups.Manager, state *configs.State) error {
|
func Exec(args []string, env []string, console string, command *exec.Cmd, container *configs.Config, cgroupManager cgroups.Manager, state *configs.State) error {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
// create a pipe so that we can syncronize with the namespaced process and
|
// create a pipe so that we can syncronize with the namespaced process and
|
||||||
|
@ -54,8 +54,9 @@ func Exec(args []string, env []string, command *exec.Cmd, container *configs.Con
|
||||||
}
|
}
|
||||||
|
|
||||||
process := processArgs{
|
process := processArgs{
|
||||||
Env: append(env[0:], container.Env...),
|
Env: append(env[0:], container.Env...),
|
||||||
Args: args,
|
Args: args,
|
||||||
|
ConsolePath: console,
|
||||||
}
|
}
|
||||||
if err := encoder.Encode(process); err != nil {
|
if err := encoder.Encode(process); err != nil {
|
||||||
return terminate(err)
|
return terminate(err)
|
||||||
|
|
|
@ -18,7 +18,7 @@ import (
|
||||||
|
|
||||||
// ExecIn reexec's cmd with _LIBCONTAINER_INITPID=PID so that it is able to run the
|
// ExecIn reexec's cmd with _LIBCONTAINER_INITPID=PID so that it is able to run the
|
||||||
// setns code in a single threaded environment joining the existing containers' namespaces.
|
// setns code in a single threaded environment joining the existing containers' namespaces.
|
||||||
func ExecIn(args []string, env []string, cmd *exec.Cmd, container *configs.Config, state *configs.State) (int, error) {
|
func ExecIn(args []string, env []string, console string, cmd *exec.Cmd, container *configs.Config, state *configs.State) (int, error) {
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
parent, child, err := newInitPipe()
|
parent, child, err := newInitPipe()
|
||||||
|
@ -50,8 +50,9 @@ func ExecIn(args []string, env []string, cmd *exec.Cmd, container *configs.Confi
|
||||||
}
|
}
|
||||||
|
|
||||||
process := processArgs{
|
process := processArgs{
|
||||||
Env: append(env[0:], container.Env...),
|
Env: append(env[0:], container.Env...),
|
||||||
Args: args,
|
Args: args,
|
||||||
|
ConsolePath: console,
|
||||||
}
|
}
|
||||||
if err := encoder.Encode(process); err != nil {
|
if err := encoder.Encode(process); err != nil {
|
||||||
return terminate(err)
|
return terminate(err)
|
||||||
|
|
|
@ -3,13 +3,16 @@ package main
|
||||||
import (
|
import (
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"io"
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
||||||
"github.com/codegangsta/cli"
|
"github.com/codegangsta/cli"
|
||||||
|
"github.com/docker/docker/pkg/term"
|
||||||
"github.com/docker/libcontainer"
|
"github.com/docker/libcontainer"
|
||||||
"github.com/docker/libcontainer/configs"
|
"github.com/docker/libcontainer/configs"
|
||||||
|
consolepkg "github.com/docker/libcontainer/console"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -50,21 +53,55 @@ func getContainer(context *cli.Context) (libcontainer.Container, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func execAction(context *cli.Context) {
|
func execAction(context *cli.Context) {
|
||||||
var exitCode int
|
var (
|
||||||
|
master *os.File
|
||||||
|
console string
|
||||||
|
err error
|
||||||
|
|
||||||
process := &libcontainer.ProcessConfig{
|
sigc = make(chan os.Signal, 10)
|
||||||
Args: context.Args(),
|
|
||||||
Env: context.StringSlice("env"),
|
stdin = os.Stdin
|
||||||
Stdin: os.Stdin,
|
stdout = os.Stdout
|
||||||
Stdout: os.Stdout,
|
stderr = os.Stderr
|
||||||
Stderr: os.Stderr,
|
|
||||||
}
|
exitCode int
|
||||||
|
)
|
||||||
|
|
||||||
container, err := getContainer(context)
|
container, err := getContainer(context)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if container.Config().Tty {
|
||||||
|
stdin = nil
|
||||||
|
stdout = nil
|
||||||
|
stderr = nil
|
||||||
|
|
||||||
|
master, console, err = consolepkg.CreateMasterAndConsole()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
go io.Copy(master, os.Stdin)
|
||||||
|
go io.Copy(os.Stdout, master)
|
||||||
|
|
||||||
|
state, err := term.SetRawTerminal(os.Stdin.Fd())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
defer term.RestoreTerminal(os.Stdin.Fd(), state)
|
||||||
|
}
|
||||||
|
|
||||||
|
process := &libcontainer.ProcessConfig{
|
||||||
|
Args: context.Args(),
|
||||||
|
Env: context.StringSlice("env"),
|
||||||
|
Stdin: stdin,
|
||||||
|
Stdout: stdout,
|
||||||
|
Stderr: stderr,
|
||||||
|
Console: console,
|
||||||
|
}
|
||||||
|
|
||||||
pid, err := container.StartProcess(process)
|
pid, err := container.StartProcess(process)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("failed to exec: %s", err)
|
log.Fatalf("failed to exec: %s", err)
|
||||||
|
@ -75,6 +112,19 @@ func execAction(context *cli.Context) {
|
||||||
log.Fatalf("Unable to find the %d process: %s", pid, err)
|
log.Fatalf("Unable to find the %d process: %s", pid, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
resizeTty(master)
|
||||||
|
|
||||||
|
for sig := range sigc {
|
||||||
|
switch sig {
|
||||||
|
case syscall.SIGWINCH:
|
||||||
|
resizeTty(master)
|
||||||
|
default:
|
||||||
|
p.Signal(sig)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
ps, err := p.Wait()
|
ps, err := p.Wait()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Unable to wait the %d process: %s", pid, err)
|
log.Fatalf("Unable to wait the %d process: %s", pid, err)
|
||||||
|
@ -92,3 +142,18 @@ func execAction(context *cli.Context) {
|
||||||
|
|
||||||
os.Exit(exitCode)
|
os.Exit(exitCode)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func resizeTty(master *os.File) {
|
||||||
|
if master == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ws, err := term.GetWinsize(os.Stdin.Fd())
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := term.SetWinsize(master.Fd(), ws); err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -21,4 +21,7 @@ type ProcessConfig struct {
|
||||||
Stdin io.Reader
|
Stdin io.Reader
|
||||||
Stdout io.Writer
|
Stdout io.Writer
|
||||||
Stderr io.Writer
|
Stderr io.Writer
|
||||||
|
|
||||||
|
// Console is the path to the pty slave for use by the master
|
||||||
|
Console string
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue