-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathJenkinsfile
298 lines (279 loc) · 10.9 KB
/
Jenkinsfile
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
def sdk = resolveDotNetSDKToolVersion("8.0")
pipeline {
options {
buildDiscarder(logRotator(numToKeepStr: '50', artifactNumToKeepStr: '50'))
disableConcurrentBuilds()
timeout(time: 60, unit: 'MINUTES')
timestamps ()
}
agent {
label 'linux || linux-large'
}
environment {
PROJECT_WEB_FOLDER = "web-ui"
PROJECT_API_FOLDER = "api"
PROJECT_TEST = "KS.FiksProtokollValidator.Tests/KS.FiksProtokollValidator.Tests.csproj"
PROJECT_CHARTNAME = "fiks-protokoll-validator"
API_APP_NAME = "fiks-protokoll-validator-api"
WEB_APP_NAME = "fiks-protokoll-validator-web"
DOCKERFILE_TESTS = "Dockerfile-run-tests"
ARTIFACTORY_CREDENTIALS = "artifactory-token-based"
DOCKER_REPO_RELEASE = "https://docker-all.artifactory.fiks.ks.no"
DOCKER_REPO = "https://docker-local-snapshots.artifactory.fiks.ks.no"
DOTNET_CLI_HOME = "/tmp/DOTNET_CLI_HOME"
TMPDIR = "${env.PWD + '\\tmpdir'}"
BUILD_OPTS = buildOpts(env.VERSION_SUFFIX)
}
parameters {
booleanParam(defaultValue: false, description: 'Skal prosjektet releases?', name: 'isRelease')
string(name: "specifiedVersion", defaultValue: "", description: "Hva er det nye versjonsnummeret (X.X.X)? Som default releases snapshot-versjonen")
text(name: "releaseNotes", defaultValue: "Ingen endringer utført", description: "Hva er endret i denne releasen?")
text(name: "securityReview", defaultValue: "Endringene har ingen sikkerhetskonsekvenser", description: "Har endringene sikkerhetsmessige konsekvenser, og hvilke tiltak er i så fall iverksatt?")
string(name: "reviewer", defaultValue: "Endringene krever ikke review", description: "Hvem har gjort review?")
}
stages {
stage('Initialize') {
steps {
script {
env.GIT_SHA = sh(returnStdout: true, script: 'git rev-parse HEAD').substring(0, 7)
env.REPO_NAME = scm.getUserRemoteConfigs()[0].getUrl().tokenize('/').last().split("\\.")[0]
env.CURRENT_VERSION = findVersionPrefix()
env.NEXT_VERSION = params.specifiedVersion == "" ? incrementVersion(env.CURRENT_VERSION) : params.specifiedVersion
if(params.isRelease) {
env.VERSION_SUFFIX = ""
env.BUILD_SUFFIX = ""
env.FULL_VERSION = env.CURRENT_VERSION
} else {
def timestamp = getTimestamp()
env.VERSION_SUFFIX = "build.${timestamp}"
env.BUILD_SUFFIX = "--version-suffix ${env.VERSION_SUFFIX}"
env.FULL_VERSION = "${CURRENT_VERSION}-${env.VERSION_SUFFIX}"
}
}
}
}
stage('Build docker images') {
parallel {
stage('API: Build and publish docker image') {
agent {
label 'linux || linux-large'
}
tools {
dotnetsdk sdk
}
environment {
NUGET_HTTP_CACHE_PATH = "${env.WORKSPACE + '@tmp/cache'}"
TMPDIR = "${env.PWD}/tmpdir"
MSBUILDDEBUGPATH = "${env.TMPDIR}"
NUGET_CONF = credentials('nuget-config')
NUGET_ACCESS_KEY = credentials("artifactory-token-based")
NUGET_ALL_REPO = 'https://artifactory.fiks.ks.no/artifactory/api/nuget/nuget-all'
DOTNET_CLI_TELEMETRY_OPTOUT = 1
COMPlus_EnableDiagnostics = 0
DOTNET_GCHeapHardLimit=20000000
}
steps {
withDotNet(sdk: sdk) {
dir("api\\KS.FiksProtokollValidator.WebAPI") {
dotnetRestore(
sdk: sdk,
showSdkInfo: true,
verbosity: 'normal',
force: true
)
dotnetPublish(
configuration: 'Release',
nologo: true,
noRestore: true,
optionsString: env.BUILD_OPTS,
outputDirectory: 'published-api'
)
script {
println("API: Building and publishing docker image version: ${env.FULL_VERSION}")
buildAndPushDockerImage(API_APP_NAME, [env.FULL_VERSION, 'latest'], [], params.isRelease, ".")
}
}
}
}
post {
success {
recordIssues enabledForFailure: true, tools: [msBuild()]
}
}
}
stage('WEB: Build and publish docker image') {
steps {
script {
println("WEB: Building and publishing docker image version: ${env.FULL_VERSION}")
buildAndPushDockerImageWeb(WEB_APP_NAME, [env.FULL_VERSION, 'latest'], [], params.isRelease, ".");
}
}
}
}
}
stage('API and WEB: Push helm chart') {
steps {
println("API and WEB: Building helm chart version: ${env.FULL_VERSION}")
buildHelmChart(PROJECT_CHARTNAME, env.FULL_VERSION)
}
}
stage('API og WEB - Snapshot: Set version') {
when {
expression { !params.isRelease }
}
steps {
script {
env.IMAGE_TAG = env.FULL_VERSION
}
}
}
stage('API og WEB: Deploy to dev') {
when {
anyOf {
branch 'master'
branch 'main'
}
expression { !params.isRelease }
}
steps {
build job: 'deployToDev', parameters: [string(name: 'chartName', value: PROJECT_CHARTNAME), string(name: 'version', value: env.FULL_VERSION)], wait: false, propagate: false
}
}
stage('API og WEB: Release. Set next version and push to git') {
when {
allOf {
expression { params.isRelease }
expression { return env.NEXT_VERSION }
expression { return env.FULL_VERSION }
}
}
steps {
gitCheckout("main")
gitTag(isRelease, env.FULL_VERSION)
prepareDotNetNoBuild(env.NEXT_VERSION)
script {
currentBuild.description = "${env.user} released version ${env.FULL_VERSION}"
}
}
post {
success {
gitPush()
createGithubRelease env.REPO_NAME, params.reviewer, params.releaseNotes, env.CURRENT_VERSION, env.user
}
}
}
}
post {
always {
dir("${PROJECT_API_FOLDER}\\bin") {
deleteDir()
}
dir("${PROJECT_WEB_FOLDER}\\bin") {
deleteDir()
}
dir("${PROJECT_TEST}\\bin") {
deleteDir()
}
dir("${env.TMPDIR}") {
deleteDir()
}
deleteDir()
}
}
}
def versionPattern() {
return java.util.regex.Pattern.compile("^(\\d+)\\.(\\d+)\\.(\\d+)(.*)?")
}
def findVersionPrefix() {
def files = findCsprojFiles()
def versions = files.collect {
echo("Checking ${it}")
return extractVersion(readFile(file: it.getPath().trim(), encoding: 'UTF-8'))
}.findAll {
return it != null && ! it.trim().isEmpty()
}
echo "Found ${versions.size()} versions"
if(versions.size() > 0 && versions[0] != "") {
def currentVersion = versions[0]
echo "Version: ${currentVersion}"
return currentVersion
} else {
throw new Exception("No versionPrefix fond in csproj files")
}
}
def findCsprojFiles() {
def files = findFiles(glob: '**/*.csproj').findAll {
return it.getName().toUpperCase().contains("TEST") == false && it.getName().toUpperCase().contains("EXAMPLE") == false
}
if(files.size() == 0) {
throw new Exception("No csproj files found")
}
echo("Found ${files.size()} csproj files")
return files
}
@NonCPS
def extractVersion(xml) {
def debom = { data ->
if(data?.length() > 0 && data[0] == '\uFEFF') return data.drop(1) else return data
}
def xmlData = debom.call(xml)
def Project = new XmlSlurper().parseText(xmlData)
def version = Project['PropertyGroup']['VersionPrefix'].text().trim()
echo("Found version ${version}")
return version
}
def incrementVersion(versionString) {
def p = versionPattern()
def m = p.matcher(versionString)
if(m.find()) {
def major = m.group(1) as Integer
def minor = m.group(2) as Integer
def patch = m.group(3) as Integer
return "${major}.${minor}.${++patch}"
} else {
return null
}
}
def buildOpts(versionSuffix) {
if(versionSuffix == null || versionSuffix.trim().isEmpty()) {
echo("No build opts will be passed to dotnet build")
return ""
} else {
def opts = "--version-suffix ${versionSuffix}"
echo("Will add build opts: ${opts}")
return opts
}
}
def getTimestamp() {
return java.time.OffsetDateTime.now().format(java.time.format.DateTimeFormatter.ofPattern("yyyyMMddHHmmssSSS"))
}
def buildAndPushDockerImageWeb(String imageName, List tags = [], List dockerArgs = [], boolean isRelease = false, String path = ".") {
def repo = isRelease ? DOCKER_REPO_RELEASE : DOCKER_REPO
dir("web-ui") {
script {
println("Building WEB code in Docker image")
if (isRelease) {
repo = 'https://docker-local.artifactory.fiks.ks.no'
} else {
repo = 'https://docker-local-snapshots.artifactory.fiks.ks.no'
}
println("WEB: npm install")
docker.image('node:16').inside() {
sh '''
npm install
'''
sh '''
npm run build -- --mode production
'''
}
println("WEB: npm install finished")
docker.withRegistry(repo, 'artifactory-token-based') {
def customImage = docker.build("${imageName}", dockerArgs.collect { "--build-arg $it" }.join(' ') + " " + path)
println("Publishing WEB image")
tags.each {
customImage.push(it)
}
}
}
}
}