trigger build from branch or sha, closed #2679

This commit is contained in:
Brad Rydzewski 2019-04-24 14:05:47 -07:00
parent cce10d5c47
commit 1c6d751d50
4 changed files with 190 additions and 48 deletions

View file

@ -7,6 +7,10 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## Unreleased ## Unreleased
### Added ### Added
- endpoint to trigger new build for default branch, by [@bradrydzewski](https://github.com/bradrydzewski). [#2679](https://github.com/drone/drone/issues/2679).
- endpoint to trigger new build for branch, by [@bradrydzewski](https://github.com/bradrydzewski). [#2679](https://github.com/drone/drone/issues/2679).
- endpoint to trigger new build for branch and sha, by [@bradrydzewski](https://github.com/bradrydzewski). [#2679](https://github.com/drone/drone/issues/2679).
## [1.1.0] - 2019-04-23 ## [1.1.0] - 2019-04-23
### Added ### Added

View file

@ -169,9 +169,9 @@ func (s Server) Handler() http.Handler {
).Post("/repair", repos.HandleRepair(s.Hooks, s.Repoz, s.Repos, s.Users, s.System.Link)) ).Post("/repair", repos.HandleRepair(s.Hooks, s.Repoz, s.Repos, s.Users, s.System.Link))
r.Route("/builds", func(r chi.Router) { r.Route("/builds", func(r chi.Router) {
r.Get("/", builds.HandleList(s.Repos, s.Builds)) r.With(acl.CheckWriteAccess()).Get("/", builds.HandleList(s.Repos, s.Builds))
// TODO(bradrydzewski) temporarily disabled until we finalize the endpoint. r.With(acl.CheckWriteAccess()).Post("/", builds.HandleCreate(s.Repos, s.Commits, s.Triggerer))
// r.Post("/", builds.HandleCreate(s.Repos, s.Commits, s.Triggerer))
r.Get("/latest", builds.HandleLast(s.Repos, s.Builds, s.Stages)) r.Get("/latest", builds.HandleLast(s.Repos, s.Builds, s.Stages))
r.Get("/{number}", builds.HandleFind(s.Repos, s.Builds, s.Stages)) r.Get("/{number}", builds.HandleFind(s.Repos, s.Builds, s.Stages))
r.Get("/{number}/logs/{stage}/{step}", logs.HandleFind(s.Repos, s.Builds, s.Stages, s.Steps, s.Logs)) r.Get("/{number}/logs/{stage}/{step}", logs.HandleFind(s.Repos, s.Builds, s.Stages, s.Steps, s.Logs))

View file

@ -15,7 +15,6 @@
package builds package builds
import ( import (
"encoding/json"
"net/http" "net/http"
"github.com/drone/drone/core" "github.com/drone/drone/core"
@ -26,13 +25,6 @@ import (
"github.com/go-chi/chi" "github.com/go-chi/chi"
) )
type createBuild struct {
Params map[string]string `json:"params"`
Commit *string `json:"commit"`
Branch *string `json:"branch"`
Ref *string `json:"ref"`
}
// HandleCreate returns an http.HandlerFunc that processes http // HandleCreate returns an http.HandlerFunc that processes http
// requests to create a build for the specified commit. // requests to create a build for the specified commit.
func HandleCreate( func HandleCreate(
@ -45,36 +37,29 @@ func HandleCreate(
ctx = r.Context() ctx = r.Context()
namespace = chi.URLParam(r, "owner") namespace = chi.URLParam(r, "owner")
name = chi.URLParam(r, "name") name = chi.URLParam(r, "name")
sha = r.FormValue("commit")
branch = r.FormValue("branch")
user, _ = request.UserFrom(ctx) user, _ = request.UserFrom(ctx)
) )
in := new(createBuild)
err := json.NewDecoder(r.Body).Decode(in)
if err != nil {
render.BadRequest(w, err)
return
}
if in.Ref == nil || in.Branch == nil {
render.BadRequestf(w, "Missing branch or ref")
return
}
if in.Params == nil {
// cannot remember if parameters must be non-nil,
// so we set the value to prevent a possible
// downstream nil pointer. just being overly cautious ...
in.Params = map[string]string{}
}
repo, err := repos.FindName(ctx, namespace, name) repo, err := repos.FindName(ctx, namespace, name)
if err != nil { if err != nil {
render.NotFound(w, err) render.NotFound(w, err)
return return
} }
// if the user does not provide a branch, assume the
// default repository branch.
if branch == "" {
branch = repo.Branch
}
// expand the branch to a git reference.
ref := scm.ExpandRef(branch, "refs/heads")
var commit *core.Commit var commit *core.Commit
if in.Commit == nil { if sha != "" {
commit, err = commits.Find(ctx, user, repo.Slug, *in.Commit) commit, err = commits.Find(ctx, user, repo.Slug, sha)
} else if in.Ref != nil { } else {
commit, err = commits.FindRef(ctx, user, repo.Slug, *in.Ref)
} else if in.Branch != nil {
ref := scm.ExpandRef(*in.Branch, "refs/heads")
commit, err = commits.FindRef(ctx, user, repo.Slug, ref) commit, err = commits.FindRef(ctx, user, repo.Slug, ref)
} }
if err != nil { if err != nil {
@ -89,26 +74,17 @@ func HandleCreate(
Timestamp: commit.Author.Date, Timestamp: commit.Author.Date,
Title: "", // we expect this to be empty. Title: "", // we expect this to be empty.
Message: commit.Message, Message: commit.Message,
Before: "", // we expect this to be empty. Before: commit.Sha,
After: commit.Sha, After: commit.Sha,
Ref: "", // set below Ref: ref,
Source: "", // set below Source: branch,
Target: "", // set below Target: branch,
Author: commit.Author.Login, Author: commit.Author.Login,
AuthorName: commit.Author.Name, AuthorName: commit.Author.Name,
AuthorEmail: commit.Author.Email, AuthorEmail: commit.Author.Email,
AuthorAvatar: commit.Author.Avatar, AuthorAvatar: commit.Author.Avatar,
Sender: "", // todo: what value should we use? Sender: user.Login,
Params: in.Params, Params: map[string]string{}, // todo: popular from query parameters
}
if in.Branch != nil {
hook.Source = *in.Branch
hook.Target = *in.Branch
}
if in.Ref != nil {
branch := scm.TrimRef(*in.Ref)
hook.Source = branch
hook.Target = branch
} }
result, err := triggerer.Trigger(r.Context(), repo, hook) result, err := triggerer.Trigger(r.Context(), repo, hook)

View file

@ -3,3 +3,165 @@
// that can be found in the LICENSE file. // that can be found in the LICENSE file.
package builds package builds
import (
"context"
"encoding/json"
"net/http/httptest"
"net/url"
"testing"
"github.com/drone/drone/core"
"github.com/drone/drone/handler/api/request"
"github.com/drone/drone/mock"
"github.com/go-chi/chi"
"github.com/golang/mock/gomock"
"github.com/google/go-cmp/cmp"
)
func TestCreate(t *testing.T) {
controller := gomock.NewController(t)
defer controller.Finish()
mockCommit := &core.Commit{
Sha: "cce10d5c4760d1d6ede99db850ab7e77efe15579",
Ref: "refs/heads/master",
Message: "updated README.md",
Link: "https://github.com/octocatl/hello-world/commit/cce10d5c4760d1d6ede99db850ab7e77efe15579",
Author: &core.Committer{
Name: "The Octocat",
Email: "octocat@github.com",
Login: "octocat",
Avatar: "https://github.com/octocat.png",
},
}
checkBuild := func(_ context.Context, _ *core.Repository, hook *core.Hook) error {
if got, want := hook.Trigger, mockUser.Login; got != want {
t.Errorf("Want hook Trigger By %s, got %s", want, got)
}
if got, want := hook.Event, core.EventPush; got != want {
t.Errorf("Want hook Event %s, got %s", want, got)
}
if got, want := hook.Link, mockCommit.Link; got != want {
t.Errorf("Want hook Link %s, got %s", want, got)
}
if got, want := hook.Message, mockCommit.Message; got != want {
t.Errorf("Want hook Message %s, got %s", want, got)
}
if got, want := hook.Before, mockCommit.Sha; got != want {
t.Errorf("Want hook Before %s, got %s", want, got)
}
if got, want := hook.After, mockCommit.Sha; got != want {
t.Errorf("Want hook After %s, got %s", want, got)
}
if got, want := hook.Ref, mockCommit.Ref; got != want {
t.Errorf("Want hook Ref %s, got %s", want, got)
}
if got, want := hook.Source, "master"; got != want {
t.Errorf("Want hook Source %s, got %s", want, got)
}
if got, want := hook.Target, "master"; got != want {
t.Errorf("Want hook Target %s, got %s", want, got)
}
if got, want := hook.Author, mockCommit.Author.Login; got != want {
t.Errorf("Want hook Author %s, got %s", want, got)
}
if got, want := hook.AuthorName, mockCommit.Author.Name; got != want {
t.Errorf("Want hook AuthorName %s, got %s", want, got)
}
if got, want := hook.AuthorEmail, mockCommit.Author.Email; got != want {
t.Errorf("Want hook AuthorEmail %s, got %s", want, got)
}
if got, want := hook.AuthorAvatar, mockCommit.Author.Avatar; got != want {
t.Errorf("Want hook AuthorAvatar %s, got %s", want, got)
}
if got, want := hook.Sender, mockUser.Login; got != want {
t.Errorf("Want hook Sender %s, got %s", want, got)
}
return nil
}
repos := mock.NewMockRepositoryStore(controller)
repos.EXPECT().FindName(gomock.Any(), gomock.Any(), mockRepo.Name).Return(mockRepo, nil)
commits := mock.NewMockCommitService(controller)
commits.EXPECT().Find(gomock.Any(), mockUser, mockRepo.Slug, mockCommit.Sha).Return(mockCommit, nil)
triggerer := mock.NewMockTriggerer(controller)
triggerer.EXPECT().Trigger(gomock.Any(), mockRepo, gomock.Any()).Return(mockBuild, nil).Do(checkBuild)
c := new(chi.Context)
c.URLParams.Add("owner", "octocat")
c.URLParams.Add("name", "hello-world")
params := &url.Values{}
params.Set("branch", "master")
params.Set("commit", mockCommit.Sha)
w := httptest.NewRecorder()
r := httptest.NewRequest("POST", "/?"+params.Encode(), nil)
r = r.WithContext(
context.WithValue(request.WithUser(r.Context(), mockUser), chi.RouteCtxKey, c),
)
HandleCreate(repos, commits, triggerer)(w, r)
if got, want := w.Code, 200; want != got {
t.Errorf("Want response code %d, got %d", want, got)
}
got, want := new(core.Build), mockBuild
json.NewDecoder(w.Body).Decode(got)
if diff := cmp.Diff(got, want); len(diff) != 0 {
t.Errorf(diff)
}
}
func TestCreate_FromHead(t *testing.T) {
controller := gomock.NewController(t)
defer controller.Finish()
mockCommit := &core.Commit{
Sha: "cce10d5c4760d1d6ede99db850ab7e77efe15579",
Ref: "refs/heads/master",
Message: "updated README.md",
Link: "https://github.com/octocatl/hello-world/commit/cce10d5c4760d1d6ede99db850ab7e77efe15579",
Author: &core.Committer{
Name: "The Octocat",
Email: "octocat@github.com",
Login: "octocat",
Avatar: "https://github.com/octocat.png",
},
}
repos := mock.NewMockRepositoryStore(controller)
repos.EXPECT().FindName(gomock.Any(), gomock.Any(), mockRepo.Name).Return(mockRepo, nil)
commits := mock.NewMockCommitService(controller)
commits.EXPECT().FindRef(gomock.Any(), mockUser, mockRepo.Slug, mockCommit.Ref).Return(mockCommit, nil)
triggerer := mock.NewMockTriggerer(controller)
triggerer.EXPECT().Trigger(gomock.Any(), mockRepo, gomock.Any()).Return(mockBuild, nil)
c := new(chi.Context)
c.URLParams.Add("owner", "octocat")
c.URLParams.Add("name", "hello-world")
w := httptest.NewRecorder()
r := httptest.NewRequest("POST", "/", nil)
r = r.WithContext(
context.WithValue(request.WithUser(r.Context(), mockUser), chi.RouteCtxKey, c),
)
HandleCreate(repos, commits, triggerer)(w, r)
if got, want := w.Code, 200; want != got {
t.Errorf("Want response code %d, got %d", want, got)
}
got, want := new(core.Build), mockBuild
json.NewDecoder(w.Body).Decode(got)
if diff := cmp.Diff(got, want); len(diff) != 0 {
t.Errorf(diff)
}
}