network.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331
  1. package models
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "reflect"
  7. "strings"
  8. "time"
  9. "github.com/gravitl/netmaker/servercfg"
  10. "github.com/go-playground/validator/v10"
  11. "github.com/gravitl/netmaker/database"
  12. )
  13. //Network Struct
  14. //At some point, need to replace all instances of Name with something else like Identifier
  15. type Network struct {
  16. AddressRange string `json:"addressrange" bson:"addressrange" validate:"required,cidr"`
  17. 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]))?$"`
  18. DisplayName string `json:"displayname,omitempty" bson:"displayname,omitempty" validate:"omitempty,min=1,max=20,displayname_valid"`
  19. NetID string `json:"netid" bson:"netid" validate:"required,min=1,max=12,netid_valid"`
  20. NodesLastModified int64 `json:"nodeslastmodified" bson:"nodeslastmodified"`
  21. NetworkLastModified int64 `json:"networklastmodified" bson:"networklastmodified"`
  22. DefaultInterface string `json:"defaultinterface" bson:"defaultinterface" validate:"min=1,max=15"`
  23. DefaultListenPort int32 `json:"defaultlistenport,omitempty" bson:"defaultlistenport,omitempty" validate:"omitempty,min=1024,max=65535"`
  24. NodeLimit int32 `json:"nodelimit" bson:"nodelimit"`
  25. DefaultPostUp string `json:"defaultpostup" bson:"defaultpostup"`
  26. DefaultPostDown string `json:"defaultpostdown" bson:"defaultpostdown"`
  27. KeyUpdateTimeStamp int64 `json:"keyupdatetimestamp" bson:"keyupdatetimestamp"`
  28. DefaultKeepalive int32 `json:"defaultkeepalive" bson:"defaultkeepalive" validate:"omitempty,max=1000"`
  29. DefaultSaveConfig string `json:"defaultsaveconfig" bson:"defaultsaveconfig" validate:"checkyesorno"`
  30. AccessKeys []AccessKey `json:"accesskeys" bson:"accesskeys"`
  31. AllowManualSignUp string `json:"allowmanualsignup" bson:"allowmanualsignup" validate:"checkyesorno"`
  32. IsLocal string `json:"islocal" bson:"islocal" validate:"checkyesorno"`
  33. IsDualStack string `json:"isdualstack" bson:"isdualstack" validate:"checkyesorno"`
  34. IsIPv4 string `json:"isipv4" bson:"isipv4" validate:"checkyesorno"`
  35. IsIPv6 string `json:"isipv6" bson:"isipv6" validate:"checkyesorno"`
  36. IsGRPCHub string `json:"isgrpchub" bson:"isgrpchub" validate:"checkyesorno"`
  37. LocalRange string `json:"localrange" bson:"localrange" validate:"omitempty,cidr"`
  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. }
  41. type SaveData struct { // put sensitive fields here
  42. NetID string `json:"netid" bson:"netid" validate:"required,min=1,max=12,netid_valid"`
  43. }
  44. const STRING_FIELD_TYPE = "string"
  45. const INT64_FIELD_TYPE = "int64"
  46. const INT32_FIELD_TYPE = "int32"
  47. const ACCESS_KEY_TYPE = "[]AccessKey"
  48. var FIELD_TYPES = []string{STRING_FIELD_TYPE, INT64_FIELD_TYPE, INT32_FIELD_TYPE, ACCESS_KEY_TYPE}
  49. var FIELDS = map[string][]string{
  50. // "id": {"ID", "string"},
  51. "addressrange": {"AddressRange", STRING_FIELD_TYPE},
  52. "addressrange6": {"AddressRange6", STRING_FIELD_TYPE},
  53. "displayname": {"DisplayName", STRING_FIELD_TYPE},
  54. "netid": {"NetID", STRING_FIELD_TYPE},
  55. "nodeslastmodified": {"NodesLastModified", INT64_FIELD_TYPE},
  56. "networklastmodified": {"NetworkLastModified", INT64_FIELD_TYPE},
  57. "defaultinterface": {"DefaultInterface", STRING_FIELD_TYPE},
  58. "defaultlistenport": {"DefaultListenPort", INT32_FIELD_TYPE},
  59. "nodelimit": {"NodeLimit", INT32_FIELD_TYPE},
  60. "defaultpostup": {"DefaultPostUp", STRING_FIELD_TYPE},
  61. "defaultpostdown": {"DefaultPostDown", STRING_FIELD_TYPE},
  62. "keyupdatetimestamp": {"KeyUpdateTimeStamp", INT64_FIELD_TYPE},
  63. "defaultkeepalive": {"DefaultKeepalive", INT32_FIELD_TYPE},
  64. "defaultsaveconfig": {"DefaultSaveConfig", STRING_FIELD_TYPE},
  65. "accesskeys": {"AccessKeys", ACCESS_KEY_TYPE},
  66. "allowmanualsignup": {"AllowManualSignUp", STRING_FIELD_TYPE},
  67. "islocal": {"IsLocal", STRING_FIELD_TYPE},
  68. "isdualstack": {"IsDualStack", STRING_FIELD_TYPE},
  69. "isipv4": {"IsIPv4", STRING_FIELD_TYPE},
  70. "isipv6": {"IsIPv6", STRING_FIELD_TYPE},
  71. "isgrpchub": {"IsGRPCHub", STRING_FIELD_TYPE},
  72. "localrange": {"LocalRange", STRING_FIELD_TYPE},
  73. "checkininterval": {"DefaultCheckInInterval", INT32_FIELD_TYPE},
  74. "defaultudpholepunch": {"DefaultUDPHolePunch", STRING_FIELD_TYPE},
  75. }
  76. func (network *Network) FieldExists(field string) bool {
  77. return len(FIELDS[field]) > 0
  78. }
  79. func (network *Network) NetIDInNetworkCharSet() bool {
  80. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_."
  81. for _, char := range network.NetID {
  82. if !strings.Contains(charset, strings.ToLower(string(char))) {
  83. return false
  84. }
  85. }
  86. return true
  87. }
  88. func (network *Network) DisplayNameInNetworkCharSet() bool {
  89. charset := "abcdefghijklmnopqrstuvwxyz1234567890-_./;% ^#()!@$*"
  90. for _, char := range network.DisplayName {
  91. if !strings.Contains(charset, strings.ToLower(string(char))) {
  92. return false
  93. }
  94. }
  95. return true
  96. }
  97. // Anyway, returns all the networks
  98. func GetNetworks() ([]Network, error) {
  99. var networks []Network
  100. collection, err := database.FetchRecords(database.NETWORKS_TABLE_NAME)
  101. if err != nil {
  102. return networks, err
  103. }
  104. for _, value := range collection {
  105. var network Network
  106. if err := json.Unmarshal([]byte(value), &network); err != nil {
  107. return networks, err
  108. }
  109. // add network our array
  110. networks = append(networks, network)
  111. }
  112. return networks, err
  113. }
  114. func (network *Network) IsNetworkDisplayNameUnique() (bool, error) {
  115. isunique := true
  116. records, err := GetNetworks()
  117. if err != nil && !database.IsEmptyRecord(err) {
  118. return false, err
  119. }
  120. for i := 0; i < len(records); i++ {
  121. if network.NetID == records[i].DisplayName {
  122. isunique = false
  123. }
  124. }
  125. return isunique, nil
  126. }
  127. //Checks to see if any other networks have the same name (id)
  128. func (network *Network) IsNetworkNameUnique() (bool, error) {
  129. isunique := true
  130. dbs, err := GetNetworks()
  131. if err != nil && !database.IsEmptyRecord(err) {
  132. return false, err
  133. }
  134. for i := 0; i < len(dbs); i++ {
  135. if network.NetID == dbs[i].NetID {
  136. isunique = false
  137. }
  138. }
  139. return isunique, nil
  140. }
  141. func (network *Network) Validate(isUpdate bool) error {
  142. v := validator.New()
  143. _ = v.RegisterValidation("netid_valid", func(fl validator.FieldLevel) bool {
  144. inCharSet := network.NetIDInNetworkCharSet()
  145. if isUpdate {
  146. return inCharSet
  147. }
  148. isFieldUnique, _ := network.IsNetworkNameUnique()
  149. return isFieldUnique && inCharSet
  150. })
  151. //
  152. _ = v.RegisterValidation("displayname_valid", func(fl validator.FieldLevel) bool {
  153. isFieldUnique, _ := network.IsNetworkDisplayNameUnique()
  154. inCharSet := network.DisplayNameInNetworkCharSet()
  155. if isUpdate {
  156. return inCharSet
  157. }
  158. return isFieldUnique && inCharSet
  159. })
  160. _ = v.RegisterValidation("checkyesorno", func(fl validator.FieldLevel) bool {
  161. return CheckYesOrNo(fl)
  162. })
  163. err := v.Struct(network)
  164. if err != nil {
  165. for _, e := range err.(validator.ValidationErrors) {
  166. fmt.Println(e)
  167. }
  168. }
  169. return err
  170. }
  171. //TODO:
  172. //Not sure if we need the below two functions. Got rid of one of the calls. May want to revisit
  173. func (network *Network) SetNodesLastModified() {
  174. network.NodesLastModified = time.Now().Unix()
  175. }
  176. func (network *Network) SetNetworkLastModified() {
  177. network.NetworkLastModified = time.Now().Unix()
  178. }
  179. func (network *Network) SetDefaults() {
  180. if network.DefaultUDPHolePunch == "" {
  181. if servercfg.IsClientMode() {
  182. network.DefaultUDPHolePunch = "yes"
  183. } else {
  184. network.DefaultUDPHolePunch = "no"
  185. }
  186. }
  187. if network.IsLocal == "" {
  188. network.IsLocal = "no"
  189. }
  190. if network.IsGRPCHub == "" {
  191. network.IsGRPCHub = "no"
  192. }
  193. if network.DisplayName == "" {
  194. network.DisplayName = network.NetID
  195. }
  196. if network.DefaultInterface == "" {
  197. if len(network.NetID) < 13 {
  198. network.DefaultInterface = "nm-" + network.NetID
  199. } else {
  200. network.DefaultInterface = network.NetID
  201. }
  202. }
  203. if network.DefaultListenPort == 0 {
  204. network.DefaultListenPort = 51821
  205. }
  206. if network.NodeLimit == 0 {
  207. network.NodeLimit = 999999999
  208. }
  209. if network.DefaultSaveConfig == "" {
  210. network.DefaultSaveConfig = "no"
  211. }
  212. if network.DefaultKeepalive == 0 {
  213. network.DefaultKeepalive = 20
  214. }
  215. //Check-In Interval for Nodes, In Seconds
  216. if network.DefaultCheckInInterval == 0 {
  217. network.DefaultCheckInInterval = 30
  218. }
  219. if network.AllowManualSignUp == "" {
  220. network.AllowManualSignUp = "no"
  221. }
  222. if network.IsDualStack == "" {
  223. network.IsDualStack = "no"
  224. }
  225. if network.IsDualStack == "yes" {
  226. network.IsIPv6 = "yes"
  227. network.IsIPv4 = "yes"
  228. } else {
  229. network.IsIPv6 = "no"
  230. network.IsIPv4 = "yes"
  231. }
  232. }
  233. func (network *Network) CopyValues(newNetwork *Network, fieldName string) {
  234. reflection := reflect.ValueOf(newNetwork)
  235. value := reflect.Indirect(reflection).FieldByName(FIELDS[fieldName][0])
  236. if value.IsValid() && len(FIELDS[fieldName]) == 2 {
  237. fieldData := FIELDS[fieldName]
  238. for _, indexVal := range FIELD_TYPES {
  239. if indexVal == fieldData[1] {
  240. currentReflection := reflect.ValueOf(network)
  241. reflect.Indirect(currentReflection).FieldByName(FIELDS[fieldName][0]).Set(value)
  242. }
  243. }
  244. }
  245. }
  246. func (currentNetwork *Network) Update(newNetwork *Network) (bool, bool, error) {
  247. if err := newNetwork.Validate(true); err != nil {
  248. return false, false, err
  249. }
  250. if newNetwork.NetID == currentNetwork.NetID {
  251. hasrangeupdate := newNetwork.AddressRange != currentNetwork.AddressRange
  252. localrangeupdate := newNetwork.LocalRange != currentNetwork.LocalRange
  253. if data, err := json.Marshal(newNetwork); err != nil {
  254. return false, false, err
  255. } else {
  256. newNetwork.SetNetworkLastModified()
  257. err = database.Insert(newNetwork.NetID, string(data), database.NETWORKS_TABLE_NAME)
  258. return hasrangeupdate, localrangeupdate, err
  259. }
  260. }
  261. // copy values
  262. return false, false, errors.New("failed to update network " + newNetwork.NetID + ", cannot change netid.")
  263. }
  264. func (network *Network) SetNetworkNodesLastModified() error {
  265. timestamp := time.Now().Unix()
  266. network.NodesLastModified = timestamp
  267. data, err := json.Marshal(&network)
  268. if err != nil {
  269. return err
  270. }
  271. err = database.Insert(network.NetID, string(data), database.NETWORKS_TABLE_NAME)
  272. if err != nil {
  273. return err
  274. }
  275. return nil
  276. }
  277. func GetNetwork(networkname string) (Network, error) {
  278. var network Network
  279. networkData, err := database.FetchRecord(database.NETWORKS_TABLE_NAME, networkname)
  280. if err != nil {
  281. return network, err
  282. }
  283. if err = json.Unmarshal([]byte(networkData), &network); err != nil {
  284. return Network{}, err
  285. }
  286. return network, nil
  287. }