backend/plugins/gitlab/api/remote_api.go (205 lines of code) (raw):
/*
Licensed to the Apache Software Foundation (ASF) under one or more
contributor license agreements. See the NOTICE file distributed with
this work for additional information regarding copyright ownership.
The ASF licenses this file to You 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.
*/
package api
import (
"fmt"
"net/http"
"net/url"
"strings"
"github.com/apache/incubator-devlake/core/errors"
"github.com/apache/incubator-devlake/core/plugin"
"github.com/apache/incubator-devlake/helpers/pluginhelper/api"
dsmodels "github.com/apache/incubator-devlake/helpers/pluginhelper/api/models"
"github.com/apache/incubator-devlake/plugins/gitlab/models"
)
const USERS_PREFIX = "user:"
type GitlabRemotePagination struct {
Page int `json:"page" mapstructure:"page"`
PerPage int `json:"per_page" mapstructure:"per_page"`
Step string `json:"step" mapstructure:"step"`
}
func (p GitlabRemotePagination) ToQuery() url.Values {
return url.Values{
"page": {fmt.Sprintf("%v", p.Page)},
"per_page": {fmt.Sprintf("%v", p.PerPage)},
}
}
func listGitlabRemoteScopes(
connection *models.GitlabConnection,
apiClient plugin.ApiClient,
groupId string,
page GitlabRemotePagination,
) (
children []dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject],
nextPage *GitlabRemotePagination,
err errors.Error,
) {
if page.Page == 0 {
page.Page = 1
}
if page.PerPage == 0 {
page.PerPage = 100
}
if page.Step == "" {
page.Step = "group"
}
// load all groups unless groupId is user's own account
if page.Step == "group" && !strings.HasPrefix(groupId, USERS_PREFIX) {
children, nextPage, err = listGitlabRemoteGroups(connection, apiClient, groupId, page)
if err != nil {
return
}
}
if groupId == "" || nextPage != nil {
return
}
// no more groups, start to load projects under the group
var moreChild []dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject]
moreChild, nextPage, err = listGitlabRemoteProjects(connection, apiClient, groupId, GitlabRemotePagination{
Page: 1,
PerPage: page.PerPage,
Step: "project",
})
if err != nil {
return
}
children = append(children, moreChild...)
return
}
func listGitlabRemoteGroups(
connection *models.GitlabConnection,
apiClient plugin.ApiClient,
groupId string,
page GitlabRemotePagination,
) (
children []dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject],
nextPage *GitlabRemotePagination,
err errors.Error,
) {
apiPath := ""
query := page.ToQuery()
var res *http.Response
if groupId == "" && page.Page == 1 {
// make users own account as a group
children = append(children, dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject]{
Type: api.RAS_ENTRY_TYPE_GROUP,
Id: USERS_PREFIX + fmt.Sprintf("%v", apiClient.GetData("UserId")),
Name: apiClient.GetData("UserName").(string),
FullName: apiClient.GetData("UserName").(string),
})
}
var parentId *string
if groupId == "" {
apiPath = "groups"
query.Set("top_level_only", "true")
} else {
apiPath = fmt.Sprintf("groups/%s/subgroups", groupId)
parentId = &groupId
}
res, err = apiClient.Get(apiPath, query, nil)
var resGroups []models.GroupResponse
errors.Must(api.UnmarshalResponse(res, &resGroups))
for _, group := range resGroups {
children = append(children, dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject]{
Type: api.RAS_ENTRY_TYPE_GROUP,
Id: fmt.Sprintf("%v", group.Id),
ParentId: parentId,
Name: group.Name,
FullName: group.FullPath,
})
}
nextPage = getNextPage(&page, res)
return
}
func listGitlabRemoteProjects(
connection *models.GitlabConnection,
apiClient plugin.ApiClient,
groupId string,
page GitlabRemotePagination,
) (
children []dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject],
nextPage *GitlabRemotePagination,
err errors.Error,
) {
apiPath := ""
query := page.ToQuery()
query.Set("archived", "false")
query.Set("min_access_level", "20")
//
if strings.HasPrefix(groupId, USERS_PREFIX) {
apiPath = fmt.Sprintf("users/%s/projects", strings.TrimPrefix(groupId, USERS_PREFIX))
} else {
apiPath = fmt.Sprintf("groups/%s/projects", groupId)
}
res, err := apiClient.Get(apiPath, query, nil)
if err != nil {
return nil, nil, err
}
var resProjects []models.GitlabApiProject
errors.Must(api.UnmarshalResponse(res, &resProjects))
for _, project := range resProjects {
children = append(children, toProjectModel(&project))
}
nextPage = getNextPage(&page, res)
return
}
func toProjectModel(project *models.GitlabApiProject) dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject] {
var parentId string
if project.Namespace.Kind == "user" {
parentId = USERS_PREFIX + fmt.Sprintf("%v", project.Owner.ID)
} else {
parentId = fmt.Sprintf("%v", project.Namespace.ID)
}
return dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject]{
Type: api.RAS_ENTRY_TYPE_SCOPE,
Id: fmt.Sprintf("%v", project.GitlabId),
ParentId: &parentId,
Name: project.Name,
FullName: project.PathWithNamespace,
Data: project.ConvertApiScope(),
}
}
func getNextPage(page *GitlabRemotePagination, res *http.Response) *GitlabRemotePagination {
if res.Header.Get("x-next-page") == "" {
return nil
}
return &GitlabRemotePagination{
Page: page.Page + 1,
PerPage: page.PerPage,
Step: page.Step,
}
}
// RemoteScopes list all available scopes on the remote server
// @Summary list all available scopes on the remote server
// @Description list all available scopes on the remote server
// @Accept application/json
// @Param connectionId path int false "connection ID"
// @Param groupId query string false "group ID"
// @Param pageToken query string false "page Token"
// @Failure 400 {object} shared.ApiBody "Bad Request"
// @Failure 500 {object} shared.ApiBody "Internal Error"
// @Success 200 {object} dsmodels.DsRemoteApiScopeList[models.GitlabProject]
// @Tags plugins/gitlab
// @Router /plugins/gitlab/connections/{connectionId}/remote-scopes [GET]
func RemoteScopes(input *plugin.ApiResourceInput) (*plugin.ApiResourceOutput, errors.Error) {
return raScopeList.Get(input)
}
func searchGitlabScopes(
apiClient plugin.ApiClient,
params *dsmodels.DsRemoteApiScopeSearchParams,
) (
children []dsmodels.DsRemoteApiScopeListEntry[models.GitlabProject],
err errors.Error,
) {
res, err := apiClient.Get(
"projects",
url.Values{
"search": []string{params.Search},
"page": []string{fmt.Sprintf("%v", params.Page)},
"per_page": []string{fmt.Sprintf("%v", params.PageSize)},
"archived": {"false"},
"min_access_level": {"20"},
},
nil,
)
if err != nil {
return nil, err
}
var resBody []models.GitlabApiProject
err = api.UnmarshalResponse(res, &resBody)
if err != nil {
return nil, err
}
for _, r := range resBody {
children = append(children, toProjectModel(&r))
}
return
}
// SearchRemoteScopes searches projects on the remote server
// @Summary searches projects on the remote server
// @Description searches projects on the remote server
// @Accept application/json
// @Param connectionId path int false "connection ID"
// @Param search query string false "search"
// @Param page query int false "page number"
// @Param pageSize query int false "page size per page"
// @Failure 400 {object} shared.ApiBody "Bad Request"
// @Failure 500 {object} shared.ApiBody "Internal Error"
// @Success 200 {object} dsmodels.DsRemoteApiScopeList[models.GitlabProject] "the parentIds are always null"
// @Tags plugins/gitlab
// @Router /plugins/gitlab/connections/{connectionId}/search-remote-scopes [GET]
func SearchRemoteScopes(input *plugin.ApiResourceInput) (*plugin.ApiResourceOutput, errors.Error) {
return raScopeSearch.Get(input)
}
// @Summary Remote server API proxy
// @Description Forward API requests to the specified remote server
// @Param connectionId path int true "connection ID"
// @Param path path string true "path to a API endpoint"
// @Router /plugins/gitlab/connections/{connectionId}/proxy/{path} [GET]
// @Tags plugins/gitlab
func Proxy(input *plugin.ApiResourceInput) (*plugin.ApiResourceOutput, errors.Error) {
return raProxy.Proxy(input)
}