server.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. package server
  2. import (
  3. "be-vpn/internal/dto"
  4. "be-vpn/internal/model"
  5. "be-vpn/internal/storage"
  6. "github.com/gin-gonic/gin"
  7. "log"
  8. "math/rand"
  9. "net/http"
  10. "strconv"
  11. "sync"
  12. "time"
  13. )
  14. var nodes = make([]*model.Node, 0)
  15. var locker = sync.RWMutex{}
  16. var totalFreeDuration = uint64((time.Hour * 1).Milliseconds() / 1000)
  17. func Config(c *gin.Context) {
  18. deviceId := c.Query("deviceId")
  19. usedDuration, err := storage.GetUsedDuration(deviceId)
  20. if err != nil {
  21. dto.Error(c, err)
  22. return
  23. }
  24. var node *model.Node
  25. if len(healthNodes()) > 0 {
  26. node = healthNodes()[0]
  27. }
  28. c.JSON(http.StatusOK, dto.ConfigResponse{
  29. Response: dto.NewOkResponse(),
  30. Data: dto.ConfigResult{
  31. FreeTrialDuration: totalFreeDuration - usedDuration,
  32. Timestamp: time.Now().Unix(),
  33. Node: convert2DtoNode(node),
  34. },
  35. })
  36. }
  37. func AddUsedDuration(c *gin.Context) {
  38. deviceId := c.Query("deviceId")
  39. usedDurationStr := c.Query("usedDuration")
  40. log.Printf("deviceId: %s, usedDuration: %s", deviceId, usedDurationStr)
  41. usedDuration, err := strconv.ParseUint(usedDurationStr, 10, 64)
  42. if err != nil {
  43. dto.Error(c, err)
  44. return
  45. }
  46. if existed, err := storage.AddUsedDuration(deviceId, usedDuration); err != nil {
  47. dto.Error(c, err)
  48. return
  49. } else {
  50. freeTrialDuration := totalFreeDuration - existed
  51. if freeTrialDuration > totalFreeDuration || freeTrialDuration < 0 {
  52. freeTrialDuration = 0
  53. }
  54. c.JSON(http.StatusOK, dto.ConfigResponse{
  55. Response: dto.NewOkResponse(),
  56. Data: dto.ConfigResult{
  57. FreeTrialDuration: freeTrialDuration,
  58. Timestamp: time.Now().Unix(),
  59. Node: convert2DtoNode(healthNodes()[0]),
  60. },
  61. })
  62. }
  63. }
  64. func Register(c *gin.Context) {
  65. locker.Lock()
  66. defer locker.Unlock()
  67. var request dto.RegisterRequest
  68. if err := c.ShouldBindJSON(&request); err != nil {
  69. dto.BadRequest(c, err)
  70. return
  71. }
  72. for _, node := range nodes {
  73. if node.Ip == request.Ip {
  74. node.Ip = request.Ip
  75. node.Secret = request.Secret
  76. node.CountryCode = request.CountryCode
  77. node.CountryName = request.CountryName
  78. node.LastUpdateTime = time.Now()
  79. c.JSON(http.StatusOK, dto.RegisterResponse{
  80. Response: dto.NewOkResponse(),
  81. Data: dto.RegisterResult{
  82. Success: true,
  83. },
  84. })
  85. return
  86. }
  87. }
  88. node := &model.Node{
  89. Ip: request.Ip,
  90. Secret: request.Secret,
  91. LastUpdateTime: time.Now(),
  92. }
  93. nodes = append(nodes, node)
  94. log.Printf("update nodes: %+v", nodes)
  95. }
  96. func List(c *gin.Context) {
  97. locker.RLock()
  98. defer locker.RUnlock()
  99. dtoNodes := make([]*dto.Node, 0)
  100. for _, node := range healthNodes() {
  101. dtoNodes = append(dtoNodes, convert2DtoNode(node))
  102. }
  103. c.JSON(http.StatusOK, dto.ListResponse{
  104. Response: dto.NewOkResponse(),
  105. Data: dtoNodes,
  106. })
  107. }
  108. func SecretRandom(c *gin.Context) {
  109. locker.RLock()
  110. defer locker.RUnlock()
  111. random := rand.Intn(len(nodes))
  112. for i, node := range nodes {
  113. if i == random {
  114. c.Header("Content-Disposition", "attachment; filename=client.ovpn")
  115. c.Data(http.StatusOK, "plain/text", []byte(node.Secret))
  116. return
  117. }
  118. }
  119. c.JSON(http.StatusNotFound, gin.H{
  120. "message": "not found",
  121. })
  122. }
  123. func Secret(c *gin.Context) {
  124. locker.RLock()
  125. defer locker.RUnlock()
  126. var request dto.DetailRequest
  127. if err := c.ShouldBindQuery(&request); err != nil {
  128. dto.BadRequest(c, err)
  129. return
  130. }
  131. for _, node := range nodes {
  132. if node.Ip == request.Ip {
  133. //secret, err := util.AesEncrypt([]byte(node.Secret))
  134. //if err != nil {
  135. // dto.Error(c, err)
  136. // return
  137. //}
  138. c.Header("Content-Disposition", "attachment; filename=client.ovpn")
  139. c.Data(http.StatusOK, "plain/text", []byte(node.Secret))
  140. return
  141. }
  142. }
  143. c.JSON(http.StatusNotFound, gin.H{
  144. "message": "not found ip",
  145. })
  146. }
  147. func Health(c *gin.Context) {
  148. c.JSON(http.StatusOK, gin.H{"status": "up"})
  149. }
  150. func healthNodes() []*model.Node {
  151. healthNodes := make([]*model.Node, 0)
  152. for _, node := range nodes {
  153. if node.LastUpdateTime.Add(10 * time.Second).After(time.Now()) {
  154. healthNodes = append(healthNodes, node)
  155. }
  156. }
  157. return healthNodes
  158. }
  159. func convert2DtoNode(node *model.Node) *dto.Node {
  160. if node == nil {
  161. return nil
  162. }
  163. icon := ""
  164. if node.CountryCode == "BR" {
  165. icon = "http://v.starttransfernow.com/static/BR.jpg"
  166. } else if node.CountryCode == "DE" {
  167. icon = "http://v.starttransfernow.com/static/DE.jpg"
  168. } else if node.CountryCode == "HK" {
  169. icon = "http://v.starttransfernow.com/static/HK.jpg"
  170. } else if node.CountryCode == "JP" {
  171. icon = "http://v.starttransfernow.com/static/JP.jpg"
  172. } else if node.CountryCode == "US" {
  173. icon = "http://v.starttransfernow.com/static/US.jpg"
  174. }
  175. return &dto.Node{
  176. Ip: node.Ip,
  177. CountryCode: node.CountryCode,
  178. CountryName: node.CountryName,
  179. Icon: icon,
  180. }
  181. }