@@ -38,6 +38,7 @@ type File struct {
38
38
Module * Module
39
39
Go * Go
40
40
Toolchain * Toolchain
41
+ Godebug []* Godebug
41
42
Require []* Require
42
43
Exclude []* Exclude
43
44
Replace []* Replace
@@ -65,6 +66,13 @@ type Toolchain struct {
65
66
Syntax * Line
66
67
}
67
68
69
+ // A Godebug is a single godebug key=value statement.
70
+ type Godebug struct {
71
+ Key string
72
+ Value string
73
+ Syntax * Line
74
+ }
75
+
68
76
// An Exclude is a single exclude statement.
69
77
type Exclude struct {
70
78
Mod module.Version
@@ -289,7 +297,7 @@ func parseToFile(file string, data []byte, fix VersionFixer, strict bool) (parse
289
297
})
290
298
}
291
299
continue
292
- case "module" , "require" , "exclude" , "replace" , "retract" :
300
+ case "module" , "godebug" , " require" , "exclude" , "replace" , "retract" :
293
301
for _ , l := range x .Line {
294
302
f .add (& errs , x , l , x .Token [0 ], l .Token , fix , strict )
295
303
}
@@ -308,7 +316,9 @@ var laxGoVersionRE = lazyregexp.New(`^v?(([1-9][0-9]*)\.(0|[1-9][0-9]*))([^0-9].
308
316
309
317
// Toolchains must be named beginning with `go1`,
310
318
// like "go1.20.3" or "go1.20.3-gccgo". As a special case, "default" is also permitted.
311
- // TODO(samthanawalla): Replace regex with https://pkg.go.dev/go/version#IsValid in 1.23+
319
+ // Note that this regexp is a much looser condition than go/version.IsValid,
320
+ // for forward compatibility.
321
+ // (This code has to be work to identify new toolchains even if we tweak the syntax in the future.)
312
322
var ToolchainRE = lazyregexp .New (`^default$|^go1($|\.)` )
313
323
314
324
func (f * File ) add (errs * ErrorList , block * LineBlock , line * Line , verb string , args []string , fix VersionFixer , strict bool ) {
@@ -384,7 +394,7 @@ func (f *File) add(errs *ErrorList, block *LineBlock, line *Line, verb string, a
384
394
if len (args ) != 1 {
385
395
errorf ("toolchain directive expects exactly one argument" )
386
396
return
387
- } else if strict && ! ToolchainRE .MatchString (args [0 ]) {
397
+ } else if ! ToolchainRE .MatchString (args [0 ]) {
388
398
errorf ("invalid toolchain version '%s': must match format go1.23.0 or default" , args [0 ])
389
399
return
390
400
}
@@ -412,6 +422,22 @@ func (f *File) add(errs *ErrorList, block *LineBlock, line *Line, verb string, a
412
422
}
413
423
f .Module .Mod = module.Version {Path : s }
414
424
425
+ case "godebug" :
426
+ if len (args ) != 1 || strings .ContainsAny (args [0 ], "\" `'," ) {
427
+ errorf ("usage: godebug key=value" )
428
+ return
429
+ }
430
+ key , value , ok := strings .Cut (args [0 ], "=" )
431
+ if ! ok {
432
+ errorf ("usage: godebug key=value" )
433
+ return
434
+ }
435
+ f .Godebug = append (f .Godebug , & Godebug {
436
+ Key : key ,
437
+ Value : value ,
438
+ Syntax : line ,
439
+ })
440
+
415
441
case "require" , "exclude" :
416
442
if len (args ) != 2 {
417
443
errorf ("usage: %s module/path v1.2.3" , verb )
@@ -654,6 +680,22 @@ func (f *WorkFile) add(errs *ErrorList, line *Line, verb string, args []string,
654
680
f .Toolchain = & Toolchain {Syntax : line }
655
681
f .Toolchain .Name = args [0 ]
656
682
683
+ case "godebug" :
684
+ if len (args ) != 1 || strings .ContainsAny (args [0 ], "\" `'," ) {
685
+ errorf ("usage: godebug key=value" )
686
+ return
687
+ }
688
+ key , value , ok := strings .Cut (args [0 ], "=" )
689
+ if ! ok {
690
+ errorf ("usage: godebug key=value" )
691
+ return
692
+ }
693
+ f .Godebug = append (f .Godebug , & Godebug {
694
+ Key : key ,
695
+ Value : value ,
696
+ Syntax : line ,
697
+ })
698
+
657
699
case "use" :
658
700
if len (args ) != 1 {
659
701
errorf ("usage: %s local/dir" , verb )
@@ -929,6 +971,15 @@ func (f *File) Format() ([]byte, error) {
929
971
// Cleanup cleans out all the cleared entries.
930
972
func (f * File ) Cleanup () {
931
973
w := 0
974
+ for _ , g := range f .Godebug {
975
+ if g .Key != "" {
976
+ f .Godebug [w ] = g
977
+ w ++
978
+ }
979
+ }
980
+ f .Godebug = f .Godebug [:w ]
981
+
982
+ w = 0
932
983
for _ , r := range f .Require {
933
984
if r .Mod .Path != "" {
934
985
f .Require [w ] = r
@@ -1027,6 +1078,45 @@ func (f *File) AddToolchainStmt(name string) error {
1027
1078
return nil
1028
1079
}
1029
1080
1081
+ // AddGodebug sets the first godebug line for key to value,
1082
+ // preserving any existing comments for that line and removing all
1083
+ // other godebug lines for key.
1084
+ //
1085
+ // If no line currently exists for key, AddGodebug adds a new line
1086
+ // at the end of the last godebug block.
1087
+ func (f * File ) AddGodebug (key , value string ) error {
1088
+ need := true
1089
+ for _ , g := range f .Godebug {
1090
+ if g .Key == key {
1091
+ if need {
1092
+ g .Value = value
1093
+ f .Syntax .updateLine (g .Syntax , "godebug" , key + "=" + value )
1094
+ need = false
1095
+ } else {
1096
+ g .Syntax .markRemoved ()
1097
+ * g = Godebug {}
1098
+ }
1099
+ }
1100
+ }
1101
+
1102
+ if need {
1103
+ f .addNewGodebug (key , value )
1104
+ }
1105
+ return nil
1106
+ }
1107
+
1108
+ // addNewGodebug adds a new godebug key=value line at the end
1109
+ // of the last godebug block, regardless of any existing godebug lines for key.
1110
+ func (f * File ) addNewGodebug (key , value string ) {
1111
+ line := f .Syntax .addLine (nil , "godebug" , key + "=" + value )
1112
+ g := & Godebug {
1113
+ Key : key ,
1114
+ Value : value ,
1115
+ Syntax : line ,
1116
+ }
1117
+ f .Godebug = append (f .Godebug , g )
1118
+ }
1119
+
1030
1120
// AddRequire sets the first require line for path to version vers,
1031
1121
// preserving any existing comments for that line and removing all
1032
1122
// other lines for path.
@@ -1334,6 +1424,16 @@ func (f *File) SetRequireSeparateIndirect(req []*Require) {
1334
1424
f .SortBlocks ()
1335
1425
}
1336
1426
1427
+ func (f * File ) DropGodebug (key string ) error {
1428
+ for _ , g := range f .Godebug {
1429
+ if g .Key == key {
1430
+ g .Syntax .markRemoved ()
1431
+ * g = Godebug {}
1432
+ }
1433
+ }
1434
+ return nil
1435
+ }
1436
+
1337
1437
func (f * File ) DropRequire (path string ) error {
1338
1438
for _ , r := range f .Require {
1339
1439
if r .Mod .Path == path {
0 commit comments