nodeGrpcController.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379
  1. package controller
  2. import (
  3. "context"
  4. "fmt"
  5. "log"
  6. "github.com/gravitl/netmaker/functions"
  7. nodepb "github.com/gravitl/netmaker/grpc"
  8. "github.com/gravitl/netmaker/models"
  9. "github.com/gravitl/netmaker/servercfg"
  10. "google.golang.org/grpc/codes"
  11. "google.golang.org/grpc/status"
  12. )
  13. type NodeServiceServer struct {
  14. nodepb.UnimplementedNodeServiceServer
  15. }
  16. func (s *NodeServiceServer) ReadNode(ctx context.Context, req *nodepb.ReadNodeReq) (*nodepb.ReadNodeRes, error) {
  17. // convert string id (from proto) to mongoDB ObjectId
  18. macaddress := req.GetMacaddress()
  19. networkName := req.GetNetwork()
  20. network, _ := functions.GetParentNetwork(networkName)
  21. node, err := GetNode(macaddress, networkName)
  22. if err != nil {
  23. return nil, status.Errorf(codes.InvalidArgument, fmt.Sprintf("Something went wrong: %v", err))
  24. }
  25. /*
  26. if node == nil {
  27. return nil, status.Errorf(codes.NotFound, fmt.Sprintf("Could not find node with Mac Address %s: %v", req.GetMacaddress(), err))
  28. }
  29. */
  30. // Cast to ReadNodeRes type
  31. response := &nodepb.ReadNodeRes{
  32. Node: &nodepb.Node{
  33. Macaddress: node.MacAddress,
  34. Name: node.Name,
  35. Address: node.Address,
  36. Address6: node.Address6,
  37. Endpoint: node.Endpoint,
  38. Password: node.Password,
  39. Nodenetwork: node.Network,
  40. Interface: node.Interface,
  41. Localaddress: node.LocalAddress,
  42. Postdown: node.PostDown,
  43. Postup: node.PostUp,
  44. Checkininterval: node.CheckInInterval,
  45. Dnsoff: !servercfg.IsDNSMode(),
  46. Ispending: node.IsPending == "yes",
  47. Isingressgateway: node.IsIngressGateway == "yes",
  48. Ingressgatewayrange: node.IngressGatewayRange,
  49. Publickey: node.PublicKey,
  50. Listenport: node.ListenPort,
  51. Keepalive: node.PersistentKeepalive,
  52. Islocal: network.IsLocal == "yes",
  53. Isdualstack: network.IsDualStack == "yes",
  54. Localrange: network.LocalRange,
  55. Udpholepunch: node.UDPHolePunch,
  56. },
  57. }
  58. return response, nil
  59. }
  60. func (s *NodeServiceServer) CreateNode(ctx context.Context, req *nodepb.CreateNodeReq) (*nodepb.CreateNodeRes, error) {
  61. // Get the protobuf node type from the protobuf request type
  62. // Essentially doing req.Node to access the struct with a nil check
  63. data := req.GetNode()
  64. // Now we have to convert this into a NodeItem type to convert into BSON
  65. node := models.Node{
  66. // ID: primitive.NilObjectID,
  67. MacAddress: data.GetMacaddress(),
  68. LocalAddress: data.GetLocaladdress(),
  69. Name: data.GetName(),
  70. Address: data.GetAddress(),
  71. Address6: data.GetAddress6(),
  72. AccessKey: data.GetAccesskey(),
  73. Endpoint: data.GetEndpoint(),
  74. PersistentKeepalive: data.GetKeepalive(),
  75. Password: data.GetPassword(),
  76. Interface: data.GetInterface(),
  77. Network: data.GetNodenetwork(),
  78. PublicKey: data.GetPublickey(),
  79. ListenPort: data.GetListenport(),
  80. UDPHolePunch: data.GetUdpholepunch(),
  81. }
  82. //Check to see if key is valid
  83. //TODO: Triple inefficient!!! This is the third call to the DB we make for networks
  84. validKey := functions.IsKeyValid(node.Network, node.AccessKey)
  85. network, err := functions.GetParentNetwork(node.Network)
  86. if err != nil {
  87. return nil, status.Errorf(codes.NotFound, fmt.Sprintf("Could not find network: %v", err))
  88. }
  89. if !validKey {
  90. //Check to see if network will allow manual sign up
  91. //may want to switch this up with the valid key check and avoid a DB call that way.
  92. if network.AllowManualSignUp == "yes" {
  93. node.IsPending = "yes"
  94. } else {
  95. return nil, status.Errorf(
  96. codes.Internal,
  97. fmt.Sprintf("Invalid key, and network does not allow no-key signups"),
  98. )
  99. }
  100. }
  101. node, err = CreateNode(node, node.Network)
  102. if err != nil {
  103. // return internal gRPC error to be handled later
  104. return nil, status.Errorf(
  105. codes.Internal,
  106. fmt.Sprintf("Internal error: %v", err),
  107. )
  108. }
  109. // return the node in a CreateNodeRes type
  110. response := &nodepb.CreateNodeRes{
  111. Node: &nodepb.Node{
  112. Macaddress: node.MacAddress,
  113. Localaddress: node.LocalAddress,
  114. Name: node.Name,
  115. Address: node.Address,
  116. Address6: node.Address6,
  117. Endpoint: node.Endpoint,
  118. Password: node.Password,
  119. Interface: node.Interface,
  120. Nodenetwork: node.Network,
  121. Dnsoff: !servercfg.IsDNSMode(),
  122. Ispending: node.IsPending == "yes",
  123. Publickey: node.PublicKey,
  124. Listenport: node.ListenPort,
  125. Keepalive: node.PersistentKeepalive,
  126. Islocal: network.IsLocal == "yes",
  127. Isdualstack: network.IsDualStack == "yes",
  128. Localrange: network.LocalRange,
  129. Udpholepunch: node.UDPHolePunch,
  130. },
  131. }
  132. err = SetNetworkNodesLastModified(node.Network)
  133. if err != nil {
  134. return nil, status.Errorf(codes.NotFound, fmt.Sprintf("Could not update network last modified date: %v", err))
  135. }
  136. return response, nil
  137. }
  138. func (s *NodeServiceServer) CheckIn(ctx context.Context, req *nodepb.CheckInReq) (*nodepb.CheckInRes, error) {
  139. // Get the protobuf node type from the protobuf request type
  140. // Essentially doing req.Node to access the struct with a nil check
  141. data := req.GetNode()
  142. //postchanges := req.GetPostchanges()
  143. // Now we have to convert this into a NodeItem type to convert into BSON
  144. log.Println("checkin data:", data)
  145. node := models.Node{
  146. // ID: primitive.NilObjectID,
  147. MacAddress: data.GetMacaddress(),
  148. Address: data.GetAddress(),
  149. Address6: data.GetAddress6(),
  150. Endpoint: data.GetEndpoint(),
  151. Network: data.GetNodenetwork(),
  152. Password: data.GetPassword(),
  153. LocalAddress: data.GetLocaladdress(),
  154. ListenPort: data.GetListenport(),
  155. PersistentKeepalive: data.GetKeepalive(),
  156. PublicKey: data.GetPublickey(),
  157. UDPHolePunch: data.GetUdpholepunch(),
  158. SaveConfig: data.GetSaveconfig(),
  159. }
  160. checkinresponse, err := NodeCheckIn(node, node.Network)
  161. if err != nil {
  162. // return internal gRPC error to be handled later
  163. if checkinresponse == (models.CheckInResponse{}) || !checkinresponse.IsPending {
  164. return nil, status.Errorf(
  165. codes.Internal,
  166. fmt.Sprintf("Internal error: %v", err),
  167. )
  168. }
  169. }
  170. // return the node in a CreateNodeRes type
  171. response := &nodepb.CheckInRes{
  172. Checkinresponse: &nodepb.CheckInResponse{
  173. Success: checkinresponse.Success,
  174. Needpeerupdate: checkinresponse.NeedPeerUpdate,
  175. Needdelete: checkinresponse.NeedDelete,
  176. Needconfigupdate: checkinresponse.NeedConfigUpdate,
  177. Needkeyupdate: checkinresponse.NeedKeyUpdate,
  178. Nodemessage: checkinresponse.NodeMessage,
  179. Ispending: checkinresponse.IsPending,
  180. },
  181. }
  182. return response, nil
  183. }
  184. func (s *NodeServiceServer) UpdateNode(ctx context.Context, req *nodepb.UpdateNodeReq) (*nodepb.UpdateNodeRes, error) {
  185. // Get the node data from the request
  186. data := req.GetNode()
  187. log.Println("DATA:", data)
  188. // Now we have to convert this into a NodeItem type to convert into BSON
  189. newnode := models.Node{
  190. // ID: primitive.NilObjectID,
  191. MacAddress: data.GetMacaddress(),
  192. Name: data.GetName(),
  193. Address: data.GetAddress(),
  194. Address6: data.GetAddress6(),
  195. LocalAddress: data.GetLocaladdress(),
  196. Endpoint: data.GetEndpoint(),
  197. Password: data.GetPassword(),
  198. PersistentKeepalive: data.GetKeepalive(),
  199. Network: data.GetNodenetwork(),
  200. Interface: data.GetInterface(),
  201. PostDown: data.GetPostdown(),
  202. PostUp: data.GetPostup(),
  203. PublicKey: data.GetPublickey(),
  204. ListenPort: data.GetListenport(),
  205. UDPHolePunch: data.GetUdpholepunch(),
  206. SaveConfig: data.GetSaveconfig(),
  207. }
  208. // Convert the Id string to a MongoDB ObjectId
  209. macaddress := newnode.MacAddress
  210. networkName := newnode.Network
  211. network, _ := functions.GetParentNetwork(networkName)
  212. log.Println("NODE SAVECONFIG:", newnode.SaveConfig)
  213. node, err := functions.GetNodeByMacAddress(networkName, macaddress)
  214. if err != nil {
  215. return nil, status.Errorf(
  216. codes.NotFound,
  217. fmt.Sprintf("Could not find node with supplied Mac Address: %v", err),
  218. )
  219. }
  220. err = node.Update(&newnode)
  221. if err != nil {
  222. return nil, status.Errorf(
  223. codes.NotFound,
  224. fmt.Sprintf("Could not update node: %v", err),
  225. )
  226. }
  227. return &nodepb.UpdateNodeRes{
  228. Node: &nodepb.Node{
  229. Macaddress: newnode.MacAddress,
  230. Localaddress: newnode.LocalAddress,
  231. Name: newnode.Name,
  232. Address: newnode.Address,
  233. Address6: newnode.Address6,
  234. Endpoint: newnode.Endpoint,
  235. Password: newnode.Password,
  236. Interface: newnode.Interface,
  237. Postdown: newnode.PostDown,
  238. Postup: newnode.PostUp,
  239. Nodenetwork: newnode.Network,
  240. Ispending: newnode.IsPending == "yes",
  241. Publickey: newnode.PublicKey,
  242. Dnsoff: !servercfg.IsDNSMode(),
  243. Listenport: newnode.ListenPort,
  244. Keepalive: newnode.PersistentKeepalive,
  245. Islocal: network.IsLocal == "yes",
  246. Isdualstack: network.IsDualStack == "yes",
  247. Localrange: network.LocalRange,
  248. Udpholepunch: newnode.UDPHolePunch,
  249. },
  250. }, nil
  251. }
  252. func (s *NodeServiceServer) DeleteNode(ctx context.Context, req *nodepb.DeleteNodeReq) (*nodepb.DeleteNodeRes, error) {
  253. macaddress := req.GetMacaddress()
  254. network := req.GetNetworkName()
  255. err := DeleteNode(macaddress, network)
  256. if err != nil {
  257. fmt.Println("Error deleting node.")
  258. fmt.Println(err)
  259. return nil, status.Errorf(codes.NotFound, fmt.Sprintf("Could not find/delete node with mac address %s", macaddress))
  260. }
  261. fmt.Println("updating network last modified of " + req.GetNetworkName())
  262. err = SetNetworkNodesLastModified(req.GetNetworkName())
  263. if err != nil {
  264. fmt.Println("Error updating Network")
  265. fmt.Println(err)
  266. return nil, status.Errorf(codes.NotFound, fmt.Sprintf("Could not update network last modified date: %v", err))
  267. }
  268. return &nodepb.DeleteNodeRes{
  269. Success: true,
  270. }, nil
  271. }
  272. func (s *NodeServiceServer) GetPeers(req *nodepb.GetPeersReq, stream nodepb.NodeService_GetPeersServer) error {
  273. // Initiate a NodeItem type to write decoded data to
  274. //data := &models.PeersResponse{}
  275. // collection.Find returns a cursor for our (empty) query
  276. peers, err := GetPeersList(req.GetNetwork())
  277. if err != nil {
  278. return status.Errorf(codes.Internal, fmt.Sprintf("Unknown internal error: %v", err))
  279. }
  280. // cursor.Next() returns a boolean, if false there are no more items and loop will break
  281. for i := 0; i < len(peers); i++ {
  282. // If no error is found send node over stream
  283. stream.Send(&nodepb.GetPeersRes{
  284. Peers: &nodepb.PeersResponse{
  285. Address: peers[i].Address,
  286. Address6: peers[i].Address6,
  287. Endpoint: peers[i].Endpoint,
  288. Egressgatewayrange: peers[i].EgressGatewayRange,
  289. Isegressgateway: peers[i].IsEgressGateway == "yes",
  290. Publickey: peers[i].PublicKey,
  291. Keepalive: peers[i].KeepAlive,
  292. Listenport: peers[i].ListenPort,
  293. Localaddress: peers[i].LocalAddress,
  294. },
  295. })
  296. }
  297. node, err := functions.GetNodeByMacAddress(req.GetNetwork(), req.GetMacaddress())
  298. if err != nil {
  299. return status.Errorf(codes.Internal, fmt.Sprintf("Could not get node: %v", err))
  300. }
  301. err = TimestampNode(node, false, true, false)
  302. if err != nil {
  303. return status.Errorf(codes.Internal, fmt.Sprintf("Internal error occurred: %v", err))
  304. }
  305. return nil
  306. }
  307. func (s *NodeServiceServer) GetExtPeers(req *nodepb.GetExtPeersReq, stream nodepb.NodeService_GetExtPeersServer) error {
  308. // Initiate a NodeItem type to write decoded data to
  309. //data := &models.PeersResponse{}
  310. // collection.Find returns a cursor for our (empty) query
  311. peers, err := GetExtPeersList(req.GetNetwork(), req.GetMacaddress())
  312. if err != nil {
  313. return status.Errorf(codes.Internal, fmt.Sprintf("Unknown internal error: %v", err))
  314. }
  315. // cursor.Next() returns a boolean, if false there are no more items and loop will break
  316. for i := 0; i < len(peers); i++ {
  317. // If no error is found send node over stream
  318. stream.Send(&nodepb.GetExtPeersRes{
  319. Extpeers: &nodepb.ExtPeersResponse{
  320. Address: peers[i].Address,
  321. Address6: peers[i].Address6,
  322. Endpoint: peers[i].Endpoint,
  323. Publickey: peers[i].PublicKey,
  324. Keepalive: peers[i].KeepAlive,
  325. Listenport: peers[i].ListenPort,
  326. Localaddress: peers[i].LocalAddress,
  327. },
  328. })
  329. }
  330. node, err := functions.GetNodeByMacAddress(req.GetNetwork(), req.GetMacaddress())
  331. if err != nil {
  332. return status.Errorf(codes.Internal, fmt.Sprintf("Could not get node: %v", err))
  333. }
  334. err = TimestampNode(node, false, true, false)
  335. if err != nil {
  336. return status.Errorf(codes.Internal, fmt.Sprintf("Internal error occurred: %v", err))
  337. }
  338. return nil
  339. }