server_unix.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. // +build freebsd linux
  2. package server
  3. import (
  4. "crypto/tls"
  5. "fmt"
  6. "net"
  7. "net/http"
  8. "strconv"
  9. "github.com/Sirupsen/logrus"
  10. "github.com/docker/go-connections/sockets"
  11. "github.com/docker/libnetwork/portallocator"
  12. systemdActivation "github.com/coreos/go-systemd/activation"
  13. )
  14. // newServer sets up the required HTTPServers and does protocol specific checking.
  15. // newServer does not set any muxers, you should set it later to Handler field
  16. func (s *Server) newServer(proto, addr string) ([]*HTTPServer, error) {
  17. var (
  18. err error
  19. ls []net.Listener
  20. )
  21. switch proto {
  22. case "fd":
  23. ls, err = listenFD(addr, s.cfg.TLSConfig)
  24. if err != nil {
  25. return nil, err
  26. }
  27. case "tcp":
  28. l, err := s.initTCPSocket(addr)
  29. if err != nil {
  30. return nil, err
  31. }
  32. ls = append(ls, l)
  33. case "unix":
  34. l, err := sockets.NewUnixSocket(addr, s.cfg.SocketGroup)
  35. if err != nil {
  36. return nil, fmt.Errorf("can't create unix socket %s: %v", addr, err)
  37. }
  38. ls = append(ls, l)
  39. default:
  40. return nil, fmt.Errorf("Invalid protocol format: %q", proto)
  41. }
  42. var res []*HTTPServer
  43. for _, l := range ls {
  44. res = append(res, &HTTPServer{
  45. &http.Server{
  46. Addr: addr,
  47. },
  48. l,
  49. })
  50. }
  51. return res, nil
  52. }
  53. func allocateDaemonPort(addr string) error {
  54. host, port, err := net.SplitHostPort(addr)
  55. if err != nil {
  56. return err
  57. }
  58. intPort, err := strconv.Atoi(port)
  59. if err != nil {
  60. return err
  61. }
  62. var hostIPs []net.IP
  63. if parsedIP := net.ParseIP(host); parsedIP != nil {
  64. hostIPs = append(hostIPs, parsedIP)
  65. } else if hostIPs, err = net.LookupIP(host); err != nil {
  66. return fmt.Errorf("failed to lookup %s address in host specification", host)
  67. }
  68. pa := portallocator.Get()
  69. for _, hostIP := range hostIPs {
  70. if _, err := pa.RequestPort(hostIP, "tcp", intPort); err != nil {
  71. return fmt.Errorf("failed to allocate daemon listening port %d (err: %v)", intPort, err)
  72. }
  73. }
  74. return nil
  75. }
  76. // listenFD returns the specified socket activated files as a slice of
  77. // net.Listeners or all of the activated files if "*" is given.
  78. func listenFD(addr string, tlsConfig *tls.Config) ([]net.Listener, error) {
  79. var (
  80. err error
  81. listeners []net.Listener
  82. )
  83. // socket activation
  84. if tlsConfig != nil {
  85. listeners, err = systemdActivation.TLSListeners(false, tlsConfig)
  86. } else {
  87. listeners, err = systemdActivation.Listeners(false)
  88. }
  89. if err != nil {
  90. return nil, err
  91. }
  92. if len(listeners) == 0 {
  93. return nil, fmt.Errorf("No sockets found")
  94. }
  95. // default to all fds just like unix:// and tcp://
  96. if addr == "" || addr == "*" {
  97. return listeners, nil
  98. }
  99. fdNum, err := strconv.Atoi(addr)
  100. if err != nil {
  101. return nil, fmt.Errorf("failed to parse systemd address, should be number: %v", err)
  102. }
  103. fdOffset := fdNum - 3
  104. if len(listeners) < int(fdOffset)+1 {
  105. return nil, fmt.Errorf("Too few socket activated files passed in")
  106. }
  107. if listeners[fdOffset] == nil {
  108. return nil, fmt.Errorf("failed to listen on systemd activated file at fd %d", fdOffset+3)
  109. }
  110. for i, ls := range listeners {
  111. if i == fdOffset || ls == nil {
  112. continue
  113. }
  114. if err := ls.Close(); err != nil {
  115. logrus.Errorf("Failed to close systemd activated file at fd %d: %v", fdOffset+3, err)
  116. }
  117. }
  118. return []net.Listener{listeners[fdOffset]}, nil
  119. }