Skip to content

Commit 0a44fca

Browse files
authored
Safely check all array accesses (#749)
1 parent c6eb278 commit 0a44fca

18 files changed

+57
-38
lines changed

.eslintrc.cjs

+1
Original file line numberDiff line numberDiff line change
@@ -56,6 +56,7 @@ module.exports = {
5656
'@typescript-eslint/ban-ts-comment': 'off',
5757
'@typescript-eslint/consistent-type-imports': 'error',
5858
'@typescript-eslint/no-duplicate-imports': 'error',
59+
'@typescript-eslint/no-non-null-assertion': 'off',
5960
'@typescript-eslint/sort-type-union-intersection-members': 'error',
6061
},
6162
},

packages/react-docgen-cli/src/commands/parse/options/loadReactDocgenPlugin.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ export default async function loadReactDocgenPlugin<T>(
77
builtins?: Record<string, T>,
88
): Promise<T> {
99
if (builtins?.[input]) {
10-
return builtins[input];
10+
return builtins[input]!;
1111
}
1212

1313
const path = resolve(process.cwd(), input);

packages/react-docgen-cli/src/commands/parse/options/loadResolvers.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -54,5 +54,5 @@ export default async function loadResolvers(
5454
});
5555
}
5656

57-
return loadResolver(input[0]);
57+
return loadResolver(input[0]!);
5858
}

packages/react-docgen/src/handlers/componentMethodsHandler.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,7 @@ const explodedImperativeHandleVisitors =
103103
// useImperativeHandle(ref, () => ({ name: () => {}, ...}))
104104
const arg = path.get('arguments')[1];
105105

