- vừa được xem lúc

[CI-CD] Triển khai CICD đơn giản cho Raspberry Pi

0 0 24

Người đăng: Thiên Đông Trần

Theo Viblo Asia

Trên công ty có ông bạn mang con Raspberry Pi B+ lên công ty dựng server để giám sát mạng, thế là mình xin ké 1 chân vào để deploy gì đấy vui vui 😁

Thế là bắt đầu tìm ý tưởng lặt vặt dựng con server nodejs gì đấy lên cho vui

Nhưng đến đoạn deploy lên, thì thấy nếu update code phải pull git về rồi chạy lại mất thời gian quá, bèn nghĩ ra kế để deploy tự động 1 cách đơn giản nhất

Ý tưởng bắt đầu nhen nhóm bằng việc từ source code mình dùng docker build ra image và push thẳng lên docker hub, và trên raspberry pi mình chạy script python để kiểm tra docker hub có update phiên bản image mới ko, nếu có thì pull image về và restart lại container 😅

Repository cho các bạn tham khảo: Repository Demo

Triển thôi nào 🤪

Tạo docker file để build image từ source nodejs

FROM node:latest WORKDIR /usr/src/app COPY package*.json ./ RUN npm install COPY dist/ ./dist/ EXPOSE 3000 CMD ["node", "dist/app.js"] 

Tạo github workflow để build và push image lên Docker Hub

name: Deploy to Docker Hub on: push: branches: - main jobs: send-notification-started: runs-on: ubuntu-latest steps: - name: Send Telegram Notification env: TELEGRAM_BOT_TOKEN: ${{ secrets.TELEGRAM_BOT_TOKEN }} TELEGRAM_CHAT_ID: "${{ secrets.TELEGRAM_GROUP_DEPLOYMENTS }}" run: | curl -X POST \ https://api.telegram.org/bot${TELEGRAM_BOT_TOKEN}/sendMessage \ -d chat_id=${TELEGRAM_CHAT_ID} \ -d text="🚀 <b>Sms Bot</b> Deployment has started!" \ -d parse_mode=HTML build: runs-on: ubuntu-latest steps: - name: Checkout code uses: actions/checkout@v2 - name: Set up Node.js uses: actions/setup-node@v2 with: node-version: '14' - name: Install dependencies and build run: | npm install npm run build - name: Login to Docker Hub uses: docker/login-action@v1 with: username: ${{ secrets.DOCKER_USERNAME }} password: ${{ secrets.DOCKER_PASSWORD }} - name: Set up QEMU run: | sudo apt-get update sudo apt-get install -y qemu-user-static - name: Set up Docker Buildx uses: docker/setup-buildx-action@v1 - name: Build image for aarch64 run: | docker buildx create --name builder --use docker buildx inspect builder --bootstrap docker buildx build --platform linux/arm64 -t ${{ secrets.DOCKER_HUB_IMAGE_NAME }}:latest --push . send-notification-successful: needs: build runs-on: ubuntu-latest if: ${{ success() && needs.build.result == 'success' }} steps: - name: Send Telegram Notification env: TELEGRAM_BOT_TOKEN: ${{ secrets.TELEGRAM_BOT_TOKEN }} TELEGRAM_CHAT_ID: "${{ secrets.TELEGRAM_GROUP_DEPLOYMENTS }}" run: | curl -X POST \ https://api.telegram.org/bot${TELEGRAM_BOT_TOKEN}/sendMessage \ -d chat_id=${TELEGRAM_CHAT_ID} \ -d text="🎉 <b>Sms Bot</b> Deployment to docker hub was successful!" \ -d parse_mode=HTML send-notification-failed: needs: build runs-on: ubuntu-latest if: ${{ failure() && needs.build.result == 'failure' }} steps: - name: Send Telegram Notification env: TELEGRAM_BOT_TOKEN: ${{ secrets.TELEGRAM_BOT_TOKEN }} TELEGRAM_CHAT_ID: "${{ secrets.TELEGRAM_GROUP_DEPLOYMENTS }}" run: | curl -X POST \ https://api.telegram.org/bot${TELEGRAM_BOT_TOKEN}/sendMessage \ -d chat_id=${TELEGRAM_CHAT_ID} \ -d text="❌ Oh no! <b>Sms Bot</b> Deployment failed! There might be something wrong with the process. Please check it out! 🛠️🔍" \ -d parse_mode=HTML

Script python để pull và restart Docker image

