<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom" xmlns:dc="http://purl.org/dc/elements/1.1/">
  <channel>
    <title>DEV Community: Ujjwol Thapa</title>
    <description>The latest articles on DEV Community by Ujjwol Thapa (@bromx).</description>
    <link>https://dev.to/bromx</link>
    <image>
      <url>https://media2.dev.to/dynamic/image/width=90,height=90,fit=cover,gravity=auto,format=auto/https:%2F%2Fdev-to-uploads.s3.amazonaws.com%2Fuploads%2Fuser%2Fprofile_image%2F3924114%2F7c477107-7ea2-4852-bdc8-337ba544603f.png</url>
      <title>DEV Community: Ujjwol Thapa</title>
      <link>https://dev.to/bromx</link>
    </image>
    <atom:link rel="self" type="application/rss+xml" href="https://dev.to/feed/bromx"/>
    <language>en</language>
    <item>
      <title>How to Set Up a CI/CD Pipeline with GitLab CI/CD: A Step-by-Step Guide</title>
      <dc:creator>Ujjwol Thapa</dc:creator>
      <pubDate>Mon, 11 May 2026 04:26:00 +0000</pubDate>
      <link>https://dev.to/bromx/how-to-set-up-a-cicd-pipeline-with-gitlab-cicd-a-step-by-step-guide-7ao</link>
      <guid>https://dev.to/bromx/how-to-set-up-a-cicd-pipeline-with-gitlab-cicd-a-step-by-step-guide-7ao</guid>
      <description>&lt;h2&gt;
  
  
  Why CI/CD Matters for Small Teams
&lt;/h2&gt;

&lt;p&gt;If you're running a small tech team in &lt;strong&gt;Nepal&lt;/strong&gt; — maybe a startup in Kathmandu or a digital agency in Lalitpur — you might think CI/CD is something "big companies" do. Here's the truth: &lt;strong&gt;CI/CD is even more valuable for small teams&lt;/strong&gt; because every hour saved on manual work is an hour you can spend building features.&lt;/p&gt;

&lt;p&gt;In this guide, I'll walk you through setting up a complete &lt;strong&gt;CI/CD pipeline with GitLab CI/CD&lt;/strong&gt;, from zero to production deployment.&lt;/p&gt;

&lt;h2&gt;
  
  
  What Is GitLab CI/CD?
&lt;/h2&gt;

&lt;p&gt;GitLab CI/CD is a built-in continuous integration and deployment tool. Every time you push code to your GitLab repository, it can automatically:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Run your tests&lt;/li&gt;
&lt;li&gt;Build your application&lt;/li&gt;
&lt;li&gt;Deploy to staging or production&lt;/li&gt;
&lt;li&gt;Run security scans&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;All defined in a single &lt;code&gt;.gitlab-ci.yml&lt;/code&gt; file.&lt;/p&gt;

&lt;h2&gt;
  
  
  Prerequisites
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;A GitLab account (free tier works fine)&lt;/li&gt;
&lt;li&gt;A project with at least a basic application&lt;/li&gt;
&lt;li&gt;Understanding of basic shell commands&lt;/li&gt;
&lt;li&gt;A server or cloud instance for deployment (AWS, Azure, or a VPS)&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Step 1: Create Your &lt;code&gt;.gitlab-ci.yml&lt;/code&gt;
&lt;/h2&gt;

&lt;p&gt;Create this file in your project root:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;stages&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;build&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;deploy&lt;/span&gt;

&lt;span class="na"&gt;variables&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;DOCKER_IMAGE&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_REGISTRY_IMAGE:$CI_COMMIT_SHA&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 1: Run tests&lt;/span&gt;
&lt;span class="na"&gt;test&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;node:20-alpine&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm ci&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm run test&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm run lint&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_MERGE_REQUEST_IID&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 2: Build Docker image&lt;/span&gt;
&lt;span class="na"&gt;build&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;build&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;docker:24.0&lt;/span&gt;
  &lt;span class="na"&gt;services&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker:24.0-dind&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker build -t $DOCKER_IMAGE .&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker push $DOCKER_IMAGE&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 3: Deploy to production&lt;/span&gt;
