Copyright 2018 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 module defines the module.Version type along with support code. The module.Version type is a simple Path, Version pair: type Version struct { Path string Version string } There are no restrictions imposed directly by use of this structure, but additional checking functions, most notably Check, verify that a particular path, version pair is valid. Escaped Paths Module paths appear as substrings of file system paths (in the download cache) and of web server URLs in the proxy protocol. In general we cannot rely on file systems to be case-sensitive, nor can we rely on web servers, since they read from file systems. That is, we cannot rely on the file system to keep rsc.io/QUOTE and rsc.io/quote separate. Windows and macOS don't. Instead, we must never require two different casings of a file path. Because we want the download cache to match the proxy protocol, and because we want the proxy protocol to be possible to serve from a tree of static files (which might be stored on a case-insensitive file system), the proxy protocol must never require two different casings of a URL path either. One possibility would be to make the escaped form be the lowercase hexadecimal encoding of the actual path bytes. This would avoid ever needing different casings of a file path, but it would be fairly illegible to most programmers when those paths appeared in the file system (including in file paths in compiler errors and stack traces) in web server logs, and so on. Instead, we want a safe escaped form that leaves most paths unaltered. The safe escaped form is to replace every uppercase letter with an exclamation mark followed by the letter's lowercase equivalent. For example, github.com/Azure/azure-sdk-for-go -> github.com/!azure/azure-sdk-for-go. github.com/GoogleCloudPlatform/cloudsql-proxy -> github.com/!google!cloud!platform/cloudsql-proxy github.com/Sirupsen/logrus -> github.com/!sirupsen/logrus. Import paths that avoid upper-case letters are left unchanged. Note that because import paths are ASCII-only and avoid various problematic punctuation (like : < and >), the escaped form is also ASCII-only and avoids the same problematic punctuation. Import paths have never allowed exclamation marks, so there is no need to define how to escape a literal !. Unicode Restrictions Today, paths are disallowed from using Unicode. Although paths are currently disallowed from using Unicode, we would like at some point to allow Unicode letters as well, to assume that file systems and URLs are Unicode-safe (storing UTF-8), and apply the !-for-uppercase convention for escaping them in the file system. But there are at least two subtle considerations. First, note that not all case-fold equivalent distinct runes form an upper/lower pair. For example, U+004B ('K'), U+006B ('k'), and U+212A ('K' for Kelvin) are three distinct runes that case-fold to each other. When we do add Unicode letters, we must not assume that upper/lower are the only case-equivalent pairs. Perhaps the Kelvin symbol would be disallowed entirely, for example. Or perhaps it would escape as "!!k", or perhaps as "(212A)". Second, it would be nice to allow Unicode marks as well as letters, but marks include combining marks, and then we must deal not only with case folding but also normalization: both U+00E9 ('é') and U+0065 U+0301 ('e' followed by combining acute accent) look the same on the page and are treated by some file systems as the same path. If we do allow Unicode marks in paths, there must be some kind of normalization to allow only one canonical encoding of any character used in an import path.
package module
IMPORTANT NOTE This file essentially defines the set of valid import paths for the go command. There are many subtle considerations, including Unicode ambiguity, security, network, and file system representations. This file also defines the set of valid module path and version combinations, another topic with many subtle considerations. Changes to the semantics in this file require approval from rsc.

import (
	
	
	
	
	
	

	
	errors 
)
A Version (for clients, a module.Version) is defined by a module path and version pair. These are stored in their plain (unescaped) form.
Path is a module path, like "golang.org/x/text" or "rsc.io/quote/v2".
Version is usually a semantic version in canonical form. There are three exceptions to this general rule. First, the top-level target of a build has no specific version and uses Version = "". Second, during MVS calculations the version "none" is used to represent the decision to take no version of a given module. Third, filesystem paths found in "replace" directives are represented by a path with an empty version.
	Version string `json:",omitempty"`
}
String returns a representation of the Version suitable for logging (Path@Version, or just Path if Version is empty).
func ( Version) () string {
	if .Version == "" {
		return .Path
	}
	return .Path + "@" + .Version
}
A ModuleError indicates an error specific to a module.
VersionError returns a ModuleError derived from a Version and error, or err itself if it is already such an error.
func ( Version,  error) error {
	var  *ModuleError
	if errors.As(, &) && .Path == .Path && .Version == .Version {
		return 
	}
	return &ModuleError{
		Path:    .Path,
		Version: .Version,
		Err:     ,
	}
}

