|
|
|
|
@ -7,6 +7,7 @@ import ( |
|
|
|
|
|
|
|
|
|
"github.com/grafana/grafana/pkg/api/dtos" |
|
|
|
|
"github.com/grafana/grafana/pkg/api/response" |
|
|
|
|
"github.com/grafana/grafana/pkg/services/auth/identity" |
|
|
|
|
contextmodel "github.com/grafana/grafana/pkg/services/contexthandler/model" |
|
|
|
|
"github.com/grafana/grafana/pkg/services/dashboards" |
|
|
|
|
"github.com/grafana/grafana/pkg/services/team" |
|
|
|
|
@ -30,12 +31,12 @@ func (hs *HTTPServer) CreateTeam(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
return response.Error(http.StatusBadRequest, "bad request data", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
t, err := hs.teamService.CreateTeam(cmd.Name, cmd.Email, c.OrgID) |
|
|
|
|
t, err := hs.teamService.CreateTeam(cmd.Name, cmd.Email, c.SignedInUser.GetOrgID()) |
|
|
|
|
if err != nil { |
|
|
|
|
if errors.Is(err, team.ErrTeamNameTaken) { |
|
|
|
|
return response.Error(409, "Team name taken", err) |
|
|
|
|
return response.Error(http.StatusConflict, "Team name taken", err) |
|
|
|
|
} |
|
|
|
|
return response.Error(500, "Failed to create Team", err) |
|
|
|
|
return response.Error(http.StatusInternalServerError, "Failed to create Team", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Clear permission cache for the user who's created the team, so that new permissions are fetched for their next call
|
|
|
|
|
@ -45,13 +46,22 @@ func (hs *HTTPServer) CreateTeam(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
// if the request is authenticated using API tokens
|
|
|
|
|
// the SignedInUser is an empty struct therefore
|
|
|
|
|
// an additional check whether it is an actual user is required
|
|
|
|
|
if c.SignedInUser.IsRealUser() { |
|
|
|
|
if err := addOrUpdateTeamMember(c.Req.Context(), hs.teamPermissionsService, c.SignedInUser.UserID, c.OrgID, t.ID, dashboards.PERMISSION_ADMIN.String()); err != nil { |
|
|
|
|
namespace, identifier := c.SignedInUser.GetNamespacedID() |
|
|
|
|
switch namespace { |
|
|
|
|
case identity.NamespaceUser: |
|
|
|
|
userID, err := strconv.ParseInt(identifier, 10, 64) |
|
|
|
|
if err != nil { |
|
|
|
|
c.Logger.Error("Could not add creator to team because user id is not a number", "error", err) |
|
|
|
|
break |
|
|
|
|
} |
|
|
|
|
if err := addOrUpdateTeamMember(c.Req.Context(), hs.teamPermissionsService, userID, c.SignedInUser.GetOrgID(), |
|
|
|
|
t.ID, dashboards.PERMISSION_ADMIN.String()); err != nil { |
|
|
|
|
c.Logger.Error("Could not add creator to team", "error", err) |
|
|
|
|
} |
|
|
|
|
} else { |
|
|
|
|
default: |
|
|
|
|
c.Logger.Warn("Could not add creator to team because is not a real user") |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return response.JSON(http.StatusOK, &util.DynMap{ |
|
|
|
|
"teamId": t.ID, |
|
|
|
|
"message": "Team created", |
|
|
|
|
@ -75,7 +85,7 @@ func (hs *HTTPServer) UpdateTeam(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
if err := web.Bind(c.Req, &cmd); err != nil { |
|
|
|
|
return response.Error(http.StatusBadRequest, "bad request data", err) |
|
|
|
|
} |
|
|
|
|
cmd.OrgID = c.OrgID |
|
|
|
|
cmd.OrgID = c.SignedInUser.GetOrgID() |
|
|
|
|
cmd.ID, err = strconv.ParseInt(web.Params(c.Req)[":teamId"], 10, 64) |
|
|
|
|
if err != nil { |
|
|
|
|
return response.Error(http.StatusBadRequest, "teamId is invalid", err) |
|
|
|
|
@ -83,9 +93,9 @@ func (hs *HTTPServer) UpdateTeam(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
|
|
|
|
|
if err := hs.teamService.UpdateTeam(c.Req.Context(), &cmd); err != nil { |
|
|
|
|
if errors.Is(err, team.ErrTeamNameTaken) { |
|
|
|
|
return response.Error(400, "Team name taken", err) |
|
|
|
|
return response.Error(http.StatusBadRequest, "Team name taken", err) |
|
|
|
|
} |
|
|
|
|
return response.Error(500, "Failed to update Team", err) |
|
|
|
|
return response.Error(http.StatusInternalServerError, "Failed to update Team", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return response.Success("Team updated") |
|
|
|
|
@ -102,7 +112,7 @@ func (hs *HTTPServer) UpdateTeam(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
// 404: notFoundError
|
|
|
|
|
// 500: internalServerError
|
|
|
|
|
func (hs *HTTPServer) DeleteTeamByID(c *contextmodel.ReqContext) response.Response { |
|
|
|
|
orgID := c.OrgID |
|
|
|
|
orgID := c.SignedInUser.GetOrgID() |
|
|
|
|
teamID, err := strconv.ParseInt(web.Params(c.Req)[":teamId"], 10, 64) |
|
|
|
|
if err != nil { |
|
|
|
|
return response.Error(http.StatusBadRequest, "teamId is invalid", err) |
|
|
|
|
@ -110,9 +120,9 @@ func (hs *HTTPServer) DeleteTeamByID(c *contextmodel.ReqContext) response.Respon |
|
|
|
|
|
|
|
|
|
if err := hs.teamService.DeleteTeam(c.Req.Context(), &team.DeleteTeamCommand{OrgID: orgID, ID: teamID}); err != nil { |
|
|
|
|
if errors.Is(err, team.ErrTeamNotFound) { |
|
|
|
|
return response.Error(404, "Failed to delete Team. ID not found", nil) |
|
|
|
|
return response.Error(http.StatusNotFound, "Failed to delete Team. ID not found", nil) |
|
|
|
|
} |
|
|
|
|
return response.Error(500, "Failed to delete Team", err) |
|
|
|
|
return response.Error(http.StatusInternalServerError, "Failed to delete Team", err) |
|
|
|
|
} |
|
|
|
|
return response.Success("Team deleted") |
|
|
|
|
} |
|
|
|
|
@ -137,7 +147,7 @@ func (hs *HTTPServer) SearchTeams(c *contextmodel.ReqContext) response.Response |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
query := team.SearchTeamsQuery{ |
|
|
|
|
OrgID: c.OrgID, |
|
|
|
|
OrgID: c.SignedInUser.GetOrgID(), |
|
|
|
|
Query: c.Query("query"), |
|
|
|
|
Name: c.Query("name"), |
|
|
|
|
Page: page, |
|
|
|
|
@ -148,7 +158,7 @@ func (hs *HTTPServer) SearchTeams(c *contextmodel.ReqContext) response.Response |
|
|
|
|
|
|
|
|
|
queryResult, err := hs.teamService.SearchTeams(c.Req.Context(), &query) |
|
|
|
|
if err != nil { |
|
|
|
|
return response.Error(500, "Failed to search Teams", err) |
|
|
|
|
return response.Error(http.StatusInternalServerError, "Failed to search Teams", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
teamIDs := map[string]bool{} |
|
|
|
|
@ -157,7 +167,7 @@ func (hs *HTTPServer) SearchTeams(c *contextmodel.ReqContext) response.Response |
|
|
|
|
teamIDs[strconv.FormatInt(team.ID, 10)] = true |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
metadata := hs.getMultiAccessControlMetadata(c, c.OrgID, "teams:id:", teamIDs) |
|
|
|
|
metadata := hs.getMultiAccessControlMetadata(c, c.SignedInUser.GetOrgID(), "teams:id:", teamIDs) |
|
|
|
|
if len(metadata) > 0 { |
|
|
|
|
for _, team := range queryResult.Teams { |
|
|
|
|
team.AccessControl = metadata[strconv.FormatInt(team.ID, 10)] |
|
|
|
|
@ -187,7 +197,7 @@ func (hs *HTTPServer) GetTeamByID(c *contextmodel.ReqContext) response.Response |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
query := team.GetTeamByIDQuery{ |
|
|
|
|
OrgID: c.OrgID, |
|
|
|
|
OrgID: c.SignedInUser.GetOrgID(), |
|
|
|
|
ID: teamId, |
|
|
|
|
SignedInUser: c.SignedInUser, |
|
|
|
|
HiddenUsers: hs.Cfg.HiddenUsers, |
|
|
|
|
@ -196,14 +206,14 @@ func (hs *HTTPServer) GetTeamByID(c *contextmodel.ReqContext) response.Response |
|
|
|
|
queryResult, err := hs.teamService.GetTeamByID(c.Req.Context(), &query) |
|
|
|
|
if err != nil { |
|
|
|
|
if errors.Is(err, team.ErrTeamNotFound) { |
|
|
|
|
return response.Error(404, "Team not found", err) |
|
|
|
|
return response.Error(http.StatusNotFound, "Team not found", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return response.Error(500, "Failed to get Team", err) |
|
|
|
|
return response.Error(http.StatusInternalServerError, "Failed to get Team", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// Add accesscontrol metadata
|
|
|
|
|
queryResult.AccessControl = hs.getAccessControlMetadata(c, c.OrgID, "teams:id:", strconv.FormatInt(queryResult.ID, 10)) |
|
|
|
|
queryResult.AccessControl = hs.getAccessControlMetadata(c, c.SignedInUser.GetOrgID(), "teams:id:", strconv.FormatInt(queryResult.ID, 10)) |
|
|
|
|
|
|
|
|
|
queryResult.AvatarURL = dtos.GetGravatarUrlWithDefault(queryResult.Email, queryResult.Name) |
|
|
|
|
return response.JSON(http.StatusOK, &queryResult) |
|
|
|
|
@ -223,7 +233,7 @@ func (hs *HTTPServer) GetTeamPreferences(c *contextmodel.ReqContext) response.Re |
|
|
|
|
return response.Error(http.StatusBadRequest, "teamId is invalid", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return hs.getPreferencesFor(c.Req.Context(), c.OrgID, 0, teamId) |
|
|
|
|
return hs.getPreferencesFor(c.Req.Context(), c.SignedInUser.GetOrgID(), 0, teamId) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// swagger:route PUT /teams/{team_id}/preferences teams updateTeamPreferences
|
|
|
|
|
@ -246,7 +256,7 @@ func (hs *HTTPServer) UpdateTeamPreferences(c *contextmodel.ReqContext) response |
|
|
|
|
return response.Error(http.StatusBadRequest, "teamId is invalid", err) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
return hs.updatePreferencesFor(c.Req.Context(), c.OrgID, 0, teamId, &dtoCmd) |
|
|
|
|
return hs.updatePreferencesFor(c.Req.Context(), c.SignedInUser.GetOrgID(), 0, teamId, &dtoCmd) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// swagger:parameters updateTeamPreferences
|
|
|
|
|
|