Skip to content

Commit fd3683f

Browse files
authoredJan 5, 2022
feat: Support arbitrary module namespace names in no-restricted-exports (#15478)
* feat: Support arbitrary module namespace names in no-restricted-exports Refs #15465 * align quotes
1 parent 6278281 commit fd3683f

File tree

4 files changed

+108
-14
lines changed

4 files changed

+108
-14
lines changed
 

‎docs/rules/no-restricted-exports.md

+16-8
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ Examples of **incorrect** code for this rule:
1818

1919
```js
2020
/*eslint no-restricted-exports: ["error", {
21-
"restrictedNamedExports": ["foo", "bar", "Baz", "a", "b", "c", "d"]
21+
"restrictedNamedExports": ["foo", "bar", "Baz", "a", "b", "c", "d", "e", "👍"]
2222
}]*/
2323

2424
export const foo = 1;
@@ -33,16 +33,20 @@ export { a };
3333
function someFunction() {}
3434
export { someFunction as b };
3535

36-
export { c } from 'some_module';
36+
export { c } from "some_module";
3737

38-
export { something as d } from 'some_module';
38+
export { "d" } from "some_module";
39+
40+
export { something as e } from "some_module";
41+
42+
export { "👍" } from "some_module";
3943
```
4044

4145
Examples of **correct** code for this rule:
4246

4347
```js
4448
/*eslint no-restricted-exports: ["error", {
45-
"restrictedNamedExports": ["foo", "bar", "Baz", "a", "b", "c", "d"]
49+
"restrictedNamedExports": ["foo", "bar", "Baz", "a", "b", "c", "d", "e", "👍"]
4650
}]*/
4751

4852
export const quux = 1;
@@ -57,9 +61,13 @@ export { a as myObject };
5761
function someFunction() {}
5862
export { someFunction };
5963

60-
export { c as someName } from 'some_module';
64+
export { c as someName } from "some_module";
65+
66+
export { "d" as " d " } from "some_module";
67+
68+
export { something } from "some_module";
6169

62-
export { something } from 'some_module';
70+
export { "👍" as thumbsUp } from "some_module";
6371
```
6472

6573
### Default exports
@@ -79,7 +87,7 @@ export { foo as default };
7987
```js
8088
/*eslint no-restricted-exports: ["error", { "restrictedNamedExports": ["default"] }]*/
8189

82-
export { default } from 'some_module';
90+
export { default } from "some_module";
8391
```
8492

8593
Examples of additional **correct** code for this rule:
@@ -102,5 +110,5 @@ export function foo() {}
102110
//----- my_module.js -----
103111
/*eslint no-restricted-exports: ["error", { "restrictedNamedExports": ["foo"] }]*/
104112

105-
export * from 'some_module'; // allowed, although this declaration exports "foo" from my_module
113+
export * from "some_module"; // allowed, although this declaration exports "foo" from my_module
106114
```

‎lib/rules/no-restricted-exports.js

+9-3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,12 @@
55

66
"use strict";
77

8+
//------------------------------------------------------------------------------
9+
// Requirements
10+
//------------------------------------------------------------------------------
11+
12+
const astUtils = require("./utils/ast-utils");
13+
814
//------------------------------------------------------------------------------
915
// Rule Definition
1016
//------------------------------------------------------------------------------
@@ -44,12 +50,12 @@ module.exports = {
4450
const restrictedNames = new Set(context.options[0] && context.options[0].restrictedNamedExports);
4551

4652
/**
47-
* Checks and reports given exported identifier.
48-
* @param {ASTNode} node exported `Identifier` node to check.
53+
* Checks and reports given exported name.
54+
* @param {ASTNode} node exported `Identifier` or string `Literal` node to check.
4955
* @returns {void}
5056
*/
5157
function checkExportedName(node) {
52-
const name = node.name;
58+
const name = astUtils.getModuleExportName(node);
5359

5460
if (restrictedNames.has(name)) {
5561
context.report({

‎lib/rules/utils/ast-utils.js

+21-1
Original file line numberDiff line numberDiff line change
@@ -769,6 +769,25 @@ function getSwitchCaseColonToken(node, sourceCode) {
769769
return sourceCode.getFirstToken(node, 1);
770770
}
771771

772+
/**
773+
* Gets ESM module export name represented by the given node.
774+
* @param {ASTNode} node `Identifier` or string `Literal` node in a position
775+
* that represents a module export name:
776+
* - `ImportSpecifier#imported`
777+
* - `ExportSpecifier#local` (if it is a re-export from another module)
778+
* - `ExportSpecifier#exported`
779+
* - `ExportAllDeclaration#exported`
780+
* @returns {string} The module export name.
781+
*/
782+
function getModuleExportName(node) {
783+
if (node.type === "Identifier") {
784+
return node.name;
785+
}
786+
787+
// string literal
788+
return node.value;
789+
}
790+
772791
//------------------------------------------------------------------------------
773792
// Public Interface
774793
//------------------------------------------------------------------------------
@@ -1898,5 +1917,6 @@ module.exports = {
18981917
equalLiteralValue,
18991918
isSameReference,
19001919
isLogicalAssignmentOperator,
1901-
getSwitchCaseColonToken
1920+
getSwitchCaseColonToken,
1921+
getModuleExportName
19021922
};

‎tests/lib/rules/no-restricted-exports.js

+62-2
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ const { RuleTester } = require("../../../lib/rule-tester");
1616
// Tests
1717
//------------------------------------------------------------------------------
1818

19-
const ruleTester = new RuleTester({ parserOptions: { ecmaVersion: 2020, sourceType: "module" } });
19+
const ruleTester = new RuleTester({ parserOptions: { ecmaVersion: 2022, sourceType: "module" } });
2020

2121
ruleTester.run("no-restricted-exports", rule, {
2222
valid: [
@@ -57,8 +57,12 @@ ruleTester.run("no-restricted-exports", rule, {
5757
{ code: "var b; export { b as a };", options: [{ restrictedNamedExports: ["x"] }] },
5858
{ code: "export { a } from 'foo';", options: [{ restrictedNamedExports: ["x"] }] },
5959
{ code: "export { b as a } from 'foo';", options: [{ restrictedNamedExports: ["x"] }] },
60+
{ code: "export { '' } from 'foo';", options: [{ restrictedNamedExports: ["undefined"] }] },
61+
{ code: "export { '' } from 'foo';", options: [{ restrictedNamedExports: [" "] }] },
62+
{ code: "export { ' ' } from 'foo';", options: [{ restrictedNamedExports: [""] }] },
63+
{ code: "export { ' a', 'a ' } from 'foo';", options: [{ restrictedNamedExports: ["a"] }] },
6064

61-
// does not mistakenly disallow non-exported identifiers that appear in named export declarations
65+
// does not mistakenly disallow non-exported names that appear in named export declarations
6266
{ code: "export var b = a;", options: [{ restrictedNamedExports: ["a"] }] },
6367
{ code: "export let [b = a] = [];", options: [{ restrictedNamedExports: ["a"] }] },
6468
{ code: "export const [b] = [a];", options: [{ restrictedNamedExports: ["a"] }] },
@@ -69,7 +73,10 @@ ruleTester.run("no-restricted-exports", rule, {
6973
{ code: "export class A { a(){} }", options: [{ restrictedNamedExports: ["a"] }] },
7074
{ code: "export class A extends B {}", options: [{ restrictedNamedExports: ["B"] }] },
7175
{ code: "var a; export { a as b };", options: [{ restrictedNamedExports: ["a"] }] },
76+
{ code: "var a; export { a as 'a ' };", options: [{ restrictedNamedExports: ["a"] }] },
7277
{ code: "export { a as b } from 'foo';", options: [{ restrictedNamedExports: ["a"] }] },
78+
{ code: "export { a as 'a ' } from 'foo';", options: [{ restrictedNamedExports: ["a"] }] },
79+
{ code: "export { 'a' as 'a ' } from 'foo';", options: [{ restrictedNamedExports: ["a"] }] },
7380

7481
// does not check source in re-export declarations
7582
{ code: "export { b } from 'a';", options: [{ restrictedNamedExports: ["a"] }] },
@@ -188,6 +195,59 @@ ruleTester.run("no-restricted-exports", rule, {
188195
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Identifier" }]
189196
},
190197

198+
// string literals
199+
{
200+
code: "let a; export { a as 'a' };",
201+
options: [{ restrictedNamedExports: ["a"] }],
202+
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Literal", column: 22 }]
203+
},
204+
{
205+
code: "let a; export { a as 'b' };",
206+
options: [{ restrictedNamedExports: ["b"] }],
207+
errors: [{ messageId: "restrictedNamed", data: { name: "b" }, type: "Literal", column: 22 }]
208+
},
209+
{
210+
code: "let a; export { a as ' b ' };",
211+
options: [{ restrictedNamedExports: [" b "] }],
212+
errors: [{ messageId: "restrictedNamed", data: { name: " b " }, type: "Literal", column: 22 }]
213+
},
214+
{
215+
code: "let a; export { a as '👍' };",
216+
options: [{ restrictedNamedExports: ["👍"] }],
217+
errors: [{ messageId: "restrictedNamed", data: { name: "👍" }, type: "Literal", column: 22 }]
218+
},
219+
{
220+
code: "export { 'a' } from 'foo';",
221+
options: [{ restrictedNamedExports: ["a"] }],
222+
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Literal" }]
223+
},
224+
{
225+
code: "export { '' } from 'foo';",
226+
options: [{ restrictedNamedExports: [""] }],
227+
errors: [{ messageId: "restrictedNamed", data: { name: "" }, type: "Literal" }]
228+
},
229+
{
230+
code: "export { ' ' } from 'foo';",
231+
options: [{ restrictedNamedExports: [" "] }],
232+
errors: [{ messageId: "restrictedNamed", data: { name: " " }, type: "Literal" }]
233+
},
234+
{
235+
code: "export { b as 'a' } from 'foo';",
236+
options: [{ restrictedNamedExports: ["a"] }],
237+
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Literal" }]
238+
},
239+
{
240+
code: "export { b as '\\u0061' } from 'foo';",
241+
options: [{ restrictedNamedExports: ["a"] }],
242+
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Literal" }]
243+
},
244+
{
245+
code: "export * as 'a' from 'foo';",
246+
options: [{ restrictedNamedExports: ["a"] }],
247+
errors: [{ messageId: "restrictedNamed", data: { name: "a" }, type: "Literal" }]
248+
},
249+
250+
191251
// destructuring
192252
{
193253
code: "export var [a] = [];",

0 commit comments

Comments
 (0)
Please sign in to comment.