analytics.go 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. package porter_app
  2. import (
  3. "context"
  4. "net/http"
  5. "github.com/porter-dev/porter/api/server/handlers"
  6. "github.com/porter-dev/porter/api/server/shared"
  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/telemetry"
  12. )
  13. type PorterAppAnalyticsHandler struct {
  14. handlers.PorterHandlerReadWriter
  15. }
  16. func NewPorterAppAnalyticsHandler(
  17. config *config.Config,
  18. decoderValidator shared.RequestDecoderValidator,
  19. writer shared.ResultWriter,
  20. ) *PorterAppAnalyticsHandler {
  21. return &PorterAppAnalyticsHandler{
  22. PorterHandlerReadWriter: handlers.NewDefaultPorterHandler(config, decoderValidator, writer),
  23. }
  24. }
  25. func (v *PorterAppAnalyticsHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  26. ctx := r.Context()
  27. user, _ := ctx.Value(types.UserScope).(*models.User)
  28. project, _ := ctx.Value(types.ProjectScope).(*models.Project)
  29. request := &types.PorterAppAnalyticsRequest{}
  30. if ok := v.DecodeAndValidate(w, r, request); !ok {
  31. return
  32. }
  33. validateApplyV2 := project.GetFeatureFlag(models.ValidateApplyV2, v.Config().LaunchDarklyClient)
  34. if request.Step == "stack-launch-start" {
  35. v.Config().AnalyticsClient.Track(analytics.StackLaunchStartTrack(&analytics.StackLaunchStartOpts{
  36. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  37. Email: user.Email,
  38. FirstName: user.FirstName,
  39. LastName: user.LastName,
  40. CompanyName: user.CompanyName,
  41. ValidateApplyV2: validateApplyV2,
  42. }))
  43. }
  44. if request.Step == "stack-launch-complete" {
  45. v.Config().AnalyticsClient.Track(analytics.StackLaunchCompleteTrack(&analytics.StackLaunchCompleteOpts{
  46. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  47. StackName: request.StackName,
  48. Email: user.Email,
  49. FirstName: user.FirstName,
  50. LastName: user.LastName,
  51. CompanyName: user.CompanyName,
  52. ValidateApplyV2: validateApplyV2,
  53. }))
  54. }
  55. if request.Step == "stack-launch-success" {
  56. v.Config().AnalyticsClient.Track(analytics.StackLaunchSuccessTrack(&analytics.StackLaunchSuccessOpts{
  57. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  58. StackName: request.StackName,
  59. Email: user.Email,
  60. FirstName: user.FirstName,
  61. LastName: user.LastName,
  62. CompanyName: user.CompanyName,
  63. ValidateApplyV2: validateApplyV2,
  64. }))
  65. }
  66. if request.Step == "stack-launch-failure" {
  67. v.Config().AnalyticsClient.Track(analytics.StackLaunchFailureTrack(&analytics.StackLaunchFailureOpts{
  68. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  69. StackName: request.StackName,
  70. Email: user.Email,
  71. FirstName: user.FirstName,
  72. LastName: user.LastName,
  73. CompanyName: user.CompanyName,
  74. ErrorMessage: request.ErrorMessage,
  75. ValidateApplyV2: validateApplyV2,
  76. }))
  77. }
  78. if request.Step == "stack-deletion" {
  79. v.Config().AnalyticsClient.Track(analytics.StackDeletionTrack(&analytics.StackDeletionOpts{
  80. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  81. StackName: request.StackName,
  82. Email: user.Email,
  83. FirstName: user.FirstName,
  84. LastName: user.LastName,
  85. CompanyName: user.CompanyName,
  86. DeleteWorkflowFile: request.DeleteWorkflowFile,
  87. ValidateApplyV2: validateApplyV2,
  88. }))
  89. }
  90. v.WriteResult(w, r, user.ToUserType())
  91. }
  92. func TrackStackBuildStatus(
  93. ctx context.Context,
  94. config *config.Config,
  95. user *models.User,
  96. project *models.Project,
  97. stackName string,
  98. errorMessage string,
  99. status types.PorterAppEventStatus,
  100. ) error {
  101. _, span := telemetry.NewSpan(ctx, "track-build-status")
  102. defer span.End()
  103. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-build-status", Value: string(status)})
  104. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-name", Value: stackName})
  105. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-error-message", Value: errorMessage})
  106. if status == types.PorterAppEventStatus_Progressing {
  107. return config.AnalyticsClient.Track(analytics.StackBuildProgressingTrack(&analytics.StackBuildOpts{
  108. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  109. StackName: stackName,
  110. Email: user.Email,
  111. FirstName: user.FirstName,
  112. LastName: user.LastName,
  113. CompanyName: user.CompanyName,
  114. ValidateApplyV2: project.ValidateApplyV2,
  115. }))
  116. }
  117. if status == types.PorterAppEventStatus_Success {
  118. return config.AnalyticsClient.Track(analytics.StackBuildSuccessTrack(&analytics.StackBuildOpts{
  119. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  120. StackName: stackName,
  121. Email: user.Email,
  122. FirstName: user.FirstName,
  123. LastName: user.LastName,
  124. CompanyName: user.CompanyName,
  125. ValidateApplyV2: project.ValidateApplyV2,
  126. }))
  127. }
  128. if status == types.PorterAppEventStatus_Failed {
  129. return config.AnalyticsClient.Track(analytics.StackBuildFailureTrack(&analytics.StackBuildOpts{
  130. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  131. StackName: stackName,
  132. ErrorMessage: errorMessage,
  133. Email: user.Email,
  134. FirstName: user.FirstName,
  135. LastName: user.LastName,
  136. CompanyName: user.CompanyName,
  137. ValidateApplyV2: project.ValidateApplyV2,
  138. }))
  139. }
  140. return nil
  141. }