analytics.go 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  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. if request.Step == "porter-app-update-failure" {
  91. v.Config().AnalyticsClient.Track(analytics.PorterAppUpdateFailureTrack(&analytics.PorterAppUpdateOpts{
  92. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  93. StackName: request.StackName,
  94. Email: user.Email,
  95. FirstName: user.FirstName,
  96. LastName: user.LastName,
  97. CompanyName: user.CompanyName,
  98. ErrorMessage: request.ErrorMessage,
  99. ErrorStackTrace: request.ErrorStackTrace,
  100. ValidateApplyV2: validateApplyV2,
  101. }))
  102. }
  103. v.WriteResult(w, r, user.ToUserType())
  104. }
  105. func TrackStackBuildStatus(
  106. ctx context.Context,
  107. config *config.Config,
  108. user *models.User,
  109. project *models.Project,
  110. stackName string,
  111. errorMessage string,
  112. status types.PorterAppEventStatus,
  113. validateApplyV2 bool,
  114. ) error {
  115. _, span := telemetry.NewSpan(ctx, "track-build-status")
  116. defer span.End()
  117. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-build-status", Value: string(status)})
  118. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-name", Value: stackName})
  119. telemetry.WithAttributes(span, telemetry.AttributeKV{Key: "porter-app-error-message", Value: errorMessage})
  120. if status == types.PorterAppEventStatus_Progressing {
  121. return config.AnalyticsClient.Track(analytics.StackBuildProgressingTrack(&analytics.StackBuildOpts{
  122. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  123. StackName: stackName,
  124. Email: user.Email,
  125. FirstName: user.FirstName,
  126. LastName: user.LastName,
  127. CompanyName: user.CompanyName,
  128. ValidateApplyV2: validateApplyV2,
  129. }))
  130. }
  131. if status == types.PorterAppEventStatus_Success {
  132. return config.AnalyticsClient.Track(analytics.StackBuildSuccessTrack(&analytics.StackBuildOpts{
  133. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  134. StackName: stackName,
  135. Email: user.Email,
  136. FirstName: user.FirstName,
  137. LastName: user.LastName,
  138. CompanyName: user.CompanyName,
  139. ValidateApplyV2: validateApplyV2,
  140. }))
  141. }
  142. if status == types.PorterAppEventStatus_Failed {
  143. return config.AnalyticsClient.Track(analytics.StackBuildFailureTrack(&analytics.StackBuildOpts{
  144. ProjectScopedTrackOpts: analytics.GetProjectScopedTrackOpts(user.ID, project.ID),
  145. StackName: stackName,
  146. ErrorMessage: errorMessage,
  147. Email: user.Email,
  148. FirstName: user.FirstName,
  149. LastName: user.LastName,
  150. CompanyName: user.CompanyName,
  151. ValidateApplyV2: validateApplyV2,
  152. }))
  153. }
  154. return nil
  155. }