add tests for pulls commit status and some fixes

This commit is contained in:
Lunny Xiao 2017-09-18 22:01:23 +08:00
parent cd11672fd1
commit a01a37c4ad
No known key found for this signature in database
GPG Key ID: C3B7C91B632F738A
7 changed files with 320 additions and 63 deletions

View File

@ -150,6 +150,24 @@ func testEditFileToNewBranch(t *testing.T, session *TestSession, user, repo, bra
return resp
}
func testEditFileToNewBranchAndSendPull(t *testing.T, session *TestSession, user, repo, branch, targetBranch, filePath string) *TestResponse {
testEditFileToNewBranch(t, session, user, repo, branch, targetBranch, filePath)
url := path.Join(user, repo, "compare", branch+"..."+targetBranch)
req := NewRequest(t, "GET", url)
resp := session.MakeRequest(t, req, http.StatusOK)
htmlDoc := NewHTMLParser(t, resp.Body)
req = NewRequestWithValues(t, "POST", url,
map[string]string{
"_csrf": htmlDoc.GetCSRF(),
"title": "pull request from " + targetBranch,
},
)
return session.MakeRequest(t, req, http.StatusFound)
}
func TestEditFile(t *testing.T) {
prepareTestEnv(t)
session := loginUser(t, "user2")

View File

@ -0,0 +1,130 @@
// Copyright 2017 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 integrations
import (
"fmt"
"net/http"
"path"
"strings"
"testing"
"code.gitea.io/gitea/models"
api "code.gitea.io/sdk/gitea"
"github.com/PuerkitoBio/goquery"
"github.com/stretchr/testify/assert"
)
var (
statesIcons = map[models.CommitStatusState]string{
models.CommitStatusPending: "circle icon yellow",
models.CommitStatusSuccess: "check icon green",
models.CommitStatusError: "warning icon red",
models.CommitStatusFailure: "remove icon red",
models.CommitStatusWarning: "warning sign icon yellow",
}
)
func TestRepoPullsWithStatus(t *testing.T) {
prepareTestEnv(t)
session := loginUser(t, "user2")
var size = 5
// create some pulls
for i := 0; i < size; i++ {
testEditFileToNewBranchAndSendPull(t, session, "user2", "repo16", "master", fmt.Sprintf("test%d", i), "readme.md")
}
// look for repo's pulls page
req := NewRequest(t, "GET", "/user2/repo16/pulls")
resp := session.MakeRequest(t, req, http.StatusOK)
doc := NewHTMLParser(t, resp.Body)
var indexes = make([]string, 0, size)
doc.doc.Find("li.item").Each(func(idx int, s *goquery.Selection) {
indexes = append(indexes, strings.TrimLeft(s.Find("div").Eq(1).Text(), "#"))
})
indexes = indexes[:5]
var status = make([]models.CommitStatusState, len(indexes))
for i := 0; i < len(indexes); i++ {
switch i {
case 0:
status[i] = models.CommitStatusPending
case 1:
status[i] = models.CommitStatusSuccess
case 2:
status[i] = models.CommitStatusError
case 3:
status[i] = models.CommitStatusFailure
case 4:
status[i] = models.CommitStatusWarning
default:
status[i] = models.CommitStatusSuccess
}
}
for i, index := range indexes {
// Request repository commits page
req = NewRequestf(t, "GET", "/user2/repo16/pulls/%s/commits", index)
resp = session.MakeRequest(t, req, http.StatusOK)
doc = NewHTMLParser(t, resp.Body)
// Get first commit URL
commitURL, exists := doc.doc.Find("#commits-table tbody tr td.sha a").Last().Attr("href")
assert.True(t, exists)
assert.NotEmpty(t, commitURL)
commitID := path.Base(commitURL)
// Call API to add status for commit
req = NewRequestWithJSON(t, "POST", "/api/v1/repos/user2/repo16/statuses/"+commitID,
api.CreateStatusOption{
State: api.StatusState(status[i]),
TargetURL: "http://test.ci/",
Description: "",
Context: "testci",
},
)
session.MakeRequest(t, req, http.StatusCreated)
req = NewRequestf(t, "GET", "/user2/repo16/pulls/%s/commits", index)
resp = session.MakeRequest(t, req, http.StatusOK)
doc = NewHTMLParser(t, resp.Body)
commitURL, exists = doc.doc.Find("#commits-table tbody tr td.sha a").Last().Attr("href")
assert.True(t, exists)
assert.NotEmpty(t, commitURL)
assert.EqualValues(t, commitID, path.Base(commitURL))
cls, ok := doc.doc.Find("#commits-table tbody tr td.message i.commit-status").Last().Attr("class")
assert.True(t, ok)
assert.EqualValues(t, "commit-status "+statesIcons[status[i]], cls)
}
req = NewRequest(t, "GET", "/user2/repo16/pulls")
resp = session.MakeRequest(t, req, http.StatusOK)
doc = NewHTMLParser(t, resp.Body)
doc.doc.Find("li.item").Each(func(i int, s *goquery.Selection) {
cls, ok := s.Find("i.commit-status").Attr("class")
assert.True(t, ok)
assert.EqualValues(t, "commit-status "+statesIcons[status[i]], cls)
})
req = NewRequest(t, "GET", "/pulls?type=all&repo=16&sort=&state=open")
resp = session.MakeRequest(t, req, http.StatusOK)
doc = NewHTMLParser(t, resp.Body)
fmt.Println(string(resp.Body))
doc.doc.Find("li.item").Each(func(i int, s *goquery.Selection) {
cls, ok := s.Find("i.commit-status").Attr("class")
assert.True(t, ok)
assert.EqualValues(t, "commit-status "+statesIcons[status[i]], cls)
})
}

View File

@ -74,3 +74,43 @@
type: 1
config: "{}"
created_unix: 1524304355
-
id: 12
repo_id: 16
type: 1
index: 0
config: "{}"
created_unix: 946684810
-
id: 13
repo_id: 16
type: 2
index: 1
config: "{\"EnableTimetracker\":false,\"AllowOnlyContributorsToTrackTime\":false}"
created_unix: 946684810
-
id: 14
repo_id: 16
type: 3
index: 2
config: "{}"
created_unix: 946684810
-
id: 15
repo_id: 16
type: 4
index: 3
config: "{}"
created_unix: 946684810
-
id: 16
repo_id: 16
type: 5
index: 4
config: "{}"
created_unix: 946684810

View File

@ -168,6 +168,7 @@ func GetLatestCommitStatuses(repoIDs []int64, shas []string) ([][]*CommitStatus,
var results = make([]struct {
ID int64
RepoID int64
SHA string
}, 0, 10*len(repoIDs))
var cond = builder.NewCond()
@ -180,7 +181,7 @@ func GetLatestCommitStatuses(repoIDs []int64, shas []string) ([][]*CommitStatus,
err := x.Table(&CommitStatus{}).
Where(cond).
Select("max( id ) as id, repo_id").
Select("max( id ) as id, repo_id, sha").
GroupBy("repo_id, sha, context").OrderBy("max( id ) desc").Find(&results)
if err != nil {
return nil, err
@ -192,10 +193,10 @@ func GetLatestCommitStatuses(repoIDs []int64, shas []string) ([][]*CommitStatus,
}
var ids = make([]int64, 0, len(results))
var repoIDsMap = make(map[int64][]int64, len(repoIDs))
var repoIDsMap = make(map[string][]int64, len(repoIDs))
for _, res := range results {
ids = append(ids, res.ID)
repoIDsMap[res.RepoID] = append(repoIDsMap[res.RepoID], res.ID)
repoIDsMap[fmt.Sprintf("%d-%s", res.RepoID, res.SHA)] = append(repoIDsMap[fmt.Sprintf("%d-%s", res.RepoID, res.SHA)], res.ID)
}
statuses := make(map[int64]*CommitStatus, len(ids))
@ -205,7 +206,7 @@ func GetLatestCommitStatuses(repoIDs []int64, shas []string) ([][]*CommitStatus,
}
for i := 0; i < len(repoIDs); i++ {
for _, id := range repoIDsMap[repoIDs[i]] {
for _, id := range repoIDsMap[fmt.Sprintf("%d-%s", repoIDs[i], shas[i])] {
returns[i] = append(returns[i], statuses[id])
}
}

View File

@ -209,44 +209,80 @@ func Issues(ctx *context.Context) {
}
}
var repoIDs = make([]int64, 0, len(issues))
var shas = make([]string, 0, len(issues))
var pullIDs = make([]int64, 0, len(issues))
// Get posters.
for i, issue := range issues {
// Check read status
if !ctx.IsSigned {
issues[i].IsRead = true
} else if err = issues[i].GetIsRead(ctx.User.ID); err != nil {
ctx.ServerError("GetIsRead", err)
return
if !isPullList {
// Get posters.
for i := 0; i < len(issues); i++ {
// Check read status
if !ctx.IsSigned {
issues[i].IsRead = true
} else if err = issues[i].GetIsRead(ctx.User.ID); err != nil {
ctx.ServerError("GetIsRead", err)
return
}
}
} else {
var repoIDs = make([]int64, 0, len(issues))
var shas = make([]string, 0, len(issues))
var pullIDs = make([]int64, 0, len(issues))
var repoCache = make(map[int64]*git.Repository)
// Get posters.
for i, issue := range issues {
// Check read status
if !ctx.IsSigned {
issues[i].IsRead = true
} else if err = issues[i].GetIsRead(ctx.User.ID); err != nil {
ctx.ServerError("GetIsRead", err)
return
}
if issue.IsPull {
if err := issue.LoadAttributes(); err != nil {
ctx.ServerError("LoadAttributes", err)
return
}
repoIDs = append(repoIDs, ctx.Repo.Repository.ID)
shas = append(shas, issue.PullRequest.MergeBase)
pullIDs = append(pullIDs, issue.ID)
var rep *git.Repository
var ok bool
if rep, ok = repoCache[issue.PullRequest.HeadRepoID]; !ok {
if err := issue.PullRequest.GetHeadRepo(); err != nil {
ctx.ServerError("GetHeadRepo", err)
return
}
rep, err = git.OpenRepository(issue.PullRequest.HeadRepo.RepoPath())
if err != nil {
ctx.ServerError("OpenRepository", err)
return
}
repoCache[issue.PullRequest.HeadRepoID] = rep
}
sha, err := rep.GetBranchCommitID(issue.PullRequest.HeadBranch)
if err != nil {
log.Error(4, "GetBranchCommitID: %v", err)
} else {
repoIDs = append(repoIDs, issue.RepoID)
shas = append(shas, sha)
pullIDs = append(pullIDs, issue.ID)
}
}
}
commitStatuses, err := models.GetLatestCommitStatuses(repoIDs, shas)
if err != nil {
ctx.ServerError("GetLatestCommitStatuses", err)
return
}
var issuesStates = make(map[int64]*models.CommitStatus, len(issues))
if len(repoIDs) > 0 {
commitStatuses, err := models.GetLatestCommitStatuses(repoIDs, shas)
if err != nil {
ctx.ServerError("GetLatestCommitStatuses", err)
return
}
var issuesStates = make(map[int64]*models.CommitStatus, len(issues))
for i, statuses := range commitStatuses {
issuesStates[pullIDs[i]] = models.CalcCommitStatus(statuses)
for i, statuses := range commitStatuses {
issuesStates[pullIDs[i]] = models.CalcCommitStatus(statuses)
}
}
ctx.Data["IssuesStates"] = issuesStates
}
ctx.Data["Issues"] = issues
ctx.Data["IssuesStates"] = issuesStates
// Get milestones.
ctx.Data["Milestones"], err = models.GetMilestonesByRepoID(repo.ID)

View File

@ -9,9 +9,11 @@ import (
"fmt"
"sort"
"code.gitea.io/git"
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/context"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/util"
@ -291,37 +293,65 @@ func Issues(ctx *context.Context) {
return
}
var repoIDs = make([]int64, 0, len(issues))
var shas = make([]string, 0, len(issues))
var pullIDs = make([]int64, 0, len(issues))
for _, issue := range issues {
issue.Repo = showReposMap[issue.RepoID]
if !isPullList {
for _, issue := range issues {
issue.Repo = showReposMap[issue.RepoID]
}
} else {
var repoIDs = make([]int64, 0, len(issues))
var shas = make([]string, 0, len(issues))
var pullIDs = make([]int64, 0, len(issues))
var repoCache = make(map[int64]*git.Repository)
for _, issue := range issues {
issue.Repo = showReposMap[issue.RepoID]
if issue.IsPull {
if err := issue.LoadAttributes(); err != nil {
ctx.ServerError("LoadAttributes", fmt.Errorf("%v", err))
return
}
repoIDs = append(repoIDs, issue.Repo.ID)
shas = append(shas, issue.PullRequest.MergeBase)
pullIDs = append(pullIDs, issue.ID)
var rep *git.Repository
var ok bool
if rep, ok = repoCache[issue.PullRequest.HeadRepoID]; !ok {
if err := issue.PullRequest.GetHeadRepo(); err != nil {
ctx.ServerError("GetHeadRepo", err)
return
}
rep, err = git.OpenRepository(issue.PullRequest.HeadRepo.RepoPath())
if err != nil {
ctx.ServerError("OpenRepository", err)
return
}
repoCache[issue.PullRequest.HeadRepoID] = rep
}
sha, err := rep.GetBranchCommitID(issue.PullRequest.HeadBranch)
if err != nil {
log.Error(4, "GetBranchCommitID: %v", err)
} else {
repoIDs = append(repoIDs, issue.RepoID)
shas = append(shas, sha)
pullIDs = append(pullIDs, issue.ID)
}
}
}
commitStatuses, err := models.GetLatestCommitStatuses(repoIDs, shas)
if err != nil {
ctx.ServerError("GetLatestCommitStatuses", err)
return
}
var issuesStates = make(map[int64]*models.CommitStatus, len(issues))
if len(repoIDs) > 0 {
commitStatuses, err := models.GetLatestCommitStatuses(repoIDs, shas)
if err != nil {
ctx.ServerError("GetLatestCommitStatuses", err)
return
}
var issuesStates = make(map[int64]*models.CommitStatus, len(issues))
for i, statuses := range commitStatuses {
issuesStates[pullIDs[i]] = models.CalcCommitStatus(statuses)
for i, statuses := range commitStatuses {
issuesStates[pullIDs[i]] = models.CalcCommitStatus(statuses)
}
}
ctx.Data["IssuesStates"] = issuesStates
}
ctx.Data["IssuesStates"] = issuesStates
issueStats, err := models.GetUserIssueStats(models.UserIssueStatsOptions{
UserID: ctxUser.ID,
RepoID: repoID,

View File

@ -1,15 +1,17 @@
{{if eq .State "pending"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status circle icon yellow"></i></a>
{{end}}
{{if eq .State "success"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status check icon green"></i></a>
{{end}}
{{if eq .State "error"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status warning icon red"></i></a>
{{end}}
{{if eq .State "failure"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status remove icon red"></i></a>
{{end}}
{{if eq .State "warning"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status warning sign icon yellow"></i></a>
{{if .}}
{{if eq .State "pending"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status circle icon yellow"></i></a>
{{end}}
{{if eq .State "success"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status check icon green"></i></a>
{{end}}
{{if eq .State "error"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status warning icon red"></i></a>
{{end}}
{{if eq .State "failure"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status remove icon red"></i></a>
{{end}}
{{if eq .State "warning"}}
<a href="{{.TargetURL}}" target=_blank><i class="commit-status warning sign icon yellow"></i></a>
{{end}}
{{end}}