// Package sys // @Link https://github.com/bufanyun/hotgo // @Copyright Copyright (c) 2022 HotGo CLI // @Author Ms <133814250@qq.com> // @License https://github.com/bufanyun/hotgo/blob/master/LICENSE // package sys import ( "context" "github.com/gogf/gf/v2/errors/gerror" "github.com/gogf/gf/v2/frame/g" "github.com/gogf/gf/v2/net/ghttp" "github.com/gogf/gf/v2/os/gtime" "github.com/gogf/gf/v2/util/grand" "hotgo/internal/consts" "hotgo/internal/dao" "hotgo/internal/library/location" "hotgo/internal/library/sms/aliyun" "hotgo/internal/model" "hotgo/internal/model/entity" "hotgo/internal/model/input/sysin" "hotgo/internal/service" "hotgo/utility/validate" "time" ) type sSysSmsLog struct{} func NewSysSmsLog() *sSysSmsLog { return &sSysSmsLog{} } func init() { service.RegisterSysSmsLog(NewSysSmsLog()) } // Delete 删除 func (s *sSysSmsLog) Delete(ctx context.Context, in sysin.SmsLogDeleteInp) error { _, err := dao.SysSmsLog.Ctx(ctx).Where("id", in.Id).Delete() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return nil } // Edit 修改/新增 func (s *sSysSmsLog) Edit(ctx context.Context, in sysin.SmsLogEditInp) (err error) { if in.Ip == "" { err = gerror.New("ip不能为空") return err } // 修改 in.UpdatedAt = gtime.Now() if in.Id > 0 { _, err = dao.SysSmsLog.Ctx(ctx).Where("id", in.Id).Data(in).Update() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return nil } // 新增 in.CreatedAt = gtime.Now() _, err = dao.SysSmsLog.Ctx(ctx).Data(in).Insert() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return nil } // Status 更新部门状态 func (s *sSysSmsLog) Status(ctx context.Context, in sysin.SmsLogStatusInp) (err error) { if in.Id <= 0 { err = gerror.New("ID不能为空") return err } if in.Status <= 0 { err = gerror.New("状态不能为空") return err } if !validate.InSliceInt(consts.StatusMap, in.Status) { err = gerror.New("状态不正确") return err } // 修改 in.UpdatedAt = gtime.Now() _, err = dao.SysSmsLog.Ctx(ctx).Where("id", in.Id).Data("status", in.Status).Update() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return nil } // MaxSort 最大排序 func (s *sSysSmsLog) MaxSort(ctx context.Context, in sysin.SmsLogMaxSortInp) (*sysin.SmsLogMaxSortModel, error) { var res sysin.SmsLogMaxSortModel if in.Id > 0 { if err := dao.SysSmsLog.Ctx(ctx).Where("id", in.Id).Order("sort desc").Scan(&res); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return nil, err } } res.Sort = res.Sort + 10 return &res, nil } // View 获取指定字典类型信息 func (s *sSysSmsLog) View(ctx context.Context, in sysin.SmsLogViewInp) (res *sysin.SmsLogViewModel, err error) { if err = dao.SysSmsLog.Ctx(ctx).Where("id", in.Id).Scan(&res); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return nil, err } return res, nil } // List 获取列表 func (s *sSysSmsLog) List(ctx context.Context, in sysin.SmsLogListInp) (list []*sysin.SmsLogListModel, totalCount int, err error) { mod := dao.SysSmsLog.Ctx(ctx) if in.Status > 0 { mod = mod.Where("status", in.Status) } totalCount, err = mod.Count() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return list, totalCount, err } if totalCount == 0 { return list, totalCount, nil } if err = mod.Page(in.Page, in.PerPage).Order("id desc").Scan(&list); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return list, totalCount, err } return list, totalCount, err } // SendCode 发送验证码 func (s *sSysSmsLog) SendCode(ctx context.Context, in sysin.SendCodeInp) (err error) { if in.Event == "" { return gerror.New("事件不能为空") } if in.Mobile == "" { return gerror.New("手机号不能为空") } var ( models *entity.SysSmsLog ) if err = dao.SysSmsLog.Ctx(ctx).Where("event", in.Event).Where("mobile", in.Mobile).Scan(&models); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } config, err := service.SysConfig().GetSms(ctx) if err != nil { return err } in.Template, err = s.GetTemplate(ctx, in.Event, config) if err != nil { return err } err = s.AllowSend(ctx, models, config) if err != nil { return err } if in.Code == "" { in.Code = grand.Digits(4) } switch config.SmsDrive { case consts.SmsDriveAliYun: err = aliyun.SendCode(ctx, in, config) if err != nil { return err } case consts.SmsDriveTencent: return gerror.Newf("暂不支持短信驱动:%v", config.SmsDrive) default: return gerror.Newf("暂不支持短信驱动:%v", config.SmsDrive) } var data = new(entity.SysSmsLog) data.Event = in.Event data.Mobile = in.Mobile data.Code = in.Code data.Ip = location.GetClientIp(ghttp.RequestFromCtx(ctx)) data.Status = consts.SmsStatusNotUsed data.CreatedAt = gtime.Now() data.UpdatedAt = gtime.Now() _, err = dao.SysSmsLog.Ctx(ctx).Data(data).Insert() if err != nil { return err } return nil } // GetTemplate 获取指定短信模板 func (s *sSysSmsLog) GetTemplate(ctx context.Context, template string, config *model.SmsConfig) (val string, err error) { if template == "" { return "", gerror.New("模板不能为空") } if config == nil { config, err = service.SysConfig().GetSms(ctx) if err != nil { return "", err } } switch config.SmsDrive { case consts.SmsDriveAliYun: if len(config.SmsAliyunTemplate) == 0 { return "", gerror.New("管理员还没有配置任何模板!") } for _, v := range config.SmsAliyunTemplate { if v.Key == template { return v.Value, nil } } case consts.SmsDriveTencent: return "", gerror.Newf("暂不支持短信驱动:%v", config.SmsDrive) default: return "", gerror.Newf("暂不支持短信驱动:%v", config.SmsDrive) } return } // AllowSend 是否允许发送 func (s *sSysSmsLog) AllowSend(ctx context.Context, models *entity.SysSmsLog, config *model.SmsConfig) (err error) { if models == nil { return nil } if config == nil { config, err = service.SysConfig().GetSms(ctx) if err != nil { return err } } if gtime.Now().Before(models.CreatedAt.Add(time.Second * time.Duration(config.SmsMinInterval))) { return gerror.New("发送频繁,请稍后再试!") } if config.SmsMaxIpLimit > 0 { count, err := dao.SysSmsLog.NowDayCount(ctx, models.Event, models.Mobile) if err != nil { return err } if count >= config.SmsMaxIpLimit { return gerror.New("今天发送短信过多,请次日后再试!") } } return } // VerifyCode 效验验证码 func (s *sSysSmsLog) VerifyCode(ctx context.Context, in sysin.VerifyCodeInp) (err error) { if in.Event == "" { return gerror.New("事件不能为空") } if in.Mobile == "" { return gerror.New("手机号不能为空") } config, err := service.SysConfig().GetSms(ctx) if err != nil { return err } var ( models *entity.SysSmsLog ) if err = dao.SysSmsLog.Ctx(ctx).Where("event", in.Event).Where("mobile", in.Mobile).Order("id desc").Scan(&models); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } if models == nil { return gerror.New("验证码错误") } if models.Times >= 10 { return gerror.New("验证码错误次数过多,请重新发送!") } if in.Event != consts.SmsTemplateCode { if models.Status == consts.SmsStatusUsed { return gerror.New("验证码已使用,请重新发送!") } } if gtime.Now().After(models.CreatedAt.Add(time.Second * time.Duration(config.SmsCodeExpire))) { return gerror.New("验证码已过期,请重新发送") } if models.Code != in.Code { _, err = dao.SysSmsLog.Ctx(ctx).Where("id", models.Id).Increment("times", 1) if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return gerror.New("验证码错误!") } _, err = dao.SysSmsLog.Ctx(ctx).Where("id", models.Id).Data(g.Map{ "times": models.Times + 1, "status": consts.SmsStatusUsed, "updated_at": gtime.Now(), }).Update() if err != nil { err = gerror.Wrap(err, consts.ErrorORM) return err } return }