main.go 9.9 KB

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