1

TLDR;如何在 buffalo 框架中在具有外键(或一般的数据库关系)的模型上创建 REST api?

我是绝对的初学者,我正在尝试使用 buffalo 框架编写一个 RESTFul 服务,遵循官方网站上给出的示例。我能够在没有数据库关系的模型上创建 RESTful api。但是当我遇到模型有外键的情况时,我被卡住了。我无法在网上找到任何文档或参考资料。我对 Go 的概念也很薄弱,您也可以在这些方面教育我。

型号:(参考:https ://gobuffalo.io/en/docs/db/relations#example

type Composer struct {
    ID              uuid.UUID `json:"id" db:"id"`
    Name            string    `json:"name" db:"name"`
    Description     string    `json:"description" db:"description"`
    CreatedAt       time.Time `json:"created_at" db:"created_at"`
    UpdatedAt       time.Time `json:"updated_at" db:"updated_at"`

}


type Track struct {
    ID          uuid.UUID           `json:"id" db:"id"`
    Title       string              `json:"title" db:"title"`
    Description string              `json:"description" db:"description"`
    Composer    Composer            `has_one:"composer" fk_id:"id"`
    CreatedAt   time.Time           `json:"created_at" db:"created_at"`
    UpdatedAt   time.Time           `json:"updated_at" db:"updated_at"`
}

资源:(参考:https ://gobuffalo.io/en/docs/resources )

type TrackResource struct {
    buffalo.Resource
}


func (v TrackResource) List(c buffalo.Context) error {
    tx, ok := c.Value("tx").(*pop.Connection)
    if !ok {
        return errors.WithStack(errors.New("no transaction found"))
    }

    pieces := &models.Tracks{}

    q := tx.PaginateFromParams(c.Params())

    if err := q.All(pieces); err != nil {
        return errors.WithStack(err)
    }

    c.Set("pagination", q.Paginator)

    return c.Render(200, r.JSON(pieces))
}


func (v TrackResource) Show(c buffalo.Context) error {
    tx, ok := c.Value("tx").(*pop.Connection)
    if !ok {
        return errors.WithStack(errors.New("no transaction found"))
    }

    piece := &models.Track{}

    if err := tx.Find(piece, c.Param("track_id")); err != nil {
        return c.Render(404, r.JSON(err))
    }

    return c.Render(200, r.JSON(piece))
}


func (v TrackResource) Create(c buffalo.Context) error {

    piece := &models.Track{}

    if err := c.Bind(piece); err != nil {
        return errors.WithStack(err)
    }

    tx, ok := c.Value("tx").(*pop.Connection)
    if !ok {
        return errors.WithStack(errors.New("no transaction found"))
    }

    verrs, err := piece.Create(tx)
    if err != nil {
        return errors.WithStack(err)
    }

    if verrs.HasAny() {
        return c.Render(422, r.JSON(verrs))
    }

    return c.Render(201, r.Auto(c, piece))
}


func (v TrackResource) Update(c buffalo.Context) error {

    tx, ok := c.Value("tx").(*pop.Connection)
    if !ok {
        return errors.WithStack(errors.New("no transaction found"))
    }

    piece := &models.Track{}

    if err := tx.Find(piece, c.Param("track_id")); err != nil {
        return c.Error(404, err)
    }

    if err := c.Bind(piece); err != nil {
        return errors.WithStack(err)
    }

    verrs, err := piece.Update(tx)
    if err != nil {
        return errors.WithStack(err)
    }

    if verrs.HasAny() {
        return c.Render(422, r.JSON(verrs))
    }

    return c.Render(200, r.JSON(piece))
}


func (v TrackResource) Destroy(c buffalo.Context) error {

    tx, ok := c.Value("tx").(*pop.Connection)
    if !ok {
        return errors.WithStack(errors.New("no transaction found"))
    }

    piece := &models.Track{}

    if err := tx.Find(piece, c.Param("track_id")); err != nil {
        return c.Error(404, err)
    }

    if err := tx.Destroy(piece); err != nil {
        return errors.WithStack(err)
    }

    return c.Render(200, r.JSON(piece))
}

当我尝试创建轨道时。我收到错误:

json: cannot unmarshal string into Go struct field Track.Composer of type models.Composer
gitlab.com/****/****/actions.TrackResource.Create

在此处输入图像描述

请帮忙。

4

2 回答 2

2

您正在尝试将Track包含 a 的 aComposer与其 ID 绑定,但Composer被定义为结构。

要使其工作,您需要实现Unmarshaler接口并定义如何将此 ID 转换为Composer您想要的结构。

于 2018-07-17T09:43:55.563 回答
0

您应该在结构中添加一个ComposerID字段Track。然后你可以使用Eageror Load,pop 会Composer为你加载相关的对象。这显示在您链接到的页面上以供您的模型参考。

于 2018-07-17T16:10:27.317 回答