&lt;span class="na"&gt;deploy&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;deploy&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;alpine:latest&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;apk add --no-cache openssh-client&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;mkdir -p ~/.ssh&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;echo "$SSH_PRIVATE_KEY" &amp;gt; ~/.ssh/id_ed25519&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;chmod 600 ~/.ssh/id_ed25519&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;ssh -o StrictHostKeyChecking=no user@your-server.com "&lt;/span&gt;
        &lt;span class="s"&gt;docker pull $DOCKER_IMAGE &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker stop app || &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="s"&gt; &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker rm app || &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="s"&gt; &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker run -d --name app -p 3000:3000 $DOCKER_IMAGE&lt;/span&gt;
      &lt;span class="s"&gt;"&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;
      &lt;span class="na"&gt;when&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;manual&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Step 2: Set Up CI/CD Variables
&lt;/h2&gt;

&lt;p&gt;In your GitLab project, go to &lt;strong&gt;Settings → CI/CD → Variables&lt;/strong&gt; and add:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Variable&lt;/th&gt;
&lt;th&gt;Description&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;SSH_PRIVATE_KEY&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;SSH key for connecting to your server&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;CI_REGISTRY_USER&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;GitLab container registry username&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;CI_REGISTRY_PASSWORD&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;GitLab container registry password&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;h2&gt;
  
  
  Step 3: Understand the Pipeline Flow
&lt;/h2&gt;

&lt;p&gt;Here's what happens on each push to &lt;code&gt;main&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;graph LR
    A[Push to main] --&amp;gt; B[test]
    B --&amp;gt; C[build]
    C --&amp;gt; D{deploy}
    D --&amp;gt;|manual trigger| E[Production]
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Test stage&lt;/strong&gt; — Installs dependencies, runs tests and linting&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Build stage&lt;/strong&gt; — Creates a Docker image and pushes to GitLab's registry&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Deploy stage&lt;/strong&gt; — SSH's into your server and deploys the new image (manual trigger for safety)&lt;/li&gt;
&lt;/ol&gt;

&lt;h2&gt;
  
  
  Step 4: Add Security Scanning
&lt;/h2&gt;

&lt;p&gt;For teams managing &lt;strong&gt;sensitive applications in Nepal&lt;/strong&gt; (fintech, health tech, etc.), add a security scan:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;security-scan&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;aquasec/trivy:latest&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;trivy image --severity HIGH,CRITICAL $DOCKER_IMAGE&lt;/span&gt;
  &lt;span class="na"&gt;allow_failure&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Real-World Results
&lt;/h2&gt;

&lt;p&gt;At &lt;strong&gt;Andmine Pvt Ltd&lt;/strong&gt;, where I work as a System Administrator, implementing CI/CD pipelines reduced our deployment time from &lt;strong&gt;~45 minutes per deployment to under 3 minutes&lt;/strong&gt;. Manual errors dropped to near zero, and our team gained confidence in shipping code on Fridays.&lt;/p&gt;

&lt;h2&gt;
  
  
  Common Pitfalls to Avoid
&lt;/h2&gt;

&lt;h3&gt;
  
  
  1. Not caching dependencies
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;test&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;cache&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;key&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;${CI_COMMIT_REF_SLUG}&lt;/span&gt;
    &lt;span class="na"&gt;paths&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;node_modules/&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  2. Deploying on every branch
&lt;/h3&gt;

&lt;p&gt;Use &lt;code&gt;rules:&lt;/code&gt; to control when pipelines run. Don't deploy &lt;code&gt;feature&lt;/code&gt; branches to production.&lt;/p&gt;

&lt;h3&gt;
  
  
  3. No manual approval for production
&lt;/h3&gt;

&lt;p&gt;The &lt;code&gt;when: manual&lt;/code&gt; flag on your deploy stage prevents accidental production deployments.&lt;/p&gt;

