mirror of
				https://github.com/go-gitea/gitea
				synced 2025-11-04 05:18:25 +00:00 
			
		
		
		
	* [Vendor] update go-ldap to v3.0.3 * update go-ldap to v3.2.4 Co-authored-by: techknowlogick <techknowlogick@gitea.io>
		
			
				
	
	
		
			621 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
			
		
		
	
	
			621 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
package ber
 | 
						|
 | 
						|
import (
 | 
						|
	"bytes"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"math"
 | 
						|
	"os"
 | 
						|
	"reflect"
 | 
						|
	"time"
 | 
						|
	"unicode/utf8"
 | 
						|
)
 | 
						|
 | 
						|
// MaxPacketLengthBytes specifies the maximum allowed packet size when calling ReadPacket or DecodePacket. Set to 0 for
 | 
						|
// no limit.
 | 
						|
var MaxPacketLengthBytes int64 = math.MaxInt32
 | 
						|
 | 
						|
type Packet struct {
 | 
						|
	Identifier
 | 
						|
	Value       interface{}
 | 
						|
	ByteValue   []byte
 | 
						|
	Data        *bytes.Buffer
 | 
						|
	Children    []*Packet
 | 
						|
	Description string
 | 
						|
}
 | 
						|
 | 
						|
type Identifier struct {
 | 
						|
	ClassType Class
 | 
						|
	TagType   Type
 | 
						|
	Tag       Tag
 | 
						|
}
 | 
						|
 | 
						|
type Tag uint64
 | 
						|
 | 
						|
const (
 | 
						|
	TagEOC              Tag = 0x00
 | 
						|
	TagBoolean          Tag = 0x01
 | 
						|
	TagInteger          Tag = 0x02
 | 
						|
	TagBitString        Tag = 0x03
 | 
						|
	TagOctetString      Tag = 0x04
 | 
						|
	TagNULL             Tag = 0x05
 | 
						|
	TagObjectIdentifier Tag = 0x06
 | 
						|
	TagObjectDescriptor Tag = 0x07
 | 
						|
	TagExternal         Tag = 0x08
 | 
						|
	TagRealFloat        Tag = 0x09
 | 
						|
	TagEnumerated       Tag = 0x0a
 | 
						|
	TagEmbeddedPDV      Tag = 0x0b
 | 
						|
	TagUTF8String       Tag = 0x0c
 | 
						|
	TagRelativeOID      Tag = 0x0d
 | 
						|
	TagSequence         Tag = 0x10
 | 
						|
	TagSet              Tag = 0x11
 | 
						|
	TagNumericString    Tag = 0x12
 | 
						|
	TagPrintableString  Tag = 0x13
 | 
						|
	TagT61String        Tag = 0x14
 | 
						|
	TagVideotexString   Tag = 0x15
 | 
						|
	TagIA5String        Tag = 0x16
 | 
						|
	TagUTCTime          Tag = 0x17
 | 
						|
	TagGeneralizedTime  Tag = 0x18
 | 
						|
	TagGraphicString    Tag = 0x19
 | 
						|
	TagVisibleString    Tag = 0x1a
 | 
						|
	TagGeneralString    Tag = 0x1b
 | 
						|
	TagUniversalString  Tag = 0x1c
 | 
						|
	TagCharacterString  Tag = 0x1d
 | 
						|
	TagBMPString        Tag = 0x1e
 | 
						|
	TagBitmask          Tag = 0x1f // xxx11111b
 | 
						|
 | 
						|
	// HighTag indicates the start of a high-tag byte sequence
 | 
						|
	HighTag Tag = 0x1f // xxx11111b
 | 
						|
	// HighTagContinueBitmask indicates the high-tag byte sequence should continue
 | 
						|
	HighTagContinueBitmask Tag = 0x80 // 10000000b
 | 
						|
	// HighTagValueBitmask obtains the tag value from a high-tag byte sequence byte
 | 
						|
	HighTagValueBitmask Tag = 0x7f // 01111111b
 | 
						|
)
 | 
						|
 | 
						|
const (
 | 
						|
	// LengthLongFormBitmask is the mask to apply to the length byte to see if a long-form byte sequence is used
 | 
						|
	LengthLongFormBitmask = 0x80
 | 
						|
	// LengthValueBitmask is the mask to apply to the length byte to get the number of bytes in the long-form byte sequence
 | 
						|
	LengthValueBitmask = 0x7f
 | 
						|
 | 
						|
	// LengthIndefinite is returned from readLength to indicate an indefinite length
 | 
						|
	LengthIndefinite = -1
 | 
						|
)
 | 
						|
 | 
						|