func ( *ModuleError) () string {
	if ,  := .Err.(*InvalidVersionError);  {
		return fmt.Sprintf("%s@%s: invalid %s: %v", .Path, .Version, .noun(), .Err)
	}
	if .Version != "" {
		return fmt.Sprintf("%s@%s: %v", .Path, .Version, .Err)
	}
	return fmt.Sprintf("module %s: %v", .Path, .Err)
}

func ( *ModuleError) () error { return .Err }
An InvalidVersionError indicates an error specific to a version, with the module path unknown or specified externally. A ModuleError may wrap an InvalidVersionError, but an InvalidVersionError must not wrap a ModuleError.
noun returns either "version" or "pseudo-version", depending on whether e.Version is a pseudo-version.
func ( *InvalidVersionError) () string {
	if .Pseudo {
		return "pseudo-version"
	}
	return "version"
}

func ( *InvalidVersionError) () string {
	return fmt.Sprintf("%s %q invalid: %s", .noun(), .Version, .Err)
}

func ( *InvalidVersionError) () error { return .Err }
Check checks that a given module path, version pair is valid. In addition to the path being a valid module path and the version being a valid semantic version, the two must correspond. For example, the path "yaml/v2" only corresponds to semantic versions beginning with "v2.".
func (,  string) error {
	if  := CheckPath();  != nil {
		return 
	}
	if !semver.IsValid() {
		return &ModuleError{
			Path: ,
			Err:  &InvalidVersionError{Version: , Err: errors.New("not a semantic version")},
		}
	}
	, ,  := SplitPathVersion()
	if  := CheckPathMajor(, );  != nil {
		return &ModuleError{Path: , Err: }
	}
	return nil
}
firstPathOK reports whether r can appear in the first element of a module path. The first element of the path must be an LDH domain name, at least for now. To avoid case ambiguity, the domain name must be entirely lower case.
func ( rune) bool {
	return  == '-' ||  == '.' ||
		'0' <=  &&  <= '9' ||
		'a' <=  &&  <= 'z'
}
pathOK reports whether r can appear in an import path element. Paths can be ASCII letters, ASCII digits, and limited ASCII punctuation: - . _ and ~. This matches what "go get" has historically recognized in import paths. TODO(rsc): We would like to allow Unicode letters, but that requires additional care in the safe encoding (see "escaped paths" above).
func ( rune) bool {
	if  < utf8.RuneSelf {
		return  == '-' ||  == '.' ||  == '_' ||  == '~' ||
			'0' <=  &&  <= '9' ||
			'A' <=  &&  <= 'Z' ||
			'a' <=  &&  <= 'z'
	}
	return false
}
fileNameOK reports whether r can appear in a file name. For now we allow all Unicode letters but otherwise limit to pathOK plus a few more punctuation characters. If we expand the set of allowed characters here, we have to work harder at detecting potential case-folding and normalization collisions. See note about "escaped paths" above.
func ( rune) bool {
Entire set of ASCII punctuation, from which we remove characters: ! " # $ % & ' ( ) * + , - . / : ; < = > ? @ [ \ ] ^ _ ` { | } ~ We disallow some shell special characters: " ' * < > ? ` | (Note that some of those are disallowed by the Windows file system as well.) We also disallow path separators / : and \ (fileNameOK is only called on path element characters). We allow spaces (U+0020) in file names.
		const  = "!#$%&()+,-.=@[]^_{}~ "
		if '0' <=  &&  <= '9' || 'A' <=  &&  <= 'Z' || 'a' <=  &&  <= 'z' {
			return true
		}
		for  := 0;  < len(); ++ {
			if rune([]) ==  {
				return true
			}
		}
		return false
It may be OK to add more ASCII punctuation here, but only carefully. For example Windows disallows < > \, and macOS disallows :, so we must not allow those.
	return unicode.IsLetter()
}
CheckPath checks that a module path is valid. A valid module path is a valid import path, as checked by CheckImportPath, with two additional constraints. First, the leading path element (up to the first slash, if any), by convention a domain name, must contain only lower-case ASCII letters, ASCII digits, dots (U+002E), and dashes (U+002D); it must contain at least one dot and cannot start with a dash. Second, for a final path element of the form /vN, where N looks numeric (ASCII digits and dots) must not begin with a leading zero, must not be /v1, and must not contain any dots. For paths beginning with "gopkg.in/", this second requirement is replaced by a requirement that the path follow the gopkg.in server's conventions.
func ( string) error {
	if  := checkPath(, false);  != nil {
		return fmt.Errorf("malformed module path %q: %v", , )
	}
	 := strings.Index(, "/")
	if  < 0 {
		 = len()
	}
	if  == 0 {
		return fmt.Errorf("malformed module path %q: leading slash", )
	}
	if !strings.Contains([:], ".") {
		return fmt.Errorf("malformed module path %q: missing dot in first path element", )
	}
	if [0] == '-' {
		return fmt.Errorf("malformed module path %q: leading dash in first path element", )
	}
	for ,  := range [:] {
		if !firstPathOK() {
			return fmt.Errorf("malformed module path %q: invalid char %q in first path element", , )
		}
	}
	if , ,  := SplitPathVersion(); ! {
		return fmt.Errorf("malformed module path %q: invalid version", )
	}
	return nil
}
CheckImportPath checks that an import path is valid. A valid import path consists of one or more valid path elements separated by slashes (U+002F). (It must not begin with nor end in a slash.) A valid path element is a non-empty string made up of ASCII letters, ASCII digits, and limited ASCII punctuation: - . _ and ~. It must not begin or end with a dot (U+002E), nor contain two dots in a row. The element prefix up to the first dot must not be a reserved file name on Windows, regardless of case (CON, com1, NuL, and so on). The element must not have a suffix of a tilde followed by one or more ASCII digits (to exclude paths elements that look like Windows short-names). CheckImportPath may be less restrictive in the future, but see the top-level package documentation for additional information about subtleties of Unicode.
func ( string) error {
	if  := checkPath(, false);  != nil {
		return fmt.Errorf("malformed import path %q: %v", , )
	}
	return nil
}
checkPath checks that a general path is valid. It returns an error describing why but not mentioning path. Because these checks apply to both module paths and import paths, the caller is expected to add the "malformed ___ path %q: " prefix. fileName indicates whether the final element of the path is a file name (as opposed to a directory name).
func ( string,  bool) error {
	if !utf8.ValidString() {
		return fmt.Errorf("invalid UTF-8")
	}
	if  == "" {
		return fmt.Errorf("empty string")
	}
	if [0] == '-' {
		return fmt.Errorf("leading dash")
	}
	if strings.Contains(, "//") {
		return fmt.Errorf("double slash")
	}
	if [len()-1] == '/' {
		return fmt.Errorf("trailing slash")
	}
	 := 0
	for ,  := range  {
		if  == '/' {
			if  := checkElem([:], );  != nil {
				return 
			}
			 =  + 1
		}
	}
	if  := checkElem([:], );  != nil {
		return 
	}
	return nil
}
checkElem checks whether an individual path element is valid. fileName indicates whether the element is a file name (not a directory name).
func ( string,  bool) error {
	if  == "" {
		return fmt.Errorf("empty path element")
	}
	if strings.Count(, ".") == len() {
		return fmt.Errorf("invalid path element %q", )
	}
	if [0] == '.' && ! {
		return fmt.Errorf("leading dot in path element")
	}
	if [len()-1] == '.' {
		return fmt.Errorf("trailing dot in path element")
	}
	 := pathOK
	if  {
		 = fileNameOK
	}
	for ,  := range  {
		if !() {
			return fmt.Errorf("invalid char %q", )
		}
	}
Windows disallows a bunch of path elements, sadly. See https://docs.microsoft.com/en-us/windows/desktop/fileio/naming-a-file
	 := 
	if  := strings.Index(, ".");  >= 0 {
		 = [:]
	}
	for ,  := range badWindowsNames {
		if strings.EqualFold(, ) {
			return fmt.Errorf("%q disallowed as path element component on Windows", )
		}
	}

don't check for Windows short-names in file names. They're only an issue for import paths.
		return nil
	}
