2015-05-11 07:45:31 +00:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"strconv"
|
|
|
|
"time"
|
|
|
|
|
2015-05-22 18:37:40 +00:00
|
|
|
"github.com/drone/drone/Godeps/_workspace/src/github.com/gin-gonic/gin"
|
2015-05-17 18:42:56 +00:00
|
|
|
"github.com/drone/drone/pkg/queue"
|
2015-05-17 20:51:42 +00:00
|
|
|
common "github.com/drone/drone/pkg/types"
|
2015-05-17 18:36:12 +00:00
|
|
|
"github.com/drone/drone/pkg/yaml/inject"
|
2015-08-19 06:59:30 +00:00
|
|
|
"github.com/drone/drone/pkg/yaml/secure"
|
2015-05-11 07:45:31 +00:00
|
|
|
// "github.com/gin-gonic/gin/binding"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetCommit accepts a request to retrieve a commit
|
|
|
|
// from the datastore for the given repository and
|
|
|
|
// commit sequence.
|
|
|
|
//
|
|
|
|
// GET /api/repos/:owner/:name/:number
|
|
|
|
//
|
2015-08-29 22:18:45 +00:00
|
|
|
func GetBuild(c *gin.Context) {
|
2015-05-11 07:45:31 +00:00
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
return
|
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build, err := store.BuildNumber(repo, num)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
2015-08-03 06:02:44 +00:00
|
|
|
return
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build.Jobs, err = store.JobList(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
2015-06-23 03:45:08 +00:00
|
|
|
c.JSON(200, build)
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-29 22:18:45 +00:00
|
|
|
// GetCommits accepts a request to retrieve a list
|
|
|
|
// of commits from the datastore for the given repository.
|
|
|
|
//
|
|
|
|
// GET /api/repos/:owner/:name/builds
|
|
|
|
//
|
|
|
|
func GetBuilds(c *gin.Context) {
|
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
builds, err := store.BuildList(repo, 20, 0)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
|
|
|
c.JSON(200, builds)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-22 03:32:22 +00:00
|
|
|
// GetPullRequest accepts a requests to retvie a pull request
|
|
|
|
// from the datastore for the given repository and
|
|
|
|
// pull request number
|
|
|
|
//
|
2015-08-29 22:18:45 +00:00
|
|
|
// GET /api/repos/:owner/:name/pulls/:number
|
2015-08-22 03:32:22 +00:00
|
|
|
//
|
2015-08-29 22:18:45 +00:00
|
|
|
// REASON: It required by GitLab, becuase we get only
|
|
|
|
// sha and ref name, but drone uses build numbers
|
2015-08-22 03:32:22 +00:00
|
|
|
func GetPullRequest(c *gin.Context) {
|
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
|
|
|
|
// get the token and verify the hook is authorized
|
|
|
|
if c.Request.FormValue("access_token") != hash(repo.FullName, repo.Hash) {
|
|
|
|
c.AbortWithStatus(403)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
build, err := store.BuildPullRequestNumber(repo, num)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
build.Jobs, err = store.JobList(build)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
|
|
|
c.JSON(200, build)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-29 22:18:45 +00:00
|
|
|
// GetCommit accepts a requests to retvie a sha and branch
|
|
|
|
// from the datastore for the given repository and
|
|
|
|
// pull request number
|
2015-05-11 07:45:31 +00:00
|
|
|
//
|
2015-08-29 22:18:45 +00:00
|
|
|
// GET /api/repos/:owner/:name/commits/:sha
|
2015-05-11 07:45:31 +00:00
|
|
|
//
|
2015-08-29 22:18:45 +00:00
|
|
|
// REASON: It required by GitLab, becuase we get only
|
|
|
|
// sha and ref name, but drone uses build numbers
|
|
|
|
func GetCommit(c *gin.Context) {
|
|
|
|
var branch string
|
|
|
|
|
2015-05-11 07:45:31 +00:00
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
2015-08-29 22:18:45 +00:00
|
|
|
sha := c.Params.ByName("sha")
|
|
|
|
|
|
|
|
// get the token and verify the hook is authorized
|
|
|
|
if c.Request.FormValue("access_token") != hash(repo.FullName, repo.Hash) {
|
|
|
|
c.AbortWithStatus(403)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
branch = c.Request.FormValue("branch")
|
|
|
|
if branch == "" {
|
|
|
|
branch = repo.Branch
|
|
|
|
}
|
|
|
|
|
|
|
|
build, err := store.BuildSha(repo, sha, branch)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
build.Jobs, err = store.JobList(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
2015-08-29 22:18:45 +00:00
|
|
|
c.JSON(200, build)
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
2015-08-29 22:18:45 +00:00
|
|
|
|
|
|
|
return
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetLogs accepts a request to retrieve logs from the
|
|
|
|
// datastore for the given repository, build and task
|
|
|
|
// number.
|
|
|
|
//
|
|
|
|
// GET /api/repos/:owner/:name/logs/:number/:task
|
|
|
|
//
|
|
|
|
func GetLogs(c *gin.Context) {
|
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
full, _ := strconv.ParseBool(c.Params.ByName("full"))
|
2015-06-23 03:45:08 +00:00
|
|
|
build, _ := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
job, _ := strconv.Atoi(c.Params.ByName("task"))
|
2015-05-11 07:45:31 +00:00
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
path := fmt.Sprintf("/logs/%s/%v/%v", repo.FullName, build, job)
|
2015-05-11 07:45:31 +00:00
|
|
|
r, err := store.GetBlobReader(path)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
2015-08-03 06:02:44 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer r.Close()
|
|
|
|
if full {
|
2015-05-11 07:45:31 +00:00
|
|
|
io.Copy(c.Writer, r)
|
|
|
|
} else {
|
|
|
|
io.Copy(c.Writer, io.LimitReader(r, 2000000))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// // PostBuildStatus accepts a request to create a new build
|
|
|
|
// // status. The created user status is returned in JSON
|
|
|
|
// // format if successful.
|
|
|
|
// //
|
|
|
|
// // POST /api/repos/:owner/:name/status/:number
|
|
|
|
// //
|
|
|
|
// func PostBuildStatus(c *gin.Context) {
|
|
|
|
// store := ToDatastore(c)
|
|
|
|
// repo := ToRepo(c)
|
|
|
|
// num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
// if err != nil {
|
|
|
|
// c.Fail(400, err)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// in := &common.Status{}
|
|
|
|
// if !c.BindWith(in, binding.JSON) {
|
|
|
|
// c.AbortWithStatus(400)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// if err := store.SetBuildStatus(repo.Name, num, in); err != nil {
|
|
|
|
// c.Fail(400, err)
|
|
|
|
// } else {
|
|
|
|
// c.JSON(201, in)
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// RunBuild accepts a request to restart an existing build.
|
|
|
|
//
|
|
|
|
// POST /api/builds/:owner/:name/builds/:number
|
|
|
|
//
|
|
|
|
func RunBuild(c *gin.Context) {
|
|
|
|
remote := ToRemote(c)
|
|
|
|
store := ToDatastore(c)
|
|
|
|
queue_ := ToQueue(c)
|
|
|
|
repo := ToRepo(c)
|
2015-06-08 00:04:57 +00:00
|
|
|
conf := ToSettings(c)
|
|
|
|
|
2015-05-11 07:45:31 +00:00
|
|
|
num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
return
|
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build, err := store.BuildNumber(repo, num)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build.Jobs, err = store.JobList(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
user, err := store.User(repo.UserID)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// must not restart a running build
|
2015-06-23 03:45:08 +00:00
|
|
|
if build.Status == common.StatePending || build.Status == common.StateRunning {
|
2015-05-11 07:45:31 +00:00
|
|
|
c.AbortWithStatus(409)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
build.Status = common.StatePending
|
|
|
|
build.Started = 0
|
|
|
|
build.Finished = 0
|
|
|
|
for _, job := range build.Jobs {
|
2015-06-19 00:36:52 +00:00
|
|
|
job.Status = common.StatePending
|
|
|
|
job.Started = 0
|
|
|
|
job.Finished = 0
|
|
|
|
job.ExitCode = 0
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
err = store.SetBuild(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(500, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
netrc, err := remote.Netrc(user)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(500, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// featch the .drone.yml file from the database
|
2015-06-23 03:45:08 +00:00
|
|
|
raw, err := remote.Script(user, repo, build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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-08-20 04:24:32 +00:00
|
|
|
encrypted, _ := secure.Parse(repo.Keys.Private, repo.Hash, string(raw))
|
2015-08-19 06:59:30 +00:00
|
|
|
if encrypted != nil && len(encrypted) != 0 {
|
|
|
|
raw = []byte(inject.InjectSafe(string(raw), encrypted))
|
|
|
|
}
|
2015-05-11 07:45:31 +00:00
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
c.JSON(202, build)
|
2015-05-11 07:45:31 +00:00
|
|
|
|
|
|
|
queue_.Publish(&queue.Work{
|
2015-06-08 00:04:57 +00:00
|
|
|
User: user,
|
|
|
|
Repo: repo,
|
2015-06-23 03:45:08 +00:00
|
|
|
Build: build,
|
2015-06-19 01:50:57 +00:00
|
|
|
Keys: repo.Keys,
|
2015-06-08 00:04:57 +00:00
|
|
|
Netrc: netrc,
|
|
|
|
Yaml: raw,
|
|
|
|
Plugins: conf.Plugins,
|
|
|
|
Env: conf.Environment,
|
2015-05-11 07:45:31 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// KillBuild accepts a request to kill a running build.
|
|
|
|
//
|
|
|
|
// DELETE /api/builds/:owner/:name/builds/:number
|
|
|
|
//
|
|
|
|
func KillBuild(c *gin.Context) {
|
|
|
|
runner := ToRunner(c)
|
|
|
|
queue := ToQueue(c)
|
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
return
|
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build, err := store.BuildNumber(repo, num)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
build.Jobs, err = store.JobList(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// must not restart a running build
|
2015-06-23 03:45:08 +00:00
|
|
|
if build.Status != common.StatePending && build.Status != common.StateRunning {
|
2015-05-11 07:45:31 +00:00
|
|
|
c.Fail(409, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove from the queue if exists
|
|
|
|
//
|
|
|
|
// TODO(bradrydzewski) this could yield a race condition
|
|
|
|
// because other threads may also be accessing these items.
|
|
|
|
for _, item := range queue.Items() {
|
2015-06-23 03:45:08 +00:00
|
|
|
if item.Repo.FullName == repo.FullName && item.Build.Number == build.Number {
|
2015-05-11 07:45:31 +00:00
|
|
|
queue.Remove(item)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
build.Status = common.StateKilled
|
|
|
|
build.Finished = time.Now().Unix()
|
|
|
|
if build.Started == 0 {
|
|
|
|
build.Started = build.Finished
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
for _, job := range build.Jobs {
|
2015-06-19 00:36:52 +00:00
|
|
|
if job.Status != common.StatePending && job.Status != common.StateRunning {
|
2015-05-11 07:45:31 +00:00
|
|
|
continue
|
|
|
|
}
|
2015-06-19 00:36:52 +00:00
|
|
|
job.Status = common.StateKilled
|
2015-06-23 03:45:08 +00:00
|
|
|
job.Started = build.Started
|
|
|
|
job.Finished = build.Finished
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
2015-06-23 03:45:08 +00:00
|
|
|
err = store.SetBuild(build)
|
2015-05-11 07:45:31 +00:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(500, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
for _, job := range build.Jobs {
|
2015-06-19 00:36:52 +00:00
|
|
|
runner.Cancel(job)
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|
|
|
|
// // get the agent from the repository so we can
|
|
|
|
// // notify the agent to kill the build.
|
|
|
|
// agent, err := store.BuildAgent(repo.FullName, build.Number)
|
|
|
|
// if err != nil {
|
|
|
|
// c.JSON(200, build)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// url_, _ := url.Parse("http://" + agent.Addr)
|
|
|
|
// url_.Path = fmt.Sprintf("/cancel/%s/%v", repo.FullName, build.Number)
|
|
|
|
// resp, err := http.Post(url_.String(), "application/json", nil)
|
|
|
|
// if err != nil {
|
|
|
|
// c.Fail(500, err)
|
|
|
|
// return
|
|
|
|
// }
|
|
|
|
// defer resp.Body.Close()
|
|
|
|
|
2015-06-23 03:45:08 +00:00
|
|
|
c.JSON(200, build)
|
2015-05-11 07:45:31 +00:00
|
|
|
}
|