get_accounts.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. package gitinstallation
  2. import (
  3. "context"
  4. "net/http"
  5. "sort"
  6. "time"
  7. "github.com/porter-dev/porter/internal/telemetry"
  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/types"
  15. "golang.org/x/oauth2"
  16. "gorm.io/gorm"
  17. )
  18. type GetGithubAppAccountsHandler struct {
  19. handlers.PorterHandlerReadWriter
  20. authz.KubernetesAgentGetter
  21. }
  22. func NewGetGithubAppAccountsHandler(
  23. config *config.Config,
  24. decoderValidator shared.RequestDecoderValidator,
  25. writer shared.ResultWriter,
  26. ) *GetGithubAppAccountsHandler {
  27. return &GetGithubAppAccountsHandler{
  28. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  29. }
  30. }
  31. func (c *GetGithubAppAccountsHandler) getOrgList(ctx context.Context,
  32. client *github.Client,
  33. orgsChan chan<- *github.Organization,
  34. errChan chan<- error,
  35. ) {
  36. defer close(orgsChan)
  37. defer close(errChan)
  38. page := 1
  39. for {
  40. select {
  41. case <-ctx.Done():
  42. return
  43. default:
  44. orgs, pages, err := client.Organizations.List(context.Background(), "", &github.ListOptions{
  45. PerPage: 100,
  46. Page: page,
  47. })
  48. if err != nil {
  49. errChan <- err
  50. return
  51. }
  52. for _, org := range orgs {
  53. orgsChan <- org
  54. }
  55. if pages.NextPage == 0 {
  56. return
  57. } else {
  58. page = pages.NextPage
  59. }
  60. }
  61. }
  62. }
  63. func (c *GetGithubAppAccountsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  64. ctx, span := telemetry.NewSpan(r.Context(), "serve-get-github-app-accounts")
  65. defer span.End()
  66. r = r.Clone(ctx)
  67. tok, err := GetGithubAppOauthTokenFromRequest(c.Config(), r)
  68. if err != nil {
  69. err = telemetry.Error(ctx, span, err, "error getting github app oauth token from request")
  70. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusForbidden))
  71. return
  72. }
  73. client := github.NewClient(c.Config().GithubAppConf.Client(oauth2.NoContext, tok))
  74. res := &types.GetGithubAppAccountsResponse{}
  75. resultChannel := make(chan *github.Organization, 10)
  76. errChan := make(chan error)
  77. ctx, cancel := context.WithTimeout(ctx, 3*time.Second)
  78. defer cancel()
  79. go c.getOrgList(ctx, client, resultChannel, errChan)
  80. resultOrErrorReader:
  81. for {
  82. select {
  83. case result, ok := <-resultChannel:
  84. if ok {
  85. res.Accounts = append(res.Accounts, *result.Login)
  86. } else {
  87. // channel has been closed now
  88. break resultOrErrorReader
  89. }
  90. case err, ok := <-errChan:
  91. if ok {
  92. err = telemetry.Error(ctx, span, err, "error getting org list")
  93. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  94. return
  95. } else {
  96. // nothing in error, must be a close event
  97. break resultOrErrorReader
  98. }
  99. }
  100. }
  101. authUser, _, err := client.Users.Get(r.Context(), "")
  102. if err != nil {
  103. err = telemetry.Error(ctx, span, err, "error getting authenticated user")
  104. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  105. return
  106. }
  107. res.Username = *authUser.Login
  108. // check if user has app installed in their account
  109. installation, err := c.Repo().GithubAppInstallation().ReadGithubAppInstallationByAccountID(*authUser.ID)
  110. if err != nil && err != gorm.ErrRecordNotFound {
  111. err = telemetry.Error(ctx, span, err, "error reading github app installation")
  112. c.HandleAPIError(w, r, apierrors.NewErrPassThroughToClient(err, http.StatusInternalServerError))
  113. return
  114. }
  115. if installation != nil {
  116. res.Accounts = append(res.Accounts, *authUser.Login)
  117. }
  118. sort.Strings(res.Accounts)
  119. c.WriteResult(w, r, res)
  120. }