var tagMap = map[Tag]string{
 | 
						|
	TagEOC:              "EOC (End-of-Content)",
 | 
						|
	TagBoolean:          "Boolean",
 | 
						|
	TagInteger:          "Integer",
 | 
						|
	TagBitString:        "Bit String",
 | 
						|
	TagOctetString:      "Octet String",
 | 
						|
	TagNULL:             "NULL",
 | 
						|
	TagObjectIdentifier: "Object Identifier",
 | 
						|
	TagObjectDescriptor: "Object Descriptor",
 | 
						|
	TagExternal:         "External",
 | 
						|
	TagRealFloat:        "Real (float)",
 | 
						|
	TagEnumerated:       "Enumerated",
 | 
						|
	TagEmbeddedPDV:      "Embedded PDV",
 | 
						|
	TagUTF8String:       "UTF8 String",
 | 
						|
	TagRelativeOID:      "Relative-OID",
 | 
						|
	TagSequence:         "Sequence and Sequence of",
 | 
						|
	TagSet:              "Set and Set OF",
 | 
						|
	TagNumericString:    "Numeric String",
 | 
						|
	TagPrintableString:  "Printable String",
 | 
						|
	TagT61String:        "T61 String",
 | 
						|
	TagVideotexString:   "Videotex String",
 | 
						|
	TagIA5String:        "IA5 String",
 | 
						|
	TagUTCTime:          "UTC Time",
 | 
						|
	TagGeneralizedTime:  "Generalized Time",
 | 
						|
	TagGraphicString:    "Graphic String",
 | 
						|
	TagVisibleString:    "Visible String",
 | 
						|
	TagGeneralString:    "General String",
 | 
						|
	TagUniversalString:  "Universal String",
 | 
						|
	TagCharacterString:  "Character String",
 | 
						|
	TagBMPString:        "BMP String",
 | 
						|
}
 | 
						|
 | 
						|
type Class uint8
 | 
						|
 | 
						|
const (
 | 
						|
	ClassUniversal   Class = 0   // 00xxxxxxb
 | 
						|
	ClassApplication Class = 64  // 01xxxxxxb
 | 
						|
	ClassContext     Class = 128 // 10xxxxxxb
 | 
						|
	ClassPrivate     Class = 192 // 11xxxxxxb
 | 
						|
	ClassBitmask     Class = 192 // 11xxxxxxb
 | 
						|
)
 | 
						|
 | 
						|
var ClassMap = map[Class]string{
 | 
						|
	ClassUniversal:   "Universal",
 | 
						|
	ClassApplication: "Application",
 | 
						|
	ClassContext:     "Context",
 | 
						|
	ClassPrivate:     "Private",
 | 
						|
}
 | 
						|
 | 
						|
type Type uint8
 | 
						|
 | 
						|
const (
 | 
						|
	TypePrimitive   Type = 0  // xx0xxxxxb
 | 
						|
	TypeConstructed Type = 32 // xx1xxxxxb
 | 
						|
	TypeBitmask     Type = 32 // xx1xxxxxb
 | 
						|
)
 | 
						|
 | 
						|
var TypeMap = map[Type]string{
 | 
						|
	TypePrimitive:   "Primitive",
 | 
						|
	TypeConstructed: "Constructed",
 | 
						|
}
 | 
						|
 | 
						|
var Debug = false
 | 
						|
 | 
						|
