formatter.go 8.3 KB

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