Skip to content

Commit 6f7906b

Browse files
adonovangopherbot
authored andcommitted
x/tools: use ast.IsGenerated throughout
Note the behavior change: the go/ast implementation checks that the special comment appears before the packge declaration; the ad hoc implementations did not. Change-Id: Ib51c498c1c73fd32c882e25f8228a6076bba7ed7 Reviewed-on: https://go-review.googlesource.com/c/tools/+/649316 LUCI-TryBot-Result: Go LUCI <[email protected]> Auto-Submit: Alan Donovan <[email protected]> Reviewed-by: Jonathan Amsterdam <[email protected]> Commit-Queue: Alan Donovan <[email protected]>
1 parent e890c1f commit 6f7906b

File tree

9 files changed

+29
-124
lines changed

9 files changed

+29
-124
lines changed

cmd/deadcode/deadcode.go

+1-40
Original file line numberDiff line numberDiff line change
@@ -175,7 +175,7 @@ func main() {
175175
}
176176
}
177177

178-
if isGenerated(file) {
178+
if ast.IsGenerated(file) {
179179
generated[p.Fset.File(file.Pos()).Name()] = true
180180
}
181181
}
@@ -414,45 +414,6 @@ func printObjects(format string, objects []any) {
414414
}
415415
}
416416

417-
// TODO(adonovan): use go1.21's ast.IsGenerated.
418-
419-
// isGenerated reports whether the file was generated by a program,
420-
// not handwritten, by detecting the special comment described
421-
// at https://go.dev/s/generatedcode.
422-
//
423-
// The syntax tree must have been parsed with the ParseComments flag.
424-
// Example:
425-
//
426-
// f, err := parser.ParseFile(fset, filename, src, parser.ParseComments|parser.PackageClauseOnly)
427-
// if err != nil { ... }
428-
// gen := ast.IsGenerated(f)
429-
func isGenerated(file *ast.File) bool {
430-
_, ok := generator(file)
431-
return ok
432-
}
433-
434-
func generator(file *ast.File) (string, bool) {
435-
for _, group := range file.Comments {
436-
for _, comment := range group.List {
437-
if comment.Pos() > file.Package {
438-
break // after package declaration
439-
}
440-
// opt: check Contains first to avoid unnecessary array allocation in Split.
441-
const prefix = "// Code generated "
442-
if strings.Contains(comment.Text, prefix) {
443-
for _, line := range strings.Split(comment.Text, "\n") {
444-
if rest, ok := strings.CutPrefix(line, prefix); ok {
445-
if gen, ok := strings.CutSuffix(rest, " DO NOT EDIT."); ok {
446-
return gen, true
447-
}
448-
}
449-
}
450-
}
451-
}
452-
}
453-
return "", false
454-
}
455-
456417
// pathSearch returns the shortest path from one of the roots to one
457418
// of the targets (along with the root itself), or zero if no path was found.
458419
func pathSearch(roots []*ssa.Function, res *rta.Result, targets map[*ssa.Function]bool) (*callgraph.Node, []*callgraph.Edge) {

copyright/copyright.go

+1-22
Original file line numberDiff line numberDiff line change
@@ -75,7 +75,7 @@ func checkFile(toolsDir, filename string) (bool, error) {
7575
return false, err
7676
}
7777
// Don't require headers on generated files.
78-
if isGenerated(fset, parsed) {
78+
if ast.IsGenerated(parsed) {
7979
return false, nil
8080
}
8181
shouldAddCopyright := true
@@ -91,24 +91,3 @@ func checkFile(toolsDir, filename string) (bool, error) {
9191
}
9292
return shouldAddCopyright, nil
9393
}
94-
95-
// Copied from golang.org/x/tools/gopls/internal/golang/util.go.
96-
// Matches cgo generated comment as well as the proposed standard:
97-
//
98-
// https://golang.org/s/generatedcode
99-
var generatedRx = regexp.MustCompile(`// .*DO NOT EDIT\.?`)
100-
101-
func isGenerated(fset *token.FileSet, file *ast.File) bool {
102-
for _, commentGroup := range file.Comments {
103-
for _, comment := range commentGroup.List {
104-
if matched := generatedRx.MatchString(comment.Text); !matched {
105-
continue
106-
}
107-
// Check if comment is at the beginning of the line in source.
108-
if pos := fset.Position(comment.Slash); pos.Column == 1 {
109-
return true
110-
}
111-
}
112-
}
113-
return false
114-
}

gopls/internal/golang/format.go

+6-5
Original file line numberDiff line numberDiff line change
@@ -35,15 +35,16 @@ func Format(ctx context.Context, snapshot *cache.Snapshot, fh file.Handle) ([]pr
3535
ctx, done := event.Start(ctx, "golang.Format")
3636
defer done()
3737

38-
// Generated files shouldn't be edited. So, don't format them
39-
if IsGenerated(ctx, snapshot, fh.URI()) {
40-
return nil, fmt.Errorf("can't format %q: file is generated", fh.URI().Path())
41-
}
42-
4338
pgf, err := snapshot.ParseGo(ctx, fh, parsego.Full)
4439
if err != nil {
4540
return nil, err
4641
}
42+
43+
// Generated files shouldn't be edited. So, don't format them.
44+
if ast.IsGenerated(pgf.File) {
45+
return nil, fmt.Errorf("can't format %q: file is generated", fh.URI().Path())
46+
}
47+
4748
// Even if this file has parse errors, it might still be possible to format it.
4849
// Using format.Node on an AST with errors may result in code being modified.
4950
// Attempt to format the source of this file instead.

gopls/internal/golang/util.go

+3-23
Original file line numberDiff line numberDiff line change
@@ -19,16 +19,11 @@ import (
1919
"golang.org/x/tools/gopls/internal/cache/parsego"
2020
"golang.org/x/tools/gopls/internal/protocol"
2121
"golang.org/x/tools/gopls/internal/util/bug"
22-
"golang.org/x/tools/gopls/internal/util/safetoken"
2322
"golang.org/x/tools/internal/tokeninternal"
2423
)
2524

26-
// IsGenerated gets and reads the file denoted by uri and reports
27-
// whether it contains a "generated file" comment as described at
28-
// https://golang.org/s/generatedcode.
29-
//
30-
// TODO(adonovan): opt: this function does too much.
31-
// Move snapshot.ReadFile into the caller (most of which have already done it).
25+
// IsGenerated reads and parses the header of the file denoted by uri
26+
// and reports whether it [ast.IsGenerated].
3227
func IsGenerated(ctx context.Context, snapshot *cache.Snapshot, uri protocol.DocumentURI) bool {
3328
fh, err := snapshot.ReadFile(ctx, uri)
3429
if err != nil {
@@ -38,17 +33,7 @@ func IsGenerated(ctx context.Context, snapshot *cache.Snapshot, uri protocol.Doc
3833
if err != nil {
3934
return false
4035
}
41-
for _, commentGroup := range pgf.File.Comments {
42-
for _, comment := range commentGroup.List {
43-
if matched := generatedRx.MatchString(comment.Text); matched {
44-
// Check if comment is at the beginning of the line in source.
45-
if safetoken.Position(pgf.Tok, comment.Slash).Column == 1 {
46-
return true
47-
}
48-
}
49-
}
50-
}
51-
return false
36+
return ast.IsGenerated(pgf.File)
5237
}
5338

5439
// adjustedObjEnd returns the end position of obj, possibly modified for
@@ -76,11 +61,6 @@ func adjustedObjEnd(obj types.Object) token.Pos {
7661
return obj.Pos() + token.Pos(nameLen)
7762
}
7863

79-
// Matches cgo generated comment as well as the proposed standard:
80-
//
81-
// https://golang.org/s/generatedcode
82-
var generatedRx = regexp.MustCompile(`// .*DO NOT EDIT\.?`)
83-
8464
// FormatNode returns the "pretty-print" output for an ast node.
8565
func FormatNode(fset *token.FileSet, n ast.Node) string {
8666
var buf strings.Builder

gopls/internal/test/integration/diagnostics/diagnostics_test.go

+12-5
Original file line numberDiff line numberDiff line change
@@ -542,27 +542,34 @@ var X = 0
542542

543543
// Tests golang/go#38467.
544544
func TestNoSuggestedFixesForGeneratedFiles_Issue38467(t *testing.T) {
545+
// This test ensures that gopls' CodeAction handler suppresses
546+
// diagnostics in generated code. Beware that many analyzers
547+
// themselves suppress diagnostics in generated files, in
548+
// particular the low-status "simplifiers" (modernize,
549+
// simplify{range,slice,compositelit}), so we use the hostport
550+
// analyzer here.
545551
const generated = `
546552
-- go.mod --
547553
module mod.com
548554
549555
go 1.12
550556
-- main.go --
557+
// Code generated by generator.go. DO NOT EDIT.
558+
551559
package main
552560
553-
// Code generated by generator.go. DO NOT EDIT.
561+
import ("fmt"; "net")
554562
555563
func _() {
556-
for i, _ := range []string{} {
557-
_ = i
558-
}
564+
addr := fmt.Sprintf("%s:%d", "localhost", 12345)
565+
net.Dial("tcp", addr)
559566
}
560567
`
561568
Run(t, generated, func(t *testing.T, env *Env) {
562569
env.OpenFile("main.go")
563570
var d protocol.PublishDiagnosticsParams
564571
env.AfterChange(
565-
Diagnostics(AtPosition("main.go", 5, 6)),
572+
Diagnostics(AtPosition("main.go", 7, 21)),
566573
ReadDiagnostics("main.go", &d),
567574
)
568575
if fixes := env.GetQuickFixes("main.go", d.Diagnostics); len(fixes) != 0 {

gopls/internal/test/marker/testdata/diagnostics/generated.txt

+2-2
Original file line numberDiff line numberDiff line change
@@ -10,10 +10,10 @@ module example.com
1010
go 1.12
1111

1212
-- generated.go --
13-
package generated
14-
1513
// Code generated by generator.go. DO NOT EDIT.
1614

15+
package generated
16+
1717
func _() {
1818
var y int //@diag("y", re"declared (and|but) not used")
1919
}

internal/refactor/inline/inline.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -102,9 +102,9 @@ func (st *state) inline() (*Result, error) {
102102
return nil, fmt.Errorf("internal error: caller syntax positions are inconsistent with file content (did you forget to use FileSet.PositionFor when computing the file name?)")
103103
}
104104

105-
// TODO(adonovan): use go1.21's ast.IsGenerated.
106105
// Break the string literal so we can use inlining in this file. :)
107-
if bytes.Contains(caller.Content, []byte("// Code generated by "+"cmd/cgo; DO NOT EDIT.")) {
106+
if ast.IsGenerated(caller.File) &&
107+
bytes.Contains(caller.Content, []byte("// Code generated by "+"cmd/cgo; DO NOT EDIT.")) {
108108
return nil, fmt.Errorf("cannot inline calls from files that import \"C\"")
109109
}
110110

refactor/rename/rename.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -491,7 +491,7 @@ func (r *renamer) update() error {
491491
for _, info := range r.packages {
492492
for _, f := range info.Files {
493493
tokenFile := r.iprog.Fset.File(f.FileStart)
494-
if filesToUpdate[tokenFile] && generated(f, tokenFile) {
494+
if filesToUpdate[tokenFile] && ast.IsGenerated(f) {
495495
generatedFileNames = append(generatedFileNames, tokenFile.Name())
496496
}
497497
}

refactor/rename/spec.go

+1-24
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,6 @@ import (
1919
"log"
2020
"os"
2121
"path/filepath"
22-
"regexp"
2322
"strconv"
2423
"strings"
2524

@@ -321,7 +320,7 @@ func findFromObjectsInFile(iprog *loader.Program, spec *spec) ([]types.Object, e
321320

322321
if spec.offset != 0 {
323322
// We cannot refactor generated files since position information is invalidated.
324-
if generated(f, thisFile) {
323+
if ast.IsGenerated(f) {
325324
return nil, fmt.Errorf("cannot rename identifiers in generated file containing DO NOT EDIT marker: %s", thisFile.Name())
326325
}
327326

@@ -566,25 +565,3 @@ func ambiguityError(fset *token.FileSet, objects []types.Object) error {
566565
return fmt.Errorf("ambiguous specifier %s matches %s",
567566
objects[0].Name(), buf.String())
568567
}
569-
570-
// Matches cgo generated comment as well as the proposed standard:
571-
//
572-
// https://golang.org/s/generatedcode
573-
var generatedRx = regexp.MustCompile(`// .*DO NOT EDIT\.?`)
574-
575-
// generated reports whether ast.File is a generated file.
576-
func generated(f *ast.File, tokenFile *token.File) bool {
577-
578-
// Iterate over the comments in the file
579-
for _, commentGroup := range f.Comments {
580-
for _, comment := range commentGroup.List {
581-
if matched := generatedRx.MatchString(comment.Text); matched {
582-
// Check if comment is at the beginning of the line in source
583-
if pos := tokenFile.Position(comment.Slash); pos.Column == 1 {
584-
return true
585-
}
586-
}
587-
}
588-
}
589-
return false
590-
}

0 commit comments

Comments
 (0)