2019-10-10 16:10:06 +02:00
|
|
|
|
#!groovy
|
2019-09-04 03:06:34 +02:00
|
|
|
|
import org.yaml.snakeyaml.Yaml
|
2019-05-28 21:27:19 +02:00
|
|
|
|
|
2019-05-28 21:32:59 +02:00
|
|
|
|
// set the build options according to the Type of build
|
2019-11-23 03:53:24 +01:00
|
|
|
|
def (options, maven_local_repo_path, maven_settings_file, maven_parent_file) = ['', '', '', '']
|
2019-10-02 17:48:37 +02:00
|
|
|
|
def agent_root_folder = '/var/lib/jenkins/.m2'
|
2019-12-03 22:51:46 +01:00
|
|
|
|
def verbose = true
|
2019-12-09 21:31:21 +01:00
|
|
|
|
def resume = params.resume
|
2019-05-29 04:04:34 +02:00
|
|
|
|
if (params.Type == 'SNAPSHOT-DRY-RUN') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
echo "Configure Maven for SNAPSHOT-DRY-RUN artifacts"
|
|
|
|
|
options = ''
|
2019-10-01 18:17:50 +02:00
|
|
|
|
maven_local_repo_path = "local-snapshots"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
maven_settings_file = "jenkins-snapshots-dry-run-settings.xml"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
maven_parent_file = "jenkins-snapshots-dry-run-settings.xml"
|
2019-05-29 04:04:34 +02:00
|
|
|
|
}
|
2019-05-28 16:40:30 +02:00
|
|
|
|
if (params.Type == 'SNAPSHOT') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
echo "Configure Maven for SNAPSHOT artifacts"
|
|
|
|
|
options = ''
|
2019-10-01 18:17:50 +02:00
|
|
|
|
maven_local_repo_path = "local-snapshots"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
maven_settings_file = "jenkins-snapshots-settings.xml"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
maven_parent_file = "jenkins-snapshots-dry-run-settings.xml"
|
2019-05-28 21:27:19 +02:00
|
|
|
|
}
|
2019-05-29 04:04:34 +02:00
|
|
|
|
if (params.Type == 'RELEASE-DRY-RUN') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
echo "Configure Maven for RELEASE-DRY-RUN artifacts"
|
|
|
|
|
options = ''
|
2019-10-01 18:17:50 +02:00
|
|
|
|
maven_local_repo_path = "local-releases"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
maven_settings_file = "jenkins-releases-dry-run-settings.xml"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
maven_parent_file = "jenkins-releases-dry-run-settings.xml"
|
2019-05-29 04:04:34 +02:00
|
|
|
|
}
|
2019-10-09 22:06:19 +02:00
|
|
|
|
if (params.Type == 'STAGING') {
|
2019-10-17 03:09:11 +02:00
|
|
|
|
echo "Configure Maven for STAGING artifacts"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
options = ''
|
2019-10-01 18:17:50 +02:00
|
|
|
|
maven_local_repo_path = "local-staging"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
maven_settings_file = "jenkins-staging-settings.xml"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
maven_parent_file = "jenkins-staging-dry-run-settings.xml"
|
2019-08-14 06:00:57 +02:00
|
|
|
|
}
|
2019-05-28 16:40:30 +02:00
|
|
|
|
if (params.Type == 'RELEASE') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
echo "Configure Maven for RELEASE artifacts"
|
|
|
|
|
options = ''
|
2019-10-11 05:35:51 +02:00
|
|
|
|
maven_local_repo_path = "local-releases"
|
|
|
|
|
maven_settings_file = "jenkins-releases-settings.xml"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
maven_parent_file = "jenkins-releases-dry-run-settings.xml"
|
2019-05-28 15:38:19 +02:00
|
|
|
|
}
|
2019-10-17 03:09:11 +02:00
|
|
|
|
|
2019-05-29 04:54:34 +02:00
|
|
|
|
echo "Use settings file at ${maven_settings_file}"
|
|
|
|
|
echo "Use local repo at ${maven_local_repo_path}"
|
2019-09-25 15:15:52 +02:00
|
|
|
|
echo "Release number: ${params.gCube_release_version}"
|
2019-10-11 17:44:16 +02:00
|
|
|
|
echo "Clean up gcube local artifacts? ${params.cleanup_gcube_artifacts}"
|
|
|
|
|
echo "Clean up all local artifacts? ${params.cleanup_local_repo}"
|
2019-12-09 21:23:03 +01:00
|
|
|
|
echo "Resume from previous build? ${params.resume}"
|
2019-10-11 17:44:16 +02:00
|
|
|
|
|
2019-05-29 04:54:34 +02:00
|
|
|
|
|
2019-10-17 03:09:11 +02:00
|
|
|
|
//locate the release file
|
2019-09-04 03:52:47 +02:00
|
|
|
|
String releaseURL = "https://code-repo.d4science.org/gCubeCI/gCubeRelease/raw/branch/master/releases/gcube-${gCube_release_version}.yaml"
|
2019-09-30 05:49:56 +02:00
|
|
|
|
|
2019-12-03 22:51:46 +01:00
|
|
|
|
if (verbose)
|
|
|
|
|
println "Querying ${releaseURL}"
|
2019-09-30 05:49:56 +02:00
|
|
|
|
|
|
|
|
|
//load the release file
|
2019-09-04 03:06:34 +02:00
|
|
|
|
def text = releaseURL.toURL().getText()
|
|
|
|
|
|
|
|
|
|
//parsing
|
|
|
|
|
def jsonConfig = new Yaml().load(text)
|
2019-12-03 22:51:46 +01:00
|
|
|
|
if (verbose)
|
|
|
|
|
println jsonConfig.inspect()
|
2019-09-30 18:57:58 +02:00
|
|
|
|
assert jsonConfig.gCube_release.Version == params.gCube_release_version: "Release versions do not match!"
|
2019-09-25 15:15:52 +02:00
|
|
|
|
echo "Building gCube v. ${jsonConfig.gCube_release.Version}"
|
2019-12-03 22:51:46 +01:00
|
|
|
|
if (verbose) {
|
|
|
|
|
echo "Found components:"
|
|
|
|
|
jsonConfig.gCube_release.Components.each { println it.key }
|
|
|
|
|
}
|
2019-08-29 22:03:45 +02:00
|
|
|
|
|
2019-12-09 22:50:50 +01:00
|
|
|
|
def report_number = env.BUILD_NUMBER -1
|
2019-12-10 04:25:44 +01:00
|
|
|
|
def previous_report_file = "${agent_root_folder}/build_jobs.${report_number}.csv"
|
|
|
|
|
echo "Previous report file: ${previous_report_file}"
|
|
|
|
|
def jobs = parseJobs(previous_report_file)
|
|
|
|
|
for (job in jobs)
|
|
|
|
|
println job
|
|
|
|
|
|
2019-12-09 22:35:11 +01:00
|
|
|
|
|
2019-05-28 16:40:30 +02:00
|
|
|
|
pipeline {
|
2019-09-04 03:06:34 +02:00
|
|
|
|
|
|
|
|
|
// see https://jenkins.io/doc/book/pipeline/syntax/#agent
|
|
|
|
|
agent {
|
2019-10-10 16:10:06 +02:00
|
|
|
|
label 'CD'
|
2019-09-04 03:06:34 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// see https://jenkins.io/doc/book/pipeline/syntax/#environment
|
|
|
|
|
environment {
|
2019-10-12 04:51:59 +02:00
|
|
|
|
//make the JVM start a bit faster with basic just-in-time compilation of the code only (-XX:*)
|
|
|
|
|
//make maven running in a multi-thread fashion (16 threads, 2 threads on each Core)
|
2019-12-03 22:51:46 +01:00
|
|
|
|
MAVEN_OPTS = "${params.build_options}"
|
2019-10-10 18:32:40 +02:00
|
|
|
|
AGENT_ROOT_FOLDER = "${agent_root_folder}"
|
|
|
|
|
MAVEN_SETTINGS_FILE = "${maven_settings_file}"
|
2019-11-23 03:53:24 +01:00
|
|
|
|
MAVEN_PARENT_FILE = "${maven_parent_file}"
|
2019-10-09 22:06:19 +02:00
|
|
|
|
MAVEN_LOCAL_REPO = "${agent_root_folder}/${maven_local_repo_path}"
|
2019-10-11 17:56:03 +02:00
|
|
|
|
CLEANUP_GCUBE_REPO = "${params.cleanup_gcube_artifacts}"
|
2019-10-11 18:11:38 +02:00
|
|
|
|
REMOVE_LOCAL_REPO = "${params.cleanup_local_repo}"
|
2019-09-30 18:57:58 +02:00
|
|
|
|
GCUBE_RELEASE_NUMBER = "${params.gCube_release_version}"
|
2019-10-09 22:48:33 +02:00
|
|
|
|
PIPELINE_BUILD_NUMBER = "${env.BUILD_NUMBER}"
|
2019-10-22 21:05:13 +02:00
|
|
|
|
TYPE = "${params.Type}"
|
2019-12-09 21:23:03 +01:00
|
|
|
|
RESUME = "${params.resume}"
|
|
|
|
|
JOB_REPORT = "${agent_root_folder}/build_jobs.${env.BUILD_NUMBER}.csv"
|
2019-12-10 04:25:44 +01:00
|
|
|
|
PREVIOUS_JOB_REPORT = "${previous_report_file}"
|
2019-09-04 03:06:34 +02:00
|
|
|
|
}
|
2019-09-30 19:00:48 +02:00
|
|
|
|
|
2019-09-04 03:06:34 +02:00
|
|
|
|
// see https://jenkins.io/doc/book/pipeline/syntax/#parameters
|
|
|
|
|
parameters {
|
2019-06-11 21:32:57 +02:00
|
|
|
|
choice(name: 'Type',
|
2019-10-10 16:56:17 +02:00
|
|
|
|
choices: ['SNAPSHOT-DRY-RUN', 'SNAPSHOT', 'STAGING', 'RELEASE-DRY-RUN', 'RELEASE'],
|
2019-06-11 21:32:57 +02:00
|
|
|
|
description: 'The type of artifacts the build is expected to generate')
|
|
|
|
|
|
2019-09-04 03:06:34 +02:00
|
|
|
|
string(name: 'gCube_release_version',
|
2019-09-04 03:51:40 +02:00
|
|
|
|
defaultValue: 'x.y.z',
|
2019-09-04 03:06:34 +02:00
|
|
|
|
description: 'The number of the gCube release to build. Sample values: 4.14, 4.15, etc.')
|
2019-10-11 14:52:48 +02:00
|
|
|
|
|
2019-10-11 15:14:17 +02:00
|
|
|
|
booleanParam(name: 'cleanup_gcube_artifacts',
|
2019-12-03 22:51:46 +01:00
|
|
|
|
defaultValue: true,
|
|
|
|
|
description: 'Wipe out the gcube artifacts from the local maven repository before the builds?')
|
2019-10-11 15:14:17 +02:00
|
|
|
|
|
2019-10-11 14:52:48 +02:00
|
|
|
|
booleanParam(name: 'cleanup_local_repo',
|
2019-12-03 22:51:46 +01:00
|
|
|
|
defaultValue: true,
|
|
|
|
|
description: 'Wipe out the local maven repository before the builds?')
|
2019-12-09 21:23:03 +01:00
|
|
|
|
|
|
|
|
|
booleanParam(name: 'resume',
|
|
|
|
|
defaultValue: false,
|
|
|
|
|
description: 'Resume from previous build?')
|
2019-05-28 16:40:30 +02:00
|
|
|
|
}
|
2019-05-30 05:24:09 +02:00
|
|
|
|
|
2019-05-30 02:46:15 +02:00
|
|
|
|
//see https://jenkins.io/doc/book/pipeline/syntax/#stages
|
2019-05-28 16:40:30 +02:00
|
|
|
|
stages {
|
2019-12-03 22:51:46 +01:00
|
|
|
|
stage('initialize') {
|
2019-05-30 04:31:08 +02:00
|
|
|
|
steps {
|
2019-05-30 05:25:00 +02:00
|
|
|
|
sh '''
|
2019-12-09 21:42:24 +01:00
|
|
|
|
echo "REMOVE_LOCAL_REPO: ${REMOVE_LOCAL_REPO}"
|
|
|
|
|
echo "CLEANUP_GCUBE_REPO: ${CLEANUP_GCUBE_REPO}"
|
|
|
|
|
if [ "$CLEANUP_GCUBE_REPO" = "true" ]; then
|
|
|
|
|
echo "Remove gCube artifacts from local repository"
|
|
|
|
|
rm -rf $MAVEN_LOCAL_REPO/org/gcube
|
|
|
|
|
fi
|
|
|
|
|
if [ "$REMOVE_LOCAL_REPO" = "true" ]; then
|
|
|
|
|
echo "Create a fresh local repository"
|
|
|
|
|
rm -rf $MAVEN_LOCAL_REPO
|
|
|
|
|
mkdir -p $MAVEN_LOCAL_REPO
|
|
|
|
|
fi
|
|
|
|
|
mv "${AGENT_ROOT_FOLDER}/settings.xml" "${AGENT_ROOT_FOLDER}/settings.${PIPELINE_BUILD_NUMBER}"
|
|
|
|
|
cp "${AGENT_ROOT_FOLDER}/${MAVEN_SETTINGS_FILE}" "${AGENT_ROOT_FOLDER}/settings.xml"
|
|
|
|
|
echo "Done with local repository and settings"
|
|
|
|
|
|
|
|
|
|
#build report
|
|
|
|
|
echo "#Build ${PIPELINE_BUILD_NUMBER}" > ${AGENT_ROOT_FOLDER}/build_commits.csv
|
|
|
|
|
echo "#Release ${GCUBE_RELEASE_NUMBER}" >> ${AGENT_ROOT_FOLDER}/build_commits.csv
|
|
|
|
|
date=`date`
|
|
|
|
|
echo "#StartTime ${date}" >> ${AGENT_ROOT_FOLDER}/build_commits.csv
|
|
|
|
|
echo -e "GroupID,ArtifactID,Version,SCM URL,Build Number,Distribution URL,Filename,Packaging" >> ${AGENT_ROOT_FOLDER}/build_commits.csv
|
|
|
|
|
|
|
|
|
|
#job report
|
|
|
|
|
echo "#Build ${PIPELINE_BUILD_NUMBER}" > $JOB_REPORT
|
|
|
|
|
echo "#StartTime ${date}" >> $JOB_REPORT
|
|
|
|
|
echo -e "JobName,Status" >> $JOB_REPORT
|
2019-05-30 05:25:00 +02:00
|
|
|
|
'''
|
2019-05-29 04:54:34 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-25 15:43:37 +02:00
|
|
|
|
// the maven-parent needs to be built (once) at each execution
|
|
|
|
|
stage('build maven-parent') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
steps {
|
2019-12-09 22:13:36 +01:00
|
|
|
|
script {
|
2019-12-09 22:35:11 +01:00
|
|
|
|
if ( ("${resume}" == 'true') && (jobs['maven-parent'] == 'SUCCESS') ) {
|
2019-12-10 03:25:54 +01:00
|
|
|
|
echo "Skipping maven-parent"
|
2019-12-09 22:00:42 +01:00
|
|
|
|
// propagate success
|
|
|
|
|
sh "echo -e \\\"maven-parent,SUCCESS\\\">> $JOB_REPORT"
|
|
|
|
|
} else {
|
|
|
|
|
def gjob = build(job: 'maven-parent', wait: true, propagate: true,
|
|
|
|
|
parameters: [[$class: 'StringParameterValue', name: 'gcube_settings', value: "${maven_parent_file}"],
|
|
|
|
|
[$class: 'StringParameterValue', name: 'local_repo', value: "${maven_local_repo_path}"],
|
|
|
|
|
[$class: 'LabelParameterValue', name: 'exec_label', label: "CD", nodeEligibility: [$class: 'AllNodeEligibility']]
|
|
|
|
|
])
|
|
|
|
|
sh "echo -e \\\"maven-parent,${gjob.getResult()}\\\">> $JOB_REPORT"
|
|
|
|
|
echo "Done with maven-parent"
|
|
|
|
|
}
|
2019-12-09 22:02:48 +01:00
|
|
|
|
|
2019-12-09 22:13:36 +01:00
|
|
|
|
}
|
2019-09-30 18:57:58 +02:00
|
|
|
|
}
|
2019-09-25 15:43:37 +02:00
|
|
|
|
}
|
2019-12-03 22:51:46 +01:00
|
|
|
|
stage('build components') {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
steps {
|
2019-12-03 22:51:46 +01:00
|
|
|
|
script {
|
|
|
|
|
jsonConfig.gCube_release.Components.each { group_name, component_list ->
|
|
|
|
|
stage("build ${group_name} components") {
|
|
|
|
|
buildComponents items: component_list?.collect { "${it.name}" },
|
2019-12-10 03:42:10 +01:00
|
|
|
|
"${maven_settings_file}", "${maven_local_repo_path}", jobs
|
2019-12-03 22:51:46 +01:00
|
|
|
|
echo "Done with ${group_name} components"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-10-03 17:25:08 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-30 18:57:58 +02:00
|
|
|
|
}
|
2019-10-10 16:43:24 +02:00
|
|
|
|
|
|
|
|
|
// post-build actions
|
2019-12-03 22:51:46 +01:00
|
|
|
|
post {
|
2019-10-10 16:43:24 +02:00
|
|
|
|
always {
|
2019-12-03 22:51:46 +01:00
|
|
|
|
script {
|
|
|
|
|
sh '''
|
2019-10-10 18:32:40 +02:00
|
|
|
|
mv "${AGENT_ROOT_FOLDER}/settings.${PIPELINE_BUILD_NUMBER}" "${AGENT_ROOT_FOLDER}/settings.xml"
|
2019-10-22 05:01:14 +02:00
|
|
|
|
mv ${AGENT_ROOT_FOLDER}/build_commits.csv ${AGENT_ROOT_FOLDER}/build_commits.${PIPELINE_BUILD_NUMBER}.csv
|
2019-10-22 14:33:50 +02:00
|
|
|
|
cp ${AGENT_ROOT_FOLDER}/build_commits.${PIPELINE_BUILD_NUMBER}.csv .
|
2019-10-10 18:26:56 +02:00
|
|
|
|
'''
|
2019-12-03 22:51:46 +01:00
|
|
|
|
}
|
|
|
|
|
echo 'The default maven settings have been restored'
|
2019-10-10 18:32:40 +02:00
|
|
|
|
|
2019-10-10 16:43:24 +02:00
|
|
|
|
}
|
|
|
|
|
success {
|
|
|
|
|
echo 'The pipeline worked!'
|
2019-10-22 21:05:13 +02:00
|
|
|
|
emailext to: 'jenkinsbuilds@d4science.org',
|
2019-12-03 22:51:46 +01:00
|
|
|
|
subject: "[Jenkins build D4S] build ${currentBuild.fullDisplayName} worked",
|
|
|
|
|
body: "Build time: ${currentBuild.durationString}. See ${env.BUILD_URL}"
|
2019-10-22 21:05:13 +02:00
|
|
|
|
emailext attachmentsPattern: "**/*.${PIPELINE_BUILD_NUMBER}.csv",
|
2019-12-03 22:51:46 +01:00
|
|
|
|
to: 'jenkinsreleases@d4science.org',
|
|
|
|
|
subject: "${TYPE} report for release ${GCUBE_RELEASE_NUMBER} (build #${PIPELINE_BUILD_NUMBER})",
|
|
|
|
|
body: "${currentBuild.fullDisplayName}. Build time: ${currentBuild.durationString}. See ${env.BUILD_URL}"
|
2019-10-10 16:43:24 +02:00
|
|
|
|
}
|
|
|
|
|
failure {
|
|
|
|
|
echo 'The pipeline has failed'
|
2019-10-22 05:28:35 +02:00
|
|
|
|
emailext attachLog: true,
|
2019-12-03 22:51:46 +01:00
|
|
|
|
to: 'jenkinsbuilds@d4science.org',
|
|
|
|
|
subject: "[Jenkins build D4S] build ${currentBuild.fullDisplayName} failed",
|
|
|
|
|
body: "Something is wrong with ${env.BUILD_URL}"
|
2019-10-10 16:43:24 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
2019-05-28 15:00:59 +02:00
|
|
|
|
}
|
2019-09-04 03:06:34 +02:00
|
|
|
|
|
2019-12-10 03:42:10 +01:00
|
|
|
|
def buildComponents(args, maven_settings_file, maven_local_repo_path, jobs) {
|
2019-09-04 19:11:51 +02:00
|
|
|
|
if (args.items) {
|
2019-09-30 18:57:58 +02:00
|
|
|
|
parallel args.items?.collectEntries { name ->
|
|
|
|
|
["${name}": {
|
2019-12-10 03:37:23 +01:00
|
|
|
|
if (name && !"NONE".equalsIgnoreCase(name)) {
|
|
|
|
|
if ( ("${resume}" == 'true') && (jobs["${name}"] == 'SUCCESS') ) {
|
|
|
|
|
echo "Skipping ${name}"
|
|
|
|
|
// propagate success
|
|
|
|
|
sh "echo -e \\\"${name},SUCCESS\\\">> $JOB_REPORT"
|
|
|
|
|
} else {
|
2019-12-10 03:47:34 +01:00
|
|
|
|
def gjob = build(job: name, wait: true, propagate: true,
|
2019-12-09 21:17:37 +01:00
|
|
|
|
parameters: [[$class: 'StringParameterValue', name: 'gcube_settings', value: "${maven_settings_file}"],
|
|
|
|
|
[$class: 'StringParameterValue', name: 'local_repo', value: "${maven_local_repo_path}"],
|
|
|
|
|
[$class: 'LabelParameterValue', name: 'exec_label', label: "CD", nodeEligibility: [$class: 'AllNodeEligibility']]
|
|
|
|
|
])
|
2019-12-10 04:25:44 +01:00
|
|
|
|
sh "echo -e \\\"${name},${gjob.getResult()}\\\" >> $JOB_REPORT"
|
2019-12-10 03:37:23 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-12-08 21:10:38 +01:00
|
|
|
|
}
|
2019-12-09 21:17:37 +01:00
|
|
|
|
]
|
2019-12-08 00:19:14 +01:00
|
|
|
|
}
|
2019-12-09 21:17:37 +01:00
|
|
|
|
}
|
2019-12-09 21:49:33 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
Check if the job was successfully completed in teh given report.
|
|
|
|
|
*/
|
|
|
|
|
@NonCPS
|
2019-12-09 22:35:11 +01:00
|
|
|
|
def parseJobs(job_file) {
|
|
|
|
|
def jobs = [:]
|
2019-12-09 22:50:50 +01:00
|
|
|
|
try {
|
|
|
|
|
new File(job_file).splitEachLine(',') { columns ->
|
2019-12-10 03:47:34 +01:00
|
|
|
|
if (columns[0].startsWith('#') || columns[0].startsWith('JobName'))
|
2019-12-09 22:50:50 +01:00
|
|
|
|
return
|
|
|
|
|
jobs["${columns[0]}"] = columns[1]
|
|
|
|
|
}
|
2019-12-09 22:54:48 +01:00
|
|
|
|
} catch(Exception e) {println "Previous job report not available"}
|
2019-12-09 22:35:11 +01:00
|
|
|
|
jobs;
|
2019-09-04 03:06:34 +02:00
|
|
|
|
}
|