enrollmentkey.go 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. package logic
  2. import (
  3. b64 "encoding/base64"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "strings"
  8. "sync"
  9. "time"
  10. "github.com/google/uuid"
  11. "github.com/gravitl/netmaker/database"
  12. "github.com/gravitl/netmaker/models"
  13. "github.com/gravitl/netmaker/servercfg"
  14. "golang.org/x/exp/slices"
  15. )
  16. // EnrollmentErrors - struct for holding EnrollmentKey error messages
  17. var EnrollmentErrors = struct {
  18. InvalidCreate error
  19. NoKeyFound error
  20. InvalidKey error
  21. NoUsesRemaining error
  22. FailedToTokenize error
  23. FailedToDeTokenize error
  24. }{
  25. InvalidCreate: fmt.Errorf("failed to create enrollment key. paramters invalid"),
  26. NoKeyFound: fmt.Errorf("no enrollmentkey found"),
  27. InvalidKey: fmt.Errorf("invalid key provided"),
  28. NoUsesRemaining: fmt.Errorf("no uses remaining"),
  29. FailedToTokenize: fmt.Errorf("failed to tokenize"),
  30. FailedToDeTokenize: fmt.Errorf("failed to detokenize"),
  31. }
  32. var (
  33. enrollmentkeyCacheMutex = &sync.RWMutex{}
  34. enrollmentkeyCacheMap = make(map[string]models.EnrollmentKey)
  35. )
  36. // CreateEnrollmentKey - creates a new enrollment key in db
  37. func CreateEnrollmentKey(uses int, expiration time.Time, networks,
  38. tags []string, groups []models.TagID, unlimited bool, relay uuid.UUID,
  39. defaultKey, autoEgress, autoAssignGw bool) (*models.EnrollmentKey, error) {
  40. newKeyID, err := getUniqueEnrollmentID()
  41. if err != nil {
  42. return nil, err
  43. }
  44. k := &models.EnrollmentKey{
  45. Value: newKeyID,
  46. Expiration: time.Time{},
  47. UsesRemaining: 0,
  48. Unlimited: unlimited,
  49. Networks: []string{},
  50. Tags: []string{},
  51. Type: models.Undefined,
  52. Relay: relay,
  53. Groups: groups,
  54. Default: defaultKey,
  55. AutoEgress: autoEgress,
  56. AutoAssignGateway: autoAssignGw,
  57. }
  58. if uses > 0 {
  59. k.UsesRemaining = uses
  60. k.Type = models.Uses
  61. } else if !expiration.IsZero() {
  62. k.Expiration = expiration
  63. k.Type = models.TimeExpiration
  64. } else if k.Unlimited {
  65. k.Type = models.Unlimited
  66. }
  67. if len(networks) > 0 {
  68. k.Networks = networks
  69. }
  70. if len(tags) > 0 {
  71. k.Tags = tags
  72. }
  73. if err := k.Validate(); err != nil {
  74. return nil, err
  75. }
  76. if relay != uuid.Nil {
  77. relayNode, err := GetNodeByID(relay.String())
  78. if err != nil {
  79. return nil, err
  80. }
  81. if !slices.Contains(k.Networks, relayNode.Network) {
  82. return nil, errors.New("relay node not in key's networks")
  83. }
  84. if !relayNode.IsRelay {
  85. return nil, errors.New("relay node is not a relay")
  86. }
  87. }
  88. if err = upsertEnrollmentKey(k); err != nil {
  89. return nil, err
  90. }
  91. return k, nil
  92. }
  93. // UpdateEnrollmentKey - updates an existing enrollment key's associated relay
  94. func UpdateEnrollmentKey(keyId string, relayId uuid.UUID, groups []models.TagID) (*models.EnrollmentKey, error) {
  95. key, err := GetEnrollmentKey(keyId)
  96. if err != nil {
  97. return nil, err
  98. }
  99. if relayId != uuid.Nil {
  100. relayNode, err := GetNodeByID(relayId.String())
  101. if err != nil {
  102. return nil, err
  103. }
  104. if !slices.Contains(key.Networks, relayNode.Network) {
  105. return nil, errors.New("relay node not in key's networks")
  106. }
  107. if !relayNode.IsRelay {
  108. return nil, errors.New("relay node is not a relay")
  109. }
  110. }
  111. key.Relay = relayId
  112. key.Groups = groups
  113. if err = upsertEnrollmentKey(&key); err != nil {
  114. return nil, err
  115. }
  116. return &key, nil
  117. }
  118. // GetAllEnrollmentKeys - fetches all enrollment keys from DB
  119. func GetAllEnrollmentKeys() ([]models.EnrollmentKey, error) {
  120. currentKeys, err := getEnrollmentKeysMap()
  121. if err != nil {
  122. return nil, err
  123. }
  124. var currentKeysList = []models.EnrollmentKey{}
  125. for k := range currentKeys {
  126. currentKeysList = append(currentKeysList, currentKeys[k])
  127. }
  128. return currentKeysList, nil
  129. }
  130. // GetEnrollmentKey - fetches a single enrollment key
  131. // returns nil and error if not found
  132. func GetEnrollmentKey(value string) (key models.EnrollmentKey, err error) {
  133. currentKeys, err := getEnrollmentKeysMap()
  134. if err != nil {
  135. return key, err
  136. }
  137. if key, ok := currentKeys[value]; ok {
  138. return key, nil
  139. }
  140. return key, EnrollmentErrors.NoKeyFound
  141. }
  142. func deleteEnrollmentkeyFromCache(key string) {
  143. enrollmentkeyCacheMutex.Lock()
  144. delete(enrollmentkeyCacheMap, key)
  145. enrollmentkeyCacheMutex.Unlock()
  146. }
  147. // DeleteEnrollmentKey - delete's a given enrollment key by value
  148. func DeleteEnrollmentKey(value string, force bool) error {
  149. key, err := GetEnrollmentKey(value)
  150. if err != nil {
  151. return err
  152. }
  153. if key.Default && !force {
  154. return errors.New("cannot delete default network key")
  155. }
  156. err = database.DeleteRecord(database.ENROLLMENT_KEYS_TABLE_NAME, value)
  157. if err == nil {
  158. if servercfg.CacheEnabled() {
  159. deleteEnrollmentkeyFromCache(value)
  160. }
  161. }
  162. return err
  163. }
  164. // TryToUseEnrollmentKey - checks first if key can be decremented
  165. // returns true if it is decremented or isvalid
  166. func TryToUseEnrollmentKey(k *models.EnrollmentKey) bool {
  167. key, err := decrementEnrollmentKey(k.Value)
  168. if err != nil {
  169. if errors.Is(err, EnrollmentErrors.NoUsesRemaining) {
  170. return k.IsValid()
  171. }
  172. } else {
  173. k.UsesRemaining = key.UsesRemaining
  174. return true
  175. }
  176. return false
  177. }
  178. // Tokenize - tokenizes an enrollment key to be used via registration
  179. // and attaches it to the Token field on the struct
  180. func Tokenize(k *models.EnrollmentKey, serverAddr string) error {
  181. if len(serverAddr) == 0 || k == nil {
  182. return EnrollmentErrors.FailedToTokenize
  183. }
  184. newToken := models.EnrollmentToken{
  185. Server: serverAddr,
  186. Value: k.Value,
  187. }
  188. data, err := json.Marshal(&newToken)
  189. if err != nil {
  190. return err
  191. }
  192. k.Token = b64.StdEncoding.EncodeToString(data)
  193. return nil
  194. }
  195. // DeTokenize - detokenizes a base64 encoded string
  196. // and finds the associated enrollment key
  197. func DeTokenize(b64Token string) (*models.EnrollmentKey, error) {
  198. if len(b64Token) == 0 {
  199. return nil, EnrollmentErrors.FailedToDeTokenize
  200. }
  201. tokenData, err := b64.StdEncoding.DecodeString(b64Token)
  202. if err != nil {
  203. return nil, err
  204. }
  205. var newToken models.EnrollmentToken
  206. err = json.Unmarshal(tokenData, &newToken)
  207. if err != nil {
  208. return nil, err
  209. }
  210. k, err := GetEnrollmentKey(newToken.Value)
  211. if err != nil {
  212. return nil, err
  213. }
  214. return &k, nil
  215. }
  216. // == private ==
  217. // decrementEnrollmentKey - decrements the uses on a key if above 0 remaining
  218. func decrementEnrollmentKey(value string) (*models.EnrollmentKey, error) {
  219. k, err := GetEnrollmentKey(value)
  220. if err != nil {
  221. return nil, err
  222. }
  223. if k.UsesRemaining == 0 {
  224. return nil, EnrollmentErrors.NoUsesRemaining
  225. }
  226. k.UsesRemaining = k.UsesRemaining - 1
  227. if err = upsertEnrollmentKey(&k); err != nil {
  228. return nil, err
  229. }
  230. return &k, nil
  231. }
  232. func upsertEnrollmentKey(k *models.EnrollmentKey) error {
  233. if k == nil {
  234. return EnrollmentErrors.InvalidKey
  235. }
  236. data, err := json.Marshal(k)
  237. if err != nil {
  238. return err
  239. }
  240. err = database.Insert(k.Value, string(data), database.ENROLLMENT_KEYS_TABLE_NAME)
  241. if err == nil {
  242. if servercfg.CacheEnabled() {
  243. storeEnrollmentkeyInCache(k.Value, *k)
  244. }
  245. }
  246. return nil
  247. }
  248. func getUniqueEnrollmentID() (string, error) {
  249. currentKeys, err := getEnrollmentKeysMap()
  250. if err != nil {
  251. return "", err
  252. }
  253. newID := RandomString(models.EnrollmentKeyLength)
  254. for _, ok := currentKeys[newID]; ok; {
  255. newID = RandomString(models.EnrollmentKeyLength)
  256. }
  257. return newID, nil
  258. }
  259. func getEnrollmentkeysFromCache() map[string]models.EnrollmentKey {
  260. return enrollmentkeyCacheMap
  261. }
  262. func storeEnrollmentkeyInCache(key string, enrollmentkey models.EnrollmentKey) {
  263. enrollmentkeyCacheMutex.Lock()
  264. enrollmentkeyCacheMap[key] = enrollmentkey
  265. enrollmentkeyCacheMutex.Unlock()
  266. }
  267. func getEnrollmentKeysMap() (map[string]models.EnrollmentKey, error) {
  268. if servercfg.CacheEnabled() {
  269. keys := getEnrollmentkeysFromCache()
  270. if len(keys) != 0 {
  271. return keys, nil
  272. }
  273. }
  274. records, err := database.FetchRecords(database.ENROLLMENT_KEYS_TABLE_NAME)
  275. if err != nil {
  276. if !database.IsEmptyRecord(err) {
  277. return nil, err
  278. }
  279. }
  280. if records == nil {
  281. records = make(map[string]string)
  282. }
  283. currentKeys := make(map[string]models.EnrollmentKey, 0)
  284. if len(records) > 0 {
  285. for k := range records {
  286. var currentKey models.EnrollmentKey
  287. if err = json.Unmarshal([]byte(records[k]), &currentKey); err != nil {
  288. continue
  289. }
  290. currentKeys[k] = currentKey
  291. if servercfg.CacheEnabled() {
  292. storeEnrollmentkeyInCache(currentKey.Value, currentKey)
  293. }
  294. }
  295. }
  296. return currentKeys, nil
  297. }
  298. func RemoveTagFromEnrollmentKeys(deletedTagID models.TagID) {
  299. keys, _ := GetAllEnrollmentKeys()
  300. for _, key := range keys {
  301. newTags := []models.TagID{}
  302. update := false
  303. for _, tagID := range key.Groups {
  304. if tagID == deletedTagID {
  305. update = true
  306. continue
  307. }
  308. newTags = append(newTags, tagID)
  309. }
  310. if update {
  311. key.Groups = newTags
  312. upsertEnrollmentKey(&key)
  313. }
  314. }
  315. }
  316. func UnlinkNetworkAndTagsFromEnrollmentKeys(network string, delete bool) error {
  317. keys, err := GetAllEnrollmentKeys()
  318. if err != nil {
  319. return fmt.Errorf("failed to retrieve keys: %w", err)
  320. }
  321. var errs []error
  322. for _, key := range keys {
  323. newNetworks := []string{}
  324. newTags := []models.TagID{}
  325. update := false
  326. // Check and update networks
  327. for _, net := range key.Networks {
  328. if net == network {
  329. update = true
  330. continue
  331. }
  332. newNetworks = append(newNetworks, net)
  333. }
  334. // Check and update tags
  335. for _, tag := range key.Groups {
  336. tagParts := strings.Split(tag.String(), ".")
  337. if len(tagParts) == 0 {
  338. continue
  339. }
  340. tagNetwork := tagParts[0]
  341. if tagNetwork == network {
  342. update = true
  343. continue
  344. }
  345. newTags = append(newTags, tag)
  346. }
  347. if update && len(newNetworks) == 0 && delete {
  348. if err := DeleteEnrollmentKey(key.Value, true); err != nil {
  349. errs = append(errs, fmt.Errorf("failed to delete key %s: %w", key.Value, err))
  350. }
  351. continue
  352. }
  353. if update {
  354. key.Networks = newNetworks
  355. key.Groups = newTags
  356. if err := upsertEnrollmentKey(&key); err != nil {
  357. errs = append(errs, fmt.Errorf("failed to update key %s: %w", key.Value, err))
  358. }
  359. }
  360. }
  361. if len(errs) > 0 {
  362. return fmt.Errorf("errors unlinking network/tags from keys: %v", errs)
  363. }
  364. return nil
  365. }