// Package sys // @Link https://github.com/bufanyun/hotgo // @Copyright Copyright (c) 2023 HotGo CLI // @Author Ms <133814250@qq.com> // @License https://github.com/bufanyun/hotgo/blob/master/LICENSE package sys import ( "context" "fmt" "github.com/gogf/gf/v2/database/gdb" "github.com/gogf/gf/v2/encoding/gjson" "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/gctx" "github.com/gogf/gf/v2/os/gtime" "github.com/gogf/gf/v2/util/gconv" "hotgo/internal/consts" "hotgo/internal/dao" "hotgo/internal/library/hgorm/handler" "hotgo/internal/library/hgorm/hook" "hotgo/internal/library/location" "hotgo/internal/library/queue" "hotgo/internal/model/entity" "hotgo/internal/model/input/adminin" "hotgo/internal/model/input/form" "hotgo/internal/model/input/sysin" "hotgo/internal/service" "hotgo/utility/convert" "hotgo/utility/excel" "hotgo/utility/useragent" ) type sSysLoginLog struct{} func NewSysLoginLog() *sSysLoginLog { return &sSysLoginLog{} } func init() { service.RegisterSysLoginLog(NewSysLoginLog()) } // Model 登录日志Orm模型 func (s *sSysLoginLog) Model(ctx context.Context, option ...*handler.Option) *gdb.Model { return handler.Model(dao.SysLoginLog.Ctx(ctx), option...) } // List 获取登录日志列表 func (s *sSysLoginLog) List(ctx context.Context, in *sysin.LoginLogListInp) (list []*sysin.LoginLogListModel, totalCount int, err error) { mod := s.Model(ctx) // 查询状态 if in.Status > 0 { mod = mod.Where(dao.SysLoginLog.Columns().Status, in.Status) } // 查询登录时间 if len(in.LoginAt) == 2 { mod = mod.WhereBetween(dao.SysLoginLog.Columns().LoginAt, in.LoginAt[0], in.LoginAt[1]) } // 查询IP地址 if in.LoginIp != "" { mod = mod.Where(dao.SysLoginLog.Columns().LoginIp, in.LoginIp) } // 用户名 if in.Username != "" { mod = mod.Where(dao.SysLoginLog.Columns().Username, in.Username) } totalCount, err = mod.Clone().Count(1) if err != nil || totalCount == 0 { return } if err = mod.Fields(sysin.LoginLogListModel{}).Hook(hook.CityLabel).Page(in.Page, in.PerPage).OrderDesc(dao.SysLoginLog.Columns().Id).Scan(&list); err != nil { err = gerror.Wrap(err, consts.ErrorORM) return list, totalCount, err } for _, v := range list { v.Os = useragent.GetOs(v.UserAgent) v.Browser = useragent.GetBrowser(v.UserAgent) v.SysLogId, err = dao.SysLog.Ctx(ctx).Fields("id").Where("req_id", v.ReqId).Value() if err != nil { return nil, 0, err } } return } // Export 导出登录日志 func (s *sSysLoginLog) Export(ctx context.Context, in *sysin.LoginLogListInp) (err error) { list, totalCount, err := s.List(ctx, in) if err != nil { return } // 字段的排序是依据tags的字段顺序,如果你不想使用默认的排序方式,可以直接定义 tags = []string{"字段名称", "字段名称2", ...} tags, err := convert.GetEntityDescTags(sysin.LoginLogExportModel{}) if err != nil { return } var ( fileName = "导出登录日志-" + gctx.CtxId(ctx) sheetName = fmt.Sprintf("索引条件共%v行,共%v页,当前导出是第%v页,本页共%v行", totalCount, form.CalPageCount(totalCount, in.PerPage), in.Page, len(list)) exports []sysin.LoginLogExportModel ) if err = gconv.Scan(list, &exports); err != nil { return } err = excel.ExportByStructs(ctx, tags, exports, fileName, sheetName) return } // Delete 删除登录日志 func (s *sSysLoginLog) Delete(ctx context.Context, in *sysin.LoginLogDeleteInp) (err error) { _, err = s.Model(ctx).WherePri(in.Id).Delete() return } // Push 推送登录日志 func (s *sSysLoginLog) Push(ctx context.Context, in *sysin.LoginLogPushInp) { if in.Response == nil { in.Response = new(adminin.LoginModel) } r := ghttp.RequestFromCtx(ctx) if r == nil { g.Log().Warningf(ctx, "ctx not http request") return } clientIp := location.GetClientIp(r) ipData, err := location.GetLocation(ctx, clientIp) if err != nil { g.Log().Debugf(ctx, "location.GetLocation clientIp:%v, err:%+v", clientIp, err) } if ipData == nil { ipData = new(location.IpLocationData) } var models entity.SysLoginLog models.ReqId = gctx.CtxId(ctx) models.MemberId = in.Response.Id models.Username = in.Response.Username models.LoginAt = gtime.Now() models.LoginIp = clientIp models.UserAgent = r.UserAgent() models.ProvinceId = ipData.ProvinceCode models.CityId = ipData.CityCode models.Status = consts.StatusEnabled if in.Err != nil { models.Status = consts.StatusDisable models.ErrMsg = in.Err.Error() } models.Response = gjson.New(consts.NilJsonToString) if in.Response != nil { models.Response = gjson.New(in.Response) } if err = queue.Push(consts.QueueLoginLogTopic, models); err != nil { g.Log().Warningf(ctx, "push LoginLog err:%+v, models:%v", err, gjson.New(models).String()) } } // RealWrite 真实写入 func (s *sSysLoginLog) RealWrite(ctx context.Context, models entity.SysLoginLog) (err error) { _, err = dao.SysLoginLog.Ctx(ctx).Data(models).OmitEmptyData().Insert() return }