Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

issue diagnostic for a templated model with a template parameter that is not used. #5494

Merged
merged 42 commits into from
Feb 28, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
dd61afc
check unused template parameter
chunyu3 Dec 29, 2024
bfec5ab
check model template parameters in CheckNode
chunyu3 Jan 2, 2025
836d107
add template parameter delaration node into usedTemplateParameter set…
chunyu3 Jan 5, 2025
4b36411
resolve template argument in union and decorator
chunyu3 Jan 6, 2025
85a1d0c
remove unused code
chunyu3 Jan 6, 2025
dc1ca22
implement codefix for unused template parameter
chunyu3 Jan 6, 2025
8ca3c08
refine codefix
chunyu3 Jan 6, 2025
a148269
add constrain to check unused template parameter for only model template
chunyu3 Jan 6, 2025
c6020d2
refine the function name
chunyu3 Jan 6, 2025
3a47d98
implement Unit test for unused template parameter
chunyu3 Jan 6, 2025
9b8dce6
Merge branch 'main' of https://github.com/microsoft/typespec into fix181
chunyu3 Jan 6, 2025
75b0e23
add changelog
chunyu3 Jan 6, 2025
09226da
format
chunyu3 Jan 6, 2025
0bc9432
support template parameter used in base model or typeof expression
chunyu3 Jan 7, 2025
ffe81e1
use templateMapper instead of deprecatd templateArguments
chunyu3 Jan 7, 2025
6fb8703
add changelog
chunyu3 Jan 7, 2025
65e728b
remove unused if check
chunyu3 Jan 7, 2025
ad609ca
Merge branch 'main' of https://github.com/microsoft/typespec into fix181
chunyu3 Jan 15, 2025
2f160b5
Update packages/compiler/src/core/messages.ts
chunyu3 Jan 15, 2025
0c86369
remove suppression
chunyu3 Jan 15, 2025
2275729
Merge branch 'main' of https://github.com/microsoft/typespec into fix181
chunyu3 Feb 20, 2025
56c8990
format
chunyu3 Feb 20, 2025
d7acad3
correct the unused-template-parameter message in the test
chunyu3 Feb 20, 2025
8b88990
remove unnessary unused-template-parameter logic in checker
chunyu3 Feb 20, 2025
cf6dd79
remove unnessary change log md
chunyu3 Feb 20, 2025
71ff8de
use build in lint for unused template parameter
chunyu3 Feb 20, 2025
059a512
Merge branch 'main' into fix181
chunyu3 Feb 20, 2025
a721abc
track unsedTemplateParamterNodes in check
chunyu3 Feb 21, 2025
8a9d527
Merge branch 'fix181' of https://github.com/chunyu3/typespec into fix181
chunyu3 Feb 21, 2025
d07a498
Merge branch 'main' into fix181
chunyu3 Feb 21, 2025
7a51567
Merge branch 'main' of https://github.com/microsoft/typespec into fix181
chunyu3 Feb 24, 2025
68fcefc
verify template parameter usage in a general place
chunyu3 Feb 24, 2025
c59a2c2
calculate unused template parameter in checkNode
chunyu3 Feb 24, 2025
f3654ce
add more unit test
chunyu3 Feb 25, 2025
5c25f07
check and set the usage of template parameters and calculate unused t…
chunyu3 Feb 25, 2025
0c91355
Merge branch 'main' into fix181
chunyu3 Feb 25, 2025
632898c
separate linter rules to different files
chunyu3 Feb 26, 2025
a04bd8f
Merge branch 'fix181' of https://github.com/chunyu3/typespec into fix181
chunyu3 Feb 26, 2025
f273eac
format
chunyu3 Feb 26, 2025
074e29c
test unused-template-parameter linter
chunyu3 Feb 26, 2025
ca909b4
Merge branch 'main' into fix181
chunyu3 Feb 26, 2025
947b1d2
Merge branch 'main' into fix181
chunyu3 Feb 26, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions .chronus/changes/fix181-2025-0-6-16-32-31.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
---
changeKind: feature
packages:
- "@typespec/compiler"
---

