main.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package main
  2. /*
  3. Two-Nodes
  4. A simple demo for showcasing the go-DDDNS working in a two nodes environment
  5. */
  6. import (
  7. "errors"
  8. "flag"
  9. "io/ioutil"
  10. "log"
  11. "net/http"
  12. "os"
  13. "os/signal"
  14. "strconv"
  15. "syscall"
  16. "time"
  17. godddns "github.com/tobychui/go-DDDNS/godddns"
  18. )
  19. var (
  20. //Flags
  21. //Current node
  22. thisNodeUUID = flag.String("nid", "node1", "The UUID of this node")
  23. //Remote nodes
  24. remoteNodeAUUID = flag.String("rida", "node2", "The UUID of the remote node A")
  25. remoteNodeAIPAddr = flag.String("ripa", "127.0.0.1", "The remote node A current IP address")
  26. remoteNodeBUUID = flag.String("ridb", "node3", "The UUID of the remote node B")
  27. remoteNodeBIPAddr = flag.String("ripb", "127.0.0.1", "The remote node B current IP address")
  28. //The following flags assume both remote node A and B are hosted on the same port and with same connection endpoint
  29. connPath = flag.String("conn", "/conn", "The relative path for connection establishment")
  30. thisNodeRecvPort = flag.Int("np", 8080, "This node receive port")
  31. remoteNodeRecvPort = flag.Int("rp", 8080, "Remote nodes (A and B) receive port")
  32. //Paramter, which we hardcoded here for now
  33. syncInterval int64 = 10
  34. serviceRouter *godddns.ServiceRouter
  35. )
  36. //Dummy function which allow all login to pass through
  37. func ValidateCred(username string, password string) bool {
  38. return true
  39. }
  40. func cleanup(configFilename string) {
  41. //Save the router record to json file
  42. js, err := serviceRouter.ExportRouterToJSON()
  43. if err != nil {
  44. panic(err)
  45. }
  46. ioutil.WriteFile(configFilename, []byte(js), 0755)
  47. //Close the service router
  48. serviceRouter.Close()
  49. }
  50. func main() {
  51. //Set-ups
  52. flag.Parse()
  53. configFilename := *thisNodeUUID + "_config.json"
  54. c := make(chan os.Signal)
  55. signal.Notify(c, os.Interrupt, syscall.SIGTERM)
  56. go func() {
  57. <-c
  58. cleanup(configFilename)
  59. os.Exit(1)
  60. }()
  61. //Step 1: we want to create a new service router based on the given information
  62. if fileExists(configFilename) {
  63. //Load service router from file
  64. s, err := godddns.NewRouterFromJSONFile(configFilename)
  65. if err != nil {
  66. log.Println("Failed to load service router from file: ")
  67. panic(err)
  68. }
  69. s.InjectAuthFunction(ValidateCred)
  70. serviceRouter = s
  71. log.Println("Service Router Config Loaded")
  72. } else {
  73. //Create a new service router
  74. serviceRouter = godddns.NewServiceRouter(godddns.RouterOptions{
  75. DeviceUUID: *thisNodeUUID,
  76. AuthFunction: ValidateCred,
  77. SyncInterval: syncInterval,
  78. })
  79. log.Println("Service Router Created")
  80. }
  81. //Optional: Enable verbal output on router
  82. serviceRouter.Options.Verbal = true
  83. //Step 2: Create HTTP listening endpoint for incoming connection requests and heartbeat
  84. go func() {
  85. http.HandleFunc("/conn", serviceRouter.HandleConnections)
  86. log.Println("HTTP Listening endpoint started on :" + strconv.Itoa(*thisNodeRecvPort))
  87. http.ListenAndServe(":"+strconv.Itoa(*thisNodeRecvPort), nil)
  88. }()
  89. if !fileExists(configFilename) {
  90. //Step 3: we want to add the remote nodes to the list
  91. //Node A
  92. nodeA := serviceRouter.NewNode(godddns.NodeOptions{
  93. NodeID: *remoteNodeAUUID,
  94. Port: *remoteNodeRecvPort,
  95. RESTInterface: *connPath,
  96. RequireHTTPS: false,
  97. })
  98. serviceRouter.AddNode(nodeA)
  99. log.Println("Remote NodeA Added")
  100. //Node B
  101. nodeB := serviceRouter.NewNode(godddns.NodeOptions{
  102. NodeID: *remoteNodeBUUID,
  103. Port: *remoteNodeRecvPort,
  104. RESTInterface: *connPath,
  105. RequireHTTPS: false,
  106. })
  107. serviceRouter.AddNode(nodeB)
  108. log.Println("Remote NodeB Added")
  109. //Step 4: Start the connection to Node 1 (Replace username and password with real value here)
  110. retryCount := 0
  111. retry := true
  112. for retry {
  113. if retryCount > 10 {
  114. log.Println("Connection to remote node " + nodeA.UUID + " failed. Skipping.")
  115. break
  116. }
  117. assignedSendingTOTP, err := nodeA.StartConnection(*remoteNodeAIPAddr, "username", "password")
  118. if err != nil {
  119. log.Println("Unable to establish connection to Node A. Retrying in 3 seconds", assignedSendingTOTP)
  120. retryCount++
  121. time.Sleep(3 * time.Second)
  122. } else {
  123. log.Println("Remote Node A TOTP exchange done:", assignedSendingTOTP)
  124. retry = false
  125. }
  126. }
  127. //Start connection to Node 2
  128. retryCount = 0
  129. retry = true
  130. for retry {
  131. if retryCount > 10 {
  132. log.Println("Connection to remote node " + nodeB.UUID + " failed. Skipping.")
  133. break
  134. }
  135. assignedSendingTOTP, err := nodeB.StartConnection(*remoteNodeBIPAddr, "username", "password")
  136. if err != nil {
  137. log.Println("Unable to establish connection to Node B. Retrying in 3 seconds", assignedSendingTOTP)
  138. retryCount++
  139. time.Sleep(3 * time.Second)
  140. } else {
  141. log.Println("Node B TOTP exchange done:", assignedSendingTOTP)
  142. retry = false
  143. }
  144. }
  145. }
  146. //Step 4: Start Heartbeat to the remote node(s)
  147. serviceRouter.StartHeartBeat()
  148. log.Println("Heartbeat schedule started")
  149. //Do a blocking loop
  150. select {}
  151. }
  152. func fileExists(filename string) bool {
  153. if _, err := os.Stat(filename); errors.Is(err, os.ErrNotExist) {
  154. return false
  155. }
  156. return true
  157. }