Merge pull request #6785 from overleaf/em-split-tests-analytics-enabled

Add "analytics enabled" setting to split tests

GitOrigin-RevId: 9ddfda9e246cac7a13361b2d3df6884212583000
This commit is contained in:
Eric Mc Sween 2022-02-28 07:57:41 -05:00 committed by Copybot
parent 95eb69c268
commit 5ded04eaea
8 changed files with 212 additions and 71 deletions

View file

@ -6,6 +6,7 @@ const crypto = require('crypto')
const _ = require('lodash') const _ = require('lodash')
const { callbackify } = require('util') const { callbackify } = require('util')
const SplitTestCache = require('./SplitTestCache') const SplitTestCache = require('./SplitTestCache')
const { SplitTest } = require('../../models/SplitTest')
const DEFAULT_VARIANT = 'default' const DEFAULT_VARIANT = 'default'
const ALPHA_PHASE = 'alpha' const ALPHA_PHASE = 'alpha'
@ -107,31 +108,37 @@ async function assignInLocalsContext(
* Get a mapping of the active split test assignments for the given user * Get a mapping of the active split test assignments for the given user
*/ */
async function getActiveAssignmentsForUser(userId) { async function getActiveAssignmentsForUser(userId) {
const user = await UserGetter.promises.getUser(userId, { splitTests: 1 }) const user = await _getUser(userId)
if (user == null || user.splitTests == null) { if (user == null) {
return {} return {}
} }
const activeAssignments = {}
for (const [splitTestName, assignments] of Object.entries(user.splitTests)) {
const splitTest = await SplitTestCache.get(splitTestName)
if (splitTest == null) {
continue
}
const currentVersion = splitTest.getCurrentVersion()
if (!currentVersion || !currentVersion.active) {
continue
}
let assignment const splitTests = await SplitTest.find({
if (Array.isArray(assignments)) { $where: 'this.versions[this.versions.length - 1].active',
assignment = _.maxBy(assignments, 'versionNumber') }).exec()
} else { const assignments = {}
// Older format is a single string rather than an array of objects for (const splitTest of splitTests) {
assignment = { variantName: assignments } const { activeForUser, selectedVariantName, phase, versionNumber } =
await _getAssignmentMetadata(user.analyticsId, user, splitTest)
if (activeForUser) {
const assignment = {
variantName: selectedVariantName,
versionNumber,
phase,
} }
activeAssignments[splitTestName] = assignment const userAssignments = user.splitTests?.[splitTest.name]
if (Array.isArray(userAssignments)) {
const userAssignment = userAssignments.find(
x => x.versionNumber === versionNumber
)
if (userAssignment) {
assignment.assignedAt = userAssignment.assignedAt
} }
return activeAssignments }
assignments[splitTest.name] = assignment
}
}
return assignments
} }
async function _getAssignment( async function _getAssignment(
@ -158,8 +165,9 @@ async function _getAssignment(
return _makeAssignment(splitTest, cachedVariant, currentVersion) return _makeAssignment(splitTest, cachedVariant, currentVersion)
} }
} }
const user = userId && (await _getUser(userId))
const { activeForUser, selectedVariantName, phase, versionNumber } = const { activeForUser, selectedVariantName, phase, versionNumber } =
await _getAssignmentMetadata(analyticsId, userId, splitTest) await _getAssignmentMetadata(analyticsId, user, splitTest)
if (activeForUser) { if (activeForUser) {
const assignmentConfig = { const assignmentConfig = {
userId, userId,
@ -181,26 +189,19 @@ async function _getAssignment(
return DEFAULT_ASSIGNMENT return DEFAULT_ASSIGNMENT
} }
async function _getAssignmentMetadata(analyticsId, userId, splitTest) { async function _getAssignmentMetadata(analyticsId, user, splitTest) {
const currentVersion = splitTest.getCurrentVersion() const currentVersion = splitTest.getCurrentVersion()
const phase = currentVersion.phase const phase = currentVersion.phase
if ([ALPHA_PHASE, BETA_PHASE].includes(phase)) {
if (userId) {
const user = await _getUser(userId)
if ( if (
(phase === ALPHA_PHASE && !(user && user.alphaProgram)) || !user ||
(phase === BETA_PHASE && !(user && user.betaProgram)) (phase === ALPHA_PHASE && !user.alphaProgram) ||
(phase === BETA_PHASE && !user.betaProgram)
) { ) {
return { return {
activeForUser: false, activeForUser: false,
} }
} }
} else { const userId = user?._id.toString()
return {
activeForUser: false,
}
}
}
const percentile = _getPercentile( const percentile = _getPercentile(
analyticsId || userId, analyticsId || userId,
splitTest.name, splitTest.name,

View file

@ -58,6 +58,8 @@ async function createSplitTest(name, configuration, info = {}) {
versionNumber: 1, versionNumber: 1,
phase: configuration.phase, phase: configuration.phase,
active: configuration.active, active: configuration.active,
analyticsEnabled:
configuration.active && configuration.analyticsEnabled,
variants: stripedVariants, variants: stripedVariants,
}, },
], ],
@ -84,6 +86,7 @@ async function updateSplitTestConfig(name, configuration) {
versionNumber: lastVersion.versionNumber + 1, versionNumber: lastVersion.versionNumber + 1,
phase: configuration.phase, phase: configuration.phase,
active: configuration.active, active: configuration.active,
analyticsEnabled: configuration.active && configuration.analyticsEnabled,
variants: updatedVariants, variants: updatedVariants,
}) })
return _saveSplitTest(splitTest) return _saveSplitTest(splitTest)

View file

@ -60,6 +60,11 @@ const VersionSchema = new Schema(
default: true, default: true,
required: true, required: true,
}, },
analyticsEnabled: {
type: Boolean,
default: true,
required: true,
},
variants: [VariantSchema], variants: [VariantSchema],
createdAt: { createdAt: {
type: Date, type: Date,

View file

@ -118,6 +118,10 @@
background: inherit; background: inherit;
left: 20px; left: 20px;
} }
&:disabled + label {
opacity: 0.5;
cursor: not-allowed;
}
} }
label { label {

View file

@ -0,0 +1,51 @@
exports.tags = ['saas']
exports.migrate = async client => {
const { db } = client
await db.splittests.updateMany(
{},
{ $set: { 'versions.$[version].analyticsEnabled': true } },
{
arrayFilters: [
{
'version.active': true,
'version.analyticsEnabled': { $exists: false },
},
],
}
)
await db.splittests.updateMany(
{},
{ $set: { 'versions.$[version].analyticsEnabled': false } },
{
arrayFilters: [
{
'version.active': false,
'version.analyticsEnabled': { $exists: false },
},
],
}
)
}
exports.rollback = async client => {
const { db } = client
await db.splittests.updateMany(
{},
{ $unset: { 'versions.$[version].analyticsEnabled': 1 } },
{
arrayFilters: [
{ 'version.active': true, 'version.analyticsEnabled': true },
],
}
)
await db.splittests.updateMany(
{},
{ $unset: { 'versions.$[version].analyticsEnabled': 1 } },
{
arrayFilters: [
{ 'version.active': false, 'version.analyticsEnabled': false },
],
}
)
}

View file

@ -3,6 +3,7 @@ chai.should()
chai.use(require('chai-as-promised')) chai.use(require('chai-as-promised'))
chai.use(require('chaid')) chai.use(require('chaid'))
chai.use(require('sinon-chai')) chai.use(require('sinon-chai'))
chai.use(require('chai-exclude'))
// Do not truncate assertion errors // Do not truncate assertion errors
chai.config.truncateThreshold = 0 chai.config.truncateThreshold = 0

View file

@ -11,12 +11,15 @@ const MODULE_PATH = Path.join(
describe('SplitTestHandler', function () { describe('SplitTestHandler', function () {
beforeEach(function () { beforeEach(function () {
this.splitTest = { this.splitTests = [
getCurrentVersion: sinon.stub().returns({ active: true }), makeSplitTest('active-test'),
} makeSplitTest('legacy-test'),
this.inactiveSplitTest = { makeSplitTest('no-analytics-test-1', { analyticsEnabled: false }),
getCurrentVersion: sinon.stub().returns({ active: false }), makeSplitTest('no-analytics-test-2', {
} analyticsEnabled: false,
versionNumber: 2,
}),
]
this.UserGetter = { this.UserGetter = {
promises: { promises: {
@ -24,19 +27,24 @@ describe('SplitTestHandler', function () {
}, },
} }
this.SplitTest = {
find: sinon.stub().returns({
exec: sinon.stub().resolves(this.splitTests),
}),
}
this.SplitTestCache = { this.SplitTestCache = {
get: sinon.stub().resolves(null), get: sinon.stub().resolves(null),
} }
this.SplitTestCache.get.withArgs('legacy-test').resolves(this.splitTest) for (const splitTest of this.splitTests) {
this.SplitTestCache.get.withArgs('other-test').resolves(this.splitTest) this.SplitTestCache.get.withArgs(splitTest.name).resolves(splitTest)
this.SplitTestCache.get }
.withArgs('inactive-test')
.resolves(this.inactiveSplitTest)
this.SplitTestHandler = SandboxedModule.require(MODULE_PATH, { this.SplitTestHandler = SandboxedModule.require(MODULE_PATH, {
requires: { requires: {
'../User/UserGetter': this.UserGetter, '../User/UserGetter': this.UserGetter,
'./SplitTestCache': this.SplitTestCache, './SplitTestCache': this.SplitTestCache,
'../../models/SplitTest': { SplitTest: this.SplitTest },
'../User/UserUpdater': {}, '../User/UserUpdater': {},
'../Analytics/AnalyticsManager': {}, '../Analytics/AnalyticsManager': {},
'./LocalsHelper': {}, './LocalsHelper': {},
@ -49,14 +57,23 @@ describe('SplitTestHandler', function () {
this.user = { this.user = {
_id: ObjectId(), _id: ObjectId(),
splitTests: { splitTests: {
'legacy-test': 'legacy-variant', 'active-test': [
'other-test': [ {
{ variantName: 'default', versionNumber: 1 }, variantName: 'default',
{ variantName: 'latest', versionNumber: 3 }, versionNumber: 1,
{ variantName: 'experiment', versionNumber: 2 }, assignedAt: 'active-test-assigned-at',
},
], ],
'legacy-test': 'legacy-variant',
'inactive-test': [{ variantName: 'trythis' }], 'inactive-test': [{ variantName: 'trythis' }],
'unknown-test': [{ variantName: 'trythis' }], 'unknown-test': [{ variantName: 'trythis' }],
'no-analytics-test-2': [
{
variantName: 'some-variant',
versionNumber: 1,
assignedAt: 'no-analytics-assigned-at',
},
],
}, },
} }
this.UserGetter.promises.getUser this.UserGetter.promises.getUser
@ -69,19 +86,36 @@ describe('SplitTestHandler', function () {
}) })
it('handles the legacy assignment format', function () { it('handles the legacy assignment format', function () {
expect(this.assignments).to.have.property('legacy-test') expect(this.assignments['legacy-test']).to.deep.equal({
expect(this.assignments['legacy-test'].variantName).to.equal( variantName: 'variant-1',
'legacy-variant' phase: 'release',
) versionNumber: 1,
})
}) })
it('returns the last assignment for each active test', function () { it('returns the current assignment for each active test', function () {
expect(this.assignments).to.have.property('other-test') expect(this.assignments['active-test']).to.deep.equal({
expect(this.assignments['other-test'].variantName).to.equal('latest') variantName: 'variant-1',
phase: 'release',
versionNumber: 1,
assignedAt: 'active-test-assigned-at',
})
}) })
it('does not return assignments for inactive tests', function () { it('returns the current assignment for tests with analytics disabled', function () {
expect(this.assignments).not.to.have.property('inactive-test') expect(this.assignments['no-analytics-test-1']).to.deep.equal({
variantName: 'variant-1',
phase: 'release',
versionNumber: 1,
})
})
it('returns the current assignment for tests with analytics disabled that had previous assignments', function () {
expect(this.assignments['no-analytics-test-2']).to.deep.equal({
variantName: 'variant-1',
phase: 'release',
versionNumber: 2,
})
}) })
it('does not return assignments for unknown tests', function () { it('does not return assignments for unknown tests', function () {
@ -89,7 +123,7 @@ describe('SplitTestHandler', function () {
}) })
}) })
describe('with an inexistent user', function () { describe('with an non-existent user', function () {
beforeEach(async function () { beforeEach(async function () {
const unknownUserId = ObjectId() const unknownUserId = ObjectId()
this.assignments = this.assignments =
@ -115,8 +149,55 @@ describe('SplitTestHandler', function () {
) )
}) })
it('returns empty assignments', function () { it('returns current assignments', function () {
expect(this.assignments).to.deep.equal({}) expect(this.assignments).to.deep.equal({
'active-test': {
phase: 'release',
variantName: 'variant-1',
versionNumber: 1,
},
'legacy-test': {
phase: 'release',
variantName: 'variant-1',
versionNumber: 1,
},
'no-analytics-test-1': {
phase: 'release',
variantName: 'variant-1',
versionNumber: 1,
},
'no-analytics-test-2': {
phase: 'release',
variantName: 'variant-1',
versionNumber: 2,
},
})
}) })
}) })
}) })
function makeSplitTest(name, opts = {}) {
const {
active = true,
analyticsEnabled = active,
phase = 'release',
versionNumber = 1,
} = opts
return {
name,
getCurrentVersion: sinon.stub().returns({
active,
analyticsEnabled,
phase,
versionNumber,
variants: [
{
name: 'variant-1',
rolloutPercent: 100,
rolloutStripes: [{ start: 0, end: 100 }],
},
],
}),
}
}

View file

@ -1,10 +1,5 @@
const sinon = require('sinon') const sinon = require('sinon')
const sinonChai = require('sinon-chai') const { expect } = require('chai')
const chai = require('chai')
const chaiAsPromised = require('chai-as-promised')
chai.use(sinonChai)
chai.use(chaiAsPromised)
const { expect } = chai
const recurly = require('recurly') const recurly = require('recurly')
const modulePath = '../../../../app/src/Features/Subscription/RecurlyClient' const modulePath = '../../../../app/src/Features/Subscription/RecurlyClient'
const SandboxedModule = require('sandboxed-module') const SandboxedModule = require('sandboxed-module')