blob: 78748c5c12eaee55127eef1ae3396c691ed4d004 [file] [log] [blame]
// Copyright 2023 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 httpmux
import (
"go/ast"
"go/constant"
"go/types"
"regexp"
"strings"
"golang.org/x/mod/semver"
"golang.org/x/tools/go/analysis"
"golang.org/x/tools/go/analysis/passes/inspect"
"golang.org/x/tools/go/analysis/passes/internal/analysisutil"
"golang.org/x/tools/go/ast/inspector"
"golang.org/x/tools/go/types/typeutil"
"golang.org/x/tools/internal/typesinternal"
)
const Doc = `report using Go 1.22 enhanced ServeMux patterns in older Go versions
The httpmux analysis is active for Go modules configured to run with Go 1.21 or
earlier versions. It reports calls to net/http.ServeMux.Handle and HandleFunc
methods whose patterns use features added in Go 1.22, like HTTP methods (such as
"GET") and wildcards. (See https://pkg.go.dev/net/http#ServeMux for details.)
Such patterns can be registered in older versions of Go, but will not behave as expected.`
var Analyzer = &analysis.Analyzer{
Name: "httpmux",
Doc: Doc,
URL: "https://pkg.go.dev/golang.org/x/tools/go/analysis/passes/httpmux",
Requires: []*analysis.Analyzer{inspect.Analyzer},
Run: run,
}
var inTest bool // So Go version checks can be skipped during testing.
func run(pass *analysis.Pass) (any, error) {
if !inTest {
// Check that Go version is 1.21 or earlier.
if goVersionAfter121(goVersion(pass.Pkg)) {
return nil, nil
}
}
if !analysisutil.Imports(pass.Pkg, "net/http") {
return nil, nil
}
// Look for calls to ServeMux.Handle or ServeMux.HandleFunc.
inspect := pass.ResultOf[inspect.Analyzer].(*inspector.Inspector)
nodeFilter := []ast.Node{
(*ast.CallExpr)(nil),
}
inspect.Preorder(nodeFilter, func(n ast.Node) {
call := n.(*ast.CallExpr)
if isServeMuxRegisterCall(pass, call) {
pat, ok := stringConstantExpr(pass, call.Args[0])
if ok && likelyEnhancedPattern(pat) {
pass.ReportRangef(call.Args[0], "possible enhanced ServeMux pattern used with Go version before 1.22 (update go.mod file?)")
}
}
})
return nil, nil
}
// isServeMuxRegisterCall reports whether call is a static call to one of:
// - net/http.Handle
// - net/http.HandleFunc
// - net/http.ServeMux.Handle
// - net/http.ServeMux.HandleFunc
// TODO(jba): consider expanding this to accommodate wrappers around these functions.
func isServeMuxRegisterCall(pass *analysis.Pass, call *ast.CallExpr) bool {
fn := typeutil.StaticCallee(pass.TypesInfo, call)
if fn == nil {
return false
}
if analysisutil.IsFunctionNamed(fn, "net/http", "Handle", "HandleFunc") {
return true
}
if !isMethodNamed(fn, "net/http", "Handle", "HandleFunc") {
return false
}
recv := fn.Type().(*types.Signature).Recv() // isMethodNamed() -> non-nil
isPtr, named := typesinternal.ReceiverNamed(recv)
return isPtr && analysisutil.IsNamedType(named, "net/http", "ServeMux")
}
// isMethodNamed reports when a function f is a method,
// in a package with the path pkgPath and the name of f is in names.
func isMethodNamed(f *types.Func, pkgPath string, names ...string) bool {
if f == nil {
return false
}
if f.Pkg() == nil || f.Pkg().Path() != pkgPath {
return false // not at pkgPath
}
if f.Type().(*types.Signature).Recv() == nil {
return false // not a method
}
for _, n := range names {
if f.Name() == n {
return true
}
}
return false // not in names
}
// stringConstantExpr returns expression's string constant value.
//
// ("", false) is returned if expression isn't a string
// constant.
func stringConstantExpr(pass *analysis.Pass, expr ast.Expr) (string, bool) {
lit := pass.TypesInfo.Types[expr].Value
if lit != nil && lit.Kind() == constant.String {
return constant.StringVal(lit), true
}
return "", false
}
// A valid wildcard must start a segment, and its name must be valid Go
// identifier.
var wildcardRegexp = regexp.MustCompile(`/\{[_\pL][_\pL\p{Nd}]*(\.\.\.)?\}`)
// likelyEnhancedPattern reports whether the ServeMux pattern pat probably
// contains either an HTTP method name or a wildcard, extensions added in Go 1.22.
func likelyEnhancedPattern(pat string) bool {
if strings.Contains(pat, " ") {
// A space in the pattern suggests that it begins with an HTTP method.
return true
}
return wildcardRegexp.MatchString(pat)
}
func goVersionAfter121(goVersion string) bool {
if goVersion == "" { // Maybe the stdlib?
return true
}
version := versionFromGoVersion(goVersion)
return semver.Compare(version, "v1.21") > 0
}
func goVersion(pkg *types.Package) string {
// types.Package.GoVersion did not exist before Go 1.21.
if p, ok := any(pkg).(interface{ GoVersion() string }); ok {
return p.GoVersion()
}
return ""
}
var (
// Regexp for matching go tags. The groups are:
// 1 the major.minor version
// 2 the patch version, or empty if none
// 3 the entire prerelease, if present
// 4 the prerelease type ("beta" or "rc")
// 5 the prerelease number
tagRegexp = regexp.MustCompile(`^go(\d+\.\d+)(\.\d+|)((beta|rc)(\d+))?$`)
)
// Copied from pkgsite/internal/stdlib.VersionForTag.
func versionFromGoVersion(goVersion string) string {
// Special cases for go1.
if goVersion == "go1" {
return "v1.0.0"
}
if goVersion == "go1.0" {
return ""
}
m := tagRegexp.FindStringSubmatch(goVersion)
if m == nil {
return ""
}
version := "v" + m[1]
if m[2] != "" {
version += m[2]
} else {
version += ".0"
}
if m[3] != "" {
version += "-" + m[4] + "." + m[5]
}
return version
}