-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy path.eslintrc.js
216 lines (216 loc) · 6.49 KB
/
.eslintrc.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
module.exports = {
// `extends`를 생략해도 이 파일이 있는 위치까지만 부모 eslintrc를 찾도록 제한함
env: {
es6: true,
node: true,
},
parser: '@typescript-eslint/parser',
extends: [
'eslint:recommended',
'plugin:@next/next/recommended',
'plugin:@typescript-eslint/recommended',
'eslint-config-prettier',
'plugin:storybook/recommended',
],
plugins: [
'@typescript-eslint',
'import',
'prettier',
'react',
'react-hooks',
'jsx-a11y',
],
settings: {
'import/resolver': {
typescript: {},
},
},
rules: {
'prettier/prettier': [
'error',
{
endOfLine: 'auto',
},
],
'react-hooks/exhaustive-deps': 'error',
'no-implicit-coercion': 'error',
// TypeScript에서 이미 잡고 있는 문제이기 때문에 + location, document 등의 global variable도 잡아서
'no-undef': 'off',
// 아래 3개의 경우, Prettier가 이미 잘 해 주고 있는 부분이기 때문에
indent: 'off',
'@typescript-eslint/indent': 'off',
semi: 'off',
// 이미 널리 쓰이고 있어 에러 수가 감당이 되지 않아 잠시 꺼둡니다.
'@typescript-eslint/no-non-null-assertion': 'off',
'@typescript-eslint/no-explicit-any': 'off',
// Strict-boolean-expression을 사용할지 아직 결론이 나지 않아서
'no-extra-boolean-cast': 'off',
// union type을 받는 함수에서 모든 경우의 수에 대해 return 해도 eslint가 추론하지 못하고 있어서 warn만 하고 있음
'getter-return': 'warn',
// 대부분의 경우 필요가 없어서
'@typescript-eslint/explicit-function-return-type': 'off',
// Hoisting을 전략적으로 사용한 경우가 많아서
'@typescript-eslint/no-use-before-define': 'off',
// 모델 정의 부분에서 class와 interface를 합치기 위해 사용하는 용법도 잡고 있어서
'@typescript-eslint/no-empty-interface': 'off',
// 모델 정의 부분에서 파라미터 프로퍼티를 잘 쓰고 있어서
'@typescript-eslint/no-parameter-properties': 'off',
// TypeScript에서 이미 잘 해주고 있어서
'react/prop-types': 'off',
// React.memo, React.forwardRef에서 사용하는 경우도 막고 있어서
'react/display-name': 'off',
// 탭내빙 어택 방지
'react/jsx-no-target-blank': 'error',
// 불필요한 Fragment 방지
'react/jsx-no-useless-fragment': 'warn',
'no-async-promise-executor': 'warn',
'@typescript-eslint/prefer-as-const': 'warn',
'@typescript-eslint/no-non-null-asserted-optional-chain': 'warn',
'@typescript-eslint/ban-types': 'warn',
'@typescript-eslint/no-inferrable-types': 'warn',
'@typescript-eslint/no-empty-function': 'off',
'@typescript-eslint/naming-convention': [
'error',
{
format: ['camelCase', 'UPPER_CASE', 'PascalCase'],
selector: 'variable',
leadingUnderscore: 'allow',
},
{
format: ['camelCase', 'PascalCase'],
selector: 'function',
filter: {
regex: '[ㄱ-ㅎ|ㅏ-ㅣ|가-힣]',
match: false,
},
},
{
format: ['PascalCase'],
selector: 'interface',
filter: {
regex: '[ㄱ-ㅎ|ㅏ-ㅣ|가-힣]',
match: false,
},
},
{
format: ['PascalCase'],
selector: 'typeAlias',
filter: {
regex: '[ㄱ-ㅎ|ㅏ-ㅣ|가-힣]',
match: false,
},
},
],
'@typescript-eslint/explicit-module-boundary-types': 'off',
'@typescript-eslint/array-type': [
'error',
{
default: 'array-simple',
},
],
'@typescript-eslint/no-unused-vars': [
'error',
{
ignoreRestSiblings: true,
},
],
'@typescript-eslint/member-ordering': [
'error',
{
default: [
'public-static-field',
'private-static-field',
'public-instance-field',
'private-instance-field',
'public-constructor',
'private-constructor',
'public-instance-method',
'private-instance-method',
],
},
],
'object-shorthand': ['error', 'always'],
'prefer-const': 'error',
'no-var': 'error',
curly: ['error', 'all'],
eqeqeq: [
'error',
'always',
{
null: 'ignore',
},
],
'import/no-duplicates': 'error',
'import/order': [
'error',
{
groups: [
'builtin',
'external',
'internal',
['index', 'sibling', 'parent'],
'object',
],
// Import 구문 순서 정하고 싶으면 추가
pathGroups: [],
pathGroupsExcludedImportTypes: ['builtin'],
'newlines-between': 'always',
alphabetize: {
order: 'asc',
caseInsensitive: true,
},
},
],
'sort-imports': [
'error',
{
ignoreDeclarationSort: true,
ignoreMemberSort: false,
},
],
'import/newline-after-import': ['error'],
'react-hooks/rules-of-hooks': 'error',
'react/jsx-uses-react': 'off',
'react/react-in-jsx-scope': 'off',
'react/jsx-key': 'warn',
// 접근성, 충분히 잡히면 error로 올릴 예정
'jsx-a11y/alt-text': [
'warn',
{
img: ['Image', 'Dialog.Image'],
},
],
'jsx-a11y/aria-props': 'error',
// 제대로된 aria-* 이름인지 확인
'jsx-a11y/aria-proptypes': 'error',
// 제대로된 aria-*의 value 인지 검사
/**
* 일부 속성은 aria 표준이 아니지만 사용중이기 때문에 warn 처리
* eg. role="text" 같은 경우 aria 표준은 아니지만 널리 사용되고 있습니다.
*
* FIXME: 아래 이슈 해결후 error로 격상 하기
* https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/issues/792
*/
'jsx-a11y/aria-role': 'warn',
// 표준 role 값인지 확인하는 룰
'jsx-a11y/role-supports-aria-props': 'warn',
// 서로 같이 쓰면 안되는 aria-* 가 있는지 확인하는룰
'jsx-a11y/aria-unsupported-elements': 'warn',
// 해당 element에서 미지원하는 aria가 들어있는지 확인하는 룰
'jsx-a11y/img-redundant-alt': [
'warn',
{
components: ['Image', 'Dialog.Image'],
words: ['사진', '이미지'],
},
],
},
overrides: [
{
files: ['**/*.js', 'scripts/**/*.ts'],
rules: {
'@typescript-eslint/no-var-requires': 'warn',
},
},
],
};