network_routes.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. package network
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "net/http"
  6. "strconv"
  7. "strings"
  8. "golang.org/x/net/context"
  9. "github.com/docker/docker/api/errors"
  10. "github.com/docker/docker/api/server/httputils"
  11. "github.com/docker/docker/api/types"
  12. "github.com/docker/docker/api/types/filters"
  13. "github.com/docker/docker/api/types/network"
  14. "github.com/docker/docker/api/types/versions"
  15. "github.com/docker/libnetwork"
  16. "github.com/docker/libnetwork/networkdb"
  17. )
  18. var (
  19. // acceptedNetworkFilters is a list of acceptable filters
  20. acceptedNetworkFilters = map[string]bool{
  21. "driver": true,
  22. "type": true,
  23. "name": true,
  24. "id": true,
  25. "label": true,
  26. "scope": true,
  27. }
  28. )
  29. func (n *networkRouter) getNetworksList(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  30. if err := httputils.ParseForm(r); err != nil {
  31. return err
  32. }
  33. filter := r.Form.Get("filters")
  34. netFilters, err := filters.FromParam(filter)
  35. if err != nil {
  36. return err
  37. }
  38. if err := netFilters.Validate(acceptedNetworkFilters); err != nil {
  39. return err
  40. }
  41. list := []types.NetworkResource{}
  42. if nr, err := n.cluster.GetNetworks(); err == nil {
  43. list = append(list, nr...)
  44. }
  45. // Combine the network list returned by Docker daemon if it is not already
  46. // returned by the cluster manager
  47. SKIP:
  48. for _, nw := range n.backend.GetNetworks() {
  49. for _, nl := range list {
  50. if nl.ID == nw.ID() {
  51. continue SKIP
  52. }
  53. }
  54. var nr *types.NetworkResource
  55. // Versions < 1.28 fetches all the containers attached to a network
  56. // in a network list api call. It is a heavy weight operation when
  57. // run across all the networks. Starting API version 1.28, this detailed
  58. // info is available for network specific GET API (equivalent to inspect)
  59. if versions.LessThan(httputils.VersionFromContext(ctx), "1.28") {
  60. nr = n.buildDetailedNetworkResources(nw, false)
  61. } else {
  62. nr = n.buildNetworkResource(nw)
  63. }
  64. list = append(list, *nr)
  65. }
  66. list, err = filterNetworks(list, netFilters)
  67. if err != nil {
  68. return err
  69. }
  70. return httputils.WriteJSON(w, http.StatusOK, list)
  71. }
  72. func (n *networkRouter) getNetwork(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  73. if err := httputils.ParseForm(r); err != nil {
  74. return err
  75. }
  76. term := vars["id"]
  77. var (
  78. verbose bool
  79. err error
  80. )
  81. if v := r.URL.Query().Get("verbose"); v != "" {
  82. if verbose, err = strconv.ParseBool(v); err != nil {
  83. err = fmt.Errorf("invalid value for verbose: %s", v)
  84. return errors.NewBadRequestError(err)
  85. }
  86. }
  87. // In case multiple networks have duplicate names, return error.
  88. // TODO (yongtang): should we wrap with version here for backward compatibility?
  89. // First find based on full ID, return immediately once one is found.
  90. // If a network appears both in swarm and local, assume it is in local first
  91. // For full name and partial ID, save the result first, and process later
  92. // in case multiple records was found based on the same term
  93. listByFullName := map[string]types.NetworkResource{}
  94. listByPartialID := map[string]types.NetworkResource{}
  95. nw := n.backend.GetNetworks()
  96. for _, network := range nw {
  97. if network.ID() == term {
  98. return httputils.WriteJSON(w, http.StatusOK, *n.buildDetailedNetworkResources(network, verbose))
  99. }
  100. if network.Name() == term {
  101. // No need to check the ID collision here as we are still in
  102. // local scope and the network ID is unique in this scope.
  103. listByFullName[network.ID()] = *n.buildDetailedNetworkResources(network, verbose)
  104. }
  105. if strings.HasPrefix(network.ID(), term) {
  106. // No need to check the ID collision here as we are still in
  107. // local scope and the network ID is unique in this scope.
  108. listByPartialID[network.ID()] = *n.buildDetailedNetworkResources(network, verbose)
  109. }
  110. }
  111. nr, _ := n.cluster.GetNetworks()
  112. for _, network := range nr {
  113. if network.ID == term {
  114. return httputils.WriteJSON(w, http.StatusOK, network)
  115. }
  116. if network.Name == term {
  117. // Check the ID collision as we are in swarm scope here, and
  118. // the map (of the listByFullName) may have already had a
  119. // network with the same ID (from local scope previously)
  120. if _, ok := listByFullName[network.ID]; !ok {
  121. listByFullName[network.ID] = network
  122. }
  123. }
  124. if strings.HasPrefix(network.ID, term) {
  125. // Check the ID collision as we are in swarm scope here, and
  126. // the map (of the listByPartialID) may have already had a
  127. // network with the same ID (from local scope previously)
  128. if _, ok := listByPartialID[network.ID]; !ok {
  129. listByPartialID[network.ID] = network
  130. }
  131. }
  132. }
  133. // Find based on full name, returns true only if no duplicates
  134. if len(listByFullName) == 1 {
  135. for _, v := range listByFullName {
  136. return httputils.WriteJSON(w, http.StatusOK, v)
  137. }
  138. }
  139. if len(listByFullName) > 1 {
  140. return fmt.Errorf("network %s is ambiguous (%d matches found based on name)", term, len(listByFullName))
  141. }
  142. // Find based on partial ID, returns true only if no duplicates
  143. if len(listByPartialID) == 1 {
  144. for _, v := range listByPartialID {
  145. return httputils.WriteJSON(w, http.StatusOK, v)
  146. }
  147. }
  148. if len(listByPartialID) > 1 {
  149. return fmt.Errorf("network %s is ambiguous (%d matches found based on ID prefix)", term, len(listByPartialID))
  150. }
  151. return libnetwork.ErrNoSuchNetwork(term)
  152. }
  153. func (n *networkRouter) postNetworkCreate(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  154. var create types.NetworkCreateRequest
  155. if err := httputils.ParseForm(r); err != nil {
  156. return err
  157. }
  158. if err := httputils.CheckForJSON(r); err != nil {
  159. return err
  160. }
  161. if err := json.NewDecoder(r.Body).Decode(&create); err != nil {
  162. return err
  163. }
  164. if nws, err := n.cluster.GetNetworksByName(create.Name); err == nil && len(nws) > 0 {
  165. return libnetwork.NetworkNameError(create.Name)
  166. }
  167. nw, err := n.backend.CreateNetwork(create)
  168. if err != nil {
  169. var warning string
  170. if _, ok := err.(libnetwork.NetworkNameError); ok {
  171. // check if user defined CheckDuplicate, if set true, return err
  172. // otherwise prepare a warning message
  173. if create.CheckDuplicate {
  174. return libnetwork.NetworkNameError(create.Name)
  175. }
  176. warning = libnetwork.NetworkNameError(create.Name).Error()
  177. }
  178. if _, ok := err.(libnetwork.ManagerRedirectError); !ok {
  179. return err
  180. }
  181. id, err := n.cluster.CreateNetwork(create)
  182. if err != nil {
  183. return err
  184. }
  185. nw = &types.NetworkCreateResponse{
  186. ID: id,
  187. Warning: warning,
  188. }
  189. }
  190. return httputils.WriteJSON(w, http.StatusCreated, nw)
  191. }
  192. func (n *networkRouter) postNetworkConnect(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  193. var connect types.NetworkConnect
  194. if err := httputils.ParseForm(r); err != nil {
  195. return err
  196. }
  197. if err := httputils.CheckForJSON(r); err != nil {
  198. return err
  199. }
  200. if err := json.NewDecoder(r.Body).Decode(&connect); err != nil {
  201. return err
  202. }
  203. return n.backend.ConnectContainerToNetwork(connect.Container, vars["id"], connect.EndpointConfig)
  204. }
  205. func (n *networkRouter) postNetworkDisconnect(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  206. var disconnect types.NetworkDisconnect
  207. if err := httputils.ParseForm(r); err != nil {
  208. return err
  209. }
  210. if err := httputils.CheckForJSON(r); err != nil {
  211. return err
  212. }
  213. if err := json.NewDecoder(r.Body).Decode(&disconnect); err != nil {
  214. return err
  215. }
  216. return n.backend.DisconnectContainerFromNetwork(disconnect.Container, vars["id"], disconnect.Force)
  217. }
  218. func (n *networkRouter) deleteNetwork(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  219. if err := httputils.ParseForm(r); err != nil {
  220. return err
  221. }
  222. if _, err := n.cluster.GetNetwork(vars["id"]); err == nil {
  223. if err = n.cluster.RemoveNetwork(vars["id"]); err != nil {
  224. return err
  225. }
  226. w.WriteHeader(http.StatusNoContent)
  227. return nil
  228. }
  229. if err := n.backend.DeleteNetwork(vars["id"]); err != nil {
  230. return err
  231. }
  232. w.WriteHeader(http.StatusNoContent)
  233. return nil
  234. }
  235. func (n *networkRouter) buildNetworkResource(nw libnetwork.Network) *types.NetworkResource {
  236. r := &types.NetworkResource{}
  237. if nw == nil {
  238. return r
  239. }
  240. info := nw.Info()
  241. r.Name = nw.Name()
  242. r.ID = nw.ID()
  243. r.Created = info.Created()
  244. r.Scope = info.Scope()
  245. if n.cluster.IsManager() {
  246. if _, err := n.cluster.GetNetwork(nw.ID()); err == nil {
  247. r.Scope = "swarm"
  248. }
  249. } else if info.Dynamic() {
  250. r.Scope = "swarm"
  251. }
  252. r.Driver = nw.Type()
  253. r.EnableIPv6 = info.IPv6Enabled()
  254. r.Internal = info.Internal()
  255. r.Attachable = info.Attachable()
  256. r.Options = info.DriverOptions()
  257. r.Containers = make(map[string]types.EndpointResource)
  258. buildIpamResources(r, info)
  259. r.Labels = info.Labels()
  260. peers := info.Peers()
  261. if len(peers) != 0 {
  262. r.Peers = buildPeerInfoResources(peers)
  263. }
  264. return r
  265. }
  266. func (n *networkRouter) buildDetailedNetworkResources(nw libnetwork.Network, verbose bool) *types.NetworkResource {
  267. if nw == nil {
  268. return &types.NetworkResource{}
  269. }
  270. r := n.buildNetworkResource(nw)
  271. epl := nw.Endpoints()
  272. for _, e := range epl {
  273. ei := e.Info()
  274. if ei == nil {
  275. continue
  276. }
  277. sb := ei.Sandbox()
  278. tmpID := e.ID()
  279. key := "ep-" + tmpID
  280. if sb != nil {
  281. key = sb.ContainerID()
  282. }
  283. r.Containers[key] = buildEndpointResource(tmpID, e.Name(), ei)
  284. }
  285. if !verbose {
  286. return r
  287. }
  288. services := nw.Info().Services()
  289. r.Services = make(map[string]network.ServiceInfo)
  290. for name, service := range services {
  291. tasks := []network.Task{}
  292. for _, t := range service.Tasks {
  293. tasks = append(tasks, network.Task{
  294. Name: t.Name,
  295. EndpointID: t.EndpointID,
  296. EndpointIP: t.EndpointIP,
  297. Info: t.Info,
  298. })
  299. }
  300. r.Services[name] = network.ServiceInfo{
  301. VIP: service.VIP,
  302. Ports: service.Ports,
  303. Tasks: tasks,
  304. LocalLBIndex: service.LocalLBIndex,
  305. }
  306. }
  307. return r
  308. }
  309. func buildPeerInfoResources(peers []networkdb.PeerInfo) []network.PeerInfo {
  310. peerInfo := make([]network.PeerInfo, 0, len(peers))
  311. for _, peer := range peers {
  312. peerInfo = append(peerInfo, network.PeerInfo{
  313. Name: peer.Name,
  314. IP: peer.IP,
  315. })
  316. }
  317. return peerInfo
  318. }
  319. func buildIpamResources(r *types.NetworkResource, nwInfo libnetwork.NetworkInfo) {
  320. id, opts, ipv4conf, ipv6conf := nwInfo.IpamConfig()
  321. ipv4Info, ipv6Info := nwInfo.IpamInfo()
  322. r.IPAM.Driver = id
  323. r.IPAM.Options = opts
  324. r.IPAM.Config = []network.IPAMConfig{}
  325. for _, ip4 := range ipv4conf {
  326. if ip4.PreferredPool == "" {
  327. continue
  328. }
  329. iData := network.IPAMConfig{}
  330. iData.Subnet = ip4.PreferredPool
  331. iData.IPRange = ip4.SubPool
  332. iData.Gateway = ip4.Gateway
  333. iData.AuxAddress = ip4.AuxAddresses
  334. r.IPAM.Config = append(r.IPAM.Config, iData)
  335. }
  336. if len(r.IPAM.Config) == 0 {
  337. for _, ip4Info := range ipv4Info {
  338. iData := network.IPAMConfig{}
  339. iData.Subnet = ip4Info.IPAMData.Pool.String()
  340. iData.Gateway = ip4Info.IPAMData.Gateway.IP.String()
  341. r.IPAM.Config = append(r.IPAM.Config, iData)
  342. }
  343. }
  344. hasIpv6Conf := false
  345. for _, ip6 := range ipv6conf {
  346. if ip6.PreferredPool == "" {
  347. continue
  348. }
  349. hasIpv6Conf = true
  350. iData := network.IPAMConfig{}
  351. iData.Subnet = ip6.PreferredPool
  352. iData.IPRange = ip6.SubPool
  353. iData.Gateway = ip6.Gateway
  354. iData.AuxAddress = ip6.AuxAddresses
  355. r.IPAM.Config = append(r.IPAM.Config, iData)
  356. }
  357. if !hasIpv6Conf {
  358. for _, ip6Info := range ipv6Info {
  359. iData := network.IPAMConfig{}
  360. iData.Subnet = ip6Info.IPAMData.Pool.String()
  361. iData.Gateway = ip6Info.IPAMData.Gateway.String()
  362. r.IPAM.Config = append(r.IPAM.Config, iData)
  363. }
  364. }
  365. }
  366. func buildEndpointResource(id string, name string, info libnetwork.EndpointInfo) types.EndpointResource {
  367. er := types.EndpointResource{}
  368. er.EndpointID = id
  369. er.Name = name
  370. ei := info
  371. if ei == nil {
  372. return er
  373. }
  374. if iface := ei.Iface(); iface != nil {
  375. if mac := iface.MacAddress(); mac != nil {
  376. er.MacAddress = mac.String()
  377. }
  378. if ip := iface.Address(); ip != nil && len(ip.IP) > 0 {
  379. er.IPv4Address = ip.String()
  380. }
  381. if ipv6 := iface.AddressIPv6(); ipv6 != nil && len(ipv6.IP) > 0 {
  382. er.IPv6Address = ipv6.String()
  383. }
  384. }
  385. return er
  386. }
  387. func (n *networkRouter) postNetworksPrune(ctx context.Context, w http.ResponseWriter, r *http.Request, vars map[string]string) error {
  388. if err := httputils.ParseForm(r); err != nil {
  389. return err
  390. }
  391. pruneFilters, err := filters.FromParam(r.Form.Get("filters"))
  392. if err != nil {
  393. return err
  394. }
  395. pruneReport, err := n.backend.NetworksPrune(pruneFilters)
  396. if err != nil {
  397. return err
  398. }
  399. return httputils.WriteJSON(w, http.StatusOK, pruneReport)
  400. }