gitea源码

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  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 org
  5. import (
  6. "net/http"
  7. "net/url"
  8. "code.gitea.io/gitea/models/db"
  9. packages_model "code.gitea.io/gitea/models/packages"
  10. repo_model "code.gitea.io/gitea/models/repo"
  11. user_model "code.gitea.io/gitea/models/user"
  12. "code.gitea.io/gitea/models/webhook"
  13. "code.gitea.io/gitea/modules/log"
  14. "code.gitea.io/gitea/modules/optional"
  15. repo_module "code.gitea.io/gitea/modules/repository"
  16. "code.gitea.io/gitea/modules/setting"
  17. "code.gitea.io/gitea/modules/structs"
  18. "code.gitea.io/gitea/modules/templates"
  19. "code.gitea.io/gitea/modules/util"
  20. "code.gitea.io/gitea/modules/web"
  21. shared_user "code.gitea.io/gitea/routers/web/shared/user"
  22. user_setting "code.gitea.io/gitea/routers/web/user/setting"
  23. "code.gitea.io/gitea/services/context"
  24. "code.gitea.io/gitea/services/forms"
  25. org_service "code.gitea.io/gitea/services/org"
  26. user_service "code.gitea.io/gitea/services/user"
  27. )
  28. const (
  29. // tplSettingsOptions template path for render settings
  30. tplSettingsOptions templates.TplName = "org/settings/options"
  31. // tplSettingsHooks template path for render hook settings
  32. tplSettingsHooks templates.TplName = "org/settings/hooks"
  33. // tplSettingsLabels template path for render labels settings
  34. tplSettingsLabels templates.TplName = "org/settings/labels"
  35. )
  36. // Settings render the main settings page
  37. func Settings(ctx *context.Context) {
  38. ctx.Data["Title"] = ctx.Tr("org.settings")
  39. ctx.Data["PageIsOrgSettings"] = true
  40. ctx.Data["PageIsSettingsOptions"] = true
  41. ctx.Data["CurrentVisibility"] = ctx.Org.Organization.Visibility
  42. ctx.Data["RepoAdminChangeTeamAccess"] = ctx.Org.Organization.RepoAdminChangeTeamAccess
  43. ctx.Data["ContextUser"] = ctx.ContextUser
  44. if _, err := shared_user.RenderUserOrgHeader(ctx); err != nil {
  45. ctx.ServerError("RenderUserOrgHeader", err)
  46. return
  47. }
  48. ctx.HTML(http.StatusOK, tplSettingsOptions)
  49. }
  50. // SettingsPost response for settings change submitted
  51. func SettingsPost(ctx *context.Context) {
  52. form := web.GetForm(ctx).(*forms.UpdateOrgSettingForm)
  53. ctx.Data["Title"] = ctx.Tr("org.settings")
  54. ctx.Data["PageIsOrgSettings"] = true
  55. ctx.Data["PageIsSettingsOptions"] = true
  56. ctx.Data["CurrentVisibility"] = ctx.Org.Organization.Visibility
  57. if ctx.HasError() {
  58. ctx.HTML(http.StatusOK, tplSettingsOptions)
  59. return
  60. }
  61. org := ctx.Org.Organization
  62. if form.Email != "" {
  63. if err := user_service.ReplacePrimaryEmailAddress(ctx, org.AsUser(), form.Email); err != nil {
  64. ctx.Data["Err_Email"] = true
  65. ctx.RenderWithErr(ctx.Tr("form.email_invalid"), tplSettingsOptions, &form)
  66. return
  67. }
  68. }
  69. opts := &user_service.UpdateOptions{
  70. FullName: optional.Some(form.FullName),
  71. Description: optional.Some(form.Description),
  72. Website: optional.Some(form.Website),
  73. Location: optional.Some(form.Location),
  74. RepoAdminChangeTeamAccess: optional.Some(form.RepoAdminChangeTeamAccess),
  75. }
  76. if ctx.Doer.IsAdmin {
  77. opts.MaxRepoCreation = optional.Some(form.MaxRepoCreation)
  78. }
  79. if err := user_service.UpdateUser(ctx, org.AsUser(), opts); err != nil {
  80. ctx.ServerError("UpdateUser", err)
  81. return
  82. }
  83. log.Trace("Organization setting updated: %s", org.Name)
  84. ctx.Flash.Success(ctx.Tr("org.settings.update_setting_success"))
  85. ctx.Redirect(ctx.Org.OrgLink + "/settings")
  86. }
  87. // SettingsAvatar response for change avatar on settings page
  88. func SettingsAvatar(ctx *context.Context) {
  89. form := web.GetForm(ctx).(*forms.AvatarForm)
  90. form.Source = forms.AvatarLocal
  91. if err := user_setting.UpdateAvatarSetting(ctx, form, ctx.Org.Organization.AsUser()); err != nil {
  92. ctx.Flash.Error(err.Error())
  93. } else {
  94. ctx.Flash.Success(ctx.Tr("org.settings.update_avatar_success"))
  95. }
  96. ctx.Redirect(ctx.Org.OrgLink + "/settings")
  97. }
  98. // SettingsDeleteAvatar response for delete avatar on settings page
  99. func SettingsDeleteAvatar(ctx *context.Context) {
  100. if err := user_service.DeleteAvatar(ctx, ctx.Org.Organization.AsUser()); err != nil {
  101. ctx.Flash.Error(err.Error())
  102. }
  103. ctx.JSONRedirect(ctx.Org.OrgLink + "/settings")
  104. }
  105. // SettingsDeleteOrgPost response for deleting an organization
  106. func SettingsDeleteOrgPost(ctx *context.Context) {
  107. if ctx.Org.Organization.Name != ctx.FormString("org_name") {
  108. ctx.JSONError(ctx.Tr("form.enterred_invalid_org_name"))
  109. return
  110. }
  111. if err := org_service.DeleteOrganization(ctx, ctx.Org.Organization, false /* no purge */); err != nil {
  112. if repo_model.IsErrUserOwnRepos(err) {
  113. ctx.JSONError(ctx.Tr("form.org_still_own_repo"))
  114. } else if packages_model.IsErrUserOwnPackages(err) {
  115. ctx.JSONError(ctx.Tr("form.org_still_own_packages"))
  116. } else {
  117. log.Error("DeleteOrganization: %v", err)
  118. ctx.JSONError(util.Iif(ctx.Doer.IsAdmin, err.Error(), string(ctx.Tr("org.settings.delete_failed"))))
  119. }
  120. return
  121. }
  122. ctx.Flash.Success(ctx.Tr("org.settings.delete_successful", ctx.Org.Organization.Name))
  123. ctx.JSONRedirect(setting.AppSubURL + "/")
  124. }
  125. // Webhooks render webhook list page
  126. func Webhooks(ctx *context.Context) {
  127. ctx.Data["Title"] = ctx.Tr("org.settings")
  128. ctx.Data["PageIsOrgSettings"] = true
  129. ctx.Data["PageIsSettingsHooks"] = true
  130. ctx.Data["BaseLink"] = ctx.Org.OrgLink + "/settings/hooks"
  131. ctx.Data["BaseLinkNew"] = ctx.Org.OrgLink + "/settings/hooks"
  132. ctx.Data["Description"] = ctx.Tr("org.settings.hooks_desc")
  133. ws, err := db.Find[webhook.Webhook](ctx, webhook.ListWebhookOptions{OwnerID: ctx.Org.Organization.ID})
  134. if err != nil {
  135. ctx.ServerError("ListWebhooksByOpts", err)
  136. return
  137. }
  138. if _, err := shared_user.RenderUserOrgHeader(ctx); err != nil {
  139. ctx.ServerError("RenderUserOrgHeader", err)
  140. return
  141. }
  142. ctx.Data["Webhooks"] = ws
  143. ctx.HTML(http.StatusOK, tplSettingsHooks)
  144. }
  145. // DeleteWebhook response for delete webhook
  146. func DeleteWebhook(ctx *context.Context) {
  147. if err := webhook.DeleteWebhookByOwnerID(ctx, ctx.Org.Organization.ID, ctx.FormInt64("id")); err != nil {
  148. ctx.Flash.Error("DeleteWebhookByOwnerID: " + err.Error())
  149. } else {
  150. ctx.Flash.Success(ctx.Tr("repo.settings.webhook_deletion_success"))
  151. }
  152. ctx.JSONRedirect(ctx.Org.OrgLink + "/settings/hooks")
  153. }
  154. // Labels render organization labels page
  155. func Labels(ctx *context.Context) {
  156. ctx.Data["Title"] = ctx.Tr("repo.labels")
  157. ctx.Data["PageIsOrgSettings"] = true
  158. ctx.Data["PageIsOrgSettingsLabels"] = true
  159. ctx.Data["LabelTemplateFiles"] = repo_module.LabelTemplateFiles
  160. if _, err := shared_user.RenderUserOrgHeader(ctx); err != nil {
  161. ctx.ServerError("RenderUserOrgHeader", err)
  162. return
  163. }
  164. ctx.HTML(http.StatusOK, tplSettingsLabels)
  165. }
  166. // SettingsRenamePost response for renaming organization
  167. func SettingsRenamePost(ctx *context.Context) {
  168. form := web.GetForm(ctx).(*forms.RenameOrgForm)
  169. if ctx.HasError() {
  170. ctx.JSONError(ctx.GetErrMsg())
  171. return
  172. }
  173. oldOrgName, newOrgName := ctx.Org.Organization.Name, form.NewOrgName
  174. if form.OrgName != oldOrgName {
  175. ctx.JSONError(ctx.Tr("form.enterred_invalid_org_name"))
  176. return
  177. }
  178. if newOrgName == oldOrgName {
  179. ctx.JSONError(ctx.Tr("org.settings.rename_no_change"))
  180. return
  181. }
  182. if err := user_service.RenameUser(ctx, ctx.Org.Organization.AsUser(), newOrgName); err != nil {
  183. if user_model.IsErrUserAlreadyExist(err) {
  184. ctx.JSONError(ctx.Tr("org.form.name_been_taken", newOrgName))
  185. } else if db.IsErrNameReserved(err) {
  186. ctx.JSONError(ctx.Tr("org.form.name_reserved", newOrgName))
  187. } else if db.IsErrNamePatternNotAllowed(err) {
  188. ctx.JSONError(ctx.Tr("org.form.name_pattern_not_allowed", newOrgName))
  189. } else {
  190. log.Error("RenameOrganization: %v", err)
  191. ctx.JSONError(util.Iif(ctx.Doer.IsAdmin, err.Error(), string(ctx.Tr("org.settings.rename_failed"))))
  192. }
  193. return
  194. }
  195. ctx.Flash.Success(ctx.Tr("org.settings.rename_success", oldOrgName, newOrgName))
  196. ctx.JSONRedirect(setting.AppSubURL + "/org/" + url.PathEscape(newOrgName) + "/settings")
  197. }
  198. // SettingsChangeVisibilityPost response for change organization visibility
  199. func SettingsChangeVisibilityPost(ctx *context.Context) {
  200. visibility, ok := structs.VisibilityModes[ctx.FormString("visibility")]
  201. if !ok {
  202. ctx.Flash.Error(ctx.Tr("invalid_data", visibility))
  203. ctx.JSONRedirect(setting.AppSubURL + "/org/" + url.PathEscape(ctx.Org.Organization.Name) + "/settings")
  204. return
  205. }
  206. if ctx.Org.Organization.Visibility == visibility {
  207. ctx.Flash.Info(ctx.Tr("nothing_has_been_changed"))
  208. ctx.JSONRedirect(setting.AppSubURL + "/org/" + url.PathEscape(ctx.Org.Organization.Name) + "/settings")
  209. return
  210. }
  211. if err := org_service.ChangeOrganizationVisibility(ctx, ctx.Org.Organization, visibility); err != nil {
  212. log.Error("ChangeOrganizationVisibility: %v", err)
  213. ctx.JSONError(ctx.Tr("error.occurred"))
  214. return
  215. }
  216. ctx.Flash.Success(ctx.Tr("org.settings.change_visibility_success", ctx.Org.Organization.Name))
  217. ctx.JSONRedirect(setting.AppSubURL + "/org/" + url.PathEscape(ctx.Org.Organization.Name) + "/settings")
  218. }