github.com/bir3/gocompiler@v0.9.2202/src/xvendor/golang.org/x/mod/sumdb/tlog/note.go (about) 1 // Copyright 2019 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 tlog 6 7 import ( 8 "bytes" 9 "encoding/base64" 10 "errors" 11 "fmt" 12 "strconv" 13 "strings" 14 "unicode/utf8" 15 ) 16 17 // A Tree is a tree description, to be signed by a go.sum database server. 18 type Tree struct { 19 N int64 20 Hash Hash 21 } 22 23 // FormatTree formats a tree description for inclusion in a note. 24 // 25 // The encoded form is three lines, each ending in a newline (U+000A): 26 // 27 // go.sum database tree 28 // N 29 // Hash 30 // 31 // where N is in decimal and Hash is in base64. 32 // 33 // A future backwards-compatible encoding may add additional lines, 34 // which the parser can ignore. 35 // A future backwards-incompatible encoding would use a different 36 // first line (for example, "go.sum database tree v2"). 37 func FormatTree(tree Tree) []byte { 38 return []byte(fmt.Sprintf("go.sum database tree\n%d\n%s\n", tree.N, tree.Hash)) 39 } 40 41 var errMalformedTree = errors.New("malformed tree note") 42 var treePrefix = []byte("go.sum database tree\n") 43 44 // ParseTree parses a formatted tree root description. 45 func ParseTree(text []byte) (tree Tree, err error) { 46 // The message looks like: 47 // 48 // go.sum database tree 49 // 2 50 // nND/nri/U0xuHUrYSy0HtMeal2vzD9V4k/BO79C+QeI= 51 // 52 // For forwards compatibility, extra text lines after the encoding are ignored. 53 if !bytes.HasPrefix(text, treePrefix) || bytes.Count(text, []byte("\n")) < 3 || len(text) > 1e6 { 54 return Tree{}, errMalformedTree 55 } 56 57 lines := strings.SplitN(string(text), "\n", 4) 58 n, err := strconv.ParseInt(lines[1], 10, 64) 59 if err != nil || n < 0 || lines[1] != strconv.FormatInt(n, 10) { 60 return Tree{}, errMalformedTree 61 } 62 63 h, err := base64.StdEncoding.DecodeString(lines[2]) 64 if err != nil || len(h) != HashSize { 65 return Tree{}, errMalformedTree 66 } 67 68 var hash Hash 69 copy(hash[:], h) 70 return Tree{n, hash}, nil 71 } 72 73 var errMalformedRecord = errors.New("malformed record data") 74 75 // FormatRecord formats a record for serving to a client 76 // in a lookup response or data tile. 77 // 78 // The encoded form is the record ID as a single number, 79 // then the text of the record, and then a terminating blank line. 80 // Record text must be valid UTF-8 and must not contain any ASCII control 81 // characters (those below U+0020) other than newline (U+000A). 82 // It must end in a terminating newline and not contain any blank lines. 83 func FormatRecord(id int64, text []byte) (msg []byte, err error) { 84 if !isValidRecordText(text) { 85 return nil, errMalformedRecord 86 } 87 msg = []byte(fmt.Sprintf("%d\n", id)) 88 msg = append(msg, text...) 89 msg = append(msg, '\n') 90 return msg, nil 91 } 92 93 // isValidRecordText reports whether text is syntactically valid record text. 94 func isValidRecordText(text []byte) bool { 95 var last rune 96 for i := 0; i < len(text); { 97 r, size := utf8.DecodeRune(text[i:]) 98 if r < 0x20 && r != '\n' || r == utf8.RuneError && size == 1 || last == '\n' && r == '\n' { 99 return false 100 } 101 i += size 102 last = r 103 } 104 if last != '\n' { 105 return false 106 } 107 return true 108 } 109 110 // ParseRecord parses a record description at the start of text, 111 // stopping immediately after the terminating blank line. 112 // It returns the record id, the record text, and the remainder of text. 113 func ParseRecord(msg []byte) (id int64, text, rest []byte, err error) { 114 // Leading record id. 115 i := bytes.IndexByte(msg, '\n') 116 if i < 0 { 117 return 0, nil, nil, errMalformedRecord 118 } 119 id, err = strconv.ParseInt(string(msg[:i]), 10, 64) 120 if err != nil { 121 return 0, nil, nil, errMalformedRecord 122 } 123 msg = msg[i+1:] 124 125 // Record text. 126 i = bytes.Index(msg, []byte("\n\n")) 127 if i < 0 { 128 return 0, nil, nil, errMalformedRecord 129 } 130 text, rest = msg[:i+1], msg[i+2:] 131 if !isValidRecordText(text) { 132 return 0, nil, nil, errMalformedRecord 133 } 134 return id, text, rest, nil 135 }