deployment.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360
  1. package dbstore
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "git.linuxforward.com/byop/byop-engine/dbmanager"
  6. "git.linuxforward.com/byop/byop-engine/models"
  7. "github.com/google/uuid"
  8. "gorm.io/gorm"
  9. )
  10. // DeploymentStore handles database operations for deployments
  11. type DeploymentStore struct {
  12. db *gorm.DB
  13. }
  14. // NewDeploymentStore creates a new DeploymentStore
  15. func NewDeploymentStore(dbManager dbmanager.DbManager) *DeploymentStore {
  16. return &DeploymentStore{
  17. db: dbManager.GetDB(),
  18. }
  19. }
  20. // Create creates a new deployment
  21. func (ds *DeploymentStore) CreateDeployment(deployment *models.Deployment) error {
  22. // Generate ID if not provided
  23. if deployment.ID == "" {
  24. deployment.ID = uuid.New().String()
  25. }
  26. // Ensure logs, metrics, and alerts config are properly JSON serialized
  27. if err := ds.serializeConfigFields(deployment); err != nil {
  28. return fmt.Errorf("failed to serialize config fields: %w", err)
  29. }
  30. // Create deployment in a transaction to handle deployed apps
  31. return ds.db.Transaction(func(tx *gorm.DB) error {
  32. // Create the deployment
  33. if err := tx.Create(deployment).Error; err != nil {
  34. return err
  35. }
  36. // Create any deployed apps in the same transaction
  37. for i := range deployment.DeployedComponents {
  38. app := &deployment.DeployedComponents[i]
  39. // Ensure each deployed app has an ID
  40. if app.ID == "" {
  41. app.ID = uuid.New().String()
  42. }
  43. // Set the deployment ID (ensure relationship is maintained)
  44. app.DeploymentID = deployment.ID
  45. // Create the deployed app
  46. if err := tx.Create(app).Error; err != nil {
  47. return err
  48. }
  49. // Handle resources if provided
  50. if app.Resources != (models.ResourceAllocation{}) {
  51. resource := models.DeployedComponentResource{
  52. ID: uuid.New().String(),
  53. DeployedComponentID: app.ID,
  54. CPU: app.Resources.CPU,
  55. CPUUsage: app.Resources.CPUUsage,
  56. Memory: app.Resources.Memory,
  57. MemoryUsage: app.Resources.MemoryUsage,
  58. Storage: app.Resources.Storage,
  59. StorageUsage: app.Resources.StorageUsage,
  60. }
  61. if err := tx.Create(&resource).Error; err != nil {
  62. return err
  63. }
  64. }
  65. }
  66. return nil
  67. })
  68. }
  69. // GetByID retrieves a deployment by ID
  70. func (ds *DeploymentStore) GetByID(id string) (*models.Deployment, error) {
  71. var deployment models.Deployment
  72. // Get deployment with all related deployed apps
  73. err := ds.db.
  74. Preload("DeployedComponents").
  75. First(&deployment, "id = ?", id).Error
  76. if err != nil {
  77. if err == gorm.ErrRecordNotFound {
  78. return nil, nil // No deployment found
  79. }
  80. return nil, fmt.Errorf("failed to get deployment: %w", err)
  81. }
  82. // Load resources for each deployed app
  83. for i, app := range deployment.DeployedComponents {
  84. var resource models.DeployedComponentResource
  85. if err := ds.db.Where("deployed_app_id = ?", app.ID).First(&resource).Error; err != nil {
  86. if err != gorm.ErrRecordNotFound {
  87. return nil, fmt.Errorf("failed to get resources for deployed app: %w", err)
  88. }
  89. } else {
  90. deployment.DeployedComponents[i].Resources = models.ResourceAllocation{
  91. CPU: resource.CPU,
  92. CPUUsage: resource.CPUUsage,
  93. Memory: resource.Memory,
  94. MemoryUsage: resource.MemoryUsage,
  95. Storage: resource.Storage,
  96. StorageUsage: resource.StorageUsage,
  97. }
  98. }
  99. }
  100. // Deserialize config fields
  101. if err := ds.deserializeConfigFields(&deployment); err != nil {
  102. return nil, fmt.Errorf("failed to deserialize config fields: %w", err)
  103. }
  104. return &deployment, nil
  105. }
  106. // Update updates an existing deployment
  107. func (ds *DeploymentStore) Update(deployment *models.Deployment) error {
  108. // Ensure logs, metrics, and alerts config are properly JSON serialized
  109. if err := ds.serializeConfigFields(deployment); err != nil {
  110. return fmt.Errorf("failed to serialize config fields: %w", err)
  111. }
  112. // Use transaction to handle deployment and deployed apps
  113. return ds.db.Transaction(func(tx *gorm.DB) error {
  114. // Update the deployment
  115. if err := tx.Save(deployment).Error; err != nil {
  116. return err
  117. }
  118. // Handle deployed apps - this is trickier as we need to compare with existing apps
  119. var existingComponents []models.DeployedComponent
  120. if err := tx.Where("deployment_id = ?", deployment.ID).Find(&existingComponents).Error; err != nil {
  121. return err
  122. }
  123. // Create a map of existing app IDs for quick lookup
  124. existingComponentMap := make(map[string]bool)
  125. for _, app := range existingComponents {
  126. existingComponentMap[app.ID] = true
  127. }
  128. // Process each app in the updated deployment
  129. for i := range deployment.DeployedComponents {
  130. app := &deployment.DeployedComponents[i]
  131. // If app has ID and exists, update it
  132. if app.ID != "" && existingComponentMap[app.ID] {
  133. if err := tx.Save(app).Error; err != nil {
  134. return err
  135. }
  136. delete(existingComponentMap, app.ID)
  137. } else {
  138. // New app, create it
  139. if app.ID == "" {
  140. app.ID = uuid.New().String()
  141. }
  142. app.DeploymentID = deployment.ID
  143. if err := tx.Create(app).Error; err != nil {
  144. return err
  145. }
  146. }
  147. // Handle resources
  148. if app.Resources != (models.ResourceAllocation{}) {
  149. var resource models.DeployedComponentResource
  150. result := tx.Where("deployed_app_id = ?", app.ID).First(&resource)
  151. if result.Error != nil && result.Error != gorm.ErrRecordNotFound {
  152. return result.Error
  153. }
  154. if result.Error == gorm.ErrRecordNotFound {
  155. // Create new resource
  156. resource = models.DeployedComponentResource{
  157. ID: uuid.New().String(),
  158. DeployedComponentID: app.ID,
  159. CPU: app.Resources.CPU,
  160. CPUUsage: app.Resources.CPUUsage,
  161. Memory: app.Resources.Memory,
  162. MemoryUsage: app.Resources.MemoryUsage,
  163. Storage: app.Resources.Storage,
  164. StorageUsage: app.Resources.StorageUsage,
  165. }
  166. if err := tx.Create(&resource).Error; err != nil {
  167. return err
  168. }
  169. } else {
  170. // Update existing resource
  171. resource.CPU = app.Resources.CPU
  172. resource.CPUUsage = app.Resources.CPUUsage
  173. resource.Memory = app.Resources.Memory
  174. resource.MemoryUsage = app.Resources.MemoryUsage
  175. resource.Storage = app.Resources.Storage
  176. resource.StorageUsage = app.Resources.StorageUsage
  177. if err := tx.Save(&resource).Error; err != nil {
  178. return err
  179. }
  180. }
  181. }
  182. }
  183. // Delete any apps that are no longer part of the deployment
  184. for appID := range existingComponentMap {
  185. if err := tx.Delete(&models.DeployedComponent{}, "id = ?", appID).Error; err != nil {
  186. return err
  187. }
  188. // Delete associated resources
  189. if err := tx.Delete(&models.DeployedComponentResource{}, "deployed_app_id = ?", appID).Error; err != nil && err != gorm.ErrRecordNotFound {
  190. return err
  191. }
  192. }
  193. return nil
  194. })
  195. }
  196. // Delete deletes a deployment by ID
  197. func (ds *DeploymentStore) Delete(id string) error {
  198. return ds.db.Transaction(func(tx *gorm.DB) error {
  199. // Delete associated DeployedComponentResources
  200. var deployedComponents []models.DeployedComponent
  201. if err := tx.Where("deployment_id = ?", id).Find(&deployedComponents).Error; err != nil {
  202. return err
  203. }
  204. for _, app := range deployedComponents {
  205. if err := tx.Delete(&models.DeployedComponentResource{}, "deployed_app_id = ?", app.ID).Error; err != nil && err != gorm.ErrRecordNotFound {
  206. return err
  207. }
  208. }
  209. // Delete deployed apps
  210. if err := tx.Delete(&models.DeployedComponent{}, "deployment_id = ?", id).Error; err != nil && err != gorm.ErrRecordNotFound {
  211. return err
  212. }
  213. // Delete the deployment itself
  214. return tx.Delete(&models.Deployment{}, "id = ?", id).Error
  215. })
  216. }
  217. // List retrieves all deployments with optional filtering
  218. func (ds *DeploymentStore) List(filter map[string]interface{}) ([]*models.Deployment, error) {
  219. var deployments []*models.Deployment
  220. // Build query from filters
  221. query := ds.db.Preload("DeployedComponents")
  222. if filter != nil {
  223. for key, value := range filter {
  224. query = query.Where(key+" = ?", value)
  225. }
  226. }
  227. // Execute query
  228. if err := query.Find(&deployments).Error; err != nil {
  229. return nil, fmt.Errorf("failed to list deployments: %w", err)
  230. }
  231. // Load resources and deserialize config for each deployment
  232. for i, deployment := range deployments {
  233. // Load resources for each deployed app
  234. for j, app := range deployment.DeployedComponents {
  235. var resource models.DeployedComponentResource
  236. if err := ds.db.Where("deployed_app_id = ?", app.ID).First(&resource).Error; err != nil {
  237. if err != gorm.ErrRecordNotFound {
  238. return nil, fmt.Errorf("failed to get resources for deployed app: %w", err)
  239. }
  240. } else {
  241. deployments[i].DeployedComponents[j].Resources = models.ResourceAllocation{
  242. CPU: resource.CPU,
  243. CPUUsage: resource.CPUUsage,
  244. Memory: resource.Memory,
  245. MemoryUsage: resource.MemoryUsage,
  246. Storage: resource.Storage,
  247. StorageUsage: resource.StorageUsage,
  248. }
  249. }
  250. }
  251. // Deserialize config fields
  252. if err := ds.deserializeConfigFields(deployments[i]); err != nil {
  253. return nil, fmt.Errorf("failed to deserialize config fields: %w", err)
  254. }
  255. }
  256. return deployments, nil
  257. }
  258. // GetByClientID retrieves deployments for a specific client
  259. func (ds *DeploymentStore) GetByClientID(clientID string) ([]*models.Deployment, error) {
  260. return ds.List(map[string]interface{}{"client_id": clientID})
  261. }
  262. // GetByUserID retrieves deployments created by a specific user
  263. func (ds *DeploymentStore) GetByUserID(userID string) ([]*models.Deployment, error) {
  264. return ds.List(map[string]interface{}{"created_by": userID})
  265. }
  266. // GetByBlueprintID retrieves deployments based on a specific blueprint
  267. func (ds *DeploymentStore) GetByBlueprintID(blueprintID string) ([]*models.Deployment, error) {
  268. return ds.List(map[string]interface{}{"blueprint_id": blueprintID})
  269. }
  270. // serializeConfigFields serializes JSON config fields to strings
  271. func (ds *DeploymentStore) serializeConfigFields(deployment *models.Deployment) error {
  272. // Serialize logs config if provided
  273. if deployment.LogsConfig == "" {
  274. logsConfig := models.LogConfiguration{
  275. Enabled: true,
  276. RetentionDays: 7,
  277. }
  278. logsConfigBytes, err := json.Marshal(logsConfig)
  279. if err != nil {
  280. return err
  281. }
  282. deployment.LogsConfig = string(logsConfigBytes)
  283. }
  284. // Serialize metrics config if provided
  285. if deployment.MetricsConfig == "" {
  286. metricsConfig := models.MetricsConfiguration{
  287. Enabled: true,
  288. RetentionDays: 30,
  289. }
  290. metricsConfigBytes, err := json.Marshal(metricsConfig)
  291. if err != nil {
  292. return err
  293. }
  294. deployment.MetricsConfig = string(metricsConfigBytes)
  295. }
  296. // Serialize alerts config if provided
  297. if deployment.AlertsConfig == "" {
  298. alertsConfig := []models.AlertConfiguration{}
  299. alertsConfigBytes, err := json.Marshal(alertsConfig)
  300. if err != nil {
  301. return err
  302. }
  303. deployment.AlertsConfig = string(alertsConfigBytes)
  304. }
  305. return nil
  306. }
  307. // deserializeConfigFields deserializes JSON config fields from strings
  308. func (ds *DeploymentStore) deserializeConfigFields(deployment *models.Deployment) error {
  309. // No need to deserialize in the store, as these fields are stored as strings
  310. // in the database and are deserialized as needed by the service layer
  311. return nil
  312. }