Skip to content

Commit

Permalink
[compiler] Expect components to have hook calls or jsx directly in body
Browse files Browse the repository at this point in the history
Summary: We can tighten our criteria for what is a component by requiring that a component or hook contain JSX or hook calls directly within its body, excluding nested functions . Currently, if we see them within the body anywhere -- including nested functions -- we treat it as a component if the other requirements are met. This change makes this stricter.

We also now expect components (but not necessarily hooks) to have return statements, and those returns must be potential React nodes (we can reject functions that return function or object literals, for example).

ghstack-source-id: a1b538c41cad14246a0ba081240b24cabc5fdbe0
Pull Request resolved: #29865
  • Loading branch information
mvitousek committed Jun 11, 2024
1 parent 9207e3d commit 4666faf
Show file tree
Hide file tree
Showing 9 changed files with 203 additions and 16 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -688,7 +688,8 @@ function getComponentOrHookLike(
if (functionName !== null && isComponentName(functionName)) {
let isComponent =
callsHooksOrCreatesJsx(node, hookPattern) &&
isValidComponentParams(node.get("params"));
isValidComponentParams(node.get("params")) &&
!returnsNonNode(node);
return isComponent ? "Component" : null;
} else if (functionName !== null && isHook(functionName, hookPattern)) {
// Hooks have hook invocations or JSX, but can take any # of arguments
Expand All @@ -708,12 +709,31 @@ function getComponentOrHookLike(
return null;
}

function skipNestedFunctions(
node: NodePath<
t.FunctionDeclaration | t.ArrowFunctionExpression | t.FunctionExpression
>
) {
return (
fn: NodePath<
t.FunctionDeclaration | t.ArrowFunctionExpression | t.FunctionExpression
>
): void => {
if (fn.node !== node.node) {
fn.skip();
}
};
}

function callsHooksOrCreatesJsx(
node: NodePath<t.Node>,
node: NodePath<
t.FunctionDeclaration | t.ArrowFunctionExpression | t.FunctionExpression
>,
hookPattern: string | null
): boolean {
let invokesHooks = false;
let createsJsx = false;

node.traverse({
JSX() {
createsJsx = true;
Expand All @@ -724,11 +744,48 @@ function callsHooksOrCreatesJsx(
invokesHooks = true;
}
},
ArrowFunctionExpression: skipNestedFunctions(node),
FunctionExpression: skipNestedFunctions(node),
FunctionDeclaration: skipNestedFunctions(node),
});

return invokesHooks || createsJsx;
}

function returnsNonNode(
node: NodePath<
t.FunctionDeclaration | t.ArrowFunctionExpression | t.FunctionExpression
>
): boolean {
let hasReturn = false;
let returnsNonNode = false;

node.traverse({
ReturnStatement(ret) {
hasReturn = true;
const argument = ret.node.argument;
if (argument == null) {
returnsNonNode = true;
} else {
switch (argument.type) {
case "ObjectExpression":
case "ArrowFunctionExpression":
case "FunctionExpression":
case "BigIntLiteral":
case "ClassExpression":
case "NewExpression": // technically `new Array()` is legit, but unlikely
returnsNonNode = true;
}
}
},
ArrowFunctionExpression: skipNestedFunctions(node),
FunctionExpression: skipNestedFunctions(node),
FunctionDeclaration: skipNestedFunctions(node),
});

return !hasReturn || returnsNonNode;
}

/*
* Gets the static name of a function AST node. For function declarations it is
* easy. For anonymous function expressions it is much harder. If you search for
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@

## Input

```javascript
// @compilationMode(infer)
function Component(props) {
const result = f(props);
function helper() {
return <foo />;
}
helper();
return result;
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};

```

## Code

```javascript
// @compilationMode(infer)
function Component(props) {
const result = f(props);
function helper() {
return <foo />;
}
helper();
return result;
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};

```
### Eval output
(kind: ok) {}
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
// @compilationMode(infer)
function Component(props) {
const result = f(props);
function helper() {
return <foo />;
}
helper();
return result;
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@

## Input

```javascript
// @compilationMode(infer)
function Component(props) {
const ignore = <foo />;
return { foo: f(props) };
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};

```

## Code

```javascript
// @compilationMode(infer)
function Component(props) {
const ignore = <foo />;
return { foo: f(props) };
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};

```
### Eval output
(kind: ok) {"foo":{}}
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
// @compilationMode(infer)
function Component(props) {
const ignore = <foo />;
return { foo: f(props) };
}

function f(props) {
return props;
}

export const FIXTURE_ENTRYPOINT = {
fn: Component,
params: [{}],
};
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
```javascript
// @compilationMode(infer)
function Component() {
"use memo";
const f = () => {
const x = {
outer() {
Expand All @@ -27,13 +28,13 @@ function Component() {
## Error

```
7 | const y = {
8 | inner() {
> 9 | return useFoo();
| ^^^^^^ InvalidReact: Hooks must be called at the top level in the body of a function component or custom hook, and may not be called within function expressions. See the Rules of Hooks (https://react.dev/warnings/invalid-hook-call-warning). Cannot call Custom within a function component (9:9)
10 | },
11 | };
12 | return y;
8 | const y = {
9 | inner() {
> 10 | return useFoo();
| ^^^^^^ InvalidReact: Hooks must be called at the top level in the body of a function component or custom hook, and may not be called within function expressions. See the Rules of Hooks (https://react.dev/warnings/invalid-hook-call-warning). Cannot call Custom within a function component (10:10)
11 | },
12 | };
13 | return y;
```
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// @compilationMode(infer)
function Component() {
"use memo";
const f = () => {
const x = {
outer() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
```javascript
// @compilationMode(infer)
function Component() {
"use memo";
const x = {
outer() {
const y = {
Expand All @@ -23,13 +24,13 @@ function Component() {
## Error

```
5 | const y = {
6 | inner() {
> 7 | return useFoo();
| ^^^^^^ InvalidReact: Hooks must be called at the top level in the body of a function component or custom hook, and may not be called within function expressions. See the Rules of Hooks (https://react.dev/warnings/invalid-hook-call-warning). Cannot call Custom within a function component (7:7)
8 | },
9 | };
10 | return y;
6 | const y = {
7 | inner() {
> 8 | return useFoo();
| ^^^^^^ InvalidReact: Hooks must be called at the top level in the body of a function component or custom hook, and may not be called within function expressions. See the Rules of Hooks (https://react.dev/warnings/invalid-hook-call-warning). Cannot call Custom within a function component (8:8)
9 | },
10 | };
11 | return y;
```
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
// @compilationMode(infer)
function Component() {
"use memo";
const x = {
outer() {
const y = {
Expand Down

0 comments on commit 4666faf

Please sign in to comment.