Skip to content

Commit 521a3cb

Browse files
authored
Simplify ESLint Config (#261)
1 parent da0c6b7 commit 521a3cb

File tree

11 files changed

+159
-77
lines changed

11 files changed

+159
-77
lines changed

.eslintrc.js

+128-15
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,135 @@
1+
/**
2+
* This is intended to be a basic starting point for linting in the Indie Stack.
3+
* It relies on recommended configs out of the box for simplicity, but you can
4+
* and should modify this configuration to best suit your team's needs.
5+
*/
6+
17
/** @type {import('eslint').Linter.Config} */
28
module.exports = {
39
root: true,
4-
extends: [
5-
"@remix-run/eslint-config",
6-
"@remix-run/eslint-config/node",
7-
"@remix-run/eslint-config/jest-testing-library",
8-
"prettier",
9-
],
10+
parserOptions: {
11+
ecmaVersion: "latest",
12+
sourceType: "module",
13+
ecmaFeatures: {
14+
jsx: true,
15+
},
16+
},
1017
env: {
11-
"cypress/globals": true,
18+
browser: true,
19+
commonjs: true,
20+
es6: true,
1221
},
13-
plugins: ["cypress"],
14-
// we're using vitest which has a very similar API to jest
15-
// (so the linting plugins work nicely), but it means we have to explicitly
16-
// set the jest version.
17-
settings: {
18-
jest: {
19-
version: 28,
22+
23+
// Base config
24+
extends: ["eslint:recommended"],
25+
26+
overrides: [
27+
// React
28+
{
29+
files: ["**/*.{js,jsx,ts,tsx}"],
30+
plugins: ["react", "jsx-a11y"],
31+
extends: [
32+
"plugin:react/recommended",
33+
"plugin:react/jsx-runtime",
34+
"plugin:jsx-a11y/recommended",
35+
"prettier",
36+
],
37+
settings: {
38+
react: {
39+
version: "detect",
40+
},
41+
formComponents: ["Form"],
42+
linkComponents: [
43+
{ name: "Link", linkAttribute: "to" },
44+
{ name: "NavLink", linkAttribute: "to" },
45+
],
46+
},
47+
rules: {
48+
"react/jsx-no-leaked-render": [
49+
"warn",
50+
{ validStrategies: ["ternary"] },
51+
],
52+
},
2053
},
21-
},
54+
55+
// Typescript
56+
{
57+
files: ["**/*.{ts,tsx}"],
58+
plugins: ["@typescript-eslint", "import"],
59+
parser: "@typescript-eslint/parser",
60+
settings: {
61+
"import/internal-regex": "^~/",
62+
"import/resolver": {
63+
node: {
64+
extensions: [".ts", ".tsx"],
65+
},
66+
typescript: {
67+
alwaysTryTypes: true,
68+
},
69+
},
70+
},
71+
extends: [
72+
"plugin:@typescript-eslint/recommended",
73+
"plugin:@typescript-eslint/stylistic",
74+
"plugin:import/recommended",
75+
"plugin:import/typescript",
76+
"prettier",
77+
],
78+
rules: {
79+
"import/order": [
80+
"error",
81+
{
82+
alphabetize: { caseInsensitive: true, order: "asc" },
83+
groups: ["builtin", "external", "internal", "parent", "sibling"],
84+
"newlines-between": "always",
85+
},
86+
],
87+
},
88+
},
89+
90+
// Markdown
91+
{
92+
files: ["**/*.md"],
93+
plugins: ["markdown"],
94+
extends: ["plugin:markdown/recommended", "prettier"],
95+
},
96+
97+
// Jest/Vitest
98+
{
99+
files: ["**/*.test.{js,jsx,ts,tsx}"],
100+
plugins: ["jest", "jest-dom", "testing-library"],
101+
extends: [
102+
"plugin:jest/recommended",
103+
"plugin:jest-dom/recommended",
104+
"plugin:testing-library/react",
105+
"prettier",
106+
],
107+
env: {
108+
"jest/globals": true,
109+
},
110+
settings: {
111+
jest: {
112+
// we're using vitest which has a very similar API to jest
113+
// (so the linting plugins work nicely), but it means we have to explicitly
114+
// set the jest version.
115+
version: 28,
116+
},
117+
},
118+
},
119+
120+
// Cypress
121+
{
122+
files: ["cypress/**/*.ts"],
123+
plugins: ["cypress"],
124+
extends: ["plugin:cypress/recommended", "prettier"],
125+
},
126+
127+
// Node
128+
{
129+
files: [".eslintrc.js", "mocks/**/*.js"],
130+
env: {
131+
node: true,
132+
},
133+
},
134+
],
22135
};

.eslintrc.repo.js

-32
This file was deleted.

.github/workflows/lint-repo.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,4 +30,4 @@ jobs:
3030
run: npm install
3131

3232
- name: 🔬 Lint
33-
run: npm run lint:repo
33+
run: npm run lint

app/models/user.server.ts

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ export async function verifyLogin(
5656
return null;
5757
}
5858

59+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
5960
const { password: _password, ...userWithoutPassword } = userWithPassword;
6061

6162
return userWithoutPassword;

app/routes/join.tsx

+1
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,7 @@ export default function Join() {
100100
ref={emailRef}
101101
id="email"
102102
required
103+
// eslint-disable-next-line jsx-a11y/no-autofocus
103104
autoFocus={true}
104105
name="email"
105106
type="email"

app/routes/login.tsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,7 @@ export default function LoginPage() {
9595
ref={emailRef}
9696
id="email"
9797
required
98+
// eslint-disable-next-line jsx-a11y/no-autofocus
9899
autoFocus={true}
99100
name="email"
100101
type="email"
@@ -160,7 +161,7 @@ export default function LoginPage() {
160161
</label>
161162
</div>
162163
<div className="text-center text-sm text-gray-500">
163-
Don't have an account?{" "}
164+
Don&apos;t have an account?{" "}
164165
<Link
165166
className="text-blue-500 underline"
166167
to={{

app/singleton.server.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,8 @@ export const singleton = <Value>(
66
name: string,
77
valueFactory: () => Value,
88
): Value => {
9-
const g = global as any;
9+
const g = global as unknown as { __singletons: Record<string, unknown> };
1010
g.__singletons ??= {};
1111
g.__singletons[name] ??= valueFactory();
12-
return g.__singletons[name];
12+
return g.__singletons[name] as Value;
1313
};

app/utils.ts

+7-2
Original file line numberDiff line numberDiff line change
@@ -44,8 +44,13 @@ export function useMatchesData(
4444
return route?.data as Record<string, unknown>;
4545
}
4646

47-
function isUser(user: any): user is User {
48-
return user && typeof user === "object" && typeof user.email === "string";
47+
function isUser(user: unknown): user is User {
48+
return (
49+
user != null &&
50+
typeof user === "object" &&
51+
"email" in user &&
52+
typeof user.email === "string"
53+
);
4954
}
5055

5156
export function useOptionalUser(): User | undefined {

cypress/support/commands.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import { faker } from "@faker-js/faker";
22

33
declare global {
4+
// eslint-disable-next-line @typescript-eslint/no-namespace
45
namespace Cypress {
56
interface Chainable {
67
/**
@@ -85,7 +86,7 @@ function deleteUserByEmail(email: string) {
8586
// Also added custom types to avoid getting detached
8687
// https://github.com/cypress-io/cypress/issues/7306#issuecomment-1152752612
8788
// ===========================================================
88-
function visitAndCheck(url: string, waitTime: number = 1000) {
89+
function visitAndCheck(url: string, waitTime = 1000) {
8990
cy.visit(url);
9091
cy.location("pathname").should("contain", url).wait(waitTime);
9192
}

package.json

+13-7
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,8 @@
77
"dev": "remix dev -c \"npm run dev:serve\"",
88
"dev:serve": "binode --require ./mocks -- @remix-run/serve:remix-serve ./build/index.js",
99
"format": "prettier --write .",
10-
"format:repo": "npm run format && npm run lint:repo -- --fix",
10+
"format:repo": "npm run format && npm run lint -- --fix",
1111
"lint": "eslint --cache --cache-location ./node_modules/.cache/eslint .",
12-
"lint:repo": "npm run lint -- --config .eslintrc.repo.js",
1312
"setup": "prisma generate && prisma migrate deploy && prisma db seed",
1413
"start": "remix-serve ./build/index.js",
1514
"start:mocks": "binode --require ./mocks -- @remix-run/serve:remix-serve ./build/index.js",
@@ -41,26 +40,33 @@
4140
"devDependencies": {
4241
"@faker-js/faker": "^8.0.2",
4342
"@remix-run/dev": "*",
44-
"@remix-run/eslint-config": "*",
4543
"@testing-library/cypress": "^9.0.0",
4644
"@testing-library/jest-dom": "^5.17.0",
4745
"@types/bcryptjs": "^2.4.2",
4846
"@types/eslint": "^8.44.2",
4947
"@types/node": "^18.17.6",
5048
"@types/react": "^18.2.20",
5149
"@types/react-dom": "^18.2.7",
50+
"@typescript-eslint/eslint-plugin": "^6.7.3",
51+
"@typescript-eslint/parser": "^6.7.3",
5252
"@vitejs/plugin-react": "^4.0.4",
5353
"@vitest/coverage-v8": "^0.34.2",
5454
"autoprefixer": "^10.4.15",
5555
"binode": "^1.0.5",
5656
"cookie": "^0.5.0",
5757
"cross-env": "^7.0.3",
5858
"cypress": "12.17.3",
59-
"eslint": "^8.47.0",
59+
"eslint": "^8.50.0",
6060
"eslint-config-prettier": "^9.0.0",
61-
"eslint-plugin-cypress": "^2.14.0",
61+
"eslint-import-resolver-typescript": "^3.6.1",
62+
"eslint-plugin-cypress": "^2.15.1",
63+
"eslint-plugin-import": "^2.28.1",
64+
"eslint-plugin-jest": "^27.4.0",
65+
"eslint-plugin-jest-dom": "^5.1.0",
66+
"eslint-plugin-jsx-a11y": "^6.7.1",
6267
"eslint-plugin-markdown": "^3.0.1",
63-
"eslint-plugin-prefer-let": "^3.0.1",
68+
"eslint-plugin-react": "^7.33.2",
69+
"eslint-plugin-testing-library": "^6.0.2",
6470
"happy-dom": "^10.10.4",
6571
"msw": "^1.2.3",
6672
"npm-run-all": "^4.1.5",
@@ -72,7 +78,7 @@
7278
"tailwindcss": "^3.3.3",
7379
"ts-node": "^10.9.1",
7480
"tsconfig-paths": "^4.2.0",
75-
"typescript": "^5.1.6",
81+
"typescript": "^5.2.2",
7682
"vite": "^4.4.9",
7783
"vite-tsconfig-paths": "^3.6.0",
7884
"vitest": "^0.34.2"

remix.init/index.js

+2-16
Original file line numberDiff line numberDiff line change
@@ -56,30 +56,17 @@ const getPackageManagerVersion = (packageManager) =>
5656

5757
const getRandomString = (length) => crypto.randomBytes(length).toString("hex");
5858

59-
const removeUnusedDependencies = (dependencies, unusedDependencies) =>
60-
Object.fromEntries(
61-
Object.entries(dependencies).filter(
62-
([key]) => !unusedDependencies.includes(key),
63-
),
64-
);
65-
6659
const updatePackageJson = ({ APP_NAME, packageJson }) => {
6760
const {
68-
devDependencies,
6961
scripts: {
62+
// eslint-disable-next-line no-unused-vars
7063
"format:repo": _repoFormatScript,
71-
"lint:repo": _repoLintScript,
7264
...scripts
7365
},
7466
} = packageJson.content;
7567

7668
packageJson.update({
7769
name: APP_NAME,
78-
devDependencies: removeUnusedDependencies(
79-
devDependencies,
80-
// packages that are only used for linting the repo
81-
["eslint-plugin-markdown", "eslint-plugin-prefer-let"],
82-
),
8370
scripts,
8471
});
8572
};
@@ -190,13 +177,12 @@ const main = async ({ packageManager, rootDirectory }) => {
190177
fs.rm(path.join(rootDirectory, ".github", "workflows", "no-response.yml")),
191178
fs.rm(path.join(rootDirectory, ".github", "dependabot.yml")),
192179
fs.rm(path.join(rootDirectory, ".github", "PULL_REQUEST_TEMPLATE.md")),
193-
fs.rm(path.join(rootDirectory, ".eslintrc.repo.js")),
194180
fs.rm(path.join(rootDirectory, "LICENSE.md")),
195181
]);
196182

197183
execSync(pm.run("setup"), { cwd: rootDirectory, stdio: "inherit" });
198184

199-
execSync(pm.run("format", "--loglevel warn"), {
185+
execSync(pm.run("format", "--log-level warn"), {
200186
cwd: rootDirectory,
201187
stdio: "inherit",
202188
});

0 commit comments

Comments
 (0)