latest_app_revisions.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. package porter_app
  2. import (
  3. "net/http"
  4. "connectrpc.com/connect"
  5. porterv1 "github.com/porter-dev/api-contracts/generated/go/porter/v1"
  6. "github.com/porter-dev/porter/api/server/handlers"
  7. "github.com/porter-dev/porter/api/server/shared"
  8. "github.com/porter-dev/porter/api/server/shared/apierrors"
  9. "github.com/porter-dev/porter/api/server/shared/config"
  10. "github.com/porter-dev/porter/api/types"
  11. "github.com/porter-dev/porter/internal/models"
  12. "github.com/porter-dev/porter/internal/porter_app"
  13. "github.com/porter-dev/porter/internal/telemetry"
  14. )
  15. // LatestAppRevisionsHandler handles requests to the /apps/{porter_app_name}/revisions endpoint
  16. type LatestAppRevisionsHandler struct {
  17. handlers.PorterHandlerReadWriter
  18. }
  19. // NewLatestAppRevisionsHandler returns a new LatestAppRevisionsHandler
  20. func NewLatestAppRevisionsHandler(
  21. config *config.Config,
  22. decoderValidator shared.RequestDecoderValidator,
  23. writer shared.ResultWriter,
  24. ) *LatestAppRevisionsHandler {
  25. return &LatestAppRevisionsHandler{
  26. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  27. }
  28. }
  29. // LatestAppRevisionsRequest represents the response from the /apps/{porter_app_name}/revisions endpoint
  30. type LatestAppRevisionsRequest struct{}
  31. // LatestRevisionWithSource is an app revision and its source porter app
  32. type LatestRevisionWithSource struct {
  33. AppRevision porter_app.Revision `json:"app_revision"`
  34. Source types.PorterApp `json:"source"`
  35. }
  36. // LatestAppRevisionsResponse represents the response from the /apps/{porter_app_name}/revisions endpoint
  37. type LatestAppRevisionsResponse struct {
  38. AppRevisions []LatestRevisionWithSource `json:"app_revisions"`
  39. }
  40. func (c *LatestAppRevisionsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  41. ctx, span := telemetry.NewSpan(r.Context(), "serve-list-app-revisions")
  42. defer span.End()
  43. project, _ := r.Context().Value(types.ProjectScope).(*models.Project)
  44. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  45. deploymentTargets, err := c.Repo().DeploymentTarget().List(project.ID)
  46. if err != nil {
  47. err = telemetry.Error(ctx, span, err, "error reading deployment targets")
  48. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  49. return
  50. }
  51. if len(deploymentTargets) == 0 {
  52. res := &LatestAppRevisionsResponse{
  53. AppRevisions: []LatestRevisionWithSource{},
  54. }
  55. c.WriteResult(w, r, res)
  56. return
  57. }
  58. if len(deploymentTargets) > 1 {
  59. err = telemetry.Error(ctx, span, err, "more than one deployment target found")
  60. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  61. return
  62. }
  63. // todo(ianedwards): once we have a way to select a deployment target, we can add it to the request
  64. deploymentTarget := deploymentTargets[0]
  65. listAppRevisionsReq := connect.NewRequest(&porterv1.LatestAppRevisionsRequest{
  66. ProjectId: int64(project.ID),
  67. DeploymentTargetId: deploymentTarget.ID.String(),
  68. })
  69. latestAppRevisionsResp, err := c.Config().ClusterControlPlaneClient.LatestAppRevisions(ctx, listAppRevisionsReq)
  70. if err != nil {
  71. err = telemetry.Error(ctx, span, err, "error getting latest app revisions")
  72. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  73. return
  74. }
  75. if latestAppRevisionsResp == nil || latestAppRevisionsResp.Msg == nil {
  76. err = telemetry.Error(ctx, span, nil, "latest app revisions response is nil")
  77. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  78. return
  79. }
  80. appRevisions := latestAppRevisionsResp.Msg.AppRevisions
  81. if appRevisions == nil {
  82. appRevisions = []*porterv1.AppRevision{}
  83. }
  84. res := &LatestAppRevisionsResponse{
  85. AppRevisions: make([]LatestRevisionWithSource, 0),
  86. }
  87. for _, revision := range appRevisions {
  88. encodedRevision, err := porter_app.EncodedRevisionFromProto(ctx, revision)
  89. if err != nil {
  90. err := telemetry.Error(ctx, span, err, "error getting encoded revision from proto")
  91. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  92. return
  93. }
  94. porterApp, err := c.Repo().PorterApp().ReadPorterAppByName(cluster.ID, revision.App.Name)
  95. if err != nil {
  96. err := telemetry.Error(ctx, span, err, "error reading porter app")
  97. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  98. return
  99. }
  100. if porterApp == nil {
  101. err := telemetry.Error(ctx, span, err, "porter app is nil")
  102. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  103. return
  104. }
  105. res.AppRevisions = append(res.AppRevisions, LatestRevisionWithSource{
  106. AppRevision: encodedRevision,
  107. Source: *porterApp.ToPorterAppType(),
  108. })
  109. }
  110. c.WriteResult(w, r, res)
  111. }