2015-04-08 15:43:59 -07:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
2015-04-16 14:45:05 -07:00
|
|
|
"io"
|
2015-04-08 15:43:59 -07:00
|
|
|
"strconv"
|
|
|
|
|
2015-04-16 14:45:05 -07:00
|
|
|
"github.com/drone/drone/common"
|
2015-04-08 15:43:59 -07:00
|
|
|
"github.com/gin-gonic/gin"
|
2015-04-16 14:45:05 -07:00
|
|
|
"github.com/gin-gonic/gin/binding"
|
2015-04-08 15:43:59 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// GetBuild accepts a request to retrieve a build
|
|
|
|
// from the datastore for the given repository and
|
|
|
|
// build number.
|
|
|
|
//
|
|
|
|
// GET /api/builds/:owner/:name/:number
|
|
|
|
//
|
|
|
|
func GetBuild(c *gin.Context) {
|
2015-04-14 22:04:38 -07:00
|
|
|
store := ToDatastore(c)
|
2015-04-08 15:43:59 -07:00
|
|
|
repo := ToRepo(c)
|
|
|
|
num, err := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
return
|
|
|
|
}
|
2015-04-14 22:04:38 -07:00
|
|
|
build, err := store.Build(repo.FullName, num)
|
2015-04-08 15:43:59 -07:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
|
|
|
c.JSON(200, build)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-16 14:45:05 -07:00
|
|
|
// GetBuilds accepts a request to retrieve a list
|
2015-04-08 15:43:59 -07:00
|
|
|
// of builds from the datastore for the given repository.
|
|
|
|
//
|
|
|
|
// GET /api/builds/:owner/:name
|
|
|
|
//
|
|
|
|
func GetBuilds(c *gin.Context) {
|
2015-04-14 22:04:38 -07:00
|
|
|
store := ToDatastore(c)
|
2015-04-08 15:43:59 -07:00
|
|
|
repo := ToRepo(c)
|
2015-04-14 22:04:38 -07:00
|
|
|
builds, err := store.BuildList(repo.FullName)
|
2015-04-08 15:43:59 -07:00
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else {
|
|
|
|
c.JSON(200, builds)
|
|
|
|
}
|
|
|
|
}
|
2015-04-16 14:45:05 -07:00
|
|
|
|
|
|
|
// GetBuildLogs 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 GetBuildLogs(c *gin.Context) {
|
|
|
|
store := ToDatastore(c)
|
|
|
|
repo := ToRepo(c)
|
|
|
|
full, _ := strconv.ParseBool(c.Params.ByName("full"))
|
|
|
|
build, _ := strconv.Atoi(c.Params.ByName("number"))
|
|
|
|
task, _ := strconv.Atoi(c.Params.ByName("task"))
|
|
|
|
|
|
|
|
r, err := store.LogReader(repo.FullName, build, task)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
} else if full {
|
|
|
|
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.SetStatus(repo.Name, num, in); err != nil {
|
|
|
|
c.Fail(400, err)
|
|
|
|
} else {
|
|
|
|
c.JSON(201, in)
|
|
|
|
}
|
|
|
|
}
|
2015-04-16 22:07:13 -07:00
|
|
|
|
|
|
|
// RunBuild accepts a request to restart an existing build.
|
|
|
|
//
|
|
|
|
// POST /api/builds/:owner/:name/builds/:number
|
|
|
|
//
|
|
|
|
func RunBuild(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
|
|
|
|
}
|
|
|
|
build, err := store.Build(repo.FullName, num)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// must not restart a running build
|
|
|
|
if build.State == common.StatePending || build.State == common.StateRunning {
|
|
|
|
c.Fail(409, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
build.State = common.StatePending
|
|
|
|
build.Started = 0
|
|
|
|
build.Finished = 0
|
|
|
|
build.Duration = 0
|
|
|
|
build.Statuses = []*common.Status{}
|
|
|
|
for _, task := range build.Tasks {
|
|
|
|
task.State = common.StatePending
|
|
|
|
task.Started = 0
|
|
|
|
task.Finished = 0
|
|
|
|
task.ExitCode = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
err = store.SetBuild(repo.FullName, build)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(500, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO push build to queue
|
|
|
|
|
|
|
|
c.JSON(202, build)
|
|
|
|
}
|
|
|
|
|
|
|
|
// KillBuild accepts a request to kill a running build.
|
|
|
|
//
|
|
|
|
// DELETE /api/builds/:owner/:name/builds/:number
|
|
|
|
//
|
|
|
|
func KillBuild(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
|
|
|
|
}
|
|
|
|
build, err := store.Build(repo.FullName, num)
|
|
|
|
if err != nil {
|
|
|
|
c.Fail(404, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// must not restart a running build
|
|
|
|
if build.State != common.StatePending && build.State != common.StateRunning {
|
|
|
|
c.Fail(409, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO tell queue to cancel build
|
|
|
|
|
|
|
|
c.Writer.WriteHeader(202)
|
|
|
|
}
|