1
0
mirror of https://github.com/rancher/os.git synced 2025-07-18 17:11:04 +00:00

Merge pull request #988 from joshwget/user-docker-improvements

Various User Docker improvements
This commit is contained in:
Darren Shepherd 2016-06-07 22:01:19 -07:00
commit 9dc25fc411

View File

@ -1,6 +1,7 @@
package userdocker package userdocker
import ( import (
"io"
"io/ioutil" "io/ioutil"
"os" "os"
"os/signal" "os/signal"
@ -10,6 +11,8 @@ import (
"golang.org/x/net/context" "golang.org/x/net/context"
"path/filepath"
log "github.com/Sirupsen/logrus" log "github.com/Sirupsen/logrus"
"github.com/docker/engine-api/types" "github.com/docker/engine-api/types"
composeClient "github.com/docker/libcompose/docker/client" composeClient "github.com/docker/libcompose/docker/client"
@ -19,7 +22,6 @@ import (
"github.com/rancher/os/config" "github.com/rancher/os/config"
rosDocker "github.com/rancher/os/docker" rosDocker "github.com/rancher/os/docker"
"github.com/rancher/os/util" "github.com/rancher/os/util"
"path/filepath"
) )
const ( const (
@ -32,15 +34,34 @@ const (
func Main() { func Main() {
cfg := config.LoadConfig() cfg := config.LoadConfig()
if err := startDocker(cfg); err != nil { execID, resp, err := startDocker(cfg)
if err != nil {
log.Fatal(err) log.Fatal(err)
} }
if err := setupTermHandler(); err != nil { process, err := getDockerProcess()
if err != nil {
log.Fatal(err) log.Fatal(err)
} }
select {} handleTerm(process)
// Wait for Docker daemon to exit
io.Copy(ioutil.Discard, resp.Reader)
resp.Close()
client, err := rosDocker.NewSystemClient()
if err != nil {
log.Fatal(err)
}
state, err := client.ContainerExecInspect(context.Background(), execID)
if err != nil {
log.Fatal(err)
}
// Proxy exit code
os.Exit(state.ExitCode)
} }
func writeCerts(cfg *config.CloudConfig) error { func writeCerts(cfg *config.CloudConfig) error {
@ -73,7 +94,7 @@ func writeCerts(cfg *config.CloudConfig) error {
return nil return nil
} }
func startDocker(cfg *config.CloudConfig) error { func startDocker(cfg *config.CloudConfig) (string, types.HijackedResponse, error) {
storageContext := cfg.Rancher.Docker.StorageContext storageContext := cfg.Rancher.Docker.StorageContext
if storageContext == "" { if storageContext == "" {
storageContext = DEFAULT_STORAGE_CONTEXT storageContext = DEFAULT_STORAGE_CONTEXT
@ -83,23 +104,23 @@ func startDocker(cfg *config.CloudConfig) error {
p, err := compose.GetProject(cfg, true) p, err := compose.GetProject(cfg, true)
if err != nil { if err != nil {
return err return "", types.HijackedResponse{}, err
} }
pid, err := waitForPid(storageContext, p) pid, err := waitForPid(storageContext, p)
if err != nil { if err != nil {
return err return "", types.HijackedResponse{}, err
} }
log.Infof("%s PID %d", storageContext, pid) log.Infof("%s PID %d", storageContext, pid)
client, err := rosDocker.NewSystemClient() client, err := rosDocker.NewSystemClient()
if err != nil { if err != nil {
return err return "", types.HijackedResponse{}, err
} }
if err := os.Remove(DOCKER_PID_FILE); err != nil && !os.IsNotExist(err) { if err := os.Remove(DOCKER_PID_FILE); err != nil && !os.IsNotExist(err) {
return err return "", types.HijackedResponse{}, err
} }
dockerCfg := cfg.Rancher.Docker dockerCfg := cfg.Rancher.Docker
@ -110,7 +131,7 @@ func startDocker(cfg *config.CloudConfig) error {
if dockerCfg.TLS { if dockerCfg.TLS {
if err := writeCerts(cfg); err != nil { if err := writeCerts(cfg); err != nil {
return err return "", types.HijackedResponse{}, err
} }
} }
@ -124,49 +145,50 @@ func startDocker(cfg *config.CloudConfig) error {
resp, err := client.ContainerExecCreate(context.Background(), types.ExecConfig{ resp, err := client.ContainerExecCreate(context.Background(), types.ExecConfig{
Container: storageContext, Container: storageContext,
Privileged: true, Privileged: true,
AttachStdin: true,
AttachStderr: true, AttachStderr: true,
AttachStdout: true, AttachStdout: true,
Detach: false,
Cmd: cmd, Cmd: cmd,
}) })
if err != nil { if err != nil {
return err return "", types.HijackedResponse{}, err
}
attachResp, err := client.ContainerExecAttach(context.Background(), resp.ID, types.ExecConfig{
Detach: false,
AttachStderr: true,
AttachStdout: true,
})
if err != nil {
return "", types.HijackedResponse{}, err
} }
if err := client.ContainerExecStart(context.Background(), resp.ID, types.ExecStartCheck{ if err := client.ContainerExecStart(context.Background(), resp.ID, types.ExecStartCheck{
Detach: false, Detach: false,
}); err != nil { }); err != nil {
return err return "", types.HijackedResponse{}, err
} }
return nil return resp.ID, attachResp, nil
} }
func setupTermHandler() error { func getDockerProcess() (*os.Process, error) {
pidBytes, err := waitForFile(DOCKER_PID_FILE) pidBytes, err := waitForFile(DOCKER_PID_FILE)
if err != nil { if err != nil {
return err return nil, err
} }
dockerPid, err := strconv.Atoi(string(pidBytes)) dockerPid, err := strconv.Atoi(string(pidBytes))
if err != nil { if err != nil {
return err return nil, err
} }
process, err := os.FindProcess(dockerPid) return os.FindProcess(dockerPid)
if err != nil {
return err
}
handleTerm(process)
return nil
} }
func handleTerm(p *os.Process) { func handleTerm(process *os.Process) {
term := make(chan os.Signal) term := make(chan os.Signal)
signal.Notify(term, syscall.SIGTERM) signal.Notify(term, syscall.SIGTERM)
go func() { go func() {
<-term <-term
p.Signal(syscall.SIGTERM) process.Signal(syscall.SIGTERM)
os.Exit(0)
}() }()
} }