Skip to content

Commit 1d8f7a8

Browse files
✨: ESLintのディレクティブコメントに関するルールを追加 (#1192)
以下の課題を改善するために、[eslint-plugin-eslint-comments](https://github.com/mysticatea/eslint-plugin-eslint-comments)を追加しました。 - ページ全体を`eslint-disable`できないようにしたい - ESLintをdisableにした理由を必ず記載するようにしたい - 不要な`eslnit-disable`を検知したい ## ✅ What's done - [x] eslint-plugin-eslint-commentsを追加 - [eslint-comments/no-restricted-disable | eslint-plugin-eslint-comments](https://mysticatea.github.io/eslint-plugin-eslint-comments/rules/no-restricted-disable.html)以外は、`error`レベルでルールに追加 - `eslint-comments/no-restricted-disable`は、disableにしたいルールが現状思いつかないので設定を入れていません - [x] ESLintエラーになった箇所の修正
1 parent e0020f5 commit 1d8f7a8

30 files changed

+189
-123
lines changed

example-app/SantokuApp/.eslintrc.js

+13-2
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,13 @@
44
module.exports = {
55
plugins: ['strict-dependencies'],
66
root: true,
7-
// universe/native: https://github.com/expo/expo/tree/master/packages/eslint-config-universe
8-
extends: ['universe/native', 'plugin:react-hooks/recommended'],
7+
extends: [
8+
// universe/native: https://github.com/expo/expo/tree/master/packages/eslint-config-universe
9+
'universe/native',
10+
'plugin:react-hooks/recommended',
11+
// eslint-comments/recommended: https://mysticatea.github.io/eslint-plugin-eslint-comments/rules/
12+
'plugin:eslint-comments/recommended',
13+
],
914
overrides: [
1015
{
1116
files: ['*.ts', '*.tsx', '*.d.ts'],
@@ -110,5 +115,11 @@ module.exports = {
110115
],
111116
// https://eslint.org/docs/latest/rules/curly
112117
curly: ['error', 'all'],
118+
// https://mysticatea.github.io/eslint-plugin-eslint-comments/rules/no-unused-disable.html
119+
'eslint-comments/no-unused-disable': 'error',
120+
// https://mysticatea.github.io/eslint-plugin-eslint-comments/rules/no-use.html
121+
'eslint-comments/no-use': ['error', {allow: ['eslint-disable-line', 'eslint-disable-next-line']}],
122+
// https://mysticatea.github.io/eslint-plugin-eslint-comments/rules/require-description.html
123+
'eslint-comments/require-description': 'error',
113124
},
114125
};

example-app/SantokuApp/jest/__mocks__/@react-navigation/native.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -19,9 +19,11 @@ const mock: jest.Mocked<NavigationProp<ParamListBase>> = {
1919

2020
Object.defineProperty(__mocks, 'navigation', {value: mock});
2121

22-
// @react-navigation/nativeのすべてのNamed Exportを列挙するのは大変なので、
23-
// ES6のexport/import形式ではなく、module.exportsを使ってexportする。
24-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
22+
/*
23+
eslint-disable-next-line @typescript-eslint/no-unsafe-assignment --
24+
@react-navigation/nativeのすべてのNamed Exportを列挙するのは大変なので、
25+
ES6のexport/import形式ではなく、module.exportsを使ってexportする。
26+
*/
2527
module.exports = {
2628
...jest.requireActual('@react-navigation/native'),
2729
useNavigation: () => mock,

example-app/SantokuApp/jest/__mocks__/expo-secure-store.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,11 @@ export const getItemAsync = jest.fn(() => '1234567890');
33
export const deleteItemAsync = jest.fn();
44
export const isAvailableAsync = jest.fn();
55

6-
// expo-secure-storeのすべてのNamed Exportを列挙するのは大変なので、
7-
// ES6のexport/import形式ではなく、module.exportsを使ってexportする。
8-
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
6+
/*
7+
eslint-disable-next-line @typescript-eslint/no-unsafe-assignment --
8+
expo-secure-storeのすべてのNamed Exportを列挙するのは大変なので、
9+
ES6のexport/import形式ではなく、module.exportsを使ってexportする。
10+
*/
911
module.exports = {
1012
...jest.requireActual('expo-secure-store'),
1113
setItemAsync,

example-app/SantokuApp/package-lock.json

+47
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

example-app/SantokuApp/package.json

+1
Original file line numberDiff line numberDiff line change
@@ -106,6 +106,7 @@
106106
"eslint-config-universe": "^11.0.0",
107107
"eslint-formatter-rdjson": "^1.0.3",
108108
"eslint-plugin-deprecation": "^1.3.2",
109+
"eslint-plugin-eslint-comments": "^3.2.0",
109110
"eslint-plugin-jest": "^27.0.0",
110111
"eslint-plugin-strict-dependencies": "^1.0.1",
111112
"expo-module-scripts": "^3.0.4",

example-app/SantokuApp/src/@types/react-native-reanimated-jest-util.d.ts

-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
declare namespace jest {
22
import Reanimated from 'react-native-reanimated';
3-
// eslint-disable-next-line import/no-duplicates
43
import {ImageStyle, TextStyle, ViewStyle} from 'react-native';
54

65
interface Matchers<R> {
@@ -14,7 +13,6 @@ declare namespace jest {
1413

1514
declare module 'react-native-reanimated/lib/reanimated2/jestUtils' {
1615
import {ReactTestInstance} from 'react-test-renderer';
17-
// eslint-disable-next-line import/no-duplicates
1816
import {ImageStyle, TextStyle, ViewStyle} from 'react-native';
1917
export const withReanimatedTimer: (test: () => unknown) => void;
2018
export const advanceAnimationByTime: (time: number) => void;

example-app/SantokuApp/src/apps/app/AppWithInitialization.tsx

+12-5
Original file line numberDiff line numberDiff line change
@@ -35,14 +35,21 @@ export const AppWithInitialization: React.FC = () => {
3535
Alert.alert(initializationResult.title, initializationResult.message);
3636
return null;
3737
} else {
38-
// RootStackNav、WithFirebaseMessagingHandlersをimportしてしまうと、アプリの初期化処理が完了する前に各画面でimportしているモジュールも読み込まれてしまうため、
39-
// アプリの初期化処理が完了した時点でrequireする。
40-
// requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
41-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
38+
/*
39+
eslint-disable-next-line @typescript-eslint/no-unsafe-member-access --
40+
RootStackNavをimportしてしまうと、アプリの初期化処理が完了する前に各画面でimportしているモジュールも読み込まれてしまうため、
41+
アプリの初期化処理が完了した時点でrequireする。
42+
requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
43+
*/
4244
const RootStackNav = require('./navigators/RootStackNav').RootStackNav as React.FC<
4345
React.PropsWithChildren<{initialData: AppInitialData}>
4446
>;
45-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
47+
/*
48+
eslint-disable-next-line @typescript-eslint/no-unsafe-member-access --
49+
WithFirebaseMessagingHandlersをimportしてしまうと、アプリの初期化処理が完了する前に各画面でimportしているモジュールも読み込まれてしまうため、
50+
アプリの初期化処理が完了した時点でrequireする。
51+
requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
52+
*/
4653
const FirebaseMessagingHandlers = require('./components/FirebaseMessagingHandlers')
4754
.FirebaseMessagingHandlers as React.FC<React.PropsWithChildren<{initialData: AppInitialData}>>;
4855

example-app/SantokuApp/src/apps/app/services/initializeFirebaseCrashlyticsAsync.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,11 @@ import {firebaseConfig} from 'bases/firebase/FirebaseConfig';
55

66
export const initializeFirebaseCrashlyticsAsync = async () => {
77
if (!firebaseConfig.isDummy) {
8-
// requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
9-
// 型は、typeofで明示するようにします。
10-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
8+
/*
9+
eslint-disable-next-line @typescript-eslint/no-unsafe-member-access --
10+
requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
11+
型は、typeofで明示するようにします。
12+
*/
1113
const crashlytics = require('@react-native-firebase/crashlytics').default as typeof crashlyticsModule;
1214
// アプリの起動IDを設定
1315
try {

example-app/SantokuApp/src/bases/core/errors/ApplicationError.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ class ApplicationErrorSubClass extends ApplicationError {}
44
const cause = new ApplicationErrorSubClass('root cause');
55
const nested = new ApplicationErrorSubClass('nested cause', cause);
66

7-
// eslint-disable-next-line jest/unbound-method
7+
// eslint-disable-next-line jest/unbound-method -- Error.captureStackTraceの復元(再代入)にしか使わないため
88
const captureStackTrace = Error.captureStackTrace;
99
describe.each([false, true])(
1010
'new ApplicationError() when captureStackTrace availability is %p',

example-app/SantokuApp/src/bases/core/errors/RuntimeError.test.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@ class RuntimeErrorSubClass extends RuntimeError {}
44
const cause = new RuntimeErrorSubClass('root cause');
55
const nested = new RuntimeErrorSubClass('nested cause', cause);
66

7-
// eslint-disable-next-line jest/unbound-method
7+
// eslint-disable-next-line jest/unbound-method -- Error.captureStackTraceの復元(再代入)にしか使わないため
88
const captureStackTrace = Error.captureStackTrace;
99
describe.each([false, true])(
1010
'new RuntimeError() when captureStackTrace availability is %p',

example-app/SantokuApp/src/bases/firebase/FirebaseConfig.test.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -29,10 +29,10 @@ const mockDummySettings = {
2929
describe('Firebase Config', () => {
3030
test('指定した名前でFirebaseインスタンスが生成されることの検証', () => {
3131
const spy = jest.spyOn(firebase, 'app');
32-
// eslint-disable-next-line no-new
32+
// eslint-disable-next-line no-new -- 作成したインスタンスは以降の検証で使用しないため
3333
new FirebaseConfig('testApp');
3434
expect(spy).toHaveBeenCalledWith('testApp');
35-
// eslint-disable-next-line no-new
35+
// eslint-disable-next-line no-new -- 作成したインスタンスは以降の検証で使用しないため
3636
new FirebaseConfig();
3737
expect(spy).toHaveBeenCalledWith(undefined);
3838
spy.mockRestore();

example-app/SantokuApp/src/bases/ui/barcode/Barcode.test.tsx

+1-2
Original file line numberDiff line numberDiff line change
@@ -13,8 +13,7 @@ const getSvgProps = (view: ReactTestInstance): SVGProps<any> => {
1313

1414
const getSvgPathProps = (view: ReactTestInstance): PathProps => {
1515
const svgView = view.children[0] as ReactTestInstance;
16-
// svgView.props.childrenがany型になってしまうため、型を強制的に指定
17-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
16+
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access -- svgView.props.childrenがany型になってしまうため、型を強制的に指定
1817
return svgView.props.children.props as PathProps;
1918
};
2019

example-app/SantokuApp/src/bases/ui/loading/Overlay.test.tsx

+3-8
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,6 @@ import {ReactTestInstance} from 'react-test-renderer';
55

66
import {Overlay} from './Overlay';
77

8-
// If advancing a timer changes the state of a component, the timer must be run within an act.
9-
// However, since act is `Thenable`, ESLint will issue a warning if you do not do something like await.
10-
// For convenience, disable the relevant rule in this file.
11-
/* eslint-disable @typescript-eslint/no-floating-promises */
12-
138
jest.useFakeTimers();
149

1510
function getStyle<T>(instance: ReactTestInstance) {
@@ -23,7 +18,7 @@ const ChildComponent = () => {
2318
const FADE_DURATION = 200;
2419

2520
describe('Overlay', () => {
26-
it('Overlayが正常にrenderできることを確認', () => {
21+
it('Overlayが正常にrenderできることを確認', async () => {
2722
render(
2823
<Overlay visible>
2924
<ChildComponent />
@@ -36,7 +31,7 @@ describe('Overlay', () => {
3631
expect(getStyle<ViewStyle>(screen.getByTestId('overlayAnimatedView')).opacity).toBe(0);
3732
expect(screen).toMatchSnapshot('render直後');
3833

39-
act(() => {
34+
await act(() => {
4035
jest.advanceTimersByTime(FADE_DURATION);
4136
});
4237

@@ -49,7 +44,7 @@ describe('Overlay', () => {
4944
</Overlay>,
5045
);
5146

52-
act(() => {
47+
await act(() => {
5348
jest.advanceTimersByTime(FADE_DURATION);
5449
});
5550

example-app/SantokuApp/src/bases/ui/overlay/FullWindowOverlay.tsx

+5-3
Original file line numberDiff line numberDiff line change
@@ -9,9 +9,11 @@ import {Platform, View} from 'react-native';
99
*/
1010
export const FullWindowOverlay = (props: {children: React.ReactNode}) => {
1111
if (Platform.OS === 'ios') {
12-
// iOS以外の場合にreact-native-screensのFullWindowOverlayを使用すると警告が出るので、iOSの場合のみ使用します。
13-
// requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
14-
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
12+
/*
13+
eslint-disable-next-line @typescript-eslint/no-unsafe-member-access --
14+
iOS以外の場合にreact-native-screensのFullWindowOverlayを使用すると警告が出るので、iOSの場合のみ使用します。
15+
requireした場合の型はanyとなってしまいESLintエラーが発生しますが無視します。
16+
*/
1517
const RNSFullWindowOverlay = require('react-native-screens').FullWindowOverlay as typeof View;
1618
return <RNSFullWindowOverlay testID="FullWindowOverlay">{props.children}</RNSFullWindowOverlay>;
1719
}

example-app/SantokuApp/src/bases/ui/overlay/OverlayBackdrop.test.tsx

-5
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,6 @@ import {SafeAreaProvider} from 'react-native-safe-area-context';
66

77
import {OVERLAY_BACKDROP_DEFAULT_ENTERING, OVERLAY_BACKDROP_DEFAULT_EXITING, OverlayBackdrop} from './OverlayBackdrop';
88

9-
// If advancing a timer changes the state of a component, the timer must be run within an act.
10-
// However, since act is `Thenable`, ESLint will issue a warning if you do not do something like await.
11-
// For convenience, disable the relevant rule in this file.
12-
/* eslint-disable @typescript-eslint/no-floating-promises */
13-
149
jest.useFakeTimers();
1510

1611
// TODO: Jest v27にアップデートできたら、withReanimatedTimerでテストを実装できるか検証する。

example-app/SantokuApp/src/bases/ui/overlay/OverlayContainer.test.tsx

-5
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,6 @@ import {SafeAreaProvider} from 'react-native-safe-area-context';
66

77
import {MODAL_CONTAINER_DEFAULT_ENTERING, MODAL_CONTAINER_DEFAULT_EXITING, OverlayContainer} from './OverlayContainer';
88

9-
// If advancing a timer changes the state of a component, the timer must be run within an act.
10-
// However, since act is `Thenable`, ESLint will issue a warning if you do not do something like await.
11-
// For convenience, disable the relevant rule in this file.
12-
/* eslint-disable @typescript-eslint/no-floating-promises */
13-
149
jest.useFakeTimers();
1510

1611
// TODO: Jest v27にアップデートできたら、withReanimatedTimerでテストを実装できるか検証する。

example-app/SantokuApp/src/bases/ui/picker/PickerBackdrop.test.tsx

+8-12
Original file line numberDiff line numberDiff line change
@@ -4,11 +4,6 @@ import {ModalProps, PressableProps} from 'react-native';
44

55
import {DEFAULT_FADE_OUT_DURATION, PickerBackdrop} from './PickerBackdrop';
66

7-
// If advancing a timer changes the state of a component, the timer must be run within an act.
8-
// However, since act is `Thenable`, ESLint will issue a warning if you do not do something like await.
9-
// For convenience, disable the relevant rule in this file.
10-
/* eslint-disable @typescript-eslint/no-floating-promises */
11-
127
jest.useFakeTimers();
138

149
// TODO: Jest v27にアップデートできたら、withReanimatedTimerでテストを実装できるか検証する。
@@ -21,7 +16,7 @@ describe('PickerBackdrop only with required props', () => {
2116
expect(screen.toJSON()).toBeNull();
2217
});
2318

24-
it('renders successfully only with required props', () => {
19+
it('renders successfully only with required props', async () => {
2520
render(<PickerBackdrop isVisible pressableProps={{testID: 'pressable'}} />);
2621
//////////////////////////////////////////////////////////////////////////////////
2722
// 初期表示
@@ -35,7 +30,8 @@ describe('PickerBackdrop only with required props', () => {
3530
// 非表示にする
3631
//////////////////////////////////////////////////////////////////////////////////
3732
screen.update(<PickerBackdrop isVisible={false} pressableProps={{testID: 'pressable'}} />);
38-
act(() => {
33+
34+
await act(() => {
3935
jest.advanceTimersByTime(DEFAULT_FADE_OUT_DURATION);
4036
});
4137
const pressable2 = screen.queryByTestId('pressable');
@@ -63,7 +59,7 @@ describe('PickerBackdrop with `onPress', () => {
6359
});
6460
//
6561
describe('PickerBackdrop with all props', () => {
66-
it('should be applied properly', () => {
62+
it('should be applied properly', async () => {
6763
const onPress = jest.fn();
6864
const onLongPress = jest.fn();
6965
const afterFadeIn = jest.fn();
@@ -113,12 +109,12 @@ describe('PickerBackdrop with all props', () => {
113109
expect(onPress).toHaveBeenCalledTimes(1);
114110

115111
// fadeInDurationで指定した時間の1msc前ではafterFadeInは実行されない
116-
act(() => {
112+
await act(() => {
117113
jest.advanceTimersByTime(199);
118114
});
119115
expect(afterFadeIn).not.toHaveBeenCalled();
120116
// fadeInDurationで指定した時間経過後は、afterFadeInが実行される
121-
act(() => {
117+
await act(() => {
122118
jest.advanceTimersByTime(1);
123119
});
124120
expect(afterFadeIn).toHaveBeenCalled();
@@ -147,12 +143,12 @@ describe('PickerBackdrop with all props', () => {
147143
screen.update(<PickerBackdrop isVisible={false} afterFadeOut={afterFadeOut} fadeOutDuration={100} />);
148144

149145
// fadeOutDurationで指定した時間の1msc前ではafterFadeOutは実行されない
150-
act(() => {
146+
await act(() => {
151147
jest.advanceTimersByTime(99);
152148
});
153149
expect(afterFadeOut).not.toHaveBeenCalled();
154150
// fadeOutDurationで指定した時間経過後は、afterFadeOutが実行される
155-
act(() => {
151+
await act(() => {
156152
jest.advanceTimersByTime(1);
157153
});
158154
expect(afterFadeOut).toHaveBeenCalled();

0 commit comments

Comments
 (0)