382 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			382 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
package gomemcached
 | 
						|
 | 
						|
import (
 | 
						|
	"encoding/binary"
 | 
						|
	"fmt"
 | 
						|
)
 | 
						|
 | 
						|
type FrameObjType int
 | 
						|
 | 
						|
const (
 | 
						|
	FrameBarrier     FrameObjType = iota
 | 
						|
	FrameDurability  FrameObjType = iota
 | 
						|
	FrameDcpStreamId FrameObjType = iota
 | 
						|
	FrameOpenTracing FrameObjType = iota
 | 
						|
)
 | 
						|
 | 
						|
type FrameInfo struct {
 | 
						|
	ObjId   FrameObjType
 | 
						|
	ObjLen  int
 | 
						|
	ObjData []byte
 | 
						|
}
 | 
						|
 | 
						|
var ErrorInvalidOp error = fmt.Errorf("Specified method is not applicable")
 | 
						|
var ErrorObjLenNotMatch error = fmt.Errorf("Object length does not match data")
 | 
						|
 | 
						|
func (f *FrameInfo) Validate() error {
 | 
						|
	switch f.ObjId {
 | 
						|
	case FrameBarrier:
 | 
						|
		if f.ObjLen != 0 {
 | 
						|
			return fmt.Errorf("Invalid FrameBarrier - length is %v\n", f.ObjLen)
 | 
						|
		} else if f.ObjLen != len(f.ObjData) {
 | 
						|
			return ErrorObjLenNotMatch
 | 
						|
		}
 | 
						|
	case FrameDurability:
 | 
						|
		if f.ObjLen != 1 && f.ObjLen != 3 {
 | 
						|
			return fmt.Errorf("Invalid FrameDurability - length is %v\n", f.ObjLen)
 | 
						|
		} else if f.ObjLen != len(f.ObjData) {
 | 
						|
			return ErrorObjLenNotMatch
 | 
						|
		}
 | 
						|
	case FrameDcpStreamId:
 | 
						|
		if f.ObjLen != 2 {
 | 
						|
			return fmt.Errorf("Invalid FrameDcpStreamId - length is %v\n", f.ObjLen)
 | 
						|
		} else if f.ObjLen != len(f.ObjData) {
 | 
						|
			return ErrorObjLenNotMatch
 | 
						|
		}
 | 
						|
	case FrameOpenTracing:
 | 
						|
		if f.ObjLen == 0 {
 | 
						|
			return fmt.Errorf("Invalid FrameOpenTracing - length must be > 0")
 | 
						|
		} else if f.ObjLen != len(f.ObjData) {
 | 
						|
			return ErrorObjLenNotMatch
 | 
						|
		}
 | 
						|
	default:
 | 
						|
		return fmt.Errorf("Unknown FrameInfo type")
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (f *FrameInfo) GetStreamId() (uint16, error) {
 | 
						|
	if f.ObjId != FrameDcpStreamId {
 | 
						|
		return 0, ErrorInvalidOp
 | 
						|
	}
 | 
						|
 | 
						|
	var output uint16
 | 
						|
	output = uint16(f.ObjData[0])
 | 
						|
	output = output << 8
 | 
						|
	output |= uint16(f.ObjData[1])
 | 
						|
	return output, nil
 | 
						|
}
 | 
						|
 | 
						|
type DurabilityLvl uint8
 | 
						|
 | 
						|
const (
 | 
						|
	DuraInvalid                    DurabilityLvl = iota // Not used (0x0)
 | 
						|
	DuraMajority                   DurabilityLvl = iota // (0x01)
 | 
						|
	DuraMajorityAndPersistOnMaster DurabilityLvl = iota // (0x02)
 | 
						|
	DuraPersistToMajority          DurabilityLvl = iota // (0x03)
 | 
						|
)
 | 
						|
 | 
						|
func (f *FrameInfo) GetDurabilityRequirements() (lvl DurabilityLvl, timeoutProvided bool, timeoutMs uint16, err error) {
 | 
						|
	if f.ObjId != FrameDurability {
 | 
						|
		err = ErrorInvalidOp
 | 
						|
		return
 | 
						|
	}
 | 
						|
	if f.ObjLen != 1 && f.ObjLen != 3 {
 | 
						|
		err = ErrorObjLenNotMatch
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	lvl = DurabilityLvl(uint8(f.ObjData[0]))
 | 
						|
 | 
						|
	if f.ObjLen == 3 {
 | 
						|
		timeoutProvided = true
 | 
						|
		timeoutMs = binary.BigEndian.Uint16(f.ObjData[1:2])
 | 
						|
	}
 | 
						|
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func incrementMarker(bitsToBeIncremented, byteIncrementCnt *int, framingElen, curObjIdx int) (int, error) {
 | 
						|
	for *bitsToBeIncremented >= 8 {
 | 
						|
		*byteIncrementCnt++
 | 
						|
		*bitsToBeIncremented -= 8
 | 
						|
	}
 | 
						|
	marker := curObjIdx + *byteIncrementCnt
 | 
						|
	if marker > framingElen {
 | 
						|
		return -1, fmt.Errorf("Out of bounds")
 | 
						|
	}
 | 
						|
	return marker, nil
 | 
						|
}
 | 
						|
 | 
						|
// Right now, halfByteRemaining will always be false, because ObjID and Len haven't gotten that large yet
 | 
						|
func (f *FrameInfo) Bytes() (output []byte, halfByteRemaining bool) {
 | 
						|
	// ObjIdentifier - 4 bits + ObjLength - 4 bits
 | 
						|
	var idAndLen uint8
 | 
						|
	idAndLen |= uint8(f.ObjId) << 4
 | 
						|
	idAndLen |= uint8(f.ObjLen)
 | 
						|
	output = append(output, byte(idAndLen))
 | 
						|
 | 
						|
	// Rest is Data
 | 
						|
	output = append(output, f.ObjData...)
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func parseFrameInfoObjects(buf []byte, framingElen int) (objs []FrameInfo, err error, halfByteRemaining bool) {
 | 
						|
	var curObjIdx int
 | 
						|
	var byteIncrementCnt int
 | 
						|
	var bitsToBeIncremented int
 | 
						|
	var marker int
 | 
						|
 | 
						|
	// Parse frameInfo objects
 | 
						|
	for curObjIdx = 0; curObjIdx < framingElen; curObjIdx += byteIncrementCnt {
 | 
						|
		byteIncrementCnt = 0
 | 
						|
		var oneFrameObj FrameInfo
 | 
						|
 | 
						|
		// First get the objId
 | 
						|
		// -------------------------
 | 
						|
		var objId int
 | 
						|
		var objHeader uint8 = buf[curObjIdx]
 | 
						|
		var objIdentifierRaw uint8
 | 
						|
		if bitsToBeIncremented == 0 {
 | 
						|
			// ObjHeader
 | 
						|
			// 0 1 2 3 4 5 6 7
 | 
						|
			// ^-----^
 | 
						|
			// ObjIdentifierRaw
 | 
						|
			objIdentifierRaw = (objHeader & 0xf0) >> 4
 | 
						|
		} else {
 | 
						|
			// ObjHeader
 | 
						|
			// 0 1 2 3 4 5 6 7
 | 
						|
			//         ^-----^
 | 
						|
			//           ObjIdentifierRaw
 | 
						|
			objIdentifierRaw = (objHeader & 0x0f)
 | 
						|
		}
 | 
						|
		bitsToBeIncremented += 4
 | 
						|
 | 
						|
		marker, err = incrementMarker(&bitsToBeIncremented, &byteIncrementCnt, framingElen, curObjIdx)
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
 | 
						|
		// Value is 0-14
 | 
						|
		objId = int(objIdentifierRaw & 0xe)
 | 
						|
		// If bit 15 is set, ID is 15 + value of next byte
 | 
						|
		if objIdentifierRaw&0x1 > 0 {
 | 
						|
			if bitsToBeIncremented > 0 {
 | 
						|
				// ObjHeader
 | 
						|
				// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
 | 
						|
				// ^-----^ ^---------------^
 | 
						|
				// ObjId1    Extension
 | 
						|
				// ^ marker
 | 
						|
				buffer := uint16(buf[marker])
 | 
						|
				buffer = buffer << 8
 | 
						|
				buffer |= uint16(buf[marker+1])
 | 
						|
				var extension uint8 = uint8(buffer & 0xff0 >> 4)
 | 
						|
				objId += int(extension)
 | 
						|
			} else {
 | 
						|
				// ObjHeader
 | 
						|
				// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
 | 
						|
				//         ^-----^ ^-------------------^
 | 
						|
				//          ObjId1    extension
 | 
						|
				//                 ^ marker
 | 
						|
				var extension uint8 = uint8(buf[marker])
 | 
						|
				objId += int(extension)
 | 
						|
			}
 | 
						|
			bitsToBeIncremented += 8
 | 
						|
		}
 | 
						|
 | 
						|
		marker, err = incrementMarker(&bitsToBeIncremented, &byteIncrementCnt, framingElen, curObjIdx)
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
		oneFrameObj.ObjId = FrameObjType(objId)
 | 
						|
 | 
						|
		// Then get the obj length
 | 
						|
		// -------------------------
 | 
						|
		var objLenRaw uint8
 | 
						|
		var objLen int
 | 
						|
		if bitsToBeIncremented > 0 {
 | 
						|
			// ObjHeader
 | 
						|
			// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
 | 
						|
			//                 ^         ^---------^
 | 
						|
			//                 marker       objLen
 | 
						|
			objLenRaw = uint8(buf[marker]) & 0x0f
 | 
						|
		} else {
 | 
						|
			// ObjHeader
 | 
						|
			// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
 | 
						|
			//                                        ^--------^
 | 
						|
			//                                          objLen
 | 
						|
			//                                        ^ marker
 | 
						|
			objLenRaw = uint8(buf[marker]) & 0xf0 >> 4
 | 
						|
		}
 | 
						|
		bitsToBeIncremented += 4
 | 
						|
 | 
						|
		marker, err = incrementMarker(&bitsToBeIncremented, &byteIncrementCnt, framingElen, curObjIdx)
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
 | 
						|
		// Length is 0-14
 | 
						|
		objLen = int(objLenRaw & 0xe)
 | 
						|
		// If bit 15 is set, lenghth is 15 + value of next byte
 | 
						|
		if objLenRaw&0x1 > 0 {
 | 
						|
			if bitsToBeIncremented == 0 {
 | 
						|
				// ObjHeader
 | 
						|
				// 12 13 14 15 16 17 18 19 20 21 22 23
 | 
						|
				// ^---------^ ^--------------------^
 | 
						|
				//   objLen        extension
 | 
						|
				//             ^ marker
 | 
						|
				var extension uint8 = uint8(buf[marker])
 | 
						|
				objLen += int(extension)
 | 
						|
			} else {
 | 
						|
				// ObjHeader
 | 
						|
				// 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
 | 
						|
				// ^--------^  ^---------------------^
 | 
						|
				//  objLen          extension
 | 
						|
				// ^ marker				var buffer uint16
 | 
						|
				buffer := uint16(buf[marker])
 | 
						|
				buffer = buffer << 8
 | 
						|
				buffer |= uint16(buf[marker+1])
 | 
						|
				var extension uint8 = uint8(buffer & 0xff0 >> 4)
 | 
						|
				objLen += int(extension)
 | 
						|
			}
 | 
						|
			bitsToBeIncremented += 8
 | 
						|
		}
 | 
						|
 | 
						|
		marker, err = incrementMarker(&bitsToBeIncremented, &byteIncrementCnt, framingElen, curObjIdx)
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
		oneFrameObj.ObjLen = objLen
 | 
						|
 | 
						|
		// The rest is N-bytes of data based on the length
 | 
						|
		if bitsToBeIncremented == 0 {
 | 
						|
			// No weird alignment needed
 | 
						|
			oneFrameObj.ObjData = buf[marker : marker+objLen]
 | 
						|
		} else {
 | 
						|
			// 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
 | 
						|
			// ^--------^  ^---------------------^ ^--------->
 | 
						|
			//  objLen          extension            data
 | 
						|
			//                          ^ marker
 | 
						|
			oneFrameObj.ObjData = ShiftByteSliceLeft4Bits(buf[marker : marker+objLen+1])
 | 
						|
		}
 | 
						|
		err = oneFrameObj.Validate()
 | 
						|
		if err != nil {
 | 
						|
			return
 | 
						|
		}
 | 
						|
		objs = append(objs, oneFrameObj)
 | 
						|
 | 
						|
		bitsToBeIncremented += 8 * objLen
 | 
						|
		marker, err = incrementMarker(&bitsToBeIncremented, &byteIncrementCnt, framingElen, curObjIdx)
 | 
						|
	}
 | 
						|
 | 
						|
	if bitsToBeIncremented > 0 {
 | 
						|
		halfByteRemaining = true
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func ShiftByteSliceLeft4Bits(slice []byte) (replacement []byte) {
 | 
						|
	var buffer uint16
 | 
						|
	var i int
 | 
						|
	sliceLen := len(slice)
 | 
						|
 | 
						|
	if sliceLen < 2 {
 | 
						|
		// Let's not shift less than 16 bits
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	replacement = make([]byte, sliceLen, cap(slice))
 | 
						|
 | 
						|
	for i = 0; i < sliceLen-1; i++ {
 | 
						|
		// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
 | 
						|
		// ^-----^ ^---------------^ ^-----------
 | 
						|
		// garbage   data byte 0       data byte 1
 | 
						|
		buffer = uint16(slice[i])
 | 
						|
		buffer = buffer << 8
 | 
						|
		buffer |= uint16(slice[i+1])
 | 
						|
		replacement[i] = uint8(buffer & 0xff0 >> 4)
 | 
						|
	}
 | 
						|
 | 
						|
	if i < sliceLen {
 | 
						|
		lastByte := slice[sliceLen-1]
 | 
						|
		lastByte = lastByte << 4
 | 
						|
		replacement[i] = lastByte
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
// The following is used to theoretically support frameInfo ObjID extensions
 | 
						|
// for completeness, but they are not very efficient though
 | 
						|
func ShiftByteSliceRight4Bits(slice []byte) (replacement []byte) {
 | 
						|
	var buffer uint16
 | 
						|
	var i int
 | 
						|
	var leftovers uint8 // 4 bits only
 | 
						|
	var replacementUnit uint16
 | 
						|
	var first bool = true
 | 
						|
	var firstLeftovers uint8
 | 
						|
	var lastLeftovers uint8
 | 
						|
	sliceLen := len(slice)
 | 
						|
 | 
						|
	if sliceLen < 2 {
 | 
						|
		// Let's not shift less than 16 bits
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	if slice[sliceLen-1]&0xf == 0 {
 | 
						|
		replacement = make([]byte, sliceLen, cap(slice))
 | 
						|
	} else {
 | 
						|
		replacement = make([]byte, sliceLen+1, cap(slice)+1)
 | 
						|
	}
 | 
						|
 | 
						|
	for i = 0; i < sliceLen-1; i++ {
 | 
						|
		buffer = binary.BigEndian.Uint16(slice[i : i+2])
 | 
						|
		// (buffer)
 | 
						|
		// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
 | 
						|
		// ^-------------^ ^-------------------^
 | 
						|
		//     data byte 0        data byte 1
 | 
						|
		//
 | 
						|
		// into
 | 
						|
		//
 | 
						|
		// 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 | 
						|
		// ^-----^ ^---------------^ ^--------------------^ ^----------^
 | 
						|
		// zeroes   data byte 0      data byte 1              zeroes
 | 
						|
 | 
						|
		if first {
 | 
						|
			// The leftover OR'ing will overwrite the first 4 bits of data byte 0. Save them
 | 
						|
			firstLeftovers = uint8(buffer & 0xf000 >> 12)
 | 
						|
			first = false
 | 
						|
		}
 | 
						|
		replacementUnit = 0
 | 
						|
		replacementUnit |= uint16(leftovers) << 12
 | 
						|
		replacementUnit |= (buffer & 0xff00) >> 4 // data byte 0
 | 
						|
		replacementUnit |= buffer & 0xff >> 4     // data byte 1 first 4 bits
 | 
						|
		lastLeftovers = uint8(buffer&0xf) << 4
 | 
						|
 | 
						|
		replacement[i+1] = byte(replacementUnit)
 | 
						|
 | 
						|
		leftovers = uint8((buffer & 0x000f) << 4)
 | 
						|
	}
 | 
						|
 | 
						|
	replacement[0] = byte(uint8(replacement[0]) | firstLeftovers)
 | 
						|
	if lastLeftovers > 0 {
 | 
						|
		replacement[sliceLen] = byte(lastLeftovers)
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 | 
						|
 | 
						|
func Merge2HalfByteSlices(src1, src2 []byte) (output []byte) {
 | 
						|
	src1Len := len(src1)
 | 
						|
	src2Len := len(src2)
 | 
						|
	output = make([]byte, src1Len+src2Len-1)
 | 
						|
 | 
						|
	var mergeByte uint8 = src1[src1Len-1]
 | 
						|
	mergeByte |= uint8(src2[0])
 | 
						|
 | 
						|
	copy(output, src1)
 | 
						|
	copy(output[src1Len:], src2[1:])
 | 
						|
 | 
						|
	output[src1Len-1] = byte(mergeByte)
 | 
						|
 | 
						|
	return
 | 
						|
}
 |