&lt;h2&gt;
  
  
  Next Steps
&lt;/h2&gt;

&lt;p&gt;Once you have the basics working:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Add &lt;strong&gt;monitoring&lt;/strong&gt; with Prometheus and Grafana&lt;/li&gt;
&lt;li&gt;Set up &lt;strong&gt;automatic rollbacks&lt;/strong&gt; on health check failures&lt;/li&gt;
&lt;li&gt;Implement &lt;strong&gt;canary deployments&lt;/strong&gt; for zero-risk releases&lt;/li&gt;
&lt;li&gt;Use &lt;strong&gt;GitLab Environments&lt;/strong&gt; to track deployment history&lt;/li&gt;
&lt;/ul&gt;




&lt;p&gt;&lt;em&gt;Need help setting up CI/CD for your team? I work with organizations across Nepal to build reliable deployment pipelines. &lt;a href="//ujjwolthapa.com.np/#contact"&gt;Get in touch&lt;/a&gt;.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>cicd</category>
      <category>gitlab</category>
      <category>devops</category>
      <category>tutorial</category>
    </item>
    <item>
      <title>How to Set Up a CI/CD Pipeline with GitLab CI/CD: A Step-by-Step Guide</title>
      <dc:creator>Ujjwol Thapa</dc:creator>
      <pubDate>Mon, 11 May 2026 04:26:00 +0000</pubDate>
      <link>https://dev.to/bromx/how-to-set-up-a-cicd-pipeline-with-gitlab-cicd-a-step-by-step-guide-2add</link>
      <guid>https://dev.to/bromx/how-to-set-up-a-cicd-pipeline-with-gitlab-cicd-a-step-by-step-guide-2add</guid>
      <description>&lt;h2&gt;
  
  
  Why CI/CD Matters for Small Teams
&lt;/h2&gt;

&lt;p&gt;If you're running a small tech team in &lt;strong&gt;Nepal&lt;/strong&gt; — maybe a startup in Kathmandu or a digital agency in Lalitpur — you might think CI/CD is something "big companies" do. Here's the truth: &lt;strong&gt;CI/CD is even more valuable for small teams&lt;/strong&gt; because every hour saved on manual work is an hour you can spend building features.&lt;/p&gt;

&lt;p&gt;In this guide, I'll walk you through setting up a complete &lt;strong&gt;CI/CD pipeline with GitLab CI/CD&lt;/strong&gt;, from zero to production deployment.&lt;/p&gt;

&lt;h2&gt;
  
  
  What Is GitLab CI/CD?
&lt;/h2&gt;

&lt;p&gt;GitLab CI/CD is a built-in continuous integration and deployment tool. Every time you push code to your GitLab repository, it can automatically:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Run your tests&lt;/li&gt;
&lt;li&gt;Build your application&lt;/li&gt;
&lt;li&gt;Deploy to staging or production&lt;/li&gt;
&lt;li&gt;Run security scans&lt;/li&gt;
&lt;/ul&gt;

&lt;p&gt;All defined in a single &lt;code&gt;.gitlab-ci.yml&lt;/code&gt; file.&lt;/p&gt;

&lt;h2&gt;
  
  
  Prerequisites
&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;A GitLab account (free tier works fine)&lt;/li&gt;
&lt;li&gt;A project with at least a basic application&lt;/li&gt;
&lt;li&gt;Understanding of basic shell commands&lt;/li&gt;
&lt;li&gt;A server or cloud instance for deployment (AWS, Azure, or a VPS)&lt;/li&gt;
&lt;/ul&gt;

&lt;h2&gt;
  
  
  Step 1: Create Your &lt;code&gt;.gitlab-ci.yml&lt;/code&gt;
&lt;/h2&gt;

&lt;p&gt;Create this file in your project root:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;stages&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;build&lt;/span&gt;
  &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;deploy&lt;/span&gt;

