2
0

enrollmentkey.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. package logic
  2. import (
  3. b64 "encoding/base64"
  4. "encoding/json"
  5. "errors"
  6. "fmt"
  7. "time"
  8. "github.com/gravitl/netmaker/database"
  9. "github.com/gravitl/netmaker/models"
  10. )
  11. // EnrollmentErrors - struct for holding EnrollmentKey error messages
  12. var EnrollmentErrors = struct {
  13. InvalidCreate error
  14. NoKeyFound error
  15. InvalidKey error
  16. NoUsesRemaining error
  17. FailedToTokenize error
  18. FailedToDeTokenize error
  19. }{
  20. InvalidCreate: fmt.Errorf("invalid enrollment key created"),
  21. NoKeyFound: fmt.Errorf("no enrollmentkey found"),
  22. InvalidKey: fmt.Errorf("invalid key provided"),
  23. NoUsesRemaining: fmt.Errorf("no uses remaining"),
  24. FailedToTokenize: fmt.Errorf("failed to tokenize"),
  25. FailedToDeTokenize: fmt.Errorf("failed to detokenize"),
  26. }
  27. // CreateEnrollmentKey - creates a new enrollment key in db
  28. func CreateEnrollmentKey(uses int, expiration time.Time, networks, tags []string, unlimited bool) (k *models.EnrollmentKey, err error) {
  29. newKeyID, err := getUniqueEnrollmentID()
  30. if err != nil {
  31. return nil, err
  32. }
  33. k = &models.EnrollmentKey{
  34. Value: newKeyID,
  35. Expiration: time.Time{},
  36. UsesRemaining: 0,
  37. Unlimited: unlimited,
  38. Networks: []string{},
  39. Tags: []string{},
  40. Type: models.Undefined,
  41. }
  42. if uses > 0 {
  43. k.UsesRemaining = uses
  44. k.Type = models.Uses
  45. } else if !expiration.IsZero() {
  46. k.Expiration = expiration
  47. k.Type = models.TimeExpiration
  48. } else if k.Unlimited {
  49. k.Type = models.Unlimited
  50. }
  51. if len(networks) > 0 {
  52. k.Networks = networks
  53. }
  54. if len(tags) > 0 {
  55. k.Tags = tags
  56. }
  57. if ok := k.Validate(); !ok {
  58. return nil, EnrollmentErrors.InvalidCreate
  59. }
  60. if err = upsertEnrollmentKey(k); err != nil {
  61. return nil, err
  62. }
  63. return
  64. }
  65. // GetAllEnrollmentKeys - fetches all enrollment keys from DB
  66. func GetAllEnrollmentKeys() ([]*models.EnrollmentKey, error) {
  67. currentKeys, err := getEnrollmentKeysMap()
  68. if err != nil {
  69. return nil, err
  70. }
  71. var currentKeysList = []*models.EnrollmentKey{}
  72. for k := range currentKeys {
  73. currentKeysList = append(currentKeysList, currentKeys[k])
  74. }
  75. return currentKeysList, nil
  76. }
  77. // GetEnrollmentKey - fetches a single enrollment key
  78. // returns nil and error if not found
  79. func GetEnrollmentKey(value string) (*models.EnrollmentKey, error) {
  80. currentKeys, err := getEnrollmentKeysMap()
  81. if err != nil {
  82. return nil, err
  83. }
  84. if key, ok := currentKeys[value]; ok {
  85. return key, nil
  86. }
  87. return nil, EnrollmentErrors.NoKeyFound
  88. }
  89. // DeleteEnrollmentKey - delete's a given enrollment key by value
  90. func DeleteEnrollmentKey(value string) error {
  91. _, err := GetEnrollmentKey(value)
  92. if err != nil {
  93. return err
  94. }
  95. return database.DeleteRecord(database.ENROLLMENT_KEYS_TABLE_NAME, value)
  96. }
  97. // TryToUseEnrollmentKey - checks first if key can be decremented
  98. // returns true if it is decremented or isvalid
  99. func TryToUseEnrollmentKey(k *models.EnrollmentKey) bool {
  100. key, err := decrementEnrollmentKey(k.Value)
  101. if err != nil {
  102. if errors.Is(err, EnrollmentErrors.NoUsesRemaining) {
  103. return k.IsValid()
  104. }
  105. } else {
  106. k.UsesRemaining = key.UsesRemaining
  107. return true
  108. }
  109. return false
  110. }
  111. // Tokenize - tokenizes an enrollment key to be used via registration
  112. // and attaches it to the Token field on the struct
  113. func Tokenize(k *models.EnrollmentKey, serverAddr string) error {
  114. if len(serverAddr) == 0 || k == nil {
  115. return EnrollmentErrors.FailedToTokenize
  116. }
  117. newToken := models.EnrollmentToken{
  118. Server: serverAddr,
  119. Value: k.Value,
  120. }
  121. data, err := json.Marshal(&newToken)
  122. if err != nil {
  123. return err
  124. }
  125. k.Token = b64.StdEncoding.EncodeToString(data)
  126. return nil
  127. }
  128. // DeTokenize - detokenizes a base64 encoded string
  129. // and finds the associated enrollment key
  130. func DeTokenize(b64Token string) (*models.EnrollmentKey, error) {
  131. if len(b64Token) == 0 {
  132. return nil, EnrollmentErrors.FailedToDeTokenize
  133. }
  134. tokenData, err := b64.StdEncoding.DecodeString(b64Token)
  135. if err != nil {
  136. return nil, err
  137. }
  138. var newToken models.EnrollmentToken
  139. err = json.Unmarshal(tokenData, &newToken)
  140. if err != nil {
  141. return nil, err
  142. }
  143. k, err := GetEnrollmentKey(newToken.Value)
  144. if err != nil {
  145. return nil, err
  146. }
  147. return k, nil
  148. }
  149. // == private ==
  150. // decrementEnrollmentKey - decrements the uses on a key if above 0 remaining
  151. func decrementEnrollmentKey(value string) (*models.EnrollmentKey, error) {
  152. k, err := GetEnrollmentKey(value)
  153. if err != nil {
  154. return nil, err
  155. }
  156. if k.UsesRemaining == 0 {
  157. return nil, EnrollmentErrors.NoUsesRemaining
  158. }
  159. k.UsesRemaining = k.UsesRemaining - 1
  160. if err = upsertEnrollmentKey(k); err != nil {
  161. return nil, err
  162. }
  163. return k, nil
  164. }
  165. func upsertEnrollmentKey(k *models.EnrollmentKey) error {
  166. if k == nil {
  167. return EnrollmentErrors.InvalidKey
  168. }
  169. data, err := json.Marshal(k)
  170. if err != nil {
  171. return err
  172. }
  173. return database.Insert(k.Value, string(data), database.ENROLLMENT_KEYS_TABLE_NAME)
  174. }
  175. func getUniqueEnrollmentID() (string, error) {
  176. currentKeys, err := getEnrollmentKeysMap()
  177. if err != nil {
  178. return "", err
  179. }
  180. newID := RandomString(models.EnrollmentKeyLength)
  181. for _, ok := currentKeys[newID]; ok; {
  182. newID = RandomString(models.EnrollmentKeyLength)
  183. }
  184. return newID, nil
  185. }
  186. func getEnrollmentKeysMap() (map[string]*models.EnrollmentKey, error) {
  187. records, err := database.FetchRecords(database.ENROLLMENT_KEYS_TABLE_NAME)
  188. if err != nil {
  189. if !database.IsEmptyRecord(err) {
  190. return nil, err
  191. }
  192. }
  193. if records == nil {
  194. records = make(map[string]string)
  195. }
  196. currentKeys := make(map[string]*models.EnrollmentKey, 0)
  197. if len(records) > 0 {
  198. for k := range records {
  199. var currentKey models.EnrollmentKey
  200. if err = json.Unmarshal([]byte(records[k]), &currentKey); err != nil {
  201. continue
  202. }
  203. currentKeys[k] = &currentKey
  204. }
  205. }
  206. return currentKeys, nil
  207. }