woodpecker/server/hooks.go

170 lines
4.1 KiB
Go
Raw Normal View History

2015-04-08 22:43:59 +00:00
package server
import (
"fmt"
2015-04-08 22:43:59 +00:00
"strings"
log "github.com/Sirupsen/logrus"
2015-04-08 22:43:59 +00:00
"github.com/drone/drone/common"
"github.com/drone/drone/common/httputil"
2015-04-30 17:39:16 +00:00
"github.com/drone/drone/parser"
2015-04-21 22:48:48 +00:00
"github.com/drone/drone/parser/inject"
"github.com/drone/drone/parser/matrix"
2015-04-24 21:25:03 +00:00
"github.com/drone/drone/queue"
2015-04-08 22:43:59 +00:00
"github.com/gin-gonic/gin"
)
// PostHook accepts a post-commit hook and parses the payload
// in order to trigger a build.
//
// GET /api/hook
//
func PostHook(c *gin.Context) {
remote := ToRemote(c)
store := ToDatastore(c)
2015-04-24 21:25:03 +00:00
queue_ := ToQueue(c)
2015-04-30 21:23:46 +00:00
sess := ToSession(c)
2015-04-08 22:43:59 +00:00
hook, err := remote.Hook(c.Request)
if err != nil {
log.Errorf("failure to parse hook. %s", err)
2015-04-08 22:43:59 +00:00
c.Fail(400, err)
return
}
if hook == nil {
c.Writer.WriteHeader(200)
return
}
if hook.Repo == nil {
log.Errorf("failure to ascertain repo from hook.")
2015-04-08 22:43:59 +00:00
c.Writer.WriteHeader(400)
return
}
2015-04-30 21:23:46 +00:00
// get the token and verify the hook is authorized
token := sess.GetLogin(c.Request)
if token == nil || token.Label != hook.Repo.FullName {
log.Errorf("invalid token sent with hook.")
c.AbortWithStatus(403)
return
}
2015-04-08 22:43:59 +00:00
// a build may be skipped if the text [CI SKIP]
// is found inside the commit message
if hook.Commit != nil && strings.Contains(hook.Commit.Message, "[CI SKIP]") {
log.Infof("ignoring hook. [ci skip] found for %s")
2015-04-08 22:43:59 +00:00
c.Writer.WriteHeader(204)
return
}
repo, err := store.RepoName(hook.Repo.Owner, hook.Repo.Name)
2015-04-08 22:43:59 +00:00
if err != nil {
log.Errorf("failure to find repo %s/%s from hook. %s", hook.Repo.Owner, hook.Repo.Name, err)
2015-04-08 22:43:59 +00:00
c.Fail(404, err)
return
}
switch {
case repo.UserID == 0:
log.Warnf("ignoring hook. repo %s has no owner.", repo.FullName)
c.Writer.WriteHeader(204)
return
case !repo.PostCommit && hook.Commit.PullRequest != "":
log.Infof("ignoring hook. repo %s is disabled.", repo.FullName)
c.Writer.WriteHeader(204)
return
case !repo.PullRequest && hook.Commit.PullRequest == "":
log.Warnf("ignoring hook. repo %s is disabled for pull requests.", repo.FullName)
2015-04-08 22:43:59 +00:00
c.Writer.WriteHeader(204)
return
}
user, err := store.User(repo.UserID)
2015-04-08 22:43:59 +00:00
if err != nil {
log.Errorf("failure to find repo owner %s. %s", repo.FullName, err)
2015-04-08 22:43:59 +00:00
c.Fail(500, err)
return
}
commit := hook.Commit
commit.State = common.StatePending
commit.RepoID = repo.ID
2015-04-08 22:43:59 +00:00
// featch the .drone.yml file from the database
raw, err := remote.Script(user, repo, commit)
2015-04-08 22:43:59 +00:00
if err != nil {
log.Errorf("failure to get .drone.yml for %s. %s", repo.FullName, err)
2015-04-08 22:43:59 +00:00
c.Fail(404, err)
return
}
2015-04-21 22:48:48 +00:00
// inject any private parameters into the .drone.yml
if repo.Params != nil && len(repo.Params) != 0 {
raw = []byte(inject.InjectSafe(string(raw), repo.Params))
2015-04-21 22:48:48 +00:00
}
axes, err := matrix.Parse(string(raw))
if err != nil {
log.Errorf("failure to calculate matrix for %s. %s", repo.FullName, err)
c.Fail(404, err)
return
}
if len(axes) == 0 {
axes = append(axes, matrix.Axis{})
}
for num, axis := range axes {
commit.Builds = append(commit.Builds, &common.Build{
CommitID: commit.ID,
Sequence: num + 1,
State: common.StatePending,
Environment: axis,
})
}
keys := &common.Keypair{
Public: repo.PublicKey,
Private: repo.PrivateKey,
2015-04-08 22:43:59 +00:00
}
2015-04-28 22:08:21 +00:00
netrc, err := remote.Netrc(user)
if err != nil {
log.Errorf("failure to generate netrc for %s. %s", repo.FullName, err)
2015-04-28 22:08:21 +00:00
c.Fail(500, err)
return
}
2015-04-08 22:43:59 +00:00
// verify the branches can be built vs skipped
2015-04-30 17:39:16 +00:00
when, _ := parser.ParseCondition(string(raw))
if commit.PullRequest != "" && when != nil && !when.MatchBranch(commit.Branch) {
log.Infof("ignoring hook. yaml file excludes repo and branch %s %s", repo.FullName, commit.Branch)
2015-04-30 17:39:16 +00:00
c.AbortWithStatus(200)
return
}
2015-04-08 22:43:59 +00:00
err = store.AddCommit(commit)
2015-04-24 21:25:03 +00:00
if err != nil {
log.Errorf("failure to save commit for %s. %s", repo.FullName, err)
2015-04-24 21:25:03 +00:00
c.Fail(500, err)
return
}
c.JSON(200, commit)
link := fmt.Sprintf(
"%s/%s/%d",
httputil.GetURL(c.Request),
repo.FullName,
commit.Sequence,
)
err = remote.Status(user, repo, commit, link)
if err != nil {
log.Errorf("error setting commit status for %s/%d", repo.FullName, commit.Sequence)
}
2015-04-24 21:25:03 +00:00
queue_.Publish(&queue.Work{
User: user,
Repo: repo,
Commit: commit,
Keys: keys,
Netrc: netrc,
Yaml: raw,
2015-04-24 21:25:03 +00:00
})
2015-04-08 22:43:59 +00:00
}