Невозможно создать связь с внешним ключом с помощью gorm, где имя поля!= Имя типа
Я пытаюсь создать отношение "принадлежащий" в моем приложении, где имя поля (и столбца БД) не совпадает с именем типа. Горм жалуется на это. Вот короткий пример, который демонстрирует проблему:
package main
import (
"fmt"
"github.com/jinzhu/gorm"
_ "github.com/mattn/go-sqlite3"
)
type Car struct {
gorm.Model
Owner User
OwnerID int
}
type User struct {
gorm.Model
}
func main() {
db, _ := gorm.Open("sqlite3", "test.db")
defer db.Close()
db.LogMode(true)
db.CreateTable(&Car{}, &User{})
u := User{}
u.ID = 1
db.Create(&u)
c := Car{}
c.ID = 1
c.Owner = u
db.Create(&c)
cars := make([]Car, 5)
db.Find(&cars)
for i := range cars {
db.Model(cars[i]).Related(&cars[i].Owner)
}
for i := range cars {
fmt.Println(cars[i].Owner.ID)
}
}
Мы ожидаем, что эта программа выведет "1" в качестве последней строки, так как это первичный ключ отношения Users, которому "принадлежит" единственный автомобиль в базе данных. Вместо этого выводится "0", и подробное ведение журнала показывает ошибку при Related(&cars[i].Owner)
шаг:
(/home/nate/gocode/src/foo/main.go:26)
[2016-04-09 19:33:46] [122.35ms] CREATE TABLE "cars" ("id" integer primary key autoincrement,"created_at" datetime,"updated_at" datetime,"deleted_at" datetime,"owner_id" integer )
(/home/nate/gocode/src/foo/main.go:26)
[2016-04-09 19:33:46] [98.96ms] CREATE INDEX idx_cars_deleted_at ON "cars"(deleted_at)
(/home/nate/gocode/src/foo/main.go:26)
[2016-04-09 19:33:46] [108.95ms] CREATE TABLE "users" ("id" integer primary key autoincrement,"created_at" datetime,"updated_at" datetime,"deleted_at" datetime )
(/home/nate/gocode/src/foo/main.go:26)
[2016-04-09 19:33:46] [125.74ms] CREATE INDEX idx_users_deleted_at ON "users"(deleted_at)
(/home/nate/gocode/src/foo/main.go:30)
[2016-04-09 19:33:46] [0.53ms] INSERT INTO "users" ("id","created_at","updated_at","deleted_at") VALUES ('1','2016-04-09T19:33:46-04:00','2016-04-09T19:33:46-04:00','<nil>')
(/home/nate/gocode/src/foo/main.go:35)
[2016-04-09 19:33:46] [1.98ms] UPDATE "users" SET "created_at" = '2016-04-09T19:33:46-04:00', "updated_at" = '2016-04-09T19:33:46-04:00', "deleted_at" = '<nil>' WHERE "users".deleted_at IS NULL AND "users"."id" = '1'
(/home/nate/gocode/src/foo/main.go:35)
[2016-04-09 19:33:46] [0.43ms] INSERT INTO "cars" ("id","created_at","updated_at","deleted_at","owner_id") VALUES ('1','2016-04-09T19:33:46-04:00','2016-04-09T19:33:46-04:00','<nil>','1')
(/home/nate/gocode/src/foo/main.go:38)
[2016-04-09 19:33:46] [0.64ms] SELECT * FROM "cars" WHERE "cars".deleted_at IS NULL
(/home/nate/gocode/src/foo/main.go:40)
[2016-04-09 19:33:46] invalid association []
0
Что здесь происходит? Есть ли какой-то специальный тег, который я должен использовать на Owner
поле? Я пытался использовать gorm:"ForeignKey:owner_id"
а также gorm:"AssociationForeignKey:owner_id"
оба дают одинаковый результат.
1 ответ
Самый простой способ в вашем случае - указать внешнее имя поля. Так что ваши
db.Model(cars[i]).Related(&cars[i].Owner)
должно быть как
db.Model(cars[i]).Related(&cars[i].Owner, "Owner")
Альтернативные способы
- переименовывать
Owner
а такжеOwnerID
поля дляUser
а такжеUserID
или же - переименовывать
User
структура дляOwner
В этих случаях GORM автоматически выберет правильные внешние ключи. Из документации ГОРМ об ассоциациях:
Если имя поля совпадает с именем типа переменной, оно может быть опущено.