2013-01-19 19:07:19 -05:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2013-03-13 03:29:40 -04:00
|
|
|
"flag"
|
2013-04-10 06:24:15 -04:00
|
|
|
"fmt"
|
2013-03-13 03:29:40 -04:00
|
|
|
"github.com/dotcloud/docker"
|
2013-05-14 18:37:35 -04:00
|
|
|
"github.com/dotcloud/docker/utils"
|
2013-04-22 18:57:31 -04:00
|
|
|
"io/ioutil"
|
2013-01-19 19:07:19 -05:00
|
|
|
"log"
|
|
|
|
"os"
|
2013-04-10 06:24:15 -04:00
|
|
|
"os/signal"
|
2013-04-22 18:57:31 -04:00
|
|
|
"strconv"
|
2013-05-23 12:09:28 -04:00
|
|
|
"strings"
|
2013-04-10 06:24:15 -04:00
|
|
|
"syscall"
|
2013-01-24 02:14:46 -05:00
|
|
|
)
|
|
|
|
|
2013-04-11 15:58:23 -04:00
|
|
|
var (
|
2013-06-04 14:00:22 -04:00
|
|
|
GITCOMMIT string
|
2013-04-11 15:58:23 -04:00
|
|
|
)
|
2013-04-01 14:12:56 -04:00
|
|
|
|
2013-01-19 19:07:19 -05:00
|
|
|
func main() {
|
2013-05-14 18:37:35 -04:00
|
|
|
if utils.SelfPath() == "/sbin/init" {
|
2013-03-13 03:29:40 -04:00
|
|
|
// Running in init mode
|
|
|
|
docker.SysInit()
|
|
|
|
return
|
2013-01-25 14:26:18 -05:00
|
|
|
}
|
2013-03-22 14:44:12 -04:00
|
|
|
// FIXME: Switch d and D ? (to be more sshd like)
|
2013-03-28 20:12:23 -04:00
|
|
|
flDaemon := flag.Bool("d", false, "Daemon mode")
|
|
|
|
flDebug := flag.Bool("D", false, "Debug mode")
|
2013-04-24 22:01:23 -04:00
|
|
|
flAutoRestart := flag.Bool("r", false, "Restart previously running containers")
|
2013-04-05 17:16:04 -04:00
|
|
|
bridgeName := flag.String("b", "", "Attach containers to a pre-existing network bridge")
|
2013-04-10 06:24:15 -04:00
|
|
|
pidfile := flag.String("p", "/var/run/docker.pid", "File containing process PID")
|
2013-06-03 21:39:00 -04:00
|
|
|
flEnableCors := flag.Bool("api-enable-cors", false, "Enable CORS requests in the remote api.")
|
2013-06-06 14:01:29 -04:00
|
|
|
flDns := flag.String("dns", "", "Set custom dns servers")
|
2013-06-19 08:31:54 -04:00
|
|
|
flHosts := docker.ListOpts{fmt.Sprintf("tcp://%s:%d", docker.DEFAULTHTTPHOST, docker.DEFAULTHTTPPORT)}
|
|
|
|
flag.Var(&flHosts, "H", "tcp://host:port to bind/connect to or unix://path/to/socket to use")
|
2013-03-13 03:29:40 -04:00
|
|
|
flag.Parse()
|
2013-06-19 08:31:54 -04:00
|
|
|
if len(flHosts) > 1 {
|
|
|
|
flHosts = flHosts[1:len(flHosts)] //trick to display a nice defaul value in the usage
|
|
|
|
}
|
|
|
|
for i, flHost := range flHosts {
|
|
|
|
flHosts[i] = utils.ParseHost(docker.DEFAULTHTTPHOST, docker.DEFAULTHTTPPORT, flHost)
|
|
|
|
}
|
|
|
|
|
2013-04-03 17:53:09 -04:00
|
|
|
if *bridgeName != "" {
|
|
|
|
docker.NetworkBridgeIface = *bridgeName
|
|
|
|
} else {
|
2013-04-04 08:33:28 -04:00
|
|
|
docker.NetworkBridgeIface = docker.DefaultNetworkBridge
|
2013-04-03 17:53:09 -04:00
|
|
|
}
|
2013-04-02 13:58:16 -04:00
|
|
|
if *flDebug {
|
|
|
|
os.Setenv("DEBUG", "1")
|
|
|
|
}
|
2013-06-04 14:00:22 -04:00
|
|
|
docker.GITCOMMIT = GITCOMMIT
|
2013-03-28 20:12:23 -04:00
|
|
|
if *flDaemon {
|
2013-03-13 03:29:40 -04:00
|
|
|
if flag.NArg() != 0 {
|
|
|
|
flag.Usage()
|
|
|
|
return
|
|
|
|
}
|
2013-06-19 08:31:54 -04:00
|
|
|
if err := daemon(*pidfile, flHosts, *flAutoRestart, *flEnableCors, *flDns); err != nil {
|
2013-03-13 03:29:40 -04:00
|
|
|
log.Fatal(err)
|
2013-05-06 05:31:22 -04:00
|
|
|
os.Exit(-1)
|
2013-03-13 03:29:40 -04:00
|
|
|
}
|
|
|
|
} else {
|
2013-06-19 08:31:54 -04:00
|
|
|
if len(flHosts) > 1 {
|
|
|
|
log.Fatal("Please specify only one -H")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
protoAddrParts := strings.SplitN(flHosts[0], "://", 2)
|
2013-06-18 14:59:56 -04:00
|
|
|
if err := docker.ParseCommands(protoAddrParts[0], protoAddrParts[1], flag.Args()...); err != nil {
|
2013-03-13 03:29:40 -04:00
|
|
|
log.Fatal(err)
|
2013-05-06 05:31:22 -04:00
|
|
|
os.Exit(-1)
|
2013-03-13 03:29:40 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-10 06:24:15 -04:00
|
|
|
func createPidFile(pidfile string) error {
|
2013-04-22 18:57:31 -04:00
|
|
|
if pidString, err := ioutil.ReadFile(pidfile); err == nil {
|
|
|
|
pid, err := strconv.Atoi(string(pidString))
|
|
|
|
if err == nil {
|
|
|
|
if _, err := os.Stat(fmt.Sprintf("/proc/%d/", pid)); err == nil {
|
|
|
|
return fmt.Errorf("pid file found, ensure docker is not running or delete %s", pidfile)
|
|
|
|
}
|
|
|
|
}
|
2013-04-10 06:24:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
file, err := os.Create(pidfile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
_, err = fmt.Fprintf(file, "%d", os.Getpid())
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func removePidFile(pidfile string) {
|
|
|
|
if err := os.Remove(pidfile); err != nil {
|
|
|
|
log.Printf("Error removing %s: %s", pidfile, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-18 14:59:56 -04:00
|
|
|
func daemon(pidfile string, protoAddrs []string, autoRestart, enableCors bool, flDns string) error {
|
2013-04-12 10:37:24 -04:00
|
|
|
if err := createPidFile(pidfile); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
defer removePidFile(pidfile)
|
2013-04-10 06:24:15 -04:00
|
|
|
|
2013-04-12 10:37:24 -04:00
|
|
|
c := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(c, os.Interrupt, os.Kill, os.Signal(syscall.SIGTERM))
|
|
|
|
go func() {
|
|
|
|
sig := <-c
|
|
|
|
log.Printf("Received signal '%v', exiting\n", sig)
|
|
|
|
removePidFile(pidfile)
|
|
|
|
os.Exit(0)
|
|
|
|
}()
|
2013-06-06 14:01:29 -04:00
|
|
|
var dns []string
|
|
|
|
if flDns != "" {
|
|
|
|
dns = []string{flDns}
|
|
|
|
}
|
2013-06-14 17:39:05 -04:00
|
|
|
server, err := docker.NewServer(autoRestart, enableCors, dns)
|
2013-04-17 21:13:43 -04:00
|
|
|
if err != nil {
|
2013-04-10 22:48:21 -04:00
|
|
|
return err
|
|
|
|
}
|
2013-06-18 14:59:56 -04:00
|
|
|
chErrors := make(chan error, len(protoAddrs))
|
|
|
|
for _, protoAddr := range protoAddrs {
|
|
|
|
protoAddrParts := strings.SplitN(protoAddr, "://", 2)
|
|
|
|
if protoAddrParts[0] == "unix" {
|
2013-06-20 17:19:09 -04:00
|
|
|
syscall.Unlink(protoAddrParts[1])
|
2013-06-18 14:59:56 -04:00
|
|
|
} else if protoAddrParts[0] == "tcp" {
|
|
|
|
if !strings.HasPrefix(protoAddrParts[1], "127.0.0.1") {
|
|
|
|
log.Println("/!\\ DON'T BIND ON ANOTHER IP ADDRESS THAN 127.0.0.1 IF YOU DON'T KNOW WHAT YOU'RE DOING /!\\")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Fatal("Invalid protocol format.")
|
|
|
|
os.Exit(-1)
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
chErrors <- docker.ListenAndServe(protoAddrParts[0], protoAddrParts[1], server, true)
|
|
|
|
}()
|
|
|
|
}
|
2013-06-20 17:19:09 -04:00
|
|
|
for i := 0; i < len(protoAddrs); i += 1 {
|
2013-06-18 14:59:56 -04:00
|
|
|
err := <-chErrors
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2013-01-19 19:07:19 -05:00
|
|
|
}
|