登录
首页 >  Golang >  Go问答

使用 mongo-driver 自定义 BSON 编组和解组

来源:stackoverflow

时间:2024-04-30 22:03:25 148浏览 收藏

来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《使用 mongo-driver 自定义 BSON 编组和解组》,介绍一下,希望对大家的知识积累有所帮助,助力实战开发!

问题内容

我有一个如下所示的结构字段。我还将相同结构的原始 protobuf 存储在数据库中。现在每次获取或保存数据到 mongo 时。当我想保存到数据库时,我必须从原型更新 reallybigraw,当我想保存到数据库时,我必须将 reallybigraw 解组到 reallybigobj 以给出响应。有没有办法可以实现一些接口或提供一些回调函数,以便 mongo 驱动程序在保存或从数据库获取数据之前自动执行此操作。

另外,我使用的是官方 golang mongo 驱动程序而不是 mgo,我已经阅读了一些可以在 mgo golang 库中完成的答案。

import (
    "github.com/golang/protobuf/jsonpb"
    "go.mongodb.org/mongo-driver/bson"
    "go.mongodb.org/mongo-driver/bson/primitive"

    proto "github.com/dinesh/api/go"
)

type ReallyBig struct {
    ID           string                 `bson:"_id,omitempty"`
    DraftID      string                 `bson:"draft_id,omitempty"`
// Marshaled ReallyBigObj proto to map[string]interface{} stored in DB
    ReallyBigRaw map[string]interface{} `bson:"raw,omitempty"`
    ReallyBigObj *proto.ReallyBig       `bson:"-"`
    CreatedAt    primitive.DateTime     `bson:"created_at,omitempty"`
    UpdatedAt    primitive.DateTime     `bson:"updated_at,omitempty"`
}

func (r *ReallyBig) GetProto() (*proto.ReallyBig, error) {
    if r.ReallyBigObj != nil {
        return r.ReallyBigObj, nil
    }
    Obj, err := getProto(r.ReallyBigRaw)
    if err != nil {
        return nil, err
    }
    r.ReallyBigObj = Obj
    return r.ReallyBigObj, nil
}

func getRaw(r *proto.ReallyBig) (map[string]interface{}, error) {
    m := jsonpb.Marshaler{}
    b := bytes.NewBuffer([]byte{})

    // marshals proto to json format
    err := m.Marshal(b, r)
    if err != nil {
        return nil, err
    }
    var raw map[string]interface{}
    // unmarshal the raw data to an interface
    err = json.Unmarshal(b.Bytes(), &raw)
    if err != nil {
        return nil, err
    }
    return raw, nil
}

func getProto(raw map[string]interface{}) (*proto.ReallyBig, error) {
    b, err := json.Marshal(raw)
    if err != nil {
        return nil, err
    }
    u := jsonpb.Unmarshaler{}
    var reallyBigProto proto.ReallyBig
    err = u.Unmarshal(bytes.NewReader(b), &recipeProto)
    if err != nil {
        return nil, err
    }
    return &reallyBigProto, nil
}

解决方案


我实现了 marshalerunmarshaler 接口。由于 mongo 驱动程序调用 marshalbsonunmarshalbson (如果类型实现 marshalerunmarshaler),我们也会陷入无限循环。为了避免这种情况,我们创建了 type 的别名。 golang 中的别名仅继承字段而不继承方法,因此我们最终调用正常的 bson.marshalbson.unmarshal

func (r *ReallyBig) MarshalBSON() ([]byte, error) {
    type ReallyBigAlias ReallyBig
    reallyBigRaw, err := getRaw(r.ReallyBigObj)
    if err != nil {
        return nil, err
    }
    r.ReallyBigRaw = reallyBigRaw
    return bson.Marshal((*ReallyBigAlias)(r))
}

func (r *ReallyBig) UnmarshalBSON(data []byte) error {
    type ReallyBigAlias ReallyBig
    err := bson.Unmarshal(data, (*ReallyBigAlias)(r))
    if err != nil {
        return err
    }
    reallyBigProto, err := getProto(r.ReallyBigRaw)
    if err != nil {
        return err
    }
    r.ReallyBigObj = reallyBigProto
    return nil
}

好了,本文到此结束,带大家了解了《使用 mongo-driver 自定义 BSON 编组和解组》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>