network.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. package models
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "strings"
  7. "time"
  8. "github.com/go-playground/validator/v10"
  9. "github.com/gravitl/netmaker/database"
  10. "github.com/gravitl/netmaker/servercfg"
  11. )
  12. // Network Struct - contains info for a given unique network
  13. //At some point, need to replace all instances of Name with something else like Identifier
  14. type Network struct {
  15. AddressRange string `json:"addressrange" bson:"addressrange" validate:"required,cidr"`
  16. AddressRange6 string `json:"addressrange6" bson:"addressrange6" validate:"regexp=^s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:)))(%.+)?s*(\/([0-9]|[1-9][0-9]|1[0-1][0-9]|12[0-8]))?$"`
  17. DisplayName string `json:"displayname,omitempty" bson:"displayname,omitempty" validate:"omitempty,min=1,max=20,displayname_valid"`
  18. NetID string `json:"netid" bson:"netid" validate:"required,min=1,max=12,netid_valid"`
  19. NodesLastModified int64 `json:"nodeslastmodified" bson:"nodeslastmodified"`
  20. NetworkLastModified int64 `json:"networklastmodified" bson:"networklastmodified"`
  21. DefaultInterface string `json:"defaultinterface" bson:"defaultinterface" validate:"min=1,max=15"`
  22. DefaultListenPort int32 `json:"defaultlistenport,omitempty" bson:"defaultlistenport,omitempty" validate:"omitempty,min=1024,max=65535"`
  23. NodeLimit int32 `json:"nodelimit" bson:"nodelimit"`
  24. DefaultPostUp string `json:"defaultpostup" bson:"defaultpostup"`
  25. DefaultPostDown string `json:"defaultpostdown" bson:"defaultpostdown"`
  26. KeyUpdateTimeStamp int64 `json:"keyupdatetimestamp" bson:"keyupdatetimestamp"`
  27. DefaultKeepalive int32 `json:"defaultkeepalive" bson:"defaultkeepalive" validate:"omitempty,max=1000"`
  28. DefaultSaveConfig string `json:"defaultsaveconfig" bson:"defaultsaveconfig" validate:"checkyesorno"`
  29. AccessKeys []AccessKey `json:"accesskeys" bson:"accesskeys"`
  30. AllowManualSignUp string `json:"allowmanualsignup" bson:"allowmanualsignup" validate:"checkyesorno"`
  31. IsLocal string `json:"islocal" bson:"islocal" validate:"checkyesorno"`
  32. IsDualStack string `json:"isdualstack" bson:"isdualstack" validate:"checkyesorno"`
  33. IsIPv4 string `json:"isipv4" bson:"isipv4" validate:"checkyesorno"`
  34. IsIPv6 string `json:"isipv6" bson:"isipv6" validate:"checkyesorno"`
  35. IsGRPCHub string `json:"isgrpchub" bson:"isgrpchub" validate:"checkyesorno"`
  36. LocalRange string `json:"localrange" bson:"localrange" validate:"omitempty,cidr"`
  37. // checkin interval is depreciated at the network level. Set on server with CHECKIN_INTERVAL
  38. DefaultCheckInInterval int32 `json:"checkininterval,omitempty" bson:"checkininterval,omitempty" validate:"omitempty,numeric,min=2,max=100000"`
  39. DefaultUDPHolePunch string `json:"defaultudpholepunch" bson:"defaultudpholepunch" validate:"checkyesorno"`
  40. DefaultExtClientDNS string `json:"defaultextclientdns" bson:"defaultextclientdns"`
  41. DefaultMTU int32 `json:"defaultmtu" bson:"defaultmtu"`
  42. }
  43. // SaveData - sensitive fields of a network that should be kept the same
  44. type SaveData struct { // put sensitive fields here
  45. NetID string `json:"netid" bson:"netid" validate:"required,min=1,max=12,netid_valid"`
  46. }
  47. // Network.NetIDInNetworkCharSet - checks if a netid of a network uses valid characters
  48. func (network *Network) NetIDInNetworkCharSet() bool {
  49. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_."
  50. for _, char := range network.NetID {
  51. if !strings.Contains(charset, strings.ToLower(string(char))) {
  52. return false
  53. }
  54. }
  55. return true
  56. }
  57. // Network.DisplayNameInNetworkCharSet - checks if displayname uses valid characters
  58. func (network *Network) DisplayNameInNetworkCharSet() bool {
  59. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_./;% ^#()!@$*"
  60. for _, char := range network.DisplayName {
  61. if !strings.Contains(charset, strings.ToLower(string(char))) {
  62. return false
  63. }
  64. }
  65. return true
  66. }
  67. // GetNetworks - returns all networks from database
  68. func GetNetworks() ([]Network, error) {
  69. var networks []Network
  70. collection, err := database.FetchRecords(database.NETWORKS_TABLE_NAME)
  71. if err != nil {
  72. return networks, err
  73. }
  74. for _, value := range collection {
  75. var network Network
  76. if err := json.Unmarshal([]byte(value), &network); err != nil {
  77. return networks, err
  78. }
  79. // add network our array
  80. networks = append(networks, network)
  81. }
  82. return networks, err
  83. }
  84. // Network.IsNetworkDisplayNameUnique - checks if displayname is unique from other networks
  85. func (network *Network) IsNetworkDisplayNameUnique() (bool, error) {
  86. isunique := true
  87. records, err := GetNetworks()
  88. if err != nil && !database.IsEmptyRecord(err) {
  89. return false, err
  90. }
  91. for i := 0; i < len(records); i++ {
  92. if network.NetID == records[i].DisplayName {
  93. isunique = false
  94. }
  95. }
  96. return isunique, nil
  97. }
  98. // Network.IsNetworkNameUnique - checks to see if any other networks have the same name (id)
  99. func (network *Network) IsNetworkNameUnique() (bool, error) {
  100. isunique := true
  101. dbs, err := GetNetworks()
  102. if err != nil && !database.IsEmptyRecord(err) {
  103. return false, err
  104. }
  105. for i := 0; i < len(dbs); i++ {
  106. if network.NetID == dbs[i].NetID {
  107. isunique = false
  108. }
  109. }
  110. return isunique, nil
  111. }
  112. // Network.Validate - validates fields of an network struct
  113. func (network *Network) Validate(isUpdate bool) error {
  114. v := validator.New()
  115. _ = v.RegisterValidation("netid_valid", func(fl validator.FieldLevel) bool {
  116. inCharSet := network.NetIDInNetworkCharSet()
  117. if isUpdate {
  118. return inCharSet
  119. }
  120. isFieldUnique, _ := network.IsNetworkNameUnique()
  121. return isFieldUnique && inCharSet
  122. })
  123. //
  124. _ = v.RegisterValidation("displayname_valid", func(fl validator.FieldLevel) bool {
  125. isFieldUnique, _ := network.IsNetworkDisplayNameUnique()
  126. inCharSet := network.DisplayNameInNetworkCharSet()
  127. if isUpdate {
  128. return inCharSet
  129. }
  130. return isFieldUnique && inCharSet
  131. })
  132. _ = v.RegisterValidation("checkyesorno", func(fl validator.FieldLevel) bool {
  133. return CheckYesOrNo(fl)
  134. })
  135. err := v.Struct(network)
  136. if err != nil {
  137. for _, e := range err.(validator.ValidationErrors) {
  138. fmt.Println(e)
  139. }
  140. }
  141. return err
  142. }
  143. // Network.SetNodesLastModified - sets nodes last modified on network, depricated
  144. func (network *Network) SetNodesLastModified() {
  145. network.NodesLastModified = time.Now().Unix()
  146. }
  147. // Network.SetNetworkLastModified - sets network last modified time
  148. func (network *Network) SetNetworkLastModified() {
  149. network.NetworkLastModified = time.Now().Unix()
  150. }
  151. // Network.SetDefaults - sets default values for a network struct
  152. func (network *Network) SetDefaults() {
  153. if network.DefaultUDPHolePunch == "" {
  154. if servercfg.IsClientMode() != "off" {
  155. network.DefaultUDPHolePunch = "yes"
  156. } else {
  157. network.DefaultUDPHolePunch = "no"
  158. }
  159. }
  160. if network.IsLocal == "" {
  161. network.IsLocal = "no"
  162. }
  163. if network.IsGRPCHub == "" {
  164. network.IsGRPCHub = "no"
  165. }
  166. if network.DisplayName == "" {
  167. network.DisplayName = network.NetID
  168. }
  169. if network.DefaultInterface == "" {
  170. if len(network.NetID) < 13 {
  171. network.DefaultInterface = "nm-" + network.NetID
  172. } else {
  173. network.DefaultInterface = network.NetID
  174. }
  175. }
  176. if network.DefaultListenPort == 0 {
  177. network.DefaultListenPort = 51821
  178. }
  179. if network.NodeLimit == 0 {
  180. network.NodeLimit = 999999999
  181. }
  182. if network.DefaultSaveConfig == "" {
  183. network.DefaultSaveConfig = "no"
  184. }
  185. if network.DefaultKeepalive == 0 {
  186. network.DefaultKeepalive = 20
  187. }
  188. //Check-In Interval for Nodes, In Seconds
  189. if network.DefaultCheckInInterval == 0 {
  190. network.DefaultCheckInInterval = 30
  191. }
  192. if network.AllowManualSignUp == "" {
  193. network.AllowManualSignUp = "no"
  194. }
  195. if network.IsDualStack == "" {
  196. network.IsDualStack = "no"
  197. }
  198. if network.IsDualStack == "yes" {
  199. network.IsIPv6 = "yes"
  200. network.IsIPv4 = "yes"
  201. } else {
  202. network.IsIPv6 = "no"
  203. network.IsIPv4 = "yes"
  204. }
  205. if network.DefaultMTU == 0 {
  206. network.DefaultMTU = 1280
  207. }
  208. }
  209. // Network.Update - updates a network with another network's fields
  210. func (currentNetwork *Network) Update(newNetwork *Network) (bool, bool, error) {
  211. if err := newNetwork.Validate(true); err != nil {
  212. return false, false, err
  213. }
  214. if newNetwork.NetID == currentNetwork.NetID {
  215. hasrangeupdate := newNetwork.AddressRange != currentNetwork.AddressRange
  216. localrangeupdate := newNetwork.LocalRange != currentNetwork.LocalRange
  217. data, err := json.Marshal(newNetwork)
  218. if err != nil {
  219. return false, false, err
  220. }
  221. newNetwork.SetNetworkLastModified()
  222. err = database.Insert(newNetwork.NetID, string(data), database.NETWORKS_TABLE_NAME)
  223. return hasrangeupdate, localrangeupdate, err
  224. }
  225. // copy values
  226. return false, false, errors.New("failed to update network " + newNetwork.NetID + ", cannot change netid.")
  227. }
  228. // Network.SetNetworkNodesLastModified - sets network nodes last modified time
  229. func (network *Network) SetNetworkNodesLastModified() error {
  230. timestamp := time.Now().Unix()
  231. network.NodesLastModified = timestamp
  232. data, err := json.Marshal(&network)
  233. if err != nil {
  234. return err
  235. }
  236. err = database.Insert(network.NetID, string(data), database.NETWORKS_TABLE_NAME)
  237. if err != nil {
  238. return err
  239. }
  240. return nil
  241. }
  242. // GetNetwork - gets a network from database
  243. func GetNetwork(networkname string) (Network, error) {
  244. var network Network
  245. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, networkname)
  246. if err != nil {
  247. return network, err
  248. }
  249. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  250. return Network{}, err
  251. }
  252. return network, nil
  253. }