Reject path components that look like Windows short-names. Those usually end in a tilde followed by one or more ASCII digits.
	if  := strings.LastIndexByte(, '~');  >= 0 &&  < len()-1 {
		 := [+1:]
		 := true
		for ,  := range  {
			if  < '0' ||  > '9' {
				 = false
				break
			}
		}
		if  {
			return fmt.Errorf("trailing tilde and digits in path element")
		}
	}

	return nil
}
CheckFilePath checks that a slash-separated file path is valid. The definition of a valid file path is the same as the definition of a valid import path except that the set of allowed characters is larger: all Unicode letters, ASCII digits, the ASCII space character (U+0020), and the ASCII punctuation characters “!#$%&()+,-.=@[]^_{}~”. (The excluded punctuation characters, " * < > ? ` ' | / \ and :, have special meanings in certain shells or operating systems.) CheckFilePath may be less restrictive in the future, but see the top-level package documentation for additional information about subtleties of Unicode.
func ( string) error {
	if  := checkPath(, true);  != nil {
		return fmt.Errorf("malformed file path %q: %v", , )
	}
	return nil
}
badWindowsNames are the reserved file path elements on Windows. See https://docs.microsoft.com/en-us/windows/desktop/fileio/naming-a-file
var badWindowsNames = []string{
	"CON",
	"PRN",
	"AUX",
	"NUL",
	"COM1",
	"COM2",
	"COM3",
	"COM4",
	"COM5",
	"COM6",
	"COM7",
	"COM8",
	"COM9",
	"LPT1",
	"LPT2",
	"LPT3",
	"LPT4",
	"LPT5",
	"LPT6",
	"LPT7",
	"LPT8",
	"LPT9",
}
SplitPathVersion returns prefix and major version such that prefix+pathMajor == path and version is either empty or "/vN" for N >= 2. As a special case, gopkg.in paths are recognized directly; they require ".vN" instead of "/vN", and for all N, not just N >= 2. SplitPathVersion returns with ok = false when presented with a path whose last path element does not satisfy the constraints applied by CheckPath, such as "example.com/pkg/v1" or "example.com/pkg/v1.2".
func ( string) (,  string,  bool) {
	if strings.HasPrefix(, "gopkg.in/") {
		return splitGopkgIn()
	}

	 := len()
	 := false
	for  > 0 && ('0' <= [-1] && [-1] <= '9' || [-1] == '.') {
		if [-1] == '.' {
			 = true
		}
		--
	}
	if  <= 1 ||  == len() || [-1] != 'v' || [-2] != '/' {
		return , "", true
	}
	,  = [:-2], [-2:]
	if  || len() <= 2 || [2] == '0' ||  == "/v1" {
		return , "", false
	}
	return , , true
}
splitGopkgIn is like SplitPathVersion but only for gopkg.in paths.
func ( string) (,  string,  bool) {
	if !strings.HasPrefix(, "gopkg.in/") {
		return , "", false
	}
	 := len()
	if strings.HasSuffix(, "-unstable") {
		 -= len("-unstable")
	}
	for  > 0 && ('0' <= [-1] && [-1] <= '9') {
		--
	}
All gopkg.in paths must end in vN for some N.
		return , "", false
	}
	,  = [:-2], [-2:]
	if len() <= 2 || [2] == '0' &&  != ".v0" {
		return , "", false
	}
	return , , true
}
MatchPathMajor reports whether the semantic version v matches the path major version pathMajor. MatchPathMajor returns true if and only if CheckPathMajor returns nil.
func (,  string) bool {
	return CheckPathMajor(, ) == nil
}
CheckPathMajor returns a non-nil error if the semantic version v does not match the path major version pathMajor.
TODO(jayconrod): return errors or panic for invalid inputs. This function (and others) was covered by integration tests for cmd/go, and surrounding code protected against invalid inputs like non-canonical versions.
	if strings.HasPrefix(, ".v") && strings.HasSuffix(, "-unstable") {
		 = strings.TrimSuffix(, "-unstable")
	}
