main.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. //go:generate goversioninfo -icon=windowsdata/resource/netmaker.ico -manifest=netclient.exe.manifest.xml -64=true -o=netclient.syso
  2. package main
  3. import (
  4. "errors"
  5. "log"
  6. "os"
  7. "os/exec"
  8. "os/signal"
  9. "strconv"
  10. "syscall"
  11. "runtime/debug"
  12. "github.com/gravitl/netmaker/netclient/command"
  13. "github.com/gravitl/netmaker/netclient/config"
  14. "github.com/gravitl/netmaker/netclient/local"
  15. "github.com/gravitl/netmaker/netclient/ncwindows"
  16. "github.com/gravitl/netmaker/netclient/netclientutils"
  17. "github.com/urfave/cli/v2"
  18. )
  19. func main() {
  20. app := cli.NewApp()
  21. app.Name = "Netclient CLI"
  22. app.Usage = "Netmaker's netclient agent and CLI. Used to perform interactions with Netmaker server and set local WireGuard config."
  23. app.Version = "v0.7.3"
  24. cliFlags := []cli.Flag{
  25. &cli.StringFlag{
  26. Name: "network",
  27. Aliases: []string{"n"},
  28. EnvVars: []string{"NETCLIENT_NETWORK"},
  29. Value: "all",
  30. Usage: "Network to perform specified action against.",
  31. },
  32. &cli.StringFlag{
  33. Name: "password",
  34. Aliases: []string{"p"},
  35. EnvVars: []string{"NETCLIENT_PASSWORD"},
  36. Value: "",
  37. Usage: "Password for authenticating with netmaker.",
  38. },
  39. &cli.StringFlag{
  40. Name: "endpoint",
  41. Aliases: []string{"e"},
  42. EnvVars: []string{"NETCLIENT_ENDPOINT"},
  43. Value: "",
  44. Usage: "Reachable (usually public) address for WireGuard (not the private WG address).",
  45. },
  46. &cli.StringFlag{
  47. Name: "macaddress",
  48. Aliases: []string{"m"},
  49. EnvVars: []string{"NETCLIENT_MACADDRESS"},
  50. Value: "",
  51. Usage: "Mac Address for this machine. Used as a unique identifier within Netmaker network.",
  52. },
  53. &cli.StringFlag{
  54. Name: "publickey",
  55. Aliases: []string{"pubkey"},
  56. EnvVars: []string{"NETCLIENT_PUBLICKEY"},
  57. Value: "",
  58. Usage: "Public Key for WireGuard Interface.",
  59. },
  60. &cli.StringFlag{
  61. Name: "privatekey",
  62. Aliases: []string{"privkey"},
  63. EnvVars: []string{"NETCLIENT_PRIVATEKEY"},
  64. Value: "",
  65. Usage: "Private Key for WireGuard Interface.",
  66. },
  67. &cli.StringFlag{
  68. Name: "port",
  69. EnvVars: []string{"NETCLIENT_PORT"},
  70. Value: "",
  71. Usage: "Port for WireGuard Interface.",
  72. },
  73. &cli.IntFlag{
  74. Name: "keepalive",
  75. EnvVars: []string{"NETCLIENT_KEEPALIVE"},
  76. Value: 0,
  77. Usage: "Default PersistentKeepAlive for Peers in WireGuard Interface.",
  78. },
  79. &cli.StringFlag{
  80. Name: "operatingsystem",
  81. Aliases: []string{"os"},
  82. EnvVars: []string{"NETCLIENT_OS"},
  83. Value: "",
  84. Usage: "Identifiable name for machine within Netmaker network.",
  85. },
  86. &cli.StringFlag{
  87. Name: "name",
  88. EnvVars: []string{"NETCLIENT_NAME"},
  89. Value: "",
  90. Usage: "Identifiable name for machine within Netmaker network.",
  91. },
  92. &cli.StringFlag{
  93. Name: "localaddress",
  94. EnvVars: []string{"NETCLIENT_LOCALADDRESS"},
  95. Value: "",
  96. Usage: "Local address for machine. Can be used in place of Endpoint for machines on the same LAN.",
  97. },
  98. &cli.StringFlag{
  99. Name: "address",
  100. Aliases: []string{"a"},
  101. EnvVars: []string{"NETCLIENT_ADDRESS"},
  102. Value: "",
  103. Usage: "WireGuard address for machine within Netmaker network.",
  104. },
  105. &cli.StringFlag{
  106. Name: "addressIPv6",
  107. Aliases: []string{"a6"},
  108. EnvVars: []string{"NETCLIENT_ADDRESSIPV6"},
  109. Value: "",
  110. Usage: "WireGuard address for machine within Netmaker network.",
  111. },
  112. &cli.StringFlag{
  113. Name: "interface",
  114. Aliases: []string{"i"},
  115. EnvVars: []string{"NETCLIENT_INTERFACE"},
  116. Value: "",
  117. Usage: "WireGuard local network interface name.",
  118. },
  119. &cli.StringFlag{
  120. Name: "apiserver",
  121. EnvVars: []string{"NETCLIENT_API_SERVER"},
  122. Value: "",
  123. Usage: "Address + GRPC Port (e.g. 1.2.3.4:50051) of Netmaker server.",
  124. },
  125. &cli.StringFlag{
  126. Name: "grpcserver",
  127. EnvVars: []string{"NETCLIENT_GRPC_SERVER"},
  128. Value: "",
  129. Usage: "Address + API Port (e.g. 1.2.3.4:8081) of Netmaker server.",
  130. },
  131. &cli.StringFlag{
  132. Name: "key",
  133. Aliases: []string{"k"},
  134. EnvVars: []string{"NETCLIENT_ACCESSKEY"},
  135. Value: "",
  136. Usage: "Access Key for signing up machine with Netmaker server during initial 'add'.",
  137. },
  138. &cli.StringFlag{
  139. Name: "token",
  140. Aliases: []string{"t"},
  141. EnvVars: []string{"NETCLIENT_ACCESSTOKEN"},
  142. Value: "",
  143. Usage: "Access Token for signing up machine with Netmaker server during initial 'add'.",
  144. },
  145. &cli.StringFlag{
  146. Name: "localrange",
  147. EnvVars: []string{"NETCLIENT_LOCALRANGE"},
  148. Value: "",
  149. Usage: "Local Range if network is local, for instance 192.168.1.0/24.",
  150. },
  151. &cli.StringFlag{
  152. Name: "dnson",
  153. EnvVars: []string{"NETCLIENT_DNS"},
  154. Value: "yes",
  155. Usage: "Sets private dns if 'yes'. Ignores if 'no'. Will retrieve from network if unset.",
  156. },
  157. &cli.StringFlag{
  158. Name: "islocal",
  159. EnvVars: []string{"NETCLIENT_IS_LOCAL"},
  160. Value: "",
  161. Usage: "Sets endpoint to local address if 'yes'. Ignores if 'no'. Will retrieve from network if unset.",
  162. },
  163. &cli.StringFlag{
  164. Name: "isdualstack",
  165. EnvVars: []string{"NETCLIENT_IS_DUALSTACK"},
  166. Value: "",
  167. Usage: "Sets ipv6 address if 'yes'. Ignores if 'no'. Will retrieve from network if unset.",
  168. },
  169. &cli.StringFlag{
  170. Name: "udpholepunch",
  171. EnvVars: []string{"NETCLIENT_UDP_HOLEPUNCH"},
  172. Value: "",
  173. Usage: "Turns on udp holepunching if 'yes'. Ignores if 'no'. Will retrieve from network if unset.",
  174. },
  175. &cli.StringFlag{
  176. Name: "ipforwarding",
  177. EnvVars: []string{"NETCLIENT_IPFORWARDING"},
  178. Value: "on",
  179. Usage: "Sets ip forwarding on if 'on'. Ignores if 'off'. On by default.",
  180. },
  181. &cli.StringFlag{
  182. Name: "postup",
  183. EnvVars: []string{"NETCLIENT_POSTUP"},
  184. Value: "",
  185. Usage: "Sets PostUp command for WireGuard.",
  186. },
  187. &cli.StringFlag{
  188. Name: "postdown",
  189. EnvVars: []string{"NETCLIENT_POSTDOWN"},
  190. Value: "",
  191. Usage: "Sets PostDown command for WireGuard.",
  192. },
  193. &cli.StringFlag{
  194. Name: "daemon",
  195. EnvVars: []string{"NETCLIENT_DAEMON"},
  196. Value: "on",
  197. Usage: "Installs daemon if 'on'. Ignores if 'off'. On by default.",
  198. },
  199. &cli.StringFlag{
  200. Name: "roaming",
  201. EnvVars: []string{"NETCLIENT_ROAMING"},
  202. Value: "on",
  203. Usage: "Checks for IP changes if 'on'. Ignores if 'off'. On by default.",
  204. },
  205. }
  206. app.Commands = []*cli.Command{
  207. {
  208. Name: "join",
  209. Usage: "Join a Netmaker network.",
  210. Flags: cliFlags,
  211. Action: func(c *cli.Context) error {
  212. cfg, pvtKey, err := config.GetCLIConfig(c)
  213. if err != nil {
  214. return err
  215. }
  216. if cfg.Network == "all" {
  217. err = errors.New("No network provided.")
  218. return err
  219. }
  220. if cfg.Server.GRPCAddress == "" {
  221. err = errors.New("No server address provided.")
  222. return err
  223. }
  224. err = command.Join(cfg, pvtKey)
  225. return err
  226. },
  227. },
  228. {
  229. Name: "leave",
  230. Usage: "Leave a Netmaker network.",
  231. Flags: cliFlags,
  232. // the action, or code that will be executed when
  233. // we execute our `ns` command
  234. Action: func(c *cli.Context) error {
  235. cfg, _, err := config.GetCLIConfig(c)
  236. if err != nil {
  237. return err
  238. }
  239. err = command.Leave(cfg)
  240. return err
  241. },
  242. },
  243. {
  244. Name: "checkin",
  245. Usage: "Checks for local changes and then checks into the specified Netmaker network to ask about remote changes.",
  246. Flags: cliFlags,
  247. // the action, or code that will be executed when
  248. // we execute our `ns` command
  249. Action: func(c *cli.Context) error {
  250. cfg, _, err := config.GetCLIConfig(c)
  251. if err != nil {
  252. return err
  253. }
  254. err = command.CheckIn(cfg)
  255. return err
  256. },
  257. },
  258. {
  259. Name: "push",
  260. Usage: "Push configuration changes to server.",
  261. Flags: cliFlags,
  262. // the action, or code that will be executed when
  263. // we execute our `ns` command
  264. Action: func(c *cli.Context) error {
  265. cfg, _, err := config.GetCLIConfig(c)
  266. if err != nil {
  267. return err
  268. }
  269. err = command.Push(cfg)
  270. return err
  271. },
  272. },
  273. {
  274. Name: "pull",
  275. Usage: "Pull latest configuration and peers from server.",
  276. Flags: cliFlags,
  277. // the action, or code that will be executed when
  278. // we execute our `ns` command
  279. Action: func(c *cli.Context) error {
  280. cfg, _, err := config.GetCLIConfig(c)
  281. if err != nil {
  282. return err
  283. }
  284. err = command.Pull(cfg)
  285. return err
  286. },
  287. },
  288. {
  289. Name: "list",
  290. Usage: "Get list of networks.",
  291. Flags: cliFlags,
  292. // the action, or code that will be executed when
  293. // we execute our `ns` command
  294. Action: func(c *cli.Context) error {
  295. cfg, _, err := config.GetCLIConfig(c)
  296. if err != nil {
  297. return err
  298. }
  299. err = command.List(cfg)
  300. return err
  301. },
  302. },
  303. {
  304. Name: "uninstall",
  305. Usage: "Uninstall the netclient system service.",
  306. Flags: cliFlags,
  307. // the action, or code that will be executed when
  308. // we execute our `ns` command
  309. Action: func(c *cli.Context) error {
  310. err := command.Uninstall()
  311. return err
  312. },
  313. },
  314. }
  315. setGarbageCollection()
  316. if netclientutils.IsWindows() {
  317. ncwindows.InitWindows()
  318. } else {
  319. // start our application
  320. out, err := local.RunCmd("id -u")
  321. if err != nil {
  322. log.Fatal(out, err)
  323. }
  324. id, err := strconv.Atoi(string(out[:len(out)-1]))
  325. if err != nil {
  326. log.Fatal(err)
  327. }
  328. if id != 0 {
  329. log.Fatal("This program must be run with elevated privileges (sudo). This program installs a SystemD service and configures WireGuard and networking rules. Please re-run with sudo/root.")
  330. }
  331. _, err = exec.LookPath("wg")
  332. if err != nil {
  333. log.Println(err)
  334. log.Fatal("WireGuard not installed. Please install WireGuard (wireguard-tools) and try again.")
  335. }
  336. }
  337. if netclientutils.IsWindows() {
  338. if !local.IsWindowsWGInstalled() {
  339. log.Fatal("Please install Windows WireGuard before using Gravitl Netclient. https://download.wireguard.com/windows-client/wireguard-installer.exe")
  340. }
  341. }
  342. if len(os.Args) == 1 && netclientutils.IsWindows() {
  343. c := make(chan os.Signal)
  344. signal.Notify(c, os.Interrupt, syscall.SIGTERM)
  345. go func() {
  346. <-c
  347. log.Println("closing Gravitl Netclient")
  348. os.Exit(0)
  349. }()
  350. command.RunUserspaceDaemon()
  351. } else {
  352. err := app.Run(os.Args)
  353. if err != nil {
  354. log.Fatal(err)
  355. }
  356. }
  357. }
  358. func setGarbageCollection(){
  359. _, gcset := os.LookupEnv("GOGC");
  360. if !gcset {
  361. debug.SetGCPercent(netclientutils.DEFAULT_GC_PERCENT)
  362. }
  363. }