create.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. package project
  2. import (
  3. "net/http"
  4. "github.com/porter-dev/porter/api/server/handlers"
  5. "github.com/porter-dev/porter/api/server/shared"
  6. "github.com/porter-dev/porter/api/server/shared/apierrors"
  7. "github.com/porter-dev/porter/api/server/shared/config"
  8. "github.com/porter-dev/porter/api/types"
  9. "github.com/porter-dev/porter/internal/analytics"
  10. "github.com/porter-dev/porter/internal/models"
  11. "github.com/porter-dev/porter/internal/repository"
  12. )
  13. type ProjectCreateHandler struct {
  14. handlers.PorterHandlerReadWriter
  15. }
  16. func NewProjectCreateHandler(
  17. config *config.Config,
  18. decoderValidator shared.RequestDecoderValidator,
  19. writer shared.ResultWriter,
  20. ) *ProjectCreateHandler {
  21. return &ProjectCreateHandler{
  22. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  23. }
  24. }
  25. func (p *ProjectCreateHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  26. request := &types.CreateProjectRequest{}
  27. ok := p.DecodeAndValidate(w, r, request)
  28. if !ok {
  29. return
  30. }
  31. // read the user from context
  32. user, _ := r.Context().Value(types.UserScope).(*models.User)
  33. proj := &models.Project{
  34. Name: request.Name,
  35. }
  36. var err error
  37. proj, role, err := CreateProjectWithUser(p.Repo().Project(), proj, user)
  38. if err != nil {
  39. p.HandleAPIError(w, r, apierrors.NewErrInternal(err))
  40. return
  41. }
  42. p.WriteResult(w, r, proj.ToProjectType())
  43. // add project to billing team
  44. teamID, err := p.Config().BillingManager.CreateTeam(proj)
  45. if err != nil {
  46. // we do not write error response, since setting up billing error can be
  47. // resolved later and may not be fatal
  48. p.HandleAPIErrorNoWrite(w, r, apierrors.NewErrInternal(err))
  49. }
  50. if teamID != "" {
  51. err = p.Config().BillingManager.AddUserToTeam(teamID, user, role)
  52. if err != nil {
  53. // we do not write error response, since setting up billing error can be
  54. // resolved later and may not be fatal
  55. p.HandleAPIErrorNoWrite(w, r, apierrors.NewErrInternal(err))
  56. }
  57. }
  58. p.Config().AnalyticsClient.Track(analytics.ProjectCreateTrack(&analytics.ProjectCreateTrackOpts{
  59. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, proj.ID),
  60. }))
  61. }
  62. func CreateProjectWithUser(
  63. projectRepo repository.ProjectRepository,
  64. proj *models.Project,
  65. user *models.User,
  66. ) (*models.Project, *models.Role, error) {
  67. proj, err := projectRepo.CreateProject(proj)
  68. if err != nil {
  69. return nil, nil, err
  70. }
  71. // create a new Role with the user as the admin
  72. role, err := projectRepo.CreateProjectRole(proj, &models.Role{
  73. Role: types.Role{
  74. UserID: user.ID,
  75. ProjectID: proj.ID,
  76. Kind: types.RoleAdmin,
  77. },
  78. })
  79. if err != nil {
  80. return nil, nil, err
  81. }
  82. // read the project again to get the model with the role attached
  83. proj, err = projectRepo.ReadProject(proj.ID)
  84. if err != nil {
  85. return nil, nil, err
  86. }
  87. return proj, role, nil
  88. }