]> source.dussan.org Git - gitea.git/commitdiff
[API] Add pagination to ListBranches (#14524)
author6543 <6543@obermui.de>
Wed, 3 Feb 2021 19:06:13 +0000 (20:06 +0100)
committerGitHub <noreply@github.com>
Wed, 3 Feb 2021 19:06:13 +0000 (20:06 +0100)
* make PaginateUserSlice generic -> PaginateSlice

* Add pagination to ListBranches

* add skip, limit to Repository.GetBranches()

* Move routers/api/v1/utils/utils PaginateSlice -> modules/util/paginate.go

* repo_module.GetBranches paginate

* fix & rename & more logging

* better description

Co-authored-by: zeripath <art27@cantab.net>
Co-authored-by: a1012112796 <1012112796@qq.com>
20 files changed:
integrations/branches_test.go
modules/context/repo.go
modules/git/repo_branch.go
modules/git/repo_branch_gogit.go
modules/git/repo_branch_nogogit.go
modules/git/repo_branch_test.go
modules/git/repo_tag_nogogit.go
modules/repository/branch.go
modules/repository/init.go
modules/util/paginate.go [new file with mode: 0644]
modules/util/paginate_test.go [new file with mode: 0644]
routers/api/v1/org/org.go
routers/api/v1/repo/branch.go
routers/api/v1/utils/utils.go
routers/repo/branch.go
routers/repo/compare.go
routers/repo/issue.go
services/mirror/mirror.go
services/pull/pull.go
templates/swagger/v1_json.tmpl

index 2b9fc8dda5d7522e620adea67ca37c7996fafa16..b2230e7031bf404d41d52ee170c318eca6083856 100644 (file)
@@ -57,7 +57,9 @@ func branchAction(t *testing.T, button string) (*HTMLDoc, string) {
 
        htmlDoc := NewHTMLParser(t, resp.Body)
        link, exists := htmlDoc.doc.Find(button).Attr("data-url")
-       assert.True(t, exists, "The template has changed")
+       if !assert.True(t, exists, "The template has changed") {
+               t.Skip()
+       }
 
        req = NewRequestWithValues(t, "POST", link, map[string]string{
                "_csrf": getCsrf(t, htmlDoc.doc),
@@ -69,7 +71,7 @@ func branchAction(t *testing.T, button string) (*HTMLDoc, string) {
        req = NewRequest(t, "GET", "/user2/repo1/branches")
        resp = session.MakeRequest(t, req, http.StatusOK)
 
-       return NewHTMLParser(t, resp.Body), url.Query()["name"][0]
+       return NewHTMLParser(t, resp.Body), url.Query().Get("name")
 }
 
 func getCsrf(t *testing.T, doc *goquery.Document) string {
index 13037f4625d318f5716cb9e1b28ae308ba1b5c71..bf149b8158ac9e708366ba01c410d167a50e1153 100644 (file)
@@ -554,7 +554,7 @@ func RepoAssignment() func(http.Handler) http.Handler {
                        }
                        ctx.Data["Tags"] = tags
 
-                       brs, err := ctx.Repo.GitRepo.GetBranches()
+                       brs, _, err := ctx.Repo.GitRepo.GetBranches(0, 0)
                        if err != nil {
                                ctx.ServerError("GetBranches", err)
                                return
@@ -747,7 +747,7 @@ func RepoRefByType(refType RepoRefType) func(http.Handler) http.Handler {
                                refName = ctx.Repo.Repository.DefaultBranch
                                ctx.Repo.BranchName = refName
                                if !ctx.Repo.GitRepo.IsBranchExist(refName) {
-                                       brs, err := ctx.Repo.GitRepo.GetBranches()
+                                       brs, _, err := ctx.Repo.GitRepo.GetBranches(0, 0)
                                        if err != nil {
                                                ctx.ServerError("GetBranches", err)
                                                return
index 25438530f5530090f9924e29d3899cc082150b07..58781eb1c71a17d54b7b17eb66a85b25d6231e85 100644 (file)
@@ -78,16 +78,17 @@ func (repo *Repository) GetBranch(branch string) (*Branch, error) {
 }
 
 // GetBranchesByPath returns a branch by it's path
-func GetBranchesByPath(path string) ([]*Branch, error) {
+// if limit = 0 it will not limit
+func GetBranchesByPath(path string, skip, limit int) ([]*Branch, int, error) {
        gitRepo, err := OpenRepository(path)
        if err != nil {
-               return nil, err
+               return nil, 0, err
        }
        defer gitRepo.Close()
 
-       brs, err := gitRepo.GetBranches()
+       brs, countAll, err := gitRepo.GetBranches(skip, limit)
        if err != nil {
-               return nil, err
+               return nil, 0, err
        }
 
        branches := make([]*Branch, len(brs))
@@ -99,7 +100,7 @@ func GetBranchesByPath(path string) ([]*Branch, error) {
                }
        }
 
-       return branches, nil
+       return branches, countAll, nil
 }
 
 // DeleteBranchOptions Option(s) for delete branch
index 65cb77a8b55fff50e589b643f6771b0dfe99d893..b00253f6ffd668ec045a952b12cb0daa480e13c5 100644 (file)
@@ -25,21 +25,32 @@ func (repo *Repository) IsBranchExist(name string) bool {
        return reference.Type() != plumbing.InvalidReference
 }
 
-// GetBranches returns all branches of the repository.
-func (repo *Repository) GetBranches() ([]string, error) {
+// GetBranches returns branches from the repository, skipping skip initial branches and
+// returning at most limit branches, or all branches if limit is 0.
+func (repo *Repository) GetBranches(skip, limit int) ([]string, int, error) {
        var branchNames []string
 
        branches, err := repo.gogitRepo.Branches()
        if err != nil {
-               return nil, err
+               return nil, 0, err
        }
 
+       i := 0
+       count := 0
        _ = branches.ForEach(func(branch *plumbing.Reference) error {
+               count++
+               if i < skip {
+                       i++
+                       return nil
+               } else if limit != 0 && count > skip+limit {
+                       return nil
+               }
+
                branchNames = append(branchNames, strings.TrimPrefix(branch.Name().String(), BranchPrefix))
                return nil
        })
 
        // TODO: Sort?
 
-       return branchNames, nil
+       return branchNames, count, nil
 }
index 5ec46d725e4b430eb7ecd3d25d2ee22de9fe55b6..0628a572859c10a568587caf77715b41b75b6d51 100644 (file)
@@ -21,14 +21,14 @@ func (repo *Repository) IsBranchExist(name string) bool {
        return IsReferenceExist(repo.Path, BranchPrefix+name)
 }
 
-// GetBranches returns all branches of the repository.
-func (repo *Repository) GetBranches() ([]string, error) {
-       return callShowRef(repo.Path, BranchPrefix, "--heads")
+// GetBranches returns branches from the repository, skipping skip initial branches and
+// returning at most limit branches, or all branches if limit is 0.
+func (repo *Repository) GetBranches(skip, limit int) ([]string, int, error) {
+       return callShowRef(repo.Path, BranchPrefix, "--heads", skip, limit)
 }
 
-func callShowRef(repoPath, prefix, arg string) ([]string, error) {
-       var branchNames []string
-
+// callShowRef return refs, if limit = 0 it will not limit
+func callShowRef(repoPath, prefix, arg string, skip, limit int) (branchNames []string, countAll int, err error) {
        stdoutReader, stdoutWriter := io.Pipe()
        defer func() {
                _ = stdoutReader.Close()
@@ -49,8 +49,21 @@ func callShowRef(repoPath, prefix, arg string) ([]string, error) {
                }
        }()
 
+       i := 0
        bufReader := bufio.NewReader(stdoutReader)
-       for {
+       for i < skip {
+               _, isPrefix, err := bufReader.ReadLine()
+               if err == io.EOF {
+                       return branchNames, i, nil
+               }
+               if err != nil {
+                       return nil, 0, err
+               }
+               if !isPrefix {
+                       i++
+               }
+       }
+       for limit == 0 || i < skip+limit {
                // The output of show-ref is simply a list:
                // <sha> SP <ref> LF
                _, err := bufReader.ReadSlice(' ')
@@ -59,24 +72,39 @@ func callShowRef(repoPath, prefix, arg string) ([]string, error) {
                        _, err = bufReader.ReadSlice(' ')
                }
                if err == io.EOF {
-                       return branchNames, nil
+                       return branchNames, i, nil
                }
                if err != nil {
-                       return nil, err
+                       return nil, 0, err
                }
 
                branchName, err := bufReader.ReadString('\n')
                if err == io.EOF {
                        // This shouldn't happen... but we'll tolerate it for the sake of peace
-                       return branchNames, nil
+                       return branchNames, i, nil
                }
                if err != nil {
-                       return nil, err
+                       return nil, i, err
                }
                branchName = strings.TrimPrefix(branchName, prefix)
                if len(branchName) > 0 {
                        branchName = branchName[:len(branchName)-1]
                }
                branchNames = append(branchNames, branchName)
+               i++
+       }
+       // count all refs
+       for limit != 0 {
+               _, isPrefix, err := bufReader.ReadLine()
+               if err == io.EOF {
+                       return branchNames, i, nil
+               }
+               if err != nil {
+                       return nil, 0, err
+               }
+               if !isPrefix {
+                       i++
+               }
        }
+       return branchNames, i, nil
 }
index 33d31aef686de12ca20a6ea1c8db7f060d37c197..05d5237e6a65b8f9818a996f7d6dc485bc052a50 100644 (file)
@@ -17,11 +17,26 @@ func TestRepository_GetBranches(t *testing.T) {
        assert.NoError(t, err)
        defer bareRepo1.Close()
 
-       branches, err := bareRepo1.GetBranches()
+       branches, countAll, err := bareRepo1.GetBranches(0, 2)
+
+       assert.NoError(t, err)
+       assert.Len(t, branches, 2)
+       assert.EqualValues(t, 3, countAll)
+       assert.ElementsMatch(t, []string{"branch1", "branch2"}, branches)
+
+       branches, countAll, err = bareRepo1.GetBranches(0, 0)
 
        assert.NoError(t, err)
        assert.Len(t, branches, 3)
+       assert.EqualValues(t, 3, countAll)
        assert.ElementsMatch(t, []string{"branch1", "branch2", "master"}, branches)
+
+       branches, countAll, err = bareRepo1.GetBranches(5, 1)
+
+       assert.NoError(t, err)
+       assert.Len(t, branches, 0)
+       assert.EqualValues(t, 3, countAll)
+       assert.ElementsMatch(t, []string{}, branches)
 }
 
 func BenchmarkRepository_GetBranches(b *testing.B) {
@@ -33,7 +48,7 @@ func BenchmarkRepository_GetBranches(b *testing.B) {
        defer bareRepo1.Close()
 
        for i := 0; i < b.N; i++ {
-               _, err := bareRepo1.GetBranches()
+               _, _, err := bareRepo1.GetBranches(0, 0)
                if err != nil {
                        b.Fatal(err)
                }
index 83cbc58e342eea028cf5608b92df7a633b41f035..b3fa5d6dc40774bd004cbad1ca69cf3884b62293 100644 (file)
@@ -13,6 +13,7 @@ func (repo *Repository) IsTagExist(name string) bool {
 }
 
 // GetTags returns all tags of the repository.
-func (repo *Repository) GetTags() ([]string, error) {
-       return callShowRef(repo.Path, TagPrefix, "--tags")
+func (repo *Repository) GetTags() (tags []string, err error) {
+       tags, _, err = callShowRef(repo.Path, TagPrefix, "--tags", 0, 0)
+       return
 }
index d369a200b05397a2b37ad6b3e0d0140200851dc7..275bae91e3f9b7795c2d5700603ce2c66fd370fb 100644 (file)
@@ -13,6 +13,9 @@ import (
 
 // GetBranch returns a branch by its name
 func GetBranch(repo *models.Repository, branch string) (*git.Branch, error) {
+       if len(branch) == 0 {
+               return nil, fmt.Errorf("GetBranch: empty string for branch")
+       }
        gitRepo, err := git.OpenRepository(repo.RepoPath())
        if err != nil {
                return nil, err
@@ -22,9 +25,10 @@ func GetBranch(repo *models.Repository, branch string) (*git.Branch, error) {
        return gitRepo.GetBranch(branch)
 }
 
-// GetBranches returns all the branches of a repository
-func GetBranches(repo *models.Repository) ([]*git.Branch, error) {
-       return git.GetBranchesByPath(repo.RepoPath())
+// GetBranches returns branches from the repository, skipping skip initial branches and
+// returning at most limit branches, or all branches if limit is 0.
+func GetBranches(repo *models.Repository, skip, limit int) ([]*git.Branch, int, error) {
+       return git.GetBranchesByPath(repo.RepoPath(), skip, limit)
 }
 
 // checkBranchName validates branch name with existing repository branches
@@ -35,7 +39,7 @@ func checkBranchName(repo *models.Repository, name string) error {
        }
        defer gitRepo.Close()
 
-       branches, err := GetBranches(repo)
+       branches, _, err := GetBranches(repo, 0, 0)
        if err != nil {
                return err
        }
index a100456e778e979b36e364ecddaedc7980d4ef3d..50cde4c0b9d17320252669338a851e3ce417bda1 100644 (file)
@@ -239,7 +239,7 @@ func adoptRepository(ctx models.DBContext, repoPath string, u *models.User, repo
 
                repo.DefaultBranch = strings.TrimPrefix(repo.DefaultBranch, git.BranchPrefix)
        }
-       branches, _ := gitRepo.GetBranches()
+       branches, _, _ := gitRepo.GetBranches(0, 0)
        found := false
        hasDefault := false
        hasMaster := false
diff --git a/modules/util/paginate.go b/modules/util/paginate.go
new file mode 100644 (file)
index 0000000..2baa716
--- /dev/null
@@ -0,0 +1,34 @@
+// Copyright 2021 The Gitea Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package util
+
+import "reflect"
+
+// PaginateSlice cut a slice as per pagination options
+// if page = 0 it do not paginate
+func PaginateSlice(list interface{}, page, pageSize int) interface{} {
+       if page <= 0 || pageSize <= 0 {
+               return list
+       }
+       if reflect.TypeOf(list).Kind() != reflect.Slice {
+               return list
+       }
+
+       listValue := reflect.ValueOf(list)
+
+       page--
+
+       if page*pageSize >= listValue.Len() {
+               return listValue.Slice(listValue.Len(), listValue.Len()).Interface()
+       }
+
+       listValue = listValue.Slice(page*pageSize, listValue.Len())
+
+       if listValue.Len() > pageSize {
+               return listValue.Slice(0, pageSize).Interface()
+       }
+
+       return listValue.Interface()
+}
diff --git a/modules/util/paginate_test.go b/modules/util/paginate_test.go
new file mode 100644 (file)
index 0000000..d962e04
--- /dev/null
@@ -0,0 +1,47 @@
+// Copyright 2021 The Gitea Authors. All rights reserved.
+// Use of this source code is governed by a MIT-style
+// license that can be found in the LICENSE file.
+
+package util
+
+import (
+       "testing"
+
+       "github.com/stretchr/testify/assert"
+)
+
+func TestPaginateSlice(t *testing.T) {
+       stringSlice := []string{"a", "b", "c", "d", "e"}
+       result, ok := PaginateSlice(stringSlice, 1, 2).([]string)
+       assert.True(t, ok)
+       assert.EqualValues(t, []string{"a", "b"}, result)
+
+       result, ok = PaginateSlice(stringSlice, 100, 2).([]string)
+       assert.True(t, ok)
+       assert.EqualValues(t, []string{}, result)
+
+       result, ok = PaginateSlice(stringSlice, 3, 2).([]string)
+       assert.True(t, ok)
+       assert.EqualValues(t, []string{"e"}, result)
+
+       result, ok = PaginateSlice(stringSlice, 1, 0).([]string)
+       assert.True(t, ok)
+       assert.EqualValues(t, []string{"a", "b", "c", "d", "e"}, result)
+
+       result, ok = PaginateSlice(stringSlice, 1, -1).([]string)
+       assert.True(t, ok)
+       assert.EqualValues(t, []string{"a", "b", "c", "d", "e"}, result)
+
+       type Test struct {
+               Val int
+       }
+
+       var testVar = []*Test{{Val: 2}, {Val: 3}, {Val: 4}}
+       testVar, ok = PaginateSlice(testVar, 1, 50).([]*Test)
+       assert.True(t, ok)
+       assert.EqualValues(t, []*Test{{Val: 2}, {Val: 3}, {Val: 4}}, testVar)
+
+       testVar, ok = PaginateSlice(testVar, 2, 2).([]*Test)
+       assert.True(t, ok)
+       assert.EqualValues(t, []*Test{{Val: 4}}, testVar)
+}
index 61e09e1126a0415cffa9cf6c094d5e00336669ee..e0f36aa1e657d013b451b37cb007d2ad236c2864 100644 (file)
@@ -13,6 +13,7 @@ import (
        "code.gitea.io/gitea/modules/context"
        "code.gitea.io/gitea/modules/convert"
        api "code.gitea.io/gitea/modules/structs"
+       "code.gitea.io/gitea/modules/util"
        "code.gitea.io/gitea/modules/web"
        "code.gitea.io/gitea/routers/api/v1/user"
        "code.gitea.io/gitea/routers/api/v1/utils"
@@ -28,9 +29,9 @@ func listUserOrgs(ctx *context.APIContext, u *models.User) {
                ctx.Error(http.StatusInternalServerError, "GetOrgsByUserID", err)
                return
        }
-       maxResults := len(orgs)
 
-       orgs = utils.PaginateUserSlice(orgs, listOptions.Page, listOptions.PageSize)
+       maxResults := len(orgs)
+       orgs, _ = util.PaginateSlice(orgs, listOptions.Page, listOptions.PageSize).([]*models.User)
 
        apiOrgs := make([]*api.Organization, len(orgs))
        for i := range orgs {
index 790464c8bc55aaa5dcda0ffd2cf3202c8a1c5ffa..451fdcf516f041044a149a3d8672242cdc0d8717 100644 (file)
@@ -17,6 +17,7 @@ import (
        repo_module "code.gitea.io/gitea/modules/repository"
        api "code.gitea.io/gitea/modules/structs"
        "code.gitea.io/gitea/modules/web"
+       "code.gitea.io/gitea/routers/api/v1/utils"
        pull_service "code.gitea.io/gitea/services/pull"
        repo_service "code.gitea.io/gitea/services/repository"
 )
@@ -284,11 +285,21 @@ func ListBranches(ctx *context.APIContext) {
        //   description: name of the repo
        //   type: string
        //   required: true
+       // - name: page
+       //   in: query
+       //   description: page number of results to return (1-based)
+       //   type: integer
+       // - name: limit
+       //   in: query
+       //   description: page size of results
+       //   type: integer
        // responses:
        //   "200":
        //     "$ref": "#/responses/BranchList"
 
-       branches, err := repo_module.GetBranches(ctx.Repo.Repository)
+       listOptions := utils.GetListOptions(ctx)
+       skip, _ := listOptions.GetStartEnd()
+       branches, totalNumOfBranches, err := repo_module.GetBranches(ctx.Repo.Repository, skip, listOptions.PageSize)
        if err != nil {
                ctx.Error(http.StatusInternalServerError, "GetBranches", err)
                return
@@ -313,6 +324,9 @@ func ListBranches(ctx *context.APIContext) {
                }
        }
 
+       ctx.SetLinkHeader(int(totalNumOfBranches), listOptions.PageSize)
+       ctx.Header().Set("X-Total-Count", fmt.Sprintf("%d", totalNumOfBranches))
+       ctx.Header().Set("Access-Control-Expose-Headers", "X-Total-Count, Link")
        ctx.JSON(http.StatusOK, &apiBranches)
 }
 
index 5732ea7f7d5001a5c9176abd2a1dd84ee22aed74..ad1a136db463af1449e9da9a82f68ee83f0a7296 100644 (file)
@@ -66,22 +66,3 @@ func GetListOptions(ctx *context.APIContext) models.ListOptions {
                PageSize: convert.ToCorrectPageSize(ctx.QueryInt("limit")),
        }
 }
-
-// PaginateUserSlice cut a slice of Users as per pagination options
-// TODO: make it generic
-func PaginateUserSlice(items []*models.User, page, pageSize int) []*models.User {
-       if page != 0 {
-               page--
-       }
-
-       if page*pageSize >= len(items) {
-               return items[len(items):]
-       }
-
-       items = items[page*pageSize:]
-
-       if len(items) > pageSize {
-               return items[:pageSize]
-       }
-       return items
-}
index 7d844abe5a0230adcb58a5051c9c0006b9b9eeb7..cf6abc08df52577cd8ef8c4eb11ed1667e0e7f02 100644 (file)
@@ -58,12 +58,14 @@ func Branches(ctx *context.Context) {
                page = 1
        }
 
-       pageSize := ctx.QueryInt("limit")
-       if pageSize <= 0 || pageSize > git.BranchesRangeSize {
-               pageSize = git.BranchesRangeSize
+       limit := ctx.QueryInt("limit")
+       if limit <= 0 || limit > git.BranchesRangeSize {
+               limit = git.BranchesRangeSize
        }
 
-       branches, branchesCount := loadBranches(ctx, page, pageSize)
+       skip := (page - 1) * limit
+       log.Debug("Branches: skip: %d limit: %d", skip, limit)
+       branches, branchesCount := loadBranches(ctx, skip, limit)
        if ctx.Written() {
                return
        }
@@ -80,6 +82,7 @@ func DeleteBranchPost(ctx *context.Context) {
        defer redirect(ctx)
        branchName := ctx.Query("name")
        if branchName == ctx.Repo.Repository.DefaultBranch {
+               log.Debug("DeleteBranch: Can't delete default branch '%s'", branchName)
                ctx.Flash.Error(ctx.Tr("repo.branch.default_deletion_failed", branchName))
                return
        }
@@ -92,16 +95,19 @@ func DeleteBranchPost(ctx *context.Context) {
        }
 
        if isProtected {
+               log.Debug("DeleteBranch: Can't delete protected branch '%s'", branchName)
                ctx.Flash.Error(ctx.Tr("repo.branch.protected_deletion_failed", branchName))
                return
        }
 
-       if !ctx.Repo.GitRepo.IsBranchExist(branchName) || branchName == ctx.Repo.Repository.DefaultBranch {
+       if !ctx.Repo.GitRepo.IsBranchExist(branchName) {
+               log.Debug("DeleteBranch: Can't delete non existing branch '%s'", branchName)
                ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", branchName))
                return
        }
 
        if err := deleteBranch(ctx, branchName); err != nil {
+               log.Error("DeleteBranch: %v", err)
                ctx.Flash.Error(ctx.Tr("repo.branch.deletion_failed", branchName))
                return
        }
@@ -129,10 +135,11 @@ func RestoreBranchPost(ctx *context.Context) {
                Env:    models.PushingEnvironment(ctx.User, ctx.Repo.Repository),
        }); err != nil {
                if strings.Contains(err.Error(), "already exists") {
+                       log.Debug("RestoreBranch: Can't restore branch '%s', since one with same name already exist", deletedBranch.Name)
                        ctx.Flash.Error(ctx.Tr("repo.branch.already_exists", deletedBranch.Name))
                        return
                }
-               log.Error("CreateBranch: %v", err)
+               log.Error("RestoreBranch: CreateBranch: %v", err)
                ctx.Flash.Error(ctx.Tr("repo.branch.restore_failed", deletedBranch.Name))
                return
        }
@@ -148,7 +155,7 @@ func RestoreBranchPost(ctx *context.Context) {
                        RepoUserName: ctx.Repo.Owner.Name,
                        RepoName:     ctx.Repo.Repository.Name,
                }); err != nil {
-               log.Error("Update: %v", err)
+               log.Error("RestoreBranch: Update: %v", err)
        }
 
        ctx.Flash.Success(ctx.Tr("repo.branch.restore_success", deletedBranch.Name))
@@ -196,16 +203,18 @@ func deleteBranch(ctx *context.Context, branchName string) error {
 }
 
 // loadBranches loads branches from the repository limited by page & pageSize.
-// NOTE: May write to context on error. page & pageSize must be > 0
-func loadBranches(ctx *context.Context, page, pageSize int) ([]*Branch, int) {
+// NOTE: May write to context on error.
+func loadBranches(ctx *context.Context, skip, limit int) ([]*Branch, int) {
        defaultBranch, err := repo_module.GetBranch(ctx.Repo.Repository, ctx.Repo.Repository.DefaultBranch)
        if err != nil {
+               log.Error("loadBranches: get default branch: %v", err)
                ctx.ServerError("GetDefaultBranch", err)
                return nil, 0
        }
 
-       rawBranches, err := repo_module.GetBranches(ctx.Repo.Repository)
+       rawBranches, totalNumOfBranches, err := repo_module.GetBranches(ctx.Repo.Repository, skip, limit)
        if err != nil {
+               log.Error("GetBranches: %v", err)
                ctx.ServerError("GetBranches", err)
                return nil, 0
        }
@@ -222,32 +231,23 @@ func loadBranches(ctx *context.Context, page, pageSize int) ([]*Branch, int) {
        repoIDToGitRepo := map[int64]*git.Repository{}
        repoIDToGitRepo[ctx.Repo.Repository.ID] = ctx.Repo.GitRepo
 
-       var totalNumOfBranches = len(rawBranches)
-       var startIndex = (page - 1) * pageSize
-       if startIndex > totalNumOfBranches {
-               startIndex = totalNumOfBranches - 1
-       }
-       var endIndex = startIndex + pageSize
-       if endIndex > totalNumOfBranches {
-               endIndex = totalNumOfBranches - 1
-       }
-
        var branches []*Branch
-       for i := startIndex; i < endIndex; i++ {
+       for i := range rawBranches {
+               if rawBranches[i].Name == defaultBranch.Name {
+                       // Skip default branch
+                       continue
+               }
+
                var branch = loadOneBranch(ctx, rawBranches[i], protectedBranches, repoIDToRepo, repoIDToGitRepo)
                if branch == nil {
                        return nil, 0
                }
 
-               if branch.Name == ctx.Repo.Repository.DefaultBranch {
-                       // Skip default branch
-                       continue
-               }
-
                branches = append(branches, branch)
        }
 
        // Always add the default branch
+       log.Debug("loadOneBranch: load default: '%s'", defaultBranch.Name)
        branches = append(branches, loadOneBranch(ctx, defaultBranch, protectedBranches, repoIDToRepo, repoIDToGitRepo))
 
        if ctx.Repo.CanWrite(models.UnitTypeCode) {
@@ -259,12 +259,13 @@ func loadBranches(ctx *context.Context, page, pageSize int) ([]*Branch, int) {
                branches = append(branches, deletedBranches...)
        }
 
-       return branches, len(rawBranches) - 1
+       return branches, totalNumOfBranches - 1
 }
 
 func loadOneBranch(ctx *context.Context, rawBranch *git.Branch, protectedBranches []*models.ProtectedBranch,
        repoIDToRepo map[int64]*models.Repository,
        repoIDToGitRepo map[int64]*git.Repository) *Branch {
+       log.Trace("loadOneBranch: '%s'", rawBranch.Name)
 
        commit, err := rawBranch.GetCommit()
        if err != nil {
index aa4b3191b2e63534069f5cbc87bb21d9e2500c7a..218f712469530e2f548cd7acdbd6c61739b0934e 100644 (file)
@@ -520,7 +520,7 @@ func getBranchesForRepo(user *models.User, repo *models.Repository) (bool, []str
        }
        defer gitRepo.Close()
 
-       branches, err := gitRepo.GetBranches()
+       branches, _, err := gitRepo.GetBranches(0, 0)
        if err != nil {
                return false, nil, err
        }
@@ -541,7 +541,7 @@ func CompareDiff(ctx *context.Context) {
        }
 
        if ctx.Data["PageIsComparePull"] == true {
-               headBranches, err := headGitRepo.GetBranches()
+               headBranches, _, err := headGitRepo.GetBranches(0, 0)
                if err != nil {
                        ctx.ServerError("GetBranches", err)
                        return
index fb7107451fed73eb7ade051ff8b6d56dc337a1ff..3bc20839aa44c564e5fd38c2be96a892ad2d94b9 100644 (file)
@@ -678,7 +678,7 @@ func RetrieveRepoMetas(ctx *context.Context, repo *models.Repository, isPull boo
                return nil
        }
 
-       brs, err := ctx.Repo.GitRepo.GetBranches()
+       brs, _, err := ctx.Repo.GitRepo.GetBranches(0, 0)
        if err != nil {
                ctx.ServerError("GetBranches", err)
                return nil
index 328e23ad2fc75de7c583da7239f8333b7408fb41..e4981b8c00e646556054c3cd77144848c6e0c51d 100644 (file)
@@ -301,7 +301,7 @@ func runSync(m *models.Mirror) ([]*mirrorSyncResult, bool) {
        }
 
        log.Trace("SyncMirrors [repo: %-v]: invalidating mirror branch caches...", m.Repo)
-       branches, err := repo_module.GetBranches(m.Repo)
+       branches, _, err := repo_module.GetBranches(m.Repo, 0, 0)
        if err != nil {
                log.Error("GetBranches: %v", err)
                return nil, false
index 92f1ff65fb2fd6289d7d2d727520e9778adf2d2c..4f742f5a1a6a8345b095de688347d859c8333116 100644 (file)
@@ -482,7 +482,7 @@ func CloseBranchPulls(doer *models.User, repoID int64, branch string) error {
 
 // CloseRepoBranchesPulls close all pull requests which head branches are in the given repository
 func CloseRepoBranchesPulls(doer *models.User, repo *models.Repository) error {
-       branches, err := git.GetBranchesByPath(repo.RepoPath())
+       branches, _, err := git.GetBranchesByPath(repo.RepoPath(), 0, 0)
        if err != nil {
                return err
        }
index 36c1c43a00775e23ddb01b90e5af6a9e25f2d3ef..fd760a28e6c4744d1934d1094c1cc00515f8fdf6 100644 (file)
             "name": "repo",
             "in": "path",
             "required": true
+          },
+          {
+            "type": "integer",
+            "description": "page number of results to return (1-based)",
+            "name": "page",
+            "in": "query"
+          },
+          {
+            "type": "integer",
+            "description": "page size of results",
+            "name": "limit",
+            "in": "query"
           }
         ],
         "responses": {