2014-08-18 16:41:43 +08:00
|
|
|
// Copyright 2014 beego Author. All Rights Reserved.
|
2014-07-03 23:40:21 +08:00
|
|
|
//
|
2014-08-18 16:41:43 +08:00
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
2014-07-03 23:40:21 +08:00
|
|
|
//
|
2014-08-18 16:41:43 +08:00
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
2014-07-03 23:40:21 +08:00
|
|
|
//
|
2014-08-18 16:41:43 +08:00
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2013-07-25 22:25:12 +08:00
|
|
|
package beego
|
|
|
|
|
|
|
|
import (
|
2013-08-10 16:33:46 +08:00
|
|
|
"html/template"
|
2013-07-25 22:25:12 +08:00
|
|
|
"net/url"
|
2014-07-03 23:40:21 +08:00
|
|
|
"reflect"
|
2013-07-25 22:25:12 +08:00
|
|
|
"testing"
|
2013-07-26 16:29:00 +08:00
|
|
|
"time"
|
2013-07-25 22:25:12 +08:00
|
|
|
)
|
|
|
|
|
2013-07-26 16:29:00 +08:00
|
|
|
func TestSubstr(t *testing.T) {
|
|
|
|
s := `012345`
|
|
|
|
if Substr(s, 0, 2) != "01" {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
if Substr(s, 0, 100) != "012345" {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
2014-08-05 23:52:06 +08:00
|
|
|
if Substr(s, 12, 100) != "012345" {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestHtml2str(t *testing.T) {
|
|
|
|
h := `<HTML><style></style><script>x<x</script></HTML><123> 123\n
|
|
|
|
|
|
|
|
|
|
|
|
\n`
|
2015-09-08 23:41:41 +08:00
|
|
|
if HTML2str(h) != "123\\n\n\\n" {
|
2013-07-26 16:29:00 +08:00
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDateFormat(t *testing.T) {
|
|
|
|
ts := "Mon, 01 Jul 2013 13:27:42 CST"
|
|
|
|
tt, _ := time.Parse(time.RFC1123, ts)
|
2014-05-30 14:12:21 -05:00
|
|
|
|
2014-05-31 14:28:44 +08:00
|
|
|
if ss := DateFormat(tt, "2006-01-02 15:04:05"); ss != "2013-07-01 13:27:42" {
|
|
|
|
t.Errorf("2013-07-01 13:27:42 does not equal %v", ss)
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDate(t *testing.T) {
|
|
|
|
ts := "Mon, 01 Jul 2013 13:27:42 CST"
|
|
|
|
tt, _ := time.Parse(time.RFC1123, ts)
|
2014-05-30 14:12:21 -05:00
|
|
|
|
2014-05-31 14:28:44 +08:00
|
|
|
if ss := Date(tt, "Y-m-d H:i:s"); ss != "2013-07-01 13:27:42" {
|
|
|
|
t.Errorf("2013-07-01 13:27:42 does not equal %v", ss)
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
2014-05-31 14:28:44 +08:00
|
|
|
if ss := Date(tt, "y-n-j h:i:s A"); ss != "13-7-1 01:27:42 PM" {
|
|
|
|
t.Errorf("13-7-1 01:27:42 PM does not equal %v", ss)
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
2014-05-31 14:28:44 +08:00
|
|
|
if ss := Date(tt, "D, d M Y g:i:s a"); ss != "Mon, 01 Jul 2013 1:27:42 pm" {
|
|
|
|
t.Errorf("Mon, 01 Jul 2013 1:27:42 pm does not equal %v", ss)
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
2014-05-31 14:28:44 +08:00
|
|
|
if ss := Date(tt, "l, d F Y G:i:s"); ss != "Monday, 01 July 2013 13:27:42" {
|
|
|
|
t.Errorf("Monday, 01 July 2013 13:27:42 does not equal %v", ss)
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-05 16:38:57 +08:00
|
|
|
func TestCompareRelated(t *testing.T) {
|
2013-07-26 16:29:00 +08:00
|
|
|
if !Compare("abc", "abc") {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
if Compare("abc", "aBc") {
|
|
|
|
t.Error("should be not equal")
|
|
|
|
}
|
|
|
|
if !Compare("1", 1) {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
2015-09-08 23:41:41 +08:00
|
|
|
if CompareNot("abc", "abc") {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
if !CompareNot("abc", "aBc") {
|
|
|
|
t.Error("should be not equal")
|
|
|
|
}
|
|
|
|
if !NotNil("a string") {
|
|
|
|
t.Error("should not be nil")
|
|
|
|
}
|
2013-07-26 16:29:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestHtmlquote(t *testing.T) {
|
2018-07-20 14:26:43 +08:00
|
|
|
h := `<' ”“&">`
|
2013-07-26 16:29:00 +08:00
|
|
|
s := `<' ”“&">`
|
|
|
|
if Htmlquote(s) != h {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHtmlunquote(t *testing.T) {
|
2018-07-20 14:26:43 +08:00
|
|
|
h := `<' ”“&">`
|
|
|
|
s := `<' ”“&">`
|
2013-07-26 16:29:00 +08:00
|
|
|
if Htmlunquote(h) != s {
|
|
|
|
t.Error("should be equal")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-25 22:25:12 +08:00
|
|
|
func TestParseForm(t *testing.T) {
|
2016-09-02 16:08:04 +08:00
|
|
|
type ExtendInfo struct {
|
2019-01-25 00:38:14 +07:00
|
|
|
Hobby []string `form:"hobby"`
|
2016-09-02 16:08:04 +08:00
|
|
|
Memo string
|
|
|
|
}
|
|
|
|
|
|
|
|
type OtherInfo struct {
|
|
|
|
Organization string `form:"organization"`
|
|
|
|
Title string `form:"title"`
|
|
|
|
ExtendInfo
|
|
|
|
}
|
|
|
|
|
2013-07-25 22:25:12 +08:00
|
|
|
type user struct {
|
2015-09-08 23:41:41 +08:00
|
|
|
ID int `form:"-"`
|
2014-11-04 16:39:17 +08:00
|
|
|
tag string `form:"tag"`
|
|
|
|
Name interface{} `form:"username"`
|
|
|
|
Age int `form:"age,text"`
|
|
|
|
Email string
|
|
|
|
Intro string `form:",textarea"`
|
|
|
|
StrBool bool `form:"strbool"`
|
|
|
|
Date time.Time `form:"date,2006-01-02"`
|
2016-09-02 16:08:04 +08:00
|
|
|
OtherInfo
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
u := user{}
|
|
|
|
form := url.Values{
|
2016-09-02 16:08:04 +08:00
|
|
|
"ID": []string{"1"},
|
|
|
|
"-": []string{"1"},
|
|
|
|
"tag": []string{"no"},
|
|
|
|
"username": []string{"test"},
|
|
|
|
"age": []string{"40"},
|
|
|
|
"Email": []string{"test@gmail.com"},
|
|
|
|
"Intro": []string{"I am an engineer!"},
|
|
|
|
"strbool": []string{"yes"},
|
|
|
|
"date": []string{"2014-11-12"},
|
|
|
|
"organization": []string{"beego"},
|
|
|
|
"title": []string{"CXO"},
|
2019-01-25 09:04:01 +07:00
|
|
|
"hobby": []string{"", "Basketball", "Football"},
|
2016-09-02 16:08:04 +08:00
|
|
|
"memo": []string{"nothing"},
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
|
|
|
if err := ParseForm(form, u); err == nil {
|
|
|
|
t.Fatal("nothing will be changed")
|
|
|
|
}
|
|
|
|
if err := ParseForm(form, &u); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2015-09-08 23:41:41 +08:00
|
|
|
if u.ID != 0 {
|
|
|
|
t.Errorf("ID should equal 0 but got %v", u.ID)
|
2013-07-25 22:50:29 +08:00
|
|
|
}
|
|
|
|
if len(u.tag) != 0 {
|
2013-08-10 11:42:25 +08:00
|
|
|
t.Errorf("tag's length should equal 0 but got %v", len(u.tag))
|
2013-07-25 22:50:29 +08:00
|
|
|
}
|
2013-07-25 22:25:12 +08:00
|
|
|
if u.Name.(string) != "test" {
|
2013-08-10 11:42:25 +08:00
|
|
|
t.Errorf("Name should equal `test` but got `%v`", u.Name.(string))
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
|
|
|
if u.Age != 40 {
|
2013-08-10 11:42:25 +08:00
|
|
|
t.Errorf("Age should equal 40 but got %v", u.Age)
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
|
|
|
if u.Email != "test@gmail.com" {
|
2013-08-10 11:42:25 +08:00
|
|
|
t.Errorf("Email should equal `test@gmail.com` but got `%v`", u.Email)
|
|
|
|
}
|
|
|
|
if u.Intro != "I am an engineer!" {
|
|
|
|
t.Errorf("Intro should equal `I am an engineer!` but got `%v`", u.Intro)
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
2017-03-17 19:24:45 +02:00
|
|
|
if !u.StrBool {
|
2014-11-04 16:39:17 +08:00
|
|
|
t.Errorf("strboll should equal `true`, but got `%v`", u.StrBool)
|
|
|
|
}
|
|
|
|
y, m, d := u.Date.Date()
|
|
|
|
if y != 2014 || m.String() != "November" || d != 12 {
|
|
|
|
t.Errorf("Date should equal `2014-11-12`, but got `%v`", u.Date.String())
|
|
|
|
}
|
2016-09-02 16:08:04 +08:00
|
|
|
if u.Organization != "beego" {
|
|
|
|
t.Errorf("Organization should equal `beego`, but got `%v`", u.Organization)
|
|
|
|
}
|
|
|
|
if u.Title != "CXO" {
|
|
|
|
t.Errorf("Title should equal `CXO`, but got `%v`", u.Title)
|
|
|
|
}
|
2019-01-25 09:04:01 +07:00
|
|
|
if u.Hobby[0] != "" {
|
|
|
|
t.Errorf("Hobby should equal ``, but got `%v`", u.Hobby[0])
|
2019-01-25 00:38:14 +07:00
|
|
|
}
|
2019-01-25 09:04:01 +07:00
|
|
|
if u.Hobby[1] != "Basketball" {
|
|
|
|
t.Errorf("Hobby should equal `Basketball`, but got `%v`", u.Hobby[1])
|
|
|
|
}
|
|
|
|
if u.Hobby[2] != "Football" {
|
|
|
|
t.Errorf("Hobby should equal `Football`, but got `%v`", u.Hobby[2])
|
2016-09-02 16:08:04 +08:00
|
|
|
}
|
|
|
|
if len(u.Memo) != 0 {
|
|
|
|
t.Errorf("Memo's length should equal 0 but got %v", len(u.Memo))
|
|
|
|
}
|
2013-07-25 22:25:12 +08:00
|
|
|
}
|
2013-08-10 16:33:46 +08:00
|
|
|
|
|
|
|
func TestRenderForm(t *testing.T) {
|
|
|
|
type user struct {
|
2015-09-08 23:41:41 +08:00
|
|
|
ID int `form:"-"`
|
2014-07-03 23:40:21 +08:00
|
|
|
Name interface{} `form:"username"`
|
|
|
|
Age int `form:"age,text,年龄:"`
|
|
|
|
Sex string
|
|
|
|
Email []string
|
|
|
|
Intro string `form:",textarea"`
|
2014-06-29 20:30:11 +02:00
|
|
|
Ignored string `form:"-"`
|
2013-08-10 16:33:46 +08:00
|
|
|
}
|
|
|
|
|
2014-07-03 23:40:21 +08:00
|
|
|
u := user{Name: "test", Intro: "Some Text"}
|
2013-08-10 16:33:46 +08:00
|
|
|
output := RenderForm(u)
|
|
|
|
if output != template.HTML("") {
|
|
|
|
t.Errorf("output should be empty but got %v", output)
|
|
|
|
}
|
|
|
|
output = RenderForm(&u)
|
|
|
|
result := template.HTML(
|
2013-08-11 22:42:35 +08:00
|
|
|
`Name: <input name="username" type="text" value="test"></br>` +
|
2013-08-12 06:46:40 +08:00
|
|
|
`年龄:<input name="age" type="text" value="0"></br>` +
|
2013-08-11 22:42:35 +08:00
|
|
|
`Sex: <input name="Sex" type="text" value=""></br>` +
|
2014-06-29 20:30:11 +02:00
|
|
|
`Intro: <textarea name="Intro">Some Text</textarea>`)
|
2013-08-10 16:33:46 +08:00
|
|
|
if output != result {
|
|
|
|
t.Errorf("output should equal `%v` but got `%v`", result, output)
|
|
|
|
}
|
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
2014-06-29 20:30:11 +02:00
|
|
|
func TestRenderFormField(t *testing.T) {
|
2016-06-28 16:24:32 -07:00
|
|
|
html := renderFormField("Label: ", "Name", "text", "Value", "", "", false)
|
2014-07-03 23:40:21 +08:00
|
|
|
if html != `Label: <input name="Name" type="text" value="Value">` {
|
|
|
|
t.Errorf("Wrong html output for input[type=text]: %v ", html)
|
|
|
|
}
|
|
|
|
|
2016-06-28 16:24:32 -07:00
|
|
|
html = renderFormField("Label: ", "Name", "textarea", "Value", "", "", false)
|
2014-07-03 23:40:21 +08:00
|
|
|
if html != `Label: <textarea name="Name">Value</textarea>` {
|
|
|
|
t.Errorf("Wrong html output for textarea: %v ", html)
|
|
|
|
}
|
2016-06-28 16:24:32 -07:00
|
|
|
|
|
|
|
html = renderFormField("Label: ", "Name", "textarea", "Value", "", "", true)
|
|
|
|
if html != `Label: <textarea name="Name" required>Value</textarea>` {
|
|
|
|
t.Errorf("Wrong html output for textarea: %v ", html)
|
|
|
|
}
|
2014-06-29 20:30:11 +02:00
|
|
|
}
|
|
|
|
|
2014-06-29 19:19:32 +02:00
|
|
|
func TestParseFormTag(t *testing.T) {
|
2014-07-03 23:40:21 +08:00
|
|
|
// create struct to contain field with different types of struct-tag `form`
|
2014-06-29 19:19:32 +02:00
|
|
|
type user struct {
|
2016-06-30 10:32:53 -07:00
|
|
|
All int `form:"name,text,年龄:"`
|
|
|
|
NoName int `form:",hidden,年龄:"`
|
|
|
|
OnlyLabel int `form:",,年龄:"`
|
|
|
|
OnlyName int `form:"name" id:"name" class:"form-name"`
|
|
|
|
Ignored int `form:"-"`
|
|
|
|
Required int `form:"name" required:"true"`
|
|
|
|
IgnoreRequired int `form:"name"`
|
|
|
|
NotRequired int `form:"name" required:"false"`
|
2014-07-03 23:40:21 +08:00
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
|
|
|
objT := reflect.TypeOf(&user{}).Elem()
|
|
|
|
|
2017-04-28 22:36:28 +08:00
|
|
|
label, name, fType, _, _, ignored, _ := parseFormTag(objT.Field(0))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "name" && label == "年龄:" && fType == "text" && !ignored) {
|
2014-06-29 19:19:32 +02:00
|
|
|
t.Errorf("Form Tag with name, label and type was not correctly parsed.")
|
2014-07-03 23:40:21 +08:00
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
2017-04-28 22:36:28 +08:00
|
|
|
label, name, fType, _, _, ignored, _ = parseFormTag(objT.Field(1))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "NoName" && label == "年龄:" && fType == "hidden" && !ignored) {
|
2014-06-29 19:19:32 +02:00
|
|
|
t.Errorf("Form Tag with label and type but without name was not correctly parsed.")
|
2014-07-03 23:40:21 +08:00
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
2017-04-28 22:36:28 +08:00
|
|
|
label, name, fType, _, _, ignored, _ = parseFormTag(objT.Field(2))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "OnlyLabel" && label == "年龄:" && fType == "text" && !ignored) {
|
2014-06-29 19:19:32 +02:00
|
|
|
t.Errorf("Form Tag containing only label was not correctly parsed.")
|
2014-07-03 23:40:21 +08:00
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
2017-04-28 22:36:28 +08:00
|
|
|
label, name, fType, id, class, ignored, _ := parseFormTag(objT.Field(3))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "name" && label == "OnlyName: " && fType == "text" && !ignored &&
|
2014-10-20 22:23:29 +08:00
|
|
|
id == "name" && class == "form-name") {
|
2014-07-03 23:40:21 +08:00
|
|
|
t.Errorf("Form Tag containing only name was not correctly parsed.")
|
|
|
|
}
|
2014-06-29 19:19:32 +02:00
|
|
|
|
2017-04-28 21:38:08 +08:00
|
|
|
_, _, _, _, _, ignored, _ = parseFormTag(objT.Field(4))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !ignored {
|
2014-07-03 23:40:21 +08:00
|
|
|
t.Errorf("Form Tag that should be ignored was not correctly parsed.")
|
|
|
|
}
|
2016-06-28 16:31:45 -07:00
|
|
|
|
2017-04-28 22:36:28 +08:00
|
|
|
_, name, _, _, _, _, required := parseFormTag(objT.Field(5))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "name" && required) {
|
2016-06-28 16:31:45 -07:00
|
|
|
t.Errorf("Form Tag containing only name and required was not correctly parsed.")
|
|
|
|
}
|
|
|
|
|
2017-04-28 21:38:08 +08:00
|
|
|
_, name, _, _, _, _, required = parseFormTag(objT.Field(6))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "name" && !required) {
|
2016-06-30 10:32:53 -07:00
|
|
|
t.Errorf("Form Tag containing only name and ignore required was not correctly parsed.")
|
|
|
|
}
|
|
|
|
|
2017-04-28 21:38:08 +08:00
|
|
|
_, name, _, _, _, _, required = parseFormTag(objT.Field(7))
|
2017-03-17 19:24:45 +02:00
|
|
|
if !(name == "name" && !required) {
|
2016-06-30 10:32:53 -07:00
|
|
|
t.Errorf("Form Tag containing only name and not required was not correctly parsed.")
|
|
|
|
}
|
|
|
|
|
2014-06-29 19:19:32 +02:00
|
|
|
}
|
2015-08-20 19:04:43 +05:00
|
|
|
|
|
|
|
func TestMapGet(t *testing.T) {
|
|
|
|
// test one level map
|
|
|
|
m1 := map[string]int64{
|
|
|
|
"a": 1,
|
|
|
|
"1": 2,
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err := MapGet(m1, "a"); err == nil {
|
|
|
|
if res.(int64) != 1 {
|
|
|
|
t.Errorf("Should return 1, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err := MapGet(m1, "1"); err == nil {
|
|
|
|
if res.(int64) != 2 {
|
|
|
|
t.Errorf("Should return 2, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err := MapGet(m1, 1); err == nil {
|
|
|
|
if res.(int64) != 2 {
|
|
|
|
t.Errorf("Should return 2, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// test 2 level map
|
2018-08-20 22:55:50 +02:00
|
|
|
m2 := M{
|
2015-08-20 19:04:43 +05:00
|
|
|
"1": map[string]float64{
|
|
|
|
"2": 3.5,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err := MapGet(m2, 1, 2); err == nil {
|
|
|
|
if res.(float64) != 3.5 {
|
|
|
|
t.Errorf("Should return 3.5, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// test 5 level map
|
2018-08-20 22:55:50 +02:00
|
|
|
m5 := M{
|
|
|
|
"1": M{
|
|
|
|
"2": M{
|
|
|
|
"3": M{
|
|
|
|
"4": M{
|
2015-08-20 19:04:43 +05:00
|
|
|
"5": 1.2,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if res, err := MapGet(m5, 1, 2, 3, 4, 5); err == nil {
|
|
|
|
if res.(float64) != 1.2 {
|
|
|
|
t.Errorf("Should return 1.2, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check whether element not exists in map
|
|
|
|
if res, err := MapGet(m5, 5, 4, 3, 2, 1); err == nil {
|
|
|
|
if res != nil {
|
|
|
|
t.Errorf("Should return nil, but return %v", res)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Errorf("Error happens %v", err)
|
|
|
|
}
|
|
|
|
}
|