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

teams.go 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  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. "net/url"
  9. "path"
  10. "strings"
  11. "code.gitea.io/gitea/models"
  12. unit_model "code.gitea.io/gitea/models/unit"
  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/modules/web"
  17. "code.gitea.io/gitea/routers/utils"
  18. "code.gitea.io/gitea/services/forms"
  19. )
  20. const (
  21. // tplTeams template path for teams list page
  22. tplTeams base.TplName = "org/team/teams"
  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. ctx.HTML(http.StatusOK, tplTeams)
  43. }
  44. // TeamsAction response for join, leave, remove, add operations to team
  45. func TeamsAction(ctx *context.Context) {
  46. uid := ctx.FormInt64("uid")
  47. if uid == 0 {
  48. ctx.Redirect(ctx.Org.OrgLink + "/teams")
  49. return
  50. }
  51. page := ctx.FormString("page")
  52. var err error
  53. switch ctx.Params(":action") {
  54. case "join":
  55. if !ctx.Org.IsOwner {
  56. ctx.Error(http.StatusNotFound)
  57. return
  58. }
  59. err = ctx.Org.Team.AddMember(ctx.User.ID)
  60. case "leave":
  61. err = ctx.Org.Team.RemoveMember(ctx.User.ID)
  62. if err != nil {
  63. if models.IsErrLastOrgOwner(err) {
  64. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  65. } else {
  66. log.Error("Action(%s): %v", ctx.Params(":action"), err)
  67. ctx.JSON(http.StatusOK, map[string]interface{}{
  68. "ok": false,
  69. "err": err.Error(),
  70. })
  71. return
  72. }
  73. }
  74. ctx.JSON(http.StatusOK,
  75. map[string]interface{}{
  76. "redirect": ctx.Org.OrgLink + "/teams/",
  77. })
  78. return
  79. case "remove":
  80. if !ctx.Org.IsOwner {
  81. ctx.Error(http.StatusNotFound)
  82. return
  83. }
  84. err = ctx.Org.Team.RemoveMember(uid)
  85. if err != nil {
  86. if models.IsErrLastOrgOwner(err) {
  87. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  88. } else {
  89. log.Error("Action(%s): %v", ctx.Params(":action"), err)
  90. ctx.JSON(http.StatusOK, map[string]interface{}{
  91. "ok": false,
  92. "err": err.Error(),
  93. })
  94. return
  95. }
  96. }
  97. ctx.JSON(http.StatusOK,
  98. map[string]interface{}{
  99. "redirect": ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName),
  100. })
  101. return
  102. case "add":
  103. if !ctx.Org.IsOwner {
  104. ctx.Error(http.StatusNotFound)
  105. return
  106. }
  107. uname := utils.RemoveUsernameParameterSuffix(strings.ToLower(ctx.FormString("uname")))
  108. var u *models.User
  109. u, err = models.GetUserByName(uname)
  110. if err != nil {
  111. if models.IsErrUserNotExist(err) {
  112. ctx.Flash.Error(ctx.Tr("form.user_not_exist"))
  113. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName))
  114. } else {
  115. ctx.ServerError(" GetUserByName", err)
  116. }
  117. return
  118. }
  119. if u.IsOrganization() {
  120. ctx.Flash.Error(ctx.Tr("form.cannot_add_org_to_team"))
  121. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName))
  122. return
  123. }
  124. if ctx.Org.Team.IsMember(u.ID) {
  125. ctx.Flash.Error(ctx.Tr("org.teams.add_duplicate_users"))
  126. } else {
  127. err = ctx.Org.Team.AddMember(u.ID)
  128. }
  129. page = "team"
  130. }
  131. if err != nil {
  132. if models.IsErrLastOrgOwner(err) {
  133. ctx.Flash.Error(ctx.Tr("form.last_org_owner"))
  134. } else {
  135. log.Error("Action(%s): %v", ctx.Params(":action"), err)
  136. ctx.JSON(http.StatusOK, map[string]interface{}{
  137. "ok": false,
  138. "err": err.Error(),
  139. })
  140. return
  141. }
  142. }
  143. switch page {
  144. case "team":
  145. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName))
  146. case "home":
  147. ctx.Redirect(ctx.Org.Organization.HomeLink())
  148. default:
  149. ctx.Redirect(ctx.Org.OrgLink + "/teams")
  150. }
  151. }
  152. // TeamsRepoAction operate team's repository
  153. func TeamsRepoAction(ctx *context.Context) {
  154. if !ctx.Org.IsOwner {
  155. ctx.Error(http.StatusNotFound)
  156. return
  157. }
  158. var err error
  159. action := ctx.Params(":action")
  160. switch action {
  161. case "add":
  162. repoName := path.Base(ctx.FormString("repo_name"))
  163. var repo *models.Repository
  164. repo, err = models.GetRepositoryByName(ctx.Org.Organization.ID, repoName)
  165. if err != nil {
  166. if models.IsErrRepoNotExist(err) {
  167. ctx.Flash.Error(ctx.Tr("org.teams.add_nonexistent_repo"))
  168. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName) + "/repositories")
  169. return
  170. }
  171. ctx.ServerError("GetRepositoryByName", err)
  172. return
  173. }
  174. err = ctx.Org.Team.AddRepository(repo)
  175. case "remove":
  176. err = ctx.Org.Team.RemoveRepository(ctx.FormInt64("repoid"))
  177. case "addall":
  178. err = ctx.Org.Team.AddAllRepositories()
  179. case "removeall":
  180. err = ctx.Org.Team.RemoveAllRepositories()
  181. }
  182. if err != nil {
  183. log.Error("Action(%s): '%s' %v", ctx.Params(":action"), ctx.Org.Team.Name, err)
  184. ctx.ServerError("TeamsRepoAction", err)
  185. return
  186. }
  187. if action == "addall" || action == "removeall" {
  188. ctx.JSON(http.StatusOK, map[string]interface{}{
  189. "redirect": ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName) + "/repositories",
  190. })
  191. return
  192. }
  193. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(ctx.Org.Team.LowerName) + "/repositories")
  194. }
  195. // NewTeam render create new team page
  196. func NewTeam(ctx *context.Context) {
  197. ctx.Data["Title"] = ctx.Org.Organization.FullName
  198. ctx.Data["PageIsOrgTeams"] = true
  199. ctx.Data["PageIsOrgTeamsNew"] = true
  200. ctx.Data["Team"] = &models.Team{}
  201. ctx.Data["Units"] = unit_model.Units
  202. ctx.HTML(http.StatusOK, tplTeamNew)
  203. }
  204. // NewTeamPost response for create new team
  205. func NewTeamPost(ctx *context.Context) {
  206. form := web.GetForm(ctx).(*forms.CreateTeamForm)
  207. ctx.Data["Title"] = ctx.Org.Organization.FullName
  208. ctx.Data["PageIsOrgTeams"] = true
  209. ctx.Data["PageIsOrgTeamsNew"] = true
  210. ctx.Data["Units"] = unit_model.Units
  211. var includesAllRepositories = form.RepoAccess == "all"
  212. t := &models.Team{
  213. OrgID: ctx.Org.Organization.ID,
  214. Name: form.TeamName,
  215. Description: form.Description,
  216. Authorize: models.ParseAccessMode(form.Permission),
  217. IncludesAllRepositories: includesAllRepositories,
  218. CanCreateOrgRepo: form.CanCreateOrgRepo,
  219. }
  220. if t.Authorize < models.AccessModeOwner {
  221. var units = make([]*models.TeamUnit, 0, len(form.Units))
  222. for _, tp := range form.Units {
  223. units = append(units, &models.TeamUnit{
  224. OrgID: ctx.Org.Organization.ID,
  225. Type: tp,
  226. })
  227. }
  228. t.Units = units
  229. }
  230. ctx.Data["Team"] = t
  231. if ctx.HasError() {
  232. ctx.HTML(http.StatusOK, tplTeamNew)
  233. return
  234. }
  235. if t.Authorize < models.AccessModeAdmin && len(form.Units) == 0 {
  236. ctx.RenderWithErr(ctx.Tr("form.team_no_units_error"), tplTeamNew, &form)
  237. return
  238. }
  239. if err := models.NewTeam(t); err != nil {
  240. ctx.Data["Err_TeamName"] = true
  241. switch {
  242. case models.IsErrTeamAlreadyExist(err):
  243. ctx.RenderWithErr(ctx.Tr("form.team_name_been_taken"), tplTeamNew, &form)
  244. default:
  245. ctx.ServerError("NewTeam", err)
  246. }
  247. return
  248. }
  249. log.Trace("Team created: %s/%s", ctx.Org.Organization.Name, t.Name)
  250. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(t.LowerName))
  251. }
  252. // TeamMembers render team members page
  253. func TeamMembers(ctx *context.Context) {
  254. ctx.Data["Title"] = ctx.Org.Team.Name
  255. ctx.Data["PageIsOrgTeams"] = true
  256. ctx.Data["PageIsOrgTeamMembers"] = true
  257. if err := ctx.Org.Team.GetMembers(&models.SearchMembersOptions{}); err != nil {
  258. ctx.ServerError("GetMembers", err)
  259. return
  260. }
  261. ctx.HTML(http.StatusOK, tplTeamMembers)
  262. }
  263. // TeamRepositories show the repositories of team
  264. func TeamRepositories(ctx *context.Context) {
  265. ctx.Data["Title"] = ctx.Org.Team.Name
  266. ctx.Data["PageIsOrgTeams"] = true
  267. ctx.Data["PageIsOrgTeamRepos"] = true
  268. if err := ctx.Org.Team.GetRepositories(&models.SearchTeamOptions{}); err != nil {
  269. ctx.ServerError("GetRepositories", err)
  270. return
  271. }
  272. ctx.HTML(http.StatusOK, tplTeamRepositories)
  273. }
  274. // EditTeam render team edit page
  275. func EditTeam(ctx *context.Context) {
  276. ctx.Data["Title"] = ctx.Org.Organization.FullName
  277. ctx.Data["PageIsOrgTeams"] = true
  278. ctx.Data["team_name"] = ctx.Org.Team.Name
  279. ctx.Data["desc"] = ctx.Org.Team.Description
  280. ctx.Data["Units"] = unit_model.Units
  281. ctx.HTML(http.StatusOK, tplTeamNew)
  282. }
  283. // EditTeamPost response for modify team information
  284. func EditTeamPost(ctx *context.Context) {
  285. form := web.GetForm(ctx).(*forms.CreateTeamForm)
  286. t := ctx.Org.Team
  287. ctx.Data["Title"] = ctx.Org.Organization.FullName
  288. ctx.Data["PageIsOrgTeams"] = true
  289. ctx.Data["Team"] = t
  290. ctx.Data["Units"] = unit_model.Units
  291. isAuthChanged := false
  292. isIncludeAllChanged := false
  293. var includesAllRepositories = form.RepoAccess == "all"
  294. if !t.IsOwnerTeam() {
  295. // Validate permission level.
  296. auth := models.ParseAccessMode(form.Permission)
  297. t.Name = form.TeamName
  298. if t.Authorize != auth {
  299. isAuthChanged = true
  300. t.Authorize = auth
  301. }
  302. if t.IncludesAllRepositories != includesAllRepositories {
  303. isIncludeAllChanged = true
  304. t.IncludesAllRepositories = includesAllRepositories
  305. }
  306. }
  307. t.Description = form.Description
  308. if t.Authorize < models.AccessModeOwner {
  309. var units = make([]models.TeamUnit, 0, len(form.Units))
  310. for _, tp := range form.Units {
  311. units = append(units, models.TeamUnit{
  312. OrgID: t.OrgID,
  313. TeamID: t.ID,
  314. Type: tp,
  315. })
  316. }
  317. err := models.UpdateTeamUnits(t, units)
  318. if err != nil {
  319. ctx.Error(http.StatusInternalServerError, "LoadIssue", err.Error())
  320. return
  321. }
  322. }
  323. t.CanCreateOrgRepo = form.CanCreateOrgRepo
  324. if ctx.HasError() {
  325. ctx.HTML(http.StatusOK, tplTeamNew)
  326. return
  327. }
  328. if t.Authorize < models.AccessModeAdmin && len(form.Units) == 0 {
  329. ctx.RenderWithErr(ctx.Tr("form.team_no_units_error"), tplTeamNew, &form)
  330. return
  331. }
  332. if err := models.UpdateTeam(t, isAuthChanged, isIncludeAllChanged); err != nil {
  333. ctx.Data["Err_TeamName"] = true
  334. switch {
  335. case models.IsErrTeamAlreadyExist(err):
  336. ctx.RenderWithErr(ctx.Tr("form.team_name_been_taken"), tplTeamNew, &form)
  337. default:
  338. ctx.ServerError("UpdateTeam", err)
  339. }
  340. return
  341. }
  342. ctx.Redirect(ctx.Org.OrgLink + "/teams/" + url.PathEscape(t.LowerName))
  343. }
  344. // DeleteTeam response for the delete team request
  345. func DeleteTeam(ctx *context.Context) {
  346. if err := models.DeleteTeam(ctx.Org.Team); err != nil {
  347. ctx.Flash.Error("DeleteTeam: " + err.Error())
  348. } else {
  349. ctx.Flash.Success(ctx.Tr("org.teams.delete_team_success"))
  350. }
  351. ctx.JSON(http.StatusOK, map[string]interface{}{
  352. "redirect": ctx.Org.OrgLink + "/teams",
  353. })
  354. }