main.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. /*
  2. * Copyright (c) 2016, Psiphon Inc.
  3. * All rights reserved.
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. *
  18. */
  19. package main
  20. import (
  21. "flag"
  22. "fmt"
  23. "io/ioutil"
  24. "os"
  25. "strconv"
  26. "strings"
  27. "github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon"
  28. "github.com/Psiphon-Labs/psiphon-tunnel-core/psiphon/server"
  29. )
  30. func main() {
  31. var generateTrafficRulesFilename, generateServerEntryFilename string
  32. var generateServerIPaddress, generateServerNetworkInterface string
  33. var generateWebServerPort int
  34. var generateProtocolPorts stringListFlag
  35. var generateLogFilename string
  36. var generateFail2BanLogFilename string
  37. var configFilename string
  38. flag.StringVar(
  39. &generateTrafficRulesFilename,
  40. "trafficRules",
  41. server.SERVER_TRAFFIC_RULES_FILENAME,
  42. "generate with this traffic rules `filename`")
  43. flag.StringVar(
  44. &generateServerEntryFilename,
  45. "serverEntry",
  46. server.SERVER_ENTRY_FILENAME,
  47. "generate with this server entry `filename`")
  48. flag.StringVar(
  49. &generateServerIPaddress,
  50. "ipaddress",
  51. server.DEFAULT_SERVER_IP_ADDRESS,
  52. "generate with this server `IP address`")
  53. flag.StringVar(
  54. &generateServerNetworkInterface,
  55. "interface",
  56. "",
  57. "generate with server IP address from this `network-interface`")
  58. flag.IntVar(
  59. &generateWebServerPort,
  60. "web",
  61. 0,
  62. "generate with web server `port`; 0 for no web server")
  63. flag.Var(
  64. &generateProtocolPorts,
  65. "protocol",
  66. "generate with `protocol:port`; flag may be repeated to enable multiple protocols")
  67. flag.StringVar(
  68. &generateLogFilename,
  69. "logFilename",
  70. "",
  71. "set application log file name and path; blank for stderr")
  72. flag.StringVar(
  73. &generateFail2BanLogFilename,
  74. "fail2BanLogFilename",
  75. "",
  76. "set Fail2Ban log file name and path; blank for stderr")
  77. flag.StringVar(
  78. &configFilename,
  79. "config",
  80. server.SERVER_CONFIG_FILENAME,
  81. "run or generate with this config `filename`")
  82. flag.Usage = func() {
  83. fmt.Fprintf(os.Stderr,
  84. "Usage:\n\n"+
  85. "%s <flags> generate generates configuration files\n"+
  86. "%s <flags> run runs configured services\n\n",
  87. os.Args[0], os.Args[0])
  88. flag.PrintDefaults()
  89. }
  90. flag.Parse()
  91. args := flag.Args()
  92. if len(args) < 1 {
  93. flag.Usage()
  94. os.Exit(1)
  95. } else if args[0] == "generate" {
  96. serverIPaddress := generateServerIPaddress
  97. if generateServerNetworkInterface != "" {
  98. var err error
  99. serverIPaddress, err = psiphon.GetInterfaceIPAddress(generateServerNetworkInterface)
  100. fmt.Printf("generate failed: %s\n", err)
  101. os.Exit(1)
  102. }
  103. tunnelProtocolPorts := make(map[string]int)
  104. for _, protocolPort := range generateProtocolPorts {
  105. parts := strings.Split(protocolPort, ":")
  106. if len(parts) == 2 {
  107. port, err := strconv.Atoi(parts[1])
  108. if err != nil {
  109. fmt.Printf("generate failed: %s\n", err)
  110. os.Exit(1)
  111. }
  112. tunnelProtocolPorts[parts[0]] = port
  113. }
  114. }
  115. configJSON, trafficRulesJSON, encodedServerEntry, err :=
  116. server.GenerateConfig(
  117. &server.GenerateConfigParams{
  118. ServerIPAddress: serverIPaddress,
  119. EnableSSHAPIRequests: true,
  120. WebServerPort: generateWebServerPort,
  121. TunnelProtocolPorts: tunnelProtocolPorts,
  122. TrafficRulesFilename: generateTrafficRulesFilename,
  123. LogFilename: generateLogFilename,
  124. Fail2BanLogFilename: generateFail2BanLogFilename,
  125. })
  126. if err != nil {
  127. fmt.Printf("generate failed: %s\n", err)
  128. os.Exit(1)
  129. }
  130. err = ioutil.WriteFile(configFilename, configJSON, 0600)
  131. if err != nil {
  132. fmt.Printf("error writing configuration file: %s\n", err)
  133. os.Exit(1)
  134. }
  135. err = ioutil.WriteFile(generateTrafficRulesFilename, trafficRulesJSON, 0600)
  136. if err != nil {
  137. fmt.Printf("error writing traffic rule configuration file: %s\n", err)
  138. os.Exit(1)
  139. }
  140. err = ioutil.WriteFile(generateServerEntryFilename, encodedServerEntry, 0600)
  141. if err != nil {
  142. fmt.Printf("error writing server entry file: %s\n", err)
  143. os.Exit(1)
  144. }
  145. } else if args[0] == "run" {
  146. configJSON, err := ioutil.ReadFile(configFilename)
  147. if err != nil {
  148. fmt.Printf("error loading configuration file: %s\n", err)
  149. os.Exit(1)
  150. }
  151. err = server.RunServices(configJSON)
  152. if err != nil {
  153. fmt.Printf("run failed: %s\n", err)
  154. os.Exit(1)
  155. }
  156. }
  157. }
  158. type stringListFlag []string
  159. func (list *stringListFlag) String() string {
  160. return strings.Join(*list, ", ")
  161. }
  162. func (list *stringListFlag) Set(flagValue string) error {
  163. *list = append(*list, flagValue)
  164. return nil
  165. }