cluster.go 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426
  1. package gorm
  2. import (
  3. "context"
  4. "github.com/porter-dev/porter/internal/models"
  5. "github.com/porter-dev/porter/internal/repository"
  6. "gorm.io/gorm"
  7. ints "github.com/porter-dev/porter/internal/models/integrations"
  8. )
  9. // ClusterRepository uses gorm.DB for querying the database
  10. type ClusterRepository struct {
  11. db *gorm.DB
  12. key *[32]byte
  13. }
  14. // NewClusterRepository returns a ClusterRepository which uses
  15. // gorm.DB for querying the database. It accepts an encryption key to encrypt
  16. // sensitive data
  17. func NewClusterRepository(db *gorm.DB, key *[32]byte) repository.ClusterRepository {
  18. return &ClusterRepository{db, key}
  19. }
  20. // CreateClusterCandidate creates a new cluster candidate
  21. func (repo *ClusterRepository) CreateClusterCandidate(
  22. cc *models.ClusterCandidate,
  23. ) (*models.ClusterCandidate, error) {
  24. err := repo.EncryptClusterCandidateData(cc, repo.key)
  25. if err != nil {
  26. return nil, err
  27. }
  28. project := &models.Project{}
  29. if err := repo.db.Where("id = ?", cc.ProjectID).First(&project).Error; err != nil {
  30. return nil, err
  31. }
  32. assoc := repo.db.Model(&project).Association("ClusterCandidates")
  33. if assoc.Error != nil {
  34. return nil, assoc.Error
  35. }
  36. if err := assoc.Append(cc); err != nil {
  37. return nil, err
  38. }
  39. // decrypt at the end to return
  40. err = repo.DecryptClusterCandidateData(cc, repo.key)
  41. if err != nil {
  42. return nil, err
  43. }
  44. return cc, nil
  45. }
  46. // ReadClusterCandidate finds a cluster candidate by id
  47. func (repo *ClusterRepository) ReadClusterCandidate(
  48. projectID, ccID uint,
  49. ) (*models.ClusterCandidate, error) {
  50. cc := &models.ClusterCandidate{}
  51. if err := repo.db.Preload("Resolvers").Where("project_id = ? AND id = ?", projectID, ccID).First(&cc).Error; err != nil {
  52. return nil, err
  53. }
  54. repo.DecryptClusterCandidateData(cc, repo.key)
  55. return cc, nil
  56. }
  57. // ListClusterCandidatesByProjectID finds all cluster candidates
  58. // for a given project id
  59. func (repo *ClusterRepository) ListClusterCandidatesByProjectID(
  60. projectID uint,
  61. ) ([]*models.ClusterCandidate, error) {
  62. ccs := []*models.ClusterCandidate{}
  63. if err := repo.db.Preload("Resolvers").Where("project_id = ?", projectID).Find(&ccs).Error; err != nil {
  64. return nil, err
  65. }
  66. for _, cc := range ccs {
  67. repo.DecryptClusterCandidateData(cc, repo.key)
  68. }
  69. return ccs, nil
  70. }
  71. // UpdateClusterCandidateCreatedClusterID updates the CreatedClusterID for
  72. // a candidate, after the candidate has been resolved.
  73. func (repo *ClusterRepository) UpdateClusterCandidateCreatedClusterID(
  74. id uint,
  75. createdClusterID uint,
  76. ) (*models.ClusterCandidate, error) {
  77. cc := &models.ClusterCandidate{}
  78. if err := repo.db.Where("id = ?", id).First(&cc).Error; err != nil {
  79. return nil, err
  80. }
  81. cc.CreatedClusterID = createdClusterID
  82. if err := repo.db.Save(cc).Error; err != nil {
  83. return nil, err
  84. }
  85. repo.DecryptClusterCandidateData(cc, repo.key)
  86. return cc, nil
  87. }
  88. // CreateCluster creates a new cluster
  89. func (repo *ClusterRepository) CreateCluster(
  90. cluster *models.Cluster,
  91. ) (*models.Cluster, error) {
  92. ctxDB := repo.db.WithContext(context.Background())
  93. err := repo.EncryptClusterData(cluster, repo.key)
  94. if err != nil {
  95. return nil, err
  96. }
  97. project := &models.Project{}
  98. if err := ctxDB.Where("id = ?", cluster.ProjectID).First(&project).Error; err != nil {
  99. return nil, err
  100. }
  101. assoc := ctxDB.Model(&project).Association("Clusters")
  102. if assoc.Error != nil {
  103. return nil, assoc.Error
  104. }
  105. if err := assoc.Append(cluster); err != nil {
  106. return nil, err
  107. }
  108. // create a token cache by default
  109. cluster.TokenCache.ClusterID = cluster.ID
  110. if err := ctxDB.Create(&cluster.TokenCache).Error; err != nil {
  111. return nil, err
  112. }
  113. cluster.TokenCacheID = cluster.TokenCache.ID
  114. if err := ctxDB.Save(cluster).Error; err != nil {
  115. return nil, err
  116. }
  117. err = repo.DecryptClusterData(cluster, repo.key)
  118. if err != nil {
  119. return nil, err
  120. }
  121. return cluster, nil
  122. }
  123. // ReadCluster finds a cluster by id
  124. func (repo *ClusterRepository) ReadCluster(
  125. projectID, clusterID uint,
  126. ) (*models.Cluster, error) {
  127. ctxDB := repo.db.WithContext(context.Background())
  128. cluster := &models.Cluster{}
  129. // preload Clusters association
  130. if err := ctxDB.Where("project_id = ? AND id = ?", projectID, clusterID).First(&cluster).Error; err != nil {
  131. return nil, err
  132. }
  133. cache := ints.ClusterTokenCache{}
  134. if cluster.TokenCacheID != 0 {
  135. if err := ctxDB.Where("id = ?", cluster.TokenCacheID).First(&cache).Error; err != nil {
  136. return nil, err
  137. }
  138. }
  139. cluster.TokenCache = cache
  140. err := repo.DecryptClusterData(cluster, repo.key)
  141. if err != nil {
  142. return nil, err
  143. }
  144. return cluster, nil
  145. }
  146. // ListClustersByProjectID finds all clusters
  147. // for a given project id
  148. func (repo *ClusterRepository) ListClustersByProjectID(
  149. projectID uint,
  150. ) ([]*models.Cluster, error) {
  151. ctxDB := repo.db.WithContext(context.Background())
  152. clusters := []*models.Cluster{}
  153. if err := ctxDB.Where("project_id = ?", projectID).Find(&clusters).Error; err != nil {
  154. return nil, err
  155. }
  156. for _, cluster := range clusters {
  157. repo.DecryptClusterData(cluster, repo.key)
  158. }
  159. return clusters, nil
  160. }
  161. // UpdateCluster modifies an existing Cluster in the database
  162. func (repo *ClusterRepository) UpdateCluster(
  163. cluster *models.Cluster,
  164. ) (*models.Cluster, error) {
  165. ctxDB := repo.db.WithContext(context.Background())
  166. err := repo.EncryptClusterData(cluster, repo.key)
  167. if err != nil {
  168. return nil, err
  169. }
  170. if err := ctxDB.Save(cluster).Error; err != nil {
  171. return nil, err
  172. }
  173. err = repo.DecryptClusterData(cluster, repo.key)
  174. if err != nil {
  175. return nil, err
  176. }
  177. return cluster, nil
  178. }
  179. // UpdateClusterTokenCache updates the token cache for a cluster
  180. func (repo *ClusterRepository) UpdateClusterTokenCache(
  181. tokenCache *ints.ClusterTokenCache,
  182. ) (*models.Cluster, error) {
  183. ctxDB := repo.db.WithContext(context.Background())
  184. if tok := tokenCache.Token; len(tok) > 0 {
  185. cipherData, err := repository.Encrypt(tok, repo.key)
  186. if err != nil {
  187. return nil, err
  188. }
  189. tokenCache.Token = cipherData
  190. }
  191. cluster := &models.Cluster{}
  192. if err := ctxDB.Where("id = ?", tokenCache.ClusterID).First(&cluster).Error; err != nil {
  193. return nil, err
  194. }
  195. if cluster.TokenCacheID == 0 {
  196. tokenCache.ClusterID = cluster.ID
  197. if err := ctxDB.Create(tokenCache).Error; err != nil {
  198. return nil, err
  199. }
  200. cluster.TokenCacheID = tokenCache.ID
  201. if err := ctxDB.Save(cluster).Error; err != nil {
  202. return nil, err
  203. }
  204. } else {
  205. prev := &ints.ClusterTokenCache{}
  206. if err := ctxDB.Where("id = ?", cluster.TokenCacheID).First(prev).Error; err != nil {
  207. return nil, err
  208. }
  209. prev.Token = tokenCache.Token
  210. prev.Expiry = tokenCache.Expiry
  211. prev.ClusterID = cluster.ID
  212. if err := ctxDB.Save(prev).Error; err != nil {
  213. return nil, err
  214. }
  215. }
  216. return cluster, nil
  217. }
  218. // DeleteCluster removes a cluster from the db
  219. func (repo *ClusterRepository) DeleteCluster(
  220. cluster *models.Cluster,
  221. ) error {
  222. // clear TokenCache association
  223. if err := repo.db.Where("id = ?", cluster.TokenCacheID).Delete(&ints.ClusterTokenCache{}).Error; err != nil {
  224. return err
  225. }
  226. if err := repo.db.Where("id = ?", cluster.ID).Delete(&models.Cluster{}).Error; err != nil {
  227. return err
  228. }
  229. return nil
  230. }
  231. // EncryptClusterData will encrypt the user's service account data before writing
  232. // to the DB
  233. func (repo *ClusterRepository) EncryptClusterData(
  234. cluster *models.Cluster,
  235. key *[32]byte,
  236. ) error {
  237. if len(cluster.CertificateAuthorityData) > 0 {
  238. cipherData, err := repository.Encrypt(cluster.CertificateAuthorityData, key)
  239. if err != nil {
  240. return err
  241. }
  242. cluster.CertificateAuthorityData = cipherData
  243. }
  244. if tok := cluster.TokenCache.Token; len(tok) > 0 {
  245. cipherData, err := repository.Encrypt(tok, key)
  246. if err != nil {
  247. return err
  248. }
  249. cluster.TokenCache.Token = cipherData
  250. }
  251. return nil
  252. }
  253. // EncryptClusterCandidateData will encrypt the service account candidate data before
  254. // writing to the DB
  255. func (repo *ClusterRepository) EncryptClusterCandidateData(
  256. cc *models.ClusterCandidate,
  257. key *[32]byte,
  258. ) error {
  259. if len(cc.AWSClusterIDGuess) > 0 {
  260. cipherData, err := repository.Encrypt(cc.AWSClusterIDGuess, key)
  261. if err != nil {
  262. return err
  263. }
  264. cc.AWSClusterIDGuess = cipherData
  265. }
  266. if len(cc.Kubeconfig) > 0 {
  267. cipherData, err := repository.Encrypt(cc.Kubeconfig, key)
  268. if err != nil {
  269. return err
  270. }
  271. cc.Kubeconfig = cipherData
  272. }
  273. return nil
  274. }
  275. // DecryptClusterData will decrypt the user's service account data before
  276. // returning it from the DB
  277. func (repo *ClusterRepository) DecryptClusterData(
  278. cluster *models.Cluster,
  279. key *[32]byte,
  280. ) error {
  281. if len(cluster.CertificateAuthorityData) > 0 {
  282. plaintext, err := repository.Decrypt(cluster.CertificateAuthorityData, key)
  283. if err != nil {
  284. return err
  285. }
  286. cluster.CertificateAuthorityData = plaintext
  287. }
  288. if tok := cluster.TokenCache.Token; len(tok) > 0 {
  289. plaintext, err := repository.Decrypt(tok, key)
  290. // in the case that the token cache is down, set empty token
  291. if err != nil {
  292. cluster.TokenCache.Token = []byte{}
  293. } else {
  294. cluster.TokenCache.Token = plaintext
  295. }
  296. }
  297. return nil
  298. }
  299. // DecryptClusterCandidateData will decrypt the service account candidate data before
  300. // returning it from the DB
  301. func (repo *ClusterRepository) DecryptClusterCandidateData(
  302. cc *models.ClusterCandidate,
  303. key *[32]byte,
  304. ) error {
  305. if len(cc.AWSClusterIDGuess) > 0 {
  306. plaintext, err := repository.Decrypt(cc.AWSClusterIDGuess, key)
  307. if err != nil {
  308. return err
  309. }
  310. cc.AWSClusterIDGuess = plaintext
  311. }
  312. if len(cc.Kubeconfig) > 0 {
  313. plaintext, err := repository.Decrypt(cc.Kubeconfig, key)
  314. if err != nil {
  315. return err
  316. }
  317. cc.Kubeconfig = plaintext
  318. }
  319. return nil
  320. }