blob: 8575827aa3ec0ddfe149a99656d48ebe1ea81e5e [file] [log] [blame]
// Copyright 2024 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package modernize
import (
"go/ast"
"go/types"
"strings"
"golang.org/x/tools/go/analysis"
"golang.org/x/tools/go/analysis/passes/inspect"
"golang.org/x/tools/go/ast/inspector"
"golang.org/x/tools/go/types/typeutil"
"golang.org/x/tools/internal/analysisinternal"
)
// The fmtappend function replaces []byte(fmt.Sprintf(...)) by
// fmt.Appendf(nil, ...).
func fmtappendf(pass *analysis.Pass) {
inspect := pass.ResultOf[inspect.Analyzer].(*inspector.Inspector)
info := pass.TypesInfo
for curFile := range filesUsing(inspect, info, "go1.19") {
for curCallExpr := range curFile.Preorder((*ast.CallExpr)(nil)) {
conv := curCallExpr.Node().(*ast.CallExpr)
tv := info.Types[conv.Fun]
if tv.IsType() && types.Identical(tv.Type, byteSliceType) {
call, ok := conv.Args[0].(*ast.CallExpr)
if ok {
obj := typeutil.Callee(info, call)
if !analysisinternal.IsFunctionNamed(obj, "fmt", "Sprintf", "Sprintln", "Sprint") {
continue
}
// Find "Sprint" identifier.
var id *ast.Ident
switch e := ast.Unparen(call.Fun).(type) {
case *ast.SelectorExpr:
id = e.Sel // "fmt.Sprint"
case *ast.Ident:
id = e // "Sprint" after `import . "fmt"`
}
pass.Report(analysis.Diagnostic{
Pos: conv.Pos(),
End: conv.End(),
Category: "fmtappendf",
Message: "Replace []byte(fmt.Sprintf...) with fmt.Appendf",
SuggestedFixes: []analysis.SuggestedFix{{
Message: "Replace []byte(fmt.Sprintf...) with fmt.Appendf",
TextEdits: []analysis.TextEdit{
{
// delete "[]byte("
Pos: conv.Pos(),
End: conv.Lparen + 1,
},
{
// remove ")"
Pos: conv.Rparen,
End: conv.Rparen + 1,
},
{
Pos: id.Pos(),
End: id.End(),
NewText: []byte(strings.Replace(obj.Name(), "Sprint", "Append", 1)),
},
{
Pos: call.Lparen + 1,
NewText: []byte("nil, "),
},
},
}},
})
}
}
}
}
}