main

mattermost/focalboard

Last updated at: 29/12/2023 09:46

telemetry_test.go

TLDR

This file contains a Go test file for the telemetry package. It includes a test function, TestTelemetry, which tests the telemetry functionality by registering a tracker and running the telemetry job. The file also includes a helper function, mockServer, which sets up a mock server for testing telemetry.

Methods

None

Classes

None

package telemetry

import (
	"bytes"
	"encoding/json"
	"io"
	"net/http"
	"net/http/httptest"
	"os"
	"strings"
	"testing"
	"time"

	"github.com/mattermost/mattermost-server/v6/shared/mlog"
	"github.com/stretchr/testify/require"
)

func mockServer() (chan []byte, *httptest.Server) {
	done := make(chan []byte, 1)

	server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
		buf := bytes.NewBuffer(nil)
		if _, err := io.Copy(buf, r.Body); err != nil {
			panic(err)
		}

		var v interface{}
		err := json.Unmarshal(buf.Bytes(), &v)
		if err != nil {
			panic(err)
		}

		b, err := json.MarshalIndent(v, "", "  ")
		if err != nil {
			panic(err)
		}

		// filter the identify message
		if strings.Contains(string(b), `"type": "identify"`) {
			return
		}

		done <- b
	}))

	return done, server
}

func TestTelemetry(t *testing.T) {
	receiveChan, server := mockServer()

	os.Setenv("RUDDER_KEY", "mock-test-rudder-key")
	os.Setenv("RUDDER_DATAPLANE_URL", server.URL)

	checkMockRudderServer := func(t *testing.T) {
		// check mock rudder server got
		got := string(<-receiveChan)
		require.Contains(t, got, "mockTrackerKey")
		require.Contains(t, got, "mockTrackerValue")
	}

	t.Run("Register tracker and run telemetry job", func(t *testing.T) {
		service := New("mockTelemetryID", mlog.CreateConsoleTestLogger(false, mlog.LvlDebug))
		service.RegisterTracker("mockTracker", func() (Tracker, error) {
			return map[string]interface{}{
				"mockTrackerKey": "mockTrackerValue",
			}, nil
		})

		service.RunTelemetryJob(time.Now().UnixNano() / int64(time.Millisecond))
		checkMockRudderServer(t)
	})

	t.Run("do telemetry if needed", func(t *testing.T) {
		service := New("mockTelemetryID", mlog.CreateConsoleTestLogger(false, mlog.LvlDebug))
		service.RegisterTracker("mockTracker", func() (Tracker, error) {
			return map[string]interface{}{
				"mockTrackerKey": "mockTrackerValue",
			}, nil
		})

		firstRun := time.Now()
		t.Run("Send once every 10 minutes for the first hour", func(t *testing.T) {
			service.doTelemetryIfNeeded(firstRun.Add(-30 * time.Minute))
			checkMockRudderServer(t)
		})

		t.Run("Send once every hour thereafter for the first 12 hours", func(t *testing.T) {
			// firstRun is 2 hours ago and timestampLastTelemetrySent is hour ago
			// need to do telemetry
			service.timestampLastTelemetrySent = time.Now().Add(-time.Hour)
			service.doTelemetryIfNeeded(firstRun.Add(-2 * time.Hour))
			checkMockRudderServer(t)

			// firstRun is 2 hours ago and timestampLastTelemetrySent is just now
			// no need to do telemetry
			service.doTelemetryIfNeeded(firstRun.Add(-2 * time.Hour))
			require.Equal(t, 0, len(receiveChan))
		})
		t.Run("Send at the 24 hour mark and every 24 hours after", func(t *testing.T) {
			// firstRun is 24 hours ago and timestampLastTelemetrySent is 24 hours ago
			// need to do telemetry
			service.timestampLastTelemetrySent = time.Now().Add(-24 * time.Hour)
			service.doTelemetryIfNeeded(firstRun.Add(-24 * time.Hour))
			checkMockRudderServer(t)

			// firstRun is 24 hours ago and timestampLastTelemetrySent is just now
			// no need to do telemetry
			service.doTelemetryIfNeeded(firstRun.Add(-24 * time.Hour))
			require.Equal(t, 0, len(receiveChan))
		})
	})
}