我使用rest api
,golang
和gin
gorp
Employee structure:
type Employee struct {
Id int64 `db:"id" json:"id"`
Firstname string `db:"firstname" json:"firstname"`
Lastname string `db:"lastname" json:"lastname"`
Dob time.Time `db:"dob" json:"dob"`
Skills []string `db:skills json:"skills"`
}
在POST
发送请求中:
func PostEmployee(c *gin.Context) {
var emp Employee
c.Bind(&emp)
skills, _ := json.Marshal(emp.Skills)
if emp.Firstname != "" && emp.Lastname != "" {
if insert, _ := dbmap.Exec(`INSERT INTO employee (firstname, lastname, dob, skills) VALUES (?, ?, ?, ?)`, emp.Firstname, emp.Lastname, emp.Dob, skills); insert != nil {
emp_id, err := insert.LastInsertId()
.....
}
......
}
将数据保存到mysql
数据库,效果很好。
用于从数据库实现的GET
请求
func GetEmployees(c *gin.Context) {
var emps []Employee
_, err := dbmap.Select(&emps, "SELECT * FROM employee")
log.Println(err)
if err == nil {
c.JSON(200, emps)
} else {
c.JSON(404, gin.H{"error": "no employee(s) into the table"})
}
GET
查询没有提供来自数据库的任何数据,log.Println(err)
日志说明:
Scan error on column index 4: unsupported Scan, storing driver.Value type []uint8 into type *[]string
有什么想法吗?
答案 0 :(得分:6)
两种方法:
1。为自定义类型实现sql.Scanner和driver.Valuer接口
好处:
注意事项:
package tgorm
import (
"database/sql/driver"
"encoding/json"
"errors"
"fmt"
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/sqlite"
"github.com/stretchr/testify/assert"
"strings"
"testing"
"time"
)
type Skills []string
func (s Skills) Value() (driver.Value, error) {
if len(s) == 0 {
return "[]", nil
}
return fmt.Sprintf(`["%s"]`, strings.Join(s, `","`)), nil
}
func (s *Skills) Scan(src interface{}) (err error) {
var skills []string
switch src.(type) {
case string:
err = json.Unmarshal([]byte(src.(string)), &skills)
case []byte:
err = json.Unmarshal(src.([]byte), &skills)
default:
return errors.New("Incompatible type for Skills")
}
if err != nil {
return
}
*s = skills
return nil
}
type Employee struct {
Id int64 `db:"id" json:"id"`
Firstname string `db:"firstname" json:"firstname"`
Lastname string `db:"lastname" json:"lastname"`
Dob time.Time `db:"dob" json:"dob"`
Skills Skills `gorm:"type:varchar(255);" db:"skills" json:"skills"`
}
func (e Employee) TableName() string {
return "employee"
}
func getMemoryDataBase() *gorm.DB {
db, err := gorm.Open("sqlite3", ":memory:")
if err != nil {
panic(err)
}
db = db.Debug()
db.AutoMigrate(Employee{})
return db
}
func TestSaveEmployee(t *testing.T) {
db := getMemoryDataBase()
emp := Employee{
Id: 1,
Firstname: "Fake",
Lastname: "Emp",
Dob: time.Time{},
Skills: []string{"C#", "GO", "C++"},
}
skills, _ := json.Marshal(emp.Skills)
err := db.Exec(`INSERT INTO employee (firstname, lastname, dob, skills) VALUES (?, ?, ?, ?)`, emp.Firstname, emp.Lastname, emp.Dob, skills).Error
assert.Nil(t, err)
var emps []Employee
err = db.Raw("SELECT * FROM employee").Scan(&emps).Error
assert.Nil(t, err)
assert.Equal(t, []Employee{emp}, emps)
}
2。将技能移到一个单独的表中,并引用该员工。
好处:
注意事项:
package subgrom
import (
"github.com/jinzhu/gorm"
_ "github.com/jinzhu/gorm/dialects/sqlite"
"github.com/stretchr/testify/assert"
"testing"
"time"
)
type Skill struct {
Id int64 `db:"id" json:"id"`
Skill string `db:"skill" json:"skill"`
EmployeeRef int64
}
type Employee struct {
Id int64 `db:"id" json:"id"`
Firstname string `db:"firstname" json:"firstname"`
Lastname string `db:"lastname" json:"lastname"`
Dob time.Time `db:"dob" json:"dob"`
Skills []Skill `db:"skills" json:"skills" gorm:"foreignkey:EmployeeRef"`
}
func (e Employee) TableName() string {
return "employee"
}
func getMemoryDataBase() *gorm.DB {
db, err := gorm.Open("sqlite3", ":memory:")
if err != nil {
panic(err)
}
db = db.Debug()
db.AutoMigrate(Employee{}, Skill{})
return db
}
func TestSaveEmployee(t *testing.T) {
db := getMemoryDataBase()
emp := Employee{
Id: 1,
Firstname: "Fake",
Lastname: "Emp",
Dob: time.Time{},
Skills: []Skill{{Skill: "C#"}, {Skill: "GO"}, {Skill: "C++"}},
}
err := db.Create(&emp).Error
assert.Nil(t, err)
var emps []Employee
err = db.Preload("Skills").Find(&emps).Error
assert.Nil(t, err)
assert.Equal(t, []Employee{emp}, emps)
}
答案 1 :(得分:0)
面临类似的问题,对我来说问题是字段“scope_t”的排序。
selectGroup = `SELECT
id,
name,
fully_qualified_name,
parent_id,
scopes,
scope_t
FROM groups `
插入数据时,我将“scope_t”放在随机位置,因此 SQL 返回并出现上述错误,原因是该位置被映射为不同的数据类型。
if err := r.db.QueryRowContext(ctx, createGroup, group.Name, group.FullyQualifiedName,
pq.Array(group.Scopes), group.ParentID, userID, scope_type).Scan(&id); err != nil {
return nil, err
}
TL;DR
在插入之前还要检查您可能会因哪种类型错误而交换值的位置。