Files
EdgeAPI/internal/rpc/services/service_http_access_log.go

226 lines
5.8 KiB
Go
Raw Permalink Normal View History

2020-10-10 11:49:21 +08:00
package services
import (
"context"
2024-07-27 14:15:25 +08:00
"sync"
2020-10-10 11:49:21 +08:00
"github.com/TeaOSLab/EdgeAPI/internal/db/models"
2022-04-17 16:18:53 +08:00
"github.com/TeaOSLab/EdgeAPI/internal/errors"
2020-10-10 11:49:21 +08:00
rpcutils "github.com/TeaOSLab/EdgeAPI/internal/rpc/utils"
2022-10-14 10:03:29 +08:00
"github.com/TeaOSLab/EdgeAPI/internal/utils/regexputils"
2020-10-10 11:49:21 +08:00
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
2022-04-17 16:18:53 +08:00
"github.com/iwind/TeaGo/dbs"
"github.com/iwind/TeaGo/lists"
2020-10-10 11:49:21 +08:00
)
2021-06-02 11:53:24 +08:00
// HTTPAccessLogService 访问日志相关服务
2020-10-10 11:49:21 +08:00
type HTTPAccessLogService struct {
2021-01-01 20:49:09 +08:00
BaseService
2020-10-10 11:49:21 +08:00
}
2021-06-02 11:53:24 +08:00
// CreateHTTPAccessLogs 创建访问日志
2020-10-10 11:49:21 +08:00
func (this *HTTPAccessLogService) CreateHTTPAccessLogs(ctx context.Context, req *pb.CreateHTTPAccessLogsRequest) (*pb.CreateHTTPAccessLogsResponse, error) {
2020-10-10 19:21:32 +08:00
// 校验请求
2021-07-11 18:05:57 +08:00
_, _, _, err := rpcutils.ValidateRequest(ctx, rpcutils.UserTypeNode)
2020-10-10 11:49:21 +08:00
if err != nil {
return nil, err
}
2021-06-02 11:53:24 +08:00
if len(req.HttpAccessLogs) == 0 {
2020-10-10 11:49:21 +08:00
return &pb.CreateHTTPAccessLogsResponse{}, nil
}
2022-07-22 15:05:30 +08:00
var tx = this.NullTx()
if this.canWriteAccessLogsToDB() {
err = models.SharedHTTPAccessLogDAO.CreateHTTPAccessLogs(tx, req.HttpAccessLogs)
if err != nil {
return nil, err
}
2020-10-10 11:49:21 +08:00
}
2022-08-09 17:35:35 +08:00
err = this.writeAccessLogsToPolicy(req.HttpAccessLogs)
2021-07-29 16:50:59 +08:00
if err != nil {
return nil, err
}
2020-10-10 11:49:21 +08:00
return &pb.CreateHTTPAccessLogsResponse{}, nil
}
2020-10-10 19:21:32 +08:00
2021-06-02 11:53:24 +08:00
// ListHTTPAccessLogs 列出单页访问日志
2020-10-10 19:21:32 +08:00
func (this *HTTPAccessLogService) ListHTTPAccessLogs(ctx context.Context, req *pb.ListHTTPAccessLogsRequest) (*pb.ListHTTPAccessLogsResponse, error) {
// 校验请求
2022-09-17 16:07:37 +08:00
_, userId, err := this.ValidateAdminAndUser(ctx, true)
2020-10-10 19:21:32 +08:00
if err != nil {
return nil, err
}
2022-07-22 15:05:30 +08:00
var tx = this.NullTx()
2021-01-01 20:49:09 +08:00
// 检查服务ID
if userId > 0 {
2022-07-22 15:05:30 +08:00
req.UserId = userId
2021-01-01 20:49:09 +08:00
// 这里不用担心serverId <= 0 的情况因为如果userId>0则只会查询当前用户下的服务不会产生安全问题
if req.ServerId > 0 {
err = models.SharedServerDAO.CheckUserServer(tx, userId, req.ServerId)
if err != nil {
return nil, err
}
2021-01-01 20:49:09 +08:00
}
}
2022-04-17 16:18:53 +08:00
accessLogs, requestId, hasMore, err := models.SharedHTTPAccessLogDAO.ListAccessLogs(tx, req.Partition, req.RequestId, req.Size, req.Day, req.HourFrom, req.HourTo, req.NodeClusterId, req.NodeId, req.ServerId, req.Reverse, req.HasError, req.FirewallPolicyId, req.FirewallRuleGroupId, req.FirewallRuleSetId, req.HasFirewallPolicy, req.UserId, req.Keyword, req.Ip, req.Domain)
2020-10-10 19:21:32 +08:00
if err != nil {
return nil, err
}
2022-07-22 15:05:30 +08:00
var result = []*pb.HTTPAccessLog{}
2021-08-10 11:15:15 +08:00
var pbNodeMap = map[int64]*pb.Node{}
var pbClusterMap = map[int64]*pb.NodeCluster{}
2020-10-10 19:21:32 +08:00
for _, accessLog := range accessLogs {
a, err := accessLog.ToPB()
if err != nil {
return nil, err
}
2021-08-10 11:15:15 +08:00
// 节点 & 集群
pbNode, ok := pbNodeMap[a.NodeId]
if ok {
a.Node = pbNode
} else {
node, err := models.SharedNodeDAO.FindEnabledNode(tx, a.NodeId)
if err != nil {
return nil, err
}
if node != nil {
pbNode = &pb.Node{Id: int64(node.Id), Name: node.Name}
var clusterId = int64(node.ClusterId)
pbCluster, ok := pbClusterMap[clusterId]
if ok {
pbNode.NodeCluster = pbCluster
} else {
cluster, err := models.SharedNodeClusterDAO.FindEnabledNodeCluster(tx, clusterId)
if err != nil {
return nil, err
}
if cluster != nil {
pbCluster = &pb.NodeCluster{
Id: int64(cluster.Id),
Name: cluster.Name,
}
pbNode.NodeCluster = pbCluster
pbClusterMap[clusterId] = pbCluster
}
}
pbNodeMap[a.NodeId] = pbNode
a.Node = pbNode
}
}
2020-10-10 19:21:32 +08:00
result = append(result, a)
}
return &pb.ListHTTPAccessLogsResponse{
2021-06-02 11:53:24 +08:00
HttpAccessLogs: result,
2021-06-04 11:41:50 +08:00
AccessLogs: result, // TODO 仅仅为了兼容当用户节点版本大于0.0.8时可以删除
2021-06-02 11:53:24 +08:00
HasMore: hasMore,
RequestId: requestId,
2020-10-10 19:21:32 +08:00
}, nil
}
2021-06-02 11:53:24 +08:00
// FindHTTPAccessLog 查找单个日志
func (this *HTTPAccessLogService) FindHTTPAccessLog(ctx context.Context, req *pb.FindHTTPAccessLogRequest) (*pb.FindHTTPAccessLogResponse, error) {
// 校验请求
2022-09-17 16:07:37 +08:00
_, userId, err := this.ValidateAdminAndUser(ctx, true)
if err != nil {
return nil, err
}
2022-07-22 15:05:30 +08:00
var tx = this.NullTx()
accessLog, err := models.SharedHTTPAccessLogDAO.FindAccessLogWithRequestId(tx, req.RequestId)
if err != nil {
return nil, err
}
if accessLog == nil {
2021-06-02 11:53:24 +08:00
return &pb.FindHTTPAccessLogResponse{HttpAccessLog: nil}, nil
}
2021-01-14 18:01:00 +08:00
// 检查权限
if userId > 0 {
err = models.SharedServerDAO.CheckUserServer(tx, userId, int64(accessLog.ServerId))
2021-01-14 18:01:00 +08:00
if err != nil {
return nil, err
}
}
a, err := accessLog.ToPB()
if err != nil {
return nil, err
}
2021-06-02 11:53:24 +08:00
return &pb.FindHTTPAccessLogResponse{HttpAccessLog: a}, nil
}
2022-04-17 16:18:53 +08:00
// FindHTTPAccessLogPartitions 查找日志分区
func (this *HTTPAccessLogService) FindHTTPAccessLogPartitions(ctx context.Context, req *pb.FindHTTPAccessLogPartitionsRequest) (*pb.FindHTTPAccessLogPartitionsResponse, error) {
2022-07-22 14:35:17 +08:00
_, err := this.ValidateAdmin(ctx)
2022-04-17 16:18:53 +08:00
if err != nil {
return nil, err
}
2022-10-14 10:03:29 +08:00
if !regexputils.YYYYMMDD.MatchString(req.Day) {
2022-04-17 16:18:53 +08:00
return nil, errors.New("invalid 'day': " + req.Day)
}
var dbList = models.AllAccessLogDBs()
if len(dbList) == 0 {
return &pb.FindHTTPAccessLogPartitionsResponse{
Partitions: nil,
}, nil
}
var partitions = []int32{}
var locker sync.Mutex
var wg = sync.WaitGroup{}
wg.Add(len(dbList))
var lastErr error
for _, db := range dbList {
go func(db *dbs.DB) {
defer wg.Done()
names, err := models.SharedHTTPAccessLogManager.FindTableNames(db, req.Day)
if err != nil {
lastErr = err
}
for _, name := range names {
var partition = models.SharedHTTPAccessLogManager.TablePartition(name)
locker.Lock()
if !lists.Contains(partitions, partition) {
partitions = append(partitions, partition)
}
locker.Unlock()
}
}(db)
}
wg.Wait()
if lastErr != nil {
return nil, lastErr
}
var reversePartitions = []int32{}
for i := len(partitions) - 1; i >= 0; i-- {
reversePartitions = append(reversePartitions, partitions[i])
}
return &pb.FindHTTPAccessLogPartitionsResponse{
Partitions: partitions,
ReversePartitions: reversePartitions,
2022-04-17 16:18:53 +08:00
}, nil
}