config.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  1. package config
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "goseg/structs"
  6. "io/ioutil"
  7. "log/slog"
  8. "net"
  9. "net/http"
  10. "os"
  11. "path/filepath"
  12. "sync"
  13. "time"
  14. )
  15. var (
  16. globalConfig structs.SysConfig
  17. logger = slog.New(slog.NewJSONHandler(os.Stdout, nil))
  18. BasePath = "/opt/nativeplanet/groundseg"
  19. Version = "v2.0.0"
  20. Ready = false
  21. VersionServerReady = false
  22. VersionInfo structs.Version
  23. Ram int
  24. Cpu int
  25. CoreTemp int
  26. Disk int
  27. WifiEnabled = false
  28. ActiveNetwork string
  29. WifiNetworks []string
  30. HttpOpen = false
  31. UploadSecret string
  32. checkInterval = 5 * time.Minute
  33. confMutex sync.Mutex
  34. versMutex sync.Mutex
  35. )
  36. // try initializing from system.json on disk
  37. func init() {
  38. pathMsg := fmt.Sprintf("Loading configs from %s", BasePath)
  39. logger.Info(pathMsg)
  40. confPath := filepath.Join(BasePath, "settings", "system.json")
  41. file, err := os.Open(confPath)
  42. if err != nil {
  43. // create a default if it doesn't exist
  44. err = createDefaultConf()
  45. if err != nil {
  46. errmsg := fmt.Sprintf("Unable to create config! %v", err)
  47. logger.Error(errmsg)
  48. }
  49. }
  50. defer file.Close()
  51. decoder := json.NewDecoder(file)
  52. err = decoder.Decode(&globalConfig)
  53. if err != nil {
  54. errmsg := fmt.Sprintf("Error decoding JSON: %v", err)
  55. logger.Error(errmsg)
  56. }
  57. }
  58. // return the global conf var
  59. func Conf() structs.SysConfig {
  60. confMutex.Lock()
  61. defer confMutex.Unlock()
  62. return globalConfig
  63. }
  64. // update by passing in a map of key:values you want to modify
  65. func UpdateConf(values map[string]interface{}) error {
  66. // mutex lock to avoid race conditions
  67. confMutex.Lock()
  68. defer confMutex.Unlock()
  69. confPath := filepath.Join(BasePath, "settings", "system.json")
  70. file, err := ioutil.ReadFile(confPath)
  71. if err != nil {
  72. errmsg := fmt.Sprintf("Unable to load config: %v", err)
  73. logger.Error(errmsg)
  74. return err
  75. }
  76. // unmarshal the config to struct
  77. var configMap map[string]interface{}
  78. if err := json.Unmarshal(file, &configMap); err != nil {
  79. errmsg := fmt.Sprintf("Error decoding JSON: %v", err)
  80. logger.Error(errmsg)
  81. return err
  82. }
  83. // update our unmarshaled struct
  84. for key, value := range values {
  85. configMap[key] = value
  86. }
  87. // marshal and persist it
  88. updatedJSON, err := json.MarshalIndent(configMap, "", " ")
  89. if err != nil {
  90. errmsg := fmt.Sprintf("Error encoding JSON: %v", err)
  91. logger.Error(errmsg)
  92. return err
  93. }
  94. // update the globalConfig var
  95. if err := json.Unmarshal(updatedJSON, &globalConfig); err != nil {
  96. errmsg := fmt.Sprintf("Error updating global config: %v", err)
  97. logger.Error(errmsg)
  98. return err
  99. }
  100. if err := ioutil.WriteFile(confPath, updatedJSON, 0644); err != nil {
  101. errmsg := fmt.Sprintf("Error writing to file: %v", err)
  102. logger.Error(errmsg)
  103. return err
  104. }
  105. return nil
  106. }
  107. // write a default conf to disk
  108. func createDefaultConf() error {
  109. defaultConfig := structs.SysConfig{
  110. Setup: "start",
  111. EndpointUrl: "api.startram.io",
  112. ApiVersion: "v1",
  113. Piers: []string{},
  114. NetCheck: "1.1.1.1:53",
  115. UpdateMode: "auto",
  116. UpdateUrl: "https://version.groundseg.app",
  117. UpdateBranch: "latest",
  118. SwapVal: 16,
  119. SwapFile: filepath.Join(BasePath, "settings", "swapfile"),
  120. KeyFile: filepath.Join(BasePath, "settings", "session.key"),
  121. Sessions: struct {
  122. Authorized map[string]structs.SessionInfo `json:"authorized"`
  123. Unauthorized map[string]structs.SessionInfo `json:"unauthorized"`
  124. }{
  125. Authorized: make(map[string]structs.SessionInfo),
  126. Unauthorized: make(map[string]structs.SessionInfo),
  127. },
  128. LinuxUpdates: struct {
  129. Value int `json:"value"`
  130. Interval string `json:"interval"`
  131. Previous bool `json:"previous"`
  132. }{
  133. Value: 1,
  134. Interval: "week",
  135. Previous: false,
  136. },
  137. DockerData: "/var/lib/docker",
  138. WgOn: false,
  139. WgRegistered: false,
  140. PwHash: "",
  141. C2cInterval: 0,
  142. FirstBoot: false,
  143. WgRegisterd: false,
  144. GsVersion: Version,
  145. CfgDir: "",
  146. UpdateInterval: 0,
  147. BinHash: "",
  148. Pubkey: "",
  149. Privkey: "",
  150. Salt: "",
  151. }
  152. path := filepath.Join(BasePath, "settings", "system.json")
  153. if err := os.MkdirAll(filepath.Dir(path), os.ModePerm); err != nil {
  154. return err
  155. }
  156. file, err := os.Create(path)
  157. if err != nil {
  158. return err
  159. }
  160. defer file.Close()
  161. encoder := json.NewEncoder(file)
  162. encoder.SetIndent("", " ")
  163. if err := encoder.Encode(&defaultConfig); err != nil {
  164. return err
  165. }
  166. return nil
  167. }
  168. // check outbound tcp connectivity
  169. // takes ip:port
  170. func NetCheck(netCheck string) bool {
  171. logger.Info("Checking internet access")
  172. internet := false
  173. timeout := 3 * time.Second
  174. conn, err := net.DialTimeout("tcp", netCheck, timeout)
  175. if err != nil {
  176. errmsg := fmt.Sprintf("Check internet access error: %v", err)
  177. logger.Error(errmsg)
  178. } else {
  179. internet = true
  180. _ = conn.Close()
  181. }
  182. return internet
  183. }
  184. // check the version server and return unmarshaled result
  185. func CheckVersion() (structs.Version, bool) {
  186. versMutex.Lock()
  187. defer versMutex.Unlock()
  188. const retries = 10
  189. const delay = time.Second
  190. url := globalConfig.UpdateUrl
  191. for i := 0; i < retries; i++ {
  192. resp, err := http.Get(url)
  193. if err != nil {
  194. errmsg := fmt.Sprintf("Unable to connect to update server: %v", err)
  195. logger.Warn(errmsg)
  196. if i < retries-1 {
  197. time.Sleep(delay)
  198. continue
  199. } else {
  200. return VersionInfo, false
  201. }
  202. }
  203. // read the body bytes
  204. body, err := ioutil.ReadAll(resp.Body)
  205. resp.Body.Close()
  206. if err != nil {
  207. errmsg := fmt.Sprintf("Error reading version info: %v", err)
  208. logger.Warn(errmsg)
  209. if i < retries-1 {
  210. time.Sleep(delay)
  211. continue
  212. } else {
  213. return VersionInfo, false
  214. }
  215. }
  216. // unmarshal values into Version struct
  217. err = json.Unmarshal(body, &VersionInfo)
  218. if err != nil {
  219. errmsg := fmt.Sprintf("Error unmarshalling JSON: %v", err)
  220. logger.Warn(errmsg)
  221. if i < retries-1 {
  222. time.Sleep(delay)
  223. continue
  224. } else {
  225. return VersionInfo, false
  226. }
  227. }
  228. // debug: re-marshal and write to disk
  229. confPath := filepath.Join(BasePath, "settings", "version_info.json")
  230. file, err := os.Create(confPath)
  231. if err != nil {
  232. errmsg := fmt.Sprintf("Failed to create file: %v", err)
  233. logger.Error(errmsg)
  234. return VersionInfo, false
  235. }
  236. defer file.Close()
  237. encoder := json.NewEncoder(file)
  238. encoder.SetIndent("", " ")
  239. if err := encoder.Encode(&VersionInfo); err != nil {
  240. errmsg := fmt.Sprintf("Failed to write JSON: %v", err)
  241. logger.Error(errmsg)
  242. }
  243. return VersionInfo, true
  244. }
  245. return VersionInfo, false
  246. }
  247. func CheckVersionLoop() {
  248. ticker := time.NewTicker(checkInterval)
  249. for {
  250. select {
  251. case <-ticker.C:
  252. latestVersion, _ := CheckVersion()
  253. currentVersion := VersionInfo
  254. if latestVersion != currentVersion {
  255. fmt.Printf("New version available! Current: %s, Latest: %s\n", currentVersion, latestVersion)
  256. // Handle the update logic here
  257. }
  258. }
  259. }
  260. }