golang GORM和Gin无法创建具有关联的对象

2024年 2月 12日 105.3k 0

golang gorm和gin无法创建具有关联的对象

问题内容

我使用 GORM 作为 ORM 以及我的 Golang API 来与我的数据库进行通信。

但是对于创建与数据库关联的实体,代码失败了(DB.Create(&data)):

2023/11/01 20:44:14 [Recovery] 2023/11/01 - 20:44:14 panic recovered:
POST /v1/product/products HTTP/1.1
Host: localhost:8080
Accept: */*
Content-Length: 589
Content-Type: application/json
User-Agent: curl/8.4.0

runtime error: invalid memory address or nil pointer dereference
/usr/lib/go/src/runtime/panic.go:261 (0x452d97)
panicmem: panic(memoryError)
/usr/lib/go/src/runtime/signal_unix.go:861 (0x452d65)
sigpanic: panicmem()
/home/grimm/go/pkg/mod/gorm.io/[email protected]/finisher_api.go:18 (0x926e7c)
(*DB).Create: if db.CreateBatchSize > 0 {
/home/grimm/Documents/beeskill/website/back_end/API_website/main.go:122 (0x99cfab)
CreatingProduct: DB.Create(&data)
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/context.go:174 (0x8c9ad9)
(*Context).Next: c.handlers[c.index](c)
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/recovery.go:102 (0x8c9ac7)
CustomRecoveryWithWriter.func1: c.Next()
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/context.go:174 (0x8c8c7d)
(*Context).Next: c.handlers[c.index](c)
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/logger.go:240 (0x8c8c4c)
LoggerWithConfig.func1: c.Next()
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/context.go:174 (0x8c7d3a)
(*Context).Next: c.handlers[c.index](c)
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/gin.go:620 (0x8c79cd)
(*Engine).handleHTTPRequest: c.Next()
/home/grimm/go/pkg/mod/github.com/gin-gonic/[email protected]/gin.go:576 (0x8c74fc)
(*Engine).ServeHTTP: engine.handleHTTPRequest(c)
/usr/lib/go/src/net/http/server.go:2938 (0x6a35ed)
serverHandler.ServeHTTP: handler.ServeHTTP(rw, req)
/usr/lib/go/src/net/http/server.go:2009 (0x69f4d3)
(*conn).serve: serverHandler{c.server}.ServeHTTP(w, w.req)
/usr/lib/go/src/runtime/asm_amd64.s:1650 (0x46ed20)
goexit: BYTE $0x90 // NOP

登录后复制

使用的代码:

package main

import (
"net/http"

"github.com/gin-gonic/gin"
"gorm.io/driver/sqlite"
"gorm.io/gorm"
)

type CategoryProduct struct {
gorm.Model
Title string `json:"title" gorm:"not null"`
Description string `json:"description"`
}

type CreateCategoryProduct struct {
gorm.Model
Title string `json:"title" binding:"required"`
Description string `json:"description" binding:"required"`
}

// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
type PriceYearProduct struct {
gorm.Model
Price float64 `json:"price" gorm:"not null"`
Description string `json:"description"`
}

type CreatePriceYearProduct struct {
gorm.Model
Price float64 `json:"price" binding:"required"`
Description string `json:"description" binding:"required"`
}

// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
type PriceMounthProduct struct {
gorm.Model
Price float64 `json:"price" gorm:"not null"`
Description string `json:"description"`
}

type CreatePriceMounthProduct struct {
gorm.Model
Price float64 `json:"price" binding:"required"`
Description string `json:"description" binding:"required"`
}

// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
// --------------------------------------------------------
type Product struct {
gorm.Model
Name string `json:"name" gorm:"not null"`
Description string `json:"description" gorm:"not null"`
PriceMounthProduct PriceMounthProduct `json:"pricemounth"`
PriceYearProduct PriceYearProduct `json:"priceyear"`
CategoryProduct CategoryProduct `json:"category"`
}

type CreateProduct struct {
gorm.Model
Name string `json:"name" binding:"required"`
Description string `json:"description" binding:"required"`
PriceMounthProduct PriceMounthProduct `json:"pricemounth" binding:"required"`
PriceYearProduct PriceYearProduct `json:"priceyear" binding:"required"`
CategoryProduct CategoryProduct `json:"category" binding:"required"`
}

var DB *gorm.DB

func ConnectDatabase() {

database, err := gorm.Open(sqlite.Open("test.db"), &gorm.Config{})

if err != nil {
panic("Failed to connect to database!")
}
// For the product and dependancies of them
err = database.AutoMigrate(&CategoryProduct{}, &PriceYearProduct{}, &PriceMounthProduct{})
if err != nil {
return
}
err = database.AutoMigrate(&Product{})
if err != nil {
return
}

DB = database
}

// POST /v1/products
// Create a product
func CreatingProduct(c *gin.Context) {
// Validate input
var input CreateProduct
if err := c.ShouldBindJSON(&input); err != nil {
c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
return
}

data := Product{
Name: input.Name,
Description: input.Description,
PriceYearProduct: input.PriceYearProduct,
PriceMounthProduct: input.PriceMounthProduct,
CategoryProduct: input.CategoryProduct,
}
DB.Create(&data)

c.JSON(http.StatusOK, gin.H{"data": data})

}

func Routes() {
var router = gin.Default()

v1 := router.Group("/v1")
{
//Product CRUD
product := v1.Group("/product")
product.POST("/products", CreatingProduct)

}

err := router.Run(":8080")
if err != nil {
return
}
}

func main() {
ConnectDatabase()
Routes()

}

登录后复制

以及用于发布一些数据的curl命令:

curl http://localhost:8080/v1/product/products --request "POST" --header "Content-Type: application/json" --data @file.json

登录后复制

file.json 的内容:

{
"name": "airplane simulation",
"description": "airplane simulation for everybody, childrens to adults.",
"priceyear": {
"price": 44.99,
"description": "pay for a year to access on irplane simulation for everybody, childrens to adults."
},
"pricemounth": {
"price": 4.99,
"description": "pay for a mounth to access on irplane simulation for everybody, childrens to adults."
},
"category": {
"title": "Airplane",
"description": "Information cataegorized on airplane."
}
}

登录后复制

我查看了官方文档,但没有更多信息......
尝试使用创建时的变量进行调试,但一切似乎都很好......

解决方法

运行代码出现错误

[error] invalid field found for struct
main.Product's field PriceMounthProduct:
define a valid foreign key for
relations or implement the Valuer/Scanner interface

登录后复制

gin日志前第一行的错误信息,可能你错过了。

所以修复它,在 Product 结构中添加正确的 foreignkey gorm 标签 就可以了。


type Product struct {
gorm.Model
Name string `json:"name" gorm:"not null"`
Description string `json:"description" gorm:"not null"`
PriceMounthProduct PriceMounthProduct `json:"pricemounth" gorm:"foreignkey:ID"`
PriceYearProduct PriceYearProduct `json:"priceyear" gorm:"foreignkey:ID"`
CategoryProduct CategoryProduct `json:"category" gorm:"foreignkey:ID"`
}

登录后复制

然后运行

curl http://localhost:8080/v1/product/products --request "POST" --header "Content-Type: application/json" --data @info.json

登录后复制

得到了

{"data":{"ID":1,"CreatedAt":"2023-11-02T13:37:24.052228+08:00","UpdatedAt":"2023-11-02T13:37:24.052228+08:00","DeletedAt":null,"name":"airplane simulation","description":"airplane simulation for everybody, childrens to adults.","pricemounth":{"ID":1,"CreatedAt":"2023-11-02T13:37:24.054792+08:00","UpdatedAt":"2023-11-02T13:37:24.054792+08:00","DeletedAt":null,"price":4.99,"description":"pay for a mounth to access on irplane simulation for everybody, childrens to adults."},"priceyear":{"ID":1,"CreatedAt":"2023-11-02T13:37:24.056352+08:00","UpdatedAt":"2023-11-02T13:37:24.056352+08:00","DeletedAt":null,"price":44.99,"description":"pay for a year to access on irplane simulation for everybody, childrens to adults."},"category":{"ID":1,"CreatedAt":"2023-11-02T13:37:24.056585+08:00","UpdatedAt":"2023-11-02T13:37:24.056585+08:00","DeletedAt":null,"title":"Airplane","description":"Information cataegorized on airplane."}}}%

登录后复制

以上就是golang GORM和Gin无法创建具有关联的对象的详细内容,更多请关注每日运维网(www.mryunwei.com)其它相关文章!

相关文章

JavaScript2024新功能:Object.groupBy、正则表达式v标志
PHP trim 函数对多字节字符的使用和限制
新函数 json_validate() 、randomizer 类扩展…20 个PHP 8.3 新特性全面解析
使用HTMX为WordPress增效:如何在不使用复杂框架的情况下增强平台功能
为React 19做准备:WordPress 6.6用户指南
如何删除WordPress中的所有评论

发布评论