Allow old bug in pseudo-versions that generated v0.0.0- pseudoversion for gopkg .v1. For example, gopkg.in/yaml.v2@v2.2.1's go.mod requires gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405.
		return nil
	}
	 := semver.Major()
	if  == "" {
		if  == "v0" ||  == "v1" || semver.Build() == "+incompatible" {
			return nil
		}
		 = "v0 or v1"
	} else if [0] == '/' || [0] == '.' {
		if  == [1:] {
			return nil
		}
		 = [1:]
	}
	return &InvalidVersionError{
		Version: ,
		Err:     fmt.Errorf("should be %s, not %s", , semver.Major()),
	}
}
PathMajorPrefix returns the major-version tag prefix implied by pathMajor. An empty PathMajorPrefix allows either v0 or v1. Note that MatchPathMajor may accept some versions that do not actually begin with this prefix: namely, it accepts a 'v0.0.0-' prefix for a '.v1' pathMajor, even though that pathMajor implies 'v1' tagging.
func ( string) string {
	if  == "" {
		return ""
	}
	if [0] != '/' && [0] != '.' {
		panic("pathMajor suffix " +  + " passed to PathMajorPrefix lacks separator")
	}
	if strings.HasPrefix(, ".v") && strings.HasSuffix(, "-unstable") {
		 = strings.TrimSuffix(, "-unstable")
	}
	 := [1:]
	if  != semver.Major() {
		panic("pathMajor suffix " +  + "passed to PathMajorPrefix is not a valid major version")
	}
	return 
}
CanonicalVersion returns the canonical form of the version string v. It is the same as semver.Canonical(v) except that it preserves the special build suffix "+incompatible".
func ( string) string {
	 := semver.Canonical()
	if semver.Build() == "+incompatible" {
		 += "+incompatible"
	}
	return 
}
Sort sorts the list by Path, breaking ties by comparing Version fields. The Version fields are interpreted as semantic versions (using semver.Compare) optionally followed by a tie-breaking suffix introduced by a slash character, like in "v0.0.1/go.mod".
func ( []Version) {
	sort.Slice(, func(,  int) bool {
		 := []
		 := []
		if .Path != .Path {
			return .Path < .Path
To help go.sum formatting, allow version/file. Compare semver prefix by semver rules, file by string order.
		 := .Version
		 := .Version
		var ,  string
		if  := strings.Index(, "/");  >= 0 {
			,  = [:], [:]
		}
		if  := strings.Index(, "/");  >= 0 {
			,  = [:], [:]
		}
		if  !=  {
			return semver.Compare(, ) < 0
		}
		return  < 
	})
}
EscapePath returns the escaped form of the given module path. It fails if the module path is invalid.
func ( string) ( string,  error) {
	if  := CheckPath();  != nil {
		return "", 
	}

	return escapeString()
}
EscapeVersion returns the escaped form of the given module version. Versions are allowed to be in non-semver form but must be valid file names and not contain exclamation marks.
func ( string) ( string,  error) {
	if  := checkElem(, true);  != nil || strings.Contains(, "!") {
		return "", &InvalidVersionError{
			Version: ,
			Err:     fmt.Errorf("disallowed version string"),
		}
	}
	return escapeString()
}

func ( string) ( string,  error) {
	 := false
	for ,  := range  {
This should be disallowed by CheckPath, but diagnose anyway. The correctness of the escaping loop below depends on it.
			return "", fmt.Errorf("internal error: inconsistency in EscapePath")
		}
		if 'A' <=  &&  <= 'Z' {
			 = true
		}
	}

	if ! {
		return , nil
	}

	var  []byte
	for ,  := range  {
		if 'A' <=  &&  <= 'Z' {
			 = append(, '!', byte(+'a'-'A'))
		} else {
			 = append(, byte())
		}
	}
	return string(), nil
}
UnescapePath returns the module path for the given escaped path. It fails if the escaped path is invalid or describes an invalid path.
func ( string) ( string,  error) {
	,  := unescapeString()
	if ! {
		return "", fmt.Errorf("invalid escaped module path %q", )
	}
	if  := CheckPath();  != nil {
		return "", fmt.Errorf("invalid escaped module path %q: %v", , )
	}
	return , nil
}
UnescapeVersion returns the version string for the given escaped version. It fails if the escaped form is invalid or describes an invalid version. Versions are allowed to be in non-semver form but must be valid file names and not contain exclamation marks.
func ( string) ( string,  error) {
	,  := unescapeString()
	if ! {
		return "", fmt.Errorf("invalid escaped version %q", )
	}
	if  := checkElem(, true);  != nil {
		return "", fmt.Errorf("invalid escaped version %q: %v", , )
	}
	return , nil
}

