exec.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. package main
  2. import (
  3. "fmt"
  4. "github.com/dotcloud/docker/pkg/libcontainer"
  5. "github.com/dotcloud/docker/pkg/libcontainer/network"
  6. "github.com/dotcloud/docker/pkg/libcontainer/utils"
  7. "github.com/dotcloud/docker/pkg/system"
  8. "github.com/dotcloud/docker/pkg/term"
  9. "io"
  10. "io/ioutil"
  11. "os"
  12. "os/exec"
  13. "syscall"
  14. )
  15. func execCommand(container *libcontainer.Container) (int, error) {
  16. master, console, err := createMasterAndConsole()
  17. if err != nil {
  18. return -1, err
  19. }
  20. command := exec.Command("nsinit", "init", console)
  21. command.SysProcAttr = &syscall.SysProcAttr{
  22. Cloneflags: uintptr(getNamespaceFlags(container.Namespaces) | syscall.CLONE_VFORK), // we need CLONE_VFORK so we can wait on the child
  23. }
  24. inPipe, err := command.StdinPipe()
  25. if err != nil {
  26. return -1, err
  27. }
  28. if err := command.Start(); err != nil {
  29. return -1, err
  30. }
  31. if err := writePidFile(command); err != nil {
  32. return -1, err
  33. }
  34. if container.Network != nil {
  35. name1, name2, err := createVethPair()
  36. if err != nil {
  37. return -1, err
  38. }
  39. if err := network.SetInterfaceMaster(name1, container.Network.Bridge); err != nil {
  40. return -1, err
  41. }
  42. if err := network.InterfaceUp(name1); err != nil {
  43. return -1, err
  44. }
  45. if err := network.SetInterfaceInNamespacePid(name2, command.Process.Pid); err != nil {
  46. return -1, err
  47. }
  48. fmt.Fprint(inPipe, name2)
  49. inPipe.Close()
  50. }
  51. go io.Copy(os.Stdout, master)
  52. go io.Copy(master, os.Stdin)
  53. ws, err := term.GetWinsize(os.Stdin.Fd())
  54. if err != nil {
  55. return -1, err
  56. }
  57. if err := term.SetWinsize(master.Fd(), ws); err != nil {
  58. return -1, err
  59. }
  60. state, err := term.SetRawTerminal(os.Stdin.Fd())
  61. if err != nil {
  62. command.Process.Kill()
  63. return -1, err
  64. }
  65. defer term.RestoreTerminal(os.Stdin.Fd(), state)
  66. if err := command.Wait(); err != nil {
  67. if _, ok := err.(*exec.ExitError); !ok {
  68. return -1, err
  69. }
  70. }
  71. return command.ProcessState.Sys().(syscall.WaitStatus).ExitStatus(), nil
  72. }
  73. func createMasterAndConsole() (*os.File, string, error) {
  74. master, err := os.OpenFile("/dev/ptmx", syscall.O_RDWR|syscall.O_NOCTTY|syscall.O_CLOEXEC, 0)
  75. if err != nil {
  76. return nil, "", err
  77. }
  78. console, err := system.Ptsname(master)
  79. if err != nil {
  80. return nil, "", err
  81. }
  82. if err := system.Unlockpt(master); err != nil {
  83. return nil, "", err
  84. }
  85. return master, console, nil
  86. }
  87. func createVethPair() (name1 string, name2 string, err error) {
  88. name1, err = utils.GenerateRandomName("dock", 4)
  89. if err != nil {
  90. return
  91. }
  92. name2, err = utils.GenerateRandomName("dock", 4)
  93. if err != nil {
  94. return
  95. }
  96. if err = network.CreateVethPair(name1, name2); err != nil {
  97. return
  98. }
  99. return
  100. }
  101. func writePidFile(command *exec.Cmd) error {
  102. return ioutil.WriteFile(".nspid", []byte(fmt.Sprint(command.Process.Pid)), 0655)
  103. }