161 lines
4.5 KiB
Go
161 lines
4.5 KiB
Go
|
|
package handler
|
|||
|
|
|
|||
|
|
import (
|
|||
|
|
"fmt"
|
|||
|
|
"net/http"
|
|||
|
|
"strings"
|
|||
|
|
"time"
|
|||
|
|
|
|||
|
|
"soul-api/internal/database"
|
|||
|
|
"soul-api/internal/model"
|
|||
|
|
"soul-api/internal/wechat"
|
|||
|
|
|
|||
|
|
"github.com/gin-gonic/gin"
|
|||
|
|
)
|
|||
|
|
|
|||
|
|
// WechatLogin POST /api/wechat/login
|
|||
|
|
func WechatLogin(c *gin.Context) {
|
|||
|
|
c.JSON(http.StatusOK, gin.H{"success": true})
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// WechatPhoneLoginReq 手机号登录请求:code 为 wx.login() 的 code,phoneCode 为 getPhoneNumber 返回的 code
|
|||
|
|
type WechatPhoneLoginReq struct {
|
|||
|
|
Code string `json:"code"` // wx.login() 得到,用于 code2session 拿 openId
|
|||
|
|
PhoneCode string `json:"phoneCode"` // getPhoneNumber 得到,用于换手机号
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
// WechatPhoneLogin POST /api/wechat/phone-login
|
|||
|
|
// 请求体:code(必填)+ phoneCode(必填)。先 code2session 得到 openId,再 getPhoneNumber 得到手机号,创建/更新用户并返回与 /api/miniprogram/login 一致的数据结构。
|
|||
|
|
func WechatPhoneLogin(c *gin.Context) {
|
|||
|
|
var req WechatPhoneLoginReq
|
|||
|
|
if err := c.ShouldBindJSON(&req); err != nil {
|
|||
|
|
c.JSON(http.StatusBadRequest, gin.H{"success": false, "error": "缺少 code 或 phoneCode"})
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
if req.Code == "" || req.PhoneCode == "" {
|
|||
|
|
c.JSON(http.StatusBadRequest, gin.H{"success": false, "error": "请提供 code 与 phoneCode"})
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
openID, sessionKey, _, err := wechat.Code2Session(req.Code)
|
|||
|
|
if err != nil {
|
|||
|
|
c.JSON(http.StatusBadRequest, gin.H{"success": false, "error": fmt.Sprintf("微信登录失败: %v", err)})
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
phoneNumber, countryCode, err := wechat.GetPhoneNumber(req.PhoneCode)
|
|||
|
|
if err != nil {
|
|||
|
|
c.JSON(http.StatusBadRequest, gin.H{"success": false, "error": fmt.Sprintf("获取手机号失败: %v", err)})
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
db := database.DB()
|
|||
|
|
var user model.User
|
|||
|
|
result := db.Where("open_id = ?", openID).First(&user)
|
|||
|
|
isNewUser := result.Error != nil
|
|||
|
|
|
|||
|
|
if isNewUser {
|
|||
|
|
referralCode := "SOUL" + strings.ToUpper(openID[len(openID)-6:])
|
|||
|
|
nickname := "微信用户" + openID[len(openID)-4:]
|
|||
|
|
avatar := ""
|
|||
|
|
hasFullBook := false
|
|||
|
|
earnings := 0.0
|
|||
|
|
pendingEarnings := 0.0
|
|||
|
|
referralCount := 0
|
|||
|
|
purchasedSections := "[]"
|
|||
|
|
phone := phoneNumber
|
|||
|
|
if countryCode != "" && countryCode != "86" {
|
|||
|
|
phone = "+" + countryCode + " " + phoneNumber
|
|||
|
|
}
|
|||
|
|
user = model.User{
|
|||
|
|
ID: openID,
|
|||
|
|
OpenID: &openID,
|
|||
|
|
SessionKey: &sessionKey,
|
|||
|
|
Nickname: &nickname,
|
|||
|
|
Avatar: &avatar,
|
|||
|
|
Phone: &phone,
|
|||
|
|
ReferralCode: &referralCode,
|
|||
|
|
HasFullBook: &hasFullBook,
|
|||
|
|
PurchasedSections: &purchasedSections,
|
|||
|
|
Earnings: &earnings,
|
|||
|
|
PendingEarnings: &pendingEarnings,
|
|||
|
|
ReferralCount: &referralCount,
|
|||
|
|
}
|
|||
|
|
if err := db.Create(&user).Error; err != nil {
|
|||
|
|
c.JSON(http.StatusInternalServerError, gin.H{"success": false, "error": "创建用户失败"})
|
|||
|
|
return
|
|||
|
|
}
|
|||
|
|
} else {
|
|||
|
|
phone := phoneNumber
|
|||
|
|
if countryCode != "" && countryCode != "86" {
|
|||
|
|
phone = "+" + countryCode + " " + phoneNumber
|
|||
|
|
}
|
|||
|
|
db.Model(&user).Updates(map[string]interface{}{"session_key": sessionKey, "phone": phone})
|
|||
|
|
user.Phone = &phone
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
var orderRows []struct {
|
|||
|
|
ProductID string `gorm:"column:product_id"`
|
|||
|
|
}
|
|||
|
|
db.Raw(`
|
|||
|
|
SELECT DISTINCT product_id FROM orders WHERE user_id = ? AND status = 'paid' AND product_type = 'section'
|
|||
|
|
`, user.ID).Scan(&orderRows)
|
|||
|
|
purchasedSections := []string{}
|
|||
|
|
for _, row := range orderRows {
|
|||
|
|
if row.ProductID != "" {
|
|||
|
|
purchasedSections = append(purchasedSections, row.ProductID)
|
|||
|
|
}
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
responseUser := map[string]interface{}{
|
|||
|
|
"id": user.ID,
|
|||
|
|
"openId": strVal(user.OpenID),
|
|||
|
|
"nickname": strVal(user.Nickname),
|
|||
|
|
"avatar": strVal(user.Avatar),
|
|||
|
|
"phone": strVal(user.Phone),
|
|||
|
|
"wechatId": strVal(user.WechatID),
|
|||
|
|
"referralCode": strVal(user.ReferralCode),
|
|||
|
|
"hasFullBook": boolVal(user.HasFullBook),
|
|||
|
|
"purchasedSections": purchasedSections,
|
|||
|
|
"earnings": floatVal(user.Earnings),
|
|||
|
|
"pendingEarnings": floatVal(user.PendingEarnings),
|
|||
|
|
"referralCount": intVal(user.ReferralCount),
|
|||
|
|
"createdAt": user.CreatedAt,
|
|||
|
|
}
|
|||
|
|
token := fmt.Sprintf("tk_%s_%d", openID[len(openID)-8:], time.Now().Unix())
|
|||
|
|
|
|||
|
|
c.JSON(http.StatusOK, gin.H{
|
|||
|
|
"success": true,
|
|||
|
|
"data": map[string]interface{}{
|
|||
|
|
"openId": openID,
|
|||
|
|
"user": responseUser,
|
|||
|
|
"token": token,
|
|||
|
|
},
|
|||
|
|
"isNewUser": isNewUser,
|
|||
|
|
})
|
|||
|
|
}
|
|||
|
|
|
|||
|
|
func strVal(p *string) string {
|
|||
|
|
if p == nil {
|
|||
|
|
return ""
|
|||
|
|
}
|
|||
|
|
return *p
|
|||
|
|
}
|
|||
|
|
func boolVal(p *bool) bool {
|
|||
|
|
if p == nil {
|
|||
|
|
return false
|
|||
|
|
}
|
|||
|
|
return *p
|
|||
|
|
}
|
|||
|
|
func floatVal(p *float64) float64 {
|
|||
|
|
if p == nil {
|
|||
|
|
return 0
|
|||
|
|
}
|
|||
|
|
return *p
|
|||
|
|
}
|
|||
|
|
func intVal(p *int) int {
|
|||
|
|
if p == nil {
|
|||
|
|
return 0
|
|||
|
|
}
|
|||
|
|
return *p
|
|||
|
|
}
|