Merge pull request #13272 from overleaf/jdt-hackathon-merge-ff

Allow merge/ replace of local feature flags

GitOrigin-RevId: 531c2b9e73da8b8ca90ec0ed334a21c584cebe59
This commit is contained in:
Jimmy Domagala-Tang 2023-08-03 10:42:52 -04:00 committed by Copybot
parent 7e90846a31
commit dfc9138dd6
2 changed files with 87 additions and 0 deletions

View file

@ -168,6 +168,45 @@ async function updateSplitTestBadgeInfo(name, badgeInfo) {
return _saveSplitTest(splitTest) return _saveSplitTest(splitTest)
} }
async function replaceSplitTests(tests) {
_checkEnvIsSafe('replace')
try {
await _deleteSplitTests()
return await SplitTest.create(tests)
} catch (error) {
throw OError.tag(error, 'Failed to replace all split tests', { tests })
}
}
async function mergeSplitTests(incomingTests, overWriteLocal) {
_checkEnvIsSafe('merge')
// this is required as the query returns models, and we need all the items to be objects,
// similar to the ones we recieve as incomingTests
const localTests = await SplitTest.find({}).lean().exec()
let merged
// we preserve the state of the local tests (baseTests)
// eg: if inTest is in phase 1, and basetest is in phase 2, the merged will be in state 2
// therefore, we can have the opposite effect by swapping the order of args (overwrite locals with sent tests)
if (overWriteLocal) {
merged = _mergeFlags(localTests, incomingTests)
} else {
merged = _mergeFlags(incomingTests, localTests)
}
try {
await _deleteSplitTests()
const success = await SplitTest.create(merged)
return success
} catch (error) {
throw OError.tag(error, 'Failed to merge all split tests, merged set was', {
merged,
})
}
}
async function switchToNextPhase({ name, comment }, userId) { async function switchToNextPhase({ name, comment }, userId) {
const splitTest = await getSplitTest({ name }) const splitTest = await getSplitTest({ name })
if (!splitTest) { if (!splitTest) {
@ -356,6 +395,47 @@ async function _saveSplitTest(splitTest) {
} }
} }
/*
* As this is only used for utility in local dev, we want to prevent this running in any other env
* since deleting all records in staging or prod would be very bad...
*/
function _checkEnvIsSafe(operation) {
if (process.env.NODE_ENV !== 'development') {
throw OError.tag(
`attempted to ${operation} all feature flags outside of local env`
)
}
}
async function _deleteSplitTests() {
_checkEnvIsSafe('delete')
let deleted
try {
deleted = await SplitTest.deleteMany({}).exec()
} catch (error) {
throw OError.tag('Failed to delete all split tests')
}
if (!deleted.acknowledged) {
throw OError.tag('Error deleting split tests, split tests have not updated')
}
}
function _mergeFlags(incomingTests, baseTests) {
// copy all base versions
const mergedSet = baseTests.map(test => test)
for (const inTest of incomingTests) {
// since name is a unique key, we can use it to compare
const newFeatureFlag = !mergedSet.some(bTest => bTest.name === inTest.name)
// only add new feature flags, instead of overwriting ones in baseTests, meaning baseTests take precendence
if (newFeatureFlag) {
mergedSet.push(inTest)
}
}
return mergedSet
}
module.exports = { module.exports = {
getSplitTest, getSplitTest,
getSplitTests, getSplitTests,
@ -366,4 +446,6 @@ module.exports = {
switchToNextPhase, switchToNextPhase,
revertToPreviousVersion, revertToPreviousVersion,
archive, archive,
replaceSplitTests,
mergeSplitTests,
} }

View file

@ -12,3 +12,8 @@
&:extend(.label); &:extend(.label);
&:extend(.label-info); &:extend(.label-info);
} }
.scrollable {
max-height: calc(100vh - 40vh);
overflow-y: auto;
}