Skip to content

Commit 81c15cf

Browse files
authored
chore(specs): add eslint rule to avoid cross-references (#3413)
1 parent 71c4f92 commit 81c15cf

13 files changed

+188
-32
lines changed

.eslintrc.cjs

+1
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,7 @@ module.exports = {
6464
'automation-custom/out-of-line-all-of': 'error',
6565
'automation-custom/out-of-line-any-of': 'error',
6666
'automation-custom/valid-acl': 'error',
67+
'automation-custom/ref-common': 'error',
6768
},
6869
},
6970
],

.github/workflows/check.yml

+3
Original file line numberDiff line numberDiff line change
@@ -139,6 +139,9 @@ jobs:
139139
- name: Test scripts
140140
run: yarn scripts:test
141141

142+
- name: Lint custom eslint plugin
143+
run: yarn workspace eslint-plugin-automation-custom lint
144+
142145
- name: Test custom eslint plugin
143146
run: yarn workspace eslint-plugin-automation-custom test
144147

eslint/package.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,14 @@
22
"name": "eslint-plugin-automation-custom",
33
"version": "1.0.0",
44
"description": "Custom rules for eslint",
5-
"main": "dist/index.js",
5+
"main": "dist/src/index.js",
66
"files": [
77
"src/**.ts"
88
],
99
"scripts": {
1010
"build": "rm -rf dist/ && tsc",
11+
"lint": "eslint --ext=ts .",
12+
"lint:fix": "eslint --ext=ts --fix .",
1113
"test": "jest"
1214
},
1315
"devDependencies": {

eslint/src/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ import { endWithDot } from './rules/endWithDot';
22
import { noFinalDot } from './rules/noFinalDot';
33
import { noNewLine } from './rules/noNewLine';
44
import { createOutOfLineRule } from './rules/outOfLineRule';
5+
import { refCommon } from './rules/refCommon';
56
import { singleQuoteRef } from './rules/singleQuoteRef';
67
import { validACL } from './rules/validACL';
78

@@ -15,6 +16,7 @@ const rules = {
1516
'single-quote-ref': singleQuoteRef,
1617
'valid-acl': validACL,
1718
'no-new-line': noNewLine,
19+
'ref-common': refCommon,
1820
};
1921

2022
// Custom parser for ESLint, to read plain text file like mustache.

eslint/src/rules/endWithDot.ts

+1-3
Original file line numberDiff line numberDiff line change
@@ -19,9 +19,7 @@ export const endWithDot: Rule.RuleModule = {
1919

2020
return {
2121
YAMLPair(node): void {
22-
if (
23-
!isPairWithKey(node, 'description')
24-
) {
22+
if (!isPairWithKey(node, 'description')) {
2523
return;
2624
}
2725

eslint/src/rules/refCommon.ts

+73
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,73 @@
1+
import type { Rule } from 'eslint';
2+
3+
import { isPairWithKey, isScalar } from '../utils';
4+
5+
const allSpecs = [
6+
'abtesting',
7+
'analytics',
8+
'crawler',
9+
'ingestion',
10+
'insights',
11+
'monitoring',
12+
'personalization',
13+
'query-suggestions',
14+
'recommend',
15+
'search',
16+
'usage',
17+
];
18+
19+
export const refCommon: Rule.RuleModule = {
20+
meta: {
21+
docs: {
22+
description:
23+
'the $ref must target the current spec, or the common spec. If you intended to use a model from an other spec, move it to the common folder',
24+
},
25+
messages: {
26+
refCommon: '$ref to another spec',
27+
},
28+
},
29+
create(context) {
30+
if (!context.sourceCode.parserServices.isYAML) {
31+
return {};
32+
}
33+
34+
return {
35+
YAMLPair(node): void {
36+
if (!isPairWithKey(node, '$ref')) {
37+
return;
38+
}
39+
40+
if (!isScalar(node.value)) {
41+
return;
42+
}
43+
44+
let ref = node.value.value;
45+
if (
46+
typeof ref !== 'string' ||
47+
ref.trim().startsWith('#/') ||
48+
ref.startsWith('./')
49+
) {
50+
return;
51+
}
52+
53+
const spec = context.filename.match(/specs\/([a-z-]+?)\//)?.[1];
54+
if (!spec) {
55+
return;
56+
}
57+
while (ref.startsWith(`../`)) {
58+
ref = ref.slice(3);
59+
}
60+
if (
61+
allSpecs.filter((s) => s !== spec).every((s) => !ref.startsWith(s))
62+
) {
63+
return;
64+
}
65+
66+
context.report({
67+
node: node as any,
68+
messageId: 'refCommon',
69+
});
70+
},
71+
};
72+
},
73+
};

eslint/src/rules/validACL.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ const ACLs = [
1616
'usage',
1717
'logs',
1818
'setUnretrievableAttributes',
19-
'admin'
19+
'admin',
2020
];
2121

2222
export const validACL: Rule.RuleModule = {

eslint/tests/noNewLine.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,6 @@ multiple lines`,
3535
`,
3636
errors: [{ messageId: 'noNewLine' }],
3737
output: `multiple new lines`,
38-
}
38+
},
3939
],
4040
});

eslint/tests/outOfLineRule.test.ts

+31-22
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ root:
4747
inside:
4848
type: string
4949
enum: [bla, blabla]
50-
`,
50+
`,
5151
errors: [{ messageId: 'enumNotOutOfLine' }],
5252
},
5353
{
@@ -60,33 +60,37 @@ root:
6060
6161
useIt:
6262
$ref: '#/root/inside/deeper'
63-
`,
63+
`,
6464
errors: [{ messageId: 'enumNotOutOfLine' }],
6565
},
6666
],
6767
});
6868

