You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

teams.go 9.9 kB

11 years ago
11 years ago
11 years ago
3 years ago
3 years ago
3 years ago
11 years ago
11 years ago
API add/generalize pagination (#9452) * paginate results * fixed deadlock * prevented breaking change * updated swagger * go fmt * fixed find topic * go mod tidy * go mod vendor with go1.13.5 * fixed repo find topics * fixed unit test * added Limit method to Engine struct; use engine variable when provided; fixed gitignore * use ItemsPerPage for default pagesize; fix GetWatchers, getOrgUsersByOrgID and GetStargazers; fix GetAllCommits headers; reverted some changed behaviors * set Page value on Home route * improved memory allocations * fixed response headers * removed logfiles * fixed import order * import order * improved swagger * added function to get models.ListOptions from context * removed pagesize diff on unit test * fixed imports * removed unnecessary struct field * fixed go fmt * scoped PR * code improvements * code improvements * go mod tidy * fixed import order * fixed commit statuses session * fixed files headers * fixed headers; added pagination for notifications * go mod tidy * go fmt * removed Private from user search options; added setting.UI.IssuePagingNum as default valeu on repo's issues list * Apply suggestions from code review Co-Authored-By: 6543 <6543@obermui.de> Co-Authored-By: zeripath <art27@cantab.net> * fixed build error * CI.restart() * fixed merge conflicts resolve * fixed conflicts resolve * improved FindTrackedTimesOptions.ToOptions() method * added backwards compatibility on ListReleases request; fixed issue tracked time ToSession * fixed build error; fixed swagger template * fixed swagger template * fixed ListReleases backwards compatibility * added page to user search route Co-authored-by: techknowlogick <matti@mdranta.net> Co-authored-by: 6543 <6543@obermui.de> Co-authored-by: zeripath <art27@cantab.net>
5 years ago
11 years ago
3 years ago
3 years ago
3 years ago
Better logging (#6038) (#6095) * Panic don't fatal on create new logger Fixes #5854 Signed-off-by: Andrew Thornton <art27@cantab.net> * partial broken * Update the logging infrastrcture Signed-off-by: Andrew Thornton <art27@cantab.net> * Reset the skip levels for Fatal and Error Signed-off-by: Andrew Thornton <art27@cantab.net> * broken ncsa * More log.Error fixes Signed-off-by: Andrew Thornton <art27@cantab.net> * Remove nal * set log-levels to lowercase * Make console_test test all levels * switch to lowercased levels * OK now working * Fix vetting issues * Fix lint * Fix tests * change default logging to match current gitea * Improve log testing Signed-off-by: Andrew Thornton <art27@cantab.net> * reset error skip levels to 0 * Update documentation and access logger configuration * Redirect the router log back to gitea if redirect macaron log but also allow setting the log level - i.e. TRACE * Fix broken level caching * Refactor the router log * Add Router logger * Add colorizing options * Adjust router colors * Only create logger if they will be used * update app.ini.sample * rename Attribute ColorAttribute * Change from white to green for function * Set fatal/error levels * Restore initial trace logger * Fix Trace arguments in modules/auth/auth.go * Properly handle XORMLogger * Improve admin/config page * fix fmt * Add auto-compression of old logs * Update error log levels * Remove the unnecessary skip argument from Error, Fatal and Critical * Add stacktrace support * Fix tests * Remove x/sync from vendors? * Add stderr option to console logger * Use filepath.ToSlash to protect against Windows in tests * Remove prefixed underscores from names in colors.go * Remove not implemented database logger This was removed from Gogs on 4 Mar 2016 but left in the configuration since then. * Ensure that log paths are relative to ROOT_PATH * use path.Join * rename jsonConfig to logConfig * Rename "config" to "jsonConfig" to make it clearer * Requested changes * Requested changes: XormLogger * Try to color the windows terminal If successful default to colorizing the console logs * fixup * Colorize initially too * update vendor * Colorize logs on default and remove if this is not a colorizing logger * Fix documentation * fix test * Use go-isatty to detect if on windows we are on msys or cygwin * Fix spelling mistake * Add missing vendors * More changes * Rationalise the ANSI writer protection * Adjust colors on advice from @0x5c * Make Flags a comma separated list * Move to use the windows constant for ENABLE_VIRTUAL_TERMINAL_PROCESSING * Ensure matching is done on the non-colored message - to simpify EXPRESSION
6 years ago
9 years ago
3 years ago
Better logging (#6038) (#6095) * Panic don't fatal on create new logger Fixes #5854 Signed-off-by: Andrew Thornton <art27@cantab.net> * partial broken * Update the logging infrastrcture Signed-off-by: Andrew Thornton <art27@cantab.net> * Reset the skip levels for Fatal and Error Signed-off-by: Andrew Thornton <art27@cantab.net> * broken ncsa * More log.Error fixes Signed-off-by: Andrew Thornton <art27@cantab.net> * Remove nal * set log-levels to lowercase * Make console_test test all levels * switch to lowercased levels * OK now working * Fix vetting issues * Fix lint * Fix tests * change default logging to match current gitea * Improve log testing Signed-off-by: Andrew Thornton <art27@cantab.net> * reset error skip levels to 0 * Update documentation and access logger configuration * Redirect the router log back to gitea if redirect macaron log but also allow setting the log level - i.e. TRACE * Fix broken level caching * Refactor the router log * Add Router logger * Add colorizing options * Adjust router colors * Only create logger if they will be used * update app.ini.sample * rename Attribute ColorAttribute * Change from white to green for function * Set fatal/error levels * Restore initial trace logger * Fix Trace arguments in modules/auth/auth.go * Properly handle XORMLogger * Improve admin/config page * fix fmt * Add auto-compression of old logs * Update error log levels * Remove the unnecessary skip argument from Error, Fatal and Critical * Add stacktrace support * Fix tests * Remove x/sync from vendors? * Add stderr option to console logger * Use filepath.ToSlash to protect against Windows in tests * Remove prefixed underscores from names in colors.go * Remove not implemented database logger This was removed from Gogs on 4 Mar 2016 but left in the configuration since then. * Ensure that log paths are relative to ROOT_PATH * use path.Join * rename jsonConfig to logConfig * Rename "config" to "jsonConfig" to make it clearer * Requested changes * Requested changes: XormLogger * Try to color the windows terminal If successful default to colorizing the console logs * fixup * Colorize initially too * update vendor * Colorize logs on default and remove if this is not a colorizing logger * Fix documentation * fix test * Use go-isatty to detect if on windows we are on msys or cygwin * Fix spelling mistake * Add missing vendors * More changes * Rationalise the ANSI writer protection * Adjust colors on advice from @0x5c * Make Flags a comma separated list * Move to use the windows constant for ENABLE_VIRTUAL_TERMINAL_PROCESSING * Ensure matching is done on the non-colored message - to simpify EXPRESSION
6 years ago
3 years ago
3 years ago
3 years ago
3 years ago
API add/generalize pagination (#9452) * paginate results * fixed deadlock * prevented breaking change * updated swagger * go fmt * fixed find topic * go mod tidy * go mod vendor with go1.13.5 * fixed repo find topics * fixed unit test * added Limit method to Engine struct; use engine variable when provided; fixed gitignore * use ItemsPerPage for default pagesize; fix GetWatchers, getOrgUsersByOrgID and GetStargazers; fix GetAllCommits headers; reverted some changed behaviors * set Page value on Home route * improved memory allocations * fixed response headers * removed logfiles * fixed import order * import order * improved swagger * added function to get models.ListOptions from context * removed pagesize diff on unit test * fixed imports * removed unnecessary struct field * fixed go fmt * scoped PR * code improvements * code improvements * go mod tidy * fixed import order * fixed commit statuses session * fixed files headers * fixed headers; added pagination for notifications * go mod tidy * go fmt * removed Private from user search options; added setting.UI.IssuePagingNum as default valeu on repo's issues list * Apply suggestions from code review Co-Authored-By: 6543 <6543@obermui.de> Co-Authored-By: zeripath <art27@cantab.net> * fixed build error * CI.restart() * fixed merge conflicts resolve * fixed conflicts resolve * improved FindTrackedTimesOptions.ToOptions() method * added backwards compatibility on ListReleases request; fixed issue tracked time ToSession * fixed build error; fixed swagger template * fixed swagger template * fixed ListReleases backwards compatibility * added page to user search route Co-authored-by: techknowlogick <matti@mdranta.net> Co-authored-by: 6543 <6543@obermui.de> Co-authored-by: zeripath <art27@cantab.net>
5 years ago
3 years ago
API add/generalize pagination (#9452) * paginate results * fixed deadlock * prevented breaking change * updated swagger * go fmt * fixed find topic * go mod tidy * go mod vendor with go1.13.5 * fixed repo find topics * fixed unit test * added Limit method to Engine struct; use engine variable when provided; fixed gitignore * use ItemsPerPage for default pagesize; fix GetWatchers, getOrgUsersByOrgID and GetStargazers; fix GetAllCommits headers; reverted some changed behaviors * set Page value on Home route * improved memory allocations * fixed response headers * removed logfiles * fixed import order * import order * improved swagger * added function to get models.ListOptions from context * removed pagesize diff on unit test * fixed imports * removed unnecessary struct field * fixed go fmt * scoped PR * code improvements * code improvements * go mod tidy * fixed import order * fixed commit statuses session * fixed files headers * fixed headers; added pagination for notifications * go mod tidy * go fmt * removed Private from user search options; added setting.UI.IssuePagingNum as default valeu on repo's issues list * Apply suggestions from code review Co-Authored-By: 6543 <6543@obermui.de> Co-Authored-By: zeripath <art27@cantab.net> * fixed build error * CI.restart() * fixed merge conflicts resolve * fixed conflicts resolve * improved FindTrackedTimesOptions.ToOptions() method * added backwards compatibility on ListReleases request; fixed issue tracked time ToSession * fixed build error; fixed swagger template * fixed swagger template * fixed ListReleases backwards compatibility * added page to user search route Co-authored-by: techknowlogick <matti@mdranta.net> Co-authored-by: 6543 <6543@obermui.de> Co-authored-by: zeripath <art27@cantab.net>
5 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
9 years ago
9 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Copyright 2019 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package org
  6. import (
  7. "net/http"
  8. "path"
  9. "strings"
  10. "code.gitea.io/gitea/modules/setting"
  11. "code.gitea.io/gitea/models"
  12. "code.gitea.io/gitea/modules/auth"
  13. "code.gitea.io/gitea/modules/base"
  14. "code.gitea.io/gitea/modules/context"
  15. "code.gitea.io/gitea/modules/log"
  16. "code.gitea.io/gitea/routers/utils"
  17. "github.com/unknwon/com"
  18. )
  19. const (
  20. // tplTeams template path for teams list page
  21. tplTeams base.TplName = "org/team/teams"
  22. tplCourseTeams base.TplName = "org/team/courseTeams"
  23. // tplTeamNew template path for create new team page
  24. tplTeamNew base.TplName = "org/team/new"
  25. // tplTeamMembers template path for showing team members page
  26. tplTeamMembers base.TplName = "org/team/members"
  27. // tplTeamRepositories template path for showing team repositories page
  28. tplTeamRepositories base.TplName = "org/team/repositories"
  29. )
  30. // Teams render teams list page
  31. func Teams(ctx *context.Context) {
  32. org := ctx.Org.Organization
  33. ctx.Data["Title"] = org.FullName
  34. ctx.Data["PageIsOrgTeams"] = true
  35. for _, t := range org.Teams {
  36. if err := t.GetMembers(&models.SearchMembersOptions{}); err != nil {
  37. ctx.ServerError("GetMembers", err)
  38. return
  39. }
  40. }
  41. ctx.Data["Teams"] = org.Teams
  42. if setting.Course.OrgName == org.Name {
  43. ctx.HTML(200, tplCourseTeams)
  44. } else {
  45. ctx.HTML(200, tplTeams)
  46. }
  47. }
  48. // TeamsAction response for join, leave, remove, add operations to team
  49. func TeamsAction(ctx *context.Context) {
  50. uid := com.StrTo(ctx.Query("uid")).MustInt64()
  51. if uid == 0 {
  52. ctx.Redirect(ctx.Org.OrgLink + "/teams")
  53. return
  54. }
  55. page := ctx.Query("page")
  56. var err error
  57. switch ctx.Params(":action") {
  58. case "join":
  59. if !ctx.Org.IsOwner {
  60. ctx.Error(404)
  61. return
  62. }
  63. err = ctx.Org.Team.AddMember(ctx.User.ID)
  64. case "leave":
  65. err = ctx.Org.Team.RemoveMember(ctx.User.ID)
  66. case "remove":
  67. if !ctx.Org.IsOwner {
  68. ctx.Error(404)
  69. return
  70. }
  71. err = ctx.Org.Team.RemoveMember(uid)
  72. page = "team"
  73. case "add":
  74. if !ctx.Org.IsOwner {
  75. ctx.Error(404)
  76. return
  77. }
  78. uname := utils.RemoveUsernameParameterSuffix(strings.ToLower(ctx.Query("uname")))
  79. var u *models.User
  80. u, err = models.GetUserByName(uname)
  81. if err != nil {
  82. if models.IsErrUserNotExist(err) {
  83. ctx.Flash.Error(ctx.Tr("form.user_not_exist"))
  84. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName)
  85. } else {
  86. ctx.ServerError(" GetUserByName", err)
  87. }
  88. return
  89. }
  90. if u.IsOrganization() {
  91. ctx.Flash.Error(ctx.Tr("form.cannot_add_org_to_team"))
  92. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName)
  93. return
  94. }
  95. if ctx.Org.Team.IsMember(u.ID) {
  96. ctx.Flash.Error(ctx.Tr("org.teams.add_duplicate_users"))
  97. } else {
  98. err = ctx.Org.Team.AddMember(u.ID)
  99. }
  100. page = "team"
  101. }
  102. if err != nil {
  103. if models.IsErrLastOrgOwner(err) {
  104. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  105. } else {
  106. log.Error("Action(%s): %v", ctx.Params(":action"), err)
  107. ctx.JSON(200, map[string]interface{}{
  108. "ok": false,
  109. "err": err.Error(),
  110. })
  111. return
  112. }
  113. }
  114. switch page {
  115. case "team":
  116. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName)
  117. case "home":
  118. ctx.Redirect(ctx.Org.Organization.HomeLink())
  119. default:
  120. ctx.Redirect(ctx.Org.OrgLink + "/teams")
  121. }
  122. }
  123. // TeamsRepoAction operate team's repository
  124. func TeamsRepoAction(ctx *context.Context) {
  125. if !ctx.Org.IsOwner {
  126. ctx.Error(404)
  127. return
  128. }
  129. var err error
  130. action := ctx.Params(":action")
  131. switch action {
  132. case "add":
  133. repoName := path.Base(ctx.Query("repo_name"))
  134. var repo *models.Repository
  135. repo, err = models.GetRepositoryByOwnerAndAlias(ctx.Org.Organization.Name, repoName)
  136. if err != nil {
  137. if models.IsErrRepoNotExist(err) {
  138. ctx.Flash.Error(ctx.Tr("org.teams.add_nonexistent_repo"))
  139. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName + "/repositories")
  140. return
  141. }
  142. ctx.ServerError("GetRepositoryByName", err)
  143. return
  144. }
  145. err = ctx.Org.Team.AddRepository(repo)
  146. case "remove":
  147. err = ctx.Org.Team.RemoveRepository(com.StrTo(ctx.Query("repoid")).MustInt64())
  148. case "addall":
  149. err = ctx.Org.Team.AddAllRepositories()
  150. case "removeall":
  151. err = ctx.Org.Team.RemoveAllRepositories()
  152. }
  153. if err != nil {
  154. log.Error("Action(%s): '%s' %v", ctx.Params(":action"), ctx.Org.Team.Name, err)
  155. ctx.ServerError("TeamsRepoAction", err)
  156. return
  157. }
  158. if action == "addall" || action == "removeall" {
  159. ctx.JSON(200, map[string]interface{}{
  160. "redirect": ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName + "/repositories",
  161. })
  162. return
  163. }
  164. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + ctx.Org.Team.LowerName + "/repositories")
  165. }
  166. // NewTeam render create new team page
  167. func NewTeam(ctx *context.Context) {
  168. ctx.Data["Title"] = ctx.Org.Organization.FullName
  169. ctx.Data["PageIsOrgTeams"] = true
  170. ctx.Data["PageIsOrgTeamsNew"] = true
  171. ctx.Data["Team"] = &models.Team{}
  172. ctx.Data["Units"] = models.Units
  173. ctx.HTML(200, tplTeamNew)
  174. }
  175. // NewTeamPost response for create new team
  176. func NewTeamPost(ctx *context.Context, form auth.CreateTeamForm) {
  177. ctx.Data["Title"] = ctx.Org.Organization.FullName
  178. ctx.Data["PageIsOrgTeams"] = true
  179. ctx.Data["PageIsOrgTeamsNew"] = true
  180. ctx.Data["Units"] = models.Units
  181. var includesAllRepositories = (form.RepoAccess == "all")
  182. t := &models.Team{
  183. OrgID: ctx.Org.Organization.ID,
  184. Name: form.TeamName,
  185. Description: form.Description,
  186. Authorize: models.ParseAccessMode(form.Permission),
  187. IncludesAllRepositories: includesAllRepositories,
  188. CanCreateOrgRepo: form.CanCreateOrgRepo,
  189. }
  190. if t.Authorize < models.AccessModeOwner {
  191. var units = make([]*models.TeamUnit, 0, len(form.Units))
  192. for _, tp := range form.Units {
  193. units = append(units, &models.TeamUnit{
  194. OrgID: ctx.Org.Organization.ID,
  195. Type: tp,
  196. })
  197. }
  198. t.Units = units
  199. }
  200. ctx.Data["Team"] = t
  201. if ctx.HasError() {
  202. ctx.HTML(200, tplTeamNew)
  203. return
  204. }
  205. if t.Authorize < models.AccessModeAdmin && len(form.Units) == 0 {
  206. ctx.RenderWithErr(ctx.Tr("form.team_no_units_error"), tplTeamNew, &form)
  207. return
  208. }
  209. if err := models.NewTeam(t); err != nil {
  210. ctx.Data["Err_TeamName"] = true
  211. switch {
  212. case models.IsErrTeamAlreadyExist(err):
  213. ctx.RenderWithErr(ctx.Tr("form.team_name_been_taken"), tplTeamNew, &form)
  214. default:
  215. ctx.ServerError("NewTeam", err)
  216. }
  217. return
  218. }
  219. log.Trace("Team created: %s/%s", ctx.Org.Organization.Name, t.Name)
  220. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + t.LowerName)
  221. }
  222. // TeamMembers render team members page
  223. func TeamMembers(ctx *context.Context) {
  224. ctx.Data["Title"] = ctx.Org.Team.Name
  225. ctx.Data["PageIsOrgTeams"] = true
  226. ctx.Data["PageIsOrgTeamMembers"] = true
  227. if err := ctx.Org.Team.GetMembers(&models.SearchMembersOptions{}); err != nil {
  228. ctx.ServerError("GetMembers", err)
  229. return
  230. }
  231. ctx.HTML(200, tplTeamMembers)
  232. }
  233. // TeamRepositories show the repositories of team
  234. func TeamRepositories(ctx *context.Context) {
  235. ctx.Data["Title"] = ctx.Org.Team.Name
  236. ctx.Data["PageIsOrgTeams"] = true
  237. ctx.Data["PageIsOrgTeamRepos"] = true
  238. if err := ctx.Org.Team.GetRepositories(&models.SearchTeamOptions{}); err != nil {
  239. ctx.ServerError("GetRepositories", err)
  240. return
  241. }
  242. ctx.HTML(200, tplTeamRepositories)
  243. }
  244. // EditTeam render team edit page
  245. func EditTeam(ctx *context.Context) {
  246. ctx.Data["Title"] = ctx.Org.Organization.FullName
  247. ctx.Data["PageIsOrgTeams"] = true
  248. ctx.Data["team_name"] = ctx.Org.Team.Name
  249. ctx.Data["desc"] = ctx.Org.Team.Description
  250. ctx.Data["Units"] = models.Units
  251. ctx.HTML(200, tplTeamNew)
  252. }
  253. // EditTeamPost response for modify team information
  254. func EditTeamPost(ctx *context.Context, form auth.CreateTeamForm) {
  255. t := ctx.Org.Team
  256. ctx.Data["Title"] = ctx.Org.Organization.FullName
  257. ctx.Data["PageIsOrgTeams"] = true
  258. ctx.Data["Team"] = t
  259. ctx.Data["Units"] = models.Units
  260. isAuthChanged := false
  261. isIncludeAllChanged := false
  262. var includesAllRepositories = (form.RepoAccess == "all")
  263. if !t.IsOwnerTeam() {
  264. // Validate permission level.
  265. auth := models.ParseAccessMode(form.Permission)
  266. t.Name = form.TeamName
  267. if t.Authorize != auth {
  268. isAuthChanged = true
  269. t.Authorize = auth
  270. }
  271. if t.IncludesAllRepositories != includesAllRepositories {
  272. isIncludeAllChanged = true
  273. t.IncludesAllRepositories = includesAllRepositories
  274. }
  275. }
  276. t.Description = form.Description
  277. if t.Authorize < models.AccessModeOwner {
  278. var units = make([]models.TeamUnit, 0, len(form.Units))
  279. for _, tp := range form.Units {
  280. units = append(units, models.TeamUnit{
  281. OrgID: t.OrgID,
  282. TeamID: t.ID,
  283. Type: tp,
  284. })
  285. }
  286. err := models.UpdateTeamUnits(t, units)
  287. if err != nil {
  288. ctx.Error(http.StatusInternalServerError, "LoadIssue", err.Error())
  289. return
  290. }
  291. }
  292. t.CanCreateOrgRepo = form.CanCreateOrgRepo
  293. if ctx.HasError() {
  294. ctx.HTML(200, tplTeamNew)
  295. return
  296. }
  297. if t.Authorize < models.AccessModeAdmin && len(form.Units) == 0 {
  298. ctx.RenderWithErr(ctx.Tr("form.team_no_units_error"), tplTeamNew, &form)
  299. return
  300. }
  301. if err := models.UpdateTeam(t, isAuthChanged, isIncludeAllChanged); err != nil {
  302. ctx.Data["Err_TeamName"] = true
  303. switch {
  304. case models.IsErrTeamAlreadyExist(err):
  305. ctx.RenderWithErr(ctx.Tr("form.team_name_been_taken"), tplTeamNew, &form)
  306. default:
  307. ctx.ServerError("UpdateTeam", err)
  308. }
  309. return
  310. }
  311. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + t.LowerName)
  312. }
  313. // DeleteTeam response for the delete team request
  314. func DeleteTeam(ctx *context.Context) {
  315. if err := models.DeleteTeam(ctx.Org.Team); err != nil {
  316. ctx.Flash.Error("DeleteTeam: " + err.Error())
  317. } else {
  318. ctx.Flash.Success(ctx.Tr("org.teams.delete_team_success"))
  319. }
  320. ctx.JSON(200, map[string]interface{}{
  321. "redirect": ctx.Org.OrgLink + "/teams",
  322. })
  323. }