main.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. package main
  2. import (
  3. "flag"
  4. "fmt"
  5. "os"
  6. "path/filepath"
  7. "strconv"
  8. "github.com/OliveTin/OliveTin/internal/api"
  9. config "github.com/OliveTin/OliveTin/internal/config"
  10. "github.com/knadh/koanf/parsers/yaml"
  11. "github.com/knadh/koanf/providers/file"
  12. "github.com/knadh/koanf/v2"
  13. log "github.com/sirupsen/logrus"
  14. )
  15. func printPwd() {
  16. pwd, err := os.Getwd()
  17. if err != nil {
  18. log.Fatalf("Error getting working directory: %v", err)
  19. }
  20. log.Infof("Working directory: %s", pwd)
  21. }
  22. func main() {
  23. resetPasswords := flag.Bool("passwords", true, "Reset passwords")
  24. flag.Parse()
  25. log.Info("Config tool started")
  26. printPwd()
  27. k := koanf.New(".")
  28. configPath, err := filepath.Abs("../config.yaml")
  29. if err != nil {
  30. log.Fatalf("Error getting absolute config path: %v", err)
  31. }
  32. log.Infof("Loading config from %s", configPath)
  33. backupOriginalConfig(configPath)
  34. err = k.Load(file.Provider(configPath), yaml.Parser())
  35. if err != nil {
  36. log.Fatalf("Error loading config: %v", err)
  37. }
  38. cfg := &config.Config{}
  39. config.AppendSource(cfg, k, configPath)
  40. if *resetPasswords {
  41. resetAllPasswords(k, cfg)
  42. }
  43. saveConfig(configPath, k)
  44. }
  45. func backupOriginalConfig(configPath string) {
  46. originalConfigPath := filepath.Join(filepath.Dir(configPath), "config.original.yaml")
  47. _, err := os.Stat(originalConfigPath)
  48. if err == nil {
  49. log.Infof("Backup already exists at %s, skipping backup to preserve original", originalConfigPath)
  50. return
  51. }
  52. if !os.IsNotExist(err) {
  53. log.Fatalf("Error checking backup file: %v", err)
  54. }
  55. data, err := os.ReadFile(configPath)
  56. if err != nil {
  57. log.Fatalf("Error reading config for backup: %v", err)
  58. }
  59. err = os.WriteFile(originalConfigPath, data, 0644)
  60. if err != nil {
  61. log.Fatalf("Error writing backup config: %v", err)
  62. }
  63. log.Infof("Original config backed up to %s", originalConfigPath)
  64. }
  65. func resetAllPasswords(k *koanf.Koanf, cfg *config.Config) {
  66. if !cfg.AuthLocalUsers.Enabled || len(cfg.AuthLocalUsers.Users) == 0 {
  67. log.Info("No local users found, skipping password reset")
  68. return
  69. }
  70. hashedPassword, err := api.CreateHash("password")
  71. if err != nil {
  72. log.Fatalf("Error creating password hash: %v", err)
  73. }
  74. usersSlice := k.Get("authLocalUsers.users")
  75. usersSliceTyped, ok := usersSlice.([]interface{})
  76. if ok && len(usersSliceTyped) > 0 {
  77. newUsersSlice := make([]interface{}, len(usersSliceTyped))
  78. for index, userValue := range usersSliceTyped {
  79. userMap, ok := userValue.(map[string]interface{})
  80. if !ok {
  81. log.Warnf("User entry at index %d is not a map, skipping", index)
  82. newUsersSlice[index] = userValue
  83. continue
  84. }
  85. oldPassword, _ := userMap["password"].(string)
  86. username, _ := userMap["username"].(string)
  87. if username == "" {
  88. username = fmt.Sprintf("user[%d]", index)
  89. }
  90. newUserMap := make(map[string]interface{})
  91. for k, v := range userMap {
  92. newUserMap[k] = v
  93. }
  94. newUserMap["password"] = hashedPassword
  95. newUsersSlice[index] = newUserMap
  96. oldHashPreview := oldPassword
  97. if len(oldPassword) > 20 {
  98. oldHashPreview = oldPassword[:20]
  99. }
  100. log.Infof("Reset password for user '%s' (old hash: %s...)", username, oldHashPreview)
  101. }
  102. err = k.Set("authLocalUsers.users", newUsersSlice)
  103. if err != nil {
  104. log.WithFields(log.Fields{
  105. "error": err,
  106. }).Fatalf("Error setting users")
  107. }
  108. } else {
  109. for index, user := range cfg.AuthLocalUsers.Users {
  110. key := "authLocalUsers.users." + strconv.Itoa(index) + ".password"
  111. err = k.Set(key, hashedPassword)
  112. if err != nil {
  113. log.WithFields(log.Fields{
  114. "error": err,
  115. }).Fatalf("Error setting user password")
  116. }
  117. oldHashPreview := user.Password
  118. if len(oldHashPreview) > 20 {
  119. oldHashPreview = oldHashPreview[:20]
  120. }
  121. log.Infof("Reset password for user '%s' (old hash: %s...)", user.Username, oldHashPreview)
  122. }
  123. }
  124. log.Infof("Reset %d password(s) to 'password'", len(cfg.AuthLocalUsers.Users))
  125. }
  126. func saveConfig(configPath string, k *koanf.Koanf) {
  127. out, err := k.Marshal(yaml.Parser())
  128. if err != nil {
  129. log.Fatalf("Error marshalling config: %v", err)
  130. }
  131. err = os.WriteFile(configPath, out, 0644)
  132. if err != nil {
  133. log.Fatalf("Error saving config: %v", err)
  134. }
  135. log.Infof("Config saved to %s", configPath)
  136. }