woodpecker/cmd/drone-build/main.go

157 lines
3.5 KiB
Go
Raw Normal View History

package main
import (
"bytes"
"encoding/json"
"flag"
"fmt"
"os"
2015-08-21 01:31:54 +00:00
"os/signal"
"strings"
2015-08-21 01:31:54 +00:00
"syscall"
2015-05-22 18:37:40 +00:00
log "github.com/drone/drone/Godeps/_workspace/src/github.com/Sirupsen/logrus"
"github.com/drone/drone/Godeps/_workspace/src/github.com/samalba/dockerclient"
2015-05-17 20:51:42 +00:00
common "github.com/drone/drone/pkg/types"
)
var (
clone = flag.Bool("clone", false, "")
build = flag.Bool("build", false, "")
publish = flag.Bool("publish", false, "")
deploy = flag.Bool("deploy", false, "")
notify = flag.Bool("notify", false, "")
2015-05-16 02:35:33 +00:00
debug = flag.Bool("debug", false, "")
)
func main() {
flag.Parse()
2015-05-16 02:35:33 +00:00
if *debug {
log.SetLevel(log.DebugLevel)
}
ctx, err := parseContext()
if err != nil {
2015-05-16 02:35:33 +00:00
log.Errorln("Error launching build container.", err)
os.Exit(1)
return
}
// creates the Docker client, connecting to the
// linked Docker daemon
docker, err := dockerclient.NewDockerClient("unix:///var/run/docker.sock", nil)
if err != nil {
2015-05-16 02:35:33 +00:00
log.Errorln("Error connecting to build server.", err)
os.Exit(1)
return
}
// creates a wrapper Docker client that uses an ambassador
// container to create a pod-like environment.
client, err := newClient(docker)
if err != nil {
2015-05-16 02:35:33 +00:00
log.Errorln("Error starting build server pod", err)
os.Exit(1)
return
}
ctx.client = client
defer client.Destroy()
2015-08-21 01:31:54 +00:00
// watch for sigkill (timeout or cancel build)
killc := make(chan os.Signal, 1)
signal.Notify(killc, syscall.SIGINT, syscall.SIGTERM)
go func() {
<-killc
log.Println("Received reques to kill this build")
client.Destroy() // possibe race here. implement lock on the other end
os.Exit(130) // cancel is treated like ctrl+c
2015-08-21 01:31:54 +00:00
}()
// performs some initial parsing and pre-processing steps
// prior to executing our build tasks.
2015-08-19 19:38:08 +00:00
createClone(ctx)
err = setup(ctx)
if err != nil {
2015-05-16 02:35:33 +00:00
log.Errorln("Error processing .drone.yml file.", err)
client.Destroy()
os.Exit(1)
return
}
2015-08-19 01:28:35 +00:00
var execs []execFunc
if *clone {
execs = append(execs, execClone)
}
if *build {
execs = append(execs, execSetup)
execs = append(execs, execCompose)
execs = append(execs, execBuild)
}
if *publish {
execs = append(execs, execPublish)
}
if *deploy {
execs = append(execs, execDeploy)
}
// Loop through and execute each step.
for i, exec_ := range execs {
code, err := exec_(ctx)
if err != nil {
fmt.Printf("00%d Error executing build\n", i+1)
fmt.Println(err)
code = 255
}
if code != 0 {
2015-06-19 00:36:52 +00:00
ctx.Job.ExitCode = code
break
}
}
// Optionally execute notification steps.
if *notify {
execNotify(ctx)
}
client.Destroy()
2015-06-19 00:36:52 +00:00
os.Exit(ctx.Job.ExitCode)
}
func createClone(c *Context) {
c.Clone = &common.Clone{
Netrc: c.Netrc,
Keypair: c.Keys,
Remote: c.Repo.Clone,
Origin: c.Repo.Clone,
}
c.Clone.Origin = c.Repo.Clone
c.Clone.Remote = c.Repo.Clone
2015-07-13 07:11:33 +00:00
c.Clone.Sha = c.Build.Commit.Sha
c.Clone.Ref = c.Build.Commit.Ref
c.Clone.Branch = c.Build.Commit.Branch
// TODO do we still need this? it should be set by the remote
if strings.HasPrefix(c.Clone.Branch, "refs/heads/") {
c.Clone.Branch = c.Clone.Branch[11:]
}
// TODO we should also pass the SourceSha, SourceBranch, etc
// to the clone object for merge requests from bitbucket, gitlab, et al
// if len(c.Commit.PullRequest) != 0 {
// }
}
func parseContext() (*Context, error) {
c := &Context{}
for i, arg := range os.Args {
if arg == "--" && len(os.Args) != i+1 {
buf := bytes.NewBufferString(os.Args[i+1])
err := json.NewDecoder(buf).Decode(c)
return c, err
}
}
err := json.NewDecoder(os.Stdin).Decode(c)
return c, err
}