network.go 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. package controlapi
  2. import (
  3. "net"
  4. "github.com/docker/docker/pkg/plugingetter"
  5. "github.com/docker/libnetwork/driverapi"
  6. "github.com/docker/libnetwork/ipamapi"
  7. "github.com/docker/swarmkit/api"
  8. "github.com/docker/swarmkit/identity"
  9. "github.com/docker/swarmkit/manager/allocator"
  10. "github.com/docker/swarmkit/manager/allocator/networkallocator"
  11. "github.com/docker/swarmkit/manager/state/store"
  12. "golang.org/x/net/context"
  13. "google.golang.org/grpc/codes"
  14. "google.golang.org/grpc/status"
  15. )
  16. func validateIPAMConfiguration(ipamConf *api.IPAMConfig) error {
  17. if ipamConf == nil {
  18. return status.Errorf(codes.InvalidArgument, "ipam configuration: cannot be empty")
  19. }
  20. _, subnet, err := net.ParseCIDR(ipamConf.Subnet)
  21. if err != nil {
  22. return status.Errorf(codes.InvalidArgument, "ipam configuration: invalid subnet %s", ipamConf.Subnet)
  23. }
  24. if ipamConf.Range != "" {
  25. ip, _, err := net.ParseCIDR(ipamConf.Range)
  26. if err != nil {
  27. return status.Errorf(codes.InvalidArgument, "ipam configuration: invalid range %s", ipamConf.Range)
  28. }
  29. if !subnet.Contains(ip) {
  30. return status.Errorf(codes.InvalidArgument, "ipam configuration: subnet %s does not contain range %s", ipamConf.Subnet, ipamConf.Range)
  31. }
  32. }
  33. if ipamConf.Gateway != "" {
  34. ip := net.ParseIP(ipamConf.Gateway)
  35. if ip == nil {
  36. return status.Errorf(codes.InvalidArgument, "ipam configuration: invalid gateway %s", ipamConf.Gateway)
  37. }
  38. if !subnet.Contains(ip) {
  39. return status.Errorf(codes.InvalidArgument, "ipam configuration: subnet %s does not contain gateway %s", ipamConf.Subnet, ipamConf.Gateway)
  40. }
  41. }
  42. return nil
  43. }
  44. func validateIPAM(ipam *api.IPAMOptions, pg plugingetter.PluginGetter) error {
  45. if ipam == nil {
  46. // It is ok to not specify any IPAM configurations. We
  47. // will choose good defaults.
  48. return nil
  49. }
  50. if err := validateDriver(ipam.Driver, pg, ipamapi.PluginEndpointType); err != nil {
  51. return err
  52. }
  53. for _, ipamConf := range ipam.Configs {
  54. if err := validateIPAMConfiguration(ipamConf); err != nil {
  55. return err
  56. }
  57. }
  58. return nil
  59. }
  60. func validateNetworkSpec(spec *api.NetworkSpec, pg plugingetter.PluginGetter) error {
  61. if spec == nil {
  62. return status.Errorf(codes.InvalidArgument, errInvalidArgument.Error())
  63. }
  64. if spec.Ingress && spec.DriverConfig != nil && spec.DriverConfig.Name != "overlay" {
  65. return status.Errorf(codes.Unimplemented, "only overlay driver is currently supported for ingress network")
  66. }
  67. if spec.Attachable && spec.Ingress {
  68. return status.Errorf(codes.InvalidArgument, "ingress network cannot be attachable")
  69. }
  70. if err := validateAnnotations(spec.Annotations); err != nil {
  71. return err
  72. }
  73. if _, ok := spec.Annotations.Labels[networkallocator.PredefinedLabel]; ok {
  74. return status.Errorf(codes.PermissionDenied, "label %s is for internally created predefined networks and cannot be applied by users",
  75. networkallocator.PredefinedLabel)
  76. }
  77. if err := validateDriver(spec.DriverConfig, pg, driverapi.NetworkPluginEndpointType); err != nil {
  78. return err
  79. }
  80. return validateIPAM(spec.IPAM, pg)
  81. }
  82. // CreateNetwork creates and returns a Network based on the provided NetworkSpec.
  83. // - Returns `InvalidArgument` if the NetworkSpec is malformed.
  84. // - Returns an error if the creation fails.
  85. func (s *Server) CreateNetwork(ctx context.Context, request *api.CreateNetworkRequest) (*api.CreateNetworkResponse, error) {
  86. if err := validateNetworkSpec(request.Spec, s.pg); err != nil {
  87. return nil, err
  88. }
  89. // TODO(mrjana): Consider using `Name` as a primary key to handle
  90. // duplicate creations. See #65
  91. n := &api.Network{
  92. ID: identity.NewID(),
  93. Spec: *request.Spec,
  94. }
  95. err := s.store.Update(func(tx store.Tx) error {
  96. if request.Spec.Ingress {
  97. if n, err := allocator.GetIngressNetwork(s.store); err == nil {
  98. return status.Errorf(codes.AlreadyExists, "ingress network (%s) is already present", n.ID)
  99. } else if err != allocator.ErrNoIngress {
  100. return status.Errorf(codes.Internal, "failed ingress network presence check: %v", err)
  101. }
  102. }
  103. return store.CreateNetwork(tx, n)
  104. })
  105. if err != nil {
  106. return nil, err
  107. }
  108. return &api.CreateNetworkResponse{
  109. Network: n,
  110. }, nil
  111. }
  112. // GetNetwork returns a Network given a NetworkID.
  113. // - Returns `InvalidArgument` if NetworkID is not provided.
  114. // - Returns `NotFound` if the Network is not found.
  115. func (s *Server) GetNetwork(ctx context.Context, request *api.GetNetworkRequest) (*api.GetNetworkResponse, error) {
  116. if request.NetworkID == "" {
  117. return nil, status.Errorf(codes.InvalidArgument, errInvalidArgument.Error())
  118. }
  119. var n *api.Network
  120. s.store.View(func(tx store.ReadTx) {
  121. n = store.GetNetwork(tx, request.NetworkID)
  122. })
  123. if n == nil {
  124. return nil, status.Errorf(codes.NotFound, "network %s not found", request.NetworkID)
  125. }
  126. return &api.GetNetworkResponse{
  127. Network: n,
  128. }, nil
  129. }
  130. // RemoveNetwork removes a Network referenced by NetworkID.
  131. // - Returns `InvalidArgument` if NetworkID is not provided.
  132. // - Returns `NotFound` if the Network is not found.
  133. // - Returns an error if the deletion fails.
  134. func (s *Server) RemoveNetwork(ctx context.Context, request *api.RemoveNetworkRequest) (*api.RemoveNetworkResponse, error) {
  135. if request.NetworkID == "" {
  136. return nil, status.Errorf(codes.InvalidArgument, errInvalidArgument.Error())
  137. }
  138. var (
  139. n *api.Network
  140. rm = s.removeNetwork
  141. )
  142. s.store.View(func(tx store.ReadTx) {
  143. n = store.GetNetwork(tx, request.NetworkID)
  144. })
  145. if n == nil {
  146. return nil, status.Errorf(codes.NotFound, "network %s not found", request.NetworkID)
  147. }
  148. if allocator.IsIngressNetwork(n) {
  149. rm = s.removeIngressNetwork
  150. }
  151. if v, ok := n.Spec.Annotations.Labels[networkallocator.PredefinedLabel]; ok && v == "true" {
  152. return nil, status.Errorf(codes.FailedPrecondition, "network %s (%s) is a swarm predefined network and cannot be removed",
  153. request.NetworkID, n.Spec.Annotations.Name)
  154. }
  155. if err := rm(n.ID); err != nil {
  156. if err == store.ErrNotExist {
  157. return nil, status.Errorf(codes.NotFound, "network %s not found", request.NetworkID)
  158. }
  159. return nil, err
  160. }
  161. return &api.RemoveNetworkResponse{}, nil
  162. }
  163. func (s *Server) removeNetwork(id string) error {
  164. return s.store.Update(func(tx store.Tx) error {
  165. services, err := store.FindServices(tx, store.ByReferencedNetworkID(id))
  166. if err != nil {
  167. return status.Errorf(codes.Internal, "could not find services using network %s: %v", id, err)
  168. }
  169. if len(services) != 0 {
  170. return status.Errorf(codes.FailedPrecondition, "network %s is in use by service %s", id, services[0].ID)
  171. }
  172. tasks, err := store.FindTasks(tx, store.ByReferencedNetworkID(id))
  173. if err != nil {
  174. return status.Errorf(codes.Internal, "could not find tasks using network %s: %v", id, err)
  175. }
  176. for _, t := range tasks {
  177. if t.DesiredState <= api.TaskStateRunning && t.Status.State <= api.TaskStateRunning {
  178. return status.Errorf(codes.FailedPrecondition, "network %s is in use by task %s", id, t.ID)
  179. }
  180. }
  181. return store.DeleteNetwork(tx, id)
  182. })
  183. }
  184. func (s *Server) removeIngressNetwork(id string) error {
  185. return s.store.Update(func(tx store.Tx) error {
  186. services, err := store.FindServices(tx, store.All)
  187. if err != nil {
  188. return status.Errorf(codes.Internal, "could not find services using network %s: %v", id, err)
  189. }
  190. for _, srv := range services {
  191. if allocator.IsIngressNetworkNeeded(srv) {
  192. return status.Errorf(codes.FailedPrecondition, "ingress network cannot be removed because service %s depends on it", srv.ID)
  193. }
  194. }
  195. return store.DeleteNetwork(tx, id)
  196. })
  197. }
  198. func filterNetworks(candidates []*api.Network, filters ...func(*api.Network) bool) []*api.Network {
  199. result := []*api.Network{}
  200. for _, c := range candidates {
  201. match := true
  202. for _, f := range filters {
  203. if !f(c) {
  204. match = false
  205. break
  206. }
  207. }
  208. if match {
  209. result = append(result, c)
  210. }
  211. }
  212. return result
  213. }
  214. // ListNetworks returns a list of all networks.
  215. func (s *Server) ListNetworks(ctx context.Context, request *api.ListNetworksRequest) (*api.ListNetworksResponse, error) {
  216. var (
  217. networks []*api.Network
  218. err error
  219. )
  220. s.store.View(func(tx store.ReadTx) {
  221. switch {
  222. case request.Filters != nil && len(request.Filters.Names) > 0:
  223. networks, err = store.FindNetworks(tx, buildFilters(store.ByName, request.Filters.Names))
  224. case request.Filters != nil && len(request.Filters.NamePrefixes) > 0:
  225. networks, err = store.FindNetworks(tx, buildFilters(store.ByNamePrefix, request.Filters.NamePrefixes))
  226. case request.Filters != nil && len(request.Filters.IDPrefixes) > 0:
  227. networks, err = store.FindNetworks(tx, buildFilters(store.ByIDPrefix, request.Filters.IDPrefixes))
  228. default:
  229. networks, err = store.FindNetworks(tx, store.All)
  230. }
  231. })
  232. if err != nil {
  233. return nil, err
  234. }
  235. if request.Filters != nil {
  236. networks = filterNetworks(networks,
  237. func(e *api.Network) bool {
  238. return filterContains(e.Spec.Annotations.Name, request.Filters.Names)
  239. },
  240. func(e *api.Network) bool {
  241. return filterContainsPrefix(e.Spec.Annotations.Name, request.Filters.NamePrefixes)
  242. },
  243. func(e *api.Network) bool {
  244. return filterContainsPrefix(e.ID, request.Filters.IDPrefixes)
  245. },
  246. func(e *api.Network) bool {
  247. return filterMatchLabels(e.Spec.Annotations.Labels, request.Filters.Labels)
  248. },
  249. )
  250. }
  251. return &api.ListNetworksResponse{
  252. Networks: networks,
  253. }, nil
  254. }