&lt;span class="na"&gt;variables&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;DOCKER_IMAGE&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_REGISTRY_IMAGE:$CI_COMMIT_SHA&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 1: Run tests&lt;/span&gt;
&lt;span class="na"&gt;test&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;node:20-alpine&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm ci&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm run test&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;npm run lint&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_MERGE_REQUEST_IID&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 2: Build Docker image&lt;/span&gt;
&lt;span class="na"&gt;build&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;build&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;docker:24.0&lt;/span&gt;
  &lt;span class="na"&gt;services&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker:24.0-dind&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker login -u $CI_REGISTRY_USER -p $CI_REGISTRY_PASSWORD $CI_REGISTRY&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker build -t $DOCKER_IMAGE .&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;docker push $DOCKER_IMAGE&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;

&lt;span class="c1"&gt;# Stage 3: Deploy to production&lt;/span&gt;
&lt;span class="na"&gt;deploy&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;deploy&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;alpine:latest&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;apk add --no-cache openssh-client&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;mkdir -p ~/.ssh&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;echo "$SSH_PRIVATE_KEY" &amp;gt; ~/.ssh/id_ed25519&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;chmod 600 ~/.ssh/id_ed25519&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;ssh -o StrictHostKeyChecking=no user@your-server.com "&lt;/span&gt;
        &lt;span class="s"&gt;docker pull $DOCKER_IMAGE &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker stop app || &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="s"&gt; &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker rm app || &lt;/span&gt;&lt;span class="kc"&gt;true&lt;/span&gt;&lt;span class="s"&gt; &amp;amp;&amp;amp;&lt;/span&gt;
        &lt;span class="s"&gt;docker run -d --name app -p 3000:3000 $DOCKER_IMAGE&lt;/span&gt;
      &lt;span class="s"&gt;"&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;
      &lt;span class="na"&gt;when&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;manual&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Step 2: Set Up CI/CD Variables
&lt;/h2&gt;

&lt;p&gt;In your GitLab project, go to &lt;strong&gt;Settings → CI/CD → Variables&lt;/strong&gt; and add:&lt;/p&gt;

&lt;div class="table-wrapper-paragraph"&gt;&lt;table&gt;
&lt;thead&gt;
&lt;tr&gt;
&lt;th&gt;Variable&lt;/th&gt;
&lt;th&gt;Description&lt;/th&gt;
&lt;/tr&gt;
&lt;/thead&gt;
&lt;tbody&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;SSH_PRIVATE_KEY&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;SSH key for connecting to your server&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;CI_REGISTRY_USER&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;GitLab container registry username&lt;/td&gt;
&lt;/tr&gt;
&lt;tr&gt;
&lt;td&gt;&lt;code&gt;CI_REGISTRY_PASSWORD&lt;/code&gt;&lt;/td&gt;
&lt;td&gt;GitLab container registry password&lt;/td&gt;
&lt;/tr&gt;
&lt;/tbody&gt;
&lt;/table&gt;&lt;/div&gt;

&lt;h2&gt;
  
  
  Step 3: Understand the Pipeline Flow
&lt;/h2&gt;

&lt;p&gt;Here's what happens on each push to &lt;code&gt;main&lt;/code&gt;:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight plaintext"&gt;&lt;code&gt;graph LR
    A[Push to main] --&amp;gt; B[test]
    B --&amp;gt; C[build]
    C --&amp;gt; D{deploy}
    D --&amp;gt;|manual trigger| E[Production]
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;ol&gt;
&lt;li&gt;
&lt;strong&gt;Test stage&lt;/strong&gt; — Installs dependencies, runs tests and linting&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Build stage&lt;/strong&gt; — Creates a Docker image and pushes to GitLab's registry&lt;/li&gt;
&lt;li&gt;
&lt;strong&gt;Deploy stage&lt;/strong&gt; — SSH's into your server and deploys the new image (manual trigger for safety)&lt;/li&gt;
&lt;/ol&gt;

&lt;h2&gt;
  
  
  Step 4: Add Security Scanning
