2016-08-08 10:40:14 +02:00
|
|
|
//These APIs provide services for manipulating Harbor project.
|
|
|
|
|
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http/httptest"
|
|
|
|
"path/filepath"
|
|
|
|
"runtime"
|
|
|
|
|
2016-09-13 11:41:32 +02:00
|
|
|
"github.com/vmware/harbor/dao"
|
|
|
|
"github.com/vmware/harbor/models"
|
|
|
|
"github.com/vmware/harbor/tests/apitests/apilib"
|
|
|
|
"github.com/vmware/harbor/utils"
|
|
|
|
// "strconv"
|
|
|
|
// "strings"
|
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
"github.com/astaxie/beego"
|
|
|
|
"github.com/dghubble/sling"
|
|
|
|
|
|
|
|
//for test env prepare
|
|
|
|
_ "github.com/vmware/harbor/auth/db"
|
|
|
|
_ "github.com/vmware/harbor/auth/ldap"
|
|
|
|
)
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
const (
|
|
|
|
jsonAcceptHeader = "application/json"
|
|
|
|
testAcceptHeader = "text/plain"
|
2016-09-02 10:05:55 +02:00
|
|
|
adminName = "admin"
|
|
|
|
adminPwd = "Harbor12345"
|
2016-08-30 12:26:41 +02:00
|
|
|
)
|
|
|
|
|
2016-09-09 10:59:53 +02:00
|
|
|
var admin, unknownUsr, testUser *usrInfo
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
type api struct {
|
|
|
|
basePath string
|
|
|
|
}
|
|
|
|
|
|
|
|
func newHarborAPI() *api {
|
|
|
|
return &api{
|
|
|
|
basePath: "",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func newHarborAPIWithBasePath(basePath string) *api {
|
|
|
|
return &api{
|
|
|
|
basePath: basePath,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type usrInfo struct {
|
|
|
|
Name string
|
|
|
|
Passwd string
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
2016-09-13 11:41:32 +02:00
|
|
|
dao.InitDatabase()
|
2016-08-08 10:40:14 +02:00
|
|
|
_, file, _, _ := runtime.Caller(1)
|
|
|
|
apppath, _ := filepath.Abs(filepath.Dir(filepath.Join(file, ".."+string(filepath.Separator))))
|
|
|
|
beego.BConfig.WebConfig.Session.SessionOn = true
|
|
|
|
beego.TestBeegoInit(apppath)
|
|
|
|
|
|
|
|
beego.Router("/api/search/", &SearchAPI{})
|
2016-08-30 12:26:41 +02:00
|
|
|
beego.Router("/api/projects/", &ProjectAPI{}, "get:List;post:Post;head:Head")
|
2016-09-02 10:05:55 +02:00
|
|
|
beego.Router("/api/projects/:id", &ProjectAPI{}, "delete:Delete;get:Get")
|
2016-09-06 10:57:24 +02:00
|
|
|
beego.Router("/api/users/?:id", &UserAPI{})
|
2016-08-08 10:40:14 +02:00
|
|
|
beego.Router("/api/users/:id([0-9]+)/password", &UserAPI{}, "put:ChangePassword")
|
2016-09-06 10:57:24 +02:00
|
|
|
beego.Router("/api/users/:id/sysadmin", &UserAPI{}, "put:ToggleUserAdminRole")
|
2016-08-30 12:26:41 +02:00
|
|
|
beego.Router("/api/projects/:id/publicity", &ProjectAPI{}, "put:ToggleProjectPublic")
|
|
|
|
beego.Router("/api/projects/:id([0-9]+)/logs/filter", &ProjectAPI{}, "post:FilterAccessLog")
|
2016-09-09 10:59:53 +02:00
|
|
|
beego.Router("/api/projects/:pid([0-9]+)/members/?:mid", &ProjectMemberAPI{}, "get:Get;post:Post;delete:Delete;put:Put")
|
2016-08-31 06:12:22 +02:00
|
|
|
beego.Router("/api/statistics", &StatisticAPI{})
|
2016-09-09 10:59:53 +02:00
|
|
|
beego.Router("/api/users/?:id", &UserAPI{})
|
2016-08-31 06:12:22 +02:00
|
|
|
beego.Router("/api/logs", &LogAPI{})
|
2016-09-09 10:59:53 +02:00
|
|
|
beego.Router("/api/repositories", &RepositoryAPI{})
|
|
|
|
beego.Router("/api/repositories/tags", &RepositoryAPI{}, "get:GetTags")
|
|
|
|
beego.Router("/api/repositories/manifests", &RepositoryAPI{}, "get:GetManifests")
|
|
|
|
beego.Router("/api/repositories/top", &RepositoryAPI{}, "get:GetTopRepos")
|
2016-09-13 10:09:13 +02:00
|
|
|
beego.Router("/api/targets/", &TargetAPI{}, "get:List")
|
|
|
|
beego.Router("/api/targets/", &TargetAPI{}, "post:Post")
|
|
|
|
beego.Router("/api/targets/:id([0-9]+)", &TargetAPI{})
|
|
|
|
beego.Router("/api/targets/:id([0-9]+)/policies/", &TargetAPI{}, "get:ListPolicies")
|
|
|
|
beego.Router("/api/targets/ping", &TargetAPI{}, "post:Ping")
|
2016-09-19 09:21:39 +02:00
|
|
|
beego.Router("/api/policies/replication/:id([0-9]+)", &RepPolicyAPI{})
|
|
|
|
beego.Router("/api/policies/replication", &RepPolicyAPI{}, "get:List")
|
|
|
|
beego.Router("/api/policies/replication", &RepPolicyAPI{}, "post:Post;delete:Delete")
|
|
|
|
beego.Router("/api/policies/replication/:id([0-9]+)/enablement", &RepPolicyAPI{}, "put:UpdateEnablement")
|
2016-08-08 10:40:14 +02:00
|
|
|
|
|
|
|
_ = updateInitPassword(1, "Harbor12345")
|
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
//Init user Info
|
|
|
|
admin = &usrInfo{adminName, adminPwd}
|
|
|
|
unknownUsr = &usrInfo{"unknown", "unknown"}
|
2016-09-09 10:59:53 +02:00
|
|
|
testUser = &usrInfo{TestUserName, TestUserPwd}
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
}
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
func request(_sling *sling.Sling, acceptHeader string, authInfo ...usrInfo) (int, []byte, error) {
|
|
|
|
_sling = _sling.Set("Accept", acceptHeader)
|
|
|
|
req, err := _sling.Request()
|
|
|
|
if err != nil {
|
|
|
|
return 400, nil, err
|
|
|
|
}
|
|
|
|
if len(authInfo) > 0 {
|
|
|
|
req.SetBasicAuth(authInfo[0].Name, authInfo[0].Passwd)
|
|
|
|
}
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
beego.BeeApp.Handlers.ServeHTTP(w, req)
|
|
|
|
body, err := ioutil.ReadAll(w.Body)
|
|
|
|
return w.Code, body, err
|
|
|
|
}
|
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
//Search for projects and repositories
|
|
|
|
//Implementation Notes
|
|
|
|
//The Search endpoint returns information about the projects and repositories
|
|
|
|
//offered at public status or related to the current logged in user.
|
|
|
|
//The response includes the project and repository list in a proper display order.
|
|
|
|
//@param q Search parameter for project and repository name.
|
|
|
|
//@return []Search
|
|
|
|
//func (a api) SearchGet (q string) (apilib.Search, error) {
|
|
|
|
func (a api) SearchGet(q string) (apilib.Search, error) {
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
_sling := sling.New().Get(a.basePath)
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
// create path and map variables
|
|
|
|
path := "/api/search"
|
|
|
|
_sling = _sling.Path(path)
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
type QueryParams struct {
|
|
|
|
Query string `url:"q,omitempty"`
|
|
|
|
}
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
_sling = _sling.QueryStruct(&QueryParams{Query: q})
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
_, body, err := request(_sling, jsonAcceptHeader)
|
2016-08-08 10:40:14 +02:00
|
|
|
var successPayload = new(apilib.Search)
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
return *successPayload, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Create a new project.
|
|
|
|
//Implementation Notes
|
|
|
|
//This endpoint is for user to create a new project.
|
|
|
|
//@param project New created project.
|
|
|
|
//@return void
|
|
|
|
//func (a api) ProjectsPost (prjUsr usrInfo, project apilib.Project) (int, error) {
|
2016-09-02 10:05:55 +02:00
|
|
|
func (a api) ProjectsPost(prjUsr usrInfo, project apilib.ProjectReq) (int, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
// create path and map variables
|
|
|
|
path := "/api/projects/"
|
2016-09-02 10:05:55 +02:00
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
// body params
|
|
|
|
_sling = _sling.BodyJSON(project)
|
2016-08-30 12:26:41 +02:00
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
return httpStatusCode, err
|
2016-08-08 10:40:14 +02:00
|
|
|
}
|
|
|
|
|
2016-08-31 06:12:22 +02:00
|
|
|
func (a api) StatisticGet(user usrInfo) (apilib.StatisticMap, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
2016-08-08 10:40:14 +02:00
|
|
|
|
2016-08-31 06:12:22 +02:00
|
|
|
// create path and map variables
|
|
|
|
path := "/api/statistics/"
|
|
|
|
fmt.Printf("project statistic path: %s\n", path)
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
var successPayload = new(apilib.StatisticMap)
|
|
|
|
code, body, err := request(_sling, jsonAcceptHeader, user)
|
|
|
|
if 200 == code && nil == err {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
return *successPayload, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a api) LogGet(user usrInfo, startTime, endTime, lines string) (int, []apilib.AccessLog, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/logs/"
|
|
|
|
fmt.Printf("logs path: %s\n", path)
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
type QueryParams struct {
|
|
|
|
StartTime string `url:"start_time,omitempty"`
|
|
|
|
EndTime string `url:"end_time,omitempty"`
|
|
|
|
Lines string `url:"lines,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{StartTime: startTime, EndTime: endTime, Lines: lines})
|
|
|
|
var successPayload []apilib.AccessLog
|
|
|
|
code, body, err := request(_sling, jsonAcceptHeader, user)
|
|
|
|
if 200 == code && nil == err {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
return code, successPayload, err
|
2016-08-08 10:40:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
////Delete a repository or a tag in a repository.
|
|
|
|
////Delete a repository or a tag in a repository.
|
|
|
|
////This endpoint let user delete repositories and tags with repo name and tag.\n
|
|
|
|
////@param repoName The name of repository which will be deleted.
|
|
|
|
////@param tag Tag of a repository.
|
|
|
|
////@return void
|
2016-08-30 12:26:41 +02:00
|
|
|
////func (a api) RepositoriesDelete(prjUsr UsrInfo, repoName string, tag string) (int, error) {
|
|
|
|
//func (a api) RepositoriesDelete(prjUsr UsrInfo, repoName string, tag string) (int, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
// _sling := sling.New().Delete(a.basePath)
|
|
|
|
|
|
|
|
// // create path and map variables
|
|
|
|
// path := "/api/repositories"
|
|
|
|
|
|
|
|
// _sling = _sling.Path(path)
|
|
|
|
|
|
|
|
// type QueryParams struct {
|
|
|
|
// RepoName string `url:"repo_name,omitempty"`
|
|
|
|
// Tag string `url:"tag,omitempty"`
|
|
|
|
// }
|
|
|
|
|
|
|
|
// _sling = _sling.QueryStruct(&QueryParams{RepoName: repoName, Tag: tag})
|
|
|
|
// // accept header
|
|
|
|
// accepts := []string{"application/json", "text/plain"}
|
|
|
|
// for key := range accepts {
|
|
|
|
// _sling = _sling.Set("Accept", accepts[key])
|
|
|
|
// break // only use the first Accept
|
|
|
|
// }
|
|
|
|
|
|
|
|
// req, err := _sling.Request()
|
|
|
|
// req.SetBasicAuth(prjUsr.Name, prjUsr.Passwd)
|
|
|
|
// //fmt.Printf("request %+v", req)
|
|
|
|
|
|
|
|
// client := &http.Client{}
|
|
|
|
// httpResponse, err := client.Do(req)
|
|
|
|
// defer httpResponse.Body.Close()
|
|
|
|
|
|
|
|
// if err != nil {
|
|
|
|
// // handle error
|
|
|
|
// }
|
|
|
|
// return httpResponse.StatusCode, err
|
|
|
|
//}
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
//Delete project by projectID
|
|
|
|
func (a api) ProjectsDelete(prjUsr usrInfo, projectID string) (int, error) {
|
|
|
|
_sling := sling.New().Delete(a.basePath)
|
|
|
|
|
|
|
|
//create api path
|
2016-09-02 10:05:55 +02:00
|
|
|
path := "api/projects/" + projectID
|
2016-08-30 12:26:41 +02:00
|
|
|
_sling = _sling.Path(path)
|
2016-09-02 10:05:55 +02:00
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, prjUsr)
|
2016-08-30 12:26:41 +02:00
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Check if the project name user provided already exists
|
|
|
|
func (a api) ProjectsHead(prjUsr usrInfo, projectName string) (int, error) {
|
|
|
|
_sling := sling.New().Head(a.basePath)
|
|
|
|
|
|
|
|
//create api path
|
|
|
|
path := "api/projects"
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
type QueryParams struct {
|
|
|
|
ProjectName string `url:"project_name,omitempty"`
|
|
|
|
}
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{ProjectName: projectName})
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
//Return specific project detail infomation
|
|
|
|
func (a api) ProjectsGetByPID(projectID string) (int, apilib.Project, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
//create api path
|
|
|
|
path := "api/projects/" + projectID
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
var successPayload apilib.Project
|
|
|
|
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader)
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
return httpStatusCode, successPayload, err
|
|
|
|
}
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
//Search projects by projectName and isPublic
|
|
|
|
func (a api) ProjectsGet(projectName string, isPublic int32) (int, []apilib.Project, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
//create api path
|
|
|
|
path := "api/projects"
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
type QueryParams struct {
|
|
|
|
ProjectName string `url:"project_name,omitempty"`
|
|
|
|
IsPubilc int32 `url:"is_public,omitempty"`
|
|
|
|
}
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{ProjectName: projectName, IsPubilc: isPublic})
|
|
|
|
|
|
|
|
var successPayload []apilib.Project
|
|
|
|
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader)
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
|
|
|
|
return httpStatusCode, successPayload, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update properties for a selected project.
|
|
|
|
func (a api) ToggleProjectPublicity(prjUsr usrInfo, projectID string, ispublic int32) (int, error) {
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/projects/" + projectID + "/publicity/"
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
type QueryParams struct {
|
|
|
|
Public int32 `json:"public,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.BodyJSON(&QueryParams{Public: ispublic})
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
return httpStatusCode, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get access logs accompany with a relevant project.
|
|
|
|
func (a api) ProjectLogsFilter(prjUsr usrInfo, projectID string, accessLog apilib.AccessLogFilter) (int, []byte, error) {
|
|
|
|
//func (a api) ProjectLogsFilter(prjUsr usrInfo, projectID string, accessLog apilib.AccessLog) (int, apilib.AccessLog, error) {
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/projects/" + projectID + "/logs/filter"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
// body params
|
|
|
|
_sling = _sling.BodyJSON(accessLog)
|
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
//var successPayload []apilib.AccessLog
|
2016-08-30 12:26:41 +02:00
|
|
|
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
/*
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
return httpStatusCode, body, err
|
|
|
|
// return httpStatusCode, successPayload, err
|
|
|
|
}
|
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
//-------------------------Member Test---------------------------------------//
|
|
|
|
|
2016-08-30 12:26:41 +02:00
|
|
|
//Return relevant role members of projectID
|
2016-09-02 10:05:55 +02:00
|
|
|
func (a api) GetProjectMembersByProID(prjUsr usrInfo, projectID string) (int, []apilib.User, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
2016-08-30 12:26:41 +02:00
|
|
|
|
|
|
|
path := "/api/projects/" + projectID + "/members/"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
var successPayload []apilib.User
|
2016-08-30 12:26:41 +02:00
|
|
|
|
2016-09-02 10:05:55 +02:00
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
return httpStatusCode, successPayload, err
|
2016-08-30 12:26:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//Add project role member accompany with projectID
|
2016-09-09 10:59:53 +02:00
|
|
|
func (a api) AddProjectMember(prjUsr usrInfo, projectID string, roles apilib.RoleParam) (int, error) {
|
2016-08-30 12:26:41 +02:00
|
|
|
_sling := sling.New().Post(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/projects/" + projectID + "/members/"
|
2016-09-09 10:59:53 +02:00
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(roles)
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, prjUsr)
|
|
|
|
return httpStatusCode, err
|
2016-08-30 12:26:41 +02:00
|
|
|
|
2016-09-09 10:59:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//Delete project role member accompany with projectID
|
|
|
|
func (a api) DeleteProjectMember(authInfo usrInfo, projectID string, userID string) (int, error) {
|
|
|
|
_sling := sling.New().Delete(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/projects/" + projectID + "/members/" + userID
|
2016-08-30 12:26:41 +02:00
|
|
|
_sling = _sling.Path(path)
|
2016-09-09 10:59:53 +02:00
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
2016-08-30 12:26:41 +02:00
|
|
|
|
2016-09-09 10:59:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
//Get role memberInfo by projectId and UserId
|
|
|
|
func (a api) GetMemByPIDUID(authInfo usrInfo, projectID string, userID string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/projects/" + projectID + "/members/" + userID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Put:update current project role members accompany with relevant project and user
|
|
|
|
func (a api) PutProjectMember(authInfo usrInfo, projectID string, userID string, roles apilib.RoleParam) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
path := "/api/projects/" + projectID + "/members/" + userID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(roles)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------Repositories Test---------------------------------------//
|
|
|
|
//Return relevant repos of projectID
|
|
|
|
func (a api) GetRepos(authInfo usrInfo, projectID string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/repositories/"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
2016-08-30 12:26:41 +02:00
|
|
|
|
2016-09-09 10:59:53 +02:00
|
|
|
type QueryParams struct {
|
|
|
|
ProjectID string `url:"project_id"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{ProjectID: projectID})
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get tags of a relevant repository
|
|
|
|
func (a api) GetReposTags(authInfo usrInfo, repoName string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/repositories/tags"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
type QueryParams struct {
|
|
|
|
RepoName string `url:"repo_name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{RepoName: repoName})
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get manifests of a relevant repository
|
|
|
|
func (a api) GetReposManifests(authInfo usrInfo, repoName string, tag string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/repositories/manifests"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
type QueryParams struct {
|
|
|
|
RepoName string `url:"repo_name"`
|
|
|
|
Tag string `url:"tag"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{RepoName: repoName, Tag: tag})
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get public repositories which are accessed most
|
|
|
|
func (a api) GetReposTop(authInfo usrInfo, count string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/repositories/top"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
type QueryParams struct {
|
|
|
|
Count string `url:"count"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{Count: count})
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
2016-09-13 10:09:13 +02:00
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//-------------------------Targets Test---------------------------------------//
|
|
|
|
//Create a new replication target
|
|
|
|
func (a api) AddTargets(authInfo usrInfo, repTarget apilib.RepTargetPost) (int, error) {
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(repTarget)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//List filters targets by name
|
|
|
|
func (a api) ListTargets(authInfo usrInfo, targetName string) (int, []apilib.RepTarget, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets?name=" + targetName
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
var successPayload []apilib.RepTarget
|
|
|
|
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
|
|
|
|
return httpStatusCode, successPayload, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Ping target by targetID
|
|
|
|
func (a api) PingTargetsByID(authInfo usrInfo, targetID string) (int, error) {
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets/ping?id=" + targetID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get target by targetID
|
|
|
|
func (a api) GetTargetByID(authInfo usrInfo, targetID string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets/" + targetID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update target by targetID
|
|
|
|
func (a api) PutTargetByID(authInfo usrInfo, targetID string, repTarget apilib.RepTargetPost) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets/" + targetID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(repTarget)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//List the target relevant policies by targetID
|
|
|
|
func (a api) GetTargetPoliciesByID(authInfo usrInfo, targetID string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets/" + targetID + "/policies/"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Delete target by targetID
|
|
|
|
func (a api) DeleteTargetsByID(authInfo usrInfo, targetID string) (int, error) {
|
|
|
|
_sling := sling.New().Delete(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/targets/" + targetID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
2016-09-19 09:21:39 +02:00
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------Replication_Policy Test--------------------------------//
|
|
|
|
|
|
|
|
//Create a new replication policy
|
|
|
|
func (a api) AddPolicy(authInfo usrInfo, repPolicy apilib.RepPolicyPost) (int, error) {
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(repPolicy)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//List policies by policyName and projectID
|
|
|
|
func (a api) ListPolicies(authInfo usrInfo, policyName string, proID string) (int, []apilib.RepPolicy, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
type QueryParams struct {
|
|
|
|
PolicyName string `url:"name"`
|
|
|
|
ProjectID string `url:"project_id"`
|
|
|
|
}
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{PolicyName: policyName, ProjectID: proID})
|
|
|
|
|
|
|
|
var successPayload []apilib.RepPolicy
|
|
|
|
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
|
|
|
|
if err == nil && httpStatusCode == 200 {
|
|
|
|
err = json.Unmarshal(body, &successPayload)
|
|
|
|
}
|
|
|
|
return httpStatusCode, successPayload, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get replication policy by policyID
|
|
|
|
func (a api) GetPolicyByID(authInfo usrInfo, policyID string) (int, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/" + policyID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update policyInfo by policyID
|
|
|
|
func (a api) PutPolicyInfoByID(authInfo usrInfo, policyID string, policyUpdate apilib.RepPolicyUpdate) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/" + policyID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(policyUpdate)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update policy enablement flag by policyID
|
|
|
|
func (a api) PutPolicyEnableByID(authInfo usrInfo, policyID string, policyEnable apilib.RepPolicyEnablementReq) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/" + policyID + "/enablement"
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
_sling = _sling.BodyJSON(policyEnable)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Delete policy by policyID
|
|
|
|
func (a api) DeletePolicyByID(authInfo usrInfo, policyID string) (int, error) {
|
|
|
|
_sling := sling.New().Delete(a.basePath)
|
|
|
|
|
|
|
|
path := "/api/policies/replication/" + policyID
|
|
|
|
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
2016-09-09 10:59:53 +02:00
|
|
|
return httpStatusCode, err
|
2016-08-30 12:26:41 +02:00
|
|
|
}
|
|
|
|
|
2016-08-08 10:40:14 +02:00
|
|
|
//Return projects created by Harbor
|
|
|
|
//func (a HarborApi) ProjectsGet (projectName string, isPublic int32) ([]Project, error) {
|
|
|
|
// }
|
|
|
|
|
|
|
|
//Check if the project name user provided already exists.
|
|
|
|
//func (a HarborApi) ProjectsHead (projectName string) (error) {
|
|
|
|
//}
|
|
|
|
|
|
|
|
//Get access logs accompany with a relevant project.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdLogsFilterPost (projectID int32, accessLog AccessLog) ([]AccessLog, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Return a project's relevant role members.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdMembersGet (projectID int32) ([]Role, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Add project role member accompany with relevant project and user.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdMembersPost (projectID int32, roles RoleParam) (error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Delete project role members accompany with relevant project and user.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdMembersUserIdDelete (projectID int32, userId int32) (error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Return role members accompany with relevant project and user.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdMembersUserIdGet (projectID int32, userId int32) ([]Role, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Update project role members accompany with relevant project and user.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdMembersUserIdPut (projectID int32, userId int32, roles RoleParam) (error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Update properties for a selected project.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) ProjectsProjectIdPut (projectID int32, project Project) (error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Get repositories accompany with relevant project and repo name.
|
2016-08-30 12:26:41 +02:00
|
|
|
//func (a HarborApi) RepositoriesGet (projectID int32, q string) ([]Repository, error) {
|
2016-08-08 10:40:14 +02:00
|
|
|
//}
|
|
|
|
|
|
|
|
//Get manifests of a relevant repository.
|
|
|
|
//func (a HarborApi) RepositoriesManifestGet (repoName string, tag string) (error) {
|
|
|
|
//}
|
|
|
|
|
|
|
|
//Get tags of a relevant repository.
|
|
|
|
//func (a HarborApi) RepositoriesTagsGet (repoName string) (error) {
|
|
|
|
//}
|
|
|
|
|
|
|
|
//Get registered users of Harbor.
|
2016-09-06 10:57:24 +02:00
|
|
|
func (a api) UsersGet(userName string, authInfo usrInfo) (int, []apilib.User, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/"
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
// body params
|
|
|
|
type QueryParams struct {
|
|
|
|
UserName string `url:"username, omitempty"`
|
|
|
|
}
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{UserName: userName})
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
var successPayLoad []apilib.User
|
|
|
|
if 200 == httpStatusCode && nil == err {
|
|
|
|
err = json.Unmarshal(body, &successPayLoad)
|
|
|
|
}
|
|
|
|
return httpStatusCode, successPayLoad, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Get registered users by userid.
|
|
|
|
func (a api) UsersGetByID(userName string, authInfo usrInfo, userID int) (int, apilib.User, error) {
|
|
|
|
_sling := sling.New().Get(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/" + fmt.Sprintf("%d", userID)
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
// body params
|
|
|
|
type QueryParams struct {
|
|
|
|
UserName string `url:"username, omitempty"`
|
|
|
|
}
|
|
|
|
_sling = _sling.QueryStruct(&QueryParams{UserName: userName})
|
|
|
|
httpStatusCode, body, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
var successPayLoad apilib.User
|
|
|
|
if 200 == httpStatusCode && nil == err {
|
|
|
|
err = json.Unmarshal(body, &successPayLoad)
|
|
|
|
}
|
|
|
|
return httpStatusCode, successPayLoad, err
|
|
|
|
}
|
2016-08-08 10:40:14 +02:00
|
|
|
|
|
|
|
//Creates a new user account.
|
2016-09-06 10:57:24 +02:00
|
|
|
func (a api) UsersPost(user apilib.User, authInfo ...usrInfo) (int, error) {
|
|
|
|
_sling := sling.New().Post(a.basePath)
|
2016-08-08 10:40:14 +02:00
|
|
|
|
2016-09-06 10:57:24 +02:00
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/"
|
2016-08-08 10:40:14 +02:00
|
|
|
|
2016-09-06 10:57:24 +02:00
|
|
|
_sling = _sling.Path(path)
|
2016-08-08 10:40:14 +02:00
|
|
|
|
2016-09-06 10:57:24 +02:00
|
|
|
// body params
|
|
|
|
_sling = _sling.BodyJSON(user)
|
|
|
|
var httpStatusCode int
|
|
|
|
var err error
|
|
|
|
if len(authInfo) > 0 {
|
|
|
|
httpStatusCode, _, err = request(_sling, jsonAcceptHeader, authInfo[0])
|
|
|
|
} else {
|
|
|
|
httpStatusCode, _, err = request(_sling, jsonAcceptHeader)
|
|
|
|
}
|
|
|
|
return httpStatusCode, err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update a registered user to change profile.
|
|
|
|
func (a api) UsersPut(userID int, profile apilib.UserProfile, authInfo usrInfo) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/" + fmt.Sprintf("%d", userID)
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
|
|
|
|
// body params
|
|
|
|
_sling = _sling.BodyJSON(profile)
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update a registered user to be an administrator of Harbor.
|
|
|
|
func (a api) UsersToggleAdminRole(userID int, authInfo usrInfo, hasAdminRole int32) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/" + fmt.Sprintf("%d", userID) + "/sysadmin"
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
type QueryParams struct {
|
|
|
|
HasAdminRole int32 `json:"has_admin_role,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
_sling = _sling.BodyJSON(&QueryParams{HasAdminRole: hasAdminRole})
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Update password of a registered user.
|
|
|
|
func (a api) UsersUpdatePassword(userID int, password apilib.Password, authInfo usrInfo) (int, error) {
|
|
|
|
_sling := sling.New().Put(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/" + fmt.Sprintf("%d", userID) + "/password"
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
// body params
|
|
|
|
_sling = _sling.BodyJSON(password)
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
|
|
|
|
|
|
|
//Mark a registered user as be removed.
|
|
|
|
func (a api) UsersDelete(userID int, authInfo usrInfo) (int, error) {
|
|
|
|
_sling := sling.New().Delete(a.basePath)
|
|
|
|
// create path and map variables
|
|
|
|
path := "/api/users/" + fmt.Sprintf("%d", userID)
|
|
|
|
_sling = _sling.Path(path)
|
|
|
|
httpStatusCode, _, err := request(_sling, jsonAcceptHeader, authInfo)
|
|
|
|
return httpStatusCode, err
|
|
|
|
}
|
2016-08-08 10:40:14 +02:00
|
|
|
func updateInitPassword(userID int, password string) error {
|
|
|
|
queryUser := models.User{UserID: userID}
|
|
|
|
user, err := dao.GetUser(queryUser)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Failed to get user, userID: %d %v", userID, err)
|
|
|
|
}
|
|
|
|
if user == nil {
|
|
|
|
return fmt.Errorf("User id: %d does not exist.", userID)
|
|
|
|
}
|
|
|
|
if user.Salt == "" {
|
2016-09-13 11:41:32 +02:00
|
|
|
user.Salt = utils.GenerateRandomString()
|
2016-08-08 10:40:14 +02:00
|
|
|
user.Password = password
|
|
|
|
err = dao.ChangeUserPassword(*user)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Failed to update user encrypted password, userID: %d, err: %v", userID, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|