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) }