123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147 |
- package main
- import (
- "flag"
- "fmt"
- "github.com/dotcloud/docker"
- "github.com/dotcloud/docker/rcli"
- "github.com/dotcloud/docker/term"
- "io"
- "log"
- "os"
- "os/signal"
- "syscall"
- )
- var (
- GIT_COMMIT string
- NO_MEMORY_LIMIT string
- )
- func main() {
- if docker.SelfPath() == "/sbin/init" {
- // Running in init mode
- docker.SysInit()
- return
- }
- // FIXME: Switch d and D ? (to be more sshd like)
- flDaemon := flag.Bool("d", false, "Daemon mode")
- flDebug := flag.Bool("D", false, "Debug mode")
- bridgeName := flag.String("b", "", "Attach containers to a pre-existing network bridge")
- pidfile := flag.String("p", "/var/run/docker.pid", "File containing process PID")
- flag.Parse()
- if *bridgeName != "" {
- docker.NetworkBridgeIface = *bridgeName
- } else {
- docker.NetworkBridgeIface = docker.DefaultNetworkBridge
- }
- if *flDebug {
- os.Setenv("DEBUG", "1")
- }
- docker.GIT_COMMIT = GIT_COMMIT
- docker.NO_MEMORY_LIMIT = NO_MEMORY_LIMIT == "1"
- if *flDaemon {
- if flag.NArg() != 0 {
- flag.Usage()
- return
- }
- if NO_MEMORY_LIMIT == "1" {
- log.Printf("WARNING: This version of docker has been compiled without memory limit support.")
- }
- if err := daemon(*pidfile); err != nil {
- log.Fatal(err)
- }
- } else {
- if err := runCommand(flag.Args()); err != nil {
- log.Fatal(err)
- }
- }
- }
- func createPidFile(pidfile string) error {
- if _, err := os.Stat(pidfile); err == nil {
- return fmt.Errorf("pid file found, ensure docker is not running or delete %s", pidfile)
- }
- 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)
- }
- }
- func daemon(pidfile string) error {
- if err := createPidFile(pidfile); err != nil {
- log.Fatal(err)
- }
- defer removePidFile(pidfile)
- 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)
- }()
- service, err := docker.NewServer()
- if err != nil {
- return err
- }
- if err := http.ListenAndServe("0.0.0.0:4243", service.restEndpoint); err != nil {
- return err
- }
- return rcli.ListenAndServe("tcp", "127.0.0.1:4242", service)
- }
- func runCommand(args []string) error {
- // 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 {
- options := conn.GetOptions()
- if options.RawTerminal &&
- term.IsTerminal(int(os.Stdin.Fd())) &&
- os.Getenv("NORAW") == "" {
- if oldState, err := rcli.SetRawTerminal(); err != nil {
- return err
- } else {
- defer rcli.RestoreTerminal(oldState)
- }
- }
- receiveStdout := docker.Go(func() error {
- _, err := io.Copy(os.Stdout, conn)
- return err
- })
- sendStdin := docker.Go(func() error {
- _, err := io.Copy(conn, os.Stdin)
- if err := conn.CloseWrite(); err != nil {
- log.Printf("Couldn't send EOF: " + err.Error())
- }
- return err
- })
- if err := <-receiveStdout; err != nil {
- return err
- }
- if !term.IsTerminal(int(os.Stdin.Fd())) {
- if err := <-sendStdin; err != nil {
- return err
- }
- }
- } else {
- return fmt.Errorf("Can't connect to docker daemon. Is 'docker -d' running on this host?")
- }
- return nil
- }
|