validate_porter_yaml.go 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. package environment
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "net/http"
  7. "strings"
  8. "github.com/google/go-github/v41/github"
  9. "github.com/porter-dev/porter/api/server/authz"
  10. "github.com/porter-dev/porter/api/server/handlers"
  11. "github.com/porter-dev/porter/api/server/shared"
  12. "github.com/porter-dev/porter/api/server/shared/apierrors"
  13. "github.com/porter-dev/porter/api/server/shared/config"
  14. "github.com/porter-dev/porter/api/server/shared/requestutils"
  15. "github.com/porter-dev/porter/api/types"
  16. "github.com/porter-dev/porter/internal/integrations/preview"
  17. "github.com/porter-dev/porter/internal/models"
  18. "gorm.io/gorm"
  19. )
  20. type ValidatePorterYAMLHandler struct {
  21. handlers.PorterHandlerReadWriter
  22. authz.KubernetesAgentGetter
  23. }
  24. func NewValidatePorterYAMLHandler(
  25. config *config.Config,
  26. decoderValidator shared.RequestDecoderValidator,
  27. writer shared.ResultWriter,
  28. ) *ValidatePorterYAMLHandler {
  29. return &ValidatePorterYAMLHandler{
  30. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  31. KubernetesAgentGetter: authz.NewOutOfClusterAgentGetter(config),
  32. }
  33. }
  34. func (c *ValidatePorterYAMLHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  35. project, _ := r.Context().Value(types.ProjectScope).(*models.Project)
  36. cluster, _ := r.Context().Value(types.ClusterScope).(*models.Cluster)
  37. envID, reqErr := requestutils.GetURLParamUint(r, "environment_id")
  38. if reqErr != nil {
  39. c.HandleAPIError(w, r, reqErr)
  40. return
  41. }
  42. req := &types.ValidatePorterYAMLRequest{}
  43. if ok := c.DecodeAndValidate(w, r, req); !ok {
  44. return
  45. }
  46. env, err := c.Repo().Environment().ReadEnvironmentByID(project.ID, cluster.ID, envID)
  47. if err != nil {
  48. if errors.Is(err, gorm.ErrRecordNotFound) {
  49. c.HandleAPIError(w, r, apierrors.NewErrNotFound(fmt.Errorf("no such environment with ID: %d", envID)))
  50. return
  51. }
  52. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("error reading environment with ID: %d. Error: %w", envID, err)))
  53. return
  54. }
  55. ghClient, err := getGithubClientFromEnvironment(c.Config(), env)
  56. if err != nil {
  57. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  58. return
  59. }
  60. res := &types.ValidatePorterYAMLResponse{
  61. Errors: []string{},
  62. }
  63. if req.Branch == "" { // get the default branch name
  64. repo, _, err := ghClient.Repositories.Get(r.Context(), env.GitRepoOwner, env.GitRepoName)
  65. if err != nil {
  66. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  67. return
  68. }
  69. req.Branch = repo.GetDefaultBranch()
  70. }
  71. fileContents, _, ghResp, err := ghClient.Repositories.GetContents(
  72. context.Background(), env.GitRepoOwner, env.GitRepoName, "porter.yaml",
  73. &github.RepositoryContentGetOptions{
  74. Ref: req.Branch,
  75. },
  76. )
  77. if ghResp.StatusCode == 404 {
  78. res.Errors = append(res.Errors, preview.ErrNoPorterYAMLFile.Error())
  79. c.WriteResult(w, r, res)
  80. return
  81. }
  82. if err != nil {
  83. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  84. return
  85. }
  86. contents, err := fileContents.GetContent()
  87. if err != nil {
  88. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  89. return
  90. }
  91. if strings.TrimSpace(contents) == "" {
  92. res.Errors = append(res.Errors, preview.ErrEmptyPorterYAMLFile.Error())
  93. c.WriteResult(w, r, res)
  94. return
  95. }
  96. for _, err := range preview.Validate(contents) {
  97. if err != nil {
  98. res.Errors = append(res.Errors, err.Error())
  99. }
  100. }
  101. c.WriteResult(w, r, res)
  102. }