mirror of
https://github.com/yu-i-i/overleaf-cep.git
synced 2026-05-23 09:09:36 +02:00
Add customer.io fields for migration comms and marketing initiatives GitOrigin-RevId: f11ffee255d9582cbfd4c7e285bd6690c0cf1e3c
270 lines
8.5 KiB
JavaScript
270 lines
8.5 KiB
JavaScript
const { RuleTester } = require('eslint')
|
|
const preferKebabUrl = require('./prefer-kebab-url')
|
|
const noUnnecessaryTrans = require('./no-unnecessary-trans')
|
|
const shouldUnescapeTrans = require('./should-unescape-trans')
|
|
const noGeneratedEditorThemes = require('./no-generated-editor-themes')
|
|
const viDoMockValidPath = require('./require-vi-doMock-valid-path')
|
|
const requireCioSnakeCaseProperties = require('./require-cio-snake-case-properties')
|
|
|
|
const ruleTester = new RuleTester({
|
|
parser: require.resolve('@typescript-eslint/parser'),
|
|
parserOptions: {
|
|
ecmaVersion: 'latest',
|
|
ecmaFeatures: { jsx: true },
|
|
},
|
|
})
|
|
|
|
ruleTester.run('prefer-kebab-url', preferKebabUrl, {
|
|
valid: [
|
|
{ code: `app.get('/foo-bar')` },
|
|
{ code: `app.get('/foo-bar/:id')` },
|
|
{ code: `router.post('/foo-bar')` },
|
|
{ code: `router.get('/foo-bar/:id/:name/:age')` },
|
|
{ code: `webRouter.get('/foo-bar/:user_id/(ProjectName)/get-info')` },
|
|
{ code: `webApp.post('/foo-bar/:user_id/(ProjectName)/get-info')` },
|
|
{
|
|
code: `router.get(/^\\/download\\/project\\/([^/]*)\\/output\\/output\\.pdf$/)`,
|
|
},
|
|
{
|
|
code: `webRouter.get(/^\\/project\\/([^/]*)\\/user\\/([0-9a-f]+)\\/build\\/([0-9a-f-]+)\\/output\\/(.*)$/)`,
|
|
},
|
|
],
|
|
invalid: [
|
|
{
|
|
code: `app.get('/fooBar')`,
|
|
errors: [{ message: 'Route path should be in kebab-case.' }],
|
|
},
|
|
{
|
|
code: `app.get('/fooBar/:id')`,
|
|
errors: [{ message: 'Route path should be in kebab-case.' }],
|
|
},
|
|
{
|
|
code: `webRouter.get('/foo_bar/:id/FooBar/:name/fooBar')`,
|
|
errors: [{ message: 'Route path should be in kebab-case.' }],
|
|
},
|
|
{
|
|
code: `router.get(/^\\/downLoad\\/pro-ject\\/([^/]*)\\/OutPut\\/out-put\\.pdf$/)`,
|
|
errors: [{ message: 'Route path should be in kebab-case.' }],
|
|
},
|
|
],
|
|
})
|
|
|
|
ruleTester.run('no-unnecessary-trans', noUnnecessaryTrans, {
|
|
valid: [
|
|
{ code: `<Trans i18nKey="test" components={{ strong: <strong/> }}/>` },
|
|
],
|
|
invalid: [
|
|
{
|
|
code: `<Trans i18nKey="test" values={{ test: 'foo '}}/>`,
|
|
errors: [{ message: `Use t('…') when there are no components` }],
|
|
},
|
|
{
|
|
code: `<Trans i18nKey="test" />`,
|
|
errors: [{ message: `Use t('…') when there are no components` }],
|
|
output: `{t('test')}`,
|
|
},
|
|
],
|
|
})
|
|
|
|
ruleTester.run('should-unescape-trans', shouldUnescapeTrans, {
|
|
valid: [
|
|
{
|
|
code: `<Trans i18nKey="test" components={{ strong: <strong/> }}/>`,
|
|
},
|
|
{
|
|
code: `<Trans i18nKey="test" values={{ foo: 'bar' }} components={{ strong: <strong/> }} shouldUnescape tOptions={{ interpolation: { escapeValue: true } }}/>`,
|
|
},
|
|
],
|
|
invalid: [
|
|
{
|
|
code: `<Trans i18nKey="test" values={{ foo: 'bar' }} components={{ strong: <strong/> }} />`,
|
|
errors: [{ message: 'Trans with values must have shouldUnescape' }],
|
|
output: `<Trans i18nKey="test" values={{ foo: 'bar' }}\nshouldUnescape components={{ strong: <strong/> }} />`,
|
|
},
|
|
{
|
|
code: `<Trans i18nKey="test" values={{ foo: 'bar' }} components={{ strong: <strong/> }} shouldUnescape />`,
|
|
errors: [
|
|
{
|
|
message:
|
|
'Trans with shouldUnescape must have tOptions.interpolation.escapeValue',
|
|
},
|
|
],
|
|
output: `<Trans i18nKey="test" values={{ foo: 'bar' }} components={{ strong: <strong/> }} shouldUnescape\ntOptions={{ interpolation: { escapeValue: true } }} />`,
|
|
},
|
|
],
|
|
})
|
|
|
|
const noGeneratedEditorThemesError =
|
|
'EditorView.theme and EditorView.baseTheme each add CSS to the page for every instance of the theme. Store the theme in a variable and reuse it instead.'
|
|
ruleTester.run('no-generated-editor-themes', noGeneratedEditorThemes, {
|
|
valid: [
|
|
{
|
|
code: `EditorView.theme({ '.cm-editor': { color: 'black' } })`,
|
|
},
|
|
{
|
|
code: `const theme = EditorView.theme({ '.cm-editor': { color: 'black' } })`,
|
|
},
|
|
],
|
|
invalid: [
|
|
{
|
|
code: `function createTheme() { return EditorView.theme({ '.cm-editor': { color: 'black' } }) }`,
|
|
errors: [
|
|
{
|
|
message: noGeneratedEditorThemesError,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
code: `() => EditorView.theme({ '.cm-editor': { color: 'black' } })`,
|
|
errors: [
|
|
{
|
|
message: noGeneratedEditorThemesError,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
code: `class Foo { createTheme() { return EditorView.theme({ '.cm-editor': { color: 'black' } }) } }`,
|
|
errors: [
|
|
{
|
|
message: noGeneratedEditorThemesError,
|
|
},
|
|
],
|
|
},
|
|
],
|
|
})
|
|
|
|
ruleTester.run('domock-require-valid-path', viDoMockValidPath, {
|
|
valid: [
|
|
{
|
|
code: 'vi.doMock("./require-vi-doMock-valid-path.js")',
|
|
filename: __filename,
|
|
},
|
|
{
|
|
code: 'const filename = "./require-vi-doMock-valid-path.js"; vi.doMock(filename);',
|
|
filename: __filename,
|
|
},
|
|
],
|
|
invalid: [
|
|
{
|
|
code: "vi.doMock('./require-vi-doMock-valid-path2')",
|
|
filename: __filename,
|
|
errors: [
|
|
{
|
|
message:
|
|
'The path "./require-vi-doMock-valid-path2" in vi.doMock() cannot be resolved relative to the current file.',
|
|
},
|
|
],
|
|
},
|
|
{
|
|
code: 'const filename = "./require-vi-doMock-valid-path2.js"; vi.doMock(filename);',
|
|
filename: __filename,
|
|
errors: [
|
|
{
|
|
message:
|
|
'The first argument of vi.doMock() must be (or resolve to) a string literal representing a path.',
|
|
},
|
|
],
|
|
},
|
|
],
|
|
})
|
|
|
|
ruleTester.run(
|
|
'require-cio-snake-case-properties',
|
|
requireCioSnakeCaseProperties,
|
|
{
|
|
valid: [
|
|
// updateUserAttributes with snake_case keys
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { plan_type: 'free', group_size: 10 })`,
|
|
},
|
|
// Modules.promises.hooks.fire with snake_case keys
|
|
{
|
|
code: `Modules.promises.hooks.fire('setUserProperties', userId, { plan_type: 'free', last_active: 123 })`,
|
|
},
|
|
// Modules.hooks.fire with snake_case keys
|
|
{
|
|
code: `Modules.hooks.fire('setUserProperties', userId, { plan_type: 'free' })`,
|
|
},
|
|
// Single-word keys are valid snake_case
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { email: 'a@b.com', role: 'admin' })`,
|
|
},
|
|
// Computed/dynamic keys are skipped
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { [dynamicKey]: true })`,
|
|
},
|
|
// Spread elements are skipped
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { ...existingAttrs })`,
|
|
},
|
|
// Unrelated function calls are not checked
|
|
{
|
|
code: `SomeOtherHandler.updateUserAttributes(userId, { camelCase: true })`,
|
|
},
|
|
// fire() with a different event name is not checked
|
|
{
|
|
code: `Modules.promises.hooks.fire('someOtherEvent', userId, { camelCase: true })`,
|
|
},
|
|
],
|
|
invalid: [
|
|
// camelCase key in updateUserAttributes
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { planType: 'free' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'planType' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
// kebab-case string key
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { 'plan-type': 'free' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'plan-type' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
// PascalCase key
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { PlanType: 'free' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'PlanType' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
// camelCase in Modules.promises.hooks.fire
|
|
{
|
|
code: `Modules.promises.hooks.fire('setUserProperties', userId, { planType: 'free' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'planType' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
// camelCase in Modules.hooks.fire
|
|
{
|
|
code: `Modules.hooks.fire('setUserProperties', userId, { planType: 'free' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'planType' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
// Multiple invalid keys report multiple errors
|
|
{
|
|
code: `CustomerIoHandler.updateUserAttributes(userId, { planType: 'free', groupSize: 10, plan_term: 'annual' })`,
|
|
errors: [
|
|
{
|
|
message: `Customer.io attribute 'planType' must be in snake_case.`,
|
|
},
|
|
{
|
|
message: `Customer.io attribute 'groupSize' must be in snake_case.`,
|
|
},
|
|
],
|
|
},
|
|
],
|
|
}
|
|
)
|