docker.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107
  1. package main
  2. import (
  3. "flag"
  4. "fmt"
  5. "github.com/dotcloud/docker"
  6. "github.com/dotcloud/docker/utils"
  7. "io/ioutil"
  8. "log"
  9. "os"
  10. "os/signal"
  11. "strconv"
  12. "syscall"
  13. )
  14. var (
  15. GIT_COMMIT string
  16. )
  17. func main() {
  18. if utils.SelfPath() == "/sbin/init" {
  19. // Running in init mode
  20. docker.SysInit()
  21. return
  22. }
  23. // FIXME: Switch d and D ? (to be more sshd like)
  24. flDaemon := flag.Bool("d", false, "Daemon mode")
  25. flDebug := flag.Bool("D", false, "Debug mode")
  26. flAutoRestart := flag.Bool("r", false, "Restart previously running containers")
  27. bridgeName := flag.String("b", "", "Attach containers to a pre-existing network bridge")
  28. pidfile := flag.String("p", "/var/run/docker.pid", "File containing process PID")
  29. flag.Parse()
  30. if *bridgeName != "" {
  31. docker.NetworkBridgeIface = *bridgeName
  32. } else {
  33. docker.NetworkBridgeIface = docker.DefaultNetworkBridge
  34. }
  35. if *flDebug {
  36. os.Setenv("DEBUG", "1")
  37. }
  38. docker.GIT_COMMIT = GIT_COMMIT
  39. if *flDaemon {
  40. if flag.NArg() != 0 {
  41. flag.Usage()
  42. return
  43. }
  44. if err := daemon(*pidfile, *flAutoRestart); err != nil {
  45. log.Fatal(err)
  46. os.Exit(-1)
  47. }
  48. } else {
  49. if err := docker.ParseCommands(flag.Args()...); err != nil {
  50. log.Fatal(err)
  51. os.Exit(-1)
  52. }
  53. }
  54. }
  55. func createPidFile(pidfile string) error {
  56. if pidString, err := ioutil.ReadFile(pidfile); err == nil {
  57. pid, err := strconv.Atoi(string(pidString))
  58. if err == nil {
  59. if _, err := os.Stat(fmt.Sprintf("/proc/%d/", pid)); err == nil {
  60. return fmt.Errorf("pid file found, ensure docker is not running or delete %s", pidfile)
  61. }
  62. }
  63. }
  64. file, err := os.Create(pidfile)
  65. if err != nil {
  66. return err
  67. }
  68. defer file.Close()
  69. _, err = fmt.Fprintf(file, "%d", os.Getpid())
  70. return err
  71. }
  72. func removePidFile(pidfile string) {
  73. if err := os.Remove(pidfile); err != nil {
  74. log.Printf("Error removing %s: %s", pidfile, err)
  75. }
  76. }
  77. func daemon(pidfile string, autoRestart bool) error {
  78. if err := createPidFile(pidfile); err != nil {
  79. log.Fatal(err)
  80. }
  81. defer removePidFile(pidfile)
  82. c := make(chan os.Signal, 1)
  83. signal.Notify(c, os.Interrupt, os.Kill, os.Signal(syscall.SIGTERM))
  84. go func() {
  85. sig := <-c
  86. log.Printf("Received signal '%v', exiting\n", sig)
  87. removePidFile(pidfile)
  88. os.Exit(0)
  89. }()
  90. server, err := docker.NewServer(autoRestart)
  91. if err != nil {
  92. return err
  93. }
  94. return docker.ListenAndServe("0.0.0.0:4243", server, true)
  95. }