2013-01-19 16:07:19 -08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2013-03-13 00:29:40 -07:00
|
|
|
"flag"
|
2013-04-10 12:24:15 +02:00
|
|
|
"fmt"
|
2013-03-13 00:29:40 -07:00
|
|
|
"github.com/dotcloud/docker"
|
|
|
|
"github.com/dotcloud/docker/rcli"
|
|
|
|
"github.com/dotcloud/docker/term"
|
|
|
|
"io"
|
2013-04-22 15:57:31 -07:00
|
|
|
"io/ioutil"
|
2013-01-19 16:07:19 -08:00
|
|
|
"log"
|
|
|
|
"os"
|
2013-04-10 12:24:15 +02:00
|
|
|
"os/signal"
|
2013-04-22 15:57:31 -07:00
|
|
|
"strconv"
|
2013-04-10 12:24:15 +02:00
|
|
|
"syscall"
|
2013-01-23 23:14:46 -08:00
|
|
|
)
|
|
|
|
|
2013-04-11 12:58:23 -07:00
|
|
|
var (
|
2013-04-18 21:08:20 -07:00
|
|
|
GIT_COMMIT string
|
2013-04-11 12:58:23 -07:00
|
|
|
)
|
2013-04-01 13:12:56 -05:00
|
|
|
|
2013-01-19 16:07:19 -08:00
|
|
|
func main() {
|
2013-03-13 00:29:40 -07:00
|
|
|
if docker.SelfPath() == "/sbin/init" {
|
|
|
|
// Running in init mode
|
|
|
|
docker.SysInit()
|
|
|
|
return
|
2013-01-25 11:26:18 -08:00
|
|
|
}
|
2013-03-22 11:44:12 -07: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 19:01:23 -07:00
|
|
|
flAutoRestart := flag.Bool("r", false, "Restart previously running containers")
|
2013-04-05 14:16:04 -07:00
|
|
|
bridgeName := flag.String("b", "", "Attach containers to a pre-existing network bridge")
|
2013-04-10 12:24:15 +02:00
|
|
|
pidfile := flag.String("p", "/var/run/docker.pid", "File containing process PID")
|
2013-03-13 00:29:40 -07:00
|
|
|
flag.Parse()
|
2013-04-03 14:53:09 -07:00
|
|
|
if *bridgeName != "" {
|
|
|
|
docker.NetworkBridgeIface = *bridgeName
|
|
|
|
} else {
|
2013-04-04 05:33:28 -07:00
|
|
|
docker.NetworkBridgeIface = docker.DefaultNetworkBridge
|
2013-04-03 14:53:09 -07:00
|
|
|
}
|
2013-04-02 10:58:16 -07:00
|
|
|
if *flDebug {
|
|
|
|
os.Setenv("DEBUG", "1")
|
|
|
|
}
|
2013-04-01 13:12:56 -05:00
|
|
|
docker.GIT_COMMIT = GIT_COMMIT
|
2013-03-28 20:12:23 -04:00
|
|
|
if *flDaemon {
|
2013-03-13 00:29:40 -07:00
|
|
|
if flag.NArg() != 0 {
|
|
|
|
flag.Usage()
|
|
|
|
return
|
|
|
|
}
|
2013-04-24 19:01:23 -07:00
|
|
|
if err := daemon(*pidfile, *flAutoRestart); err != nil {
|
2013-03-13 00:29:40 -07:00
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err := runCommand(flag.Args()); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-10 12:24:15 +02:00
|
|
|
func createPidFile(pidfile string) error {
|
2013-04-22 15:57:31 -07: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 12:24:15 +02: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-04-24 19:01:23 -07:00
|
|
|
func daemon(pidfile string, autoRestart bool) error {
|
2013-04-12 07:37:24 -07:00
|
|
|
if err := createPidFile(pidfile); err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
defer removePidFile(pidfile)
|
2013-04-10 12:24:15 +02:00
|
|
|
|
2013-04-12 07:37:24 -07: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-04-10 12:24:15 +02:00
|
|
|
|
2013-04-24 19:01:23 -07:00
|
|
|
service, err := docker.NewServer(autoRestart)
|
2013-03-13 00:29:40 -07:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return rcli.ListenAndServe("tcp", "127.0.0.1:4242", service)
|
|
|
|
}
|
|
|
|
|
2013-04-03 12:25:19 -07:00
|
|
|
func runCommand(args []string) error {
|
2013-03-13 00:29:40 -07:00
|
|
|
// FIXME: we want to use unix sockets here, but net.UnixConn doesn't expose
|
|
|
|
// CloseWrite(), which we need to cleanly signal that stdin is closed without
|
|
|
|
// closing the connection.
|
|
|
|
// See http://code.google.com/p/go/issues/detail?id=3345
|
|
|
|
if conn, err := rcli.Call("tcp", "127.0.0.1:4242", args...); err == nil {
|
2013-04-03 12:25:19 -07:00
|
|
|
options := conn.GetOptions()
|
|
|
|
if options.RawTerminal &&
|
|
|
|
term.IsTerminal(int(os.Stdin.Fd())) &&
|
|
|
|
os.Getenv("NORAW") == "" {
|
2013-04-04 18:36:34 -07:00
|
|
|
if oldState, err := rcli.SetRawTerminal(); err != nil {
|
2013-04-03 12:25:19 -07:00
|
|
|
return err
|
|
|
|
} else {
|
2013-04-04 18:36:34 -07:00
|
|
|
defer rcli.RestoreTerminal(oldState)
|
2013-04-03 12:25:19 -07:00
|
|
|
}
|
|
|
|
}
|
2013-03-28 20:12:23 -04:00
|
|
|
receiveStdout := docker.Go(func() error {
|
2013-03-13 00:29:40 -07:00
|
|
|
_, err := io.Copy(os.Stdout, conn)
|
|
|
|
return err
|
|
|
|
})
|
2013-03-28 20:12:23 -04:00
|
|
|
sendStdin := docker.Go(func() error {
|
2013-03-13 00:29:40 -07:00
|
|
|
_, err := io.Copy(conn, os.Stdin)
|
|
|
|
if err := conn.CloseWrite(); err != nil {
|
|
|
|
log.Printf("Couldn't send EOF: " + err.Error())
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
})
|
2013-03-28 20:12:23 -04:00
|
|
|
if err := <-receiveStdout; err != nil {
|
2013-03-13 00:29:40 -07:00
|
|
|
return err
|
|
|
|
}
|
2013-03-27 23:56:36 -07:00
|
|
|
if !term.IsTerminal(int(os.Stdin.Fd())) {
|
2013-03-28 20:12:23 -04:00
|
|
|
if err := <-sendStdin; err != nil {
|
2013-03-13 00:29:40 -07:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2013-04-15 11:44:11 -07:00
|
|
|
return fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
|
2013-03-13 00:29:40 -07:00
|
|
|
}
|
|
|
|
return nil
|
2013-01-19 16:07:19 -08:00
|
|
|
}
|