jenkins as code 与go语言学习
前言
最近看jenkins as code这个概念在很多文章中提起,持续交付中八大原则也有把一切都放入版本管理,最近准备把我们公司用的一些jenkins上的job的配置也放到git中,由于https://github.com/jenkinsci/job-dsl-plugin的支持是groovy。我不懂groovy,所以找了一些网上的groovy脚本改了改,并且通过go template实现这个模板,还能练习使用go。
所有用的到文件简介
一共用到3个文件,只有一个模板,如果有多个情景可以多写几个模板
bogon:jenkins-dsl hongzhi.wang$ tree
.
├── config.yaml #配置文件
├── generate_dsl.go #go 源码文件
└── springboot.tpl #go template 文件
配置文件
这次准备通过yaml文件实现模板的渲染,配置文件如下,通过yaml来配置文件比较清晰。
- department_name: java
cn_name: java组
apps:
- app_name: app-1
nodes:
- node1
- node2
- node3
ansible_playbook_name: deploy-springboot-jar.yml
jvm_size: 2048m
template_name: springboot.tpl
- app_name: app-2
nodes:
- node1
- node2
- node3
ansible_playbook_name: deploy-springboot-jar.yml
jvm_size: 512m
template_name: springboot.tpl
- department_name: java2
cn_name: java2组
apps:
- app_name: app-4
nodes:
- node1
- node2
- node3
ansible_playbook_name: deploy-springboot-war.yml
jvm_size: 2048m
template_name: springboot.tpl
GO Template
这里面主要是用到了自定函数ProcessNodes
,其他的都是模板的正常用法,还有就是我们的jenkins主要是传参数拉代码,然后通过ansible-playbook实现应用部署。
def app_name = '{{ .App.AppName }}'
def gitUrl = "git@gitlab.wis.com:{{ .Name }}/{{ .App.AppName }}.git"
job("{{ .Name }}-${app_name}") {
description()
keepDependencies(false)
parameters {
choiceParam("Mode", ["release", "deploy"], "")
choiceParam("app_name", ["${app_name}"], "")
choiceParam("NODES", {{ ProcessNodes .App.Nodes }}, "")
gitParam('VERSION') {
sortMode('DESCENDING')
tagFilter('*')
}
}
scm {
git {
remote {
url(gitUrl)
credentials("git")
}
branch("\$VERSION")
}
}
disabled(false)
concurrentBuild(false)
steps {
maven{
configure { node ->
node / 'mavenName' ('maven')
}
goals('clean')
}
maven{
configure { node ->
node / 'mavenName' ('maven')
}
goals('install -Pproduct')
}
shell("""
source /opt/py3/bin/activate
if [ \$Mode == release ]
then
START_TASK="copy app code"
fi
echo \$JOB_NAME
cd /ansible/
echo \$WORKSPACE
for host in \$NODES
do
ansible-playbook --start-at-task="\$START_TASK" {{ .App.AnsiblePlaybookName }} --extra-vars "target_host=\$host workspace=\$WORKSPACE app_name=\$app_name version=\$VERSION jvm_size={{ .App.JvmSize }}"
done
""")
}
}
publishers {
postBuildScripts {
steps {
steps {
shell("""
/ops/scripts/dingding.py {{ .Name }} `git show -s --pretty=%cN` `git tag -ln \$VERSION` \$NODES
""")
}
}
markBuildUnstable(false)
}
}
}
listView("{{ .CnName }}") {
jobs {
regex(/{{ .Name }}-.+/)
}
columns {
status()
weather()
name()
lastSuccess()
lastFailure()
lastDuration()
buildButton()
}
}
GO源码
这个源码主要按照配置文件,把已经去掉的部门目录还有应用groovy脚本给删了,并且推送到远程分支,groovy脚本的名字里面不能有-
,因为groovy的脚本名字会解析成java的类名,-
是不能放在java语言的类名里面了。所以只能把应用名字都的-
转成_
。
package main
import (
"io/ioutil"
"github.com/ghodss/yaml"
"fmt"
"text/template"
"os"
"log"
"strings"
"os/exec"
"bytes"
)
type Department struct {
Name string `json:"department_name"`
CnName string `json:"cn_name"`
Apps []App `json:"apps"`
}
type App struct {
AppName string `json:"app_name"`
Nodes []string `json:"nodes"`
PythonVersion string `json:"python_version"`
JvmSize string `json:"jvm_size"`
TemplateName string `json:"template_name"`
AnsiblePlaybookName string `json:"ansible_playbook_name"`
DestPath string `json:"dest_path"`
TomcatGitUrl string `json:"tomcat_git_url"`
TomcatPort string `json:"tomcat_port"`
TomcatAdminPort string `json:"tomcat_admin_port"`
TomcatName string `json:"tomcat_name"`
TomcatWebApp string `json:"tomcat_web_app"`
}
func In(i string,l []string) bool{
for _, item:= range l{
if item == i{
return true
}
}
return false
}
func checkErr(err error) {
if err != nil{
log.Fatal(err)
}
}
func ProcessNodes(l []string) string {
if len(l) == 1{
return fmt.Sprintf(`["%s"]`,strings.Join(l," "))
}
return fmt.Sprintf(`["%s","%s"]`,l[0],strings.Join(l," "))
}
func AppNameToFileName(appname string) string {
appname = strings.Replace(appname,"-","_",-1)
return fmt.Sprintf("%s.groovy",appname)
}
func SysCommand(command string){
commandList := strings.Fields(command)
var out bytes.Buffer
cmd := exec.Command(commandList[0],commandList[1:]...)
cmd.Stdout = &out
cmd.Stderr = &out
err := cmd.Run()
if err != nil{
log.Printf("commd error: %v",command)
}
log.Println(out.String())
}
func main() {
content,err := ioutil.ReadFile("config.yaml")
checkErr(err)
var d []Department
err = yaml.Unmarshal(content, &d)
checkErr(err)
DepartmentNames := []string{}
funcMap := template.FuncMap{
"ProcessNodes": ProcessNodes,
}
for _, department := range d{
err := os.MkdirAll(department.Name,0755)
checkErr(err)
DepartmentNames = append(DepartmentNames, department.Name)
if AppFileNames := []string{}; department.Apps != nil{
for _, app := range department.Apps{
FileName := AppNameToFileName(app.AppName)
AppFileNames = append(AppFileNames,FileName)
data := map[string]interface{}{
"Name":department.Name,
"CnName":department.CnName,
"App":app,
}
templ,err := template.New(app.TemplateName).Funcs(funcMap).ParseFiles(app.TemplateName)
checkErr(err)
f, err := os.Create(fmt.Sprintf("%s/%s",department.Name, FileName))
defer f.Close()
checkErr(err)
err = templ.Execute(f, data)
checkErr(err)
}
ExistFiles,err := ioutil.ReadDir(department.Name)
checkErr(err)
for _,v := range ExistFiles{
if ! In(v.Name(),AppFileNames){
log.Println("going to delete file: ",v.Name())
SysCommand(fmt.Sprintf("git rm %s/%s",department.Name,v.Name()))
}
}
}
}
ExistDirs,err := ioutil.ReadDir("./")
checkErr(err)
for _,dir := range ExistDirs{
if dir.IsDir()&& dir.Name()!=".idea" && dir.Name()!=".git" && ! In(dir.Name(),DepartmentNames){
log.Println("going to delete dir: ",dir.Name())
log.Printf("git rm -r %s\n",dir.Name())
SysCommand(fmt.Sprintf("git rm -r %s",dir.Name()))
checkErr(err)
}
}
SysCommand("git add * ")
SysCommand("git commit -m 'update' ")
//SysCommand("git push ")
log.Println("finished")
}
templ,err := template.New(app.TemplateName).Funcs(funcMap).ParseFiles(app.TemplateName)
这个之所以这样是因为要传入自定的函数,parsefile的返回值是两个不能直接用Funcs。这个就要详见Stack Overflow上的答案了。
脚本运行的结果
bogon:jenkins-dsl hongzhi.wang$ go run generate_dsl.go
bogon:jenkins-dsl hongzhi.wang$ tree
.
├── config.yaml
├── generate_dsl.go
├── java
│ ├── app_1.groovy
│ └── app_2.groovy
├── java2
│ └── app_4.groovy
└── springboot.tpl
剩余的工作
这个脚本把代码推送到gitlab之后,gitlab通过webhook触发一个jenkins-seed-job,这个job再运行groovy脚本,这个groovy脚本运行之后我们线上的 job 就和 YAML 文件描述的一致了,我们是一个部门对应一个jenkins-seed-job,一个webhook。如果config.yaml太大了,可以把一些部门的配置放到另一个源码库里。
steps {
dsl {
ignoreExisting(false)
removeAction("DELETE")
removeViewAction("IGNORE")
lookupStrategy("JENKINS_ROOT")
}
}
更新于2019年3月
最近由于有的组需要频繁发版,所以把部署的权限给了这些组的负责人,但是我们运维又需要知道他们发版的时间和具体原因,所以我们这边的方法是通过 http 调用发送响应的信息到对应的组的钉钉群。调研了一下,直接通过 curl 调用钉钉的接口变量老是传不过去,而且 curl 调用不是很灵活,所以通过在 jenkins 服务器上放置 python 脚本的方式发送钉钉消息,脚本传入的参数分别是组名,git commiter 名字,git tag 和对应 tag 的详细描述,部署的生产服务器,脚本根据组名来分辨具体发送到哪个组。只有构建成功的作业才会发送到钉钉。
publishers {
postBuildScripts {
steps {
steps {
shell("""
/ops/scripts/dingding.py {{ .Name }} `git show -s --pretty=%cN` `git tag -ln \$VERSION` \$NODES
""")
}
}
markBuildUnstable(false)
}
}
}
总结
这次主要是把jenkins的job管理纳入版本控制,并且练习一下go。这里主要是做个笔记o( ̄︶ ̄)o。
重构
最近由于新的需求把上面go的代码重构了一下,如下:
package main
import (
"bytes"
"fmt"
"github.com/ghodss/yaml"
"io/ioutil"
"log"
"os"
"os/exec"
"strings"
"text/template"
)
const (
ImageTemplatePath = "image_dsl"
)
var (
TemplateName = "common.tpl"
SystemDirs = []string{".git", "image_dsl", ".idea", "templates"}
)
func main() {
SysCommandErrorExit("git pull")
content, err := ioutil.ReadFile("config.yaml")
checkErr(err)
var company Company
err = yaml.Unmarshal(content, &company.Departments)
checkErr(err)
company.JenkinsDsl()
company.CleanInactiveDepartments()
SysCommand("git add * ")
SysCommand("git commit -m 'update' ")
SysCommand("git push ")
log.Println("finished")
}
type Company struct {
Departments []Department `json:"departments"`
}
func (c Company) DepartmentNames() (DepartmentNames []string) {
for _, department := range c.Departments {
DepartmentNames = append(DepartmentNames, department.Name)
}
return
}
func (c Company) JenkinsDsl() {
for _, department := range c.Departments {
department.JenkinsDsl()
}
}
func (c Company) CleanInactiveDepartments() {
ExistDirs, err := ioutil.ReadDir("./")
checkErr(err)
for _, dir := range ExistDirs {
if dir.IsDir() && ! In(dir.Name(), SystemDirs) && ! In(dir.Name(), c.DepartmentNames()) {
log.Println("going to delete dir: ", dir.Name())
log.Printf("git rm -r %s\n", dir.Name())
SysCommand(fmt.Sprintf("git rm -r %s", dir.Name()))
}
}
}
type Department struct {
Name string `json:"department_name"`
CnName string `json:"cn_name"`
Apps []App `json:"apps"`
}
func (d Department) JenkinsDsl() {
d.DepartmentDir()
if d.Apps != nil {
for _, app := range d.Apps {
app.JenkinsDsl(d)
}
d.CleanOfflineApps()
}
}
func (d Department) CleanOfflineApps() {
ExistFiles, err := ioutil.ReadDir(d.Name)
checkErr(err)
for _, v := range ExistFiles {
if ! In(v.Name(), d.AppFileNames()) {
log.Println("going to delete file: ", v.Name())
SysCommand(fmt.Sprintf("git rm %s/%s", d.Name, v.Name()))
}
}
}
func (d Department) AppFileNames() (AppFileNames []string) {
for _, app := range d.Apps {
AppFileNames = append(AppFileNames, AppNameToFileName(app.AppName))
}
return
}
func (d Department) DepartmentDir() {
err := os.MkdirAll(d.Name, 0755)
checkErr(err)
}
type App struct {
AppName string `json:"app_name"`
Department Department `json:"department"`
BuildImage bool `json:"build_image"`
ArtifactFormat string `json:"artifact_format"`
JvmSize string `json:"jvm_size"`
JavaVersion string `json:"java_version"`
Clusters []Cluster `json:"clusters"`
ImageTPL string `json:"image_tpl"`
Values map[string]interface{} `json:"values"`
}
type Cluster struct {
Name string `json:"name"`
ReplicaCount int `json:"replicaCount"`
ReplicaCountMax int `json:"replicaCountMAX"`
}
func (a App) JenkinsDsl(d Department) {
a.Department = d
f := a.file()
err := a.Template().Execute(f, a.TemplateData())
defer f.Close()
checkErr(err)
}
func (a App) ValuesYamlString() (values string) {
Values, err := yaml.Marshal(a.Values)
checkErr(err)
return string(Values)
}
func (a App) TemplateData() (data map[string]interface{}) {
data = map[string]interface{}{
"Name": a.Department.Name,
"CnName": a.Department.CnName,
"App": a,
"Values": a.ValuesYamlString(),
}
return
}
func (a App) Template() (templ *template.Template) {
helmTpl := fmt.Sprintf("templates/helm_dsl/%s", TemplateName)
templ, err := template.New(TemplateName).ParseFiles(helmTpl)
checkErr(err)
return
}
func (a App) file() (*os.File) {
FileName := AppNameToFileName(a.AppName)
f, err := os.Create(fmt.Sprintf("%s/%s", a.Department.Name, FileName))
checkErr(err)
return f
}
func In(i string, l []string) bool {
for _, item := range l {
if item == i {
return true
}
}
return false
}
func checkErr(err error) {
if err != nil {
log.Fatal(err)
}
}
func AppNameToFileName(appname string) string {
appname = strings.Replace(appname, "-", "_", -1)
return fmt.Sprintf("%s.groovy", appname)
}
func SysCommandErrorExit(command string) {
commandList := strings.Fields(command)
var out bytes.Buffer
cmd := exec.Command(commandList[0], commandList[1:]...)
cmd.Stdout = &out
cmd.Stderr = &out
err := cmd.Run()
if err != nil {
log.Fatalln("command error: %v", command)
}
log.Println(out.String())
}
func SysCommand(command string) {
commandList := strings.Fields(command)
var out bytes.Buffer
cmd := exec.Command(commandList[0], commandList[1:]...)
cmd.Stdout = &out
cmd.Stderr = &out
err := cmd.Run()
if err != nil {
log.Printf("command error: %v", command)
}
log.Println(out.String())
}
jenkins as code 与go语言学习的更多相关文章
- GO学习-(3) VS Code配置Go语言开发环境
VS Code配置Go语言开发环境 VS Code配置Go语言开发环境 说在前面的话,Go语言是采用UTF8编码的,理论上使用任何文本编辑器都能做Go语言开发.大家可以根据自己的喜好自行选择.编辑器/ ...
- HTML语言学习笔记(会更新)
# HTML语言学习笔记(会更新) 一个html文件是由一系列的元素和标签组成的. 标签: 1.<html></html> 表示该文件为超文本标记语言(HTML)编写的.成对出 ...
- 大一上学期C语言学习心得总结
经过一个学期的C语言学习,大体算是在这个编程语言上入了门,能够通过一些代码解决特定的问题.当然,每次成功将问题转换成代码都小有激动,虽然只是在黑框上输出了一些数字或是字符串. 编程,虽然还不是很懂,但 ...
- 2017-05-4-C语言学习笔记
C语言学习笔记... ------------------------------------ Hello C语言:什么是程序:程序是指:完成某件事的既定方式和过程.计算机中的程序是指:为了让计算机执 ...
- R语言学习 第四篇:函数和流程控制
变量用于临时存储数据,而函数用于操作数据,实现代码的重复使用.在R中,函数只是另一种数据类型的变量,可以被分配,操作,甚至把函数作为参数传递给其他函数.分支控制和循环控制,和通用编程语言的风格很相似, ...
- Go语言学习之路
我关于Go语言的博客原本发布于我的个人网站:wwww.liwenzhouu.com.但是被某些人抄怕了,没办法只好搬运到博客园. 我的Go语言学习之路 2015年底我因为工作原因接触到了Go语言,那时 ...
- Java语言学习day02--6月29日
Java语言学习day02###01常用的DOS命令 * A: 常用的DOS命令 * a: 打开Dos控制台 * win+r--cmd--回车 * b: 常用dos命令 * cd.. : 退回到上一级 ...
- C语言学习 第八次作业总结
本次作业其实没有新的内容,主要就是复习上一次的一维数组的相关内容.冯老师布置了5道题目,其中涉及到一些比较简单的排序或者是查找的方法.因为数据很少,所以直接使用for循环遍历就可以了. 关于本次作业, ...
- C语言学习 第七次作业总结
C语言学习 第七次作业总结 数组可以分为数组和多下标数组(在传统的国内C语言书本中,将其称为二/多维数组). 数组名称 在之前的课程中,大家应该都有印象,对于int a这样的定义,会为变量 a 声明一 ...
随机推荐
- superset在 centos 7安装运行
参考:1.http://blog.csdn.net/u014729236/article/details/76302888?locationNum=2&fps=1 2.https://www. ...
- eclipse-jee-kepler 如何设置编译compiler为1.8
最新下载了jdk1.8,想在eclipse里面用一下 jdk1.8的新特性 但是,貌似eclipse(eclipse-jee-kepler-SR2-win32-x86_64.zip)最高的编译级别为: ...
- 关于web前端中遇到的html,css小知识点
容器溢出: 语法:overflow: visible | hidden | scroll | auto | inherit; visible:默认值,溢出内容不会被裁剪,正常显示 hidden: 溢出 ...
- 后台商品搜索功能开发SQL
在做后台的商品搜索功能开发时遇到了一些问题记录下来 版本一 <select id="SelectByNameAndParentId resultMap="Base_resul ...
- The declared package does not match the expected package Java
今天使用vscode 编写java代码做测试时候,发现这个问题,大概总结一下. 目录结构 bao -> Point.java test.java package bao; public clas ...
- T4设计时模板调试
在Visual Studio内调试T4设计时模板有多个方法:安装使用带调试功能的第三方工具,利用System.Diagnostics.Debugger实时调试器,VS内置的T4调试工具.使用第三方工具 ...
- Quartz.Net进阶之七:QuartzNet其他的功能简述
一.介绍 今天是这个系列的最后一篇文章了,主要功能说的差不多了,我们来看看其他相关的内容.话说回来,虽然是这个系列的最后一篇文章,并不代表Quartz的东西就这么点,学习阶段,就这些了,如果以后有了使 ...
- 安装Pygame(Python3.6,windows)
1. 本机为python3.6的环境 2. 到pygame官网下载对应系统,对应python版本的pygame文件,下载地址:https://pypi.python.org/pypi/Pygame/1 ...
- Collection<T> 的一个坑
当前所在的公司偏好使用 Collection<T>(System.Collections.ObjectModel), 这货比起List<T>不仅少了很多实用方法, 而且还有一个 ...
- window下maven的环境搭建
一.下载 官网下载地址 二.maven的安装配置 1.环境变量的配置 1)新建环境变量MAVEN_HOME,值如下: D:\install\develop\apache-maven- 2)修改环境变量 ...