...

Source file src/pkg/cmd/go/internal/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 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	}
   136	

View as plain text