📄 no-unsafe-assignment.ts¶
📊 Analysis Summary¶
| Metric | Count |
|---|---|
| 🔧 Functions | 7 |
| 📦 Imports | 13 |
| 📊 Variables & Constants | 8 |
| 🎯 Enums | 1 |
📚 Table of Contents¶
🛠️ File Location:¶
📂 packages/eslint-plugin/src/rules/no-unsafe-assignment.ts
📦 Imports¶
| Name | Source |
|---|---|
TSESTree |
@typescript-eslint/utils |
AST_NODE_TYPES |
@typescript-eslint/utils |
createRule |
../util |
getConstrainedTypeAtLocation |
../util |
getContextualType |
../util |
getParserServices |
../util |
getThisExpression |
../util |
isTypeAnyArrayType |
../util |
isTypeAnyType |
../util |
isTypeUnknownType |
../util |
isUnsafeAssignment |
../util |
nullThrows |
../util |
NullThrowsReasons |
../util |
Variables & Constants¶
| Name | Type | Kind | Value | Exported |
|---|---|---|---|---|
didReport |
boolean |
let/var | false |
✗ |
receiverElement |
any |
const | receiverNode.elements[receiverIndex] |
✗ |
senderType |
any |
const | tupleElements[receiverIndex] as ts.Type | undefined |
✗ |
properties |
Map<unknown, unknown> |
const | `new Map( | |
| senderType | ||||
| .getProperties() | ||||
| .map(property => [ | ||||
| property.getName(), | ||||
| checker.getTypeOfSymbolAtLocation(property, senderNode), | ||||
| ]), | ||||
| )` | ✗ | |||
didReport |
boolean |
let/var | false |
✗ |
key |
string |
let/var | *not shown* |
✗ |
receiverType |
any |
const | `comparisonType === ComparisonType.Contextual | |
| ? (getContextualType(checker, receiverTsNode as ts.Expression) ?? | ||||
| services.getTypeAtLocation(receiverNode)) | ||||
| : services.getTypeAtLocation(receiverNode)` | ✗ | |||
messageId |
'anyAssignment' | 'anyAssignmentThis' |
let/var | 'anyAssignment' |
✗ |
Functions¶
checkArrayDestructureHelper(receiverNode: TSESTree.Node, senderNode: TSESTree.Node): boolean¶
Code
function checkArrayDestructureHelper(
receiverNode: TSESTree.Node,
senderNode: TSESTree.Node,
): boolean {
if (receiverNode.type !== AST_NODE_TYPES.ArrayPattern) {
return false;
}
const senderTsNode = services.esTreeNodeToTSNodeMap.get(senderNode);
const senderType = services.getTypeAtLocation(senderNode);
return checkArrayDestructure(receiverNode, senderType, senderTsNode);
}
- Parameters:
receiverNode: TSESTree.NodesenderNode: TSESTree.Node- Return Type:
boolean - Calls:
services.esTreeNodeToTSNodeMap.getservices.getTypeAtLocationcheckArrayDestructure
checkArrayDestructure(receiverNode: TSESTree.ArrayPattern, senderType: ts.Type, senderNode: ts.Node): boolean¶
Code
function checkArrayDestructure(
receiverNode: TSESTree.ArrayPattern,
senderType: ts.Type,
senderNode: ts.Node,
): boolean {
// any array
// const [x] = ([] as any[]);
if (isTypeAnyArrayType(senderType, checker)) {
context.report({
node: receiverNode,
messageId: 'unsafeArrayPattern',
data: createData(senderType),
});
return false;
}
if (!checker.isTupleType(senderType)) {
return true;
}
const tupleElements = checker.getTypeArguments(senderType);
// tuple with any
// const [x] = [1 as any];
let didReport = false;
for (
let receiverIndex = 0;
receiverIndex < receiverNode.elements.length;
receiverIndex += 1
) {
const receiverElement = receiverNode.elements[receiverIndex];
if (!receiverElement) {
continue;
}
if (receiverElement.type === AST_NODE_TYPES.RestElement) {
// don't handle rests as they're not a 1:1 assignment
continue;
}
const senderType = tupleElements[receiverIndex] as ts.Type | undefined;
if (!senderType) {
continue;
}
// check for the any type first so we can handle [[[x]]] = [any]
if (isTypeAnyType(senderType)) {
context.report({
node: receiverElement,
messageId: 'unsafeArrayPatternFromTuple',
data: createData(senderType),
});
// we want to report on every invalid element in the tuple
didReport = true;
} else if (receiverElement.type === AST_NODE_TYPES.ArrayPattern) {
didReport = checkArrayDestructure(
receiverElement,
senderType,
senderNode,
);
} else if (receiverElement.type === AST_NODE_TYPES.ObjectPattern) {
didReport = checkObjectDestructure(
receiverElement,
senderType,
senderNode,
);
}
}
return didReport;
}
- Parameters:
receiverNode: TSESTree.ArrayPatternsenderType: ts.TypesenderNode: ts.Node- Return Type:
boolean - Calls:
isTypeAnyArrayType (from ../util)context.reportcreateDatachecker.isTupleTypechecker.getTypeArgumentsisTypeAnyType (from ../util)checkArrayDestructurecheckObjectDestructure- Internal Comments:
checkObjectDestructureHelper(receiverNode: TSESTree.Node, senderNode: TSESTree.Node): boolean¶
Code
function checkObjectDestructureHelper(
receiverNode: TSESTree.Node,
senderNode: TSESTree.Node,
): boolean {
if (receiverNode.type !== AST_NODE_TYPES.ObjectPattern) {
return false;
}
const senderTsNode = services.esTreeNodeToTSNodeMap.get(senderNode);
const senderType = services.getTypeAtLocation(senderNode);
return checkObjectDestructure(receiverNode, senderType, senderTsNode);
}
- Parameters:
receiverNode: TSESTree.NodesenderNode: TSESTree.Node- Return Type:
boolean - Calls:
services.esTreeNodeToTSNodeMap.getservices.getTypeAtLocationcheckObjectDestructure
checkObjectDestructure(receiverNode: TSESTree.ObjectPattern, senderType: ts.Type, senderNode: ts.Node): boolean¶
Code
function checkObjectDestructure(
receiverNode: TSESTree.ObjectPattern,
senderType: ts.Type,
senderNode: ts.Node,
): boolean {
const properties = new Map(
senderType
.getProperties()
.map(property => [
property.getName(),
checker.getTypeOfSymbolAtLocation(property, senderNode),
]),
);
let didReport = false;
for (const receiverProperty of receiverNode.properties) {
if (receiverProperty.type === AST_NODE_TYPES.RestElement) {
// don't bother checking rest
continue;
}
let key: string;
if (!receiverProperty.computed) {
key =
receiverProperty.key.type === AST_NODE_TYPES.Identifier
? receiverProperty.key.name
: String(receiverProperty.key.value);
} else if (receiverProperty.key.type === AST_NODE_TYPES.Literal) {
key = String(receiverProperty.key.value);
} else if (
receiverProperty.key.type === AST_NODE_TYPES.TemplateLiteral &&
receiverProperty.key.quasis.length === 1
) {
key = receiverProperty.key.quasis[0].value.cooked;
} else {
// can't figure out the name, so skip it
continue;
}
const senderType = properties.get(key);
if (!senderType) {
continue;
}
// check for the any type first so we can handle {x: {y: z}} = {x: any}
if (isTypeAnyType(senderType)) {
context.report({
node: receiverProperty.value,
messageId: 'unsafeArrayPatternFromTuple',
data: createData(senderType),
});
didReport = true;
} else if (
receiverProperty.value.type === AST_NODE_TYPES.ArrayPattern
) {
didReport = checkArrayDestructure(
receiverProperty.value,
senderType,
senderNode,
);
} else if (
receiverProperty.value.type === AST_NODE_TYPES.ObjectPattern
) {
didReport = checkObjectDestructure(
receiverProperty.value,
senderType,
senderNode,
);
}
}
return didReport;
}
- Parameters:
receiverNode: TSESTree.ObjectPatternsenderType: ts.TypesenderNode: ts.Node- Return Type:
boolean - Calls:
senderType .getProperties() .mapproperty.getNamechecker.getTypeOfSymbolAtLocationStringproperties.getisTypeAnyType (from ../util)context.reportcreateDatacheckArrayDestructurecheckObjectDestructure- Internal Comments:
checkAssignment(receiverNode: TSESTree.Node, senderNode: TSESTree.Expression, reportingNode: TSESTree.Node, comparisonType: ComparisonType): boolean¶
Code
function checkAssignment(
receiverNode: TSESTree.Node,
senderNode: TSESTree.Expression,
reportingNode: TSESTree.Node,
comparisonType: ComparisonType,
): boolean {
const receiverTsNode = services.esTreeNodeToTSNodeMap.get(receiverNode);
const receiverType =
comparisonType === ComparisonType.Contextual
? (getContextualType(checker, receiverTsNode as ts.Expression) ??
services.getTypeAtLocation(receiverNode))
: services.getTypeAtLocation(receiverNode);
const senderType = services.getTypeAtLocation(senderNode);
if (isTypeAnyType(senderType)) {
// handle cases when we assign any ==> unknown.
if (isTypeUnknownType(receiverType)) {
return false;
}
let messageId: 'anyAssignment' | 'anyAssignmentThis' = 'anyAssignment';
if (!isNoImplicitThis) {
// `var foo = this`
const thisExpression = getThisExpression(senderNode);
if (
thisExpression &&
isTypeAnyType(
getConstrainedTypeAtLocation(services, thisExpression),
)
) {
messageId = 'anyAssignmentThis';
}
}
context.report({
node: reportingNode,
messageId,
data: createData(senderType),
});
return true;
}
if (comparisonType === ComparisonType.None) {
return false;
}
const result = isUnsafeAssignment(
senderType,
receiverType,
checker,
senderNode,
);
if (!result) {
return false;
}
const { receiver, sender } = result;
context.report({
node: reportingNode,
messageId: 'unsafeAssignment',
data: createData(sender, receiver),
});
return true;
}
- Parameters:
receiverNode: TSESTree.NodesenderNode: TSESTree.ExpressionreportingNode: TSESTree.NodecomparisonType: ComparisonType- Return Type:
boolean - Calls:
services.esTreeNodeToTSNodeMap.getgetContextualType (from ../util)services.getTypeAtLocationisTypeAnyType (from ../util)isTypeUnknownType (from ../util)getThisExpression (from ../util)getConstrainedTypeAtLocation (from ../util)context.reportcreateDataisUnsafeAssignment (from ../util)- Internal Comments:
getComparisonType(typeAnnotation: TSESTree.TSTypeAnnotation | undefined): ComparisonType¶
Code
function getComparisonType(
typeAnnotation: TSESTree.TSTypeAnnotation | undefined,
): ComparisonType {
return typeAnnotation
? // if there's a type annotation, we can do a comparison
ComparisonType.Basic
: // no type annotation means the variable's type will just be inferred, thus equal
ComparisonType.None;
}
- Parameters:
typeAnnotation: TSESTree.TSTypeAnnotation | undefined- Return Type:
ComparisonType
createData(senderType: ts.Type, receiverType: ts.Type): Readonly<Record<string, unknown>> | undefined¶
Code
function createData(
senderType: ts.Type,
receiverType?: ts.Type,
): Readonly<Record<string, unknown>> | undefined {
if (receiverType) {
return {
receiver: `\`${checker.typeToString(receiverType)}\``,
sender: `\`${checker.typeToString(senderType)}\``,
};
}
return {
sender: tsutils.isIntrinsicErrorType(senderType)
? 'error typed'
: '`any`',
};
}
- Parameters:
senderType: ts.TypereceiverType: ts.Type- Return Type:
Readonly<Record<string, unknown>> | undefined - Calls:
checker.typeToStringtsutils.isIntrinsicErrorType
Enums¶
const enum ComparisonType¶
Enum Code
Members¶
| Name | Value | Description |
|---|---|---|
None |
auto | / Do no assignment comparison */ |
Basic |
auto | / Use the receiver's type for comparison */ |
Contextual |
auto | / Use the sender's contextual type for comparison */ |