Skip to main content
Redhat Developers  Logo
  • Products

    Platforms

    • Red Hat Enterprise Linux
      Red Hat Enterprise Linux Icon
    • Red Hat AI
      Red Hat AI
    • Red Hat OpenShift
      Openshift icon
    • Red Hat Ansible Automation Platform
      Ansible icon
    • View All Red Hat Products

    Featured

    • Red Hat build of OpenJDK
    • Red Hat Developer Hub
    • Red Hat JBoss Enterprise Application Platform
    • Red Hat OpenShift Dev Spaces
    • Red Hat OpenShift Local
    • Red Hat Developer Sandbox

      Try Red Hat products and technologies without setup or configuration fees for 30 days with this shared Openshift and Kubernetes cluster.
    • Try at no cost
  • Technologies

    Featured

    • AI/ML
      AI/ML Icon
    • Linux
      Linux Icon
    • Kubernetes
      Cloud icon
    • Automation
      Automation Icon showing arrows moving in a circle around a gear
    • View All Technologies
    • Programming Languages & Frameworks

      • Java
      • Python
      • JavaScript
    • System Design & Architecture

      • Red Hat architecture and design patterns
      • Microservices
      • Event-Driven Architecture
      • Databases
    • Developer Productivity

      • Developer productivity
      • Developer Tools
      • GitOps
    • Automated Data Processing

      • AI/ML
      • Data Science
      • Apache Kafka on Kubernetes
    • Platform Engineering

      • DevOps
      • DevSecOps
      • Ansible automation for applications and services
    • Secure Development & Architectures

      • Security
      • Secure coding
  • Learn

    Featured

    • Kubernetes & Cloud Native
      Openshift icon
    • Linux
      Rhel icon
    • Automation
      Ansible cloud icon
    • AI/ML
      AI/ML Icon
    • View All Learning Resources

    E-Books

    • GitOps Cookbook
    • Podman in Action
    • Kubernetes Operators
    • The Path to GitOps
    • View All E-books

    Cheat Sheets

    • Linux Commands
    • Bash Commands
    • Git
    • systemd Commands
    • View All Cheat Sheets

    Documentation

    • Product Documentation
    • API Catalog
    • Legacy Documentation
  • Developer Sandbox

    Developer Sandbox

    • Access Red Hat’s products and technologies without setup or configuration, and start developing quicker than ever before with our new, no-cost sandbox environments.
    • Explore Developer Sandbox

    Featured Developer Sandbox activities

    • Get started with your Developer Sandbox
    • OpenShift virtualization and application modernization using the Developer Sandbox
    • Explore all Developer Sandbox activities

    Ready to start developing apps?

    • Try at no cost
  • Blog
  • Events
  • Videos

Deploying Node.js applications to Kubernetes with Nodeshift and Minikube

March 9, 2021
Lucas Holmquist
Related topics:
Developer ToolsNode.jsKubernetes

