189 lines
7.2 KiB
Groovy
189 lines
7.2 KiB
Groovy
pipeline {
|
|
agent any
|
|
|
|
environment {
|
|
PROD_ENV = "/opt/env/.env.uzxarid"
|
|
IMAGE_NAME = "uzxarid"
|
|
TEST_TAG = "test"
|
|
PROD_TAG = "latest"
|
|
CONTAINER_DB = "uzxarid_db_test"
|
|
CONTAINER_WEB = "uzxarid_web_test"
|
|
CONTAINER_REDIS = "uzxarid_redis_test"
|
|
STACK_NAME = "uzxarid"
|
|
}
|
|
|
|
stages {
|
|
stage('Check Commit Message') {
|
|
steps {
|
|
script {
|
|
def commitMsg = sh(
|
|
script: "git log -1 --pretty=%B",
|
|
returnStdout: true
|
|
).trim()
|
|
|
|
if (commitMsg.contains("[ci skip]")) {
|
|
echo "Commit message contains [ci skip], aborting pipeline 🚫"
|
|
currentBuild.result = 'ABORTED'
|
|
error("Pipeline aborted because of [ci skip]")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
stage('Checkout Code') {
|
|
steps {
|
|
git branch: 'dev', credentialsId: 'muhammadvadud', url: 'https://gitea.felixits.uz/uzxarid/backend.git'
|
|
}
|
|
}
|
|
stage('Prepare') {
|
|
steps {
|
|
script {
|
|
env.GIT_MESSAGE = sh(
|
|
script: "git log -1 --pretty=%B ${env.GIT_COMMIT}",
|
|
returnStdout: true
|
|
).trim()
|
|
}
|
|
}
|
|
}
|
|
stage("Update files") {
|
|
when {
|
|
expression { currentBuild.currentResult == "SUCCESS" }
|
|
}
|
|
steps {
|
|
withCredentials([usernamePassword(credentialsId: 'dockerhub', usernameVariable: 'DOCKER_USER', passwordVariable: 'DOCKER_PASS')]) {
|
|
sh """
|
|
sed -i 's|image: ${DOCKER_USER}/${IMAGE_NAME}:.*|image: ${DOCKER_USER}/${IMAGE_NAME}:${BUILD_NUMBER}|' stack.yaml
|
|
sed -i 's/return HttpResponse("OK.*"/return HttpResponse("OK: #${GIT_COMMIT}"/' config/urls.py
|
|
"""
|
|
// git config --global user.email "admin@jscorp.uz"
|
|
// git config --global user.name "Jenkins"
|
|
// if ! git diff --quiet stack.yaml; then
|
|
// git add stack.yaml
|
|
// git commit -m "feat(swarm) Update image tag to ${BUILD_NUMBER} [ci skip]"
|
|
// git push origin main
|
|
// else
|
|
// echo "No changes in stack.yaml"
|
|
// fi
|
|
}
|
|
|
|
}
|
|
}
|
|
stage('Build Image') {
|
|
steps {
|
|
sh '''
|
|
if [ -e ${PROD_ENV} ]; then
|
|
echo env exists
|
|
else
|
|
mkdir -p $(dirname ${PROD_ENV})
|
|
cp ./.env.example ${PROD_ENV}
|
|
fi
|
|
cp ${PROD_ENV} ./.env
|
|
'''
|
|
sh """
|
|
docker build -t ${IMAGE_NAME}:${PROD_TAG} --build-arg SCRIPT=entrypoint-server.sh -f ./docker/Dockerfile.web .
|
|
"""
|
|
}
|
|
}
|
|
|
|
|
|
stage('Start Test DB') {
|
|
steps {
|
|
sh """
|
|
docker run -d --rm --name ${CONTAINER_DB} -e POSTGRES_PASSWORD=postgres -e POSTGRES_DB=testdb postgres:16
|
|
docker run -d --rm --name ${CONTAINER_REDIS} redis
|
|
echo "⏳ Waiting for database..."
|
|
for i in {1..30}; do
|
|
if docker exec ${CONTAINER_DB} pg_isready -U postgres >/dev/null 2>&1; then
|
|
echo "✅ Database ready"
|
|
break
|
|
fi
|
|
echo "Database not ready yet... retrying..."
|
|
sleep 2
|
|
done
|
|
"""
|
|
}
|
|
}
|
|
|
|
stage('Run Migrations & Tests') {
|
|
steps {
|
|
sh """
|
|
docker run --rm --name ${CONTAINER_WEB} --link ${CONTAINER_DB}:db --link ${CONTAINER_REDIS}:redis \
|
|
-e DB_HOST=db \
|
|
-e DB_PORT=5432 \
|
|
-e DB_NAME=testdb \
|
|
-e DB_USER=postgres \
|
|
-e DB_PASSWORD=postgres \
|
|
-e DJANGO_SETTINGS_MODULE=config.settings.test \
|
|
${IMAGE_NAME}:${PROD_TAG} \
|
|
sh -c "python manage.py migrate && pytest -v"
|
|
"""
|
|
}
|
|
}
|
|
|
|
stage('Publish to DockerHub') {
|
|
when {
|
|
expression { currentBuild.currentResult == "SUCCESS" }
|
|
}
|
|
steps {
|
|
withCredentials([usernamePassword(credentialsId: 'dockerhub', usernameVariable: 'DOCKER_USER', passwordVariable: 'DOCKER_PASS')]) {
|
|
sh '''
|
|
echo "${DOCKER_PASS}" | docker login -u "${DOCKER_USER}" --password-stdin
|
|
docker tag ${IMAGE_NAME}:${PROD_TAG} ${DOCKER_USER}/${IMAGE_NAME}:${BUILD_NUMBER}
|
|
docker tag ${IMAGE_NAME}:${PROD_TAG} ${DOCKER_USER}/${IMAGE_NAME}:${PROD_TAG}
|
|
docker push ${DOCKER_USER}/${IMAGE_NAME}:${BUILD_NUMBER}
|
|
docker push ${DOCKER_USER}/${IMAGE_NAME}:${PROD_TAG}
|
|
'''
|
|
}
|
|
}
|
|
}
|
|
stage('Deploy stack') {
|
|
when {
|
|
expression { currentBuild.currentResult == "SUCCESS" }
|
|
}
|
|
steps {
|
|
withCredentials([usernamePassword(credentialsId: 'dockerhub', usernameVariable: 'DOCKER_USER', passwordVariable: 'DOCKER_PASS')]) {
|
|
sh '''
|
|
docker stack deploy -c stack.yaml ${STACK_NAME}
|
|
'''
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
post {
|
|
always {
|
|
sh """
|
|
docker stop ${CONTAINER_DB} || true
|
|
docker stop ${CONTAINER_REDIS} || true
|
|
"""
|
|
echo "Pipeline finished: ${currentBuild.currentResult}"
|
|
}
|
|
|
|
success {
|
|
withCredentials([
|
|
string(credentialsId: 'bot-token', variable: 'BOT_TOKEN'),
|
|
string(credentialsId: 'chat-id', variable: 'CHAT_ID')
|
|
]) {
|
|
sh '''
|
|
curl -s -X POST https://api.telegram.org/bot${BOT_TOKEN}/sendMessage \
|
|
-d chat_id=${CHAT_ID} \
|
|
-d text="✅ SUCCESS: ${JOB_NAME} #${BUILD_NUMBER}\nRepo: ${GIT_URL}\nBranch: ${GIT_BRANCH}\nCommit: ${GIT_COMMIT}\nMessage: ${GIT_MESSAGE}"
|
|
'''
|
|
}
|
|
}
|
|
|
|
failure {
|
|
withCredentials([
|
|
string(credentialsId: 'bot-token', variable: 'BOT_TOKEN'),
|
|
string(credentialsId: 'chat-id', variable: 'CHAT_ID')
|
|
]) {
|
|
sh '''
|
|
curl -s -X POST https://api.telegram.org/bot${BOT_TOKEN}/sendMessage \
|
|
-d chat_id=${CHAT_ID} \
|
|
-d text="🚨 FAILED: ${JOB_NAME} #${BUILD_NUMBER}\nRepo: ${GIT_URL}\nBranch: ${GIT_BRANCH}\nCommit: ${GIT_COMMIT}\nMessage: ${GIT_MESSAGE}"
|
|
'''
|
|
}
|
|
}
|
|
}
|
|
}
|