revisions.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  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. // RevisionStatus describes the status of a revision
  41. type RevisionStatus struct {
  42. // PredeployStarted is true if the predeploy process has started
  43. PredeployStarted bool `json:"predeploy_started"`
  44. // PredeploySuccessful is true if the predeploy process has completed successfully
  45. PredeploySuccessful bool `json:"predeploy_successful"`
  46. // PredeployFailed is true if the predeploy process has failed
  47. PredeployFailed bool `json:"predeploy_failed"`
  48. // InstallStarted is true if the install process has started
  49. InstallStarted bool `json:"install_started"`
  50. // InstallSuccessful is true if the install process has completed successfully
  51. InstallSuccessful bool `json:"install_successful"`
  52. // InstallFailed is true if the install process has failed
  53. InstallFailed bool `json:"install_failed"`
  54. // DeploymentStarted is true if the deployment process has started
  55. DeploymentStarted bool `json:"deployment_started"`
  56. // DeploymentSuccessful is true if the deployment process has completed successfully
  57. DeploymentSuccessful bool `json:"deployment_successful"`
  58. // DeploymentFailed is true if the deployment process has failed
  59. DeploymentFailed bool `json:"deployment_failed"`
  60. // IsInTerminalStatus is true if the revision is in a terminal status
  61. IsInTerminalStatus bool `json:"is_in_terminal_status"`
  62. }
  63. // DeploymentTarget is a simplified version of the deployment target struct
  64. type DeploymentTarget struct {
  65. ID string `json:"id"`
  66. Name string `json:"name"`
  67. }
  68. // GetAppRevisionInput is the input struct for GetAppRevisions
  69. type GetAppRevisionInput struct {
  70. ProjectID uint
  71. AppRevisionID uuid.UUID
  72. CCPClient porterv1connect.ClusterControlPlaneServiceClient
  73. }
  74. // GetAppRevision returns a single app revision by id
  75. func GetAppRevision(ctx context.Context, inp GetAppRevisionInput) (Revision, error) {
  76. ctx, span := telemetry.NewSpan(ctx, "get-app-revision")
  77. defer span.End()
  78. var revision Revision
  79. if inp.ProjectID == 0 {
  80. return revision, telemetry.Error(ctx, span, nil, "must provide a project id")
  81. }
  82. if inp.AppRevisionID == uuid.Nil {
  83. return revision, telemetry.Error(ctx, span, nil, "must provide an app revision id")
  84. }
  85. getRevisionReq := connect.NewRequest(&porterv1.GetAppRevisionRequest{
  86. ProjectId: int64(inp.ProjectID),
  87. AppRevisionId: inp.AppRevisionID.String(),
  88. })
  89. ccpResp, err := inp.CCPClient.GetAppRevision(ctx, getRevisionReq)
  90. if err != nil {
  91. return revision, telemetry.Error(ctx, span, err, "error getting app revision")
  92. }
  93. if ccpResp == nil || ccpResp.Msg == nil {
  94. return revision, telemetry.Error(ctx, span, nil, "get app revision response is nil")
  95. }
  96. appRevisionProto := ccpResp.Msg.AppRevision
  97. revision, err = EncodedRevisionFromProto(ctx, appRevisionProto)
  98. if err != nil {
  99. return revision, telemetry.Error(ctx, span, err, "error converting app revision from proto")
  100. }
  101. return revision, nil
  102. }
  103. // EncodedRevisionFromProto converts an AppRevision proto object into a Revision object
  104. func EncodedRevisionFromProto(ctx context.Context, appRevision *porterv1.AppRevision) (Revision, error) {
  105. ctx, span := telemetry.NewSpan(ctx, "encoded-revision-from-proto")
  106. defer span.End()
  107. var revision Revision
  108. if appRevision == nil {
  109. return revision, telemetry.Error(ctx, span, nil, "current app revision definition is nil")
  110. }
  111. appProto := appRevision.App
  112. if appProto == nil {
  113. return revision, telemetry.Error(ctx, span, nil, "app proto is nil")
  114. }
  115. encoded, err := helpers.MarshalContractObject(ctx, appProto)
  116. if err != nil {
  117. return revision, telemetry.Error(ctx, span, err, "error marshalling app proto back to json")
  118. }
  119. b64 := base64.StdEncoding.EncodeToString(encoded)
  120. status, err := appRevisionStatusFromProto(appRevision.Status)
  121. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "status", Value: string(status)})
  122. if err != nil {
  123. _ = telemetry.Error(ctx, span, nil, "unknown revision type") // flagged as an error for visibility
  124. }
  125. appInstanceIdStr := appRevision.AppInstanceId
  126. appInstanceId, err := uuid.Parse(appInstanceIdStr)
  127. if err != nil {
  128. return revision, telemetry.Error(ctx, span, err, "error parsing app instance id")
  129. }
  130. revision = Revision{
  131. B64AppProto: b64,
  132. Status: status,
  133. ID: appRevision.Id,
  134. RevisionNumber: appRevision.RevisionNumber,
  135. CreatedAt: appRevision.CreatedAt.AsTime(),
  136. UpdatedAt: appRevision.UpdatedAt.AsTime(),
  137. DeploymentTarget: DeploymentTarget{ID: appRevision.DeploymentTargetId},
  138. AppInstanceID: appInstanceId,
  139. }
  140. return revision, nil
  141. }
  142. // AttachEnvToRevisionInput is the input struct for AttachEnvToRevision
  143. type AttachEnvToRevisionInput struct {
  144. ProjectID uint
  145. ClusterID int
  146. Revision Revision
  147. DeploymentTarget deployment_target.DeploymentTarget
  148. K8SAgent *kubernetes.Agent
  149. PorterAppRepository repository.PorterAppRepository
  150. }
  151. // AttachEnvToRevision attaches the environment variables from the app's default env group to a revision
  152. // These are the variables that are displayed to the user in the UI as associated with the app rather than an env group
  153. func AttachEnvToRevision(ctx context.Context, inp AttachEnvToRevisionInput) (Revision, error) {
  154. ctx, span := telemetry.NewSpan(ctx, "attach-env-to-revision")
  155. defer span.End()
  156. revision := inp.Revision
  157. if inp.ProjectID == 0 {
  158. return revision, telemetry.Error(ctx, span, nil, "must provide a project id")
  159. }
  160. if inp.ClusterID == 0 {
  161. return revision, telemetry.Error(ctx, span, nil, "must provide a cluster id")
  162. }
  163. if inp.K8SAgent == nil {
  164. return revision, telemetry.Error(ctx, span, nil, "k8s agent is nil")
  165. }
  166. decoded, err := base64.StdEncoding.DecodeString(revision.B64AppProto)
  167. if err != nil {
  168. return revision, telemetry.Error(ctx, span, err, "error decoding app proto")
  169. }
  170. appDef := &porterv1.PorterApp{}
  171. err = helpers.UnmarshalContractObject(decoded, appDef)
  172. if err != nil {
  173. return revision, telemetry.Error(ctx, span, err, "error unmarshalling app proto")
  174. }
  175. envName, err := AppEnvGroupName(ctx, appDef.Name, inp.Revision.DeploymentTarget.ID, uint(inp.ClusterID), inp.PorterAppRepository)
  176. if err != nil {
  177. return revision, telemetry.Error(ctx, span, err, "error getting app env group name")
  178. }
  179. envNameFilter := []string{envName}
  180. envFromProtoInp := AppEnvironmentFromProtoInput{
  181. ProjectID: inp.ProjectID,
  182. ClusterID: inp.ClusterID,
  183. App: appDef,
  184. K8SAgent: inp.K8SAgent,
  185. DeploymentTarget: inp.DeploymentTarget,
  186. }
  187. envGroups, err := AppEnvironmentFromProto(ctx, envFromProtoInp, WithEnvGroupFilter(envNameFilter), WithSecrets())
  188. if err != nil {
  189. return revision, telemetry.Error(ctx, span, err, "error getting app environment from revision")
  190. }
  191. if len(envGroups) > 1 {
  192. return revision, telemetry.Error(ctx, span, err, "multiple app envs groups returned for same name")
  193. }
  194. if len(envGroups) == 1 {
  195. revision.Env = envGroups[0]
  196. }
  197. return revision, nil
  198. }
  199. func appRevisionStatusFromProto(status string) (models.AppRevisionStatus, error) {
  200. appRevisionStatus := models.AppRevisionStatus_Unknown
  201. switch status {
  202. case string(models.AppRevisionStatus_ImageAvailable):
  203. appRevisionStatus = models.AppRevisionStatus_ImageAvailable
  204. case string(models.AppRevisionStatus_AwaitingBuild):
  205. appRevisionStatus = models.AppRevisionStatus_AwaitingBuild
  206. case string(models.AppRevisionStatus_AwaitingPredeploy):
  207. appRevisionStatus = models.AppRevisionStatus_AwaitingPredeploy
  208. case string(models.AppRevisionStatus_InstallSuccessful):
  209. appRevisionStatus = models.AppRevisionStatus_InstallSuccessful
  210. case string(models.AppRevisionStatus_InstallProgressing):
  211. appRevisionStatus = models.AppRevisionStatus_InstallProgressing
  212. case string(models.AppRevisionStatus_AwaitingInstall):
  213. appRevisionStatus = models.AppRevisionStatus_AwaitingInstall
  214. case string(models.AppRevisionStatus_BuildCanceled):
  215. appRevisionStatus = models.AppRevisionStatus_BuildCanceled
  216. case string(models.AppRevisionStatus_BuildFailed):
  217. appRevisionStatus = models.AppRevisionStatus_BuildFailed
  218. case string(models.AppRevisionStatus_PredeployFailed):
  219. appRevisionStatus = models.AppRevisionStatus_PredeployFailed
  220. case string(models.AppRevisionStatus_PredeploySuccessful):
  221. appRevisionStatus = models.AppRevisionStatus_PredeploySuccessful
  222. case string(models.AppRevisionStatus_PredeployProgressing):
  223. appRevisionStatus = models.AppRevisionStatus_PredeployProgressing
  224. case string(models.AppRevisionStatus_InstallFailed):
  225. appRevisionStatus = models.AppRevisionStatus_InstallFailed
  226. case string(models.AppRevisionStatus_Created):
  227. appRevisionStatus = models.AppRevisionStatus_Created
  228. case string(models.AppRevisionStatus_BuildSuccessful):
  229. appRevisionStatus = models.AppRevisionStatus_BuildSuccessful
  230. case string(models.AppRevisionStatus_ApplyFailed):
  231. appRevisionStatus = models.AppRevisionStatus_ApplyFailed
  232. case string(models.AppRevisionStatus_UpdateFailed):
  233. appRevisionStatus = models.AppRevisionStatus_UpdateFailed
  234. case string(models.AppRevisionStatus_DeploymentProgressing):
  235. appRevisionStatus = models.AppRevisionStatus_DeploymentProgressing
  236. case string(models.AppRevisionStatus_DeploymentSuccessful):
  237. appRevisionStatus = models.AppRevisionStatus_DeploymentSuccessful
  238. case string(models.AppRevisionStatus_DeploymentFailed):
  239. appRevisionStatus = models.AppRevisionStatus_DeploymentFailed
  240. default:
  241. return appRevisionStatus, errors.New("unknown app revision status")
  242. }
  243. return appRevisionStatus, nil
  244. }