mirror of
https://github.com/woodpecker-ci/woodpecker.git
synced 2024-12-30 10:11:23 +02:00
parent
7646e9e5d3
commit
0c6c0dfbbc
@ -132,8 +132,11 @@ func setupHandlers() {
|
|||||||
queueRunner := queue.NewBuildRunner(docker.New(), timeout)
|
queueRunner := queue.NewBuildRunner(docker.New(), timeout)
|
||||||
queue := queue.Start(workers, queueRunner)
|
queue := queue.Start(workers, queueRunner)
|
||||||
|
|
||||||
hookHandler := handler.NewHookHandler(queue)
|
var (
|
||||||
gitlab := handler.NewGitlabHandler(queue)
|
github = handler.NewGithubHandler(queue)
|
||||||
|
gitlab = handler.NewGitlabHandler(queue)
|
||||||
|
bitbucket = handler.NewBitbucketHandler(queue)
|
||||||
|
)
|
||||||
|
|
||||||
m := pat.New()
|
m := pat.New()
|
||||||
m.Get("/login", handler.ErrorHandler(handler.Login))
|
m.Get("/login", handler.ErrorHandler(handler.Login))
|
||||||
@ -209,10 +212,10 @@ func setupHandlers() {
|
|||||||
m.Get("/account/admin/users", handler.AdminHandler(handler.AdminUserList))
|
m.Get("/account/admin/users", handler.AdminHandler(handler.AdminUserList))
|
||||||
|
|
||||||
// handlers for GitHub post-commit hooks
|
// handlers for GitHub post-commit hooks
|
||||||
m.Post("/hook/github.com", handler.ErrorHandler(hookHandler.HookGithub))
|
m.Post("/hook/github.com", handler.ErrorHandler(github.Hook))
|
||||||
|
|
||||||
// handlers for Bitbucket post-commit hooks
|
// handlers for Bitbucket post-commit hooks
|
||||||
m.Post("/hook/bitbucket.org", handler.ErrorHandler(hookHandler.HookBitbucket))
|
m.Post("/hook/bitbucket.org", handler.ErrorHandler(bitbucket.Hook))
|
||||||
|
|
||||||
// handlers for GitLab post-commit hooks
|
// handlers for GitLab post-commit hooks
|
||||||
m.Post("/hook/gitlab", handler.ErrorHandler(gitlab.Hook))
|
m.Post("/hook/gitlab", handler.ErrorHandler(gitlab.Hook))
|
||||||
|
116
pkg/handler/bitbucket.go
Normal file
116
pkg/handler/bitbucket.go
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
package handler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/drone/drone/pkg/build/script"
|
||||||
|
"github.com/drone/drone/pkg/database"
|
||||||
|
. "github.com/drone/drone/pkg/model"
|
||||||
|
"github.com/drone/drone/pkg/queue"
|
||||||
|
"github.com/drone/go-bitbucket/bitbucket"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BitbucketHandler struct {
|
||||||
|
queue *queue.Queue
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewBitbucketHandler(queue *queue.Queue) *BitbucketHandler {
|
||||||
|
return &BitbucketHandler{
|
||||||
|
queue: queue,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Processes a generic POST-RECEIVE Bitbucket hook and
|
||||||
|
// attempts to trigger a build.
|
||||||
|
func (h *BitbucketHandler) Hook(w http.ResponseWriter, r *http.Request) error {
|
||||||
|
// get the payload from the request
|
||||||
|
payload := r.FormValue("payload")
|
||||||
|
|
||||||
|
// parse the post-commit hook
|
||||||
|
hook, err := bitbucket.ParseHook([]byte(payload))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// get the repo from the URL
|
||||||
|
repoId := r.FormValue("id")
|
||||||
|
|
||||||
|
// get the repo from the database, return error if not found
|
||||||
|
repo, err := database.GetRepoSlug(repoId)
|
||||||
|
if err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the user that owns the repository
|
||||||
|
user, err := database.GetUser(repo.UserID)
|
||||||
|
if err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify that the commit doesn't already exist.
|
||||||
|
// We should never build the same commit twice.
|
||||||
|
_, err = database.GetCommitHash(hook.Commits[len(hook.Commits)-1].Hash, repo.ID)
|
||||||
|
if err != nil && err != sql.ErrNoRows {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusBadGateway), http.StatusBadGateway)
|
||||||
|
}
|
||||||
|
|
||||||
|
commit := &Commit{}
|
||||||
|
commit.RepoID = repo.ID
|
||||||
|
commit.Branch = hook.Commits[len(hook.Commits)-1].Branch
|
||||||
|
commit.Hash = hook.Commits[len(hook.Commits)-1].Hash
|
||||||
|
commit.Status = "Pending"
|
||||||
|
commit.Created = time.Now().UTC()
|
||||||
|
commit.Message = hook.Commits[len(hook.Commits)-1].Message
|
||||||
|
commit.Timestamp = time.Now().UTC().String()
|
||||||
|
commit.SetAuthor(hook.Commits[len(hook.Commits)-1].Author)
|
||||||
|
|
||||||
|
// get the github settings from the database
|
||||||
|
settings := database.SettingsMust()
|
||||||
|
|
||||||
|
// create the Bitbucket client
|
||||||
|
client := bitbucket.New(
|
||||||
|
settings.BitbucketKey,
|
||||||
|
settings.BitbucketSecret,
|
||||||
|
user.BitbucketToken,
|
||||||
|
user.BitbucketSecret,
|
||||||
|
)
|
||||||
|
|
||||||
|
// get the yaml from the database
|
||||||
|
raw, err := client.Sources.Find(repo.Owner, repo.Name, commit.Hash, ".drone.yml")
|
||||||
|
if err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
||||||
|
}
|
||||||
|
|
||||||
|
// parse the build script
|
||||||
|
buildscript, err := script.ParseBuild([]byte(raw.Data), repo.Params)
|
||||||
|
if err != nil {
|
||||||
|
msg := "Could not parse your .drone.yml file. It needs to be a valid drone yaml file.\n\n" + err.Error() + "\n"
|
||||||
|
if err := saveFailedBuild(commit, msg); err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
return RenderText(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
||||||
|
}
|
||||||
|
|
||||||
|
// save the commit to the database
|
||||||
|
if err := database.SaveCommit(commit); err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
|
||||||
|
// save the build to the database
|
||||||
|
build := &Build{}
|
||||||
|
build.Slug = "1" // TODO
|
||||||
|
build.CommitID = commit.ID
|
||||||
|
build.Created = time.Now().UTC()
|
||||||
|
build.Status = "Pending"
|
||||||
|
if err := database.SaveBuild(build); err != nil {
|
||||||
|
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
||||||
|
}
|
||||||
|
|
||||||
|
// send the build to the queue
|
||||||
|
h.queue.Add(&queue.BuildTask{Repo: repo, Commit: commit, Build: build, Script: buildscript})
|
||||||
|
|
||||||
|
// OK!
|
||||||
|
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
||||||
|
}
|
@ -3,6 +3,7 @@ package handler
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/drone/drone/pkg/channel"
|
"github.com/drone/drone/pkg/channel"
|
||||||
"github.com/drone/drone/pkg/database"
|
"github.com/drone/drone/pkg/database"
|
||||||
@ -59,3 +60,37 @@ func CommitShow(w http.ResponseWriter, r *http.Request, u *User, repo *Repo) err
|
|||||||
// render the repository template.
|
// render the repository template.
|
||||||
return RenderTemplate(w, "repo_commit.html", &data)
|
return RenderTemplate(w, "repo_commit.html", &data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Helper method for saving a failed build or commit in the case where it never starts to build.
|
||||||
|
// This can happen if the yaml is bad or doesn't exist.
|
||||||
|
func saveFailedBuild(commit *Commit, msg string) error {
|
||||||
|
|
||||||
|
// Set the commit to failed
|
||||||
|
commit.Status = "Failure"
|
||||||
|
commit.Created = time.Now().UTC()
|
||||||
|
commit.Finished = commit.Created
|
||||||
|
commit.Duration = 0
|
||||||
|
if err := database.SaveCommit(commit); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// save the build to the database
|
||||||
|
build := &Build{}
|
||||||
|
build.Slug = "1" // TODO: This should not be hardcoded
|
||||||
|
build.CommitID = commit.ID
|
||||||
|
build.Created = time.Now().UTC()
|
||||||
|
build.Finished = build.Created
|
||||||
|
commit.Duration = 0
|
||||||
|
build.Status = "Failure"
|
||||||
|
build.Stdout = msg
|
||||||
|
if err := database.SaveBuild(build); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Should the status be Error instead of Failure?
|
||||||
|
|
||||||
|
// TODO: Do we need to update the branch table too?
|
||||||
|
|
||||||
|
return nil
|
||||||
|
|
||||||
|
}
|
||||||
|
@ -10,23 +10,22 @@ import (
|
|||||||
"github.com/drone/drone/pkg/database"
|
"github.com/drone/drone/pkg/database"
|
||||||
. "github.com/drone/drone/pkg/model"
|
. "github.com/drone/drone/pkg/model"
|
||||||
"github.com/drone/drone/pkg/queue"
|
"github.com/drone/drone/pkg/queue"
|
||||||
"github.com/drone/go-bitbucket/bitbucket"
|
|
||||||
"github.com/drone/go-github/github"
|
"github.com/drone/go-github/github"
|
||||||
)
|
)
|
||||||
|
|
||||||
type HookHandler struct {
|
type GithubHandler struct {
|
||||||
queue *queue.Queue
|
queue *queue.Queue
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHookHandler(queue *queue.Queue) *HookHandler {
|
func NewGithubHandler(queue *queue.Queue) *GithubHandler {
|
||||||
return &HookHandler{
|
return &GithubHandler{
|
||||||
queue: queue,
|
queue: queue,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Processes a generic POST-RECEIVE GitHub hook and
|
// Processes a generic POST-RECEIVE GitHub hook and
|
||||||
// attempts to trigger a build.
|
// attempts to trigger a build.
|
||||||
func (h *HookHandler) HookGithub(w http.ResponseWriter, r *http.Request) error {
|
func (h *GithubHandler) Hook(w http.ResponseWriter, r *http.Request) error {
|
||||||
// handle github ping
|
// handle github ping
|
||||||
if r.Header.Get("X-Github-Event") == "ping" {
|
if r.Header.Get("X-Github-Event") == "ping" {
|
||||||
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
||||||
@ -35,7 +34,7 @@ func (h *HookHandler) HookGithub(w http.ResponseWriter, r *http.Request) error {
|
|||||||
// if this is a pull request route
|
// if this is a pull request route
|
||||||
// to a different handler
|
// to a different handler
|
||||||
if r.Header.Get("X-Github-Event") == "pull_request" {
|
if r.Header.Get("X-Github-Event") == "pull_request" {
|
||||||
h.PullRequestHookGithub(w, r)
|
h.PullRequestHook(w, r)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -176,7 +175,7 @@ func (h *HookHandler) HookGithub(w http.ResponseWriter, r *http.Request) error {
|
|||||||
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (h *HookHandler) PullRequestHookGithub(w http.ResponseWriter, r *http.Request) {
|
func (h *GithubHandler) PullRequestHook(w http.ResponseWriter, r *http.Request) {
|
||||||
// get the payload of the message
|
// get the payload of the message
|
||||||
// this should contain a json representation of the
|
// this should contain a json representation of the
|
||||||
// repository and commit details
|
// repository and commit details
|
||||||
@ -291,130 +290,3 @@ func (h *HookHandler) PullRequestHookGithub(w http.ResponseWriter, r *http.Reque
|
|||||||
// OK!
|
// OK!
|
||||||
RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Processes a generic POST-RECEIVE Bitbucket hook and
|
|
||||||
// attempts to trigger a build.
|
|
||||||
func (h *HookHandler) HookBitbucket(w http.ResponseWriter, r *http.Request) error {
|
|
||||||
// get the payload from the request
|
|
||||||
payload := r.FormValue("payload")
|
|
||||||
|
|
||||||
// parse the post-commit hook
|
|
||||||
hook, err := bitbucket.ParseHook([]byte(payload))
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// get the repo from the URL
|
|
||||||
repoId := r.FormValue("id")
|
|
||||||
|
|
||||||
// get the repo from the database, return error if not found
|
|
||||||
repo, err := database.GetRepoSlug(repoId)
|
|
||||||
if err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the user that owns the repository
|
|
||||||
user, err := database.GetUser(repo.UserID)
|
|
||||||
if err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Verify that the commit doesn't already exist.
|
|
||||||
// We should never build the same commit twice.
|
|
||||||
_, err = database.GetCommitHash(hook.Commits[len(hook.Commits)-1].Hash, repo.ID)
|
|
||||||
if err != nil && err != sql.ErrNoRows {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusBadGateway), http.StatusBadGateway)
|
|
||||||
}
|
|
||||||
|
|
||||||
commit := &Commit{}
|
|
||||||
commit.RepoID = repo.ID
|
|
||||||
commit.Branch = hook.Commits[len(hook.Commits)-1].Branch
|
|
||||||
commit.Hash = hook.Commits[len(hook.Commits)-1].Hash
|
|
||||||
commit.Status = "Pending"
|
|
||||||
commit.Created = time.Now().UTC()
|
|
||||||
commit.Message = hook.Commits[len(hook.Commits)-1].Message
|
|
||||||
commit.Timestamp = time.Now().UTC().String()
|
|
||||||
commit.SetAuthor(hook.Commits[len(hook.Commits)-1].Author)
|
|
||||||
|
|
||||||
// get the github settings from the database
|
|
||||||
settings := database.SettingsMust()
|
|
||||||
|
|
||||||
// create the Bitbucket client
|
|
||||||
client := bitbucket.New(
|
|
||||||
settings.BitbucketKey,
|
|
||||||
settings.BitbucketSecret,
|
|
||||||
user.BitbucketToken,
|
|
||||||
user.BitbucketSecret,
|
|
||||||
)
|
|
||||||
|
|
||||||
// get the yaml from the database
|
|
||||||
raw, err := client.Sources.Find(repo.Owner, repo.Name, commit.Hash, ".drone.yml")
|
|
||||||
if err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusNotFound), http.StatusNotFound)
|
|
||||||
}
|
|
||||||
|
|
||||||
// parse the build script
|
|
||||||
buildscript, err := script.ParseBuild([]byte(raw.Data), repo.Params)
|
|
||||||
if err != nil {
|
|
||||||
msg := "Could not parse your .drone.yml file. It needs to be a valid drone yaml file.\n\n" + err.Error() + "\n"
|
|
||||||
if err := saveFailedBuild(commit, msg); err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
|
||||||
}
|
|
||||||
return RenderText(w, http.StatusText(http.StatusBadRequest), http.StatusBadRequest)
|
|
||||||
}
|
|
||||||
|
|
||||||
// save the commit to the database
|
|
||||||
if err := database.SaveCommit(commit); err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
|
||||||
}
|
|
||||||
|
|
||||||
// save the build to the database
|
|
||||||
build := &Build{}
|
|
||||||
build.Slug = "1" // TODO
|
|
||||||
build.CommitID = commit.ID
|
|
||||||
build.Created = time.Now().UTC()
|
|
||||||
build.Status = "Pending"
|
|
||||||
if err := database.SaveBuild(build); err != nil {
|
|
||||||
return RenderText(w, http.StatusText(http.StatusInternalServerError), http.StatusInternalServerError)
|
|
||||||
}
|
|
||||||
|
|
||||||
// send the build to the queue
|
|
||||||
h.queue.Add(&queue.BuildTask{Repo: repo, Commit: commit, Build: build, Script: buildscript})
|
|
||||||
|
|
||||||
// OK!
|
|
||||||
return RenderText(w, http.StatusText(http.StatusOK), http.StatusOK)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Helper method for saving a failed build or commit in the case where it never starts to build.
|
|
||||||
// This can happen if the yaml is bad or doesn't exist.
|
|
||||||
func saveFailedBuild(commit *Commit, msg string) error {
|
|
||||||
|
|
||||||
// Set the commit to failed
|
|
||||||
commit.Status = "Failure"
|
|
||||||
commit.Created = time.Now().UTC()
|
|
||||||
commit.Finished = commit.Created
|
|
||||||
commit.Duration = 0
|
|
||||||
if err := database.SaveCommit(commit); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// save the build to the database
|
|
||||||
build := &Build{}
|
|
||||||
build.Slug = "1" // TODO: This should not be hardcoded
|
|
||||||
build.CommitID = commit.ID
|
|
||||||
build.Created = time.Now().UTC()
|
|
||||||
build.Finished = build.Created
|
|
||||||
commit.Duration = 0
|
|
||||||
build.Status = "Failure"
|
|
||||||
build.Stdout = msg
|
|
||||||
if err := database.SaveBuild(build); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Should the status be Error instead of Failure?
|
|
||||||
|
|
||||||
// TODO: Do we need to update the branch table too?
|
|
||||||
|
|
||||||
return nil
|
|
||||||
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user