Skip to content

Commit

Permalink
Support changed files for Gitea PRs (#1342)
Browse files Browse the repository at this point in the history
- add tests to fetch changed files
- ignore error if gitea version is to low
- adjust docs accordingly

Co-authored-by: 6543 <6543@obermui.de>
Co-authored-by: Lauris BH <lauris@nix.lv>
  • Loading branch information
3 people authored Oct 28, 2022
1 parent 36e4291 commit 8f183c8
Show file tree
Hide file tree
Showing 10 changed files with 142 additions and 12 deletions.
5 changes: 5 additions & 0 deletions .golangci.yml
Original file line number Diff line number Diff line change
Expand Up @@ -37,3 +37,8 @@ issues:
linters:
- deadcode
- unused
# gin force us to use string as context key
- path: server/store/context.go
linters:
- staticcheck
- revive
3 changes: 1 addition & 2 deletions docs/docs/20-usage/20-pipeline-syntax.md
Original file line number Diff line number Diff line change
Expand Up @@ -412,8 +412,7 @@ when:

:::info
Path conditions are applied only to **push** and **pull_request** events.
It is currently **only available** for GitHub, GitLab.
Gitea only supports **push** at the moment ([go-gitea/gitea#18228](https://github.com/go-gitea/gitea/pull/18228)).
It is currently **only available** for GitHub, GitLab and Gitea (version 1.18.0 and newer)
:::

Execute a step only on a pipeline with certain files being changed:
Expand Down
2 changes: 1 addition & 1 deletion docs/docs/30-administration/11-forges/10-overview.md
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,4 @@
| [Multi pipeline](../../20-usage/25-multi-pipeline.md) | :white_check_mark: | :white_check_mark: | :white_check_mark: | :x: | :x: | :x: | :x: |
| [when.path filter](../../20-usage/20-pipeline-syntax.md#path) | :white_check_mark: | :white_check_mark:¹ | :white_check_mark: | :x: | :x: | :x: | :x: |

¹) [except for pull requests](https://github.com/woodpecker-ci/woodpecker/issues/754)
¹) for Gitea versions 1.17 or lower not for pull requests
2 changes: 1 addition & 1 deletion go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ module github.com/woodpecker-ci/woodpecker
go 1.18

require (
code.gitea.io/sdk/gitea v0.15.1-0.20220831004139-a0127ed0e7fe
code.gitea.io/sdk/gitea v0.15.1-0.20221016183512-2d9ee57af1e0
codeberg.org/6543/go-yaml2json v0.3.0
github.com/antonmedv/expr v1.9.0
github.com/bmatcuk/doublestar/v4 v4.2.0
Expand Down
4 changes: 2 additions & 2 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,8 @@ cloud.google.com/go/storage v1.5.0/go.mod h1:tpKbwo567HUNpVclU5sGELwQWBDZ8gh0Zeo
cloud.google.com/go/storage v1.6.0/go.mod h1:N7U0C8pVQ/+NIKOBQyamJIeKQKkZ+mxpohlUTyfDhBk=
cloud.google.com/go/storage v1.8.0/go.mod h1:Wv1Oy7z6Yz3DshWRJFhqM/UCfaWIRTdp0RXyy7KQOVs=
cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9ullr3+Kg0=
code.gitea.io/sdk/gitea v0.15.1-0.20220831004139-a0127ed0e7fe h1:PeLyxnUZE85QuJtBZ4P8qCQcgWG5Ked67mlNgr0WkCQ=
code.gitea.io/sdk/gitea v0.15.1-0.20220831004139-a0127ed0e7fe/go.mod h1:aRmrQC3CAHdJAU1LQt0C9zqzqI8tUB/5oQtNE746aYE=
code.gitea.io/sdk/gitea v0.15.1-0.20221016183512-2d9ee57af1e0 h1:AKpsCoOtVrWWBtANM9319pwCB5ihx1Sdvr1HSbAwr54=
code.gitea.io/sdk/gitea v0.15.1-0.20221016183512-2d9ee57af1e0/go.mod h1:ndkDk99BnfiUCCYEUhpNzi0lpmApXlwRFqClBlOlEBg=
codeberg.org/6543/go-yaml2json v0.3.0 h1:BlvjmY0Gous8P+rr8aBdgPYnIfUAqFepF8q7Tp0R5t8=
codeberg.org/6543/go-yaml2json v0.3.0/go.mod h1:mz61q14LWF4ZABrgMEDMmk3t9dPi6zgR1uBh2VKV2RQ=
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
Expand Down
36 changes: 32 additions & 4 deletions server/remote/gitea/fixtures/handler.go
Original file line number Diff line number Diff line change
Expand Up @@ -28,11 +28,12 @@ func Handler() http.Handler {
e := gin.New()
e.GET("/api/v1/repos/:owner/:name", getRepo)
e.GET("/api/v1/repositories/:id", getRepoByID)
e.GET("/api/v1/repos/:owner/:name/raw/:commit/:file", getRepoFile)
e.GET("/api/v1/repos/:owner/:name/raw/:file", getRepoFile)
e.POST("/api/v1/repos/:owner/:name/hooks", createRepoHook)
e.GET("/api/v1/repos/:owner/:name/hooks", listRepoHooks)
e.DELETE("/api/v1/repos/:owner/:name/hooks/:id", deleteRepoHook)
e.POST("/api/v1/repos/:owner/:name/statuses/:commit", createRepoCommitStatus)
e.GET("/api/v1/repos/:owner/:name/pulls/:index/files", getPRFiles)
e.GET("/api/v1/user/repos", getUserRepos)
e.GET("/api/v1/version", getVersion)

Expand Down Expand Up @@ -69,10 +70,13 @@ func createRepoCommitStatus(c *gin.Context) {
}

func getRepoFile(c *gin.Context) {
if c.Param("file") == "file_not_found" {
file := c.Param("file")
ref := c.Query("ref")

if file == "file_not_found" {
c.String(404, "")
}
if c.Param("commit") == "v1.0.0" || c.Param("commit") == "9ecad50" {
if ref == "v1.0.0" || ref == "9ecad50" {
c.String(200, repoFilePayload)
}
c.String(404, "")
Expand Down Expand Up @@ -116,7 +120,16 @@ func getUserRepos(c *gin.Context) {
}

func getVersion(c *gin.Context) {
c.JSON(200, map[string]interface{}{"version": "1.12"})
c.JSON(200, map[string]interface{}{"version": "1.18.0"})
}

func getPRFiles(c *gin.Context) {
page := c.Query("page")
if page == "1" {
c.String(200, prFilesPayload)
} else {
c.String(200, "[]")
}
}

const listRepoHookPayloads = `
Expand Down Expand Up @@ -175,3 +188,18 @@ const userRepoPayload = `
}
]
`

const prFilesPayload = `
[
{
"filename": "README.md",
"status": "changed",
"additions": 2,
"deletions": 0,
"changes": 2,
"html_url": "http://localhost/username/repo/src/commit/e79e4b0e8d9dd6f72b70e776c3317db7c19ca0fd/README.md",
"contents_url": "http://localhost:3000/api/v1/repos/username/repo/contents/README.md?ref=e79e4b0e8d9dd6f72b70e776c3317db7c19ca0fd",
"raw_url": "http://localhost/username/repo/raw/commit/e79e4b0e8d9dd6f72b70e776c3317db7c19ca0fd/README.md"
}
]
`
62 changes: 61 additions & 1 deletion server/remote/gitea/gitea.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ import (
"github.com/woodpecker-ci/woodpecker/server/model"
"github.com/woodpecker-ci/woodpecker/server/remote"
"github.com/woodpecker-ci/woodpecker/server/remote/common"
"github.com/woodpecker-ci/woodpecker/server/store"
)

const (
Expand Down Expand Up @@ -475,7 +476,23 @@ func (c *Gitea) BranchHead(ctx context.Context, u *model.User, r *model.Repo, br
// Hook parses the incoming Gitea hook and returns the Repository and Pipeline
// details. If the hook is unsupported nil values are returned.
func (c *Gitea) Hook(ctx context.Context, r *http.Request) (*model.Repo, *model.Pipeline, error) {
return parseHook(r)
repo, pipeline, err := parseHook(r)
if err != nil {
return nil, nil, err
}

if pipeline.Event == model.EventPull && len(pipeline.ChangedFiles) == 0 {
index, err := strconv.ParseInt(strings.Split(pipeline.Ref, "/")[2], 10, 64)
if err != nil {
return nil, nil, err
}
pipeline.ChangedFiles, err = c.getChangedFilesForPR(ctx, repo, index)
if err != nil {
log.Error().Err(err).Msgf("could not get changed files for PR %s#%d", repo.FullName, index)
}
}

return repo, pipeline, nil
}

// OrgMembership returns if user is member of organization and if user
Expand Down Expand Up @@ -540,3 +557,46 @@ func getStatus(status model.StatusValue) gitea.StatusState {
return gitea.StatusFailure
}
}

func (c *Gitea) getChangedFilesForPR(ctx context.Context, repo *model.Repo, index int64) ([]string, error) {
_store, ok := store.TryFromContext(ctx)
if !ok {
log.Error().Msg("could not get store from context")
return []string{}, nil
}

repo, err := _store.GetRepoNameFallback(repo.RemoteID, repo.FullName)
if err != nil {
return nil, err
}

user, err := _store.GetUser(repo.UserID)
if err != nil {
return nil, err
}

client, err := c.newClientToken(ctx, user.Token)
if err != nil {
return nil, err
}

if client.CheckServerVersionConstraint("1.18.0") != nil {
// version too low
log.Debug().Msg("Gitea version does not support getting changed files for PRs")
return []string{}, nil
}

return common.Paginate(func(page int) ([]string, error) {
giteaFiles, _, err := client.ListPullRequestFiles(repo.Owner, repo.Name, index,
gitea.ListPullRequestFilesOptions{ListOptions: gitea.ListOptions{Page: page}})
if err != nil {
return nil, err
}

var files []string
for _, file := range giteaFiles {
files = append(files, file.Filename)
}
return files, nil
})
}
25 changes: 24 additions & 1 deletion server/remote/gitea/gitea_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,15 +16,21 @@
package gitea

import (
"bytes"
"context"
"net/http"
"net/http/httptest"
"testing"

"github.com/franela/goblin"
"github.com/gin-gonic/gin"
"github.com/stretchr/testify/mock"
"github.com/woodpecker-ci/woodpecker/shared/utils"

"github.com/woodpecker-ci/woodpecker/server/model"
"github.com/woodpecker-ci/woodpecker/server/remote/gitea/fixtures"
"github.com/woodpecker-ci/woodpecker/server/store"
mocks_store "github.com/woodpecker-ci/woodpecker/server/store/mocks"
)

func Test_gitea(t *testing.T) {
Expand All @@ -36,7 +42,9 @@ func Test_gitea(t *testing.T) {
SkipVerify: true,
})

ctx := context.Background()
mockStore := mocks_store.NewStore(t)
ctx := store.InjectToContext(context.Background(), mockStore)

g := goblin.Goblin(t)
g.Describe("Gitea", func() {
g.After(func() {
Expand Down Expand Up @@ -154,6 +162,21 @@ func Test_gitea(t *testing.T) {
g.It("Should return push details")
g.It("Should handle a parsing error")
})

g.It("Given a PR hook", func() {
buf := bytes.NewBufferString(fixtures.HookPullRequest)
req, _ := http.NewRequest("POST", "/hook", buf)
req.Header = http.Header{}
req.Header.Set(hookEvent, hookPullRequest)
mockStore.On("GetRepoNameFallback", mock.Anything, mock.Anything).Return(fakeRepo, nil)
mockStore.On("GetUser", mock.Anything).Return(fakeUser, nil)
r, b, err := c.Hook(ctx, req)
g.Assert(r).IsNotNil()
g.Assert(b).IsNotNil()
g.Assert(err).IsNil()
g.Assert(b.Event).Equal(model.EventPull)
g.Assert(utils.EqualStringSlice(b.ChangedFiles, []string{"README.md"})).IsTrue()
})
})
}

Expand Down
11 changes: 11 additions & 0 deletions server/remote/gitea/parse_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,17 @@ func Test_parser(t *testing.T) {
g.Assert(b).IsNil()
g.Assert(err).IsNil()
})
g.It("given a PR hook", func() {
buf := bytes.NewBufferString(fixtures.HookPullRequest)
req, _ := http.NewRequest("POST", "/hook", buf)
req.Header = http.Header{}
req.Header.Set(hookEvent, hookPullRequest)
r, b, err := parseHook(req)
g.Assert(r).IsNotNil()
g.Assert(b).IsNotNil()
g.Assert(err).IsNil()
g.Assert(b.Event).Equal(model.EventPull)
})
g.Describe("given a push hook", func() {
g.It("should extract repository and pipeline details", func() {
buf := bytes.NewBufferString(fixtures.HookPush)
Expand Down
4 changes: 4 additions & 0 deletions server/store/context.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,3 +41,7 @@ func TryFromContext(c context.Context) (Store, bool) {
func ToContext(c Setter, store Store) {
c.Set(key, store)
}

func InjectToContext(ctx context.Context, store Store) context.Context {
return context.WithValue(ctx, key, store)
}

0 comments on commit 8f183c8

Please sign in to comment.