Support diagnostics for unused template parameter
32 changes: 31 additions & 1 deletion packages/compiler/src/core/checker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,9 @@ export interface Checker {
/** @internal */
getTypeOrValueForNode(node: Node): Type | Value | null;

/** @internal */
getTemplateParameterUsageMap(): Map<TemplateParameterDeclarationNode, boolean>;

readonly errorType: ErrorType;
readonly voidType: VoidType;
readonly neverType: NeverType;
Expand Down Expand Up @@ -401,6 +404,11 @@ export function createChecker(program: Program, resolver: NameResolver): Checker

let evalContext: EvalContext | undefined = undefined;

/**
* Tracking the template parameters used or not.
*/
const templateParameterUsageMap = new Map<TemplateParameterDeclarationNode, boolean>();

const checker: Checker = {
getTypeForNode,
checkProgram,
Expand Down Expand Up @@ -433,6 +441,7 @@ export function createChecker(program: Program, resolver: NameResolver): Checker
getValueForNode,
getTypeOrValueForNode,
getValueExactType,
getTemplateParameterUsageMap,
isTypeAssignableTo: undefined!,
};
const relation = createTypeRelationChecker(program, checker);
Expand All @@ -441,6 +450,10 @@ export function createChecker(program: Program, resolver: NameResolver): Checker
const projectionMembers = createProjectionMembers(checker);
return checker;

function getTemplateParameterUsageMap(): Map<TemplateParameterDeclarationNode, boolean> {
return templateParameterUsageMap;
}

function wrapInstantiationDiagnostic(
diagnostic: Diagnostic,
templateMapper?: TypeMapper,
Expand Down Expand Up @@ -1066,6 +1079,10 @@ export function createChecker(program: Program, resolver: NameResolver): Checker
const grandParentNode = parentNode.parent;
const links = getSymbolLinks(node.symbol);

if (!templateParameterUsageMap.has(node)) {
templateParameterUsageMap.set(node, false);
}

if (pendingResolutions.has(getNodeSym(node), ResolutionKind.Constraint)) {
if (mapper === undefined) {
reportCheckerDiagnostic(
Expand Down Expand Up @@ -1112,7 +1129,6 @@ export function createChecker(program: Program, resolver: NameResolver): Checker
);
}
}

return mapper ? mapper.getMappedType(type) : type;
}

Expand Down Expand Up @@ -1516,6 +1532,20 @@ export function createChecker(program: Program, resolver: NameResolver): Checker
node: TypeReferenceNode | MemberExpressionNode | IdentifierNode,
mapper: TypeMapper | undefined,
instantiateTemplates = true,
): Type | Value | IndeterminateEntity | null {
const entity = checkTypeOrValueReferenceSymbolWorker(sym, node, mapper, instantiateTemplates);

if (entity !== null && isType(entity) && entity.kind === "TemplateParameter") {
templateParameterUsageMap.set(entity.node, true);
}
return entity;
}

