porter_app_event.go 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. package gorm
  2. import (
  3. "context"
  4. "encoding/json"
  5. "errors"
  6. "strconv"
  7. "time"
  8. "github.com/porter-dev/porter/internal/telemetry"
  9. "github.com/google/uuid"
  10. "github.com/porter-dev/porter/internal/models"
  11. "github.com/porter-dev/porter/internal/repository"
  12. "github.com/porter-dev/porter/internal/repository/gorm/helpers"
  13. "gorm.io/gorm"
  14. )
  15. // PorterAppEventRepository uses gorm.DB for querying the database
  16. type PorterAppEventRepository struct {
  17. db *gorm.DB
  18. }
  19. // NewPorterAppEventRepository returns a PorterAppEventRepository which uses
  20. // gorm.DB for querying the database
  21. func NewPorterAppEventRepository(db *gorm.DB) repository.PorterAppEventRepository {
  22. return &PorterAppEventRepository{db}
  23. }
  24. func (repo *PorterAppEventRepository) ListEventsByPorterAppID(ctx context.Context, porterAppID uint, opts ...helpers.QueryOption) ([]*models.PorterAppEvent, helpers.PaginatedResult, error) {
  25. apps := []*models.PorterAppEvent{}
  26. paginatedResult := helpers.PaginatedResult{}
  27. id := strconv.Itoa(int(porterAppID))
  28. if id == "" {
  29. return nil, paginatedResult, errors.New("invalid porter app id supplied")
  30. }
  31. db := repo.db.Model(&models.PorterAppEvent{})
  32. resultDB := db.Where("porter_app_id = ?", id).Order("created_at DESC")
  33. resultDB = resultDB.Scopes(helpers.Paginate(db, &paginatedResult, opts...))
  34. if err := resultDB.Find(&apps).Error; err != nil {
  35. if !errors.Is(err, gorm.ErrRecordNotFound) {
  36. return nil, paginatedResult, err
  37. }
  38. }
  39. return apps, paginatedResult, nil
  40. }
  41. // ListEventsByPorterAppIDAndDeploymentTargetID returns a list of events for a given porter app id and deployment target id
  42. func (repo *PorterAppEventRepository) ListEventsByPorterAppIDAndDeploymentTargetID(ctx context.Context, porterAppID uint, deploymentTargetID uuid.UUID, opts ...helpers.QueryOption) ([]*models.PorterAppEvent, helpers.PaginatedResult, error) {
  43. ctx, span := telemetry.NewSpan(ctx, "list-events-by-porter-app-id-and-deployment-target-id")
  44. defer span.End()
  45. telemetry.WithAttributes(span,
  46. telemetry.AttributeKV{Key: "porter-app-id", Value: porterAppID},
  47. telemetry.AttributeKV{Key: "deployment-target-id", Value: deploymentTargetID},
  48. )
  49. apps := []*models.PorterAppEvent{}
  50. paginatedResult := helpers.PaginatedResult{}
  51. id := strconv.Itoa(int(porterAppID))
  52. if id == "" {
  53. return nil, paginatedResult, telemetry.Error(ctx, span, nil, "invalid porter app id supplied")
  54. }
  55. db := repo.db.Model(&models.PorterAppEvent{})
  56. resultDB := db.Where("porter_app_id = ? AND deployment_target_id = ?", id, deploymentTargetID).Order("created_at DESC")
  57. resultDB = resultDB.Scopes(helpers.Paginate(db, &paginatedResult, opts...))
  58. if err := resultDB.Find(&apps).Error; err != nil {
  59. if !errors.Is(err, gorm.ErrRecordNotFound) {
  60. return nil, paginatedResult, telemetry.Error(ctx, span, err, "error finding events by porter app id and deployment target id")
  61. }
  62. }
  63. return apps, paginatedResult, nil
  64. }
  65. func (repo *PorterAppEventRepository) CreateEvent(ctx context.Context, appEvent *models.PorterAppEvent) error {
  66. if appEvent.ID == uuid.Nil {
  67. appEvent.ID = uuid.New()
  68. }
  69. if appEvent.CreatedAt.IsZero() {
  70. appEvent.CreatedAt = time.Now().UTC()
  71. }
  72. if appEvent.UpdatedAt.IsZero() {
  73. appEvent.UpdatedAt = time.Now().UTC()
  74. }
  75. if appEvent.PorterAppID == 0 {
  76. return errors.New("invalid porter app id supplied to create event")
  77. }
  78. if err := repo.db.Create(appEvent).Error; err != nil {
  79. return err
  80. }
  81. return nil
  82. }
  83. // UpdateEvent will set all values in the database to the values of the passed in appEvent
  84. func (repo *PorterAppEventRepository) UpdateEvent(ctx context.Context, appEvent *models.PorterAppEvent) error {
  85. if appEvent.PorterAppID == 0 {
  86. return errors.New("invalid porter app id supplied to update event")
  87. }
  88. if appEvent.ID == uuid.Nil {
  89. return errors.New("invalid porter app event id supplied to update event")
  90. }
  91. if appEvent.UpdatedAt.IsZero() {
  92. appEvent.UpdatedAt = time.Now().UTC()
  93. }
  94. if err := repo.db.Model(appEvent).Updates(appEvent).Error; err != nil {
  95. return err
  96. }
  97. return nil
  98. }
  99. func (repo *PorterAppEventRepository) ReadEvent(ctx context.Context, id uuid.UUID) (models.PorterAppEvent, error) {
  100. appEvent := models.PorterAppEvent{}
  101. if id == uuid.Nil {
  102. return appEvent, errors.New("invalid porter app event id supplied")
  103. }
  104. strID := id.String()
  105. if err := repo.db.Where("id = ?", strID).First(&appEvent).Error; err != nil {
  106. return appEvent, err
  107. }
  108. return appEvent, nil
  109. }
  110. func (repo *PorterAppEventRepository) ReadDeployEventByRevision(ctx context.Context, porterAppID uint, revision float64) (models.PorterAppEvent, error) {
  111. appEvent := models.PorterAppEvent{}
  112. if porterAppID == 0 {
  113. return appEvent, errors.New("invalid porter app ID supplied")
  114. }
  115. // Convert porterAppID to string
  116. strAppID := strconv.Itoa(int(porterAppID))
  117. // Convert revision to JSON number string
  118. revJSON, err := json.Marshal(revision)
  119. if err != nil {
  120. return appEvent, errors.New("unable to marshal revision")
  121. }
  122. strRevision := string(revJSON)
  123. if err := repo.db.Where("porter_app_id = ? AND type = 'DEPLOY' AND metadata->>'revision' = ?", strAppID, strRevision).First(&appEvent).Error; err != nil {
  124. return appEvent, err
  125. }
  126. return appEvent, nil
  127. }
  128. // ReadDeployEventByAppRevisionID returns a deploy event for a given porter app id and app revision ID
  129. func (repo *PorterAppEventRepository) ReadDeployEventByAppRevisionID(ctx context.Context, porterAppID uint, appRevisionID string) (models.PorterAppEvent, error) {
  130. appEvent := models.PorterAppEvent{}
  131. if porterAppID == 0 {
  132. return appEvent, errors.New("invalid porter app ID supplied")
  133. }
  134. if appRevisionID == "" {
  135. return appEvent, errors.New("no app revision ID supplied")
  136. }
  137. // Convert porterAppID to string
  138. strAppID := strconv.Itoa(int(porterAppID))
  139. if err := repo.db.Where("porter_app_id = ? AND type = 'DEPLOY' AND metadata->>'app_revision_id' = ?", strAppID, appRevisionID).First(&appEvent).Error; err != nil {
  140. return appEvent, err
  141. }
  142. return appEvent, nil
  143. }