func PrintBytes(out io.Writer, buf []byte, indent string) {
 | 
						|
	dataLines := make([]string, (len(buf)/30)+1)
 | 
						|
	numLines := make([]string, (len(buf)/30)+1)
 | 
						|
 | 
						|
	for i, b := range buf {
 | 
						|
		dataLines[i/30] += fmt.Sprintf("%02x ", b)
 | 
						|
		numLines[i/30] += fmt.Sprintf("%02d ", (i+1)%100)
 | 
						|
	}
 | 
						|
 | 
						|
	for i := 0; i < len(dataLines); i++ {
 | 
						|
		_, _ = out.Write([]byte(indent + dataLines[i] + "\n"))
 | 
						|
		_, _ = out.Write([]byte(indent + numLines[i] + "\n\n"))
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func WritePacket(out io.Writer, p *Packet) {
 | 
						|
	printPacket(out, p, 0, false)
 | 
						|
}
 | 
						|
 | 
						|
func PrintPacket(p *Packet) {
 | 
						|
	printPacket(os.Stdout, p, 0, false)
 | 
						|
}
 | 
						|
 | 
						|
func printPacket(out io.Writer, p *Packet, indent int, printBytes bool) {
 | 
						|
	indentStr := ""
 | 
						|
 | 
						|
	for len(indentStr) != indent {
 | 
						|
		indentStr += " "
 | 
						|
	}
 | 
						|
 | 
						|
	classStr := ClassMap[p.ClassType]
 | 
						|
 | 
						|
	tagTypeStr := TypeMap[p.TagType]
 | 
						|
 | 
						|
	tagStr := fmt.Sprintf("0x%02X", p.Tag)
 | 
						|
 | 
						|
	if p.ClassType == ClassUniversal {
 | 
						|
		tagStr = tagMap[p.Tag]
 | 
						|
	}
 | 
						|
 | 
						|
	value := fmt.Sprint(p.Value)
 | 
						|
	description := ""
 | 
						|
 | 
						|
	if p.Description != "" {
 | 
						|
		description = p.Description + ": "
 | 
						|
	}
 | 
						|
 | 
						|
	_, _ = fmt.Fprintf(out, "%s%s(%s, %s, %s) Len=%d %q\n", indentStr, description, classStr, tagTypeStr, tagStr, p.Data.Len(), value)
 | 
						|
 | 
						|
	if printBytes {
 | 
						|
		PrintBytes(out, p.Bytes(), indentStr)
 | 
						|
	}
 | 
						|
 | 
						|
	for _, child := range p.Children {
 | 
						|
		printPacket(out, child, indent+1, printBytes)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// ReadPacket reads a single Packet from the reader.
 | 
						|
func ReadPacket(reader io.Reader) (*Packet, error) {
 | 
						|
	p, _, err := readPacket(reader)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return p, nil
 | 
						|
}
 | 
						|
 | 
						|
func DecodeString(data []byte) string {
 | 
						|
	return string(data)
 | 
						|
}
 | 
						|
 | 
						|
func ParseInt64(bytes []byte) (ret int64, err error) {
 | 
						|
	if len(bytes) > 8 {
 | 
						|
		// We'll overflow an int64 in this case.
 | 
						|
		err = fmt.Errorf("integer too large")
 | 
						|
		return
 | 
						|
	}
 | 
						|
	for bytesRead := 0; bytesRead < len(bytes); bytesRead++ {
 | 
						|
		ret <<= 8
 | 
						|
		ret |= int64(bytes[bytesRead])
 | 
						|
	}
 | 
						|
 | 
						|
	// Shift up and down in order to sign extend the result.
 | 
						|
	ret <<= 64 - uint8(len(bytes))*8
 | 
						|
	ret >>= 64 - uint8(len(bytes))*8
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func encodeInteger(i int64) []byte {
 | 
						|
	n := int64Length(i)
 | 
						|
	out := make([]byte, n)
 | 
						|
 | 
						|
	var j int
 | 
						|
	for ; n > 0; n-- {
 | 
						|
		out[j] = byte(i >> uint((n-1)*8))
 | 
						|
		j++
 | 
						|
	}
 | 
						|
 | 
						|
	return out
 | 
						|
}
 | 
						|
 | 
						|
func int64Length(i int64) (numBytes int) {
 | 
						|
	numBytes = 1
 | 
						|
 | 
						|
	for i > 127 {
 | 
						|
		numBytes++
 | 
						|
		i >>= 8
 | 
						|
	}
 | 
						|
 | 
						|
	for i < -128 {
 | 
						|
		numBytes++
 | 
						|
		i >>= 8
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// DecodePacket decodes the given bytes into a single Packet
 | 
						|
// If a decode error is encountered, nil is returned.
 | 
						|
func DecodePacket(data []byte) *Packet {
 | 
						|
	p, _, _ := readPacket(bytes.NewBuffer(data))
 | 
						|
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
// DecodePacketErr decodes the given bytes into a single Packet
 | 
						|
// If a decode error is encountered, nil is returned.
 | 
						|
func DecodePacketErr(data []byte) (*Packet, error) {
 | 
						|
	p, _, err := readPacket(bytes.NewBuffer(data))
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return p, nil
 | 
						|
}
 | 
						|
 | 
						|
// readPacket reads a single Packet from the reader, returning the number of bytes read.
 | 
						|
func readPacket(reader io.Reader) (*Packet, int, error) {
 | 
						|
	identifier, length, read, err := readHeader(reader)
 | 
						|
	if err != nil {
 | 
						|
		return nil, read, err
 | 
						|
	}
 | 
						|
 | 
						|
	p := &Packet{
 | 
						|
		Identifier: identifier,
 | 
						|
	}
 | 
						|
 | 
						|
	p.Data = new(bytes.Buffer)
 | 
						|
	p.Children = make([]*Packet, 0, 2)
 | 
						|
	p.Value = nil
 | 
						|
 | 
						|
	if p.TagType == TypeConstructed {
 | 
						|
		// TODO: if universal, ensure tag type is allowed to be constructed
 | 
						|
 | 
						|
		// Track how much content we've read
 | 
						|
		contentRead := 0
 | 
						|
		for {
 | 
						|
			if length != LengthIndefinite {
 | 
						|
				// End if we've read what we've been told to
 | 
						|
				if contentRead == length {
 | 
						|
					break
 | 
						|
				}
 | 
						|
				// Detect if a packet boundary didn't fall on the expected length
 | 
						|
				if contentRead > length {
 | 
						|
					return nil, read, fmt.Errorf("expected to read %d bytes, read %d", length, contentRead)
 | 
						|
				}
 | 
						|
			}
 | 
						|
 | 
						|
			// Read the next packet
 | 
						|
			child, r, err := readPacket(reader)
 | 
						|
			if err != nil {
 | 
						|
				return nil, read, err
 | 
						|
			}
 | 
						|
			contentRead += r
 | 
						|
			read += r
 | 
						|
 | 
						|
			// Test is this is the EOC marker for our packet
 | 
						|
			if isEOCPacket(child) {
 | 
						|
				if length == LengthIndefinite {
 | 
						|
					break
 | 
						|
				}
 | 
						|
				return nil, read, errors.New("eoc child not allowed with definite length")
 | 
						|
			}
 | 
						|
 | 
						|
			// Append and continue
 | 
						|
			p.AppendChild(child)
 | 
						|
		}
 | 
						|
		return p, read, nil
 | 
						|
	}
 | 
						|
 | 
						|
	if length == LengthIndefinite {
 | 
						|
		return nil, read, errors.New("indefinite length used with primitive type")
 | 
						|
	}
 | 
						|
 | 
						|
	// Read definite-length content
 | 
						|
	if MaxPacketLengthBytes > 0 && int64(length) > MaxPacketLengthBytes {
 | 
						|
		return nil, read, fmt.Errorf("length %d greater than maximum %d", length, MaxPacketLengthBytes)
 | 
						|
	}
 | 
						|
	content := make([]byte, length)
 | 
						|
	if length > 0 {
 | 
						|
		_, err := io.ReadFull(reader, content)
 | 
						|
		if err != nil {
 | 
						|
			if err == io.EOF {
 | 
						|
				return nil, read, io.ErrUnexpectedEOF
 | 
						|
			}
 | 
						|
			return nil, read, err
 | 
						|
		}
 | 
						|
		read += length
 | 
						|
	}
 | 
						|
 | 
						|
	if p.ClassType == ClassUniversal {
 | 
						|
		p.Data.Write(content)
 | 
						|
		p.ByteValue = content
 | 
						|
 | 
						|
		switch p.Tag {
 | 
						|
		case TagEOC:
 | 
						|
		case TagBoolean:
 | 
						|
			val, _ := ParseInt64(content)
 | 
						|
 | 
						|
			p.Value = val != 0
 | 
						|
		case TagInteger:
 | 
						|
			p.Value, _ = ParseInt64(content)
 | 
						|
		case TagBitString:
 | 
						|
		case TagOctetString:
 | 
						|
			// the actual string encoding is not known here
 | 
						|
			// (e.g. for LDAP content is already an UTF8-encoded
 | 
						|
			// string). Return the data without further processing
 | 
						|
			p.Value = DecodeString(content)
 | 
						|
		case TagNULL:
 | 
						|
		case TagObjectIdentifier:
 | 
						|
		case TagObjectDescriptor:
 | 
						|
		case TagExternal:
 | 
						|
		case TagRealFloat:
 | 
						|
			p.Value, err = ParseReal(content)
 | 
						|
		case TagEnumerated:
 | 
						|
			p.Value, _ = ParseInt64(content)
 | 
						|
		case TagEmbeddedPDV:
 | 
						|
		case TagUTF8String:
 | 
						|
			val := DecodeString(content)
 | 
						|
			if !utf8.Valid([]byte(val)) {
 | 
						|
				err = errors.New("invalid UTF-8 string")
 | 
						|
			} else {
 | 
						|
				p.Value = val
 | 
						|
			}
 | 
						|
		case TagRelativeOID:
 | 
						|
		case TagSequence:
 | 
						|
		case TagSet:
 | 
						|
		case TagNumericString:
 | 
						|
		case TagPrintableString:
 | 
						|
			val := DecodeString(content)
 | 
						|
			if err = isPrintableString(val); err == nil {
 | 
						|
				p.Value = val
 | 
						|
			}
 | 
						|
		case TagT61String:
 | 
						|
		case TagVideotexString:
 | 
						|
		case TagIA5String:
 | 
						|
			val := DecodeString(content)
 | 
						|
			for i, c := range val {
 | 
						|
				if c >= 0x7F {
 | 
						|
					err = fmt.Errorf("invalid character for IA5String at pos %d: %c", i, c)
 | 
						|
					break
 | 
						|
				}
 | 
						|
			}
 | 
						|
			if err == nil {
 | 
						|
				p.Value = val
 | 
						|
			}
 | 
						|
		case TagUTCTime:
 | 
						|
		case TagGeneralizedTime:
 | 
						|
			p.Value, err = ParseGeneralizedTime(content)
 | 
						|
		case TagGraphicString:
 | 
						|
		case TagVisibleString:
 | 
						|
		case TagGeneralString:
 | 
						|
		case TagUniversalString:
 | 
						|
		case TagCharacterString:
 | 
						|
		case TagBMPString:
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		p.Data.Write(content)
 | 
						|
	}
 | 
						|
 | 
						|
	return p, read, err
 | 
						|
}
 | 
						|
 | 
						|
func isPrintableString(val string) error {
 | 
						|
	for i, c := range val {
 | 
						|
		switch {
 | 
						|
		case c >= 'a' && c <= 'z':
 | 
						|
		case c >= 'A' && c <= 'Z':
 | 
						|
		case c >= '0' && c <= '9':
 | 
						|
		default:
 | 
						|
			switch c {
 | 
						|
			case '\'', '(', ')', '+', ',', '-', '.', '=', '/', ':', '?', ' ':
 | 
						|
			default:
 | 
						|
				return fmt.Errorf("invalid character in position %d", i)
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (p *Packet) Bytes() []byte {
 | 
						|
	var out bytes.Buffer
 | 
						|
 | 
						|
	out.Write(encodeIdentifier(p.Identifier))
 | 
						|
	out.Write(encodeLength(p.Data.Len()))
 | 
						|
	out.Write(p.Data.Bytes())
 | 
						|
 | 
						|
	return out.Bytes()
 | 
						|
}
 | 
						|
 | 
						|
func (p *Packet) AppendChild(child *Packet) {
 | 
						|
	p.Data.Write(child.Bytes())
 | 
						|
	p.Children = append(p.Children, child)
 | 
						|
}
 | 
						|
 | 
						|
func Encode(classType Class, tagType Type, tag Tag, value interface{}, description string) *Packet {
 | 
						|
	p := new(Packet)
 | 
						|
 | 
						|
	p.ClassType = classType
 | 
						|
	p.TagType = tagType
 | 
						|
	p.Tag = tag
 | 
						|
	p.Data = new(bytes.Buffer)
 | 
						|
 | 
						|
	p.Children = make([]*Packet, 0, 2)
 | 
						|
 | 
						|
	p.Value = value
 | 
						|
	p.Description = description
 | 
						|
 | 
						|
	if value != nil {
 | 
						|
		v := reflect.ValueOf(value)
 | 
						|
 | 
						|
		if classType == ClassUniversal {
 | 
						|
			switch tag {
 | 
						|
			case TagOctetString:
 | 
						|
				sv, ok := v.Interface().(string)
 | 
						|
 | 
						|
				if ok {
 | 
						|
					p.Data.Write([]byte(sv))
 | 
						|
				}
 | 
						|
			case TagEnumerated:
 | 
						|
				bv, ok := v.Interface().([]byte)
 | 
						|
				if ok {
 | 
						|
					p.Data.Write(bv)
 | 
						|
				}
 | 
						|
			case TagEmbeddedPDV:
 | 
						|
				bv, ok := v.Interface().([]byte)
 | 
						|
				if ok {
 | 
						|
					p.Data.Write(bv)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else if classType == ClassContext {
 | 
						|
			switch tag {
 | 
						|
			case TagEnumerated:
 | 
						|
				bv, ok := v.Interface().([]byte)
 | 
						|
				if ok {
 | 
						|
					p.Data.Write(bv)
 | 
						|
				}
 | 
						|
			case TagEmbeddedPDV:
 | 
						|
				bv, ok := v.Interface().([]byte)
 | 
						|
				if ok {
 | 
						|
					p.Data.Write(bv)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
func NewSequence(description string) *Packet {
 | 
						|
	return Encode(ClassUniversal, TypeConstructed, TagSequence, nil, description)
 | 
						|
}
 | 
						|
 | 
						|
func NewBoolean(classType Class, tagType Type, tag Tag, value bool, description string) *Packet {
 | 
						|
	intValue := int64(0)
 | 
						|
 | 
						|
	if value {
 | 
						|
		intValue = 1
 | 
						|
	}
 | 
						|
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
 | 
						|
	p.Value = value
 | 
						|
	p.Data.Write(encodeInteger(intValue))
 | 
						|
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
// NewLDAPBoolean returns a RFC 4511-compliant Boolean packet.
 | 
						|
func NewLDAPBoolean(classType Class, tagType Type, tag Tag, value bool, description string) *Packet {
 | 
						|
	intValue := int64(0)
 | 
						|
 | 
						|
	if value {
 | 
						|
		intValue = 255
 | 
						|
	}
 | 
						|
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
 | 
						|
	p.Value = value
 | 
						|
	p.Data.Write(encodeInteger(intValue))
 | 
						|
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
func NewInteger(classType Class, tagType Type, tag Tag, value interface{}, description string) *Packet {
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
 | 
						|
	p.Value = value
 | 
						|
	switch v := value.(type) {
 | 
						|
	case int:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case uint:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case int64:
 | 
						|
		p.Data.Write(encodeInteger(v))
 | 
						|
	case uint64:
 | 
						|
		// TODO : check range or add encodeUInt...
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case int32:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case uint32:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case int16:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case uint16:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case int8:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	case uint8:
 | 
						|
		p.Data.Write(encodeInteger(int64(v)))
 | 
						|
	default:
 | 
						|
		// TODO : add support for big.Int ?
 | 
						|
		panic(fmt.Sprintf("Invalid type %T, expected {u|}int{64|32|16|8}", v))
 | 
						|
	}
 | 
						|
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
func NewString(classType Class, tagType Type, tag Tag, value, description string) *Packet {
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
 | 
						|
	p.Value = value
 | 
						|
	p.Data.Write([]byte(value))
 | 
						|
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
func NewGeneralizedTime(classType Class, tagType Type, tag Tag, value time.Time, description string) *Packet {
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
	var s string
 | 
						|
	if value.Nanosecond() != 0 {
 | 
						|
		s = value.Format(`20060102150405.000000000Z`)
 | 
						|
	} else {
 | 
						|
		s = value.Format(`20060102150405Z`)
 | 
						|
	}
 | 
						|
	p.Value = s
 | 
						|
	p.Data.Write([]byte(s))
 | 
						|
	return p
 | 
						|
}
 | 
						|
 | 
						|
func NewReal(classType Class, tagType Type, tag Tag, value interface{}, description string) *Packet {
 | 
						|
	p := Encode(classType, tagType, tag, nil, description)
 | 
						|
 | 
						|
	switch v := value.(type) {
 | 
						|
	case float64:
 | 
						|
		p.Data.Write(encodeFloat(v))
 | 
						|
	case float32:
 | 
						|
		p.Data.Write(encodeFloat(float64(v)))
 | 
						|
	default:
 | 
						|
		panic(fmt.Sprintf("Invalid type %T, expected float{64|32}", v))
 | 
						|
	}
 | 
						|
	return p
 | 
						|
}
 |