config.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. package config
  2. import (
  3. "errors"
  4. "fmt"
  5. "github.com/cnlh/nps/lib/common"
  6. "github.com/cnlh/nps/lib/file"
  7. "regexp"
  8. "strings"
  9. )
  10. type CommonConfig struct {
  11. Server string
  12. VKey string
  13. Tp string //bridgeType kcp or tcp
  14. AutoReconnection bool
  15. ProxyUrl string
  16. Client *file.Client
  17. }
  18. type LocalServer struct {
  19. Type string
  20. Port int
  21. Ip string
  22. Password string
  23. Target string
  24. }
  25. type Config struct {
  26. content string
  27. title []string
  28. CommonConfig *CommonConfig
  29. Hosts []*file.Host
  30. Tasks []*file.Tunnel
  31. Healths []*file.Health
  32. LocalServer []*LocalServer
  33. }
  34. func NewConfig(path string) (c *Config, err error) {
  35. c = new(Config)
  36. var b []byte
  37. if b, err = common.ReadAllFromFile(path); err != nil {
  38. return
  39. } else {
  40. if c.content, err = common.ParseStr(string(b)); err != nil {
  41. return nil, err
  42. }
  43. if c.title, err = getAllTitle(c.content); err != nil {
  44. return
  45. }
  46. var nowIndex int
  47. var nextIndex int
  48. var nowContent string
  49. for i := 0; i < len(c.title); i++ {
  50. nowIndex = strings.Index(c.content, c.title[i]) + len(c.title[i])
  51. if i < len(c.title)-1 {
  52. nextIndex = strings.Index(c.content, c.title[i+1])
  53. } else {
  54. nextIndex = len(c.content)
  55. }
  56. nowContent = c.content[nowIndex:nextIndex]
  57. if strings.Index(getTitleContent(c.title[i]), "secret") == 0 && !strings.Contains(nowContent, "mode") {
  58. local := delLocalService(nowContent)
  59. local.Type = "secret"
  60. c.LocalServer = append(c.LocalServer, local)
  61. continue
  62. }
  63. //except mode
  64. if strings.Index(getTitleContent(c.title[i]), "p2p") == 0 && !strings.Contains(nowContent, "mode") {
  65. local := delLocalService(nowContent)
  66. local.Type = "p2p"
  67. c.LocalServer = append(c.LocalServer, local)
  68. continue
  69. }
  70. //health set
  71. if strings.Index(getTitleContent(c.title[i]), "health") == 0 {
  72. c.Healths = append(c.Healths, dealHealth(nowContent))
  73. continue
  74. }
  75. switch c.title[i] {
  76. case "[common]":
  77. c.CommonConfig = dealCommon(nowContent)
  78. default:
  79. if strings.Index(nowContent, "host") > -1 {
  80. h := dealHost(nowContent)
  81. h.Remark = getTitleContent(c.title[i])
  82. c.Hosts = append(c.Hosts, h)
  83. } else {
  84. t := dealTunnel(nowContent)
  85. t.Remark = getTitleContent(c.title[i])
  86. c.Tasks = append(c.Tasks, t)
  87. }
  88. }
  89. }
  90. }
  91. return
  92. }
  93. func getTitleContent(s string) string {
  94. re, _ := regexp.Compile(`[\[\]]`)
  95. return re.ReplaceAllString(s, "")
  96. }
  97. func dealCommon(s string) *CommonConfig {
  98. c := &CommonConfig{}
  99. c.Client = file.NewClient("", true, true)
  100. c.Client.Cnf = new(file.Config)
  101. for _, v := range splitStr(s) {
  102. item := strings.Split(v, "=")
  103. if len(item) == 0 {
  104. continue
  105. } else if len(item) == 1 {
  106. item = append(item, "")
  107. }
  108. switch item[0] {
  109. case "server_addr":
  110. c.Server = item[1]
  111. case "vkey":
  112. c.VKey = item[1]
  113. case "conn_type":
  114. c.Tp = item[1]
  115. case "auto_reconnection":
  116. c.AutoReconnection = common.GetBoolByStr(item[1])
  117. case "basic_username":
  118. c.Client.Cnf.U = item[1]
  119. case "basic_password":
  120. c.Client.Cnf.P = item[1]
  121. case "web_password":
  122. c.Client.WebPassword = item[1]
  123. case "web_username":
  124. c.Client.WebUserName = item[1]
  125. case "compress":
  126. c.Client.Cnf.Compress = common.GetBoolByStr(item[1])
  127. case "crypt":
  128. c.Client.Cnf.Crypt = common.GetBoolByStr(item[1])
  129. case "proxy_url":
  130. c.ProxyUrl = item[1]
  131. case "rate_limit":
  132. c.Client.RateLimit = common.GetIntNoErrByStr(item[1])
  133. case "flow_limit":
  134. c.Client.Flow.FlowLimit = int64(common.GetIntNoErrByStr(item[1]))
  135. case "max_conn":
  136. c.Client.MaxConn = common.GetIntNoErrByStr(item[1])
  137. case "remark":
  138. c.Client.Remark = item[1]
  139. }
  140. }
  141. return c
  142. }
  143. func dealHost(s string) *file.Host {
  144. h := &file.Host{}
  145. h.Target = new(file.Target)
  146. var headerChange string
  147. for _, v := range splitStr(s) {
  148. item := strings.Split(v, "=")
  149. if len(item) == 0 {
  150. continue
  151. } else if len(item) == 1 {
  152. item = append(item, "")
  153. }
  154. switch strings.TrimSpace(item[0]) {
  155. case "host":
  156. h.Host = item[1]
  157. case "target_addr":
  158. h.Target.TargetStr = strings.Replace(item[1], ",", "\n", -1)
  159. case "host_change":
  160. h.HostChange = item[1]
  161. case "scheme":
  162. h.Scheme = item[1]
  163. case "location":
  164. h.Location = item[1]
  165. default:
  166. if strings.Contains(item[0], "header") {
  167. headerChange += strings.Replace(item[0], "header_", "", -1) + ":" + item[1] + "\n"
  168. }
  169. h.HeaderChange = headerChange
  170. }
  171. }
  172. return h
  173. }
  174. func dealHealth(s string) *file.Health {
  175. h := &file.Health{}
  176. for _, v := range splitStr(s) {
  177. item := strings.Split(v, "=")
  178. if len(item) == 0 {
  179. continue
  180. } else if len(item) == 1 {
  181. item = append(item, "")
  182. }
  183. switch strings.TrimSpace(item[0]) {
  184. case "health_check_timeout":
  185. h.HealthCheckTimeout = common.GetIntNoErrByStr(item[1])
  186. case "health_check_max_failed":
  187. h.HealthMaxFail = common.GetIntNoErrByStr(item[1])
  188. case "health_check_interval":
  189. h.HealthCheckInterval = common.GetIntNoErrByStr(item[1])
  190. case "health_http_url":
  191. h.HttpHealthUrl = item[1]
  192. case "health_check_type":
  193. h.HealthCheckType = item[1]
  194. case "health_check_target":
  195. h.HealthCheckTarget = item[1]
  196. }
  197. }
  198. return h
  199. }
  200. func dealTunnel(s string) *file.Tunnel {
  201. t := &file.Tunnel{}
  202. t.Target = new(file.Target)
  203. for _, v := range splitStr(s) {
  204. item := strings.Split(v, "=")
  205. if len(item) == 0 {
  206. continue
  207. } else if len(item) == 1 {
  208. item = append(item, "")
  209. }
  210. switch strings.TrimSpace(item[0]) {
  211. case "server_port":
  212. t.Ports = item[1]
  213. case "server_ip":
  214. t.ServerIp = item[1]
  215. case "mode":
  216. t.Mode = item[1]
  217. case "target_port", "target_addr":
  218. t.Target.TargetStr = strings.Replace(item[1], ",", "\n", -1)
  219. case "target_ip":
  220. t.TargetAddr = item[1]
  221. case "password":
  222. t.Password = item[1]
  223. case "local_path":
  224. t.LocalPath = item[1]
  225. case "strip_pre":
  226. t.StripPre = item[1]
  227. }
  228. }
  229. return t
  230. }
  231. func delLocalService(s string) *LocalServer {
  232. l := new(LocalServer)
  233. for _, v := range splitStr(s) {
  234. item := strings.Split(v, "=")
  235. if len(item) == 0 {
  236. continue
  237. } else if len(item) == 1 {
  238. item = append(item, "")
  239. }
  240. switch item[0] {
  241. case "local_port":
  242. l.Port = common.GetIntNoErrByStr(item[1])
  243. case "local_ip":
  244. l.Ip = item[1]
  245. case "password":
  246. l.Password = item[1]
  247. case "target_addr":
  248. l.Target = item[1]
  249. }
  250. }
  251. return l
  252. }
  253. func getAllTitle(content string) (arr []string, err error) {
  254. var re *regexp.Regexp
  255. re, err = regexp.Compile(`\[.+?\]`)
  256. if err != nil {
  257. return
  258. }
  259. arr = re.FindAllString(content, -1)
  260. m := make(map[string]bool)
  261. for _, v := range arr {
  262. if _, ok := m[v]; ok {
  263. err = errors.New(fmt.Sprintf("Item names %s are not allowed to be duplicated", v))
  264. return
  265. }
  266. m[v] = true
  267. }
  268. return
  269. }
  270. func splitStr(s string) (configDataArr []string) {
  271. if common.IsWindows() {
  272. configDataArr = strings.Split(s, "\r\n")
  273. }
  274. if len(configDataArr) < 3 {
  275. configDataArr = strings.Split(s, "\n")
  276. }
  277. return
  278. }