https://wangzitian0.github.io/2013/06/29/zero-to-one-1/mysql
https://github.com/gothinkster/golang-gin-realworld-example-appgit
golang-gin-realworld-example-app/ ├── articles │ ├── doc.go │ ├── models.go │ ├── routers.go │ ├── serializers.go │ └── validators.go ├── common │ ├── database.go │ ├── unit_test.go │ └── utils.go ├── doc.go ├── main.go ├── scripts │ ├── coverage.sh │ └── gofmt.sh ├── users │ ├── doc.go │ ├── middlewares.go │ ├── models.go │ ├── routers.go │ ├── serializers.go │ ├── unit_test.go │ └── validators.go └── vendor └── vendor.json
https://github.com/gothinkster/golang-gin-realworld-example-app/blob/master/common/database.gogithub
package common import ( "github.com/jinzhu/gorm" _ "github.com/jinzhu/gorm/dialects/mysql" _ "github.com/jinzhu/gorm/dialects/sqlite" ) // 定義結構體Database type Database struct { *gorm.DB } // 定義全局變量DB var DB *gorm.DB // 初始化數據庫 func Init() *gorm.DB { // 鏈接數據庫 db, err := gorm.Open("sqlite3", "./../gorm.db") //db, err := gorm.Open("mysql", "user:password@/dbname?charset=utf8mb4&parseTime=True&loc=Local") if err != nil { fmt.Println("db err: ", err) } // 設置閒置的鏈接數 db.DB().SetMaxIdleConns(10) // 設置最大打開的鏈接數 db.DB().SetMaxOpenConns(100) // 啓用Logger,顯示詳細日誌 db.LogMode(true) DB = db return DB } // 函數,獲取數據庫鏈接,創建鏈接池 func GetDB() *gorm.DB { return DB }
https://github.com/gothinkster/golang-gin-realworld-example-app/blob/master/common/utils.gogolang
package common import ( "fmt" "math/rand" "time" "github.com/dgrijalva/jwt-go" "gopkg.in/go-playground/validator.v8" "github.com/gin-gonic/gin/binding" "gopkg.in/gin-gonic/gin.v1" ) var letters = []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789") // 函數 生成隨機字符串 func RandString(n int) string { b := make([]rune, n) for i := range b { b[i] = letters[rand.Intn(len(letters))] } return string(b) } // Keep this two config private, it should not expose to open source const NBSecretPassword = "A String Very Very Very Strong!!@##$!@#$" const NBRandomPassword = "A String Very Very Very Niubilty!!@##$!@#4" // 函數 生成請求頭使用的jwt_token func GenToken(id uint) string { jwt_token := jwt.New(jwt.GetSigningMethod("HS256")) // Set some claims jwt_token.Claims = jwt.MapClaims{ "id": id, "exp": time.Now().Add(time.Hour * 24).Unix(), } // Sign and get the complete encoded token as a string token, _ := jwt_token.SignedString([]byte(NBSecretPassword)) return token } // 自定義錯誤類型 // {"database": {"hello":"no such table", error: "not_exists"}} type CommonError struct { Errors map[string]interface{} `json:"errors"` } // 通常錯誤處理 func NewError(key string, err error) CommonError { res := CommonError{} res.Errors = make(map[string]interface{}) res.Errors[key] = err.Error() return res } // gin的validator錯誤處理 // https://github.com/go-playground/validator/blob/v9/_examples/translations/main.go func NewValidatorError(err error) CommonError { res := CommonError{} res.Errors = make(map[string]interface{}) errs := err.(validator.ValidationErrors) for _, v := range errs { // can translate each error one at a time. //fmt.Println("gg",v.NameNamespace) if v.Param != "" { res.Errors[v.Field] = fmt.Sprintf("{%v: %v}", v.Tag, v.Param) } else { res.Errors[v.Field] = fmt.Sprintf("{key: %v}", v.Tag) } } return res } // 變動gin-gonic的Bind函數,改變 c.MustBindWith() -> c.ShouldBindWith(). func Bind(c *gin.Context, obj interface{}) error { b := binding.Default(c.Request.Method, c.ContentType()) //return c.MustBindWith(obj, b) return c.ShouldBindWith(obj, b) }