Dynamic configuration for static Vue 2 apps without SSR

June 27, 2019

This article explains how to allow our Vue.js (version 2.x) app to accept configurations at run time instead of build time using Docker & NGINX.

If you want the solution, you can skip to the Solution section.

Why ?

At Scalair, we use Docker to deploy our applications. Some of our applications are Vue.js static front ends, served through NGINX. We use Gitlab CI to test & deploy our code to Amazon EKS. Each Gitlab pipeline builds our app, then pushes the Docker image to Amazon ECR.

Our Vue apps had configurations like this, thanks to Vue.js client-side env vars :

const API_URL = process.env.VUE_APP_API_URL;

Due to the lack of Server-Side Rendering in our Vue app, we used to give environment variables to the Vue CLI at build time in order to provide configurations, e.g.: an API URL.

We also had a Dockerfile that we used to build using docker build --build-arg API_URL=https://api.example.com . :

# Build stage
FROM mhart/alpine-node:10 as build-stage

WORKDIR /app

COPY package.json package-lock.json .npmrc ./
RUN npm ci

COPY . .

ARG API_URL
ENV VUE_APP_API_URL $API_URL

RUN ./node_modules/.bin/vue-cli-service build --mode production

# Production stage
FROM nginx:1.17.0-alpine as production-stage
COPY --from=build-stage /app/dist /usr/share/nginx/html
EXPOSE 80

CMD nginx -g "daemon off;"

This multi-stage build will build the static files, then serve them with NGINX.

One concern we had was the consistency of our images : a Docker image would differ from one environment to an other because of the --build-arg parameter. The VUE_APP_ env vars only work at build time.

Solution

Let the NGINX container do the job !

We replaced process.env.VUE_APP_API_URL with '{{ API_URL }}' :

// This will act as a "template" that will be replaced with an actual URL.
const API_URL = '{{ API_URL }}';

Then we removed the build args & replaced the {{ templates }} before running NGINX.

Final Dockerfile :

# Build stage
FROM mhart/alpine-node:10 as build-stage

WORKDIR /app

COPY package.json package-lock.json .npmrc ./
RUN npm ci

COPY . .

RUN ./node_modules/.bin/vue-cli-service build --mode production

# Production stage
FROM nginx:1.17.0-alpine as production-stage
COPY --from=build-stage /app/dist /usr/share/nginx/html
EXPOSE 80

# Using `sed` to replace {{ API_URL }} with the actual API URL,
# which is given to the container at RUN TIME !
CMD sed -i -e "s/{{ API_URL }}/$API_URL/g" /usr/share/nginx/html/js/app.*.js && \
    nginx -g "daemon off;"

Finally, we run the container using these commands :

docker build -t my-vue-app .
docker run -e API_URL=https://api.example.com my-vue-app

Conclusion

This is more a temporary solution than an actual long-term solution.

If you are facing the same issue we had, you should definitely look into Server-Side Rendering. SSR allows a lot of behaviors that you can't do with a single static app rendered in the browser. This is a good way to dynamically change parts of your frontend and prevent crappy hacks like we did.

Was this article helpful ?

Feel free to drop a 👍 !


Profile picture

I'm a backend dev who enjoys the DevOps culture, home automation and innovation. I like to experiment and try to build unique/helpful/funny things. Feel free to check out my GitHub.