Share:

    In a previous article, I showed how easy it was to deploy a Node.js application during development to Red Hat OpenShift using the Nodeshift command-line interface (CLI). In this article, we will take a look at using Nodeshift to deploy Node.js applications to vanilla Kubernetes—specifically, with Minikube.

    Getting started

    If you want to follow along with this tutorial, you will need to run Minikube. I won't cover the setup process, but Minikube's documentation can guide you through it. For the tutorial, I also assume that you have installed Node.js and Node Package Manager (npm).

    The code samples we'll use are available on GitHub. Our example is a very basic Node.js application with a Dockerfile. In fact, it is taken from the Dockerizing a Node.js web app guide on Nodejs.org.

    The Nodeshift CLI

    As the Nodeshift module readme states, Nodeshift is an opinionated command-line application and programmable API that you can use to deploy Node.js applications to Red Hat OpenShift. You can easily run it using the npx command, and it will create the appropriate YAML files to deploy your application.

    Nodeshift is a great tool to use if you are developing against an OpenShift cluster, which uses the Source-to-Image (S2I) workflow. In short, Nodeshift creates an OpenShift BuildConfig, which calls a Node.js S2I image to build your Node application. In most cases, you can achieve this by running npm install. The build result is put into an OpenShift ImageStream that resides in the internal OpenShift container registry. This image is then used to deploy your application.

    But what about deploying to a vanilla Kubernetes cluster that doesn’t know anything about BuildConfigs, ImageStreams, or S2I? Well, as of Nodeshift's 7.3 release, you can now deploy your Node.js applications to Minikube.

    Deploying Node.js to Minikube

    Before we look at how Nodeshift works for deploying a Node.js application to Minikube, let’s take a minute for a high-level overview of deploying to Kubernetes.

    First, you will create an application container image, which you can do with Docker. Once you have a container image, you'll need to push that image to a container registry that your cluster has access to, something like Docker Hub. Once the image is available, you must then specify that image in your deployment YAML and create a service to expose the application.

    This flow starts to be more cumbersome when you start iterating on your code. It isn’t really development-friendly if you need to run a Docker build and push that new image to Docker Hub every time. Not to mention that you also need to update your deployment with the new version of the image to ensure it redeploys.

    Nodeshift's goal is to make developers' lives easier when deploying to OpenShift and Kubernetes. Let's see how Nodeshift helps with each of those unwieldy steps.

    Minikube's internal Docker server

    A major difference between OpenShift and Kubernetes is that there is no easy way to run S2I builds on plain Kubernetes. We also don’t want to run a Docker build and push to Docker Hub every time we change our code. Fortunately, Minikube gives us an alternative.

    Minikube has its own internal Docker server that we can connect to using the Docker Engine API. We can use this server to run our Docker build in the environment, which means that we don’t have to push the image to an external resource like Docker Hub. We can then use this image in our deployment.

    To get access to the internal Docker server, Minikube has a command to export some environment variables to add to your terminal shell. This command is minikube docker-env, which might output something like this:

    export DOCKER_TLS_VERIFY="1"
    export DOCKER_HOST="tcp://192.168.39.12:2376"
    export DOCKER_CERT_PATH="/home/lucasholmquist/.minikube/certs"
    export MINIKUBE_ACTIVE_DOCKERD="minikube"
    
    # To point your shell to minikube's docker-daemon, run:
    # eval $(minikube -p minikube docker-env)
    

    Making it easier with Nodeshift

    Nodeshift abstracts the details we don’t really care about so we can focus on our applications. In this case, we don’t want to think about how to connect to Minikube's internal server or how to run Docker commands by hand, and we don’t want to think about updating our deployment YAML every time we build a new image to redeploy it.

    Using the Nodeshift CLI with the --kube flag simplifies those tasks. Let’s see how it works using our example application.

    We will use npx to deploy the Node.js application to Minikube, so we don’t need to install anything globally. Run it like this in the example directory:

    $ npx nodeshift --kube
    

    Nodeshift creates a service and deployment by default if none are provided. Also, note that the type of service it creates is a LoadBalancer, which allows us to expose our application without using ingress.

    The Nodeshift CLI runs the same goals for a Kubernetes deploy as it does for an OpenShift deploy. The key difference comes during the build phase. Instead of creating an OpenShift BuildConfig and running an S2I process on the cluster, Nodeshift uses the dockerode module to connect to Minikube's internal Docker server and run a build using the provided Dockerfile. The built image is now in that internal registry, ready to be deployed by the deployment YAML that the Nodeshift CLI creates. Nodeshift also adds a randomly-generated number to the deployment's metadata, which is then applied during every redeploy. This will trigger Minikube to redeploy the application with the new image.

    The following is an example log output:

    ~/develop/nodeshift-starters/basic-node-app-dockerized» npx nodeshift --kube                                        
    
    2021-02-09T20:03:18.405Z INFO loading configuration
    2021-02-09T20:03:18.452Z INFO Using the kubernetes flag.
    2021-02-09T20:03:18.762Z INFO using namespace default at https://192.168.39.12:8443
    2021-02-09T20:03:18.763Z WARNING a file property was not found in your package.json, archiving the current directory.
    2021-02-09T20:03:18.773Z INFO creating archive of .dockerignore, .gitignore, Dockerfile, README.md, package-lock.json, package.json, server.js
    2021-02-09T20:03:18.774Z INFO Building Docker Image
    2021-02-09T20:03:18.848Z TRACE {"stream":"Step 1/7 : FROM node:14"}
    2021-02-09T20:03:18.848Z TRACE {"stream":"\n"}
    2021-02-09T20:03:18.849Z TRACE {"stream":" ---\u003e cb544c4472e9\n"}
    2021-02-09T20:03:18.849Z TRACE {"stream":"Step 2/7 : WORKDIR /usr/src/app"}
    2021-02-09T20:03:18.849Z TRACE {"stream":"\n"}
    2021-02-09T20:03:18.849Z TRACE {"stream":" ---\u003e Using cache\n"}
    2021-02-09T20:03:18.849Z TRACE {"stream":" ---\u003e 57c9e3a4e918\n"}
    2021-02-09T20:03:18.849Z TRACE {"stream":"Step 3/7 : COPY package*.json ./"}
    2021-02-09T20:03:18.850Z TRACE {"stream":"\n"}
    2021-02-09T20:03:19.050Z TRACE {"stream":" ---\u003e 742050ca3266\n"}
    2021-02-09T20:03:19.050Z TRACE {"stream":"Step 4/7 : RUN npm install"}
    2021-02-09T20:03:19.050Z TRACE {"stream":"\n"}
    2021-02-09T20:03:19.109Z TRACE {"stream":" ---\u003e Running in f3477d5f2b00\n"}
    2021-02-09T20:03:21.739Z TRACE {"stream":"\u001b[91mnpm WARN basic-node-app-dockerized@1.0.0 No description\n\u001b[0m"}
    2021-02-09T20:03:21.744Z TRACE {"stream":"\u001b[91mnpm WARN basic-node-app-dockerized@1.0.0 No repository field.\n\u001b[0m"}
    2021-02-09T20:03:21.745Z TRACE {"stream":"\u001b[91m\n\u001b[0m"}
    2021-02-09T20:03:21.746Z TRACE {"stream":"added 50 packages from 37 contributors and audited 50 packages in 1.387s\n"}
    2021-02-09T20:03:21.780Z TRACE {"stream":"found 0 vulnerabilities\n\n"}
    2021-02-09T20:03:22.303Z TRACE {"stream":"Removing intermediate container f3477d5f2b00\n"}
    2021-02-09T20:03:22.303Z TRACE {"stream":" ---\u003e afb97a82c035\n"}
    2021-02-09T20:03:22.303Z TRACE {"stream":"Step 5/7 : COPY . ."}
    2021-02-09T20:03:22.303Z TRACE {"stream":"\n"}
    2021-02-09T20:03:22.481Z TRACE {"stream":" ---\u003e 1a451003c472\n"}
    2021-02-09T20:03:22.481Z TRACE {"stream":"Step 6/7 : EXPOSE 8080"}
    2021-02-09T20:03:22.482Z TRACE {"stream":"\n"}
    2021-02-09T20:03:22.545Z TRACE {"stream":" ---\u003e Running in a76389d44b59\n"}
    2021-02-09T20:03:22.697Z TRACE {"stream":"Removing intermediate container a76389d44b59\n"}
    2021-02-09T20:03:22.697Z TRACE {"stream":" ---\u003e 8ee240b7f9ab\n"}
    2021-02-09T20:03:22.697Z TRACE {"stream":"Step 7/7 : CMD [ \"node\", \"server.js\" ]"}
    2021-02-09T20:03:22.698Z TRACE {"stream":"\n"}
    2021-02-09T20:03:22.759Z TRACE {"stream":" ---\u003e Running in 1f7325ab3c64\n"}
    2021-02-09T20:03:22.911Z TRACE {"stream":"Removing intermediate container 1f7325ab3c64\n"}
    2021-02-09T20:03:22.912Z TRACE {"stream":" ---\u003e d7f5d1e95592\n"}
    2021-02-09T20:03:22.912Z TRACE {"aux":{"ID":"sha256:d7f5d1e9559242f767b54b168c36df5c7cbce6ebc7eb1145d7f6292f20e8cda2"}}
    2021-02-09T20:03:22.913Z TRACE {"stream":"Successfully built d7f5d1e95592\n"}
    2021-02-09T20:03:22.929Z TRACE {"stream":"Successfully tagged basic-node-app-dockerized:latest\n"}
    2021-02-09T20:03:22.933Z WARNING No .nodeshift directory
    2021-02-09T20:03:22.954Z INFO openshift.yaml and openshift.json written to /home/lucasholmquist/develop/nodeshift-starters/basic-node-app-dockerized/tmp/nodeshift/resource/
    2021-02-09T20:03:22.975Z INFO creating new service basic-node-app-dockerized
    2021-02-09T20:03:22.979Z TRACE Deployment Applied
    2021-02-09T20:03:23.036Z INFO Application running at: http://192.168.39.12:30076
    2021-02-09T20:03:23.036Z INFO complete
    
    

    Following the deployment, the Nodeshift CLI also provides the URL where the application is running in the console output. The output might look something like this:

    ...
    INFO Application running at http://192.168.39.12:30769
    ...
    

    Navigating to the URL provided returns "Hello World."

    Conclusion

    This article gave a brief overview of the Nodeshift CLI's support for deploying to Minikube. In the future, we plan to add more Kubernetes platforms and other developer-friendly features, like possibly having the Nodeshift CLI create a default Dockerfile if there isn’t one.

    If you like what you see and want to learn more, check out the Nodeshift project. As always, if there are more features you would like to see, create an issue over on GitHub. To learn more about what Red Hat is up to on the Node.js front, check out our Node.js landing page.

    Last updated: March 8, 2021

    Recent Posts

    • The odo CLI is deprecated: What developers need to know

    • Exposing OpenShift networks using BGP

    • Camel integration quarterly digest: Q3 2025

    • How to run I/O workloads on OpenShift Virtualization VMs

    • How spec-driven development improves AI coding quality

    Red Hat Developers logo LinkedIn YouTube Twitter Facebook

    Platforms

    • Red Hat AI
    • Red Hat Enterprise Linux
    • Red Hat OpenShift
    • Red Hat Ansible Automation Platform
    • See all products

    Build

    • Developer Sandbox
    • Developer Tools
    • Interactive Tutorials
    • API Catalog

    Quicklinks

    • Learning Resources
    • E-books
    • Cheat Sheets
    • Blog
    • Events
    • Newsletter

    Communicate

    • About us
    • Contact sales
    • Find a partner
    • Report a website issue
    • Site Status Dashboard
    • Report a security problem

    RED HAT DEVELOPER

    Build here. Go anywhere.

    We serve the builders. The problem solvers who create careers with code.

    Join us if you’re a developer, software engineer, web designer, front-end designer, UX designer, computer scientist, architect, tester, product manager, project manager or team lead.

    Sign me up

    Red Hat legal and privacy links

    • About Red Hat
    • Jobs
    • Events
    • Locations
    • Contact Red Hat
    • Red Hat Blog
    • Inclusion at Red Hat
    • Cool Stuff Store
    • Red Hat Summit
    © 2025 Red Hat

    Red Hat legal and privacy links

    • Privacy statement
    • Terms of use
    • All policies and guidelines
    • Digital accessibility

    Report a website issue