revisions.go 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. package porter_app
  2. import (
  3. "context"
  4. "encoding/base64"
  5. "errors"
  6. "time"
  7. "connectrpc.com/connect"
  8. "github.com/google/uuid"
  9. "github.com/porter-dev/api-contracts/generated/go/helpers"
  10. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  11. "github.com/porter-dev/api-contracts/generated/go/porter/v1/porterv1connect"
  12. "github.com/porter-dev/porter/internal/deployment_target"
  13. "github.com/porter-dev/porter/internal/kubernetes"
  14. "github.com/porter-dev/porter/internal/kubernetes/environment_groups"
  15. "github.com/porter-dev/porter/internal/models"
  16. "github.com/porter-dev/porter/internal/repository"
  17. "github.com/porter-dev/porter/internal/telemetry"
  18. )
  19. // Revision represents the data for a single revision
  20. type Revision struct {
  21. // ID is the revision id
  22. ID string `json:"id"`
  23. // B64AppProto is the base64 encoded app proto definition
  24. B64AppProto string `json:"b64_app_proto"`
  25. // Status is the status of the revision
  26. Status models.AppRevisionStatus `json:"status"`
  27. // RevisionNumber is the revision number with respect to the app and deployment target
  28. RevisionNumber uint64 `json:"revision_number"`
  29. // CreatedAt is the time the revision was created
  30. CreatedAt time.Time `json:"created_at"`
  31. // UpdatedAt is the time the revision was updated
  32. UpdatedAt time.Time `json:"updated_at"`
  33. // DeploymentTargetID is the id of the deployment target the revision is associated with
  34. DeploymentTarget DeploymentTarget `json:"deployment_target"`
  35. // Env is the environment variables for the revision
  36. Env environment_groups.EnvironmentGroup `json:"env,omitempty"`
  37. // AppInstanceID is the id of the app instance the revision is associated with
  38. AppInstanceID uuid.UUID `json:"app_instance_id"`
  39. }
  40. // DeploymentTarget is a simplified version of the deployment target struct
  41. type DeploymentTarget struct {
  42. ID string `json:"id"`
  43. Name string `json:"name"`
  44. }
  45. // GetAppRevisionInput is the input struct for GetAppRevisions
  46. type GetAppRevisionInput struct {
  47. ProjectID uint
  48. AppRevisionID uuid.UUID
  49. CCPClient porterv1connect.ClusterControlPlaneServiceClient
  50. }
  51. // GetAppRevision returns a single app revision by id
  52. func GetAppRevision(ctx context.Context, inp GetAppRevisionInput) (Revision, error) {
  53. ctx, span := telemetry.NewSpan(ctx, "get-app-revision")
  54. defer span.End()
  55. var revision Revision
  56. if inp.ProjectID == 0 {
  57. return revision, telemetry.Error(ctx, span, nil, "must provide a project id")
  58. }
  59. if inp.AppRevisionID == uuid.Nil {
  60. return revision, telemetry.Error(ctx, span, nil, "must provide an app revision id")
  61. }
  62. getRevisionReq := connect.NewRequest(&porterv1.GetAppRevisionRequest{
  63. ProjectId: int64(inp.ProjectID),
  64. AppRevisionId: inp.AppRevisionID.String(),
  65. })
  66. ccpResp, err := inp.CCPClient.GetAppRevision(ctx, getRevisionReq)
  67. if err != nil {
  68. return revision, telemetry.Error(ctx, span, err, "error getting app revision")
  69. }
  70. if ccpResp == nil || ccpResp.Msg == nil {
  71. return revision, telemetry.Error(ctx, span, nil, "get app revision response is nil")
  72. }
  73. appRevisionProto := ccpResp.Msg.AppRevision
  74. revision, err = EncodedRevisionFromProto(ctx, appRevisionProto)
  75. if err != nil {
  76. return revision, telemetry.Error(ctx, span, err, "error converting app revision from proto")
  77. }
  78. return revision, nil
  79. }
  80. // EncodedRevisionFromProto converts an AppRevision proto object into a Revision object
  81. func EncodedRevisionFromProto(ctx context.Context, appRevision *porterv1.AppRevision) (Revision, error) {
  82. ctx, span := telemetry.NewSpan(ctx, "encoded-revision-from-proto")
  83. defer span.End()
  84. var revision Revision
  85. if appRevision == nil {
  86. return revision, telemetry.Error(ctx, span, nil, "current app revision definition is nil")
  87. }
  88. appProto := appRevision.App
  89. if appProto == nil {
  90. return revision, telemetry.Error(ctx, span, nil, "app proto is nil")
  91. }
  92. encoded, err := helpers.MarshalContractObject(ctx, appProto)
  93. if err != nil {
  94. return revision, telemetry.Error(ctx, span, err, "error marshalling app proto back to json")
  95. }
  96. b64 := base64.StdEncoding.EncodeToString(encoded)
  97. status, err := appRevisionStatusFromProto(appRevision.Status)
  98. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "status", Value: string(status)})
  99. if err != nil {
  100. _ = telemetry.Error(ctx, span, nil, "unknown revision type") // flagged as an error for visibility
  101. }
  102. appInstanceIdStr := appRevision.AppInstanceId
  103. appInstanceId, err := uuid.Parse(appInstanceIdStr)
  104. if err != nil {
  105. return revision, telemetry.Error(ctx, span, err, "error parsing app instance id")
  106. }
  107. revision = Revision{
  108. B64AppProto: b64,
  109. Status: status,
  110. ID: appRevision.Id,
  111. RevisionNumber: appRevision.RevisionNumber,
  112. CreatedAt: appRevision.CreatedAt.AsTime(),
  113. UpdatedAt: appRevision.UpdatedAt.AsTime(),
  114. DeploymentTarget: DeploymentTarget{ID: appRevision.DeploymentTargetId},
  115. AppInstanceID: appInstanceId,
  116. }
  117. return revision, nil
  118. }
  119. // AttachEnvToRevisionInput is the input struct for AttachEnvToRevision
  120. type AttachEnvToRevisionInput struct {
  121. ProjectID uint
  122. ClusterID int
  123. Revision Revision
  124. DeploymentTarget deployment_target.DeploymentTarget
  125. K8SAgent *kubernetes.Agent
  126. PorterAppRepository repository.PorterAppRepository
  127. }
  128. // AttachEnvToRevision attaches the environment variables from the app's default env group to a revision
  129. // These are the variables that are displayed to the user in the UI as associated with the app rather than an env group
  130. func AttachEnvToRevision(ctx context.Context, inp AttachEnvToRevisionInput) (Revision, error) {
  131. ctx, span := telemetry.NewSpan(ctx, "attach-env-to-revision")
  132. defer span.End()
  133. revision := inp.Revision
  134. if inp.ProjectID == 0 {
  135. return revision, telemetry.Error(ctx, span, nil, "must provide a project id")
  136. }
  137. if inp.ClusterID == 0 {
  138. return revision, telemetry.Error(ctx, span, nil, "must provide a cluster id")
  139. }
  140. if inp.K8SAgent == nil {
  141. return revision, telemetry.Error(ctx, span, nil, "k8s agent is nil")
  142. }
  143. decoded, err := base64.StdEncoding.DecodeString(revision.B64AppProto)
  144. if err != nil {
  145. return revision, telemetry.Error(ctx, span, err, "error decoding app proto")
  146. }
  147. appDef := &porterv1.PorterApp{}
  148. err = helpers.UnmarshalContractObject(decoded, appDef)
  149. if err != nil {
  150. return revision, telemetry.Error(ctx, span, err, "error unmarshalling app proto")
  151. }
  152. envName, err := AppEnvGroupName(ctx, appDef.Name, inp.Revision.DeploymentTarget.ID, uint(inp.ClusterID), inp.PorterAppRepository)
  153. if err != nil {
  154. return revision, telemetry.Error(ctx, span, err, "error getting app env group name")
  155. }
  156. envNameFilter := []string{envName}
  157. envFromProtoInp := AppEnvironmentFromProtoInput{
  158. ProjectID: inp.ProjectID,
  159. ClusterID: inp.ClusterID,
  160. App: appDef,
  161. K8SAgent: inp.K8SAgent,
  162. DeploymentTarget: inp.DeploymentTarget,
  163. }
  164. envGroups, err := AppEnvironmentFromProto(ctx, envFromProtoInp, WithEnvGroupFilter(envNameFilter), WithSecrets())
  165. if err != nil {
  166. return revision, telemetry.Error(ctx, span, err, "error getting app environment from revision")
  167. }
  168. if len(envGroups) > 1 {
  169. return revision, telemetry.Error(ctx, span, err, "multiple app envs groups returned for same name")
  170. }
  171. if len(envGroups) == 1 {
  172. revision.Env = envGroups[0]
  173. }
  174. return revision, nil
  175. }
  176. func appRevisionStatusFromProto(status string) (models.AppRevisionStatus, error) {
  177. appRevisionStatus := models.AppRevisionStatus_Unknown
  178. switch status {
  179. case string(models.AppRevisionStatus_ImageAvailable):
  180. appRevisionStatus = models.AppRevisionStatus_ImageAvailable
  181. case string(models.AppRevisionStatus_AwaitingBuild):
  182. appRevisionStatus = models.AppRevisionStatus_AwaitingBuild
  183. case string(models.AppRevisionStatus_AwaitingPredeploy):
  184. appRevisionStatus = models.AppRevisionStatus_AwaitingPredeploy
  185. case string(models.AppRevisionStatus_Deployed):
  186. appRevisionStatus = models.AppRevisionStatus_Deployed
  187. case string(models.AppRevisionStatus_Deploying):
  188. appRevisionStatus = models.AppRevisionStatus_Deploying
  189. case string(models.AppRevisionStatus_AwaitingDeploy):
  190. appRevisionStatus = models.AppRevisionStatus_AwaitingDeploy
  191. case string(models.AppRevisionStatus_BuildCanceled):
  192. appRevisionStatus = models.AppRevisionStatus_BuildCanceled
  193. case string(models.AppRevisionStatus_BuildFailed):
  194. appRevisionStatus = models.AppRevisionStatus_BuildFailed
  195. case string(models.AppRevisionStatus_PredeployFailed):
  196. appRevisionStatus = models.AppRevisionStatus_PredeployFailed
  197. case string(models.AppRevisionStatus_PredeploySuccessful):
  198. appRevisionStatus = models.AppRevisionStatus_PredeploySuccessful
  199. case string(models.AppRevisionStatus_PredeployProgressing):
  200. appRevisionStatus = models.AppRevisionStatus_PredeployProgressing
  201. case string(models.AppRevisionStatus_DeployFailed):
  202. appRevisionStatus = models.AppRevisionStatus_DeployFailed
  203. case string(models.AppRevisionStatus_Created):
  204. appRevisionStatus = models.AppRevisionStatus_Created
  205. case string(models.AppRevisionStatus_BuildSuccessful):
  206. appRevisionStatus = models.AppRevisionStatus_BuildSuccessful
  207. case string(models.AppRevisionStatus_ApplyFailed):
  208. appRevisionStatus = models.AppRevisionStatus_ApplyFailed
  209. case string(models.AppRevisionStatus_UpdateFailed):
  210. appRevisionStatus = models.AppRevisionStatus_UpdateFailed
  211. default:
  212. return appRevisionStatus, errors.New("unknown app revision status")
  213. }
  214. return appRevisionStatus, nil
  215. }