Rainbond/builder/exector/plugin_dockerfile.go

151 lines
5.6 KiB
Go
Raw Normal View History

2018-03-14 14:12:26 +08:00
// Copyright (C) 2014-2018 Goodrain Co., Ltd.
2017-11-07 11:40:44 +08:00
// RAINBOND, Application Management Platform
2018-03-14 14:33:31 +08:00
2017-11-07 11:40:44 +08:00
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version. For any non-GPL usage of Rainbond,
// one or multiple Commercial Licenses authorized by Goodrain Co., Ltd.
// must be obtained first.
2018-03-14 14:33:31 +08:00
2017-11-07 11:40:44 +08:00
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
2018-03-14 14:33:31 +08:00
2017-11-07 11:40:44 +08:00
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
package exector
import (
"fmt"
"os"
"strings"
"github.com/goodrain/rainbond/builder/sources"
"github.com/goodrain/rainbond/util"
2018-03-07 22:52:06 +08:00
"github.com/goodrain/rainbond/db"
"github.com/goodrain/rainbond/event"
2017-11-07 11:40:44 +08:00
2018-03-07 22:52:06 +08:00
"github.com/docker/engine-api/types"
2017-11-07 11:40:44 +08:00
"github.com/pquerna/ffjson/ffjson"
"github.com/goodrain/rainbond/builder/model"
2017-11-07 11:40:44 +08:00
"github.com/Sirupsen/logrus"
)
const (
cloneTimeout = 60
buildingTimeout = 180
formatSourceDir = "/cache/build/%s/source/%s"
)
2017-11-07 11:40:44 +08:00
func (e *exectorManager) pluginDockerfileBuild(in []byte) {
var tb model.BuildPluginTaskBody
if err := ffjson.Unmarshal(in, &tb); err != nil {
logrus.Errorf("unmarshal taskbody error, %v", err)
return
}
eventID := tb.EventID
logger := event.GetManager().GetLogger(eventID)
logger.Info("从dockerfile构建插件任务开始执行", map[string]string{"step": "builder-exector", "status": "starting"})
2017-11-07 11:40:44 +08:00
go func() {
logrus.Info("start exec build plugin from image worker")
defer event.GetManager().ReleaseLogger(logger)
2018-03-08 09:35:20 +08:00
for retry := 0; retry < 2; retry++ {
err := e.runD(&tb, logger)
2017-11-07 11:40:44 +08:00
if err != nil {
2018-01-08 15:41:13 +08:00
logrus.Errorf("exec plugin build from dockerfile error:%s", err.Error())
logger.Info("dockerfile构建插件任务执行失败开始重试", map[string]string{"step": "builder-exector", "status": "failure"})
2017-11-07 11:40:44 +08:00
} else {
2018-01-08 15:41:13 +08:00
return
2017-11-07 11:40:44 +08:00
}
}
2018-01-08 15:41:13 +08:00
version, err := db.GetManager().TenantPluginBuildVersionDao().GetBuildVersionByVersionID(tb.PluginID, tb.VersionID)
if err != nil {
logrus.Errorf("get version error, %v", err)
}
version.Status = "failure"
if err := db.GetManager().TenantPluginBuildVersionDao().UpdateModel(version); err != nil {
logrus.Errorf("update version error, %v", err)
}
2018-01-10 11:00:37 +08:00
logger.Error("dockerfile构建插件任务执行失败", map[string]string{"step": "callback", "status": "failure"})
2017-11-07 11:40:44 +08:00
}()
}
func (e *exectorManager) runD(t *model.BuildPluginTaskBody, logger event.Logger) error {
2017-11-07 11:40:44 +08:00
logger.Info("开始拉取代码", map[string]string{"step": "build-exector"})
sourceDir := fmt.Sprintf(formatSourceDir, t.TenantID, t.VersionID)
if t.Repo == "" {
t.Repo = "master"
}
2018-03-07 23:01:32 +08:00
if !util.DirIsEmpty(sourceDir) {
os.RemoveAll(sourceDir)
}
if err := util.CheckAndCreateDir(sourceDir); err != nil {
return err
}
2018-03-07 22:52:06 +08:00
if _, err := sources.GitClone(sources.CodeSourceInfo{RepositoryURL: t.GitURL, Branch: t.Repo}, sourceDir, logger, 4); err != nil {
2018-01-05 15:12:11 +08:00
logger.Error("拉取代码失败", map[string]string{"step": "builder-exector", "status": "failure"})
2018-03-07 22:52:06 +08:00
logrus.Errorf("[plugin]git clone code error %v", err)
2017-11-07 11:40:44 +08:00
return err
}
if !checkDockerfile(sourceDir) {
2018-01-05 15:12:11 +08:00
logger.Error("代码未检测到dockerfile暂不支持构建任务即将退出", map[string]string{"step": "builder-exector", "status": "failure"})
2017-11-07 11:40:44 +08:00
logrus.Error("代码未检测到dockerfile")
return fmt.Errorf("have no dockerfile")
}
logger.Info("代码检测为dockerfile开始编译", map[string]string{"step": "build-exector"})
2018-03-07 22:52:06 +08:00
mm := strings.Split(t.GitURL, "/")
n1 := strings.Split(mm[len(mm)-1], ".")[0]
buildImageName := strings.ToLower(fmt.Sprintf("goodrain.me/%s_%s", n1, t.VersionID))
buildOptions := types.ImageBuildOptions{
Tags: []string{buildImageName},
Remove: true,
}
if noCache := os.Getenv("NO_CACHE"); noCache != "" {
buildOptions.NoCache = true
} else {
buildOptions.NoCache = false
}
logger.Info("开始构建镜像", map[string]string{"step": "builder-exector"})
err := sources.ImageBuild(e.DockerClient, sourceDir, buildOptions, logger, 5)
2017-11-07 11:40:44 +08:00
if err != nil {
2018-03-07 22:52:06 +08:00
logger.Error(fmt.Sprintf("构造镜像%s失败: %s", buildImageName, err.Error()), map[string]string{"step": "builder-exector", "status": "failure"})
logrus.Errorf("[plugin]build image error: %s", err.Error())
2017-11-07 11:40:44 +08:00
return err
}
2018-03-07 22:52:06 +08:00
logger.Info("镜像构建成功,开始推送镜像至仓库", map[string]string{"step": "builder-exector"})
err = sources.ImagePush(e.DockerClient, buildImageName, "", "", logger, 2)
2018-03-07 22:52:06 +08:00
if err != nil {
logger.Error("推送镜像失败", map[string]string{"step": "builder-exector"})
logrus.Errorf("push image error: %s", err.Error())
2017-11-07 11:40:44 +08:00
return err
}
logger.Info("推送镜像完成", map[string]string{"step": "build-exector"})
version, err := db.GetManager().TenantPluginBuildVersionDao().GetBuildVersionByVersionID(t.PluginID, t.VersionID)
if err != nil {
logrus.Errorf("get version error, %v", err)
return err
}
2018-03-07 22:52:06 +08:00
version.BuildLocalImage = buildImageName
2017-11-07 11:40:44 +08:00
version.Status = "complete"
if err := db.GetManager().TenantPluginBuildVersionDao().UpdateModel(version); err != nil {
logrus.Errorf("update version error, %v", err)
return err
}
2017-12-25 16:03:51 +08:00
logger.Info("从dockerfile构建插件完成", map[string]string{"step": "last", "status": "success"})
2017-11-07 11:40:44 +08:00
return nil
}
func checkDockerfile(sourceDir string) bool {
if _, err := os.Stat(fmt.Sprintf("%s/Dockerfile", sourceDir)); os.IsNotExist(err) {
return false
}
return true
}