import docker
import time
import logging
from logging.handlers import RotatingFileHandler
from telegram import Bot
import requests TELEGRAM_BOT_TOKEN = ''
TELEGRAM_CHAT_ID = '' def send_telegram_message(message): bot = Bot(token=TELEGRAM_BOT_TOKEN) bot.send_message(chat_id=TELEGRAM_CHAT_ID, text=message, parse_mode='HTML') log_formatter = logging.Formatter('%(asctime)s - %(levelname)s: %(message)s')
log_handler = RotatingFileHandler('logs.log', maxBytes=1024 * 1024, backupCount=5)
log_handler.setFormatter(log_formatter)
logger = logging.getLogger()
logger.addHandler(log_handler)
logger.setLevel(logging.INFO) def get_image_info(image_name): try: response = requests.get(f"https://hub.docker.com/v2/repositories/{image_name}/tags/latest") if response.status_code == 200: image_info = response.json() return image_info else: print(f"Failed to get image info, status code: {response.status_code}") return None except requests.RequestException as e: print(f"Request error: {e}") return None def check_and_update_image(container_name, image_name): client = docker.from_env() logger.info(f"Start pull image..") try: version = get_image_info(image_name) digest = image_name + '@' + version['digest'] container = client.containers.get(container_name) current_image_id = container.image.id current_image_digest = container.image.attrs['RepoDigests'][0] if 'RepoDigests' in container.image.attrs else None logger.info(f"current_image_id {current_image_id}") logger.info(f"current_image_digest {current_image_digest}") logger.info(f"digest {digest}") if current_image_digest != digest: logger.info("Updating...") send_telegram_message("🚀 <b>Auto Pull Docker - Raspberry pi</b> Deployment has started!") client.images.pull(image_name) container.stop() container.remove() client.containers.run(image_name, detach=True, name=container_name, volumes={ '/home/dongtran/py/.env': {'bind': '/usr/src/app/.env', 'mode': 'rw'} }) logger.info("Container update successful.") send_telegram_message("🚀 <b>Auto Pull Docker - Raspberry pi</b> Deployment on raspberry pi sucessful!") else: logger.info("No action.") except docker.errors.NotFound as e: logger.error(f"Container '{container_name}' not found: {e}") return except docker.errors.APIError as e: logger.error(f"APIError: {e}") return except docker.errors.ImageNotFound: logger.error(f"Image '{image_name}' not exist") return if __name__ == "__main__": container_name = "" image_name = "" while True: check_and_update_image(container_name, image_name) time.sleep(30)

Như vậy chúng ta đã có thể deploy ứng dụng 1 cách tự động trên raspberry pi rồi 🥲

Các bạn có thắc mắc comment bên dưới nha 😍😍

Bình luận

Bài viết tương tự

- vừa được xem lúc

CI-CD Lab - Sử dụng Jenkins, Nginx deploy dự án ReactJS (UmiJS) lên AWS EC2

1. Lập trình viên không nên chỉ biết code. Lập trình viên phải biết sửa máy giặt, sửa điện, sửa ống nước, có người yêu ,.... và phải biết triển khai code lên internet. Khi đã biết triển khai code lên internet (được sếp đánh giá rất cao rồi ) thì lại phải học tiếp CI - CD để build và test code tự độn

0 0 259

- vừa được xem lúc

[CI-CD] Triển Khai Ứng Dụng Node.js lên Cloud Run với GitHub Actions

Cloud Run của Google Cloud Platform (GCP) cung cấp một nền tảng serverless để chạy các container được quản lý tự động. Kết hợp với GitHub Actions, bạn có thể tự động hóa quy trình triển khai (CI/CD) c

0 0 25

- vừa được xem lúc

[CI-CD] Triển khai Ứng dụng trên Nginx với Kubernetes Engine

Tiếp nối với bài hôm trước Triển khai Ứng dụng lên Kubernetes Engine của Google Cloud Platform thì hôm nay mình sẽ giới thiệu cách để sử dụng thêm Nginx . Repository để các bạn có thể kéo về tham khả

0 0 31

- vừa được xem lúc

Tổng hợp kiến thức cần biết về Docker và Docker Swarm (Phần 1 - Tổng quan về Docker)

I. Docker là gì.

0 0 41

- vừa được xem lúc

PHP websites sử dụng Docker Containers với PHP, Apache2 và MySQL

Xem lại series về các lệnh cơ bản trong docker: PHẦN 1, PHẦN 2, PHẦN 3. Docker là một nền tảng để cung cấp cách để building, deploying và running ứng dụng dễ dàng hơn bằng cách sử dụng các containers.

0 0 63

- vừa được xem lúc

[Docker] Golang Create and Build Image đơn giản.

Giới Thiệu. Trong một vài năm gần đây thì ngôn ngữ lập trình Golang cực kì hot, vì hot nến thị trường công việc cũng như tuyển dụng vô cùng đa dạng.

0 0 38