449 lines
12 KiB
Go
449 lines
12 KiB
Go
|
package member
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"fmt"
|
||
|
"net/url"
|
||
|
"time"
|
||
|
"unicode/utf8"
|
||
|
|
||
|
"go-common/app/interface/main/account/model"
|
||
|
accModel "go-common/app/service/main/account/model"
|
||
|
coModel "go-common/app/service/main/coin/model"
|
||
|
ftModel "go-common/app/service/main/filter/model/rpc"
|
||
|
locModel "go-common/app/service/main/location/model"
|
||
|
meModel "go-common/app/service/main/member/model"
|
||
|
"go-common/library/ecode"
|
||
|
"go-common/library/log"
|
||
|
"go-common/library/net/metadata"
|
||
|
"go-common/library/queue/databus/report"
|
||
|
|
||
|
"github.com/pkg/errors"
|
||
|
)
|
||
|
|
||
|
var monitoredIPCountry = map[string]string{
|
||
|
"台湾": "台湾 IP",
|
||
|
"香港": "香港 IP",
|
||
|
"美国": "美国 IP",
|
||
|
"加拿大": "加拿大 IP",
|
||
|
}
|
||
|
|
||
|
var monitoredTelCountry = map[int64]string{
|
||
|
1: "美国/加拿大手机号",
|
||
|
}
|
||
|
|
||
|
var upNameCostCoins = 6.0
|
||
|
|
||
|
// Account get Account info.
|
||
|
func (s *Service) Account(c context.Context, mid int64, ip string) (acc *model.Account, err error) {
|
||
|
var (
|
||
|
nickFree *model.NickFree
|
||
|
mb *meModel.Member
|
||
|
)
|
||
|
marg := &meModel.ArgMemberMid{Mid: mid, RemoteIP: ip}
|
||
|
if mb, err = s.memRPC.Member(c, marg); err != nil {
|
||
|
log.Error("service.memberRPC.MyInfo(%v) error(%v)", marg, err)
|
||
|
return
|
||
|
}
|
||
|
if nickFree, err = s.NickFree(c, mid); err != nil {
|
||
|
return
|
||
|
}
|
||
|
acc = &model.Account{}
|
||
|
acc.Mid = mid
|
||
|
acc.Birthday = mb.Birthday.Time().Format("2006-01-02")
|
||
|
acc.Uname = mb.Name
|
||
|
acc.Face = mb.Face
|
||
|
acc.Sign = mb.Sign
|
||
|
acc.Sex = int8(mb.Sex)
|
||
|
acc.NickFree = nickFree.NickFree
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// UpdateFace Update Face
|
||
|
func (s *Service) UpdateFace(c context.Context, mid int64, faceFile []byte, ftype string) (string, error) {
|
||
|
ip := metadata.String(c, metadata.RemoteIP)
|
||
|
// check
|
||
|
profile, err := s.accRPC.Profile3(c, &accModel.ArgMid{Mid: mid})
|
||
|
if err != nil {
|
||
|
return "", errors.WithStack(err)
|
||
|
}
|
||
|
//判断是否绑定手机号
|
||
|
if !s.validateTelStatus(profile.TelStatus) {
|
||
|
return "", ecode.MemberPhoneRequired
|
||
|
}
|
||
|
if profile.Silence != 0 {
|
||
|
return "", ecode.MemberBlocked
|
||
|
}
|
||
|
//Upload bfs
|
||
|
faceURL, err := s.accDao.UploadImage(c, ftype, faceFile, s.c.FaceBFS)
|
||
|
if err != nil {
|
||
|
log.Error("s.bfsDao.Upload(%d) error(%v)", mid, err)
|
||
|
return "", errors.WithStack(err)
|
||
|
}
|
||
|
URL, err := url.Parse(faceURL)
|
||
|
if err != nil {
|
||
|
return "", errors.WithStack(err)
|
||
|
}
|
||
|
inMonitor := s.ensureMonitor(c, mid, ip)
|
||
|
arg := &meModel.ArgAddPropertyReview{
|
||
|
Mid: mid,
|
||
|
New: URL.Path,
|
||
|
State: meModel.ReviewStateQueuing,
|
||
|
Property: meModel.ReviewPropertyFace,
|
||
|
}
|
||
|
if inMonitor {
|
||
|
arg.State = meModel.ReviewStateWait
|
||
|
return profile.Face, s.memRPC.AddPropertyReview(c, arg)
|
||
|
}
|
||
|
if err := s.memRPC.AddPropertyReview(c, arg); err != nil {
|
||
|
log.Error("s.memRPC.AddPropertyReview(%d,%s) error(%v)", mid, faceFile, err)
|
||
|
return "", errors.WithStack(err)
|
||
|
}
|
||
|
if err := s.memRPC.SetFace(c, &meModel.ArgUpdateFace{Mid: mid, Face: URL.Path}); err != nil {
|
||
|
log.Error("s.memRPC.SetFace(%d,%s) error(%v)", mid, faceURL, err)
|
||
|
return "", errors.WithStack(err)
|
||
|
}
|
||
|
return faceURL, nil
|
||
|
}
|
||
|
|
||
|
// UpdateName .
|
||
|
func (s *Service) UpdateName(c context.Context, mid int64, name, appkey string) error {
|
||
|
ip := metadata.String(c, metadata.RemoteIP)
|
||
|
_, inWhiteList := s.nickFreeAppKeys[appkey]
|
||
|
if inWhiteList {
|
||
|
return s.updateNameWithinWhiteList(c, mid, name, ip)
|
||
|
}
|
||
|
return s.updateName(c, mid, name, ip)
|
||
|
}
|
||
|
|
||
|
// updateNameWithinWhiteList 白名单 appkey 不扣硬币
|
||
|
func (s *Service) updateNameWithinWhiteList(c context.Context, mid int64, name, ip string) error {
|
||
|
if err := s.nameIsValid(c, mid, name, ip); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
profile, err := s.accRPC.Profile3(c, &accModel.ArgMid{Mid: mid})
|
||
|
if err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
if err := s.permitName(c, profile, ip); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if profile.Name == name {
|
||
|
log.Info("Update name is same to origin: mid: %d, name: %s, origin: %s", mid, name, profile.Name)
|
||
|
return nil
|
||
|
}
|
||
|
inMonitor := s.ensureMonitor(c, mid, ip)
|
||
|
remark := "appkey白名单修改昵称"
|
||
|
// 在监控列表里就加入添加审核列表
|
||
|
if inMonitor {
|
||
|
saveUpNameLog(mid, profile.Name, name, remark, inMonitor, ip)
|
||
|
return errors.WithStack(s.memRPC.AddPropertyReview(c, &meModel.ArgAddPropertyReview{
|
||
|
Mid: mid,
|
||
|
New: name,
|
||
|
State: meModel.ReviewStateWait,
|
||
|
Property: meModel.ReviewPropertyName,
|
||
|
Extra: map[string]interface{}{"nick_free": true},
|
||
|
}))
|
||
|
}
|
||
|
//修改昵称
|
||
|
if err := s.passDao.UpdateName(c, mid, name, ip); err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
saveUpNameLog(mid, profile.Name, name, remark, inMonitor, ip)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
//UpdateName update name.
|
||
|
func (s *Service) updateName(c context.Context, mid int64, name, ip string) error {
|
||
|
if err := s.nameIsValid(c, mid, name, ip); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
profile, err := s.accRPC.Profile3(c, &accModel.ArgMid{Mid: mid})
|
||
|
if err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
if err = s.permitName(c, profile, ip); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if profile.Name == name {
|
||
|
log.Info("Update name is same to origin: mid: %d, name: %s, origin: %s", mid, name, profile.Name)
|
||
|
return nil
|
||
|
}
|
||
|
// 判断是否改昵称免费
|
||
|
nickFree, err := s.NickFree(c, mid)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
remark := "快速注册修改昵称"
|
||
|
if !nickFree.NickFree {
|
||
|
coins, coinErr := s.coinRPC.UserCoins(c, &coModel.ArgCoinInfo{Mid: mid, RealIP: ip})
|
||
|
if coinErr != nil {
|
||
|
return errors.WithStack(coinErr)
|
||
|
}
|
||
|
if coins < upNameCostCoins {
|
||
|
return ecode.UpdateUnameMoneyIsNot
|
||
|
}
|
||
|
remark = "修改昵称"
|
||
|
}
|
||
|
inMonitor := s.ensureMonitor(c, mid, ip)
|
||
|
// 在监控列表里就加入添加审核列表
|
||
|
if inMonitor {
|
||
|
saveUpNameLog(mid, profile.Name, name, remark, inMonitor, ip)
|
||
|
return errors.WithStack(s.memRPC.AddPropertyReview(c, &meModel.ArgAddPropertyReview{
|
||
|
Mid: mid,
|
||
|
New: name,
|
||
|
State: meModel.ReviewStateWait,
|
||
|
Property: meModel.ReviewPropertyName,
|
||
|
Extra: map[string]interface{}{"nick_free": nickFree.NickFree},
|
||
|
}))
|
||
|
}
|
||
|
//修改昵称
|
||
|
if err = s.passDao.UpdateName(c, mid, name, ip); err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
saveUpNameLog(mid, profile.Name, name, remark, inMonitor, ip)
|
||
|
if nickFree.NickFree {
|
||
|
return errors.WithStack(s.memRPC.SetNickUpdated(c, &meModel.ArgMemberMid{Mid: mid}))
|
||
|
}
|
||
|
//扣除硬币
|
||
|
if _, err = s.coinRPC.ModifyCoin(c, &coModel.ArgModifyCoin{
|
||
|
Mid: mid,
|
||
|
Count: -upNameCostCoins,
|
||
|
Reason: fmt.Sprintf("UPDATE:NICK:%s=>%s", profile.Name, name),
|
||
|
IP: ip,
|
||
|
}); err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *Service) monitorByIP(ctx context.Context, mid int64, ip string) (bool, string) {
|
||
|
IP, err := s.locRPC.Info(ctx, &locModel.ArgIP{IP: ip})
|
||
|
if err != nil || IP == nil {
|
||
|
log.Error("Failed to get ip info with ip: %s: %+v", ip, err)
|
||
|
return false, ""
|
||
|
}
|
||
|
descr, shouldMonitor := monitoredIPCountry[IP.Country]
|
||
|
if !shouldMonitor {
|
||
|
return false, ""
|
||
|
}
|
||
|
return true, descr
|
||
|
}
|
||
|
|
||
|
func (s *Service) monitorByTel(ctx context.Context, mid int64, ip string) (bool, string) {
|
||
|
p, err := s.passDao.QueryByMid(ctx, mid, ip)
|
||
|
if err != nil {
|
||
|
log.Error("Failed to query by mid form pasport: mid: %d: %+v", mid, err)
|
||
|
return false, ""
|
||
|
}
|
||
|
descr, shouldMonitor := monitoredTelCountry[p.CountryCode]
|
||
|
if !shouldMonitor {
|
||
|
return false, ""
|
||
|
}
|
||
|
return true, descr
|
||
|
}
|
||
|
|
||
|
func (s *Service) shouldMonitor(ctx context.Context, mid int64, ip string) (bool, string) {
|
||
|
should, descr := s.monitorByIP(ctx, mid, ip)
|
||
|
if should {
|
||
|
return true, descr
|
||
|
}
|
||
|
|
||
|
should, descr = s.monitorByTel(ctx, mid, ip)
|
||
|
if should {
|
||
|
return true, descr
|
||
|
}
|
||
|
|
||
|
return false, ""
|
||
|
}
|
||
|
|
||
|
func (s *Service) ensureMonitor(ctx context.Context, mid int64, ip string) bool {
|
||
|
inMonitor, _ := s.memRPC.IsInMonitor(ctx, &meModel.ArgMid{Mid: mid})
|
||
|
if inMonitor {
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
should, descr := s.shouldMonitor(ctx, mid, ip)
|
||
|
if !should {
|
||
|
return false
|
||
|
}
|
||
|
if err := s.memRPC.AddUserMonitor(ctx, &meModel.ArgAddUserMonitor{
|
||
|
Mid: mid,
|
||
|
Operator: "system",
|
||
|
Remark: fmt.Sprintf("系统自动导入-%s", descr),
|
||
|
}); err != nil {
|
||
|
log.Error("Failed to add user moniter: mid: %d: %+v", mid, err)
|
||
|
}
|
||
|
return true
|
||
|
}
|
||
|
|
||
|
// UpdateSex update sex.
|
||
|
func (s *Service) UpdateSex(c context.Context, mid, sex int64) (err error) {
|
||
|
ip := metadata.String(c, metadata.RemoteIP)
|
||
|
return s.accDao.UpdateSex(c, mid, sex, ip)
|
||
|
}
|
||
|
|
||
|
//UpdateSign update sign.
|
||
|
func (s *Service) UpdateSign(c context.Context, mid int64, sign string) error {
|
||
|
ip := metadata.String(c, metadata.RemoteIP)
|
||
|
// 签名最长 70 个字符
|
||
|
if utf8.RuneCountInString(sign) > 70 {
|
||
|
return ecode.MemberSignOverLimit
|
||
|
}
|
||
|
// 签名不能包含 emoji
|
||
|
if model.HasEmoji(sign) {
|
||
|
return ecode.MemberSignHasEmoji
|
||
|
}
|
||
|
|
||
|
// 过滤敏感词
|
||
|
res, err := s.filterRPC.Filter(c, &ftModel.ArgFilter{Area: "sign", Message: sign})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// 大于 20 认为包含敏感词
|
||
|
if res.Level >= 20 {
|
||
|
return ecode.MemberSignSensitive
|
||
|
}
|
||
|
|
||
|
// 检查是否绑定手机
|
||
|
profile, err := s.accRPC.Profile3(c, &accModel.ArgMid{Mid: mid})
|
||
|
if err != nil {
|
||
|
return errors.WithStack(err)
|
||
|
}
|
||
|
if !s.validateTelStatus(profile.TelStatus) {
|
||
|
return ecode.MemberPhoneRequired
|
||
|
}
|
||
|
// 检查是否被禁言
|
||
|
if profile.Silence != 0 {
|
||
|
return ecode.MemberBlocked
|
||
|
}
|
||
|
// 如果和老的一模一样就没必要更新了
|
||
|
if profile.Sign == sign {
|
||
|
log.Info("Update sign is same to origin: mid: %d, sign: %s, origin: %s", mid, sign, profile.Sign)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
inMonitor := s.ensureMonitor(c, mid, ip)
|
||
|
// 不在监控列表里就直接更新
|
||
|
if !inMonitor {
|
||
|
return errors.WithStack(s.memRPC.SetSign(c, &meModel.ArgUpdateSign{
|
||
|
Mid: mid,
|
||
|
Sign: sign,
|
||
|
RemoteIP: ip,
|
||
|
}))
|
||
|
}
|
||
|
// 否则就加入监控列表
|
||
|
return errors.WithStack(s.memRPC.AddPropertyReview(c, &meModel.ArgAddPropertyReview{
|
||
|
Mid: mid,
|
||
|
New: sign,
|
||
|
State: meModel.ReviewStateWait,
|
||
|
Property: meModel.ReviewPropertySign,
|
||
|
}))
|
||
|
}
|
||
|
|
||
|
// UpdateBirthday update birthday.
|
||
|
func (s *Service) UpdateBirthday(c context.Context, mid int64, birthday string) (err error) {
|
||
|
ip := metadata.String(c, metadata.RemoteIP)
|
||
|
return s.accDao.UpdateBirthday(c, mid, ip, birthday)
|
||
|
}
|
||
|
|
||
|
// NickFree .
|
||
|
func (s *Service) NickFree(c context.Context, mid int64) (nickFree *model.NickFree, err error) {
|
||
|
var (
|
||
|
isRegFast bool
|
||
|
nickUpdated bool
|
||
|
ip = metadata.String(c, metadata.RemoteIP)
|
||
|
)
|
||
|
sarg := &meModel.ArgMemberMid{Mid: mid}
|
||
|
if nickUpdated, err = s.memRPC.NickUpdated(c, sarg); err != nil {
|
||
|
log.Error("s.memRPC.IsUpNickFree(%v) error (%v)", sarg, err)
|
||
|
return
|
||
|
}
|
||
|
nickFree = &model.NickFree{}
|
||
|
if nickUpdated {
|
||
|
return
|
||
|
}
|
||
|
if isRegFast, err = s.passDao.FastReg(c, mid, ip); err != nil {
|
||
|
return
|
||
|
}
|
||
|
if isRegFast {
|
||
|
nickFree.NickFree = true
|
||
|
}
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func saveUpNameLog(mid int64, oName, nName, remark string, isMonitor bool, ip string) {
|
||
|
report.User(&report.UserInfo{
|
||
|
Mid: mid,
|
||
|
Business: model.UpNameLogID,
|
||
|
Action: model.UpNameAction,
|
||
|
IP: ip,
|
||
|
Ctime: time.Now(),
|
||
|
Index: []interface{}{0, 0, 0, oName, nName, remark},
|
||
|
Content: map[string]interface{}{
|
||
|
"is_monitor": isMonitor,
|
||
|
"old_name": oName,
|
||
|
"new_name": nName,
|
||
|
"reason": fmt.Sprintf("修改昵称(原昵称:%s 新昵称:%s)", oName, nName),
|
||
|
"remark": remark,
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func (s *Service) permitName(c context.Context, profile *accModel.Profile, ip string) error {
|
||
|
if !s.validateTelStatus(profile.TelStatus) {
|
||
|
return ecode.MemberPhoneRequired
|
||
|
}
|
||
|
// 检查是否被禁言
|
||
|
if profile.Silence != 0 {
|
||
|
return ecode.MemberBlocked
|
||
|
}
|
||
|
//昵称锁定,是否官方认证
|
||
|
if profile.Official.Role != 0 {
|
||
|
log.Info("update name fail, name is official, mid: %d", profile.Mid)
|
||
|
return ecode.UpdateUnameHadOfficial
|
||
|
}
|
||
|
pProfile, err := s.passDao.QueryByMid(c, profile.Mid, ip)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if pProfile.NickLock == 1 {
|
||
|
log.Info("update name fail, name is locked, mid: %d", profile.Mid)
|
||
|
return ecode.UpdateUnameHadLocked
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *Service) nameIsValid(c context.Context, mid int64, name, ip string) error {
|
||
|
if len(name) > 30 || utf8.RuneCountInString(name) > 16 {
|
||
|
return ecode.UpdateUnameTooLong
|
||
|
}
|
||
|
if utf8.RuneCountInString(name) < 3 {
|
||
|
return ecode.UpdateUnameTooShort
|
||
|
}
|
||
|
if !model.ValidName(name) {
|
||
|
return ecode.UpdateUnameFormat
|
||
|
}
|
||
|
// 判断昵称是否重复
|
||
|
if err := s.passDao.TestUserName(c, name, mid, ip); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// 过滤敏感词
|
||
|
res, err := s.filterRPC.Filter(c, &ftModel.ArgFilter{Area: "member", Message: name})
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// 大于 20 认为包含敏感词
|
||
|
if res.Level >= 20 {
|
||
|
return ecode.UpdateUnameSensitive
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *Service) validateTelStatus(status int32) bool {
|
||
|
if s.c.Switch.UpdatePropertyPhoneRequired && status == 0 {
|
||
|
return false
|
||
|
}
|
||
|
return true
|
||
|
}
|