network_linux.go 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. // +build linux
  2. package libcontainer
  3. import (
  4. "fmt"
  5. "io/ioutil"
  6. "net"
  7. "path/filepath"
  8. "strconv"
  9. "strings"
  10. "github.com/docker/libcontainer/configs"
  11. "github.com/docker/libcontainer/netlink"
  12. "github.com/docker/libcontainer/utils"
  13. )
  14. var strategies = map[string]networkStrategy{
  15. "veth": &veth{},
  16. "loopback": &loopback{},
  17. }
  18. // networkStrategy represents a specific network configuration for
  19. // a container's networking stack
  20. type networkStrategy interface {
  21. create(*network, int) error
  22. initialize(*network) error
  23. detach(*configs.Network) error
  24. attach(*configs.Network) error
  25. }
  26. // getStrategy returns the specific network strategy for the
  27. // provided type.
  28. func getStrategy(tpe string) (networkStrategy, error) {
  29. s, exists := strategies[tpe]
  30. if !exists {
  31. return nil, fmt.Errorf("unknown strategy type %q", tpe)
  32. }
  33. return s, nil
  34. }
  35. // Returns the network statistics for the network interfaces represented by the NetworkRuntimeInfo.
  36. func getNetworkInterfaceStats(interfaceName string) (*NetworkInterface, error) {
  37. out := &NetworkInterface{Name: interfaceName}
  38. // This can happen if the network runtime information is missing - possible if the
  39. // container was created by an old version of libcontainer.
  40. if interfaceName == "" {
  41. return out, nil
  42. }
  43. type netStatsPair struct {
  44. // Where to write the output.
  45. Out *uint64
  46. // The network stats file to read.
  47. File string
  48. }
  49. // Ingress for host veth is from the container. Hence tx_bytes stat on the host veth is actually number of bytes received by the container.
  50. netStats := []netStatsPair{
  51. {Out: &out.RxBytes, File: "tx_bytes"},
  52. {Out: &out.RxPackets, File: "tx_packets"},
  53. {Out: &out.RxErrors, File: "tx_errors"},
  54. {Out: &out.RxDropped, File: "tx_dropped"},
  55. {Out: &out.TxBytes, File: "rx_bytes"},
  56. {Out: &out.TxPackets, File: "rx_packets"},
  57. {Out: &out.TxErrors, File: "rx_errors"},
  58. {Out: &out.TxDropped, File: "rx_dropped"},
  59. }
  60. for _, netStat := range netStats {
  61. data, err := readSysfsNetworkStats(interfaceName, netStat.File)
  62. if err != nil {
  63. return nil, err
  64. }
  65. *(netStat.Out) = data
  66. }
  67. return out, nil
  68. }
  69. // Reads the specified statistics available under /sys/class/net/<EthInterface>/statistics
  70. func readSysfsNetworkStats(ethInterface, statsFile string) (uint64, error) {
  71. data, err := ioutil.ReadFile(filepath.Join("/sys/class/net", ethInterface, "statistics", statsFile))
  72. if err != nil {
  73. return 0, err
  74. }
  75. return strconv.ParseUint(strings.TrimSpace(string(data)), 10, 64)
  76. }
  77. // loopback is a network strategy that provides a basic loopback device
  78. type loopback struct {
  79. }
  80. func (l *loopback) create(n *network, nspid int) error {
  81. return nil
  82. }
  83. func (l *loopback) initialize(config *network) error {
  84. iface, err := net.InterfaceByName("lo")
  85. if err != nil {
  86. return err
  87. }
  88. return netlink.NetworkLinkUp(iface)
  89. }
  90. func (l *loopback) attach(n *configs.Network) (err error) {
  91. return nil
  92. }
  93. func (l *loopback) detach(n *configs.Network) (err error) {
  94. return nil
  95. }
  96. // veth is a network strategy that uses a bridge and creates
  97. // a veth pair, one that is attached to the bridge on the host and the other
  98. // is placed inside the container's namespace
  99. type veth struct {
  100. }
  101. func (v *veth) detach(n *configs.Network) (err error) {
  102. bridge, err := net.InterfaceByName(n.Bridge)
  103. if err != nil {
  104. return err
  105. }
  106. host, err := net.InterfaceByName(n.HostInterfaceName)
  107. if err != nil {
  108. return err
  109. }
  110. if err := netlink.DelFromBridge(host, bridge); err != nil {
  111. return err
  112. }
  113. return nil
  114. }
  115. // attach a container network interface to an external network
  116. func (v *veth) attach(n *configs.Network) (err error) {
  117. bridge, err := net.InterfaceByName(n.Bridge)
  118. if err != nil {
  119. return err
  120. }
  121. host, err := net.InterfaceByName(n.HostInterfaceName)
  122. if err != nil {
  123. return err
  124. }
  125. if err := netlink.AddToBridge(host, bridge); err != nil {
  126. return err
  127. }
  128. if err := netlink.NetworkSetMTU(host, n.Mtu); err != nil {
  129. return err
  130. }
  131. if n.HairpinMode {
  132. if err := netlink.SetHairpinMode(host, true); err != nil {
  133. return err
  134. }
  135. }
  136. if err := netlink.NetworkLinkUp(host); err != nil {
  137. return err
  138. }
  139. return nil
  140. }
  141. func (v *veth) create(n *network, nspid int) (err error) {
  142. tmpName, err := v.generateTempPeerName()
  143. if err != nil {
  144. return err
  145. }
  146. n.TempVethPeerName = tmpName
  147. defer func() {
  148. if err != nil {
  149. netlink.NetworkLinkDel(n.HostInterfaceName)
  150. netlink.NetworkLinkDel(n.TempVethPeerName)
  151. }
  152. }()
  153. if n.Bridge == "" {
  154. return fmt.Errorf("bridge is not specified")
  155. }
  156. if err := netlink.NetworkCreateVethPair(n.HostInterfaceName, n.TempVethPeerName, n.TxQueueLen); err != nil {
  157. return err
  158. }
  159. if err := v.attach(&n.Network); err != nil {
  160. return err
  161. }
  162. child, err := net.InterfaceByName(n.TempVethPeerName)
  163. if err != nil {
  164. return err
  165. }
  166. return netlink.NetworkSetNsPid(child, nspid)
  167. }
  168. func (v *veth) generateTempPeerName() (string, error) {
  169. return utils.GenerateRandomName("veth", 7)
  170. }
  171. func (v *veth) initialize(config *network) error {
  172. peer := config.TempVethPeerName
  173. if peer == "" {
  174. return fmt.Errorf("peer is not specified")
  175. }
  176. child, err := net.InterfaceByName(peer)
  177. if err != nil {
  178. return err
  179. }
  180. if err := netlink.NetworkLinkDown(child); err != nil {
  181. return err
  182. }
  183. if err := netlink.NetworkChangeName(child, config.Name); err != nil {
  184. return err
  185. }
  186. // get the interface again after we changed the name as the index also changes.
  187. if child, err = net.InterfaceByName(config.Name); err != nil {
  188. return err
  189. }
  190. if config.MacAddress != "" {
  191. if err := netlink.NetworkSetMacAddress(child, config.MacAddress); err != nil {
  192. return err
  193. }
  194. }
  195. ip, ipNet, err := net.ParseCIDR(config.Address)
  196. if err != nil {
  197. return err
  198. }
  199. if err := netlink.NetworkLinkAddIp(child, ip, ipNet); err != nil {
  200. return err
  201. }
  202. if config.IPv6Address != "" {
  203. if ip, ipNet, err = net.ParseCIDR(config.IPv6Address); err != nil {
  204. return err
  205. }
  206. if err := netlink.NetworkLinkAddIp(child, ip, ipNet); err != nil {
  207. return err
  208. }
  209. }
  210. if err := netlink.NetworkSetMTU(child, config.Mtu); err != nil {
  211. return err
  212. }
  213. if err := netlink.NetworkLinkUp(child); err != nil {
  214. return err
  215. }
  216. if config.Gateway != "" {
  217. if err := netlink.AddDefaultGw(config.Gateway, config.Name); err != nil {
  218. return err
  219. }
  220. }
  221. if config.IPv6Gateway != "" {
  222. if err := netlink.AddDefaultGw(config.IPv6Gateway, config.Name); err != nil {
  223. return err
  224. }
  225. }
  226. return nil
  227. }