Merge branch 'main' into oguzhan/acc-connector

This commit is contained in:
Charles Driesler
2025-08-05 11:46:37 +01:00
11 changed files with 145 additions and 2 deletions
+2 -1
View File
@@ -16,6 +16,7 @@
"**/{ava,babel,nyc}.config.{js,cjs,mjs}",
"**/jest.config.{js,cjs,mjs,ts}",
"**/{karma,rollup,webpack}.config.js",
"**/.{eslint,mocha}rc.{js,cjs}"
"**/.{eslint,mocha}rc.{js,cjs}",
"**/domain/**/*.{ts,js}"
]
}
+1 -1
View File
@@ -21,7 +21,7 @@ const aliases = {
/**
* EXTENSIONS TO EVALUATE FOR EXTENSIONLESS IMPORTS
*/
const extensions = ['.ts', '.js', '.mjs', '.cjs', '.json', '.d.ts']
const extensions = ['.ts', '.js', '.mjs', '.cjs', '.json']
// Register the module hooks
register('./esmLoader.js', {
@@ -0,0 +1,125 @@
import { describe, it, expect, afterEach, beforeEach } from 'vitest'
import { type FeatureFlags, parseFeatureFlags } from './index.js'
const originalDisableAllFfs = process.env.DISABLE_ALL_FFS || ''
const originalEnableAllFfs = process.env.ENABLE_ALL_FFS || ''
const setDisableAllFfs = (value: boolean) => {
process.env.DISABLE_ALL_FFS = value.toString()
}
const setEnableAllFfs = (value: boolean) => {
process.env.ENABLE_ALL_FFS = value.toString()
}
const resetEnv = () => {
process.env.DISABLE_ALL_FFS = originalDisableAllFfs
process.env.ENABLE_ALL_FFS = originalEnableAllFfs
}
describe('parseFeatureFlags', () => {
beforeEach(() => {
// Disable global "ALL FFs"/"NO FFs" modes for these tests, cause they break em
setDisableAllFfs(false)
setEnableAllFfs(false)
})
afterEach(() => {
resetEnv()
})
it('returns all defaults as false', () => {
const flags = parseFeatureFlags({})
for (const key of Object.keys(flags)) {
expect(flags[key as keyof FeatureFlags]).toBe(false)
}
})
it('parses explicit true/false values', () => {
const flags = parseFeatureFlags({
FF_AUTOMATE_MODULE_ENABLED: 'true',
FF_GENDOAI_MODULE_ENABLED: 'false',
FF_SAVED_VIEWS_ENABLED: 'true'
})
expect(flags.FF_AUTOMATE_MODULE_ENABLED).toBe(true)
expect(flags.FF_GENDOAI_MODULE_ENABLED).toBe(false)
expect(flags.FF_SAVED_VIEWS_ENABLED).toBe(true)
})
it('DISABLE_ALL_FFS disables all flags unless forceInputs is true', () => {
setDisableAllFfs(true)
const flags = parseFeatureFlags(
{
FF_AUTOMATE_MODULE_ENABLED: 'true',
FF_SAVED_VIEWS_ENABLED: 'true'
},
{ forceInputs: false }
)
for (const key of Object.keys(flags)) {
expect(flags[key as keyof FeatureFlags]).toBe(false)
}
})
it('ENABLE_ALL_FFS enables all flags unless forceInputs is true', () => {
setEnableAllFfs(true)
const flags = parseFeatureFlags(
{
FF_AUTOMATE_MODULE_ENABLED: 'false',
FF_SAVED_VIEWS_ENABLED: 'false'
},
{ forceInputs: false }
)
for (const key of Object.keys(flags)) {
expect(flags[key as keyof FeatureFlags]).toBe(true)
}
})
it('forceInputs=true preserves explicit input values even with DISABLE_ALL_FFS', () => {
setDisableAllFfs(true)
const flags = parseFeatureFlags(
{
FF_AUTOMATE_MODULE_ENABLED: 'true',
FF_SAVED_VIEWS_ENABLED: 'false'
},
{ forceInputs: true }
)
expect(flags.FF_AUTOMATE_MODULE_ENABLED).toBe(true)
expect(flags.FF_SAVED_VIEWS_ENABLED).toBe(false)
// All others should be false
for (const key of Object.keys(flags)) {
if (key !== 'FF_AUTOMATE_MODULE_ENABLED' && key !== 'FF_SAVED_VIEWS_ENABLED') {
expect(flags[key as keyof FeatureFlags]).toBe(false)
}
}
})
it('forceInputs=true preserves explicit input values even with ENABLE_ALL_FFS', () => {
setEnableAllFfs(true)
const flags = parseFeatureFlags(
{
FF_AUTOMATE_MODULE_ENABLED: 'false',
FF_SAVED_VIEWS_ENABLED: 'true'
},
{ forceInputs: true }
)
expect(flags.FF_AUTOMATE_MODULE_ENABLED).toBe(false)
expect(flags.FF_SAVED_VIEWS_ENABLED).toBe(true)
// All others should be true
for (const key of Object.keys(flags)) {
if (key !== 'FF_AUTOMATE_MODULE_ENABLED' && key !== 'FF_SAVED_VIEWS_ENABLED') {
expect(flags[key as keyof FeatureFlags]).toBe(true)
}
}
})
it('it can handle empty string env vars', () => {
const flags = parseFeatureFlags({
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-expect-error
FF_AUTOMATE_MODULE_ENABLED: ''
})
expect(flags.FF_AUTOMATE_MODULE_ENABLED).toBe(false)
})
})
+10
View File
@@ -25,6 +25,16 @@ export const parseFeatureFlags = (
): FeatureFlags => {
const { forceInputs = true } = options || {}
// Clean up input: unset empty values
for (const key of Object.keys(input)) {
const typedKey = key as keyof FeatureFlags
const typedVal = input[typedKey] as unknown
if (typedVal === undefined || typedVal === '') {
delete input[typedKey]
}
}
//INFO
// As a convention all feature flags should be prefixed with a FF_
const res = parseEnv(input, {
@@ -139,6 +139,11 @@
"type": "boolean",
"description": "Enables the dedicated Rhino based file importer. This is not part of the deployment.",
"default": false
},
"savedViewsEnabled": {
"type": "boolean",
"description": "Enables the ability to create and manage saved views",
"default": false
}
}
},
+2
View File
@@ -77,6 +77,8 @@ featureFlags:
accIntegrationEnabled: false
## @param featureFlags.rhinoFileImporterEnabled Enables the dedicated Rhino based file importer. This is not part of the deployment.
rhinoFileImporterEnabled: false
## @param featureFlags.savedViewsEnabled Enables the ability to create and manage saved views
savedViewsEnabled: false
analytics:
## @param analytics.enabled Enable or disable analytics