formatter.go 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. package service
  2. import (
  3. "context"
  4. "encoding/json"
  5. "fmt"
  6. "github.com/AlekSi/pointer"
  7. v1 "github.com/go-nunu/nunu-layout-advanced/api/v1"
  8. "github.com/go-nunu/nunu-layout-advanced/internal/model"
  9. "github.com/go-nunu/nunu-layout-advanced/internal/repository"
  10. "go.uber.org/zap"
  11. "maps"
  12. "sort"
  13. "strconv"
  14. "strings"
  15. "github.com/spf13/cast"
  16. )
  17. type FormatterService interface {
  18. FormatBackendData(ctx context.Context, req *v1.GameShieldBackendArrayRequest, output map[string]v1.SendGameShieldBackend, keyCounter int) (string, error)
  19. FormatPort(ctx context.Context, req interface{}) []int
  20. OldFormat(ctx context.Context, req *[]model.GameShieldBackend) (map[string]v1.SendGameShieldBackend, error)
  21. TidyFormatBackendData(ctx context.Context, req *v1.GameShieldBackendArrayRequest, keyCounter int) (map[string]v1.SendGameShieldBackend, error)
  22. Sort(ctx context.Context, mapData map[string]v1.SendGameShieldBackend) (map[string]v1.SendGameShieldBackend, error)
  23. ValidateBackendData(ctx context.Context, mapData map[string]v1.SendGameShieldBackend, hostId int) error
  24. }
  25. func NewFormatterService(
  26. service *Service,
  27. gameShieldPublicIpService GameShieldPublicIpService,
  28. gameShieldBackendRepository repository.GameShieldBackendRepository,
  29. hostService HostService,
  30. ) FormatterService {
  31. return &formatterService{
  32. Service: service,
  33. gameShieldPublicIpService: gameShieldPublicIpService,
  34. gameShieldBackendRepository: gameShieldBackendRepository,
  35. hostService: hostService,
  36. }
  37. }
  38. type formatterService struct {
  39. *Service
  40. gameShieldPublicIpService GameShieldPublicIpService
  41. gameShieldBackendRepository repository.GameShieldBackendRepository
  42. hostService HostService
  43. }
  44. func (service *formatterService) FormatBackendData(ctx context.Context, req *v1.GameShieldBackendArrayRequest, oldFormat map[string]v1.SendGameShieldBackend, keyCounter int) (string, error) {
  45. formData, err := service.TidyFormatBackendData(ctx, req, keyCounter)
  46. for _, v := range formData {
  47. v.Type = ""
  48. v.Remark = ""
  49. }
  50. if err != nil {
  51. return "", err
  52. }
  53. maps.Copy(formData, oldFormat)
  54. // 验证
  55. err = service.ValidateBackendData(ctx, formData, req.HostId)
  56. if err != nil {
  57. return "", err
  58. }
  59. sortedOutput, err := service.Sort(ctx, formData)
  60. if err != nil {
  61. return "", err
  62. }
  63. jsonBytes, err := json.MarshalIndent(sortedOutput, "", " ")
  64. if err != nil {
  65. return "", err
  66. }
  67. return string(jsonBytes), nil
  68. }
  69. // FormatPort 格式化端口
  70. func (service *formatterService) FormatPort(ctx context.Context, req interface{}) []int {
  71. if req == nil {
  72. return []int{}
  73. }
  74. reqStr := cast.ToString(req)
  75. if reqStr == "" {
  76. return []int{}
  77. }
  78. reqStr = strings.ReplaceAll(reqStr, ",", ",")
  79. // 分割字符串并转换为整数
  80. var res []int
  81. for _, v := range strings.Split(reqStr, ",") {
  82. // 去除空格
  83. v = strings.TrimSpace(v)
  84. if v != "" {
  85. port := cast.ToInt(v)
  86. res = append(res, port)
  87. }
  88. }
  89. return res
  90. }
  91. // OldFormat 旧格式
  92. func (service *formatterService) OldFormat(ctx context.Context, req *[]model.GameShieldBackend) (map[string]v1.SendGameShieldBackend, error) {
  93. res := make(map[string]v1.SendGameShieldBackend)
  94. var UdpSessionTimeout string
  95. var MaxBandwidth string
  96. for _, v := range *req {
  97. addr := fmt.Sprintf("%s:%s", v.SourceMachineIP, v.ConnectPort)
  98. sdkPort, err := strconv.Atoi(v.SdkPort)
  99. if err != nil {
  100. return nil, err
  101. }
  102. if v.Protocol == "udp" {
  103. UdpSessionTimeout = "300s"
  104. } else {
  105. UdpSessionTimeout = ""
  106. }
  107. keyName := fmt.Sprintf("key%d", v.KeySort)
  108. if v.Type != "pc" {
  109. v.SdkIp = ""
  110. }
  111. if v.MaxBandwidth == 1 {
  112. MaxBandwidth = "100m"
  113. } else {
  114. MaxBandwidth = ""
  115. }
  116. res[keyName] = v1.SendGameShieldBackend{
  117. Addr: []string{addr},
  118. Protocol: v.Protocol,
  119. ProxyAddr: v.ProxyAddr,
  120. SdkPort: sdkPort,
  121. UdpSessionTimeout: UdpSessionTimeout,
  122. SdkIp: v.SdkIp,
  123. MaxBandwidth: MaxBandwidth,
  124. Host: v.Host,
  125. }
  126. }
  127. return res, nil
  128. }
  129. func (service *formatterService) TidyFormatBackendData(ctx context.Context, req *v1.GameShieldBackendArrayRequest, keyCounter int) (map[string]v1.SendGameShieldBackend, error) {
  130. output := make(map[string]v1.SendGameShieldBackend)
  131. userIp, err := service.gameShieldPublicIpService.GetUserIp(ctx, req.Uid)
  132. if err != nil {
  133. return nil, err
  134. }
  135. for _, item := range req.Items {
  136. // 提取必要字段
  137. sourceIP := item.SourceMachineIP // 假设结构体中有这个字段
  138. if sourceIP == "" {
  139. return nil, fmt.Errorf("没有有效源IP的配置") // 跳过没有有效源IP的配置
  140. }
  141. protocol := item.Protocol // 假设结构体中有这个字段
  142. if protocol == "" {
  143. return nil, fmt.Errorf("没有有效协议的配置") // 跳过没有有效协议的配置
  144. }
  145. // 获取端口数组
  146. conPorts := service.FormatPort(ctx, item.ConnectPort)
  147. sdkPorts := service.FormatPort(ctx, item.SdkPort)
  148. // 验证端口数量
  149. if len(sdkPorts) > 0 && len(conPorts) != len(sdkPorts) {
  150. return nil, fmt.Errorf("端口数量不匹配")
  151. }
  152. // 处理每一对端口
  153. for i := 0; i < len(conPorts); i++ {
  154. keyCounter++
  155. key := fmt.Sprintf("key%d", keyCounter)
  156. // 使用数组中的具体端口
  157. addr := fmt.Sprintf("%s:%d", sourceIP, conPorts[i])
  158. itemMap := v1.SendGameShieldBackend{
  159. Addr: []string{addr},
  160. Protocol: protocol,
  161. Type: item.Type,
  162. }
  163. // 设置主机名(如果存在)
  164. if item.Protocol == "http" && item.Host != "" {
  165. itemMap.Host = item.Host
  166. }
  167. // 根据协议设置不同属性
  168. if protocol != "udp" {
  169. if item.RealIp == "agent" {
  170. itemMap.AgentAddr = fmt.Sprintf("%s:%s", sourceIP, "23350")
  171. }
  172. itemMap.ProxyAddr = userIp + ":32353"
  173. } else {
  174. itemMap.ProxyAddr = ""
  175. itemMap.UdpSessionTimeout = "300s"
  176. }
  177. if item.Type != "pc" {
  178. itemMap.SdkIp = ""
  179. } else {
  180. itemMap.SdkIp = item.SdkIp
  181. }
  182. if pointer.GetInt(item.MaxBandwidth) == 1 {
  183. itemMap.MaxBandwidth = "100m"
  184. } else {
  185. itemMap.MaxBandwidth = ""
  186. }
  187. // 设置SDK端口 - 使用数组中的具体端口
  188. if len(sdkPorts) != 0 {
  189. if sdkPorts[i] <= 1024 {
  190. if item.Type == "mobile" {
  191. return nil, fmt.Errorf("移动端不支持SSH端口")
  192. }
  193. }
  194. itemMap.SdkPort = sdkPorts[i]
  195. }
  196. if len(item.Remark) > 0 {
  197. itemMap.Remark = item.Remark
  198. }
  199. output[key] = itemMap
  200. }
  201. }
  202. return output, nil
  203. }
  204. func (service *formatterService) Sort(ctx context.Context, mapData map[string]v1.SendGameShieldBackend) (map[string]v1.SendGameShieldBackend, error) {
  205. var keys []int
  206. for key := range mapData {
  207. intKey, err := strconv.Atoi(strings.TrimPrefix(key, "key"))
  208. if err != nil {
  209. return nil, err
  210. }
  211. keys = append(keys, intKey)
  212. }
  213. // 2. 排序键
  214. sort.Ints(keys)
  215. // 3. 创建一个新的 output 切片或 map 来存储排序后的值
  216. sortedOutput := make(map[string]v1.SendGameShieldBackend)
  217. // 4. 按排序后的键遍历 map,并存储对应的值到 sortedOutput
  218. for _, key := range keys {
  219. sortedOutput["key"+strconv.Itoa(key)] = mapData["key"+strconv.Itoa(key)]
  220. }
  221. return sortedOutput, nil
  222. }
  223. // 验证后端数据
  224. func (service *formatterService) ValidateBackendData(ctx context.Context, data map[string]v1.SendGameShieldBackend, hostId int) error {
  225. // 获取配置限制
  226. configCount, err := service.hostService.GetGameShieldConfig(ctx, hostId)
  227. if err != nil {
  228. return fmt.Errorf("获取配置限制失败: %w", err)
  229. }
  230. // 提取源机IP和SDK端口
  231. sourceIPs := make(map[string]bool)
  232. ruleEntriesCount := int64(0)
  233. maxBandwidthCount := int64(0)
  234. // 分协议检查SDK端口
  235. tcpHttpPorts := make(map[int]bool) // TCP和HTTP共用
  236. udpPorts := make(map[int]bool) // UDP单独使用
  237. for _, item := range data {
  238. // 计算规则条目数
  239. ruleEntriesCount += int64(len(item.Addr))
  240. // 计算源机IP数
  241. for _, addr := range item.Addr {
  242. service.logger.Info(addr)
  243. parts := strings.Split(addr, ":")
  244. if len(parts) > 0 {
  245. if parts[0] != "" {
  246. sourceIPs[parts[0]] = true
  247. }
  248. }
  249. }
  250. // 计算最大带宽设置数
  251. if item.MaxBandwidth != "" {
  252. maxBandwidthCount++
  253. }
  254. // 根据协议类型检查SDK端口重复
  255. if item.SdkPort != 0 {
  256. // 根据协议字段来判断SDK端口是否重复
  257. switch strings.ToUpper(item.Protocol) {
  258. case "TCP", "HTTP":
  259. if tcpHttpPorts[item.SdkPort] {
  260. return fmt.Errorf("TCP/HTTP SDK端口%d重复,每个端口只能配置一次", item.SdkPort)
  261. }
  262. tcpHttpPorts[item.SdkPort] = true
  263. case "UDP":
  264. if udpPorts[item.SdkPort] {
  265. return fmt.Errorf("UDP SDK端口%d重复,每个端口只能配置一次", item.SdkPort)
  266. }
  267. udpPorts[item.SdkPort] = true
  268. default:
  269. return fmt.Errorf("不支持的协议类型: %s", item.Protocol)
  270. }
  271. }
  272. }
  273. // 验证源机数量
  274. service.logger.Info("配置源机数量", zap.Int64("count", configCount.SourceMachinesCount))
  275. service.logger.Info("合并后源机数量", zap.Int("count", len(sourceIPs)))
  276. service.logger.Info("源机列表", zap.Any("list", sourceIPs))
  277. if int64(len(sourceIPs)) > configCount.SourceMachinesCount {
  278. return fmt.Errorf("超出最大源机数量,当前配置允许%d个源机,合并后有%d个源机",
  279. configCount.SourceMachinesCount, len(sourceIPs))
  280. }
  281. // 验证规则条目数
  282. if ruleEntriesCount > configCount.RuleEntriesCount {
  283. return fmt.Errorf("超出最大规则数量,当前配置允许%d个规则,合并后有%d个规则",
  284. configCount.RuleEntriesCount, ruleEntriesCount)
  285. }
  286. // 验证最大带宽设置数
  287. if maxBandwidthCount > configCount.MaxBandwidthCount {
  288. return fmt.Errorf("超出最大带宽数量,当前配置允许%d个带宽设置,合并后有%d个",
  289. configCount.MaxBandwidthCount, maxBandwidthCount)
  290. }
  291. return nil
  292. }