112 lines
3.8 KiB
Go
112 lines
3.8 KiB
Go
// Copyright [2019] LinkedIn Corp. Licensed under the Apache License, Version
|
|
// 2.0 (the "License"); you may not use this file except in compliance with the
|
|
// License. You may obtain a copy of the License at
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
package goavro
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
)
|
|
|
|
// Fixed does not have child objects, therefore whatever namespace it defines is
|
|
// just to store its name in the symbol table.
|
|
func makeFixedCodec(st map[string]*Codec, enclosingNamespace string, schemaMap map[string]interface{}) (*Codec, error) {
|
|
c, err := registerNewCodec(st, schemaMap, enclosingNamespace)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("Fixed ought to have valid name: %s", err)
|
|
}
|
|
size, err := sizeFromSchemaMap(c.typeName, schemaMap)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
c.nativeFromBinary = func(buf []byte) (interface{}, []byte, error) {
|
|
if buflen := uint(len(buf)); size > buflen {
|
|
return nil, nil, fmt.Errorf("cannot decode binary fixed %q: schema size exceeds remaining buffer size: %d > %d (short buffer)", c.typeName, size, buflen)
|
|
}
|
|
return buf[:size], buf[size:], nil
|
|
}
|
|
|
|
c.binaryFromNative = func(buf []byte, datum interface{}) ([]byte, error) {
|
|
var someBytes []byte
|
|
switch d := datum.(type) {
|
|
case []byte:
|
|
someBytes = d
|
|
case string:
|
|
someBytes = []byte(d)
|
|
default:
|
|
return nil, fmt.Errorf("cannot encode binary fixed %q: expected []byte or string; received: %T", c.typeName, datum)
|
|
}
|
|
if count := uint(len(someBytes)); count != size {
|
|
return nil, fmt.Errorf("cannot encode binary fixed %q: datum size ought to equal schema size: %d != %d", c.typeName, count, size)
|
|
}
|
|
return append(buf, someBytes...), nil
|
|
}
|
|
|
|
c.nativeFromTextual = func(buf []byte) (interface{}, []byte, error) {
|
|
if buflen := uint(len(buf)); size > buflen {
|
|
return nil, nil, fmt.Errorf("cannot decode textual fixed %q: schema size exceeds remaining buffer size: %d > %d (short buffer)", c.typeName, size, buflen)
|
|
}
|
|
var datum interface{}
|
|
var err error
|
|
datum, buf, err = bytesNativeFromTextual(buf)
|
|
if err != nil {
|
|
return nil, buf, err
|
|
}
|
|
datumBytes := datum.([]byte)
|
|
if count := uint(len(datumBytes)); count != size {
|
|
return nil, nil, fmt.Errorf("cannot decode textual fixed %q: datum size ought to equal schema size: %d != %d", c.typeName, count, size)
|
|
}
|
|
return datum, buf, err
|
|
}
|
|
|
|
c.textualFromNative = func(buf []byte, datum interface{}) ([]byte, error) {
|
|
var someBytes []byte
|
|
switch d := datum.(type) {
|
|
case []byte:
|
|
someBytes = d
|
|
case string:
|
|
someBytes = []byte(d)
|
|
default:
|
|
return nil, fmt.Errorf("cannot encode textual fixed %q: expected []byte or string; received: %T", c.typeName, datum)
|
|
}
|
|
if count := uint(len(someBytes)); count != size {
|
|
return nil, fmt.Errorf("cannot encode textual fixed %q: datum size ought to equal schema size: %d != %d", c.typeName, count, size)
|
|
}
|
|
return bytesTextualFromNative(buf, someBytes)
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
func sizeFromSchemaMap(typeName *name, schemaMap map[string]interface{}) (uint, error) {
|
|
// Fixed type must have size
|
|
sizeRaw, ok := schemaMap["size"]
|
|
if !ok {
|
|
return 0, fmt.Errorf("Fixed %q ought to have size key", typeName)
|
|
}
|
|
var size uint
|
|
switch val := sizeRaw.(type) {
|
|
case string:
|
|
s, err := strconv.ParseUint(val, 10, 0)
|
|
if err != nil {
|
|
return 0, fmt.Errorf("Fixed %q size ought to be number greater than zero: %v", typeName, sizeRaw)
|
|
}
|
|
size = uint(s)
|
|
case float64:
|
|
if val <= 0 {
|
|
return 0, fmt.Errorf("Fixed %q size ought to be number greater than zero: %v", typeName, sizeRaw)
|
|
}
|
|
size = uint(val)
|
|
default:
|
|
return 0, fmt.Errorf("Fixed %q size ought to be number greater than zero: %v", typeName, sizeRaw)
|
|
}
|
|
return size, nil
|
|
}
|