config.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. package config
  2. // code for managing groundseg and container configurations
  3. import (
  4. "encoding/json"
  5. "fmt"
  6. "goseg/defaults"
  7. "goseg/structs"
  8. "io/ioutil"
  9. "log/slog"
  10. "net"
  11. "os"
  12. "path/filepath"
  13. "runtime"
  14. "sync"
  15. "time"
  16. )
  17. var (
  18. logger = slog.New(slog.NewJSONHandler(os.Stdout, nil))
  19. // global settings config (accessed via funcs)
  20. globalConfig structs.SysConfig
  21. // base path for installation (override default with env var)
  22. BasePath = os.Getenv("GS_BASE_PATH")
  23. // only amd64 or arm64
  24. Architecture = getArchitecture()
  25. // struct of /retrieve blob
  26. StartramConfig structs.StartramRetrieve
  27. // unused for now, set with `./groundseg dev`
  28. DebugMode = false
  29. Ready = false
  30. // representation of desired/actual container states
  31. GSContainers = make(map[string]structs.ContainerState)
  32. DockerDir = "/var/lib/docker/volumes/"
  33. // version server check
  34. checkInterval = 5 * time.Minute
  35. confMutex sync.Mutex
  36. contMutex sync.Mutex
  37. versMutex sync.Mutex
  38. )
  39. // try initializing from system.json on disk
  40. func init() {
  41. logger.Info("Starting GroundSeg")
  42. logger.Info("Urbit is love <3")
  43. for _, arg := range os.Args[1:] {
  44. // trigger this with `./groundseg dev`
  45. if arg == "dev" {
  46. logger.Info("Starting GroundSeg in debug mode")
  47. DebugMode = true
  48. }
  49. }
  50. if BasePath == "" {
  51. // default base path
  52. BasePath = "/opt/nativeplanet/groundseg"
  53. }
  54. pathMsg := fmt.Sprintf("Loading configs from %s", BasePath)
  55. logger.Info(pathMsg)
  56. confPath := filepath.Join(BasePath, "settings", "system.json")
  57. file, err := os.Open(confPath)
  58. if err != nil {
  59. // create a default if it doesn't exist
  60. err = createDefaultConf()
  61. if err != nil {
  62. // panic if we can't create it
  63. errmsg := fmt.Sprintf("Unable to create config! Please elevate permissions. %v", err)
  64. logger.Error(errmsg)
  65. panic(errmsg)
  66. }
  67. // generate and insert wireguard keys
  68. wgPriv, wgPub, err := WgKeyGen()
  69. if err != nil {
  70. logger.Error(fmt.Sprintf("%v",err))
  71. } else {
  72. err = UpdateConf(map[string]interface{}{
  73. "Pubkey": wgPub,
  74. "Privkey": wgPriv,
  75. })
  76. if err != nil {
  77. logger.Error(fmt.Sprintf("%v",err))
  78. }
  79. }
  80. }
  81. defer file.Close()
  82. // read the sysconfig to memory
  83. decoder := json.NewDecoder(file)
  84. err = decoder.Decode(&globalConfig)
  85. if err != nil {
  86. errmsg := fmt.Sprintf("Error decoding JSON: %v", err)
  87. logger.Error(errmsg)
  88. }
  89. }
  90. // return the global conf var
  91. func Conf() structs.SysConfig {
  92. confMutex.Lock()
  93. defer confMutex.Unlock()
  94. return globalConfig
  95. }
  96. // tell if we're amd64 or arm64
  97. func getArchitecture() string {
  98. switch runtime.GOARCH {
  99. case "arm64", "aarch64":
  100. return "arm64"
  101. default:
  102. return "amd64"
  103. }
  104. }
  105. // update by passing in a map of key:values you want to modify
  106. func UpdateConf(values map[string]interface{}) error {
  107. // mutex lock to avoid race conditions
  108. confMutex.Lock()
  109. defer confMutex.Unlock()
  110. confPath := filepath.Join(BasePath, "settings", "system.json")
  111. file, err := ioutil.ReadFile(confPath)
  112. if err != nil {
  113. errmsg := fmt.Sprintf("Unable to load config: %v", err)
  114. logger.Error(errmsg)
  115. return err
  116. }
  117. // unmarshal the config to struct
  118. var configMap map[string]interface{}
  119. if err := json.Unmarshal(file, &configMap); err != nil {
  120. errmsg := fmt.Sprintf("Error decoding JSON: %v", err)
  121. logger.Error(errmsg)
  122. return err
  123. }
  124. // update our unmarshaled struct
  125. for key, value := range values {
  126. configMap[key] = value
  127. }
  128. // marshal and persist it
  129. updatedJSON, err := json.MarshalIndent(configMap, "", " ")
  130. if err != nil {
  131. errmsg := fmt.Sprintf("Error encoding JSON: %v", err)
  132. logger.Error(errmsg)
  133. return err
  134. }
  135. // update the globalConfig var
  136. if err := json.Unmarshal(updatedJSON, &globalConfig); err != nil {
  137. errmsg := fmt.Sprintf("Error updating global config: %v", err)
  138. logger.Error(errmsg)
  139. return err
  140. }
  141. if err := ioutil.WriteFile(confPath, updatedJSON, 0644); err != nil {
  142. errmsg := fmt.Sprintf("Error writing to file: %v", err)
  143. logger.Error(errmsg)
  144. return err
  145. }
  146. return nil
  147. }
  148. // we keep map[string]structs.ContainerState in memory to keep track of the containers
  149. // eg if they're running and whether they should be
  150. // modify the desired/actual state of containers
  151. func UpdateContainerState(name string, containerState structs.ContainerState) {
  152. contMutex.Lock()
  153. defer contMutex.Unlock()
  154. GSContainers[name] = containerState
  155. res, _ := json.Marshal(containerState)
  156. logger.Info(fmt.Sprintf("%s:%s", name, string(res)))
  157. }
  158. // get the current container state
  159. func GetContainerState() map[string]structs.ContainerState {
  160. contMutex.Lock()
  161. defer contMutex.Unlock()
  162. return GSContainers
  163. }
  164. // write a default conf to disk
  165. func createDefaultConf() error {
  166. defaultConfig := defaults.SysConfig(BasePath)
  167. path := filepath.Join(BasePath, "settings", "system.json")
  168. if err := os.MkdirAll(filepath.Dir(path), os.ModePerm); err != nil {
  169. return err
  170. }
  171. file, err := os.Create(path)
  172. if err != nil {
  173. return err
  174. }
  175. defer file.Close()
  176. encoder := json.NewEncoder(file)
  177. encoder.SetIndent("", " ")
  178. if err := encoder.Encode(&defaultConfig); err != nil {
  179. return err
  180. }
  181. return nil
  182. }
  183. // check outbound tcp connectivity
  184. // takes ip:port
  185. func NetCheck(netCheck string) bool {
  186. logger.Info("Checking internet access")
  187. internet := false
  188. timeout := 3 * time.Second
  189. conn, err := net.DialTimeout("tcp", netCheck, timeout)
  190. if err != nil {
  191. errmsg := fmt.Sprintf("Check internet access error: %v", err)
  192. logger.Error(errmsg)
  193. } else {
  194. internet = true
  195. _ = conn.Close()
  196. }
  197. return internet
  198. }