driver.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. package remote
  2. import (
  3. "context"
  4. "fmt"
  5. "net"
  6. "github.com/containerd/log"
  7. "github.com/docker/docker/libnetwork/discoverapi"
  8. "github.com/docker/docker/libnetwork/driverapi"
  9. "github.com/docker/docker/libnetwork/drivers/remote/api"
  10. "github.com/docker/docker/libnetwork/scope"
  11. "github.com/docker/docker/libnetwork/types"
  12. "github.com/docker/docker/pkg/plugingetter"
  13. "github.com/docker/docker/pkg/plugins"
  14. "github.com/pkg/errors"
  15. )
  16. // remote driver must implement the discover-API.
  17. var _ discoverapi.Discover = (*driver)(nil)
  18. type driver struct {
  19. endpoint *plugins.Client
  20. networkType string
  21. }
  22. type maybeError interface {
  23. GetError() string
  24. }
  25. func newDriver(name string, client *plugins.Client) *driver {
  26. return &driver{networkType: name, endpoint: client}
  27. }
  28. // Register makes sure a remote driver is registered with r when a network
  29. // driver plugin is activated.
  30. func Register(r driverapi.Registerer, pg plugingetter.PluginGetter) error {
  31. newPluginHandler := func(name string, client *plugins.Client) {
  32. // negotiate driver capability with client
  33. d := newDriver(name, client)
  34. c, err := d.getCapabilities()
  35. if err != nil {
  36. log.G(context.TODO()).Errorf("error getting capability for %s due to %v", name, err)
  37. return
  38. }
  39. if err = r.RegisterDriver(name, d, *c); err != nil {
  40. log.G(context.TODO()).Errorf("error registering driver for %s due to %v", name, err)
  41. }
  42. }
  43. // Unit test code is unaware of a true PluginStore. So we fall back to v1 plugins.
  44. handleFunc := plugins.Handle
  45. if pg != nil {
  46. handleFunc = pg.Handle
  47. activePlugins := pg.GetAllManagedPluginsByCap(driverapi.NetworkPluginEndpointType)
  48. for _, ap := range activePlugins {
  49. client, err := getPluginClient(ap)
  50. if err != nil {
  51. return err
  52. }
  53. newPluginHandler(ap.Name(), client)
  54. }
  55. }
  56. handleFunc(driverapi.NetworkPluginEndpointType, newPluginHandler)
  57. return nil
  58. }
  59. func getPluginClient(p plugingetter.CompatPlugin) (*plugins.Client, error) {
  60. if v1, ok := p.(plugingetter.PluginWithV1Client); ok {
  61. return v1.Client(), nil
  62. }
  63. pa, ok := p.(plugingetter.PluginAddr)
  64. if !ok {
  65. return nil, errors.Errorf("unknown plugin type %T", p)
  66. }
  67. if pa.Protocol() != plugins.ProtocolSchemeHTTPV1 {
  68. return nil, errors.Errorf("unsupported plugin protocol %s", pa.Protocol())
  69. }
  70. addr := pa.Addr()
  71. client, err := plugins.NewClientWithTimeout(addr.Network()+"://"+addr.String(), nil, pa.Timeout())
  72. if err != nil {
  73. return nil, errors.Wrap(err, "error creating plugin client")
  74. }
  75. return client, nil
  76. }
  77. // Get capability from client
  78. func (d *driver) getCapabilities() (*driverapi.Capability, error) {
  79. var capResp api.GetCapabilityResponse
  80. if err := d.call("GetCapabilities", nil, &capResp); err != nil {
  81. return nil, err
  82. }
  83. c := &driverapi.Capability{}
  84. switch capResp.Scope {
  85. case scope.Global, scope.Local:
  86. c.DataScope = capResp.Scope
  87. default:
  88. return nil, fmt.Errorf("invalid capability: expecting 'local' or 'global', got %s", capResp.Scope)
  89. }
  90. switch capResp.ConnectivityScope {
  91. case scope.Global, scope.Local:
  92. c.ConnectivityScope = capResp.ConnectivityScope
  93. case "":
  94. c.ConnectivityScope = c.DataScope
  95. default:
  96. return nil, fmt.Errorf("invalid capability: expecting 'local' or 'global', got %s", capResp.Scope)
  97. }
  98. return c, nil
  99. }
  100. // Config is not implemented for remote drivers, since it is assumed
  101. // to be supplied to the remote process out-of-band (e.g., as command
  102. // line arguments).
  103. func (d *driver) Config(option map[string]interface{}) error {
  104. return &driverapi.ErrNotImplemented{}
  105. }
  106. func (d *driver) call(methodName string, arg interface{}, retVal maybeError) error {
  107. method := driverapi.NetworkPluginEndpointType + "." + methodName
  108. err := d.endpoint.Call(method, arg, retVal)
  109. if err != nil {
  110. return err
  111. }
  112. if e := retVal.GetError(); e != "" {
  113. return fmt.Errorf("remote: %s", e)
  114. }
  115. return nil
  116. }
  117. func (d *driver) NetworkAllocate(id string, options map[string]string, ipV4Data, ipV6Data []driverapi.IPAMData) (map[string]string, error) {
  118. create := &api.AllocateNetworkRequest{
  119. NetworkID: id,
  120. Options: options,
  121. IPv4Data: ipV4Data,
  122. IPv6Data: ipV6Data,
  123. }
  124. retVal := api.AllocateNetworkResponse{}
  125. err := d.call("AllocateNetwork", create, &retVal)
  126. return retVal.Options, err
  127. }
  128. func (d *driver) NetworkFree(id string) error {
  129. fr := &api.FreeNetworkRequest{NetworkID: id}
  130. return d.call("FreeNetwork", fr, &api.FreeNetworkResponse{})
  131. }
  132. func (d *driver) EventNotify(etype driverapi.EventType, nid, tableName, key string, value []byte) {
  133. }
  134. func (d *driver) DecodeTableEntry(tablename string, key string, value []byte) (string, map[string]string) {
  135. return "", nil
  136. }
  137. func (d *driver) CreateNetwork(id string, options map[string]interface{}, nInfo driverapi.NetworkInfo, ipV4Data, ipV6Data []driverapi.IPAMData) error {
  138. create := &api.CreateNetworkRequest{
  139. NetworkID: id,
  140. Options: options,
  141. IPv4Data: ipV4Data,
  142. IPv6Data: ipV6Data,
  143. }
  144. return d.call("CreateNetwork", create, &api.CreateNetworkResponse{})
  145. }
  146. func (d *driver) DeleteNetwork(nid string) error {
  147. return d.call("DeleteNetwork", &api.DeleteNetworkRequest{NetworkID: nid}, &api.DeleteNetworkResponse{})
  148. }
  149. func (d *driver) CreateEndpoint(nid, eid string, ifInfo driverapi.InterfaceInfo, epOptions map[string]interface{}) error {
  150. if ifInfo == nil {
  151. return errors.New("must not be called with nil InterfaceInfo")
  152. }
  153. reqIface := &api.EndpointInterface{}
  154. if ifInfo.Address() != nil {
  155. reqIface.Address = ifInfo.Address().String()
  156. }
  157. if ifInfo.AddressIPv6() != nil {
  158. reqIface.AddressIPv6 = ifInfo.AddressIPv6().String()
  159. }
  160. if ifInfo.MacAddress() != nil {
  161. reqIface.MacAddress = ifInfo.MacAddress().String()
  162. }
  163. create := &api.CreateEndpointRequest{
  164. NetworkID: nid,
  165. EndpointID: eid,
  166. Interface: reqIface,
  167. Options: epOptions,
  168. }
  169. var res api.CreateEndpointResponse
  170. if err := d.call("CreateEndpoint", create, &res); err != nil {
  171. return err
  172. }
  173. inIface, err := parseInterface(res)
  174. if err != nil {
  175. return err
  176. }
  177. if inIface == nil {
  178. // Remote driver did not set any field
  179. return nil
  180. }
  181. if inIface.MacAddress != nil {
  182. if err := ifInfo.SetMacAddress(inIface.MacAddress); err != nil {
  183. return errorWithRollback(fmt.Sprintf("driver modified interface MAC address: %v", err), d.DeleteEndpoint(nid, eid))
  184. }
  185. }
  186. if inIface.Address != nil {
  187. if err := ifInfo.SetIPAddress(inIface.Address); err != nil {
  188. return errorWithRollback(fmt.Sprintf("driver modified interface address: %v", err), d.DeleteEndpoint(nid, eid))
  189. }
  190. }
  191. if inIface.AddressIPv6 != nil {
  192. if err := ifInfo.SetIPAddress(inIface.AddressIPv6); err != nil {
  193. return errorWithRollback(fmt.Sprintf("driver modified interface address: %v", err), d.DeleteEndpoint(nid, eid))
  194. }
  195. }
  196. return nil
  197. }
  198. func errorWithRollback(msg string, err error) error {
  199. rollback := "rolled back"
  200. if err != nil {
  201. rollback = "failed to roll back: " + err.Error()
  202. }
  203. return fmt.Errorf("%s; %s", msg, rollback)
  204. }
  205. func (d *driver) DeleteEndpoint(nid, eid string) error {
  206. deleteRequest := &api.DeleteEndpointRequest{
  207. NetworkID: nid,
  208. EndpointID: eid,
  209. }
  210. return d.call("DeleteEndpoint", deleteRequest, &api.DeleteEndpointResponse{})
  211. }
  212. func (d *driver) EndpointOperInfo(nid, eid string) (map[string]interface{}, error) {
  213. info := &api.EndpointInfoRequest{
  214. NetworkID: nid,
  215. EndpointID: eid,
  216. }
  217. var res api.EndpointInfoResponse
  218. if err := d.call("EndpointOperInfo", info, &res); err != nil {
  219. return nil, err
  220. }
  221. return res.Value, nil
  222. }
  223. // Join method is invoked when a Sandbox is attached to an endpoint.
  224. func (d *driver) Join(nid, eid string, sboxKey string, jinfo driverapi.JoinInfo, options map[string]interface{}) error {
  225. join := &api.JoinRequest{
  226. NetworkID: nid,
  227. EndpointID: eid,
  228. SandboxKey: sboxKey,
  229. Options: options,
  230. }
  231. var (
  232. res api.JoinResponse
  233. err error
  234. )
  235. if err = d.call("Join", join, &res); err != nil {
  236. return err
  237. }
  238. ifaceName := res.InterfaceName
  239. if iface := jinfo.InterfaceName(); iface != nil && ifaceName != nil {
  240. if err := iface.SetNames(ifaceName.SrcName, ifaceName.DstPrefix); err != nil {
  241. return errorWithRollback(fmt.Sprintf("failed to set interface name: %s", err), d.Leave(nid, eid))
  242. }
  243. }
  244. var addr net.IP
  245. if res.Gateway != "" {
  246. if addr = net.ParseIP(res.Gateway); addr == nil {
  247. return fmt.Errorf(`unable to parse Gateway "%s"`, res.Gateway)
  248. }
  249. if jinfo.SetGateway(addr) != nil {
  250. return errorWithRollback(fmt.Sprintf("failed to set gateway: %v", addr), d.Leave(nid, eid))
  251. }
  252. }
  253. if res.GatewayIPv6 != "" {
  254. if addr = net.ParseIP(res.GatewayIPv6); addr == nil {
  255. return fmt.Errorf(`unable to parse GatewayIPv6 "%s"`, res.GatewayIPv6)
  256. }
  257. if jinfo.SetGatewayIPv6(addr) != nil {
  258. return errorWithRollback(fmt.Sprintf("failed to set gateway IPv6: %v", addr), d.Leave(nid, eid))
  259. }
  260. }
  261. if len(res.StaticRoutes) > 0 {
  262. routes, err := parseStaticRoutes(res)
  263. if err != nil {
  264. return err
  265. }
  266. for _, route := range routes {
  267. if jinfo.AddStaticRoute(route.Destination, route.RouteType, route.NextHop) != nil {
  268. return errorWithRollback(fmt.Sprintf("failed to set static route: %v", route), d.Leave(nid, eid))
  269. }
  270. }
  271. }
  272. if res.DisableGatewayService {
  273. jinfo.DisableGatewayService()
  274. }
  275. return nil
  276. }
  277. // Leave method is invoked when a Sandbox detaches from an endpoint.
  278. func (d *driver) Leave(nid, eid string) error {
  279. leave := &api.LeaveRequest{
  280. NetworkID: nid,
  281. EndpointID: eid,
  282. }
  283. return d.call("Leave", leave, &api.LeaveResponse{})
  284. }
  285. // ProgramExternalConnectivity is invoked to program the rules to allow external connectivity for the endpoint.
  286. func (d *driver) ProgramExternalConnectivity(nid, eid string, options map[string]interface{}) error {
  287. data := &api.ProgramExternalConnectivityRequest{
  288. NetworkID: nid,
  289. EndpointID: eid,
  290. Options: options,
  291. }
  292. err := d.call("ProgramExternalConnectivity", data, &api.ProgramExternalConnectivityResponse{})
  293. if err != nil && plugins.IsNotFound(err) {
  294. // It is not mandatory yet to support this method
  295. return nil
  296. }
  297. return err
  298. }
  299. // RevokeExternalConnectivity method is invoked to remove any external connectivity programming related to the endpoint.
  300. func (d *driver) RevokeExternalConnectivity(nid, eid string) error {
  301. data := &api.RevokeExternalConnectivityRequest{
  302. NetworkID: nid,
  303. EndpointID: eid,
  304. }
  305. err := d.call("RevokeExternalConnectivity", data, &api.RevokeExternalConnectivityResponse{})
  306. if err != nil && plugins.IsNotFound(err) {
  307. // It is not mandatory yet to support this method
  308. return nil
  309. }
  310. return err
  311. }
  312. func (d *driver) Type() string {
  313. return d.networkType
  314. }
  315. func (d *driver) IsBuiltIn() bool {
  316. return false
  317. }
  318. // DiscoverNew is a notification for a new discovery event, such as a new node joining a cluster
  319. func (d *driver) DiscoverNew(dType discoverapi.DiscoveryType, data interface{}) error {
  320. if dType != discoverapi.NodeDiscovery {
  321. return nil
  322. }
  323. notif := &api.DiscoveryNotification{
  324. DiscoveryType: dType,
  325. DiscoveryData: data,
  326. }
  327. return d.call("DiscoverNew", notif, &api.DiscoveryResponse{})
  328. }
  329. // DiscoverDelete is a notification for a discovery delete event, such as a node leaving a cluster
  330. func (d *driver) DiscoverDelete(dType discoverapi.DiscoveryType, data interface{}) error {
  331. if dType != discoverapi.NodeDiscovery {
  332. return nil
  333. }
  334. notif := &api.DiscoveryNotification{
  335. DiscoveryType: dType,
  336. DiscoveryData: data,
  337. }
  338. return d.call("DiscoverDelete", notif, &api.DiscoveryResponse{})
  339. }
  340. func parseStaticRoutes(r api.JoinResponse) ([]*types.StaticRoute, error) {
  341. routes := make([]*types.StaticRoute, len(r.StaticRoutes))
  342. for i, inRoute := range r.StaticRoutes {
  343. var err error
  344. outRoute := &types.StaticRoute{RouteType: inRoute.RouteType}
  345. if inRoute.Destination != "" {
  346. if outRoute.Destination, err = types.ParseCIDR(inRoute.Destination); err != nil {
  347. return nil, err
  348. }
  349. }
  350. if inRoute.NextHop != "" {
  351. outRoute.NextHop = net.ParseIP(inRoute.NextHop)
  352. if outRoute.NextHop == nil {
  353. return nil, fmt.Errorf("failed to parse nexthop IP %s", inRoute.NextHop)
  354. }
  355. }
  356. routes[i] = outRoute
  357. }
  358. return routes, nil
  359. }
  360. // parseInterfaces validates all the parameters of an Interface and returns them.
  361. func parseInterface(r api.CreateEndpointResponse) (*api.Interface, error) {
  362. var outIf *api.Interface
  363. inIf := r.Interface
  364. if inIf != nil {
  365. var err error
  366. outIf = &api.Interface{}
  367. if inIf.Address != "" {
  368. if outIf.Address, err = types.ParseCIDR(inIf.Address); err != nil {
  369. return nil, err
  370. }
  371. }
  372. if inIf.AddressIPv6 != "" {
  373. if outIf.AddressIPv6, err = types.ParseCIDR(inIf.AddressIPv6); err != nil {
  374. return nil, err
  375. }
  376. }
  377. if inIf.MacAddress != "" {
  378. if outIf.MacAddress, err = net.ParseMAC(inIf.MacAddress); err != nil {
  379. return nil, err
  380. }
  381. }
  382. }
  383. return outIf, nil
  384. }