登录
首页 >  Golang >  Go问答

Postman 成功发送到 Gin,但我的 Android 应用程序发送失败

来源:stackoverflow

时间:2024-02-06 09:00:23 341浏览 收藏

本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《Postman 成功发送到 Gin,但我的 Android 应用程序发送失败》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~

问题内容

我不太确定哪一方是问题的原因,但似乎 gin 收到了请求,但不允许来自 android 应用程序的请求。我尝试 post 的端点与具有类型字段的整体模型相关,并且该类型字段“告诉”gin 这是必需的字段(我的工作流程对于我的系统的这一部分来说有点复杂)。在 android 方面,我有一个该整体模型的对应模型,因为我将它用于应用程序中的某些内容。以前,我只需要从服务器获取与整体模型相关的数据。现在,我正在为其实现 post 。服务器模型的 android 副本如下所示:

@entity(tablename = "thing", indices = [index(value = ["id"], unique = true)])
data class thing(
    @primarykey(autogenerate = false) @columninfo(name = "id") var id: string,
    @serializedname("createdat") @columninfo(name = "created_at") var created_at: string,
    @columninfo(name = "name") var name: string,
    @columninfo(name = "type_id") var type_id: string,
    @columninfo("type") var type: thingtype? = null,
    @columninfo(
        name = "some_field_1",
        defaultvalue = "0"
    ) var some_field_1: float = 0.0f,
    @columninfo(
        name = "some_field_2",
        defaultvalue = "0"
    ) var some_field_2: float = 0.0f,
    @columninfo(
        name = "some_field_3",
        defaultvalue = "0"
    ) var some_field_3: float = 0.0f,
    @columninfo(
        name = "some_field_4",
        defaultvalue = "0"
    ) var some_field_4: float = 0.0f,
    @columninfo(
        name = "some_field_5",
        defaultvalue = "0"
    ) var some_field_5: float = 0.0f,
    @columninfo(
        name = "parent_id",
        defaultvalue = "0"
    ) var parent_id: string
)

这是整体服务器模型的一个片段:

type thing struct {
    gorm.model
    id            string    `gorm:"primarykey; size:40;" json:"id"`
    name          string    `json:"name"`
    typeid        string    `gorm:"size:40; index; not null" json:"type_id"`
    type          type      `gorm:"foreignkey:typeid; constraint:onupdate:cascade,ondelete:cascade" json:"type"`
    somefield1    float32   `json:"some_field_1"`
    somefield2    float32   `json:"some_field_2"`
    somefield3    float32   `json:"some_field_3"`
    somefield4    float32   `json:"some_field_4"`
    somefield5    float32   `json:"some_field_5"`
    parentid      float32   `json:"parent_id"`
}

然后我 post a thing 使用 retrofit 如下:

@post("/thing_info")
suspend fun postthinginfo(
    @headermap headers: map,
    @body data: thing
): response

val thing =
    thing(
        "",
        "",
        name = binding.txtname.text.tostring(),
        type_id = typeid,
        some_field_1 = binding.txtsomefield1.text.tostring()
            .tofloat(),
        some_field_2 = binding.txtsomefield2.text.tostring()
            .tofloat(),
        some_field_3 = binding.txtsomefield3.text.tostring()
            .tofloat(),
        parent_id = parentid,
    )

//then the usual retrofit boilerplate

使用这些代码,我只看到请求收到 201,但没有错误,即使我为进程中出现的每个错误对象返回一些内容。

这是 golang 代码顺便说一句:

ginGine.POST("/thing_info", func(ctx *gin.Context) {
    errMsg := clnHelpers.ValidateAccessToken(ctx.GetHeader("access_token"))

    if errMsg != "" {
        ctx.JSON(http.StatusBadRequest, gin.H{
            "error": errMsg,
        })
        return
    }

    var thing_info models.Thing

    err := ctx.ShouldBind(&thing_info)
    if err != nil {
        ctx.JSON(http.StatusBadRequest, gin.H{
            "error": err.Error(),
        })
        return
    }

    data := &pb.ThingInfo{
        Name:                        thing_info.Name,
        TypeId:                      thing_info.TypeId,
        //other field association
    }

    res, err := srvClient.service.CreateThingInfo(ctx, &pb.CreateThingInfoRequest{
        ThingInfo: data,
    })

    if err != nil {
        ctx.JSON(http.StatusBadRequest, gin.H{
            "error": err.Error(),
        })
        return
    }

    ctx.JSON(http.StatusCreated, gin.H{
        "thing_info": res.ThingInfo,
    })
})

正确答案


终于找到了这个问题的答案,我认为它与gorm有关。因此,在我实际的 android 模型中,我向其中添加了 created_at ,因为我需要它在 android 端进行一些排序。因为它是由 gorm 自动添加的字段,所以我认为当 posting a thing 时我真的不需要为其提供值,我什至不需要将它传递给 postman。最初,我只是向它传递一个空字符串,但它不喜欢它。然后我想有

var thing_info models.thinginfo

err := ctx.shouldbind(&thing_info)
if err != nil {
    ctx.json(http.statusbadrequest, gin.h{
        "error": err.error(),
    })
    return
}

在我的代码中会给我返回错误,就像我迄今为止遇到的所有其他错误一样。但事实并非如此,不知何故,之前的错误没有返回给我。然后我插入了一个好的ol'

fmt.Printf("thing_info bind err: %v\n", err)

在 if 语句中,最终得到一个错误(讽刺的是很高兴发现一个错误)。看起来,即使 created_atgorm 添加的字段,它仍然试图将我从 android 应用程序传递的空字符串转换为日期和时间。经过一些测试,似乎传递一个看起来像 2023-08-07t4:46:00z 的字符串是可以接受的,幸运的是,gorm 似乎忽略了该值,并输入了记录的实际创建日期时间。

理论要掌握,实操不能落!以上关于《Postman 成功发送到 Gin,但我的 Android 应用程序发送失败》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

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