driver.go 11 KB

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