enrollmentkey.go 8.4 KB

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