get_accounts.go 3.1 KB

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