&lt;/h2&gt;

&lt;p&gt;For teams managing &lt;strong&gt;sensitive applications in Nepal&lt;/strong&gt; (fintech, health tech, etc.), add a security scan:&lt;br&gt;
&lt;/p&gt;

&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;security-scan&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;stage&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;test&lt;/span&gt;
  &lt;span class="na"&gt;image&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;aquasec/trivy:latest&lt;/span&gt;
  &lt;span class="na"&gt;script&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;trivy image --severity HIGH,CRITICAL $DOCKER_IMAGE&lt;/span&gt;
  &lt;span class="na"&gt;allow_failure&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="kc"&gt;true&lt;/span&gt;
  &lt;span class="na"&gt;rules&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="na"&gt;if&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;$CI_COMMIT_BRANCH == "main"&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h2&gt;
  
  
  Real-World Results
&lt;/h2&gt;

&lt;p&gt;At &lt;strong&gt;Andmine Pvt Ltd&lt;/strong&gt;, where I work as a System Administrator, implementing CI/CD pipelines reduced our deployment time from &lt;strong&gt;~45 minutes per deployment to under 3 minutes&lt;/strong&gt;. Manual errors dropped to near zero, and our team gained confidence in shipping code on Fridays.&lt;/p&gt;

&lt;h2&gt;
  
  
  Common Pitfalls to Avoid
&lt;/h2&gt;

&lt;h3&gt;
  
  
  1. Not caching dependencies
&lt;/h3&gt;



&lt;div class="highlight js-code-highlight"&gt;
&lt;pre class="highlight yaml"&gt;&lt;code&gt;&lt;span class="na"&gt;test&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
  &lt;span class="na"&gt;cache&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
    &lt;span class="na"&gt;key&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt; &lt;span class="s"&gt;${CI_COMMIT_REF_SLUG}&lt;/span&gt;
    &lt;span class="na"&gt;paths&lt;/span&gt;&lt;span class="pi"&gt;:&lt;/span&gt;
      &lt;span class="pi"&gt;-&lt;/span&gt; &lt;span class="s"&gt;node_modules/&lt;/span&gt;
&lt;/code&gt;&lt;/pre&gt;

&lt;/div&gt;



&lt;h3&gt;
  
  
  2. Deploying on every branch
&lt;/h3&gt;

&lt;p&gt;Use &lt;code&gt;rules:&lt;/code&gt; to control when pipelines run. Don't deploy &lt;code&gt;feature&lt;/code&gt; branches to production.&lt;/p&gt;

&lt;h3&gt;
  
  
  3. No manual approval for production
&lt;/h3&gt;

&lt;p&gt;The &lt;code&gt;when: manual&lt;/code&gt; flag on your deploy stage prevents accidental production deployments.&lt;/p&gt;

&lt;h2&gt;
  
  
  Next Steps
&lt;/h2&gt;

&lt;p&gt;Once you have the basics working:&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;Add &lt;strong&gt;monitoring&lt;/strong&gt; with Prometheus and Grafana&lt;/li&gt;
&lt;li&gt;Set up &lt;strong&gt;automatic rollbacks&lt;/strong&gt; on health check failures&lt;/li&gt;
&lt;li&gt;Implement &lt;strong&gt;canary deployments&lt;/strong&gt; for zero-risk releases&lt;/li&gt;
&lt;li&gt;Use &lt;strong&gt;GitLab Environments&lt;/strong&gt; to track deployment history&lt;/li&gt;
&lt;/ul&gt;




&lt;p&gt;&lt;em&gt;Need help setting up CI/CD for your team? I work with organizations across Nepal to build reliable deployment pipelines. &lt;a href="//ujjwolthapa.com.np/#contact"&gt;Get in touch&lt;/a&gt;.&lt;/em&gt;&lt;/p&gt;

</description>
      <category>cicd</category>
      <category>gitlab</category>
      <category>devops</category>
      <category>tutorial</category>
    </item>
  </channel>
</rss>
