mysql - 使用BeforeUpdate的gorm哈希密码不会使用哈希值更新密码

标签 mysql go go-gorm

我有一个非常简单的User模型,下面是我的整个main.go

package main

import (
  "fmt"

  "github.com/jinzhu/gorm"
  _ "github.com/jinzhu/gorm/dialects/mysql"
  "golang.org/x/crypto/bcrypt"
)

// User : the user data model
type User struct {
  gorm.Model

  Username string `gorm:"type:varchar(40);unique" json:"username,omitempty"`
  Password string `gorm:"size:255" json:"password,omitempty"`
}

// BeforeSave : hook before a user is saved
func (u *User) BeforeSave(scope *gorm.Scope) (err error) {
  fmt.Println("before save")
  fmt.Println(u.Password)
  if u.Password != "" {
     hash, err := MakePassword(u.Password)
     if err != nil {
        return nil
     }
     u.Password = hash
  }

  fmt.Println(u.Password)
  return
}

// MakePassword : Encrypt user password
func MakePassword(password string) (string, error) {
  bytes, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
  return string(bytes), err
}

func main() {
  var connectionString = fmt.Sprintf(
    "%s:%s@/%s?charset=utf8&parseTime=True&loc=Local",
    "root", "password", "project",
  )

  db, _ := gorm.Open("mysql", connectionString)
  db.AutoMigrate(&User{})
  db.Save(&User{
    Username: "name1",
    Password: "123",
  })

  db.Model(&User{
    Model: gorm.Model{ID: 1},
  }).Update(&User{
    Username: "name2",
    Password: "12345",
  })
}
第一次创建密码时,便会创建密码并将其另存为哈希字符串(我注释掉了第二个更新部分并对其进行了测试),但是在更新时,密码另存为纯文本。
从控制台中,我可以看到Println消息,触发了BeforeUpdate,并且看到了它的纯口令和MD5,但是在数据库中,它被另存为纯文本。
before save
123
$2a$10$Vknv/uu7tAPRQSddPVlQ7OodIHZJmRPKktjb98U8U5.GT/OLQeQE2
before save
12345
$2a$10$K0ZkLH7slfiFmkOe5DTKr.DGNvR6HtpjxCS/1svf2ZEvfTXVkMkvu
关于如何解决该问题的任何想法?

最佳答案

According to the docs为了更改要更新的值,必须通过gorm Scope参数设置它们。在这种情况下,User结构不用于修改。您应该改用SetColumn。

// BeforeUpdate : hook before a user is updated
func (u *User) BeforeUpdate(scope *gorm.Scope) (err error) {
    fmt.Println("before update")
    fmt.Println(u.Password)

    if u.Password != "" {
        hash, err := MakePassword(u.Password)
        if err != nil {
            return nil
        }
        scope.SetColumn("Password", hash)
    }

    fmt.Println(u.Password)
    return
}

关于mysql - 使用BeforeUpdate的gorm哈希密码不会使用哈希值更新密码,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63484061/

相关文章:

适用于 Windows 7 64 位操作系统的 MYSQL 安装程序?

GORM 不会将 bool 字段更新为 false

pointers - 用指针索引-指针如何与 slice 一起使用?

go - 关于 goroutine 在 goroutine 中的行为

json - 如何在 Go 中从 interface{} 解码到 interface{}

GORM Golang 如何优化这段代码

go - 如何使用结构显示所有记录

mysql - sql error using between operator with date between two dates

php - MySQL where NOT IN 名称数组?

php - 当我选择一个不存在的表时,PDO 没有捕获错误