Gin(四):表单数据提交和模型绑定 - Go语言中文社区

Gin(四):表单数据提交和模型绑定


640?wx_fmt=jpeg


上一个章节中已经开始逐渐搭建了一个 web 页面,现在我们开始逐步完善页面上的功能,首先要完成的是登录和注册功能。

640?接受表单数据640

注册页面的 HTML 元素不在详细写出,具体页面代码可以直接参考 Github 上代码。

页面完成后布局:

640?wx_fmt=jpeg


注册页面有三个输入框,分别为 email ,passwordpassword again

完善后端 Gin 代码。我们在 initRouteruserGroup 中编写新的接口。

 userRouter.POST("/register", handler.UserRegister)

编写完新的接口就要开始编写 Handler

 func UserRegister(context *gin.Context) {	
     email := context.PostForm("email")	
     password := context.DefaultPostForm("password", "Wa123456")	
     passwordAgain := context.DefaultPostForm("password-again", "Wa123456")	
     println("email", email, "password", password, "password again", passwordAgain)	
 }

UserRegister 方法中采用新的方式来接受 Post 请求提交的表单参数,PostFormDefaultPostFormPostForm 直接接受参数,而 DefaultPostForm 可以设置一个默认值,如果前端没有进行传值,那么我们可以设置默认值,如上面的代码,如果前端没有将密码传输过来我们可以设置一个默认密码。

当我们运行并且输入的时候,在控制台上可以清楚的看到我们在表单上的输入。

当我们项目功能完善的时候,就可以完善我们的单元测试。

此时的单元测试交之前有点复杂。

首先我们要构造一个结构,该结构是为了帮助我们将我们要提交的信息存放到表单中,同时要指定请求头信息。

 func TestUserPostForm(t *testing.T) {	
     value := url.Values{}	
     value.Add("email", "youngxhui@gmail.com")	
     value.Add("password", "1234")	
     value.Add("password-again", "1234")	
     w := httptest.NewRecorder()	
     req, _ := http.NewRequest(http.MethodPost, "/user/register", bytes.NewBufferString(value.Encode()))	
     req.Header.Add("Content-Type", "application/x-www-form-urlencoded; param=value")	
     router.ServeHTTP(w, req)	
     assert.Equal(t, http.StatusOK, w.Code)	
 }

单元测试编写完成后可以运行单元测试,发现控制台答应了我们在测试中写的数据。

640?模型绑定640

上例中我们的表单仅仅传输了三个参数,如果后期项目出现了十多个参数,每次写一遍都很花费时间,也很消耗经历。下面就对该方法进行改善

Gin 中提供了 模型绑定,将我们的表单数据与我们的模型进行一样绑定。GIn会将数据统一封装到模型中,方便我们日后使用。

首先定义我们的模型,新建 model 文件夹,建立 userModel.go

 package model	
 	
 type UserModel struct {	
     Email         string `form:"email"`	
     Password      string `form:"password"`	
     PasswordAgain string `form:"password-again"`	
 }

通过 form:"email" 来对表单中的  email 输入数据进行绑定。然后需要修改一下 Handler 方法。

 func UserRegister(context *gin.Context) {	
     var user model.UserModel	
     if err := context.ShouldBind(&user); err != nil {	
         println("err ->", err.Error())	
         return	
 }	
     println("email", user.Email, "password", user.Password, "password again", user.PasswordAgain)	
 }

此时我们的模型绑定已经写好,运行 TestUserPostForm 测试用例,测试用例可以完美的通过。说明我们的模型绑定方法是正确的。同时模型绑定还是从 jsonxmlyml  等格式数据的绑定,日后会有介绍和说明。当然也可以通过浏览器中的注册表单进行提交。

640?数据校验640

做后端开发的人都明白一个道理:永远不要相信前端传过来的数据。所有的数据在进过后端时,务必要进行数据的校验。

在模型中可用 binding 来对数据进行校验。Gin 对于数据校验使用的是 validator.v8 库,该库提供多种校验方法。通过 binding:"" 方式来进行对数据的校验。

我们将 UserModel 进行修改,添加一些规则,邮箱验证和密码校验,要求第二次重复密码要和第一次密码一致。更多的校验规则可以看官方文档

 type UserModel struct {	
     Email         string `form:"email" binding:"email"`	
     Password      string `form:"password"`	
     PasswordAgain string `form:"password-again" binding:"eqfield=Password"`	
 }

我们重新写一个测试用例用来测试邮箱和密码校验是否有效。

 func TestUserPostFormEmailErrorAndPasswordError(t *testing.T) {	
     value := url.Values{}	
     value.Add("email", "youngxhui")	
     value.Add("password", "1234")	
     value.Add("password-again", "qwer")	
     w := httptest.NewRecorder()	
     req, _ := http.NewRequest(http.MethodPost, "/user/register", bytes.NewBufferString(value.Encode()))	
     req.Header.Add("Content-Type", "application/x-www-form-urlencoded; param=value")	
     router.ServeHTTP(w, req)	
     assert.Equal(t, http.StatusOK, w.Code)	
 }

运行测试,发现测试虽然通过了,但是会有两行 error 信息

 err ->  Key: 'UserModel.Email' Error:Field validation for 'Email' failed on the 'email' tag	
 Key: 'UserModel.PasswordAgain' Error:Field validation for 'PasswordAgain' failed on the 'eqfield' tag

该信息说明了我们的 EmailPasswordAgain 信息校验没有通过。

640?使用Log和重定向640

测试通过是因为无论我们代码如何都会返回 200 状态码,这是不符合http 状态码的规范的,所以我们要对http状态码进行规范化。同时我们之前的代码中一直使用 Printf 来打印日志信息,也是不规范的,因为 Printf 打印的日志信息相对局限,所以应该选用 Log 进行日志打印。

 func UserRegister(context *gin.Context) {	
     var user model.UserModel	
     if err := context.ShouldBind(&user); err != nil {	
         log.Println("err ->", err.Error())	
         context.String(http.StatusBadRequest, "输入的数据不合法")	
     } else {	
         log.Println("email", user.Email, "password", user.Password, "password again", user.PasswordAgain)	
         context.Redirect(http.StatusMovedPermanently, "/")	
   }	
 }

首先我们将原来只用 Println 打印的数据都改成了 log 去打印数据。

同时将原来的状态码都进行了更改,不同的状态码代表不同的请求响应结果。

最后在请求成功的时候我们对路由进行了重定向,将页面转跳到首页。

同时我们也要将测试用例里的返回状态码进行修改。

640✍总结640

本节将表单提交,模型绑定和数据校验有了一个相对细致的介绍,代码中也通过不同的测试用例来检查代码是否正确。

640?‍?本章节代码640

请点击阅读原文

往期推荐

Gin(一):Hello Gin

Gin(二):路由Router

Gin(三):搭配模板tmpl



640?wx_fmt=jpeg


好看的人才能点640



版权声明:本文来源CSDN,感谢博主原创文章,遵循 CC 4.0 by-sa 版权协议,转载请附上原文出处链接和本声明。
原文链接:https://blog.csdn.net/codeSquare/article/details/99288697
站方申明:本站部分内容来自社区用户分享,若涉及侵权,请联系站方删除。
  • 发表于 2020-03-01 17:36:43
  • 阅读 ( 1764 )
  • 分类:Go Web框架

0 条评论

请先 登录 后评论

官方社群

GO教程

猜你喜欢