github.com/go-asm/go@v1.21.1-0.20240213172139-40c5ead50c48/cmd/go/vcs/discovery.go (about) 1 // Copyright 2012 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 package vcs 6 7 import ( 8 "encoding/xml" 9 "fmt" 10 "io" 11 "strings" 12 ) 13 14 // charsetReader returns a reader that converts from the given charset to UTF-8. 15 // Currently it only supports UTF-8 and ASCII. Otherwise, it returns a meaningful 16 // error which is printed by go get, so the user can find why the package 17 // wasn't downloaded if the encoding is not supported. Note that, in 18 // order to reduce potential errors, ASCII is treated as UTF-8 (i.e. characters 19 // greater than 0x7f are not rejected). 20 func charsetReader(charset string, input io.Reader) (io.Reader, error) { 21 switch strings.ToLower(charset) { 22 case "utf-8", "ascii": 23 return input, nil 24 default: 25 return nil, fmt.Errorf("can't decode XML document using charset %q", charset) 26 } 27 } 28 29 // parseMetaGoImports returns meta imports from the HTML in r. 30 // Parsing ends at the end of the <head> section or the beginning of the <body>. 31 func parseMetaGoImports(r io.Reader, mod ModuleMode) ([]metaImport, error) { 32 d := xml.NewDecoder(r) 33 d.CharsetReader = charsetReader 34 d.Strict = false 35 var imports []metaImport 36 for { 37 t, err := d.RawToken() 38 if err != nil { 39 if err != io.EOF && len(imports) == 0 { 40 return nil, err 41 } 42 break 43 } 44 if e, ok := t.(xml.StartElement); ok && strings.EqualFold(e.Name.Local, "body") { 45 break 46 } 47 if e, ok := t.(xml.EndElement); ok && strings.EqualFold(e.Name.Local, "head") { 48 break 49 } 50 e, ok := t.(xml.StartElement) 51 if !ok || !strings.EqualFold(e.Name.Local, "meta") { 52 continue 53 } 54 if attrValue(e.Attr, "name") != "go-import" { 55 continue 56 } 57 if f := strings.Fields(attrValue(e.Attr, "content")); len(f) == 3 { 58 imports = append(imports, metaImport{ 59 Prefix: f[0], 60 VCS: f[1], 61 RepoRoot: f[2], 62 }) 63 } 64 } 65 66 // Extract mod entries if we are paying attention to them. 67 var list []metaImport 68 var have map[string]bool 69 if mod == PreferMod { 70 have = make(map[string]bool) 71 for _, m := range imports { 72 if m.VCS == "mod" { 73 have[m.Prefix] = true 74 list = append(list, m) 75 } 76 } 77 } 78 79 // Append non-mod entries, ignoring those superseded by a mod entry. 80 for _, m := range imports { 81 if m.VCS != "mod" && !have[m.Prefix] { 82 list = append(list, m) 83 } 84 } 85 return list, nil 86 } 87 88 // attrValue returns the attribute value for the case-insensitive key 89 // `name', or the empty string if nothing is found. 90 func attrValue(attrs []xml.Attr, name string) string { 91 for _, a := range attrs { 92 if strings.EqualFold(a.Name.Local, name) { 93 return a.Value 94 } 95 } 96 return "" 97 }