69-
7069
// oneOf should allow `type: 'null'`
71-
ruleTester.run('out-of-line-oneOf-null', createOutOfLineRule({ property: 'oneOf' }), {
72-
valid: [
73-
`
70+
ruleTester.run(
71+
'out-of-line-oneOf-null',
72+
createOutOfLineRule({ property: 'oneOf' }),
73+
{
74+
valid: [
75+
`
7476
simple:
7577
oneOf:
7678
- type: string
7779
- type: 'null'
78-
`,`
80+
`,
81+
`
7982
obj:
8083
type: object
8184
properties:
8285
name:
8386
oneOf:
8487
- type: string
8588
- type: 'null'
86-
`],
87-
invalid: [
88-
{
89-
code: `
89+
`,
90+
],
91+
invalid: [
92+
{
93+
code: `
9094
simple:
9195
type: object
9296
properties:
@@ -95,21 +99,26 @@ simple:
9599
- type: string
96100
- type: null
97101
`,
98-
errors: [{ messageId: 'oneOfNotOutOfLine' }],
99-
},
100-
],
101-
});
102+
errors: [{ messageId: 'oneOfNotOutOfLine' }],
103+
},
104+
],
105+
}
106+
);
102107

103108
// allow enum to be nullable
104-
ruleTester.run('out-of-line-enum-null', createOutOfLineRule({ property: 'enum' }), {
105-
valid: [
106-
`
109+
ruleTester.run(
110+
'out-of-line-enum-null',
111+
createOutOfLineRule({ property: 'enum' }),
112+
{
113+
valid: [
114+
`
107115
simple:
108116
oneOf:
109117
- type: string
110118
enum: [bla, blabla]
111119
- type: 'null'
112-
`],
113-
invalid: [],
114-
});
115-
120+
`,
121+
],
122+
invalid: [],
123+
}
124+
);

eslint/tests/refCommon.test.ts

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
import { RuleTester } from 'eslint';
2+
3+
import { refCommon } from '../src/rules/refCommon';
4+
5+
const ruleTester = new RuleTester({
6+
parser: require.resolve('yaml-eslint-parser'),
7+
});
8+
9+
ruleTester.run('ref-common', refCommon, {
10+
valid: [
11+
{
12+
filename: 'api-client-automation/specs/search/path/test.yml',
13+
code: `
14+
local:
15+
$ref: '#/components/schemas/Example'
16+
`,
17+
},
18+
{
19+
filename: 'renamedRepo/specs/search/path/test.yml',
20+
code: `
21+
local:
22+
$ref: '#/components/schemas/Example'
23+
`,
24+
},
25+
{
26+
filename: 'api-client-automation/specs/search/path/test.yml',
27+
code: `
28+
sameFolder:
29+
$ref: './schemas/Example'
30+
`,
31+
},
32+
{
33+
filename: 'api-client-automation/specs/search/path/test.yml',
34+
code: `
35+
external:
36+
$ref: '../../common/schemas/Example'
37+
`,
38+
},
39+
{
40+
filename: 'api-client-automation/specs/search/path/test.yml',
41+
code: `
42+
external:
43+
$ref: '../../../search/schemas/Example'
44+
`,
45+
},
46+
{
47+
filename: 'api-client-automation/specs/search/path/test.yml',
48+
code: `
49+
nested:
50+
type: object
51+
properties:
52+
nested:
53+
$ref: '#/components/schemas/Example'
54+
`,
55+
},
56+
],
57+
invalid: [
58+
{
59+
filename: 'api-client-automation/specs/search/path/test.yml',
60+
code: `
61+
out:
62+
$ref: '../../recommend/schemas/Example'
63+
`,
64+
errors: [{ messageId: 'refCommon' }],
65+
},
66+
],
67+
});

eslint/tests/validACL.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import { RuleTester } from 'eslint';
2-
import { validACL } from '../src/rules/validACL';
32

3+
import { validACL } from '../src/rules/validACL';
44

55
const ruleTester = new RuleTester({
66
parser: require.resolve('yaml-eslint-parser'),

eslint/tsconfig.json

+1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
},
66
"include": [
77
"src/**/*.ts",
8+
"tests/**/*.ts"
89
],
910
"exclude": [
1011
"node_modules",

scripts/formatter.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -27,9 +27,6 @@ export async function formatter(language: string, cwd: string): Promise<void> {
2727
case 'go':
2828
await run('goimports -w . && golangci-lint run --fix', { cwd, language });
2929
break;
30-
case 'javascript':
31-
await run(`yarn eslint --ext=ts,json ${cwd} --fix --no-error-on-unmatched-pattern`);
32-
break;
3330
case 'java':
3431
await run(
3532
`find . -type f -name "*.java" | xargs java --add-exports jdk.compiler/com.sun.tools.javac.api=ALL-UNNAMED \
@@ -42,6 +39,9 @@ export async function formatter(language: string, cwd: string): Promise<void> {
4239
{ cwd, language },
4340
);
4441
break;
42+
case 'javascript':
43+
await run(`yarn eslint --ext=ts,json ${cwd} --fix --no-error-on-unmatched-pattern`);
44+
break;
4545
case 'kotlin':
4646
await run(`./gradle/gradlew -p ${cwd} spotlessApply`, { language });
4747
break;

0 commit comments

Comments
 (0)