app/pubsub-integration/avro/avro.go (64 lines of code) (raw):

// Copyright 2023 Google LLC // // 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 // // https://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. // See the License for the specific language governing permissions and // limitations under the License. // Package avro provides API for handling avro package avro import ( "fmt" "log" "os" "reflect" "github.com/linkedin/goavro/v2" ) // NewCodedecFromFile creates codec from avro schema file func NewCodedecFromFile(path string) (*goavro.Codec, error) { schema, err := os.ReadFile(path) if err != nil { return nil, err } codec, err := goavro.NewCodec(string(schema)) if err != nil { return nil, err } return codec, nil } // EncodeToJSON encodes data to JSON using given avro codec func EncodeToJSON(codec *goavro.Codec, data map[string]interface{}) ([]byte, error) { json, err := codec.TextualFromNative(nil, data) if err != nil { log.Println("fail to encode data=", data, "err=", err) } return json, err } // DecodeFromJSON decodes json using given avro codec func DecodeFromJSON(codec *goavro.Codec, json []byte) (map[string]interface{}, error) { native, _, err := codec.NativeFromTextual(json) if err != nil { log.Println("fail to decode json=", json, "err=", err) return nil, err } data, ok := native.(map[string]interface{}) if !ok { log.Println("fail to decode json=", json, "err=", err) return nil, err } return data, err } // GetValue gets value from given map using given key and converts to given type func GetValue[T any](data map[string]interface{}, key string, valueType T) (T, error) { val, ok := data[key] if !ok { return valueType, fmt.Errorf("the key %s does not exist", key) } return toType(val, valueType) } func toType[T any](data interface{}, valueType T) (T, error) { value, ok := data.(T) if !ok { return valueType, fmt.Errorf("the type of %v is %v, but %v is expected", data, reflect.TypeOf(data), reflect.TypeOf(valueType)) } return value, nil } // GetFloatTypeValue gets float32 value from given map using given key func GetFloatTypeValue(data map[string]interface{}, key string) (float32, error) { valueMap, err := GetValue(data, key, map[string]interface{}{}) if err != nil { return 0, err } value, err := GetValue(valueMap, "float", float32(0)) if err != nil { return 0, err } return value, nil }