eslint-plugin-esm 0.2.0 → 0.2.1

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,11 @@
1
1
  # eslint-plugin-esm
2
2
 
3
+ ## 0.2.1
4
+
5
+ ### Patch Changes
6
+
7
+ - 2b04bbd: fix(eslint-plugin-esm): allow module existing in peerDependencies for `no-phantom-dep-imports` rule
8
+
3
9
  ## 0.2.0
4
10
 
5
11
  ### Minor Changes
package/README.md CHANGED
@@ -1,6 +1,6 @@
1
1
  # eslint-plugin-esm
2
2
 
3
- [![](https://img.shields.io/npm/l/eslint-plugin-esm.svg)](https://github.com/zanminkian/git-validator/blob/main/LICENSE)
3
+ [![](https://img.shields.io/npm/l/eslint-plugin-esm.svg)](https://github.com/zanminkian/fenge/blob/main/LICENSE)
4
4
  [![](https://img.shields.io/npm/v/eslint-plugin-esm.svg)](https://www.npmjs.com/package/eslint-plugin-esm)
5
5
  [![](https://img.shields.io/npm/dm/eslint-plugin-esm.svg)](https://www.npmjs.com/package/eslint-plugin-esm)
6
6
  [![](https://packagephobia.com/badge?p=eslint-plugin-esm)](https://packagephobia.com/result?p=eslint-plugin-esm)
@@ -8,6 +8,12 @@
8
8
 
9
9
  ESLint plugin for linting ESM (import/export syntax)
10
10
 
11
+ ## Feature
12
+
13
+ - Faster than `eslint-plugin-import`.
14
+ - Easier than `eslint-plugin-import`
15
+ - Zero dependencies.
16
+
11
17
  ## Requirement
12
18
 
13
19
  - ESLint >= 8.57.0
@@ -33,7 +39,7 @@ export default [
33
39
  rules: {
34
40
  "esm/no-git-ignored-imports": "error"
35
41
  ...
36
- // Visit https://github.com/zanminkian/git-validator/tree/main/packages/eslint-plugin-esm/doc/rules for more other rules
42
+ // Visit https://github.com/zanminkian/fenge/tree/main/packages/eslint-plugin-esm/doc/rules for more other rules
37
43
  },
38
44
  },
39
45
  ...
@@ -53,7 +59,7 @@ Config `package.json`
53
59
 
54
60
  ## Rules
55
61
 
56
- Click [here](https://github.com/zanminkian/git-validator/tree/main/packages/eslint-plugin-esm/doc/rules).
62
+ Click [here](https://github.com/zanminkian/fenge/tree/main/packages/eslint-plugin-esm/doc/rules).
57
63
 
58
64
  ## License
59
65
 
@@ -1 +1 @@
1
- {"version":3,"file":"no-phantom-dep-imports.d.ts","sourceRoot":"","sources":["../../src/rules/no-phantom-dep-imports.ts"],"names":[],"mappings":"AA4CA,eAAO,MAAM,mBAAmB;;;CA8D9B,CAAC"}
1
+ {"version":3,"file":"no-phantom-dep-imports.d.ts","sourceRoot":"","sources":["../../src/rules/no-phantom-dep-imports.ts"],"names":[],"mappings":"AA4CA,eAAO,MAAM,mBAAmB;;;CA+D9B,CAAC"}
@@ -49,7 +49,7 @@ export const noPhantomDepImports = createRule({
49
49
  ],
50
50
  create: (context) => create(context, (filename, source, node) => {
51
51
  const { allowDevDependencies = false, } = context.options[0] ?? {};
52
- // ignore `import {foo} from './'`
52
+ // ignore `import {foo} from './'` and `import {foo} from 'node:foo'`
53
53
  if (getSourceType(source) !== "module") {
54
54
  return false;
55
55
  }
@@ -62,6 +62,10 @@ export const noPhantomDepImports = createRule({
62
62
  isObject(pkgJson.content.dependencies)
63
63
  ? pkgJson.content.dependencies
64
64
  : {};
65
+ const peerDep = "peerDependencies" in pkgJson.content &&
66
+ isObject(pkgJson.content.peerDependencies)
67
+ ? pkgJson.content.peerDependencies
68
+ : {};
65
69
  const devDep = "devDependencies" in pkgJson.content &&
66
70
  isObject(pkgJson.content.devDependencies)
67
71
  ? pkgJson.content.devDependencies
@@ -70,20 +74,18 @@ export const noPhantomDepImports = createRule({
70
74
  .split("/")
71
75
  .slice(0, source.startsWith("@") ? 2 : 1)
72
76
  .join("/");
77
+ const isInDep = moduleName in dep || moduleName in peerDep;
78
+ const isInDev = moduleName in devDep;
73
79
  if ("importKind" in node && node.importKind === "type") {
74
80
  return moduleName.startsWith("@") && moduleName.includes("/")
75
- ? !(moduleName in dep ||
76
- moduleName in devDep ||
81
+ ? !(isInDep ||
82
+ isInDev ||
77
83
  `@types/${moduleName.slice(1).replace("/", "_")}` in devDep)
78
- : !(moduleName in dep ||
79
- moduleName in devDep ||
80
- `@types/${moduleName}` in devDep);
84
+ : !(isInDep || isInDev || `@types/${moduleName}` in devDep);
81
85
  }
82
86
  else {
83
- return allowDevDependencies
84
- ? !(moduleName in dep || moduleName in devDep)
85
- : !(moduleName in dep);
87
+ return allowDevDependencies ? !(isInDep || isInDev) : !isInDep;
86
88
  }
87
89
  }),
88
90
  });
89
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"no-phantom-dep-imports.js","sourceRoot":"","sources":["../../src/rules/no-phantom-dep-imports.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\nimport { create, createRule, getRuleName, getSourceType } from \"../common.js\";\n\nfunction isObject(value: unknown) {\n  return value !== null && typeof value === \"object\";\n}\n\nfunction isFile(filePath: string) {\n  try {\n    return fs.statSync(filePath).isFile();\n  } catch (e) {\n    return false;\n  }\n}\n\nconst cache = new Map<string, { path: string; content: object } | undefined>(); // key is dir, value is package.json\nfunction getPkgJson(\n  dir: string,\n): { path: string; content: object } | undefined {\n  if (cache.has(dir)) {\n    return cache.get(dir);\n  }\n  const pkgJsonPath = path.join(dir, \"package.json\");\n  if (isFile(pkgJsonPath)) {\n    const content: unknown = JSON.parse(fs.readFileSync(pkgJsonPath, \"utf8\"));\n    const result = isObject(content)\n      ? { path: pkgJsonPath, content }\n      : undefined;\n    cache.set(dir, result);\n    return result;\n  }\n\n  // if it is a directory\n  if (dir === process.cwd() || dir === \"/\") {\n    // stop here\n    cache.set(dir, undefined);\n    return undefined;\n  }\n\n  return getPkgJson(path.join(dir, \"..\"));\n}\n\nexport const noPhantomDepImports = createRule({\n  name: getRuleName(import.meta.url),\n  message:\n    \"Disallow importing from a module which the nearest `package.json` doesn't include it.\",\n  schema: [\n    {\n      type: \"object\",\n      properties: {\n        allowDevDependencies: { type: \"boolean\" },\n      },\n      additionalProperties: false,\n    },\n  ],\n  create: (context) =>\n    create(context, (filename, source, node) => {\n      const {\n        allowDevDependencies = false,\n      }: { allowDevDependencies: boolean } = context.options[0] ?? {};\n\n      // ignore `import {foo} from './'`\n      if (getSourceType(source) !== \"module\") {\n        return false;\n      }\n      const pkgJson = getPkgJson(path.dirname(filename));\n      // cannot find package.json file\n      if (!pkgJson) {\n        return true;\n      }\n      const dep =\n        \"dependencies\" in pkgJson.content &&\n        isObject(pkgJson.content.dependencies)\n          ? pkgJson.content.dependencies\n          : {};\n      const devDep =\n        \"devDependencies\" in pkgJson.content &&\n        isObject(pkgJson.content.devDependencies)\n          ? pkgJson.content.devDependencies\n          : {};\n\n      const moduleName = source\n        .split(\"/\")\n        .slice(0, source.startsWith(\"@\") ? 2 : 1)\n        .join(\"/\");\n\n      if (\"importKind\" in node && node.importKind === \"type\") {\n        return moduleName.startsWith(\"@\") && moduleName.includes(\"/\")\n          ? !(\n              moduleName in dep ||\n              moduleName in devDep ||\n              `@types/${moduleName.slice(1).replace(\"/\", \"_\")}` in devDep\n            )\n          : !(\n              moduleName in dep ||\n              moduleName in devDep ||\n              `@types/${moduleName}` in devDep\n            );\n      } else {\n        return allowDevDependencies\n          ? !(moduleName in dep || moduleName in devDep)\n          : !(moduleName in dep);\n      }\n    }),\n});\n"]}
91
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"no-phantom-dep-imports.js","sourceRoot":"","sources":["../../src/rules/no-phantom-dep-imports.ts"],"names":[],"mappings":"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","sourcesContent":["import fs from \"node:fs\";\nimport path from \"node:path\";\nimport process from \"node:process\";\nimport { create, createRule, getRuleName, getSourceType } from \"../common.js\";\n\nfunction isObject(value: unknown) {\n  return value !== null && typeof value === \"object\";\n}\n\nfunction isFile(filePath: string) {\n  try {\n    return fs.statSync(filePath).isFile();\n  } catch (e) {\n    return false;\n  }\n}\n\nconst cache = new Map<string, { path: string; content: object } | undefined>(); // key is dir, value is package.json\nfunction getPkgJson(\n  dir: string,\n): { path: string; content: object } | undefined {\n  if (cache.has(dir)) {\n    return cache.get(dir);\n  }\n  const pkgJsonPath = path.join(dir, \"package.json\");\n  if (isFile(pkgJsonPath)) {\n    const content: unknown = JSON.parse(fs.readFileSync(pkgJsonPath, \"utf8\"));\n    const result = isObject(content)\n      ? { path: pkgJsonPath, content }\n      : undefined;\n    cache.set(dir, result);\n    return result;\n  }\n\n  // if it is a directory\n  if (dir === process.cwd() || dir === \"/\") {\n    // stop here\n    cache.set(dir, undefined);\n    return undefined;\n  }\n\n  return getPkgJson(path.join(dir, \"..\"));\n}\n\nexport const noPhantomDepImports = createRule({\n  name: getRuleName(import.meta.url),\n  message:\n    \"Disallow importing from a module which the nearest `package.json` doesn't include it.\",\n  schema: [\n    {\n      type: \"object\",\n      properties: {\n        allowDevDependencies: { type: \"boolean\" },\n      },\n      additionalProperties: false,\n    },\n  ],\n  create: (context) =>\n    create(context, (filename, source, node) => {\n      const {\n        allowDevDependencies = false,\n      }: { allowDevDependencies: boolean } = context.options[0] ?? {};\n\n      // ignore `import {foo} from './'` and `import {foo} from 'node:foo'`\n      if (getSourceType(source) !== \"module\") {\n        return false;\n      }\n      const pkgJson = getPkgJson(path.dirname(filename));\n      // cannot find package.json file\n      if (!pkgJson) {\n        return true;\n      }\n      const dep =\n        \"dependencies\" in pkgJson.content &&\n        isObject(pkgJson.content.dependencies)\n          ? pkgJson.content.dependencies\n          : {};\n      const peerDep =\n        \"peerDependencies\" in pkgJson.content &&\n        isObject(pkgJson.content.peerDependencies)\n          ? pkgJson.content.peerDependencies\n          : {};\n      const devDep =\n        \"devDependencies\" in pkgJson.content &&\n        isObject(pkgJson.content.devDependencies)\n          ? pkgJson.content.devDependencies\n          : {};\n\n      const moduleName = source\n        .split(\"/\")\n        .slice(0, source.startsWith(\"@\") ? 2 : 1)\n        .join(\"/\");\n\n      const isInDep = moduleName in dep || moduleName in peerDep;\n      const isInDev = moduleName in devDep;\n      if (\"importKind\" in node && node.importKind === \"type\") {\n        return moduleName.startsWith(\"@\") && moduleName.includes(\"/\")\n          ? !(\n              isInDep ||\n              isInDev ||\n              `@types/${moduleName.slice(1).replace(\"/\", \"_\")}` in devDep\n            )\n          : !(isInDep || isInDev || `@types/${moduleName}` in devDep);\n      } else {\n        return allowDevDependencies ? !(isInDep || isInDev) : !isInDep;\n      }\n    }),\n});\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "eslint-plugin-esm",
3
- "version": "0.2.0",
3
+ "version": "0.2.1",
4
4
  "description": "ESLint plugin for linting ESM (import/export syntax)",
5
5
  "keywords": [
6
6
  "eslint",
@@ -14,10 +14,10 @@
14
14
  "es6",
15
15
  "module"
16
16
  ],
17
- "homepage": "https://github.com/zanminkian/git-validator/tree/main/packages/eslint-plugin-esm",
17
+ "homepage": "https://github.com/zanminkian/fenge/tree/main/packages/eslint-plugin-esm",
18
18
  "repository": {
19
19
  "type": "git",
20
- "url": "git+https://github.com/zanminkian/git-validator.git",
20
+ "url": "git+https://github.com/zanminkian/fenge.git",
21
21
  "directory": "packages/eslint-plugin-esm"
22
22
  },
23
23
  "license": "MIT",
@@ -29,7 +29,7 @@
29
29
  "@types/eslint": "8.56.11",
30
30
  "@types/estree": "1.0.6",
31
31
  "@types/json-schema": "7.0.15",
32
- "@types/node": "22.5.5",
32
+ "@types/node": "22.7.5",
33
33
  "@typescript-eslint/parser": "7.16.1",
34
34
  "eslint": "8.57.0",
35
35
  "outdent": "0.8.0"
@@ -61,7 +61,7 @@ export const noPhantomDepImports = createRule({
61
61
  allowDevDependencies = false,
62
62
  }: { allowDevDependencies: boolean } = context.options[0] ?? {};
63
63
 
64
- // ignore `import {foo} from './'`
64
+ // ignore `import {foo} from './'` and `import {foo} from 'node:foo'`
65
65
  if (getSourceType(source) !== "module") {
66
66
  return false;
67
67
  }
@@ -75,6 +75,11 @@ export const noPhantomDepImports = createRule({
75
75
  isObject(pkgJson.content.dependencies)
76
76
  ? pkgJson.content.dependencies
77
77
  : {};
78
+ const peerDep =
79
+ "peerDependencies" in pkgJson.content &&
80
+ isObject(pkgJson.content.peerDependencies)
81
+ ? pkgJson.content.peerDependencies
82
+ : {};
78
83
  const devDep =
79
84
  "devDependencies" in pkgJson.content &&
80
85
  isObject(pkgJson.content.devDependencies)
@@ -86,22 +91,18 @@ export const noPhantomDepImports = createRule({
86
91
  .slice(0, source.startsWith("@") ? 2 : 1)
87
92
  .join("/");
88
93
 
94
+ const isInDep = moduleName in dep || moduleName in peerDep;
95
+ const isInDev = moduleName in devDep;
89
96
  if ("importKind" in node && node.importKind === "type") {
90
97
  return moduleName.startsWith("@") && moduleName.includes("/")
91
98
  ? !(
92
- moduleName in dep ||
93
- moduleName in devDep ||
99
+ isInDep ||
100
+ isInDev ||
94
101
  `@types/${moduleName.slice(1).replace("/", "_")}` in devDep
95
102
  )
96
- : !(
97
- moduleName in dep ||
98
- moduleName in devDep ||
99
- `@types/${moduleName}` in devDep
100
- );
103
+ : !(isInDep || isInDev || `@types/${moduleName}` in devDep);
101
104
  } else {
102
- return allowDevDependencies
103
- ? !(moduleName in dep || moduleName in devDep)
104
- : !(moduleName in dep);
105
+ return allowDevDependencies ? !(isInDep || isInDev) : !isInDep;
105
106
  }
106
107
  }),
107
108
  });