mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2025-01-03 06:08:42 +00:00
Introduce and use Pagination helper func (#1236)
This commit is contained in:
parent
fd6923fe20
commit
c7fd1eb9d4
3 changed files with 73 additions and 57 deletions
|
@ -23,3 +23,28 @@ func ExtractHostFromCloneURL(cloneURL string) (string, error) {
|
||||||
|
|
||||||
return host, nil
|
return host, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Paginate iterates over a func call until it does not return new items and return it as list
|
||||||
|
func Paginate[T any](get func(page int) ([]T, error)) ([]T, error) {
|
||||||
|
items := make([]T, 0, 10)
|
||||||
|
page := 1
|
||||||
|
lenFirstBatch := -1
|
||||||
|
|
||||||
|
for {
|
||||||
|
batch, err := get(page)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
items = append(items, batch...)
|
||||||
|
|
||||||
|
if page == 1 {
|
||||||
|
lenFirstBatch = len(batch)
|
||||||
|
} else if len(batch) < lenFirstBatch || len(batch) == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
page++
|
||||||
|
}
|
||||||
|
|
||||||
|
return items, nil
|
||||||
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@ package common_test
|
||||||
import (
|
import (
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
"github.com/woodpecker-ci/woodpecker/server/remote/common"
|
"github.com/woodpecker-ci/woodpecker/server/remote/common"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -16,3 +17,29 @@ func Test_Netrc(t *testing.T) {
|
||||||
t.Errorf("Expected host to be git.example.com, got %s", host)
|
t.Errorf("Expected host to be git.example.com, got %s", host)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestPaginate(t *testing.T) {
|
||||||
|
apiExec := 0
|
||||||
|
apiMock := func(page int) []int {
|
||||||
|
apiExec++
|
||||||
|
switch page {
|
||||||
|
case 0, 1:
|
||||||
|
return []int{11, 12, 13}
|
||||||
|
case 2:
|
||||||
|
return []int{21, 22, 23}
|
||||||
|
case 3:
|
||||||
|
return []int{31, 32}
|
||||||
|
default:
|
||||||
|
return []int{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
result, _ := common.Paginate(func(page int) ([]int, error) {
|
||||||
|
return apiMock(page), nil
|
||||||
|
})
|
||||||
|
|
||||||
|
assert.EqualValues(t, 3, apiExec)
|
||||||
|
if assert.Len(t, result, 8) {
|
||||||
|
assert.EqualValues(t, []int{11, 12, 13, 21, 22, 23, 31, 32}, result)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -192,10 +192,7 @@ func (c *Gitea) Teams(ctx context.Context, u *model.User) ([]*model.Team, error)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
teams := make([]*model.Team, 0, perPage)
|
return common.Paginate(func(page int) ([]*model.Team, error) {
|
||||||
|
|
||||||
page := 1
|
|
||||||
for {
|
|
||||||
orgs, _, err := client.ListMyOrgs(
|
orgs, _, err := client.ListMyOrgs(
|
||||||
gitea.ListOrgsOptions{
|
gitea.ListOrgsOptions{
|
||||||
ListOptions: gitea.ListOptions{
|
ListOptions: gitea.ListOptions{
|
||||||
|
@ -204,21 +201,12 @@ func (c *Gitea) Teams(ctx context.Context, u *model.User) ([]*model.Team, error)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
if err != nil {
|
teams := make([]*model.Team, 0, len(orgs))
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, org := range orgs {
|
for _, org := range orgs {
|
||||||
teams = append(teams, toTeam(org, c.URL))
|
teams = append(teams, toTeam(org, c.URL))
|
||||||
}
|
}
|
||||||
|
return teams, err
|
||||||
if len(orgs) < perPage {
|
})
|
||||||
break
|
|
||||||
}
|
|
||||||
page++
|
|
||||||
}
|
|
||||||
|
|
||||||
return teams, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TeamPerm is not supported by the Gitea driver.
|
// TeamPerm is not supported by the Gitea driver.
|
||||||
|
@ -255,17 +243,13 @@ func (c *Gitea) Repo(ctx context.Context, u *model.User, id model.RemoteID, owne
|
||||||
// Repos returns a list of all repositories for the Gitea account, including
|
// Repos returns a list of all repositories for the Gitea account, including
|
||||||
// organization repositories.
|
// organization repositories.
|
||||||
func (c *Gitea) Repos(ctx context.Context, u *model.User) ([]*model.Repo, error) {
|
func (c *Gitea) Repos(ctx context.Context, u *model.User) ([]*model.Repo, error) {
|
||||||
repos := make([]*model.Repo, 0, perPage)
|
|
||||||
|
|
||||||
client, err := c.newClientToken(ctx, u.Token)
|
client, err := c.newClientToken(ctx, u.Token)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// Gitea SDK forces us to read repo list paginated.
|
return common.Paginate(func(page int) ([]*model.Repo, error) {
|
||||||
page := 1
|
repos, _, err := client.ListMyRepos(
|
||||||
for {
|
|
||||||
all, _, err := client.ListMyRepos(
|
|
||||||
gitea.ListReposOptions{
|
gitea.ListReposOptions{
|
||||||
ListOptions: gitea.ListOptions{
|
ListOptions: gitea.ListOptions{
|
||||||
Page: page,
|
Page: page,
|
||||||
|
@ -273,22 +257,12 @@ func (c *Gitea) Repos(ctx context.Context, u *model.User) ([]*model.Repo, error)
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
if err != nil {
|
result := make([]*model.Repo, 0, len(repos))
|
||||||
return nil, err
|
for _, repo := range repos {
|
||||||
|
result = append(result, toRepo(repo))
|
||||||
}
|
}
|
||||||
|
return result, err
|
||||||
for _, repo := range all {
|
})
|
||||||
repos = append(repos, toRepo(repo))
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(all) < perPage {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
// Last page was not empty so more repos may be available - continue loop.
|
|
||||||
page++
|
|
||||||
}
|
|
||||||
|
|
||||||
return repos, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Perm returns the user permissions for the named Gitea repository.
|
// Perm returns the user permissions for the named Gitea repository.
|
||||||
|
@ -462,27 +436,17 @@ func (c *Gitea) Branches(ctx context.Context, u *model.User, r *model.Repo) ([]s
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
branches := make([]string, 0)
|
branches, err := common.Paginate(func(page int) ([]string, error) {
|
||||||
|
branches, _, err := client.ListRepoBranches(r.Owner, r.Name,
|
||||||
page := 1
|
gitea.ListRepoBranchesOptions{ListOptions: gitea.ListOptions{Page: page}})
|
||||||
|
result := make([]string, 0, len(branches))
|
||||||
for {
|
for i := range branches {
|
||||||
giteaBranches, _, err := client.ListRepoBranches(r.Owner, r.Name, gitea.ListRepoBranchesOptions{
|
result[i] = branches[i].Name
|
||||||
ListOptions: gitea.ListOptions{
|
|
||||||
Page: page,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if len(giteaBranches) > 0 {
|
|
||||||
for _, branch := range giteaBranches {
|
|
||||||
branches = append(branches, branch.Name)
|
|
||||||
}
|
|
||||||
page++
|
|
||||||
} else {
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
return result, err
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return branches, nil
|
return branches, nil
|
||||||
|
|
Loading…
Reference in a new issue