gitea源码

config.go 8.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Copyright 2019 The Gitea Authors. All rights reserved.
  3. // SPDX-License-Identifier: MIT
  4. package admin
  5. import (
  6. "net/http"
  7. "net/url"
  8. "strconv"
  9. "strings"
  10. system_model "code.gitea.io/gitea/models/system"
  11. "code.gitea.io/gitea/modules/cache"
  12. "code.gitea.io/gitea/modules/git"
  13. "code.gitea.io/gitea/modules/json"
  14. "code.gitea.io/gitea/modules/log"
  15. "code.gitea.io/gitea/modules/setting"
  16. "code.gitea.io/gitea/modules/setting/config"
  17. "code.gitea.io/gitea/modules/templates"
  18. "code.gitea.io/gitea/modules/util"
  19. "code.gitea.io/gitea/services/context"
  20. "code.gitea.io/gitea/services/mailer"
  21. "gitea.com/go-chi/session"
  22. )
  23. const (
  24. tplConfig templates.TplName = "admin/config"
  25. tplConfigSettings templates.TplName = "admin/config_settings/config_settings"
  26. )
  27. // SendTestMail send test mail to confirm mail service is OK
  28. func SendTestMail(ctx *context.Context) {
  29. email := ctx.FormString("email")
  30. // Send a test email to the user's email address and redirect back to Config
  31. if err := mailer.SendTestMail(email); err != nil {
  32. ctx.Flash.Error(ctx.Tr("admin.config.test_mail_failed", email, err))
  33. } else {
  34. ctx.Flash.Info(ctx.Tr("admin.config.test_mail_sent", email))
  35. }
  36. ctx.Redirect(setting.AppSubURL + "/-/admin/config")
  37. }
  38. // TestCache test the cache settings
  39. func TestCache(ctx *context.Context) {
  40. elapsed, err := cache.Test()
  41. if err != nil {
  42. ctx.Flash.Error(ctx.Tr("admin.config.cache_test_failed", err))
  43. } else {
  44. if elapsed > cache.SlowCacheThreshold {
  45. ctx.Flash.Warning(ctx.Tr("admin.config.cache_test_slow", elapsed))
  46. } else {
  47. ctx.Flash.Info(ctx.Tr("admin.config.cache_test_succeeded", elapsed))
  48. }
  49. }
  50. ctx.Redirect(setting.AppSubURL + "/-/admin/config")
  51. }
  52. func shadowPasswordKV(cfgItem, splitter string) string {
  53. fields := strings.Split(cfgItem, splitter)
  54. for i := range fields {
  55. if strings.HasPrefix(fields[i], "password=") {
  56. fields[i] = "password=******"
  57. break
  58. }
  59. }
  60. return strings.Join(fields, splitter)
  61. }
  62. func shadowURL(provider, cfgItem string) string {
  63. u, err := url.Parse(cfgItem)
  64. if err != nil {
  65. log.Error("Shadowing Password for %v failed: %v", provider, err)
  66. return cfgItem
  67. }
  68. if u.User != nil {
  69. atIdx := strings.Index(cfgItem, "@")
  70. if atIdx > 0 {
  71. colonIdx := strings.LastIndex(cfgItem[:atIdx], ":")
  72. if colonIdx > 0 {
  73. return cfgItem[:colonIdx+1] + "******" + cfgItem[atIdx:]
  74. }
  75. }
  76. }
  77. return cfgItem
  78. }
  79. func shadowPassword(provider, cfgItem string) string {
  80. switch provider {
  81. case "redis":
  82. return shadowPasswordKV(cfgItem, ",")
  83. case "mysql":
  84. // root:@tcp(localhost:3306)/macaron?charset=utf8
  85. atIdx := strings.Index(cfgItem, "@")
  86. if atIdx > 0 {
  87. colonIdx := strings.Index(cfgItem[:atIdx], ":")
  88. if colonIdx > 0 {
  89. return cfgItem[:colonIdx+1] + "******" + cfgItem[atIdx:]
  90. }
  91. }
  92. return cfgItem
  93. case "postgres":
  94. // user=jiahuachen dbname=macaron port=5432 sslmode=disable
  95. if !strings.HasPrefix(cfgItem, "postgres://") {
  96. return shadowPasswordKV(cfgItem, " ")
  97. }
  98. fallthrough
  99. case "couchbase":
  100. return shadowURL(provider, cfgItem)
  101. // postgres://pqgotest:password@localhost/pqgotest?sslmode=verify-full
  102. // Notice: use shadowURL
  103. }
  104. return cfgItem
  105. }
  106. // Config show admin config page
  107. func Config(ctx *context.Context) {
  108. ctx.Data["Title"] = ctx.Tr("admin.config_summary")
  109. ctx.Data["PageIsAdminConfig"] = true
  110. ctx.Data["PageIsAdminConfigSummary"] = true
  111. ctx.Data["CustomConf"] = setting.CustomConf
  112. ctx.Data["AppUrl"] = setting.AppURL
  113. ctx.Data["AppBuiltWith"] = setting.AppBuiltWith
  114. ctx.Data["Domain"] = setting.Domain
  115. ctx.Data["OfflineMode"] = setting.OfflineMode
  116. ctx.Data["RunUser"] = setting.RunUser
  117. ctx.Data["RunMode"] = util.ToTitleCase(setting.RunMode)
  118. ctx.Data["GitVersion"] = git.DefaultFeatures().VersionInfo()
  119. ctx.Data["AppDataPath"] = setting.AppDataPath
  120. ctx.Data["RepoRootPath"] = setting.RepoRootPath
  121. ctx.Data["CustomRootPath"] = setting.CustomPath
  122. ctx.Data["LogRootPath"] = setting.Log.RootPath
  123. ctx.Data["ScriptType"] = setting.ScriptType
  124. ctx.Data["ReverseProxyAuthUser"] = setting.ReverseProxyAuthUser
  125. ctx.Data["ReverseProxyAuthEmail"] = setting.ReverseProxyAuthEmail
  126. ctx.Data["SSH"] = setting.SSH
  127. ctx.Data["LFS"] = setting.LFS
  128. ctx.Data["Service"] = setting.Service
  129. ctx.Data["DbCfg"] = setting.Database
  130. ctx.Data["Webhook"] = setting.Webhook
  131. ctx.Data["MailerEnabled"] = false
  132. if setting.MailService != nil {
  133. ctx.Data["MailerEnabled"] = true
  134. ctx.Data["Mailer"] = setting.MailService
  135. }
  136. ctx.Data["CacheAdapter"] = setting.CacheService.Adapter
  137. ctx.Data["CacheInterval"] = setting.CacheService.Interval
  138. ctx.Data["CacheConn"] = shadowPassword(setting.CacheService.Adapter, setting.CacheService.Conn)
  139. ctx.Data["CacheItemTTL"] = setting.CacheService.TTL
  140. sessionCfg := setting.SessionConfig
  141. if sessionCfg.Provider == "VirtualSession" {
  142. var realSession session.Options
  143. if err := json.Unmarshal([]byte(sessionCfg.ProviderConfig), &realSession); err != nil {
  144. log.Error("Unable to unmarshall session config for virtual provider config: %s\nError: %v", sessionCfg.ProviderConfig, err)
  145. }
  146. sessionCfg.Provider = realSession.Provider
  147. sessionCfg.ProviderConfig = realSession.ProviderConfig
  148. sessionCfg.CookieName = realSession.CookieName
  149. sessionCfg.CookiePath = realSession.CookiePath
  150. sessionCfg.Gclifetime = realSession.Gclifetime
  151. sessionCfg.Maxlifetime = realSession.Maxlifetime
  152. sessionCfg.Secure = realSession.Secure
  153. sessionCfg.Domain = realSession.Domain
  154. }
  155. sessionCfg.ProviderConfig = shadowPassword(sessionCfg.Provider, sessionCfg.ProviderConfig)
  156. ctx.Data["SessionConfig"] = sessionCfg
  157. ctx.Data["Git"] = setting.Git
  158. ctx.Data["AccessLogTemplate"] = setting.Log.AccessLogTemplate
  159. ctx.Data["LogSQL"] = setting.Database.LogSQL
  160. ctx.Data["Loggers"] = log.GetManager().DumpLoggers()
  161. config.GetDynGetter().InvalidateCache()
  162. prepareStartupProblemsAlert(ctx)
  163. ctx.HTML(http.StatusOK, tplConfig)
  164. }
  165. func ConfigSettings(ctx *context.Context) {
  166. ctx.Data["Title"] = ctx.Tr("admin.config_settings")
  167. ctx.Data["PageIsAdminConfig"] = true
  168. ctx.Data["PageIsAdminConfigSettings"] = true
  169. ctx.Data["DefaultOpenWithEditorAppsString"] = setting.DefaultOpenWithEditorApps().ToTextareaString()
  170. ctx.HTML(http.StatusOK, tplConfigSettings)
  171. }
  172. func ChangeConfig(ctx *context.Context) {
  173. cfg := setting.Config()
  174. marshalBool := func(v string) ([]byte, error) {
  175. b, _ := strconv.ParseBool(v)
  176. return json.Marshal(b)
  177. }
  178. marshalString := func(emptyDefault string) func(v string) ([]byte, error) {
  179. return func(v string) ([]byte, error) {
  180. return json.Marshal(util.IfZero(v, emptyDefault))
  181. }
  182. }
  183. marshalOpenWithApps := func(value string) ([]byte, error) {
  184. // TODO: move the block alongside OpenWithEditorAppsType.ToTextareaString
  185. lines := strings.Split(value, "\n")
  186. var openWithEditorApps setting.OpenWithEditorAppsType
  187. for _, line := range lines {
  188. line = strings.TrimSpace(line)
  189. if line == "" {
  190. continue
  191. }
  192. displayName, openURL, ok := strings.Cut(line, "=")
  193. displayName, openURL = strings.TrimSpace(displayName), strings.TrimSpace(openURL)
  194. if !ok || displayName == "" || openURL == "" {
  195. continue
  196. }
  197. openWithEditorApps = append(openWithEditorApps, setting.OpenWithEditorApp{
  198. DisplayName: strings.TrimSpace(displayName),
  199. OpenURL: strings.TrimSpace(openURL),
  200. })
  201. }
  202. return json.Marshal(openWithEditorApps)
  203. }
  204. marshallers := map[string]func(string) ([]byte, error){
  205. cfg.Picture.DisableGravatar.DynKey(): marshalBool,
  206. cfg.Picture.EnableFederatedAvatar.DynKey(): marshalBool,
  207. cfg.Repository.OpenWithEditorApps.DynKey(): marshalOpenWithApps,
  208. cfg.Repository.GitGuideRemoteName.DynKey(): marshalString(cfg.Repository.GitGuideRemoteName.DefaultValue()),
  209. }
  210. _ = ctx.Req.ParseForm()
  211. configKeys := ctx.Req.Form["key"]
  212. configValues := ctx.Req.Form["value"]
  213. configSettings := map[string]string{}
  214. loop:
  215. for i, key := range configKeys {
  216. if i >= len(configValues) {
  217. ctx.JSONError(ctx.Tr("admin.config.set_setting_failed", key))
  218. break loop
  219. }
  220. value := configValues[i]
  221. marshaller, hasMarshaller := marshallers[key]
  222. if !hasMarshaller {
  223. ctx.JSONError(ctx.Tr("admin.config.set_setting_failed", key))
  224. break loop
  225. }
  226. marshaledValue, err := marshaller(value)
  227. if err != nil {
  228. ctx.JSONError(ctx.Tr("admin.config.set_setting_failed", key))
  229. break loop
  230. }
  231. configSettings[key] = string(marshaledValue)
  232. }
  233. if ctx.Written() {
  234. return
  235. }
  236. if err := system_model.SetSettings(ctx, configSettings); err != nil {
  237. ctx.ServerError("SetSettings", err)
  238. return
  239. }
  240. config.GetDynGetter().InvalidateCache()
  241. ctx.JSONOK()
  242. }