rerun_workflow.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package gitinstallation
  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/handlers"
  10. "github.com/porter-dev/porter/api/server/shared"
  11. "github.com/porter-dev/porter/api/server/shared/apierrors"
  12. "github.com/porter-dev/porter/api/server/shared/config"
  13. )
  14. var ErrNoWorkflowRuns = errors.New("no previous workflow runs found")
  15. type RerunWorkflowHandler struct {
  16. handlers.PorterHandlerReadWriter
  17. }
  18. func NewRerunWorkflowHandler(
  19. config *config.Config,
  20. decoderValidator shared.RequestDecoderValidator,
  21. writer shared.ResultWriter,
  22. ) *RerunWorkflowHandler {
  23. return &RerunWorkflowHandler{
  24. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  25. }
  26. }
  27. func (c *RerunWorkflowHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  28. owner, name, ok := GetOwnerAndNameParams(c, w, r)
  29. if !ok {
  30. return
  31. }
  32. filename := r.URL.Query().Get("filename")
  33. release_name := r.URL.Query().Get("release_name")
  34. if filename == "" && release_name == "" {
  35. c.HandleAPIError(w, r, apierrors.NewErrInternal(fmt.Errorf("filename and release name are both empty")))
  36. return
  37. }
  38. if filename == "" {
  39. if c.Config().ServerConf.InstanceName != "" {
  40. filename = fmt.Sprintf("porter_%s_%s.yml", strings.Replace(
  41. strings.ToLower(release_name), "-", "_", -1),
  42. strings.ToLower(c.Config().ServerConf.InstanceName),
  43. )
  44. } else {
  45. filename = fmt.Sprintf("porter_%s.yml", strings.Replace(
  46. strings.ToLower(release_name), "-", "_", -1),
  47. )
  48. }
  49. }
  50. client, err := GetGithubAppClientFromRequest(c.Config(), r)
  51. if err != nil {
  52. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  53. return
  54. }
  55. latestWorkflowRun, err := getLatestWorkflowRun(client, owner, name, filename)
  56. if err != nil && errors.Is(err, ErrNoWorkflowRuns) {
  57. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, 400))
  58. return
  59. } else if err != nil {
  60. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  61. return
  62. }
  63. if latestWorkflowRun.GetStatus() == "in_progress" || latestWorkflowRun.GetStatus() == "queued" {
  64. w.WriteHeader(409)
  65. c.WriteResult(w, r, latestWorkflowRun.GetHTMLURL())
  66. return
  67. }
  68. _, err = client.Actions.RerunWorkflowByID(r.Context(), owner, name, latestWorkflowRun.GetID())
  69. if err != nil {
  70. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  71. return
  72. }
  73. latestWorkflowRun, err = getLatestWorkflowRun(client, owner, name, filename)
  74. if err != nil {
  75. c.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  76. return
  77. }
  78. c.WriteResult(w, r, latestWorkflowRun.GetHTMLURL())
  79. }
  80. func getLatestWorkflowRun(client *github.Client, owner, repo, filename string) (*github.WorkflowRun, error) {
  81. workflowRuns, _, err := client.Actions.ListWorkflowRunsByFileName(
  82. context.Background(), owner, repo, filename, &github.ListWorkflowRunsOptions{
  83. ListOptions: github.ListOptions{
  84. Page: 1,
  85. PerPage: 1,
  86. },
  87. },
  88. )
  89. if err != nil {
  90. return nil, err
  91. }
  92. if workflowRuns.GetTotalCount() == 0 {
  93. return nil, ErrNoWorkflowRuns
  94. }
  95. return workflowRuns.WorkflowRuns[0], nil
  96. }