-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add
no-anonymous-default-export
rule
- Loading branch information
1 parent
c975742
commit 2a12e67
Showing
6 changed files
with
202 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
# no-anonymous-default-export | ||
|
||
Reports if a module's default export is unnamed. This includes several types of unnamed data types; literals, object expressions, arrays, anonymous functions, arrow functions, and anonymous class declarations. | ||
|
||
Ensuring that default exports are named helps improve the grepability of the codebase by encouraging the re-use of the same identifier for the module's default export at its declaration site and at its import sites. | ||
|
||
## Options | ||
|
||
By default, all types of anonymous default exports are forbidden, but any types can be selectively allowed by toggling them on in the options. | ||
|
||
The complete default configuration looks like this. | ||
|
||
```js | ||
"import/no-anonymous-default-export": ["error", { | ||
"allowArray": false, | ||
"allowArrowFunction": false, | ||
"allowAnonymousClass": false, | ||
"allowAnonymousFunction": false, | ||
"allowLiteral": false, | ||
"allowObject": false | ||
}] | ||
``` | ||
|
||
## Rule Details | ||
|
||
### Fail | ||
```js | ||
export default [] | ||
|
||
export default () => {} | ||
|
||
export default class {} | ||
|
||
export default function () {} | ||
|
||
export default 123 | ||
|
||
export default {} | ||
``` | ||
|
||
### Pass | ||
```js | ||
const foo = 123 | ||
export default foo | ||
|
||
export default function foo() {} | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowArray": true}] */ | ||
export default [] | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowArrowFunction": true}] */ | ||
export default () => {} | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowAnonymousClass": true}] */ | ||
export default class {} | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowAnonymousFunction": true}] */ | ||
export default function () {} | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowLiteral": true}] */ | ||
export default 123 | ||
|
||
/* eslint import/no-anonymous-default-export: [2, {"allowObject": true}] */ | ||
export default {} | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,84 @@ | ||
/** | ||
* @fileoverview Rule to disallow anonymous default exports. | ||
* @author Duncan Beevers | ||
*/ | ||
|
||
const defs = { | ||
ArrayExpression: { | ||
option: 'allowArray', | ||
message: 'an array', | ||
}, | ||
ArrowFunctionExpression: { | ||
option: 'allowArrowFunction', | ||
message: 'an arrow function', | ||
}, | ||
ClassDeclaration: { | ||
option: 'allowAnonymousClass', | ||
message: 'an anonymous class', | ||
forbid: (node) => !node.declaration.id, | ||
}, | ||
FunctionDeclaration: { | ||
option: 'allowAnonymousFunction', | ||
message: 'an anonymous function', | ||
forbid: (node) => !node.declaration.id, | ||
}, | ||
Literal: { | ||
option: 'allowLiteral', | ||
message: 'a literal', | ||
}, | ||
ObjectExpression: { | ||
option: 'allowObject', | ||
message: 'an object expression', | ||
}, | ||
} | ||
|
||
const defaultOptions = Object.keys(defs). | ||
map((key) => defs[key]). | ||
reduce((acc, def) => { | ||
acc[def.options] = false | ||
|
||
return acc | ||
}) | ||
|
||
const schemaProperties = Object.keys(defs). | ||
map((key) => defs[key]). | ||
reduce((acc, def) => { | ||
acc[def.option] = { | ||
description: 'If `false`, will not report default export of ' + def.message, | ||
type: 'boolean', | ||
default: true, | ||
} | ||
|
||
return acc | ||
}, {}) | ||
|
||
module.exports = { | ||
meta: { | ||
schema: [ | ||
{ | ||
type: 'object', | ||
properties: schemaProperties, | ||
'additionalProperties': false, | ||
}, | ||
], | ||
}, | ||
|
||
create: function (context) { | ||
const options = Object.assign({}, defaultOptions, context.options[0]) | ||
|
||
return { | ||
'ExportDefaultDeclaration': (node) => { | ||
const def = defs[node.declaration.type] | ||
|
||
// Unrecognized node type or else allowed by configuration | ||
if (!def || options[def.option]) { | ||
return | ||
} | ||
|
||
if (!def.forbid || def.forbid(node)) { | ||
context.report({ node, message: 'Unexpected default export of ' + def.message }) | ||
} | ||
}, | ||
} | ||
}, | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,47 @@ | ||
import { test, SYNTAX_CASES } from '../utils' | ||
|
||
import { RuleTester } from 'eslint' | ||
|
||
var ruleTester = new RuleTester() | ||
var rule = require('rules/no-anonymous-default-export') | ||
|
||
ruleTester.run('no-anonymous-default-export', rule, { | ||
valid: [ | ||
// Exports with identifiers are valid | ||
test({ code: 'const foo = 123\nexport default foo' }), | ||
test({ code: 'export default function foo() {}'}), | ||
test({ code: 'export default class MyClass {}'}), | ||
|
||
// Allow each forbidden type with appropriate option | ||
test({ code: 'export default []', options: [{ allowArray: true }] }), | ||
test({ code: 'export default () => {}', options: [{ allowArrowFunction: true }] }), | ||
test({ code: 'export default class {}', options: [{ allowAnonymousClass: true }] }), | ||
test({ code: 'export default function() {}', options: [{ allowAnonymousFunction: true }] }), | ||
test({ code: 'export default 123', options: [{ allowLiteral: true }] }), | ||
test({ code: 'export default `foo`', options: [{ allowLiteral: true }] }), | ||
test({ code: 'export default {}', options: [{ allowObject: true }] }), | ||
|
||
// Allow forbidden types with multiple options | ||
test({ code: 'export default 123', options: [{ allowLiteral: true, allowObject: true }] }), | ||
test({ code: 'export default {}', options: [{ allowLiteral: true, allowObject: true }] }), | ||
|
||
// Sanity check unrelated export syntaxes | ||
test({ code: 'export * from \'foo\'' }), | ||
test({ code: 'const foo = 123\nexport { foo }' }), | ||
test({ code: 'const foo = 123\nexport { foo as default }' }), | ||
|
||
...SYNTAX_CASES, | ||
], | ||
|
||
invalid: [ | ||
test({ code: 'export default []', errors: [{ message: 'Unexpected default export of an array' }] }), | ||
test({ code: 'export default () => {}', errors: [{ message: 'Unexpected default export of an arrow function' }] }), | ||
test({ code: 'export default class {}', errors: [{ message: 'Unexpected default export of an anonymous class' }] }), | ||
test({ code: 'export default 123', errors: [{ message: 'Unexpected default export of a literal' }] }), | ||
test({ code: 'export default {}', errors: [{ message: 'Unexpected default export of an object expression' }] }), | ||
test({ code: 'export default function() {}', errors: [{ message: 'Unexpected default export of an anonymous function' }] }), | ||
|
||
// Test failure with non-covering exception | ||
test({ code: 'export default 123', options: [{ allowObject: true }], errors: [{ message: 'Unexpected default export of a literal' }] }), | ||
], | ||
}) |