@akinon/eslint-plugin-projectzero 1.55.0 → 1.56.0-rc.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,13 @@
1
1
  # @akinon/eslint-plugin-projectzero
2
2
 
3
+ ## 1.56.0-rc.0
4
+
5
+ ### Minor Changes
6
+
7
+ - 4d64ef7e: ZERO-2611: Add check-sentry-options rule to eslint-plugin-projectzero
8
+ - 5a4c6076: ZERO-2764: Add case-warning rule to eslint-plugin-projectzero
9
+ - 64699d3f: ZERO-2761: Fix invalid import for plugin module
10
+
3
11
  ## 1.55.0
4
12
 
5
13
  ## 1.54.0
package/configs/core.ts CHANGED
@@ -10,6 +10,8 @@ export default {
10
10
  '@akinon/projectzero/urls-without-slash': 'error',
11
11
  '@akinon/projectzero/invalid-imports': 'error',
12
12
  '@akinon/projectzero/check-menu-depth': 'error',
13
+ '@akinon/projectzero/case-warning': 'warn',
14
+ '@akinon/projectzero/check-sentry-options': 'error',
13
15
  'no-restricted-syntax': [
14
16
  'error',
15
17
  {
@@ -12,6 +12,8 @@ exports.default = {
12
12
  '@akinon/projectzero/urls-without-slash': 'error',
13
13
  '@akinon/projectzero/invalid-imports': 'error',
14
14
  '@akinon/projectzero/check-menu-depth': 'error',
15
+ '@akinon/projectzero/case-warning': 'warn',
16
+ '@akinon/projectzero/check-sentry-options': 'error',
15
17
  'no-restricted-syntax': [
16
18
  'error',
17
19
  {
@@ -0,0 +1,32 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const utils_1 = require("@typescript-eslint/utils");
4
+ exports.default = utils_1.ESLintUtils.RuleCreator.withoutDocs({
5
+ create(context) {
6
+ return {
7
+ CallExpression(node) {
8
+ if (node.callee.type === 'MemberExpression' &&
9
+ node.callee.property.type === 'Identifier' &&
10
+ (node.callee.property.name === 'toLocaleUpperCase' ||
11
+ node.callee.property.name === 'toLocaleLowerCase')) {
12
+ const args = node.arguments;
13
+ if (args.length === 0) {
14
+ context.report({
15
+ node,
16
+ messageId: 'noLocale'
17
+ });
18
+ }
19
+ }
20
+ }
21
+ };
22
+ },
23
+ meta: {
24
+ messages: {
25
+ noLocale: 'toLocaleUpperCase() or toLocaleLowerCase() should specify a locale for consistent behavior across different environments.'
26
+ },
27
+ type: 'problem',
28
+ fixable: 'code',
29
+ schema: []
30
+ },
31
+ defaultOptions: []
32
+ });
@@ -0,0 +1,62 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const utils_1 = require("@typescript-eslint/utils");
4
+ exports.default = utils_1.ESLintUtils.RuleCreator.withoutDocs({
5
+ create(context) {
6
+ return {
7
+ CallExpression(node) {
8
+ const callee = node.callee;
9
+ if (callee.object &&
10
+ callee.object.name === 'Sentry' &&
11
+ callee.property &&
12
+ callee.property.name === 'init') {
13
+ const optionsArgument = node
14
+ .arguments[0];
15
+ if (optionsArgument && optionsArgument.type === 'ObjectExpression') {
16
+ optionsArgument.properties.forEach((property) => {
17
+ if (property.type === 'Property') {
18
+ const key = property.key.name;
19
+ if (key === 'dsn') {
20
+ const dsnValue = property.value;
21
+ if (dsnValue.type === 'Literal' && !dsnValue.value) {
22
+ context.report({
23
+ node: property,
24
+ messageId: 'invalidDsn'
25
+ });
26
+ }
27
+ if (dsnValue.type === 'MemberExpression') {
28
+ const objectName = dsnValue.object
29
+ .name;
30
+ const propertyName = dsnValue.property.name;
31
+ if (objectName === 'process' && propertyName === 'env') {
32
+ return;
33
+ }
34
+ }
35
+ }
36
+ if (key === 'tracesSampleRate') {
37
+ const traceRate = property.value.value;
38
+ if (traceRate !== 1.0) {
39
+ context.report({
40
+ node: property,
41
+ messageId: 'incorrectTraceRate'
42
+ });
43
+ }
44
+ }
45
+ }
46
+ });
47
+ }
48
+ }
49
+ }
50
+ };
51
+ },
52
+ meta: {
53
+ messages: {
54
+ invalidDsn: 'Sentry DSN should not be empty.',
55
+ incorrectTraceRate: 'Sentry tracesSampleRate should be set to 1.0.'
56
+ },
57
+ type: 'problem',
58
+ fixable: 'code',
59
+ schema: []
60
+ },
61
+ defaultOptions: []
62
+ });
@@ -15,6 +15,8 @@ const check_middleware_order_1 = __importDefault(require("./check-middleware-ord
15
15
  const urls_without_slash_1 = __importDefault(require("./urls-without-slash"));
16
16
  const invalid_imports_1 = __importDefault(require("./invalid-imports"));
17
17
  const check_menu_depth_1 = __importDefault(require("./check-menu-depth"));
18
+ const case_warning_1 = __importDefault(require("./case-warning"));
19
+ const check_sentry_options_1 = __importDefault(require("./check-sentry-options"));
18
20
  const rules = {
19
21
  'client-url': client_url_1.default,
20
22
  'image-import': image_import_1.default,
@@ -26,6 +28,8 @@ const rules = {
26
28
  'check-middleware-order': check_middleware_order_1.default,
27
29
  'urls-without-slash': urls_without_slash_1.default,
28
30
  'invalid-imports': invalid_imports_1.default,
29
- 'check-menu-depth': check_menu_depth_1.default
31
+ 'check-menu-depth': check_menu_depth_1.default,
32
+ 'case-warning': case_warning_1.default,
33
+ 'check-sentry-options': check_sentry_options_1.default
30
34
  };
31
35
  exports.rules = rules;
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@akinon/eslint-plugin-projectzero",
3
- "version": "1.55.0",
3
+ "version": "1.56.0-rc.0",
4
4
  "private": false,
5
5
  "description": "ESLint plugin for Project Zero Next",
6
6
  "main": "dist/index.js",
@@ -0,0 +1,39 @@
1
+ import { ESLintUtils } from '@typescript-eslint/utils';
2
+ import { TSESTree } from '@typescript-eslint/types';
3
+
4
+ type MessageIds = 'noLocale';
5
+ type Options = readonly [];
6
+
7
+ export default ESLintUtils.RuleCreator.withoutDocs<Options, MessageIds>({
8
+ create(context) {
9
+ return {
10
+ CallExpression(node: TSESTree.CallExpression) {
11
+ if (
12
+ node.callee.type === 'MemberExpression' &&
13
+ node.callee.property.type === 'Identifier' &&
14
+ (node.callee.property.name === 'toLocaleUpperCase' ||
15
+ node.callee.property.name === 'toLocaleLowerCase')
16
+ ) {
17
+ const args = node.arguments;
18
+
19
+ if (args.length === 0) {
20
+ context.report({
21
+ node,
22
+ messageId: 'noLocale'
23
+ });
24
+ }
25
+ }
26
+ }
27
+ };
28
+ },
29
+ meta: {
30
+ messages: {
31
+ noLocale:
32
+ 'toLocaleUpperCase() or toLocaleLowerCase() should specify a locale for consistent behavior across different environments.'
33
+ },
34
+ type: 'problem',
35
+ fixable: 'code',
36
+ schema: []
37
+ },
38
+ defaultOptions: []
39
+ });
@@ -0,0 +1,72 @@
1
+ import { ESLintUtils, TSESTree } from '@typescript-eslint/utils';
2
+
3
+ export default ESLintUtils.RuleCreator.withoutDocs({
4
+ create(context) {
5
+ return {
6
+ CallExpression(node: TSESTree.CallExpression) {
7
+ const callee = node.callee as TSESTree.MemberExpression;
8
+
9
+ if (
10
+ callee.object &&
11
+ (callee.object as TSESTree.Identifier).name === 'Sentry' &&
12
+ callee.property &&
13
+ (callee.property as TSESTree.Identifier).name === 'init'
14
+ ) {
15
+ const optionsArgument = node
16
+ .arguments[0] as TSESTree.ObjectExpression;
17
+
18
+ if (optionsArgument && optionsArgument.type === 'ObjectExpression') {
19
+ optionsArgument.properties.forEach((property) => {
20
+ if (property.type === 'Property') {
21
+ const key = (property.key as TSESTree.Identifier).name;
22
+
23
+ if (key === 'dsn') {
24
+ const dsnValue = property.value;
25
+
26
+ if (dsnValue.type === 'Literal' && !dsnValue.value) {
27
+ context.report({
28
+ node: property,
29
+ messageId: 'invalidDsn'
30
+ });
31
+ }
32
+
33
+ if (dsnValue.type === 'MemberExpression') {
34
+ const objectName = (dsnValue.object as TSESTree.Identifier)
35
+ .name;
36
+ const propertyName = (
37
+ dsnValue.property as TSESTree.Identifier
38
+ ).name;
39
+
40
+ if (objectName === 'process' && propertyName === 'env') {
41
+ return;
42
+ }
43
+ }
44
+ }
45
+
46
+ if (key === 'tracesSampleRate') {
47
+ const traceRate = (property.value as TSESTree.Literal).value;
48
+ if (traceRate !== 1.0) {
49
+ context.report({
50
+ node: property,
51
+ messageId: 'incorrectTraceRate'
52
+ });
53
+ }
54
+ }
55
+ }
56
+ });
57
+ }
58
+ }
59
+ }
60
+ };
61
+ },
62
+ meta: {
63
+ messages: {
64
+ invalidDsn: 'Sentry DSN should not be empty.',
65
+ incorrectTraceRate: 'Sentry tracesSampleRate should be set to 1.0.'
66
+ },
67
+ type: 'problem',
68
+ fixable: 'code',
69
+ schema: []
70
+ },
71
+ defaultOptions: []
72
+ });
package/rules/index.ts CHANGED
@@ -9,6 +9,8 @@ import checkMiddlewareOrder from './check-middleware-order';
9
9
  import urlsWithoutSlash from './urls-without-slash';
10
10
  import invalidImports from './invalid-imports';
11
11
  import checkMenuDepth from './check-menu-depth';
12
+ import caseWarning from './case-warning';
13
+ import checkSentryOptions from './check-sentry-options';
12
14
 
13
15
  const rules = {
14
16
  'client-url': clientUrl,
@@ -21,7 +23,9 @@ const rules = {
21
23
  'check-middleware-order': checkMiddlewareOrder,
22
24
  'urls-without-slash': urlsWithoutSlash,
23
25
  'invalid-imports': invalidImports,
24
- 'check-menu-depth': checkMenuDepth
26
+ 'check-menu-depth': checkMenuDepth,
27
+ 'case-warning': caseWarning,
28
+ 'check-sentry-options': checkSentryOptions
25
29
  };
26
30
 
27
31
  export { rules };