106-
if (arg && !arg.isFunction()) {
106+
if (!arg || !arg.isFunction()) {
107107
return path.skip();
108108
}
109109

packages/react-docgen/src/resolver/FindAllDefinitionsResolver.ts

+8-4
Original file line numberDiff line numberDiff line change
@@ -53,20 +53,24 @@ const explodedVisitors = visitors.explode<TraverseState>({
5353
ClassDeclaration: { enter: classVisitor },
5454
CallExpression: {
5555
enter: function (path, state): void {
56+
const argument = path.get('arguments')[0];
57+
58+
if (!argument) {
59+
return;
60+
}
61+
5662
if (isReactForwardRefCall(path)) {
5763
// If the the inner function was previously identified as a component
5864
// replace it with the parent node
59-
const inner = resolveToValue(
60-
path.get('arguments')[0],
61-
) as ComponentNodePath;
65+
const inner = resolveToValue(argument) as ComponentNodePath;
6266

6367
state.foundDefinitions.delete(inner);
6468
state.foundDefinitions.add(path);
6569

6670
// Do not traverse into arguments
6771
return path.skip();
6872
} else if (isReactCreateClassCall(path)) {
69-
const resolvedPath = resolveToValue(path.get('arguments')[0]);
73+
const resolvedPath = resolveToValue(argument);
7074

7175
if (resolvedPath.isObjectExpression()) {
7276
state.foundDefinitions.add(resolvedPath);

packages/react-docgen/src/utils/docblock.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,7 @@ export function getDocblock(path: NodePath, trailing = false): string | null {
3535
}
3636

3737
if (comments.length > 0) {
38-
return parseDocblock(comments[comments.length - 1].value);
38+
return parseDocblock(comments[comments.length - 1]!.value);
3939
}
4040

4141
return null;
@@ -50,7 +50,7 @@ export function getDoclets(str: string): Record<string, string> {
5050
let match: RegExpExecArray | null;
5151

5252
while ((match = DOCLET_PATTERN.exec(str))) {
53-
doclets[match[1]] = match[2] || true;
53+
doclets[match[1]!] = match[2] || true;
5454
}
5555

5656
return doclets;

packages/react-docgen/src/utils/getFlowType.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -159,7 +159,7 @@ function handleGenericTypeAnnotation(
159159
if (
160160
typeParams &&
161161
typeParams[type.name] &&
162-
typeParams[type.name].isGenericTypeAnnotation()
162+
typeParams[type.name]!.isGenericTypeAnnotation()
163163
) {
164164
return type;
165165
}

packages/react-docgen/src/utils/getMemberExpressionValuePath.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,8 @@ function resolveName(path: NodePath): string | undefined {
1717
' declarations.',
1818
);
1919
}
20-
const id = declarations[0].get('id');
20+
// VariableDeclarator always has at least one declaration, hence the non-null-assertion
21+
const id = declarations[0]!.get('id');
2122

2223
if (id.isIdentifier()) {
2324
return id.node.name;

packages/react-docgen/src/utils/getTSType.ts

+13-9
Original file line numberDiff line numberDiff line change
@@ -207,15 +207,19 @@ function handleTSTypeLiteral(
207207
} else if (param.isTSCallSignatureDeclaration()) {
208208
type.signature.constructor = handleTSFunctionType(param, typeParams);
209209
} else if (param.isTSIndexSignature() && typeAnnotation.hasNode()) {
210-
const idTypeAnnotation = param
211-
.get('parameters')[0]
212-
.get('typeAnnotation') as NodePath<TSTypeAnnotation | null | undefined>;
213-
214-
if (idTypeAnnotation.hasNode()) {
215-
type.signature.properties.push({
216-
key: getTSTypeWithResolvedTypes(idTypeAnnotation, typeParams),
217-
value: getTSTypeWithRequirements(typeAnnotation, typeParams),
218-
});
210+
const parameters = param.get('parameters');
211+
212+
if (parameters[0]) {
213+
const idTypeAnnotation = parameters[0].get(
214+
'typeAnnotation',
215+
) as NodePath<TSTypeAnnotation | null | undefined>;
216+
217+
if (idTypeAnnotation.hasNode()) {
218+
type.signature.properties.push({
219+
key: getTSTypeWithResolvedTypes(idTypeAnnotation, typeParams),
220+
value: getTSTypeWithRequirements(typeAnnotation, typeParams),
221+
});
222+
}
219223
}
220224
}
221225
});

packages/react-docgen/src/utils/getTypeFromReactComponent.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,9 @@ export default (path: NodePath): NodePath | null => {
5252
if (superTypes.hasNode()) {
5353
const params = superTypes.get('params');
5454

55-
typePath = params[params.length === 3 ? 1 : 0];
55+
if (params.length >= 1) {
56+
typePath = params[params.length === 3 ? 1 : 0]!;
57+
}
5658
} else {
5759
const propsMemberPath = getMemberValuePath(path, 'props');
5860

packages/react-docgen/src/utils/getTypeParameters.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -56,7 +56,7 @@ export default function getTypeParameters(
5656
typeName.isIdentifier() &&
5757
inputParams[typeName.node.name]
5858
) {
59-
resolvedTypePath = inputParams[typeName.node.name];
59+
resolvedTypePath = inputParams[typeName.node.name]!;
6060
}
6161

6262
params[key] = resolvedTypePath;

packages/react-docgen/src/utils/postProcessDocumentation.ts

+2-4
Original file line numberDiff line numberDiff line change
@@ -6,10 +6,8 @@ export default function (
66
const props = documentation.props;
77

88
if (props) {
9-
// props with default values should not be required
10-
Object.keys(props).forEach((prop) => {
11-
const propInfo = props[prop];
12-
9+
Object.values(props).forEach((propInfo) => {
10+
// props with default values should not be required
1311
if (propInfo.defaultValue) {
1412
propInfo.required = false;
1513
}

packages/react-docgen/src/utils/resolveHOC.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ export default function resolveHOC(path: NodePath): NodePath {
2020

2121
if (argumentLength && argumentLength > 0) {
2222
const args = path.get('arguments');
23-
const firstArg = args[0];
23+
const firstArg = args[0]!;
2424

2525
// If the first argument is one of these types then the component might be the last argument
2626
// If there are all identifiers then we cannot figure out exactly and have to assume it is the first
@@ -31,7 +31,7 @@ export default function resolveHOC(path: NodePath): NodePath {
3131
firstArg.isArrayExpression() ||
3232
firstArg.isSpreadElement())
3333
) {
34-
return resolveHOC(resolveToValue(args[argumentLength - 1]));
34+
return resolveHOC(resolveToValue(args[argumentLength - 1]!));
3535
}
3636

3737
return resolveHOC(resolveToValue(firstArg));

packages/react-docgen/src/utils/resolveObjectKeysToArray.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -142,7 +142,11 @@ export default function resolveObjectKeysToArray(
142142
path: NodePath,
143143
): string[] | null {
144144
if (isObjectKeysCall(path)) {
145-
const objectExpression = resolveToValue(path.get('arguments')[0]);
145+
const argument = path.get('arguments')[0];
146+
const objectExpression = resolveToValue(
147+
// isObjectKeysCall already asserts that there is at least one argument, hence the non-null-assertion
148+
argument!,
149+
);
146150
const values = resolveObjectToNameArray(objectExpression);
147151

148152
if (values) {

packages/react-docgen/src/utils/resolveObjectValuesToArray.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -99,7 +99,11 @@ export default function resolveObjectValuesToArray(
9999
path: NodePath,
100100
): string[] | null {
101101
if (isObjectValuesCall(path)) {
102-
const objectExpression = resolveToValue(path.get('arguments')[0]);
102+
const argument = path.get('arguments')[0];
103+
const objectExpression = resolveToValue(
104+
// isObjectValuesCall already asserts that there is at least one argument, hence the non-null-assertion
105+
argument!,
106+
);
103107
const values = resolveObjectToPropMap(objectExpression);
104108

105109
if (values) {

packages/react-docgen/src/utils/resolveToValue.ts

+3-5
Original file line numberDiff line numberDiff line change
@@ -12,9 +12,9 @@ import initialize from './ts-types/index.js';
1212
import getNameOrValue from './getNameOrValue.js';
1313

1414
function findScopePath(
15-
bindingIdentifiers: Array<NodePath<Identifier>>,
15+
bindingIdentifiers: Array<NodePath<Identifier>> | undefined,
1616
): NodePath | null {
17-
if (bindingIdentifiers && bindingIdentifiers.length >= 1) {
17+
if (bindingIdentifiers && bindingIdentifiers[0]) {
1818
const resolvedParentPath = bindingIdentifiers[0].parentPath;
1919

2020
if (
@@ -136,9 +136,7 @@ export default function resolveToValue(path: NodePath): NodePath {
136136
// block where it is defined (i.e. we are not traversing into statements)
137137
resolvedPath = findLastAssignedValue(binding.scope.path, path);
138138
if (!resolvedPath) {
139-
const bindingMap = binding.path.getOuterBindingIdentifierPaths(
140-
true,
141-
) as Record<string, Array<NodePath<Identifier>>>;
139+
const bindingMap = binding.path.getOuterBindingIdentifierPaths(true);
142140

143141
resolvedPath = findScopePath(bindingMap[path.node.name]);
144142
}

packages/react-docgen/src/utils/ts-types/index.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ declare module '@babel/traverse' {
3232
export interface Scope {
3333
typeBindings: Record<string, TypeBinding>;
3434
getTypeBinding(name: string): TypeBinding | undefined;
35-
getOwnTypeBinding(name: string): TypeBinding;
35+
getOwnTypeBinding(name: string): TypeBinding | undefined;
3636
registerTypeBinding(
3737
this: BaseScope,
3838
typeKind: TypeKind,
@@ -128,7 +128,10 @@ function getTypeBinding(
128128
return undefined;
129129
}
130130

131-
function getOwnTypeBinding(this: BaseScope, name: string): TypeBinding {
131+
function getOwnTypeBinding(
132+
this: BaseScope,
133+
name: string,
134+
): TypeBinding | undefined {
132135
return this.typeBindings[name];
133136
}
134137

tsconfig.base.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
"strict": true,
99
"noImplicitAny": false,
1010
"noImplicitReturns": true,
11-
"noUncheckedIndexedAccess": false,
11+
"noUncheckedIndexedAccess": true,
1212
"noUnusedLocals": true,
1313
"noUnusedParameters": true,
1414
"moduleResolution": "node16",

0 commit comments

Comments
 (0)