docker.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. package main
  2. import (
  3. "flag"
  4. "github.com/dotcloud/docker"
  5. "github.com/dotcloud/docker/rcli"
  6. "github.com/dotcloud/docker/term"
  7. "io"
  8. "log"
  9. "os"
  10. "os/signal"
  11. )
  12. var GIT_COMMIT string
  13. func main() {
  14. if docker.SelfPath() == "/sbin/init" {
  15. // Running in init mode
  16. docker.SysInit()
  17. return
  18. }
  19. // FIXME: Switch d and D ? (to be more sshd like)
  20. flDaemon := flag.Bool("d", false, "Daemon mode")
  21. flDebug := flag.Bool("D", false, "Debug mode")
  22. bridgeName := flag.String("br", "", "")
  23. flag.Parse()
  24. if *bridgeName != "" {
  25. docker.NetworkBridgeIface = *bridgeName
  26. } else {
  27. docker.NetworkBridgeIface = "lxcbr0"
  28. }
  29. if *flDebug {
  30. os.Setenv("DEBUG", "1")
  31. }
  32. docker.GIT_COMMIT = GIT_COMMIT
  33. if *flDaemon {
  34. if flag.NArg() != 0 {
  35. flag.Usage()
  36. return
  37. }
  38. if err := daemon(); err != nil {
  39. log.Fatal(err)
  40. }
  41. } else {
  42. if err := runCommand(flag.Args()); err != nil {
  43. log.Fatal(err)
  44. }
  45. }
  46. }
  47. func daemon() error {
  48. service, err := docker.NewServer()
  49. if err != nil {
  50. return err
  51. }
  52. return rcli.ListenAndServe("tcp", "127.0.0.1:4242", service)
  53. }
  54. func runCommand(args []string) error {
  55. var oldState *term.State
  56. var err error
  57. if term.IsTerminal(int(os.Stdin.Fd())) && os.Getenv("NORAW") == "" {
  58. oldState, err = term.MakeRaw(int(os.Stdin.Fd()))
  59. if err != nil {
  60. return err
  61. }
  62. defer term.Restore(int(os.Stdin.Fd()), oldState)
  63. c := make(chan os.Signal, 1)
  64. signal.Notify(c, os.Interrupt)
  65. go func() {
  66. for _ = range c {
  67. term.Restore(int(os.Stdin.Fd()), oldState)
  68. log.Printf("\nSIGINT received\n")
  69. os.Exit(0)
  70. }
  71. }()
  72. }
  73. // FIXME: we want to use unix sockets here, but net.UnixConn doesn't expose
  74. // CloseWrite(), which we need to cleanly signal that stdin is closed without
  75. // closing the connection.
  76. // See http://code.google.com/p/go/issues/detail?id=3345
  77. if conn, err := rcli.Call("tcp", "127.0.0.1:4242", args...); err == nil {
  78. receiveStdout := docker.Go(func() error {
  79. _, err := io.Copy(os.Stdout, conn)
  80. return err
  81. })
  82. sendStdin := docker.Go(func() error {
  83. _, err := io.Copy(conn, os.Stdin)
  84. if err := conn.CloseWrite(); err != nil {
  85. log.Printf("Couldn't send EOF: " + err.Error())
  86. }
  87. return err
  88. })
  89. if err := <-receiveStdout; err != nil {
  90. return err
  91. }
  92. if !term.IsTerminal(int(os.Stdin.Fd())) {
  93. if err := <-sendStdin; err != nil {
  94. return err
  95. }
  96. }
  97. } else {
  98. service, err := docker.NewServer()
  99. if err != nil {
  100. return err
  101. }
  102. if err := rcli.LocalCall(service, os.Stdin, os.Stdout, args...); err != nil {
  103. return err
  104. }
  105. }
  106. if oldState != nil {
  107. term.Restore(int(os.Stdin.Fd()), oldState)
  108. }
  109. return nil
  110. }