Skip to content

Commit

Permalink
Add forbid-foreign-prop-types rule
Browse files Browse the repository at this point in the history
People may want to use babel-plugin-transform-react-remove-prop-types
to remove propTypes from their components in production builds, as an
optimization. The `forbib-foreign-prop-types` rule forbids using
another component's prop types unless they are explicitly
imported/exported, which makes that optimization less prone to error.

Fixes jsx-eslint#696
  • Loading branch information
iancmyers committed Jan 31, 2017
1 parent 5827897 commit d7f5e49
Show file tree
Hide file tree
Showing 5 changed files with 184 additions and 0 deletions.
1 change: 1 addition & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ Finally, enable all of the rules that you would like to use. Use [our preset](#
* [react/display-name](docs/rules/display-name.md): Prevent missing `displayName` in a React component definition
* [react/forbid-component-props](docs/rules/forbid-component-props.md): Forbid certain props on Components
* [react/forbid-prop-types](docs/rules/forbid-prop-types.md): Forbid certain propTypes
* [react/forbid-foreign-prop-types](docs/rules/forbid-foreign-prop-types.md): Forbid foreign propTypes
* [react/no-array-index-key](docs/rules/no-array-index-key.md): Prevent using Array index in `key` props
* [react/no-children-prop](docs/rules/no-children-prop.md): Prevent passing children as props
* [react/no-danger](docs/rules/no-danger.md): Prevent usage of dangerous JSX properties
Expand Down
28 changes: 28 additions & 0 deletions docs/rules/forbid-foreign-prop-types.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
# Forbid foreign propTypes (forbid-foreign-prop-types)

This rule forbids using another component's prop types unless they are explicitly imported/exported. This allows people who want to use babel-plugin-transform-react-remove-prop-types to remove propTypes from their components in production builds, to do so safely.

## Rule Details

This rule checks all objects and ensures that the `propTypes` property is not used.

The following patterns are considered warnings:

```js
import SomeComponent from './SomeComponent';
SomeComponent.propTypes;

var { propTypes } = SomeComponent;

SomeComponent['propTypes'];
```

The following patterns are not considered warnings:

```js
import {propTypes: someComponentPropTypes}, SomeComponent from './SomeComponent';
```

## When not to use

This rule aims to make a certain production optimization, removing prop types, less prone to error. This rule may not be relevant to you if you do not wish to make use of this optimization.
1 change: 1 addition & 0 deletions index.js
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@ var allRules = {
'no-direct-mutation-state': require('./lib/rules/no-direct-mutation-state'),
'forbid-component-props': require('./lib/rules/forbid-component-props'),
'forbid-prop-types': require('./lib/rules/forbid-prop-types'),
'forbid-foreign-prop-types': require('./lib/rules/forbid-foreign-prop-types'),
'prefer-es6-class': require('./lib/rules/prefer-es6-class'),
'jsx-key': require('./lib/rules/jsx-key'),
'no-string-refs': require('./lib/rules/no-string-refs'),
Expand Down
52 changes: 52 additions & 0 deletions lib/rules/forbid-foreign-prop-types.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,52 @@
/**
* @fileoverview Forbid using another component's propTypes
* @author Ian Christian Myers
*/
'use strict';

// ------------------------------------------------------------------------------
// Constants
// ------------------------------------------------------------------------------


// ------------------------------------------------------------------------------
// Rule Definition
// ------------------------------------------------------------------------------

module.exports = {
meta: {
docs: {
description: 'Forbid using another component\'s propTypes',
category: 'Best Practices',
recommended: false
}
},

create: function(context) {

return {
MemberExpression: function(node) {
if (node.property && node.property.type === 'Identifier' && node.property.name === 'propTypes' ||
node.property && node.property.type === 'Literal' && node.property.value === 'propTypes') {
context.report({
node: node.property,
message: 'Using another component\'s propTypes is forbidden'
});
}
},

ObjectPattern: function(node) {
var propTypesNode = node.properties.find(function(property) {
return property.type === 'Property' && property.key.name === 'propTypes';
});

if (propTypesNode) {
context.report({
node: propTypesNode,
message: 'Using another component\'s propTypes is forbidden'
});
}
}
};
}
};
102 changes: 102 additions & 0 deletions tests/lib/rules/forbid-foreign-prop-types.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
/**
* @fileoverview Tests for forbid-foreign-prop-types
*/
'use strict';

// -----------------------------------------------------------------------------
// Requirements
// -----------------------------------------------------------------------------

var rule = require('../../../lib/rules/forbid-foreign-prop-types');
var RuleTester = require('eslint').RuleTester;

var parserOptions = {
ecmaVersion: 6,
ecmaFeatures: {
experimentalObjectRestSpread: true,
jsx: true
}
};

require('babel-eslint');

// -----------------------------------------------------------------------------
// Tests
// -----------------------------------------------------------------------------

var ERROR_MESSAGE = 'Using another component\'s propTypes is forbidden';

var ruleTester = new RuleTester();
ruleTester.run('forbid-foreign-prop-types', rule, {

valid: [{
code: 'import { propTypes } from "SomeComponent";',
parser: 'babel-eslint'
},{
code: 'import { propTypes: someComponentPropTypes } from "SomeComponent";',
parser: 'babel-eslint'
}],

invalid: [{
code: [
'var Foo = React.createClass({',
' propTypes: Bar.propTypes,',
' render: function() {',
' return <Foo className="bar" />;',
' }',
'});'
].join('\n'),
parserOptions: parserOptions,
errors: [{
message: ERROR_MESSAGE,
type: 'Identifier'
}]
},
{
code: [
'var Foo = React.createClass({',
' propTypes: Bar["propTypes"],',
' render: function() {',
' return <Foo className="bar" />;',
' }',
'});'
].join('\n'),
parserOptions: parserOptions,
errors: [{
message: ERROR_MESSAGE,
type: 'Literal'
}]
},
{
code: [
'var { propTypes } = SomeComponent',
'var Foo = React.createClass({',
' propTypes,',
' render: function() {',
' return <Foo className="bar" />;',
' }',
'});'
].join('\n'),
parserOptions: parserOptions,
errors: [{
message: ERROR_MESSAGE,
type: 'Property'
}]
},
{
code: [
'var { propTypes: typesOfProps } = SomeComponent',
'var Foo = React.createClass({',
' propTypes: typesOfProps,',
' render: function() {',
' return <Foo className="bar" />;',
' }',
'});'
].join('\n'),
parserOptions: parserOptions,
errors: [{
message: ERROR_MESSAGE,
type: 'Property'
}]
}]
});

0 comments on commit d7f5e49

Please sign in to comment.