harbor/api/member.go

220 lines
7.3 KiB
Go
Raw Normal View History

2016-02-01 12:59:10 +01:00
/*
Copyright (c) 2016 VMware, Inc. All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
2016-02-26 11:54:14 +01:00
2016-02-01 12:59:10 +01:00
package api
import (
"net/http"
"strconv"
2016-02-01 12:59:10 +01:00
"github.com/vmware/harbor/dao"
"github.com/vmware/harbor/models"
"github.com/astaxie/beego"
)
2016-02-26 11:35:55 +01:00
// ProjectMemberAPI handles request to /api/projects/{}/members/{}
2016-02-01 12:59:10 +01:00
type ProjectMemberAPI struct {
BaseAPI
memberID int
currentUserID int
2016-02-01 12:59:10 +01:00
project *models.Project
}
type memberReq struct {
Username string `json:"user_name"`
UserID int `json:"user_id"`
2016-02-01 12:59:10 +01:00
Roles []int `json:"roles"`
}
2016-02-26 11:35:55 +01:00
// Prepare validates the URL and parms
2016-02-01 12:59:10 +01:00
func (pma *ProjectMemberAPI) Prepare() {
pid, err := strconv.ParseInt(pma.Ctx.Input.Param(":pid"), 10, 64)
if err != nil {
beego.Error("Error parsing project id:", pid, ", error:", err)
pma.CustomAbort(http.StatusBadRequest, "invalid project Id")
2016-02-01 12:59:10 +01:00
return
}
2016-02-26 04:26:54 +01:00
p, err := dao.GetProjectByID(pid)
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Error occurred in GetProjectById:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
if p == nil {
beego.Warning("Project with id:", pid, "does not exist.")
pma.CustomAbort(http.StatusNotFound, "Project does not exist")
2016-02-01 12:59:10 +01:00
}
pma.project = p
pma.currentUserID = pma.ValidateUser()
2016-02-01 12:59:10 +01:00
mid := pma.Ctx.Input.Param(":mid")
if mid == "current" {
pma.memberID = pma.currentUserID
2016-02-01 12:59:10 +01:00
} else if len(mid) == 0 {
pma.memberID = 0
2016-02-01 12:59:10 +01:00
} else if len(mid) > 0 {
memberID, err := strconv.Atoi(mid)
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Invalid member Id, error:", err)
pma.CustomAbort(http.StatusBadRequest, "Invalid member id")
2016-02-01 12:59:10 +01:00
}
pma.memberID = memberID
2016-02-01 12:59:10 +01:00
}
}
2016-02-26 11:35:55 +01:00
// Get ...
2016-02-01 12:59:10 +01:00
func (pma *ProjectMemberAPI) Get() {
2016-02-26 03:15:01 +01:00
pid := pma.project.ProjectID
2016-02-26 11:35:55 +01:00
if !checkProjectPermission(pma.currentUserID, pid) {
beego.Warning("Current user, user id :", pma.currentUserID, "does not have permission for project, id:", pid)
pma.RenderError(http.StatusForbidden, "")
2016-02-01 12:59:10 +01:00
return
}
if pma.memberID == 0 { //member id not set return list of the members
2016-02-01 12:59:10 +01:00
username := pma.GetString("username")
queryUser := models.User{Username: "%" + username + "%"}
2016-02-24 11:16:16 +01:00
userList, err := dao.GetUserByProject(pid, queryUser)
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Failed to query database for member list, error:", err)
pma.RenderError(http.StatusInternalServerError, "Internal Server Error")
2016-02-01 12:59:10 +01:00
return
}
pma.Data["json"] = userList
} else { //return detail of a member
2016-02-26 03:15:01 +01:00
roleList, err := dao.GetUserProjectRoles(models.User{UserID: pma.memberID}, pid)
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Error occurred in GetUserProjectRoles:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
//return empty role list to indicate if a user is not a member
result := make(map[string]interface{})
2016-02-26 03:15:01 +01:00
user, err := dao.GetUser(models.User{UserID: pma.memberID})
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Error occurred in GetUser:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
result["user_name"] = user.Username
result["user_id"] = pma.memberID
2016-02-01 12:59:10 +01:00
result["roles"] = roleList
pma.Data["json"] = result
}
pma.ServeJSON()
}
2016-02-26 11:35:55 +01:00
// Post ...
2016-02-01 12:59:10 +01:00
func (pma *ProjectMemberAPI) Post() {
2016-02-26 03:15:01 +01:00
pid := pma.project.ProjectID
userQuery := models.User{UserID: pma.currentUserID, RoleID: models.PROJECTADMIN}
2016-02-01 12:59:10 +01:00
rolelist, err := dao.GetUserProjectRoles(userQuery, pid)
if err != nil {
beego.Error("Error occurred in GetUserProjectRoles:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
if len(rolelist) == 0 {
beego.Warning("Current user, id:", pma.currentUserID, "does not have project admin role for project, id:", pid)
pma.RenderError(http.StatusForbidden, "")
2016-02-01 12:59:10 +01:00
return
}
var req memberReq
pma.DecodeJSONReq(&req)
2016-02-01 12:59:10 +01:00
username := req.Username
2016-02-26 11:35:55 +01:00
userID := checkUserExists(username)
if userID <= 0 {
2016-02-01 12:59:10 +01:00
beego.Warning("User does not exist, user name:", username)
pma.RenderError(http.StatusNotFound, "User does not exist")
2016-02-01 12:59:10 +01:00
return
}
2016-02-26 03:15:01 +01:00
rolelist, err = dao.GetUserProjectRoles(models.User{UserID: userID}, pid)
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Error occurred in GetUserProjectRoles:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
if len(rolelist) > 0 {
beego.Warning("user is already added to project, user id:", userID, ", project id:", pid)
pma.RenderError(http.StatusConflict, "user is ready in project")
2016-02-01 12:59:10 +01:00
return
}
for _, rid := range req.Roles {
err = dao.AddUserProjectRole(userID, pid, int(rid))
2016-02-01 12:59:10 +01:00
if err != nil {
beego.Error("Failed to update DB to add project user role, project id:", pid, ", user id:", userID, ", role id:", rid)
pma.RenderError(http.StatusInternalServerError, "Failed to update data in database")
2016-02-01 12:59:10 +01:00
return
}
}
}
2016-02-26 11:35:55 +01:00
// Put ...
2016-02-01 12:59:10 +01:00
func (pma *ProjectMemberAPI) Put() {
2016-02-26 03:15:01 +01:00
pid := pma.project.ProjectID
mid := pma.memberID
2016-02-26 03:15:01 +01:00
userQuery := models.User{UserID: pma.currentUserID, RoleID: models.PROJECTADMIN}
2016-02-01 12:59:10 +01:00
rolelist, err := dao.GetUserProjectRoles(userQuery, pid)
if err != nil {
beego.Error("Error occurred in GetUserProjectRoles:", err)
pma.CustomAbort(http.StatusInternalServerError, "Internal error.")
2016-02-01 12:59:10 +01:00
}
if len(rolelist) == 0 {
beego.Warning("Current user, id:", pma.currentUserID, ", does not have project admin role for project, id:", pid)
pma.RenderError(http.StatusForbidden, "")
2016-02-01 12:59:10 +01:00
return
}
var req memberReq
pma.DecodeJSONReq(&req)
2016-02-26 03:15:01 +01:00
roleList, err := dao.GetUserProjectRoles(models.User{UserID: mid}, pid)
2016-02-01 12:59:10 +01:00
if len(roleList) == 0 {
beego.Warning("User is not in project, user id:", mid, ", project id:", pid)
pma.RenderError(http.StatusNotFound, "user not exist in project")
2016-02-01 12:59:10 +01:00
return
}
//TODO: delete and insert should in one transaction
//delete user project role record for the given user
err = dao.DeleteUserProjectRoles(mid, pid)
if err != nil {
beego.Error("Failed to delete project roles for user, user id:", mid, ", project id: ", pid, ", error: ", err)
pma.RenderError(http.StatusInternalServerError, "Failed to update data in DB")
2016-02-01 12:59:10 +01:00
return
}
//insert roles in request
for _, rid := range req.Roles {
err = dao.AddUserProjectRole(mid, pid, int(rid))
if err != nil {
beego.Error("Failed to update DB to add project user role, project id:", pid, ", user id:", mid, ", role id:", rid)
pma.RenderError(http.StatusInternalServerError, "Failed to update data in database")
2016-02-01 12:59:10 +01:00
return
}
}
}
2016-02-26 11:35:55 +01:00
// Delete ...
2016-02-01 12:59:10 +01:00
func (pma *ProjectMemberAPI) Delete() {
2016-02-26 03:15:01 +01:00
pid := pma.project.ProjectID
mid := pma.memberID
2016-02-26 03:15:01 +01:00
userQuery := models.User{UserID: pma.currentUserID, RoleID: models.PROJECTADMIN}
2016-02-01 12:59:10 +01:00
rolelist, err := dao.GetUserProjectRoles(userQuery, pid)
if len(rolelist) == 0 {
beego.Warning("Current user, id:", pma.currentUserID, ", does not have project admin role for project, id:", pid)
pma.RenderError(http.StatusForbidden, "")
2016-02-01 12:59:10 +01:00
return
}
err = dao.DeleteUserProjectRoles(mid, pid)
if err != nil {
beego.Error("Failed to delete project roles for user, user id:", mid, ", project id:", pid, ", error:", err)
pma.RenderError(http.StatusInternalServerError, "Failed to update data in DB")
2016-02-01 12:59:10 +01:00
return
}
}