internal/ls/change/tracker.go (349 lines of code) (raw):
package change
import (
"context"
"slices"
"github.com/microsoft/typescript-go/internal/ast"
"github.com/microsoft/typescript-go/internal/astnav"
"github.com/microsoft/typescript-go/internal/collections"
"github.com/microsoft/typescript-go/internal/core"
"github.com/microsoft/typescript-go/internal/format"
"github.com/microsoft/typescript-go/internal/ls/lsconv"
"github.com/microsoft/typescript-go/internal/lsp/lsproto"
"github.com/microsoft/typescript-go/internal/printer"
"github.com/microsoft/typescript-go/internal/scanner"
"github.com/microsoft/typescript-go/internal/stringutil"
)
type NodeOptions struct {
// Text to be inserted before the new node
Prefix string
// Text to be inserted after the new node
Suffix string
// Text of inserted node will be formatted with this indentation, otherwise indentation will be inferred from the old node
indentation *int
// Text of inserted node will be formatted with this delta, otherwise delta will be inferred from the new node kind
delta *int
LeadingTriviaOption
TrailingTriviaOption
joiner string
}
type LeadingTriviaOption int
const (
LeadingTriviaOptionNone LeadingTriviaOption = 0
LeadingTriviaOptionExclude LeadingTriviaOption = 1
LeadingTriviaOptionIncludeAll LeadingTriviaOption = 2
LeadingTriviaOptionJSDoc LeadingTriviaOption = 3
LeadingTriviaOptionStartLine LeadingTriviaOption = 4
)
type TrailingTriviaOption int
const (
TrailingTriviaOptionNone TrailingTriviaOption = 0
TrailingTriviaOptionExclude TrailingTriviaOption = 1
TrailingTriviaOptionExcludeWhitespace TrailingTriviaOption = 2
TrailingTriviaOptionInclude TrailingTriviaOption = 3
)
type trackerEditKind int
const (
trackerEditKindText trackerEditKind = 1
trackerEditKindRemove trackerEditKind = 2
trackerEditKindReplaceWithSingleNode trackerEditKind = 3
trackerEditKindReplaceWithMultipleNodes trackerEditKind = 4
)
type trackerEdit struct {
kind trackerEditKind
lsproto.Range
NewText string // kind == text
*ast.Node // single
nodes []*ast.Node // multiple
options NodeOptions
}
type Tracker struct {
// initialized with
formatSettings *format.FormatCodeSettings
newLine string
converters *lsconv.Converters
ctx context.Context
*printer.EmitContext
*ast.NodeFactory
changes *collections.MultiMap[*ast.SourceFile, *trackerEdit]
deletedNodes []deletedNode
// created during call to getChanges
writer *printer.ChangeTrackerWriter
// printer
}
type deletedNode struct {
sourceFile *ast.SourceFile
node *ast.Node
}
func NewTracker(ctx context.Context, compilerOptions *core.CompilerOptions, formatOptions *format.FormatCodeSettings, converters *lsconv.Converters) *Tracker {
emitContext := printer.NewEmitContext()
newLine := compilerOptions.NewLine.GetNewLineCharacter()
ctx = format.WithFormatCodeSettings(ctx, formatOptions, newLine) // !!! formatSettings in context?
return &Tracker{
EmitContext: emitContext,
NodeFactory: &emitContext.Factory.NodeFactory,
changes: &collections.MultiMap[*ast.SourceFile, *trackerEdit]{},
ctx: ctx,
converters: converters,
formatSettings: formatOptions,
newLine: newLine,
}
}
// GetChanges returns the accumulated text edits.
// Note: after calling this, the Tracker object must be discarded!
func (t *Tracker) GetChanges() map[string][]*lsproto.TextEdit {
t.finishDeleteDeclarations()
// !!! finishClassesWithNodesInsertedAtStart
changes := t.getTextChangesFromChanges()
// !!! changes for new files
return changes
}
func (t *Tracker) ReplaceNode(sourceFile *ast.SourceFile, oldNode *ast.Node, newNode *ast.Node, options *NodeOptions) {
if options == nil {
// defaults to `useNonAdjustedPositions`
options = &NodeOptions{
LeadingTriviaOption: LeadingTriviaOptionExclude,
TrailingTriviaOption: TrailingTriviaOptionExclude,
}
}
t.ReplaceRange(sourceFile, t.getAdjustedRange(sourceFile, oldNode, oldNode, options.LeadingTriviaOption, options.TrailingTriviaOption), newNode, *options)
}
func (t *Tracker) ReplaceRange(sourceFile *ast.SourceFile, lsprotoRange lsproto.Range, newNode *ast.Node, options NodeOptions) {
t.changes.Add(sourceFile, &trackerEdit{kind: trackerEditKindReplaceWithSingleNode, Range: lsprotoRange, options: options, Node: newNode})
}
func (t *Tracker) ReplaceRangeWithText(sourceFile *ast.SourceFile, lsprotoRange lsproto.Range, text string) {
t.changes.Add(sourceFile, &trackerEdit{kind: trackerEditKindText, Range: lsprotoRange, NewText: text})
}
func (t *Tracker) ReplaceRangeWithNodes(sourceFile *ast.SourceFile, lsprotoRange lsproto.Range, newNodes []*ast.Node, options NodeOptions) {
if len(newNodes) == 1 {
t.ReplaceRange(sourceFile, lsprotoRange, newNodes[0], options)
return
}
t.changes.Add(sourceFile, &trackerEdit{kind: trackerEditKindReplaceWithMultipleNodes, Range: lsprotoRange, nodes: newNodes, options: options})
}
func (t *Tracker) InsertText(sourceFile *ast.SourceFile, pos lsproto.Position, text string) {
t.ReplaceRangeWithText(sourceFile, lsproto.Range{Start: pos, End: pos}, text)
}
func (t *Tracker) InsertNodeAt(sourceFile *ast.SourceFile, pos core.TextPos, newNode *ast.Node, options NodeOptions) {
lsPos := t.converters.PositionToLineAndCharacter(sourceFile, pos)
t.ReplaceRange(sourceFile, lsproto.Range{Start: lsPos, End: lsPos}, newNode, options)
}
func (t *Tracker) InsertNodesAt(sourceFile *ast.SourceFile, pos core.TextPos, newNodes []*ast.Node, options NodeOptions) {
lsPos := t.converters.PositionToLineAndCharacter(sourceFile, pos)
t.ReplaceRangeWithNodes(sourceFile, lsproto.Range{Start: lsPos, End: lsPos}, newNodes, options)
}
func (t *Tracker) InsertNodeAfter(sourceFile *ast.SourceFile, after *ast.Node, newNode *ast.Node) {
endPosition := t.endPosForInsertNodeAfter(sourceFile, after, newNode)
t.InsertNodeAt(sourceFile, endPosition, newNode, t.getInsertNodeAfterOptions(sourceFile, after))
}
func (t *Tracker) InsertNodesAfter(sourceFile *ast.SourceFile, after *ast.Node, newNodes []*ast.Node) {
endPosition := t.endPosForInsertNodeAfter(sourceFile, after, newNodes[0])
t.InsertNodesAt(sourceFile, endPosition, newNodes, t.getInsertNodeAfterOptions(sourceFile, after))
}
func (t *Tracker) InsertNodeBefore(sourceFile *ast.SourceFile, before *ast.Node, newNode *ast.Node, blankLineBetween bool) {
t.InsertNodeAt(sourceFile, core.TextPos(t.getAdjustedStartPosition(sourceFile, before, LeadingTriviaOptionNone, false)), newNode, t.getOptionsForInsertNodeBefore(before, newNode, blankLineBetween))
}
// InsertModifierBefore inserts a modifier token (like 'type') before a node with a trailing space.
func (t *Tracker) InsertModifierBefore(sourceFile *ast.SourceFile, modifier ast.Kind, before *ast.Node) {
pos := astnav.GetStartOfNode(before, sourceFile, false)
token := sourceFile.GetOrCreateToken(modifier, pos, pos, before.Parent)
t.InsertNodeAt(sourceFile, core.TextPos(pos), token, NodeOptions{Suffix: " "})
}
// Delete queues a node for deletion with smart handling of list items, imports, etc.
// The actual deletion happens in finishDeleteDeclarations during GetChanges.
func (t *Tracker) Delete(sourceFile *ast.SourceFile, node *ast.Node) {
t.deletedNodes = append(t.deletedNodes, deletedNode{sourceFile: sourceFile, node: node})
}
// DeleteRange deletes a text range from the source file.
func (t *Tracker) DeleteRange(sourceFile *ast.SourceFile, textRange core.TextRange) {
lspRange := t.converters.ToLSPRange(sourceFile, textRange)
t.ReplaceRangeWithText(sourceFile, lspRange, "")
}
// DeleteNode deletes a node immediately with specified trivia options.
// Stop! Consider using Delete instead, which has logic for deleting nodes from delimited lists.
func (t *Tracker) DeleteNode(sourceFile *ast.SourceFile, node *ast.Node, leadingTrivia LeadingTriviaOption, trailingTrivia TrailingTriviaOption) {
rng := t.getAdjustedRange(sourceFile, node, node, leadingTrivia, trailingTrivia)
t.ReplaceRangeWithText(sourceFile, rng, "")
}
// DeleteNodeRange deletes a range of nodes with specified trivia options.
func (t *Tracker) DeleteNodeRange(sourceFile *ast.SourceFile, startNode *ast.Node, endNode *ast.Node, leadingTrivia LeadingTriviaOption, trailingTrivia TrailingTriviaOption) {
startPosition := t.getAdjustedStartPosition(sourceFile, startNode, leadingTrivia, false)
endPosition := t.getAdjustedEndPosition(sourceFile, endNode, trailingTrivia)
startPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(startPosition))
endPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(endPosition))
t.ReplaceRangeWithText(sourceFile, lsproto.Range{Start: startPos, End: endPos}, "")
}
// finishDeleteDeclarations processes all queued deletions with smart handling for lists and trailing commas.
func (t *Tracker) finishDeleteDeclarations() {
deletedNodesInLists := make(map[*ast.Node]bool)
for _, deleted := range t.deletedNodes {
// Skip if this node is contained within another deleted node
isContained := false
for _, other := range t.deletedNodes {
if other.sourceFile == deleted.sourceFile && other.node != deleted.node &&
rangeContainsRangeExclusive(other.node, deleted.node) {
isContained = true
break
}
}
if isContained {
continue
}
deleteDeclaration(t, deletedNodesInLists, deleted.sourceFile, deleted.node)
}
// Handle trailing commas for last elements in lists
for node := range deletedNodesInLists {
sourceFile := ast.GetSourceFileOfNode(node)
list := format.GetContainingList(node, sourceFile)
if list == nil || node != list.Nodes[len(list.Nodes)-1] {
continue
}
lastNonDeletedIndex := -1
for i := len(list.Nodes) - 2; i >= 0; i-- {
if !deletedNodesInLists[list.Nodes[i]] {
lastNonDeletedIndex = i
break
}
}
if lastNonDeletedIndex != -1 {
startPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(list.Nodes[lastNonDeletedIndex].End()))
endPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(t.startPositionToDeleteNodeInList(sourceFile, list.Nodes[lastNonDeletedIndex+1])))
t.ReplaceRangeWithText(sourceFile, lsproto.Range{Start: startPos, End: endPos}, "")
}
}
}
func (t *Tracker) endPosForInsertNodeAfter(sourceFile *ast.SourceFile, after *ast.Node, newNode *ast.Node) core.TextPos {
if (needSemicolonBetween(after, newNode)) && (rune(sourceFile.Text()[after.End()-1]) != ';') {
// check if previous statement ends with semicolon
// if not - insert semicolon to preserve the code from changing the meaning due to ASI
endPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(after.End()))
t.ReplaceRange(sourceFile,
lsproto.Range{Start: endPos, End: endPos},
sourceFile.GetOrCreateToken(ast.KindSemicolonToken, after.End(), after.End(), after.Parent),
NodeOptions{},
)
}
return core.TextPos(t.getAdjustedEndPosition(sourceFile, after, TrailingTriviaOptionNone))
}
/**
* This function should be used to insert nodes in lists when nodes don't carry separators as the part of the node range,
* i.e. arguments in arguments lists, parameters in parameter lists etc.
* Note that separators are part of the node in statements and class elements.
*/
func (t *Tracker) InsertNodeInListAfter(sourceFile *ast.SourceFile, after *ast.Node, newNode *ast.Node, containingList []*ast.Node) {
if len(containingList) == 0 {
containingList = format.GetContainingList(after, sourceFile).Nodes
}
index := slices.Index(containingList, after)
if index < 0 {
return
}
if index != len(containingList)-1 {
// any element except the last one
// use next sibling as an anchor
if nextToken := astnav.GetTokenAtPosition(sourceFile, after.End()); nextToken != nil && isSeparator(after, nextToken) {
// for list
// a, b, c
// create change for adding 'e' after 'a' as
// - find start of next element after a (it is b)
// - use next element start as start and end position in final change
// - build text of change by formatting the text of node + whitespace trivia of b
// in multiline case it will work as
// a,
// b,
// c,
// result - '*' denotes leading trivia that will be inserted after new text (displayed as '#')
// a,
// insertedtext<separator>#
// ###b,
// c,
nextNode := containingList[index+1]
startPos := scanner.SkipTriviaEx(sourceFile.Text(), nextNode.Pos(), &scanner.SkipTriviaOptions{StopAfterLineBreak: false, StopAtComments: true})
// write separator and leading trivia of the next element as suffix
suffix := scanner.TokenToString(nextToken.Kind) + sourceFile.Text()[nextToken.End():startPos]
t.InsertNodeAt(sourceFile, core.TextPos(startPos), newNode, NodeOptions{Suffix: suffix})
}
return
}
afterStart := astnav.GetStartOfNode(after, sourceFile, false)
afterStartLinePosition := format.GetLineStartPositionForPosition(afterStart, sourceFile)
// insert element after the last element in the list that has more than one item
// pick the element preceding the after element to:
// - pick the separator
// - determine if list is a multiline
multilineList := false
// if list has only one element then we'll format is as multiline if node has comment in trailing trivia, or as singleline otherwise
// i.e. var x = 1 // this is x
// | new element will be inserted at this position
separator := ast.KindCommaToken // SyntaxKind.CommaToken | SyntaxKind.SemicolonToken
if len(containingList) != 1 {
// otherwise, if list has more than one element, pick separator from the list
tokenBeforeInsertPosition := astnav.FindPrecedingToken(sourceFile, after.Pos())
separator = core.IfElse(isSeparator(after, tokenBeforeInsertPosition), tokenBeforeInsertPosition.Kind, ast.KindCommaToken)
// determine if list is multiline by checking lines of after element and element that precedes it.
afterMinusOneStartLinePosition := format.GetLineStartPositionForPosition(astnav.GetStartOfNode(containingList[index-1], sourceFile, false), sourceFile)
multilineList = afterMinusOneStartLinePosition != afterStartLinePosition
}
if hasCommentsBeforeLineBreak(sourceFile.Text(), after.End()) || printer.GetLinesBetweenPositions(sourceFile, containingList[0].Pos(), containingList[len(containingList)-1].End()) != 0 {
// in this case we'll always treat containing list as multiline
multilineList = true
}
separatorString := scanner.TokenToString(separator)
end := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(after.End()))
if !multilineList {
t.ReplaceRange(sourceFile, lsproto.Range{Start: end, End: end}, newNode, NodeOptions{Prefix: separatorString})
return
}
// insert separator immediately following the 'after' node to preserve comments in trailing trivia
// !!! formatcontext
t.ReplaceRange(sourceFile, lsproto.Range{Start: end, End: end}, sourceFile.GetOrCreateToken(separator, after.End(), after.End()+len(separatorString), after.Parent), NodeOptions{})
// use the same indentation as 'after' item
indentation := format.FindFirstNonWhitespaceColumn(afterStartLinePosition, afterStart, sourceFile, t.formatSettings)
// insert element before the line break on the line that contains 'after' element
insertPos := scanner.SkipTriviaEx(sourceFile.Text(), after.End(), &scanner.SkipTriviaOptions{StopAfterLineBreak: true, StopAtComments: false})
// find position before "\n" or "\r\n"
for insertPos != after.End() && stringutil.IsLineBreak(rune(sourceFile.Text()[insertPos-1])) {
insertPos--
}
insertLSPos := t.converters.PositionToLineAndCharacter(sourceFile, core.TextPos(insertPos))
t.ReplaceRange(
sourceFile,
lsproto.Range{Start: insertLSPos, End: insertLSPos},
newNode,
NodeOptions{
indentation: ptrTo(indentation),
Prefix: t.newLine,
},
)
}
// InsertImportSpecifierAtIndex inserts a new import specifier at the specified index in a NamedImports list
func (t *Tracker) InsertImportSpecifierAtIndex(sourceFile *ast.SourceFile, newSpecifier *ast.Node, namedImports *ast.Node, index int) {
namedImportsNode := namedImports.AsNamedImports()
elements := namedImportsNode.Elements.Nodes
if index >= len(elements) {
// Insert at the end (after the last element)
t.InsertNodeInListAfter(sourceFile, elements[len(elements)-1], newSpecifier, elements)
} else if index > 0 {
// Insert after the element at index-1
t.InsertNodeInListAfter(sourceFile, elements[index-1], newSpecifier, elements)
} else {
// Insert before the first element
firstElement := elements[0]
multiline := printer.GetLinesBetweenPositions(sourceFile, firstElement.Pos(), namedImports.Parent.Parent.Pos()) != 0
t.InsertNodeBefore(sourceFile, firstElement, newSpecifier, multiline)
}
}
func (t *Tracker) InsertAtTopOfFile(sourceFile *ast.SourceFile, insert []*ast.Statement, blankLineBetween bool) {
if len(insert) == 0 {
return
}
pos := t.getInsertionPositionAtSourceFileTop(sourceFile)
options := NodeOptions{}
if pos != 0 {
options.Prefix = t.newLine
}
if len(sourceFile.Text()) == 0 || !stringutil.IsLineBreak(rune(sourceFile.Text()[pos])) {
options.Suffix = t.newLine
}
if blankLineBetween {
options.Suffix += t.newLine
}
if len(insert) == 1 {
t.InsertNodeAt(sourceFile, core.TextPos(pos), insert[0], options)
} else {
t.InsertNodesAt(sourceFile, core.TextPos(pos), insert, options)
}
}
func (t *Tracker) getInsertNodeAfterOptions(sourceFile *ast.SourceFile, node *ast.Node) NodeOptions {
newLineChar := t.newLine
var options NodeOptions
switch node.Kind {
case ast.KindParameter:
// default opts
options = NodeOptions{}
case ast.KindClassDeclaration, ast.KindModuleDeclaration:
options = NodeOptions{Prefix: newLineChar, Suffix: newLineChar}
case ast.KindVariableDeclaration, ast.KindStringLiteral, ast.KindIdentifier:
options = NodeOptions{Prefix: ", "}
case ast.KindPropertyAssignment:
options = NodeOptions{Suffix: "," + newLineChar}
case ast.KindExportKeyword:
options = NodeOptions{Prefix: " "}
default:
if !(ast.IsStatement(node) || ast.IsClassOrTypeElement(node)) {
// Else we haven't handled this kind of node yet -- add it
panic("unimplemented node type " + node.Kind.String() + " in changeTracker.getInsertNodeAfterOptions")
}
options = NodeOptions{Suffix: newLineChar}
}
if node.End() == sourceFile.End() && ast.IsStatement(node) {
options.Prefix = "\n" + options.Prefix
}
return options
}
func (t *Tracker) getOptionsForInsertNodeBefore(before *ast.Node, inserted *ast.Node, blankLineBetween bool) NodeOptions {
if ast.IsStatement(before) || ast.IsClassOrTypeElement(before) {
if blankLineBetween {
return NodeOptions{Suffix: t.newLine + t.newLine}
}
return NodeOptions{Suffix: t.newLine}
} else if before.Kind == ast.KindVariableDeclaration {
// insert `x = 1, ` into `const x = 1, y = 2;
return NodeOptions{Suffix: ", "}
} else if before.Kind == ast.KindParameter {
if inserted.Kind == ast.KindParameter {
return NodeOptions{Suffix: ", "}
}
return NodeOptions{}
} else if (before.Kind == ast.KindStringLiteral && before.Parent != nil && before.Parent.Kind == ast.KindImportDeclaration) || before.Kind == ast.KindNamedImports {
return NodeOptions{Suffix: ", "}
} else if before.Kind == ast.KindImportSpecifier {
suffix := ","
if blankLineBetween {
suffix += t.newLine
} else {
suffix += " "
}
return NodeOptions{Suffix: suffix}
}
// We haven't handled this kind of node yet -- add it
panic("unimplemented node type " + before.Kind.String() + " in changeTracker.getOptionsForInsertNodeBefore")
}
func ptrTo[T any](v T) *T {
return &v
}
func rangeContainsRangeExclusive(outer *ast.Node, inner *ast.Node) bool {
return outer.Pos() < inner.Pos() && inner.End() < outer.End()
}
func isSeparator(node *ast.Node, candidate *ast.Node) bool {
return candidate != nil && node.Parent != nil && (candidate.Kind == ast.KindCommaToken || (candidate.Kind == ast.KindSemicolonToken && node.Parent.Kind == ast.KindObjectLiteralExpression))
}