SendSmsCode.go 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. package youngee_talent_service
  2. import (
  3. "bytes"
  4. "crypto/sha256"
  5. "crypto/tls"
  6. "encoding/base64"
  7. "encoding/json"
  8. "fmt"
  9. openapi "github.com/alibabacloud-go/darabonba-openapi/v2/client"
  10. dysmsapi20170525 "github.com/alibabacloud-go/dysmsapi-20170525/v4/client"
  11. util "github.com/alibabacloud-go/tea-utils/v2/service"
  12. "github.com/alibabacloud-go/tea/tea"
  13. "github.com/gogf/gf/frame/g"
  14. "github.com/gogf/gf/net/ghttp"
  15. uuid "github.com/satori/go.uuid"
  16. "io/ioutil"
  17. "math/rand"
  18. "net/http"
  19. "net/url"
  20. "strings"
  21. "time"
  22. "youngmini_server/app/model/youngee_talent_model"
  23. "youngmini_server/app/utils"
  24. )
  25. // 无需修改,用于格式化鉴权头域,给"X-WSSE"参数赋值
  26. const WSSE_HEADER_FORMAT = "UsernameToken Username=\"%s\",PasswordDigest=\"%s\",Nonce=\"%s\",Created=\"%s\""
  27. // 无需修改,用于格式化鉴权头域,给"Authorization"参数赋值
  28. const AUTH_HEADER_VALUE = "WSSE realm=\"SDP\",profile=\"UsernameToken\",type=\"Appkey\""
  29. type SendcodeReq struct {
  30. Phone string `json:"phone"`
  31. }
  32. // 接口调用此函数
  33. func SendSmsCode(r *ghttp.Request) *TalentHttpResult {
  34. l := SendcodeReq{}
  35. err := r.ParseForm(&l)
  36. if err != nil {
  37. return &TalentHttpResult{Code: -1, Msg: "请求体解析失败"}
  38. }
  39. vcode := GetCode()
  40. err = SendCodeImpAli(l.Phone, vcode)
  41. if err != nil {
  42. return &TalentHttpResult{Code: -1, Msg: "短信发送失败"}
  43. }
  44. //验证码存redis
  45. _ = SetRedis(l.Phone, vcode)
  46. return &TalentHttpResult{Code: 0, Msg: "sendSmsCodeSuccess", Data: vcode}
  47. }
  48. func SetRedis(phone string, vcode string) error {
  49. _, err := g.Redis().Do("SETEX", getRedisKey(phone), 120, vcode) //第三个参数以秒为单位
  50. if err != nil {
  51. fmt.Println("redis存储错误")
  52. }
  53. return nil
  54. }
  55. func getRedisKey(phone string) string {
  56. return fmt.Sprintf("%s%s", "c_user:", phone)
  57. }
  58. // 发送验证码具体实现
  59. //func SendCodeImp(phone string, vcode string) error {
  60. // //必填,请参考"开发准备"获取如下数据,替换为实际值
  61. // apiAddress := "https://smsapi.cn-south-1.myhuaweicloud.com:443/sms/batchSendSms/v1" //APP接入地址(在控制台"应用管理"页面获取)+接口访问URI
  62. // appKey := "NETTvTJJie9ax03v9K5T4DFB9EV6" //APP_Key
  63. // appSecret := "txi9kXIrxW0dVNMyAulrJf7XFNP7" //APP_Secret
  64. // sender := "8823022707732" //国内短信签名通道号或国际/港澳台短信通道号
  65. // templateId := "7103cdd480d14d0aa8c68954a7dbeb6e" //模板ID
  66. //
  67. // //条件必填,国内短信关注,当templateId指定的模板类型为通用模板时生效且必填,必须是已审核通过的,与模板类型一致的签名名称
  68. // //国际/港澳台短信不用关注该参数
  69. // signature := "样叽" //签名名称
  70. //
  71. // //必填,全局号码格式(包含国家码),示例:+86151****6789,多个号码之间用英文逗号分隔
  72. // receiver := "+86" + phone //短信接收人号码
  73. //
  74. // //选填,短信状态报告接收地址,推荐使用域名,为空或者不填表示不接收状态报告
  75. // statusCallBack := ""
  76. //
  77. // /*
  78. // * 选填,使用无变量模板时请赋空值 string templateParas = "";
  79. // * 单变量模板示例:模板内容为"您的验证码是${1}"时,templateParas可填写为"[\"369751\"]"
  80. // * 双变量模板示例:模板内容为"您有${1}件快递请到${2}领取"时,templateParas可填写为"[\"3\",\"人民公园正门\"]"
  81. // * 模板中的每个变量都必须赋值,且取值不能为空
  82. // * 查看更多模板和变量规范:产品介绍>模板和变量规范
  83. // */
  84. // templateParas := "[\"" + vcode + "\"]" //模板变量,此处以单变量验证码短信为例,请客户自行生成6位验证码,并定义为字符串类型,以杜绝首位0丢失的问题(例如:002569变成了2569)。
  85. // // templateParas := "[\"12345678\"]" //模板变量,此处以单变量验证码短信为例,请客户自行生成6位验证码,并定义为字符串类型,以杜绝首位0丢失的问题(例如:002569变成了2569)。
  86. //
  87. // body := buildRequestBody(sender, receiver, templateId, templateParas, statusCallBack, signature)
  88. // headers := make(map[string]string)
  89. // headers["Content-Type"] = "application/x-www-form-urlencoded"
  90. // headers["Authorization"] = AUTH_HEADER_VALUE
  91. // headers["X-WSSE"] = buildWsseHeader(appKey, appSecret)
  92. // resp, err := post(apiAddress, []byte(body), headers)
  93. // if err != nil {
  94. // return err
  95. // }
  96. // fmt.Println(resp)
  97. // return nil
  98. //}
  99. func SendCodeImpAli(phone string, vcode string) error {
  100. // 1. 创建客户端
  101. config := &openapi.Config{
  102. AccessKeyId: tea.String("LTAI5tDfTQeZ2ufN8mhsWQNA"),
  103. AccessKeySecret: tea.String("DKLd2hGbmJPcNki8p6VlU6mV2HXVvO"),
  104. Endpoint: tea.String("dysmsapi.aliyuncs.com"),
  105. }
  106. client, err := dysmsapi20170525.NewClient(config)
  107. if err != nil {
  108. return fmt.Errorf("创建客户端失败: %v", err)
  109. }
  110. // 2. 准备请求
  111. sendSmsRequest := &dysmsapi20170525.SendSmsRequest{
  112. PhoneNumbers: tea.String(phone),
  113. SignName: tea.String("北京智子时空科技"),
  114. TemplateCode: tea.String("SMS_481650308"),
  115. TemplateParam: tea.String(fmt.Sprintf(`{"code":"%s"}`, vcode)),
  116. }
  117. // 3. 发送请求
  118. runtime := &util.RuntimeOptions{}
  119. resp, err := client.SendSmsWithOptions(sendSmsRequest, runtime)
  120. if err != nil {
  121. // 错误处理
  122. if sdkErr, ok := err.(*tea.SDKError); ok {
  123. var data interface{}
  124. if err := json.NewDecoder(strings.NewReader(tea.StringValue(sdkErr.Data))).Decode(&data); err == nil {
  125. if m, ok := data.(map[string]interface{}); ok {
  126. if recommend, exists := m["Recommend"]; exists {
  127. return fmt.Errorf("短信发送失败: %s, 建议: %v", tea.StringValue(sdkErr.Message), recommend)
  128. }
  129. }
  130. }
  131. return fmt.Errorf("短信发送失败: %s", tea.StringValue(sdkErr.Message))
  132. }
  133. return fmt.Errorf("短信发送失败: %v", err)
  134. }
  135. // 4. 响应
  136. fmt.Printf("短信发送成功,响应: %v\n", resp)
  137. return nil
  138. }
  139. func buildRequestBody(sender, receiver, templateId, templateParas, statusCallBack, signature string) string {
  140. param := "from=" + url.QueryEscape(sender) + "&to=" + url.QueryEscape(receiver) + "&templateId=" + url.QueryEscape(templateId)
  141. if templateParas != "" {
  142. param += "&templateParas=" + url.QueryEscape(templateParas)
  143. }
  144. if statusCallBack != "" {
  145. param += "&statusCallback=" + url.QueryEscape(statusCallBack)
  146. }
  147. if signature != "" {
  148. param += "&signature=" + url.QueryEscape(signature)
  149. }
  150. return param
  151. }
  152. func post(url string, param []byte, headers map[string]string) (string, error) {
  153. tr := &http.Transport{
  154. TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
  155. }
  156. client := &http.Client{Transport: tr}
  157. req, err := http.NewRequest("POST", url, bytes.NewBuffer(param))
  158. if err != nil {
  159. return "", err
  160. }
  161. for key, header := range headers {
  162. req.Header.Set(key, header)
  163. }
  164. resp, err := client.Do(req)
  165. defer resp.Body.Close()
  166. body, err := ioutil.ReadAll(resp.Body)
  167. if err != nil {
  168. return "", err
  169. }
  170. return string(body), nil
  171. }
  172. func buildWsseHeader(appKey, appSecret string) string {
  173. var cTime = time.Now().Format("2006-01-02T15:04:05Z")
  174. var nonce = uuid.NewV4().String()
  175. nonce = strings.ReplaceAll(nonce, "-", "")
  176. h := sha256.New()
  177. h.Write([]byte(nonce + cTime + appSecret))
  178. passwordDigestBase64Str := base64.StdEncoding.EncodeToString(h.Sum(nil))
  179. return fmt.Sprintf(WSSE_HEADER_FORMAT, appKey, passwordDigestBase64Str, nonce, cTime)
  180. }
  181. // 随机生成6位数验证码
  182. func GetCode() string {
  183. rnd := rand.New(rand.NewSource(time.Now().UnixNano()))
  184. vcode := fmt.Sprintf("%06v", rnd.Int31n(1000000))
  185. return vcode
  186. }
  187. type ChangeReq struct {
  188. Phone string `json:"phone"`
  189. Vcode string `json:"vcode"`
  190. }
  191. // 更换手机号
  192. func ChangePhone(r *ghttp.Request) *TalentHttpResult {
  193. tid, err := utils.SessionTalentInfo.GetTalentIdFromSession(r)
  194. if err != nil {
  195. return &TalentHttpResult{Code: -1, Msg: "Get talent info failed"}
  196. }
  197. l := ChangeReq{}
  198. err = r.ParseForm(&l)
  199. if err != nil {
  200. fmt.Printf("前端数据解析错误")
  201. }
  202. //手机号是否被绑定
  203. existingTalent := &youngee_talent_model.TalentInfo{}
  204. record, err := g.DB().Model(existingTalent).Where("talent_phone_number = ?", l.Phone).One()
  205. if err != nil {
  206. fmt.Println("查询错误:", err)
  207. return &TalentHttpResult{Code: -3, Msg: "查询失败"}
  208. }
  209. // 检查是否找到了记录
  210. if record != nil {
  211. fmt.Println("手机号已存在:", record["talent_phone_number"])
  212. return &TalentHttpResult{Code: -1, Msg: "手机号已绑定"}
  213. } else {
  214. VcodeKey := fmt.Sprintf("%s%s", "c_user:", l.Phone)
  215. Vcode, _ := g.Redis().DoVar("GET", VcodeKey)
  216. //验证码正确
  217. if Vcode.String() == l.Vcode {
  218. //修改youngee_talent_info表中手机号
  219. _, err := g.DB().Model(existingTalent).Where("id=?", tid).Update(g.Map{"talent_phone_number": l.Phone})
  220. if err != nil {
  221. fmt.Println("更新失败")
  222. return &TalentHttpResult{Code: -4, Msg: "手机号更新失败"}
  223. }
  224. return &TalentHttpResult{Code: 0, Msg: "更新成功"}
  225. } else {
  226. return &TalentHttpResult{Code: -2, Msg: "验证码错误"}
  227. }
  228. }
  229. }