function checkTypeOrValueReferenceSymbolWorker(
sym: Sym,
node: TypeReferenceNode | MemberExpressionNode | IdentifierNode,
mapper: TypeMapper | undefined,
instantiateTemplates = true,
): Type | Value | IndeterminateEntity | null {
if (sym.flags & SymbolFlags.Const) {
return getValueForNode(sym.declarations[0], mapper);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { defineCodeFix, getSourceLocation } from "../diagnostics.js";
import { TemplateParameterDeclarationNode } from "../types.js";

/**
* Quick fix that remove unused template parameter.
*/
export function removeUnusedTemplateParameterCodeFix(node: TemplateParameterDeclarationNode) {
return defineCodeFix({
id: "remove-unused-template-parameter",
label: `Remove unused template parameter ${node.id.sv}`,
fix: (context) => {
let location = getSourceLocation(node);
const parent = node.parent;
if (parent) {
const length = parent.templateParameters.length;
if (length === 1) {
location = {
file: location.file,
pos: parent.templateParametersRange.pos,
end: parent.templateParametersRange.end,
};
} else {
const index = parent.templateParameters.findIndex((param) => param === node);
if (index !== -1) {
if (index !== parent.templateParameters.length - 1) {
location = {
file: location.file,
pos: location.pos,
end: parent.templateParameters[index + 1].pos,
};
} else {
location = {
file: location.file,
pos: parent.templateParameters[index - 1].end,
end: location.end,
};
}
}
}
}
return context.replaceText(location, "");
},
});
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
import { removeUnusedTemplateParameterCodeFix } from "../compiler-code-fixes/remove-unused-template-parameter.codefix.js";
import { createLinterRule, paramMessage } from "../library.js";

export const builtInLinterRule_UnusedTemplateParameter = `unused-template-parameter`;

/** @internal */
export function createUnusedTemplateParameterLinterRule() {
return createLinterRule({
name: builtInLinterRule_UnusedTemplateParameter,
severity: "warning",
description: "Linter rules for unused template parameter.",
messages: {
default: paramMessage`Templates should use all specified parameters, and parameter '${"parameterName"}' does not exist in type '${"type"}'. Consider removing this parameter.`,
},
create(context) {
return {
root: (_root) => {
const templateParameters = context.program.checker
.getTemplateParameterUsageMap()
.entries();
for (const [templateParameter, used] of templateParameters) {
if (!used) {
context.reportDiagnostic({
format: {
parameterName: templateParameter.id.sv,
type: templateParameter.parent?.symbol.name ?? "",
},
target: templateParameter,
codefixes: [removeUnusedTemplateParameterCodeFix(templateParameter)],
});
}
}
},
};
},
});
}
39 changes: 39 additions & 0 deletions packages/compiler/src/core/linter-rules/unused-using.rule.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
import { removeUnusedCodeCodeFix } from "../compiler-code-fixes/remove-unused-code.codefix.js";
import { createLinterRule, paramMessage } from "../library.js";
import { NameResolver } from "../name-resolver.js";
import { IdentifierNode, MemberExpressionNode, SyntaxKind } from "../types.js";

export const builtInLinterRule_UnusedUsing = `unused-using`;

/** @internal */
export function createUnusedUsingLinterRule(nameResolver: NameResolver) {
return createLinterRule({
name: builtInLinterRule_UnusedUsing,
severity: "warning",
description: "Linter rules for unused using statement.",
messages: {
default: paramMessage`'using ${"code"}' is declared but never be used.`,
},
create(context) {
return {
root: (_root) => {
const getUsingName = (node: MemberExpressionNode | IdentifierNode): string => {
if (node.kind === SyntaxKind.MemberExpression) {
return `${getUsingName(node.base)}${node.selector}${node.id.sv}`;
} else {
// identifier node
return node.sv;
}
};
nameResolver.getUnusedUsings().forEach((target) => {
context.reportDiagnostic({
format: { code: getUsingName(target.name) },
target,
codefixes: [removeUnusedCodeCodeFix(target)],
});
});
},
};
},
});
}
48 changes: 7 additions & 41 deletions packages/compiler/src/core/linter.ts
Original file line number Diff line number Diff line change
@@ -1,26 +1,24 @@
import { removeUnusedCodeCodeFix } from "./compiler-code-fixes/remove-unused-code.codefix.js";
import { DiagnosticCollector, compilerAssert, createDiagnosticCollector } from "./diagnostics.js";
import { getLocationContext } from "./helpers/location-context.js";
import { createLinterRule, defineLinter, paramMessage } from "./library.js";
import { defineLinter } from "./library.js";
import { createUnusedTemplateParameterLinterRule } from "./linter-rules/unused-template-parameter.rule.js";
import { createUnusedUsingLinterRule } from "./linter-rules/unused-using.rule.js";
import { createDiagnostic } from "./messages.js";
import { NameResolver } from "./name-resolver.js";
import type { Program } from "./program.js";
import { EventEmitter, mapEventEmitterToNodeListener, navigateProgram } from "./semantic-walker.js";
import {
Diagnostic,
DiagnosticMessages,
IdentifierNode,
LinterDefinition,
LinterResolvedDefinition,
LinterRule,
LinterRuleContext,
LinterRuleDiagnosticReport,
LinterRuleSet,
MemberExpressionNode,
NoTarget,
RuleRef,
SemanticNodeListener,
SyntaxKind,
} from "./types.js";

type LinterLibraryInstance = { linter: LinterResolvedDefinition };
Expand Down Expand Up @@ -267,7 +265,6 @@ export function createLinterRuleContext<N extends string, DM extends DiagnosticM
}

export const builtInLinterLibraryName = `@typespec/compiler`;
export const builtInLinterRule_UnusedUsing = `unused-using`;
export function createBuiltInLinterLibrary(nameResolver: NameResolver): LinterLibraryInstance {
const builtInLinter: LinterResolvedDefinition = resolveLinterDefinition(
builtInLinterLibraryName,
Expand All @@ -276,42 +273,11 @@ export function createBuiltInLinterLibrary(nameResolver: NameResolver): LinterLi
return { linter: builtInLinter };
}
function createBuiltInLinter(nameResolver: NameResolver): LinterDefinition {
const unusedUsingLinterRule = createUnusedUsingLinterRule();
const unusedUsingLinterRule = createUnusedUsingLinterRule(nameResolver);

const unusedTemplateParameterLinterRule = createUnusedTemplateParameterLinterRule();

return defineLinter({
rules: [unusedUsingLinterRule],
rules: [unusedUsingLinterRule, unusedTemplateParameterLinterRule],
});

function createUnusedUsingLinterRule() {
return createLinterRule({
name: builtInLinterRule_UnusedUsing,
severity: "warning",
description: "Linter rules for unused using statement.",
messages: {
default: paramMessage`'using ${"code"}' is declared but never be used.`,
},
create(context) {
return {
root: (_root) => {
const getUsingName = (node: MemberExpressionNode | IdentifierNode): string => {
if (node.kind === SyntaxKind.MemberExpression) {
return `${getUsingName(node.base)}${node.selector}${node.id.sv}`;
} else {
// identifier node
return node.sv;
}
};
nameResolver.getUnusedUsings().forEach((target) => {
context.reportDiagnostic({
messageId: "default",
format: { code: getUsingName(target.name) },
target,
codefixes: [removeUnusedCodeCodeFix(target)],
});
});
},
};
},
});
}
}
21 changes: 21 additions & 0 deletions packages/compiler/src/core/name-resolver.ts
Original file line number Diff line number Diff line change
Expand Up @@ -126,6 +126,12 @@ export interface NameResolver {
/** Return augment decorator nodes that are bound to this symbol */
getAugmentDecoratorsForSym(symbol: Sym): AugmentDecoratorStatementNode[];

/** Return template parameters which are not used. */
getUnusedTemplateParameterDeclarationNodes(): Set<TemplateParameterDeclarationNode>;

/** Set unused template parameter */
setUnusedTemplateParameterDeclarationNode(node: TemplateParameterDeclarationNode): void;

/**
* Resolve the member expression using the given symbol as base.
* This can be used to follow the name resolution for template instance which are not statically linked.
Expand Down Expand Up @@ -190,6 +196,11 @@ export function createResolver(program: Program): NameResolver {
const nullSym = createSymbol(undefined, "null", SymbolFlags.None);
const augmentDecoratorsForSym = new Map<Sym, AugmentDecoratorStatementNode[]>();

/**
* Tracking the template parameters that are not used.
*/
const unusedTemplateParameterDeclarationNodes = new Set<TemplateParameterDeclarationNode>();

return {
symbols: { global: globalNamespaceSym, null: nullSym },
resolveProgram() {
Expand Down Expand Up @@ -233,6 +244,8 @@ export function createResolver(program: Program): NameResolver {

getAugmentDecoratorsForSym,
getUnusedUsings,
getUnusedTemplateParameterDeclarationNodes,
setUnusedTemplateParameterDeclarationNode,
};

function getUnusedUsings(): UsingStatementNode[] {
Expand Down Expand Up @@ -268,6 +281,14 @@ export function createResolver(program: Program): NameResolver {
return mergedSymbols.get(sym) || sym;
}

function setUnusedTemplateParameterDeclarationNode(node: TemplateParameterDeclarationNode): void {
if (!node) return;
unusedTemplateParameterDeclarationNodes.add(node);
}

function getUnusedTemplateParameterDeclarationNodes(): Set<TemplateParameterDeclarationNode> {
return unusedTemplateParameterDeclarationNodes;
}
/**
* @internal
*/
Expand Down
15 changes: 14 additions & 1 deletion packages/compiler/src/server/compile-service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,9 @@ import {
joinPaths,
parse,
} from "../core/index.js";
import { builtInLinterLibraryName, builtInLinterRule_UnusedUsing } from "../core/linter.js";
import { builtInLinterRule_UnusedTemplateParameter } from "../core/linter-rules/unused-template-parameter.rule.js";
import { builtInLinterRule_UnusedUsing } from "../core/linter-rules/unused-using.rule.js";
import { builtInLinterLibraryName } from "../core/linter.js";
import { compile as compileProgram } from "../core/program.js";
import { doIO, loadFile, resolveTspMain } from "../utils/misc.js";
import { serverOptions } from "./constants.js";
Expand Down Expand Up @@ -119,6 +121,17 @@ export function createCompileService({
options.linterRuleSet.enable[unusedUsingRule] = true;
}

// add linter rule for unused template parameter if user didn't configure it explicitly
const unusedTemplateParameterRule = `${builtInLinterLibraryName}/${builtInLinterRule_UnusedTemplateParameter}`;
if (
options.linterRuleSet?.enable?.[unusedTemplateParameterRule] === undefined &&
options.linterRuleSet?.disable?.[unusedTemplateParameterRule] === undefined
) {
options.linterRuleSet ??= {};
options.linterRuleSet.enable ??= {};
options.linterRuleSet.enable[unusedTemplateParameterRule] = true;
}

log({ level: "debug", message: `compiler options resolved`, detail: options });

if (!fileService.upToDate(document)) {
Expand Down
Loading
Loading