func ( string) (string, bool) {
	var  []byte

	 := false
	for ,  := range  {
		if  >= utf8.RuneSelf {
			return "", false
		}
		if  {
			 = false
			if  < 'a' || 'z' <  {
				return "", false
			}
			 = append(, byte(+'A'-'a'))
			continue
		}
		if  == '!' {
			 = true
			continue
		}
		if 'A' <=  &&  <= 'Z' {
			return "", false
		}
		 = append(, byte())
	}
	if  {
		return "", false
	}
	return string(), true
}
MatchPrefixPatterns reports whether any path prefix of target matches one of the glob patterns (as defined by path.Match) in the comma-separated globs list. This implements the algorithm used when matching a module path to the GOPRIVATE environment variable, as described by 'go help module-private'. It ignores any empty or malformed patterns in the list.
func (,  string) bool {
Extract next non-empty glob in comma-separated list.
		var  string
		if  := strings.Index(, ",");  >= 0 {
			,  = [:], [+1:]
		} else {
			,  = , ""
		}
		if  == "" {
			continue
		}
A glob with N+1 path elements (N slashes) needs to be matched against the first N+1 path elements of target, which end just before the N+1'th slash.
		 := strings.Count(, "/")
Walk target, counting slashes, truncating at the N+1'th slash.
		for  := 0;  < len(); ++ {
			if [] == '/' {
				if  == 0 {
					 = [:]
					break
				}
				--
			}
		}
Not enough prefix elements.
			continue
		}
		,  := path.Match(, )
		if  {
			return true
		}
	}
	return false