Gorm 关系错误:需要为关系定义有效的外键或者需要实现 Valuer/Scanner 接口

Tek*_*ein 3 sqlite go go-gorm

我正在对使用 Gorm 时出现的问题进行故障排除。我的 sqlite3 数据库和 Go 数据模型一切都工作得很好,但是当我遇到一些依赖项问题时,无法在构建环境中“进入”,所以我尝试从供应商文件夹中复制/删除一些包,然后重新“去”直到我让构建工作起来......但是现在当我在自己的机器上编译和运行时,我遇到了以前从未遇到过的问题。

当我尝试类似的操作时(已检查 configID 以确保它具有有效的条目):

    var config models.ConfigurationDescription

    // Find the Configuration
    results := db.Where("id = ?", configID).
        Preload("Location").
        Find(&config)
Run Code Online (Sandbox Code Playgroud)

Gorm 抛出以下错误:

 "invalid field found for struct `models.ConfigurationDescription`'s field Location, need to define a valid foreign key for relations or it need to implement the Valuer/Scanner interface"
Run Code Online (Sandbox Code Playgroud)

这就是我定义数据模型的方式(在依赖关系混乱之前它运行良好):

package models

type LocationDescription struct {
    ID       int    `json:"locationID"`
    Name     string `json:"name"`
    IsActive bool   `json:"isActive"`
}

func (LocationDescription) TableName() string { return "locations" }

type ConfigurationDescription struct {
    ID         int                 `json:"configurationID"`
    Name       string              `json:"name"`
    IsActive   bool                `json:"isActive"`
    LocationID int                 `json:"-"`
    Location   LocationDescription `json:"location,omitempty" gorm:"foreignKey:ID;references:LocationID"`
}

func (ConfigurationDescription) TableName() string { return "configurations" }
Run Code Online (Sandbox Code Playgroud)

这是使用以下设置针对 sqlite 数据库运行的:

CREATE TABLE IF NOT EXISTS locations (
    id           INTEGER PRIMARY KEY AUTOINCREMENT,
    name         TEXT,
    latitude     REAL CHECK (latitude > 0),
    longitude    REAL CHECK (latitude > 0),
    is_active    INTEGER DEFAULT 0
);

CREATE TABLE IF NOT EXISTS configurations (
    id               INTEGER PRIMARY KEY AUTOINCREMENT,
    location_id      INTEGER,
    name             TEXT,
    is_active        INTEGER DEFAULT 0,
    CONSTRAINT location_fk FOREIGN KEY (location_id) REFERENCES locations(id) ON DELETE CASCADE
);
Run Code Online (Sandbox Code Playgroud)

我知道它正在工作,并且我没有更改任何代码。我看到的所有内容看起来都遵循文档,因此依赖项的更新破坏了某些内容是没有意义的,因为似乎没有发生任何重大更改...... ..

和!真正的亮点是它有效:

var location models.Location
DB.Where("is_active = ?", true).
        Preload("Configurations").
        Find(&location)
Run Code Online (Sandbox Code Playgroud)

(这些结构和数据库实际上有更多的字段,但为了简单起见,我将它们修剪掉了)

type Location struct {
    ID             int             `json:"locationID"`
    Name           string          `json:"name"`
    Latitude       null.Float      `json:"latitude,omitempty"`
    Longitude      null.Float      `json:"longitude,omitempty"`
    IsActive       null.Bool       `json:"isActive" gorm:"default:false"`
    Configurations []Configuration `json:"configurations,omitempty" gorm:"foreignKey:LocationID;references:ID"`
}

type Configuration struct {
    ID             int             `json:"configurationID"`
    LocationID     int             `json:"locationID"`
    Name           string          `json:"name"`
    IsActive       null.Bool       `json:"isActive" gorm:"default:false"`
}
Run Code Online (Sandbox Code Playgroud)

那么问题来了,有谁知道这可能是什么原因造成的吗?更重要的是,对于如何解决这个问题有什么建议吗?

Eze*_*uns 15

在我看来,您的 LocationDescription 字段的标签方式错误。

首先,这是一种属于关系模式。

  • foreignKey应命名连接到外部实体的模型本地关键字段。
  • references应命名外部实体的主键或唯一键。

尝试使用:

type ConfigurationDescription struct {
    ID         int                 `json:"configurationID"`
    Name       string              `json:"name"`
    IsActive   bool                `json:"isActive"`
    LocationID int                 `json:"-"`
    Location   LocationDescription `json:"location,omitempty" gorm:"foreignKey:LocationID;references:ID"`
}
Run Code Online (Sandbox Code Playgroud)