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

Extract method #16960

Closed
wants to merge 60 commits into from
Closed
Show file tree
Hide file tree
Changes from 53 commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
be27a1a
Rebased EM WIP
RyanCavanaugh Jun 30, 2017
e798257
Change refactor api in fourslash
RyanCavanaugh Jul 5, 2017
bf176a6
Add comments
RyanCavanaugh Jul 5, 2017
0130ebb
Better baselines
RyanCavanaugh Jul 5, 2017
bbf5cd7
Rename test we we can run it independent of unit tests
RyanCavanaugh Jul 5, 2017
9dd6bef
Change fourslash refactor API
RyanCavanaugh Jul 5, 2017
f21b7ae
Remove invalid assert
RyanCavanaugh Jul 5, 2017
674b1bf
Fix undefined computation of length
RyanCavanaugh Jul 5, 2017
9367473
Improved type baselines for EM .js
RyanCavanaugh Jul 5, 2017
1ea9972
Add validation to fourslash test
RyanCavanaugh Jul 5, 2017
1787fa8
Move text to diagnostic messages
RyanCavanaugh Jul 5, 2017
fb89c00
Merge branch 'master' into extract-method
RyanCavanaugh Jul 5, 2017
4aef5e7
Properly widen literal param types + emit correct return statements f…
RyanCavanaugh Jul 5, 2017
b5c5c40
Don't offer to extract lone identifiers
RyanCavanaugh Jul 6, 2017
b1fa49f
Don't fail if span is zero-length
RyanCavanaugh Jul 6, 2017
825237b
Don't crash when span is empty
RyanCavanaugh Jul 6, 2017
aa6f3aa
Restore assert and fix synthetic type node creation
RyanCavanaugh Jul 18, 2017
b7e4451
Re-use noop
RyanCavanaugh Jul 18, 2017
419a4c2
Address portion of CR feedback
RyanCavanaugh Jul 18, 2017
4218738
Only compute changes for requested refactors
RyanCavanaugh Jul 19, 2017
6d2bc90
Address more PR comments
RyanCavanaugh Jul 19, 2017
9a24549
Merge master
RyanCavanaugh Jul 19, 2017
271ecb1
Remove WS
RyanCavanaugh Jul 19, 2017
bdefcc3
Emit return types in ctxly-typed functions and don't emit annotations…
RyanCavanaugh Jul 19, 2017
6428b1e
Fix broken test
RyanCavanaugh Jul 19, 2017
8af8ac2
Handle fn params and duplicated scope names properly
RyanCavanaugh Jul 19, 2017
ede6671
Re-use isJS result
RyanCavanaugh Jul 19, 2017
55b28ad
You may not extract-method on exported declarations
RyanCavanaugh Jul 19, 2017
2d9eaa9
Correctly apply overlapped edits in fourslash
RyanCavanaugh Jul 20, 2017
2f2c780
Fix some parenting issues & start detecting stranded references
RyanCavanaugh Jul 20, 2017
63193da
Add return type annotation and regression test for a prior assert
RyanCavanaugh Jul 20, 2017
889292a
Detect would-be stranded references to extracted declarations
RyanCavanaugh Jul 24, 2017
e229d42
Tests
RyanCavanaugh Jul 24, 2017
7b2dd8a
Fix line ending
RyanCavanaugh Jul 24, 2017
bf6c0a9
Function blocks aren't statements!
RyanCavanaugh Jul 25, 2017
3424312
Disallow certain classes of refactors
RyanCavanaugh Jul 25, 2017
ff18df3
Add test
RyanCavanaugh Jul 25, 2017
7f29fcd
Handle static methods and static method name conflicts
RyanCavanaugh Jul 25, 2017
d39d95b
Merge
RyanCavanaugh Jul 25, 2017
dda5496
No extracting other non-statement blocks
RyanCavanaugh Jul 28, 2017
e88ff21
More disallowed extraction position
RyanCavanaugh Jul 28, 2017
bd80005
Block more nonworking extraction locations
RyanCavanaugh Jul 28, 2017
ff716d0
Properly enumerate class blocks
RyanCavanaugh Jul 28, 2017
1291cfd
Add tests task
RyanCavanaugh Jul 28, 2017
a26790f
Don't re-use nodes
RyanCavanaugh Jul 28, 2017
56e1d6d
Correctly detect JavaScriptness
RyanCavanaugh Jul 28, 2017
7f3a0a8
Create multi-line bodies
RyanCavanaugh Jul 31, 2017
003cd0a
Add support for validating specific refactor actions' availability
RyanCavanaugh Aug 3, 2017
a76664c
More testcases
RyanCavanaugh Aug 3, 2017
42e0652
Accept baselines
RyanCavanaugh Aug 3, 2017
46a84c3
Innumerable bugfixes
RyanCavanaugh Aug 3, 2017
a97ab93
Merge branch 'master' into extract-method
RyanCavanaugh Aug 3, 2017
37ae957
Add additional test
RyanCavanaugh Aug 3, 2017
2adb2b3
Fix some class scenarios (static methods, readonly inits in ctor)
RyanCavanaugh Aug 4, 2017
ba37830
Prevent extraction of exported variables
RyanCavanaugh Aug 4, 2017
4dafb40
No extraction of ambient declarations
RyanCavanaugh Aug 4, 2017
c1c7e83
Put __return first
RyanCavanaugh Aug 4, 2017
a4c41b8
Don't wrongly exclude accessor in indexed access exprs
RyanCavanaugh Aug 4, 2017
b5a88c1
Share selection logic; fix typos
RyanCavanaugh Aug 4, 2017
568d5f6
Merge branch 'master' into extract-method
RyanCavanaugh Aug 4, 2017
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 .vscode/tasks.json
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,13 @@
"problemMatcher": [
"$tsc"
]
},
{
"taskName": "tests",
"showOutput": "silent",
"problemMatcher": [
"$tsc"
]
}
]
}
1 change: 1 addition & 0 deletions Jakefile.js
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,7 @@ var harnessSources = harnessCoreSources.concat([
"projectErrors.ts",
"matchFiles.ts",
"initializeTSConfig.ts",
"extractMethods.ts",
"printer.ts",
"textChanges.ts",
"telemetry.ts",
Expand Down
7 changes: 3 additions & 4 deletions src/compiler/checker.ts
Original file line number Diff line number Diff line change
Expand Up @@ -223,11 +223,10 @@ namespace ts {
getSuggestionForNonexistentProperty: (node, type) => unescapeLeadingUnderscores(getSuggestionForNonexistentProperty(node, type)),
getSuggestionForNonexistentSymbol: (location, name, meaning) => unescapeLeadingUnderscores(getSuggestionForNonexistentSymbol(location, escapeLeadingUnderscores(name), meaning)),
getBaseConstraintOfType,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this the same as resolveNameAtLocation, but with a changed argument order and no looking up of parse tree nodes (I can only assume so it works on synthesized nodes with parent pointers setup)? Can we not just remove the looking up of parse tree nodes in resolveNameAtLocation if that's the case? Oh, nameArg is also undefined. Can that not just be added as an optional parameter to this existing function?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@rbuckton would know authoritatively, but as I understand it, we check isParseTreeNode on public API requests because the checker does not work well with synthesized nodes, since we don't perform binding on synthesized nodes as part of transforms (for performance reasons?) and we may inspect the literal text as part of some checker operations. That is, the checker API is only supposed to work with trees parsed and bound from an actual source file.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@aozgaa is correct. If the checker receives a node that did not originate from a parse tree, we attempt to find the parse tree node from which it was created. Passing in a synthesized node to the checker without a parse-tree check will likely result in compiler crashes.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That said, resolveNameAtLocation (nee. resolveName) is marked /* @internal */ and isn't used in transforms. The guard is mostly present in case we decide to use it at some point down the road.

getJsxNamespace: () => unescapeLeadingUnderscores(getJsxNamespace()),
resolveNameAtLocation(location: Node, name: string, meaning: SymbolFlags): Symbol | undefined {
location = getParseTreeNode(location);
return resolveName(location, escapeLeadingUnderscores(name), meaning, /*nameNotFoundMessage*/ undefined, escapeLeadingUnderscores(name));
resolveName(name, location, meaning) {
return resolveName(location, name as __String, meaning, /*nameNotFoundMessage*/ undefined, /*nameArg*/ undefined);
},
getJsxNamespace: () => unescapeLeadingUnderscores(getJsxNamespace()),
};

const tupleTypes: GenericType[] = [];
Expand Down
10 changes: 10 additions & 0 deletions src/compiler/diagnosticMessages.json
Original file line number Diff line number Diff line change
Expand Up @@ -3673,5 +3673,15 @@
"Convert function '{0}' to class": {
"category": "Message",
"code": 95002
},

"Extract function": {
"category": "Message",
"code": 95003
},

"Extract function into '{0}'": {
"category": "Message",
"code": 95004
}
}
2 changes: 1 addition & 1 deletion src/compiler/transformers/es2015.ts
Original file line number Diff line number Diff line change
Expand Up @@ -394,7 +394,7 @@ namespace ts {
function shouldVisitNode(node: Node): boolean {
return (node.transformFlags & TransformFlags.ContainsES2015) !== 0
|| convertedLoopState !== undefined
|| (hierarchyFacts & HierarchyFacts.ConstructorWithCapturedSuper && isStatement(node))
|| (hierarchyFacts & HierarchyFacts.ConstructorWithCapturedSuper && isStatementOrBlock(node))
|| (isIterationStatement(node, /*lookInLabeledStatements*/ false) && shouldConvertIterationStatementBody(node))
|| isTypeScriptClassWrapper(node);
}
Expand Down
3 changes: 1 addition & 2 deletions src/compiler/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2606,9 +2606,8 @@ namespace ts {
* Does not include properties of primitive types.
*/
/* @internal */ getAllPossiblePropertiesOfType(type: Type): Symbol[];

/* @internal */ resolveName(name: string, location: Node, meaning: SymbolFlags): Symbol | undefined;
/* @internal */ getJsxNamespace(): string;
/* @internal */ resolveNameAtLocation(location: Node, name: string, meaning: SymbolFlags): Symbol | undefined;
}

export enum NodeBuilderFlags {
Expand Down
49 changes: 40 additions & 9 deletions src/compiler/utilities.ts
Original file line number Diff line number Diff line change
Expand Up @@ -693,7 +693,7 @@ namespace ts {
// At this point, node is either a qualified name or an identifier
Debug.assert(node.kind === SyntaxKind.Identifier || node.kind === SyntaxKind.QualifiedName || node.kind === SyntaxKind.PropertyAccessExpression,
"'node' was expected to be a qualified name, identifier or property access in 'isPartOfTypeNode'.");
// falls through
// falls through
case SyntaxKind.QualifiedName:
case SyntaxKind.PropertyAccessExpression:
case SyntaxKind.ThisKeyword:
Expand Down Expand Up @@ -978,7 +978,7 @@ namespace ts {
if (!includeArrowFunctions) {
continue;
}
// falls through
// falls through
case SyntaxKind.FunctionDeclaration:
case SyntaxKind.FunctionExpression:
case SyntaxKind.ModuleDeclaration:
Expand Down Expand Up @@ -1037,7 +1037,7 @@ namespace ts {
if (!stopOnFunctions) {
continue;
}
// falls through
// falls through
case SyntaxKind.PropertyDeclaration:
case SyntaxKind.PropertySignature:
case SyntaxKind.MethodDeclaration:
Expand Down Expand Up @@ -1221,7 +1221,7 @@ namespace ts {
if (node.parent.kind === SyntaxKind.TypeQuery || isJSXTagName(node)) {
return true;
}
// falls through
// falls through
case SyntaxKind.NumericLiteral:
case SyntaxKind.StringLiteral:
case SyntaxKind.ThisKeyword:
Expand Down Expand Up @@ -1509,8 +1509,8 @@ namespace ts {
parent.parent.kind === SyntaxKind.VariableStatement;
const variableStatementNode =
isInitializerOfVariableDeclarationInStatement ? parent.parent.parent :
isVariableOfVariableDeclarationStatement ? parent.parent :
undefined;
isVariableOfVariableDeclarationStatement ? parent.parent :
undefined;
if (variableStatementNode) {
getJSDocCommentsAndTagsWorker(variableStatementNode);
}
Expand Down Expand Up @@ -1624,7 +1624,7 @@ namespace ts {
if (isInJavaScriptFile(node)) {
if (node.type && node.type.kind === SyntaxKind.JSDocVariadicType ||
forEach(getJSDocParameterTags(node),
t => t.typeExpression && t.typeExpression.type.kind === SyntaxKind.JSDocVariadicType)) {
t => t.typeExpression && t.typeExpression.type.kind === SyntaxKind.JSDocVariadicType)) {
return true;
}
}
Expand Down Expand Up @@ -1917,7 +1917,7 @@ namespace ts {
if (node.asteriskToken) {
flags |= FunctionFlags.Generator;
}
// falls through
// falls through
case SyntaxKind.ArrowFunction:
if (hasModifier(node, ModifierFlags.Async)) {
flags |= FunctionFlags.Async;
Expand Down Expand Up @@ -5133,6 +5133,19 @@ namespace ts {
return isUnaryExpressionKind(skipPartiallyEmittedExpressions(node).kind);
}

/* @internal */
export function isUnaryExpressionWithWrite(expr: Node): expr is PrefixUnaryExpression | PostfixUnaryExpression {
switch (expr.kind) {
case SyntaxKind.PostfixUnaryExpression:
return true;
case SyntaxKind.PrefixUnaryExpression:
return (<PrefixUnaryExpression>expr).operator === SyntaxKind.PlusPlusToken ||
(<PrefixUnaryExpression>expr).operator === SyntaxKind.MinusMinusToken;
default:
return false;
}
}

function isExpressionKind(kind: SyntaxKind) {
return kind === SyntaxKind.ConditionalExpression
|| kind === SyntaxKind.YieldExpression
Expand Down Expand Up @@ -5347,7 +5360,25 @@ namespace ts {
const kind = node.kind;
return isStatementKindButNotDeclarationKind(kind)
|| isDeclarationStatementKind(kind)
|| kind === SyntaxKind.Block;
|| isBlockStatement(node);
}

/* @internal */
export function isStatementOrBlock(node: Node): node is Statement {
const kind = node.kind;
return isStatementKindButNotDeclarationKind(kind)
|| isDeclarationStatementKind(kind)
|| node.kind === SyntaxKind.Block;
}

function isBlockStatement(node: Node): node is Block {
if (node.kind !== SyntaxKind.Block) return false;
if (node.parent !== undefined) {
if (node.parent.kind === SyntaxKind.TryStatement || node.parent.kind === SyntaxKind.CatchClause) {
return false;
}
}
return !isFunctionBlock(node);
}

// Module references
Expand Down
91 changes: 75 additions & 16 deletions src/harness/fourslash.ts
Original file line number Diff line number Diff line change
Expand Up @@ -187,6 +187,9 @@ namespace FourSlash {

// The current caret position in the active file
public currentCaretPosition = 0;
// The position of the end of the current selection, or -1 if nothing is selected
public selectionEnd = -1;

public lastKnownMarker = "";

// The file that's currently 'opened'
Expand Down Expand Up @@ -433,11 +436,19 @@ namespace FourSlash {

public goToPosition(pos: number) {
this.currentCaretPosition = pos;
this.selectionEnd = -1;
}

public select(startMarker: string, endMarker: string) {
const start = this.getMarkerByName(startMarker), end = this.getMarkerByName(endMarker);
this.goToPosition(start.position);
this.selectionEnd = end.position;
}

public moveCaretRight(count = 1) {
this.currentCaretPosition += count;
this.currentCaretPosition = Math.min(this.currentCaretPosition, this.getFileContent(this.activeFile.fileName).length);
this.selectionEnd = -1;
}

// Opens a file given its 0-based index or fileName
Expand Down Expand Up @@ -980,9 +991,9 @@ namespace FourSlash {
}

for (const reference of expectedReferences) {
const {fileName, start, end} = reference;
const { fileName, start, end } = reference;
if (reference.marker && reference.marker.data) {
const {isWriteAccess, isDefinition} = reference.marker.data;
const { isWriteAccess, isDefinition } = reference.marker.data;
this.verifyReferencesWorker(actualReferences, fileName, start, end, isWriteAccess, isDefinition);
}
else {
Expand Down Expand Up @@ -1193,7 +1204,7 @@ namespace FourSlash {
displayParts: ts.SymbolDisplayPart[],
documentation: ts.SymbolDisplayPart[],
tags: ts.JSDocTagInfo[]
) {
) {

const actualQuickInfo = this.languageService.getQuickInfoAtPosition(this.activeFile.fileName, this.currentCaretPosition);
assert.equal(actualQuickInfo.kind, kind, this.messageAtLastKnownMarker("QuickInfo kind"));
Expand Down Expand Up @@ -1789,19 +1800,16 @@ namespace FourSlash {
// We get back a set of edits, but langSvc.editScript only accepts one at a time. Use this to keep track
// of the incremental offset from each edit to the next. We assume these edit ranges don't overlap

edits = edits.sort((a, b) => a.span.start - b.span.start);
for (let i = 0; i < edits.length - 1; i++) {
const firstEditSpan = edits[i].span;
const firstEditEnd = firstEditSpan.start + firstEditSpan.length;
assert.isTrue(firstEditEnd <= edits[i + 1].span.start);
}
// Copy this so we don't ruin someone else's copy
edits = JSON.parse(JSON.stringify(edits));

// Get a snapshot of the content of the file so we can make sure any formatting edits didn't destroy non-whitespace characters
const oldContent = this.getFileContent(fileName);
let runningOffset = 0;

for (const edit of edits) {
const offsetStart = edit.span.start + runningOffset;
for (let i = 0; i < edits.length; i++) {
const edit = edits[i];
const offsetStart = edit.span.start;
const offsetEnd = offsetStart + edit.span.length;
this.editScriptAndUpdateMarkers(fileName, offsetStart, offsetEnd, edit.newText);
const editDelta = edit.newText.length - edit.span.length;
Expand All @@ -1816,8 +1824,13 @@ namespace FourSlash {
}
}
runningOffset += editDelta;
// TODO: Consider doing this at least some of the time for higher fidelity. Currently causes a failure (bug 707150)
// this.languageService.getScriptLexicalStructure(fileName);

// Update positions of any future edits affected by this change
for (let j = i + 1; j < edits.length; j++) {
if (edits[j].span.start >= edits[i].span.start) {
edits[j].span.start += editDelta;
}
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Could you eliminate the need to do this by walking the list backwards?

}

if (isFormattingEdit) {
Expand Down Expand Up @@ -1901,7 +1914,7 @@ namespace FourSlash {
this.goToPosition(len);
}

public goToRangeStart({fileName, start}: Range) {
public goToRangeStart({ fileName, start }: Range) {
this.openFile(fileName);
this.goToPosition(start);
}
Expand Down Expand Up @@ -2075,7 +2088,7 @@ namespace FourSlash {
return result;
}

private rangeText({fileName, start, end}: Range): string {
private rangeText({ fileName, start, end }: Range): string {
return this.getFileContent(fileName).slice(start, end);
}

Expand Down Expand Up @@ -2361,7 +2374,7 @@ namespace FourSlash {
private applyCodeActions(actions: ts.CodeAction[], index?: number): void {
if (index === undefined) {
if (!(actions && actions.length === 1)) {
this.raiseError(`Should find exactly one codefix, but ${actions ? actions.length : "none"} found. ${actions ? actions.map(a => `${Harness.IO.newLine()} "${a.description}"`) : "" }`);
this.raiseError(`Should find exactly one codefix, but ${actions ? actions.length : "none"} found. ${actions ? actions.map(a => `${Harness.IO.newLine()} "${a.description}"`) : ""}`);
}
index = 0;
}
Expand Down Expand Up @@ -2736,6 +2749,26 @@ namespace FourSlash {
}
}

public verifyRefactorAvailable(negative: boolean, name?: string, subName?: string) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

negative [](start = 39, length = 8)

We seem to have an existing pattern, but this is a particularly pointed example of using a predicate with a negative name.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is to unify the verify.something and verify.not.something functions

const selection = {
pos: this.currentCaretPosition,
end: this.selectionEnd === -1 ? this.currentCaretPosition : this.selectionEnd
};

let refactors = this.languageService.getApplicableRefactors(this.activeFile.fileName, selection) || [];
if (name) {
refactors = refactors.filter(r => r.name === name && (subName === undefined || r.actions.some(a => a.name === subName)));
}
const isAvailable = refactors.length > 0;

if (negative && isAvailable) {
this.raiseError(`verifyApplicableRefactorAvailableForRange failed - expected no refactor but found some.`);
}
if (!negative && !isAvailable) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

if [](start = 12, length = 2)

else if

this.raiseError(`verifyApplicableRefactorAvailableForRange failed - expected a refactor but found none.`);
}
}

public verifyApplicableRefactorAvailableForRange(negative: boolean) {
const ranges = this.getRanges();
if (!(ranges && ranges.length === 1)) {
Expand All @@ -2752,6 +2785,20 @@ namespace FourSlash {
}
}

public applyRefactor(refactorName: string, actionName: string) {
const range = { pos: this.currentCaretPosition, end: this.selectionEnd === -1 ? this.currentCaretPosition : this.selectionEnd };
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

{ pos: this.currentCaretPosition, end: this.selectionEnd === -1 ? this.currentCaretPosition : this.selectionEnd } [](start = 26, length = 113)

Duplicated from above?

const refactors = this.languageService.getApplicableRefactors(this.activeFile.fileName, range);
const refactor = ts.find(refactors, r => r.name === refactorName);
if (!refactor) {
this.raiseError(`The expected refactor: ${refactorName} is not available at the marker location.`);
}

const editInfo = this.languageService.getEditsForRefactor(this.activeFile.fileName, this.formatCodeSettings, range, refactorName, actionName);
for (const edit of editInfo.edits) {
this.applyEdits(edit.fileName, edit.textChanges, /*isFormattingEdit*/ false);
}
}

public verifyFileAfterApplyingRefactorAtMarker(
markerName: string,
expectedContent: string,
Expand Down Expand Up @@ -3496,6 +3543,10 @@ namespace FourSlashInterface {
public file(indexOrName: any, content?: string, scriptKindName?: string): void {
this.state.openFile(indexOrName, content, scriptKindName);
}

public select(startMarker: string, endMarker: string) {
this.state.select(startMarker, endMarker);
}
}

export class VerifyNegatable {
Expand Down Expand Up @@ -3617,6 +3668,10 @@ namespace FourSlashInterface {
public applicableRefactorAvailableForRange() {
this.state.verifyApplicableRefactorAvailableForRange(this.negative);
}

public refactorAvailable(name?: string, subName?: string) {
this.state.verifyRefactorAvailable(this.negative, name, subName);
}
}

export class Verify extends VerifyNegatable {
Expand Down Expand Up @@ -4012,6 +4067,10 @@ namespace FourSlashInterface {
public disableFormatting() {
this.state.enableFormatting = false;
}

public applyRefactor(refactorName: string, actionName: string) {
this.state.applyRefactor(refactorName, actionName);
}
}

export class Debug {
Expand Down
Loading