163 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
		
		
			
		
	
	
			163 lines
		
	
	
		
			4.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
|   | // Copyright 2012 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 packet | ||
|  | 
 | ||
|  | import ( | ||
|  | 	"bytes" | ||
|  | 	"io" | ||
|  | 	"io/ioutil" | ||
|  | 
 | ||
|  | 	"golang.org/x/crypto/openpgp/errors" | ||
|  | ) | ||
|  | 
 | ||
|  | // OpaquePacket represents an OpenPGP packet as raw, unparsed data. This is | ||
|  | // useful for splitting and storing the original packet contents separately, | ||
|  | // handling unsupported packet types or accessing parts of the packet not yet | ||
|  | // implemented by this package. | ||
|  | type OpaquePacket struct { | ||
|  | 	// Packet type | ||
|  | 	Tag uint8 | ||
|  | 	// Reason why the packet was parsed opaquely | ||
|  | 	Reason error | ||
|  | 	// Binary contents of the packet data | ||
|  | 	Contents []byte | ||
|  | } | ||
|  | 
 | ||
|  | func (op *OpaquePacket) parse(r io.Reader) (err error) { | ||
|  | 	op.Contents, err = ioutil.ReadAll(r) | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | // Serialize marshals the packet to a writer in its original form, including | ||
|  | // the packet header. | ||
|  | func (op *OpaquePacket) Serialize(w io.Writer) (err error) { | ||
|  | 	err = serializeHeader(w, packetType(op.Tag), len(op.Contents)) | ||
|  | 	if err == nil { | ||
|  | 		_, err = w.Write(op.Contents) | ||
|  | 	} | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | // Parse attempts to parse the opaque contents into a structure supported by | ||
|  | // this package. If the packet is not known then the result will be another | ||
|  | // OpaquePacket. | ||
|  | func (op *OpaquePacket) Parse() (p Packet, err error) { | ||
|  | 	hdr := bytes.NewBuffer(nil) | ||
|  | 	err = serializeHeader(hdr, packetType(op.Tag), len(op.Contents)) | ||
|  | 	if err != nil { | ||
|  | 		op.Reason = err | ||
|  | 		return op, err | ||
|  | 	} | ||
|  | 	p, err = Read(io.MultiReader(hdr, bytes.NewBuffer(op.Contents))) | ||
|  | 	if err != nil { | ||
|  | 		op.Reason = err | ||
|  | 		p = op | ||
|  | 	} | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | // OpaqueReader reads OpaquePackets from an io.Reader. | ||
|  | type OpaqueReader struct { | ||
|  | 	r io.Reader | ||
|  | } | ||
|  | 
 | ||
|  | func NewOpaqueReader(r io.Reader) *OpaqueReader { | ||
|  | 	return &OpaqueReader{r: r} | ||
|  | } | ||
|  | 
 | ||
|  | // Read the next OpaquePacket. | ||
|  | func (or *OpaqueReader) Next() (op *OpaquePacket, err error) { | ||
|  | 	tag, _, contents, err := readHeader(or.r) | ||
|  | 	if err != nil { | ||
|  | 		return | ||
|  | 	} | ||
|  | 	op = &OpaquePacket{Tag: uint8(tag), Reason: err} | ||
|  | 	err = op.parse(contents) | ||
|  | 	if err != nil { | ||
|  | 		consumeAll(contents) | ||
|  | 	} | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | // OpaqueSubpacket represents an unparsed OpenPGP subpacket, | ||
|  | // as found in signature and user attribute packets. | ||
|  | type OpaqueSubpacket struct { | ||
|  | 	SubType  uint8 | ||
|  | 	Contents []byte | ||
|  | } | ||
|  | 
 | ||
|  | // OpaqueSubpackets extracts opaque, unparsed OpenPGP subpackets from | ||
|  | // their byte representation. | ||
|  | func OpaqueSubpackets(contents []byte) (result []*OpaqueSubpacket, err error) { | ||
|  | 	var ( | ||
|  | 		subHeaderLen int | ||
|  | 		subPacket    *OpaqueSubpacket | ||
|  | 	) | ||
|  | 	for len(contents) > 0 { | ||
|  | 		subHeaderLen, subPacket, err = nextSubpacket(contents) | ||
|  | 		if err != nil { | ||
|  | 			break | ||
|  | 		} | ||
|  | 		result = append(result, subPacket) | ||
|  | 		contents = contents[subHeaderLen+len(subPacket.Contents):] | ||
|  | 	} | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | func nextSubpacket(contents []byte) (subHeaderLen int, subPacket *OpaqueSubpacket, err error) { | ||
|  | 	// RFC 4880, section 5.2.3.1 | ||
|  | 	var subLen uint32 | ||
|  | 	if len(contents) < 1 { | ||
|  | 		goto Truncated | ||
|  | 	} | ||
|  | 	subPacket = &OpaqueSubpacket{} | ||
|  | 	switch { | ||
|  | 	case contents[0] < 192: | ||
|  | 		subHeaderLen = 2 // 1 length byte, 1 subtype byte | ||
|  | 		if len(contents) < subHeaderLen { | ||
|  | 			goto Truncated | ||
|  | 		} | ||
|  | 		subLen = uint32(contents[0]) | ||
|  | 		contents = contents[1:] | ||
|  | 	case contents[0] < 255: | ||
|  | 		subHeaderLen = 3 // 2 length bytes, 1 subtype | ||
|  | 		if len(contents) < subHeaderLen { | ||
|  | 			goto Truncated | ||
|  | 		} | ||
|  | 		subLen = uint32(contents[0]-192)<<8 + uint32(contents[1]) + 192 | ||
|  | 		contents = contents[2:] | ||
|  | 	default: | ||
|  | 		subHeaderLen = 6 // 5 length bytes, 1 subtype | ||
|  | 		if len(contents) < subHeaderLen { | ||
|  | 			goto Truncated | ||
|  | 		} | ||
|  | 		subLen = uint32(contents[1])<<24 | | ||
|  | 			uint32(contents[2])<<16 | | ||
|  | 			uint32(contents[3])<<8 | | ||
|  | 			uint32(contents[4]) | ||
|  | 		contents = contents[5:] | ||
|  | 	} | ||
|  | 	if subLen > uint32(len(contents)) || subLen == 0 { | ||
|  | 		goto Truncated | ||
|  | 	} | ||
|  | 	subPacket.SubType = contents[0] | ||
|  | 	subPacket.Contents = contents[1:subLen] | ||
|  | 	return | ||
|  | Truncated: | ||
|  | 	err = errors.StructuralError("subpacket truncated") | ||
|  | 	return | ||
|  | } | ||
|  | 
 | ||
|  | func (osp *OpaqueSubpacket) Serialize(w io.Writer) (err error) { | ||
|  | 	buf := make([]byte, 6) | ||
|  | 	n := serializeSubpacketLength(buf, len(osp.Contents)+1) | ||
|  | 	buf[n] = osp.SubType | ||
|  | 	if _, err = w.Write(buf[:n+1]); err != nil { | ||
|  | 		return | ||
|  | 	} | ||
|  | 	_, err = w.Write(osp.Contents) | ||
|  | 	return | ||
|  | } |