Giant blob of minor changes
[dotfiles/.git] / .config / coc / extensions / coc-go-data / tools / pkg / mod / golang.org / x / mod@v0.3.0 / sumdb / tlog / note.go
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 }