2014-08-01 13:34:06 -04:00
|
|
|
// +build daemon
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2015-01-21 19:55:05 -05:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
2014-10-24 13:12:35 -04:00
|
|
|
log "github.com/Sirupsen/logrus"
|
2015-02-04 16:22:38 -05:00
|
|
|
"github.com/docker/docker/autogen/dockerversion"
|
2014-08-11 11:44:31 -04:00
|
|
|
"github.com/docker/docker/builder"
|
2014-08-01 13:34:06 -04:00
|
|
|
"github.com/docker/docker/builtins"
|
2014-08-08 05:12:39 -04:00
|
|
|
"github.com/docker/docker/daemon"
|
2014-08-08 16:18:18 -04:00
|
|
|
_ "github.com/docker/docker/daemon/execdriver/lxc"
|
|
|
|
_ "github.com/docker/docker/daemon/execdriver/native"
|
2014-08-01 13:34:06 -04:00
|
|
|
"github.com/docker/docker/engine"
|
2015-02-06 13:18:49 -05:00
|
|
|
"github.com/docker/docker/pkg/homedir"
|
2014-08-01 13:34:06 -04:00
|
|
|
flag "github.com/docker/docker/pkg/mflag"
|
2014-08-06 04:12:22 -04:00
|
|
|
"github.com/docker/docker/pkg/signal"
|
2014-08-20 11:31:24 -04:00
|
|
|
"github.com/docker/docker/registry"
|
2015-01-21 19:55:05 -05:00
|
|
|
"github.com/docker/docker/utils"
|
2014-08-01 13:34:06 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
const CanDaemon = true
|
|
|
|
|
2014-08-09 21:18:32 -04:00
|
|
|
var (
|
2014-10-06 21:54:52 -04:00
|
|
|
daemonCfg = &daemon.Config{}
|
|
|
|
registryCfg = ®istry.Options{}
|
2014-08-09 21:18:32 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
daemonCfg.InstallFlags()
|
2014-10-06 21:54:52 -04:00
|
|
|
registryCfg.InstallFlags()
|
2014-08-09 21:18:32 -04:00
|
|
|
}
|
|
|
|
|
2015-01-22 14:22:31 -05:00
|
|
|
func migrateKey() (err error) {
|
2015-01-21 19:55:05 -05:00
|
|
|
// Migrate trust key if exists at ~/.docker/key.json and owned by current user
|
2015-02-06 13:18:49 -05:00
|
|
|
oldPath := filepath.Join(homedir.Get(), ".docker", defaultTrustKeyFile)
|
2015-01-21 19:55:05 -05:00
|
|
|
newPath := filepath.Join(getDaemonConfDir(), defaultTrustKeyFile)
|
2015-01-22 14:22:31 -05:00
|
|
|
if _, statErr := os.Stat(newPath); os.IsNotExist(statErr) && utils.IsFileOwner(oldPath) {
|
|
|
|
defer func() {
|
|
|
|
// Ensure old path is removed if no error occurred
|
|
|
|
if err == nil {
|
|
|
|
err = os.Remove(oldPath)
|
|
|
|
} else {
|
|
|
|
log.Warnf("Key migration failed, key file not removed at %s", oldPath)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2015-01-21 19:55:05 -05:00
|
|
|
if err := os.MkdirAll(getDaemonConfDir(), os.FileMode(0644)); err != nil {
|
2015-01-22 14:22:31 -05:00
|
|
|
return fmt.Errorf("Unable to create daemon configuration directory: %s", err)
|
2015-01-21 19:55:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
newFile, err := os.OpenFile(newPath, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error creating key file %q: %s", newPath, err)
|
|
|
|
}
|
|
|
|
defer newFile.Close()
|
|
|
|
|
|
|
|
oldFile, err := os.Open(oldPath)
|
|
|
|
if err != nil {
|
2015-01-22 14:22:31 -05:00
|
|
|
return fmt.Errorf("error opening key file %q: %s", oldPath, err)
|
2015-01-21 19:55:05 -05:00
|
|
|
}
|
2015-01-22 14:22:31 -05:00
|
|
|
defer oldFile.Close()
|
2015-01-21 19:55:05 -05:00
|
|
|
|
|
|
|
if _, err := io.Copy(newFile, oldFile); err != nil {
|
|
|
|
return fmt.Errorf("error copying key: %s", err)
|
|
|
|
}
|
|
|
|
|
2015-01-22 14:22:31 -05:00
|
|
|
log.Infof("Migrated key from %s to %s", oldPath, newPath)
|
2015-01-21 19:55:05 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-08-01 13:34:06 -04:00
|
|
|
func mainDaemon() {
|
|
|
|
if flag.NArg() != 0 {
|
|
|
|
flag.Usage()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
eng := engine.New()
|
2014-08-06 04:12:22 -04:00
|
|
|
signal.Trap(eng.Shutdown)
|
2014-08-20 11:31:24 -04:00
|
|
|
|
2015-01-21 19:55:05 -05:00
|
|
|
if err := migrateKey(); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
2014-11-17 14:23:41 -05:00
|
|
|
daemonCfg.TrustKeyPath = *flTrustKey
|
|
|
|
|
2014-08-01 13:34:06 -04:00
|
|
|
// Load builtins
|
|
|
|
if err := builtins.Register(eng); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2014-08-20 11:31:24 -04:00
|
|
|
// load registry service
|
2014-10-06 21:54:52 -04:00
|
|
|
if err := registry.NewService(registryCfg).Install(eng); err != nil {
|
2014-08-20 11:31:24 -04:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2014-08-01 13:34:06 -04:00
|
|
|
// load the daemon in the background so we can immediately start
|
|
|
|
// the http api so that connections don't fail while the daemon
|
|
|
|
// is booting
|
2015-03-06 15:44:31 -05:00
|
|
|
daemonWait := make(chan struct{})
|
2014-08-01 13:34:06 -04:00
|
|
|
go func() {
|
2015-03-10 15:18:43 -04:00
|
|
|
defer close(daemonWait)
|
|
|
|
|
2014-08-09 21:18:32 -04:00
|
|
|
d, err := daemon.NewDaemon(daemonCfg, eng)
|
2014-08-08 05:12:39 -04:00
|
|
|
if err != nil {
|
2015-03-06 15:44:31 -05:00
|
|
|
log.Error(err)
|
|
|
|
return
|
2014-08-08 05:12:39 -04:00
|
|
|
}
|
2015-03-06 15:44:31 -05:00
|
|
|
|
2014-10-21 03:00:25 -04:00
|
|
|
log.Infof("docker daemon: %s %s; execdriver: %s; graphdriver: %s",
|
|
|
|
dockerversion.VERSION,
|
|
|
|
dockerversion.GITCOMMIT,
|
|
|
|
d.ExecutionDriver().Name(),
|
|
|
|
d.GraphDriver().String(),
|
|
|
|
)
|
|
|
|
|
2014-08-08 05:12:39 -04:00
|
|
|
if err := d.Install(eng); err != nil {
|
2015-03-06 15:44:31 -05:00
|
|
|
log.Error(err)
|
|
|
|
return
|
2014-08-01 13:34:06 -04:00
|
|
|
}
|
2014-08-11 11:44:31 -04:00
|
|
|
|
|
|
|
b := &builder.BuilderJob{eng, d}
|
|
|
|
b.Install()
|
|
|
|
|
2014-08-01 13:34:06 -04:00
|
|
|
// after the daemon is done setting up we can tell the api to start
|
|
|
|
// accepting connections
|
|
|
|
if err := eng.Job("acceptconnections").Run(); err != nil {
|
2015-03-06 15:44:31 -05:00
|
|
|
log.Error(err)
|
|
|
|
return
|
2014-08-01 13:34:06 -04:00
|
|
|
}
|
2015-03-06 15:44:31 -05:00
|
|
|
|
|
|
|
log.Debugf("daemon finished")
|
2014-08-01 13:34:06 -04:00
|
|
|
}()
|
|
|
|
|
|
|
|
// Serve api
|
2014-08-11 18:30:01 -04:00
|
|
|
job := eng.Job("serveapi", flHosts...)
|
2014-08-01 13:34:06 -04:00
|
|
|
job.SetenvBool("Logging", true)
|
2015-01-29 21:55:22 -05:00
|
|
|
job.SetenvBool("EnableCors", daemonCfg.EnableCors)
|
2015-02-09 02:15:07 -05:00
|
|
|
job.Setenv("CorsHeaders", daemonCfg.CorsHeaders)
|
2014-08-01 13:34:06 -04:00
|
|
|
job.Setenv("Version", dockerversion.VERSION)
|
2015-01-29 21:55:22 -05:00
|
|
|
job.Setenv("SocketGroup", daemonCfg.SocketGroup)
|
2014-08-01 13:34:06 -04:00
|
|
|
|
|
|
|
job.SetenvBool("Tls", *flTls)
|
|
|
|
job.SetenvBool("TlsVerify", *flTlsVerify)
|
|
|
|
job.Setenv("TlsCa", *flCa)
|
|
|
|
job.Setenv("TlsCert", *flCert)
|
|
|
|
job.Setenv("TlsKey", *flKey)
|
|
|
|
job.SetenvBool("BufferRequests", true)
|
2015-03-06 15:44:31 -05:00
|
|
|
err := job.Run()
|
|
|
|
|
|
|
|
// Wait for the daemon startup goroutine to finish
|
|
|
|
// This makes sure we can actually cleanly shutdown the daemon
|
|
|
|
log.Infof("waiting for daemon to initialize")
|
|
|
|
<-daemonWait
|
|
|
|
eng.Shutdown()
|
|
|
|
if err != nil {
|
|
|
|
// log errors here so the log output looks more consistent
|
|
|
|
log.Fatalf("shutting down daemon due to errors: %v", err)
|
2014-08-01 13:34:06 -04:00
|
|
|
}
|
2015-03-06 15:44:31 -05:00
|
|
|
|
2014-08-01 13:34:06 -04:00
|
|
|
}
|