Deploying your project from GitHub to VPS using Jenkins for CI/CD
Introduction
In this quick tutorial, I’ll show you how I implemented CI/CD in my Nextjs 14 project on my new VPS server. This blog post is also valid for react projects or applications using nodejs. So you might want to ask, what is Jenkins or CI/CD? So you have pushed your fancy little project to your new VPS server, but you need a way to continually push updates/fixes to your application, that is where Jenkins come in for Continuous Integration/Continuous Delivery (CI/CD).
HEADS UP: An ideal hardware requirement for this is a VPS with at least 2gb of ram. Anything lower than that? you may run into memory issues during build processes. This is a step by step tutorial, please follow religiously.
Getting a VPS server
If you are yet to purchase a VPS, you can quickly purchase from one these cloud providers. But I would recommend getting one (here). It’s cheap and very easy to get started. After signing up, your new VPS credentials (ip address and login) will be sent to your email address.
Initial Server Setup
First, SSH into your server
Bashssh root@your_server_ip
Create a New User
You may want to create a new user, if you prefer.
Bashadduser your_username
Add user to sudo group
Bashusermod -aG sudo your_username
Switch to the new user
Bashsu - your_username
Basic Security Setup
Update system packages
Bashsudo apt update & upgrade -y
Install git and setup firewall
Bashsudo apt install -y curl git ufw
Configuring the firewall to allow some ports we’ll need
Bashsudo ufw allow OpenSSH sudo ufw allow 80 sudo ufw allow 8080 sudo ufw allow 443 sudo ufw enable
Install Node and PM2
We’ll be using nodejs for our Next.js deployment) and PM2 for production process management, you can read more about pm2 here
Bash1# Install Node.js using nvm (Node Version Manager) 2curl -o- https://raw.githubusercontent.com/nvm-sh/nvm/v0.39.0/install.sh | bash 3source ~/.bashrc 4 5# Install Node.js LTS version 6nvm install --lts 7 8# Install PM2 globally 9npm install -g pm2
Install & Configure Nginx
Bash1# Install Nginx 2sudo apt install nginx -y 3 4# Start Nginx 5sudo systemctl start nginx 6sudo systemctl enable nginx
Let’s create nginx configuration for your app
Bashsudo nano /etc/nginx/sites-available/your-app
Add this to the configuration (replace your-domain.com with your actual domain name):
Bash1server { 2 listen 80; 3 server_name your-domain.com www.your-domain.com; 4 5 location / { 6 proxy_pass http://localhost:3000; 7 proxy_http_version 1.1; 8 proxy_set_header Upgrade $http_upgrade; 9 proxy_set_header Connection 'upgrade'; 10 proxy_set_header Host $host; 11 proxy_cache_bypass $http_upgrade; 12 } 13}
Enable this configuration
Bashsudo ln -s /etc/nginx/sites-available/your-app /etc/nginx/sites-enabled/ sudo nginx -t sudo systemctl restart nginx
Setup Jenkins
Now let’s install Jenkins and other dependencies
Bashsudo apt update sudo apt upgrade -y
Install Java (It’s a Jenkins requirement)
Bashsudo apt install openjdk-17-jre-headless -y
Install Jenkins
Bash1# Add Jenkins repository 2curl -fsSL https://pkg.jenkins.io/debian-stable/jenkins.io-2023.key | sudo tee \ 3 /usr/share/keyrings/jenkins-keyring.adk > /dev/null 4 5echo deb [signed-by=/usr/share/keyrings/jenkins-keyring.adk] \ 6 https://pkg.jenkins.io/debian-stable binary/ | sudo tee \ 7 /etc/apt/sources.list.d/jenkins.list > /dev/null 8 9# Install Jenkins 10sudo apt update 11sudo apt install jenkins -y
Start and enable Jenkins
Bashsudo systemctl start jenkins sudo systemctl enable jenkins
Now we need to login to jenkins
- 
Access Jenkins at http://your_server_ip:8080
- 
Retrieve initial admin password: 
Bashsudo cat /var/lib/jenkins/secrets/initialAdminPassword
- 
Install recommended plugins 
- 
Create an admin user 
- 
Create a file name called Jenkinsfileat the root in your GitHub repository and paste the below in it.
Bash1pipeline { 2 agent any 3 4 tools { 5 nodejs "NodeJS" 6 } 7 8 stages { 9 stage('Checkout') { 10 steps { 11 checkout scm 12 } 13 } 14 15 stage('Install Dependencies') { 16 steps { 17 sh 'npm install' 18 } 19 } 20 21 stage('Build') { 22 steps { 23 sh 'npm run build' 24 } 25 } 26 27 stage('Deploy') { 28 steps { 29 sh ''' 30 pm2 delete app_name|| true 31 pm2 start npm --name "app_name" -- start 32 ''' 33 } 34 } 35 } 36}
Configure GitHub Webhook
- 
In your GitHub repository settings, go to Webhooks 
- 
Add a new webhook: 
- 
Payload URL: http://your_server_ip:8080/github-webhook/
- 
Content type: application/json
- 
Select events: Push and Pull Request events 
Configure Jenkins Installation
For setting up Jenkins for our Next.js project, you'll need to configure several key components/plugins
- 
Go to http://your_server_ip:8080. On you Jenkins dashboard, Go to "Manage Jenkins" > "Plugins"
- 
Install NodeJS Plugin 
- 
To configure the nodejs installation, From the dashboard Go to "Manage Jenkins" > "Tools" - 
Name it (e.g., "NodeJS") 
- 
Choose a recent Node.js version 
- 
Check "Install automatically" 
 
- 
- 
For other plugins to Install, navigate to "Manage Jenkins" > "Plugins": 
- 
Install these: - 
GitHub Integration Plugin 
- 
SSH Agent Plugin (for deployment) 
 
- 
- 
Create a New Pipeline Job - 
Click "New Item" 
- 
Choose "Pipeline" 
- 
Configure: - 
Select “Pipeline" 
- 
Choose "Pipeline Script from SCM" 
- 
Select “SCM” 
- 
Choose “git” 
- 
Enter Repository URL 
- 
For Credentials, click “add” button 
- 
Enter your giithub username 
- 
The password should be your github token. see how to create one here (Ensure you check webhook while creating your token): (https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens) 
- 
Specify your production branch (main/master/production), whatever works for you. 
- 
Apply changes and save 
 
- 
 
- 
- 
Once you are sure everything is set up, now click on “build now”. This is will execute everything you have in the Jenkinsfile. You can check the “Console Output” for the steps it run.
Starting the Application
To view the Jenkins cloned project
Bashcd /var/lib/jenkins/workspace # View it's content ls -1
Start the application
Bashpm2 start npm --name "your_app_name"
Voila! Your app is now running and can be accessed at http://your_server_ip:3000
Conclusion
Thank you for reading to this point, I hope you were able to successfully configure and deploy your application. Keep building and deploying. Until next time, happy coding. ✌🏼
If you have questions, please feel free to drop them in the comments, I’ll do my best to send a response ASAP

