i'm trying to set up a dynamic env variable based on environment.
what im trying to do is have a env-variable for a URL to use for livepreview that is localhost when running locally, a specific URL when the project is deployed to dev environment and another URL when the project is deployed to a prod environment.
we are using gitlab and openshift, and i have defined the variable in gitlabs CI/CD settings with different value per environment, but on my deployed app it's still rendering as "undefined".
have tried importing and running "dotenv.config()" in my server.ts-file and payload.config.ts file.
I believe I might have to add the variable to my Dockerfile, but not sure how to do that when I want the variable to be dynamic based on environment. Would love any help/pointers :)
Tried creating two Dockerfiles, one for dev and one for prod and adding the variable in there as well - it's still showing up as undefined in the browser
they are prefixed with PAYLOAD_PUBLIC, and they make their way to the deployment in openshift just fine.
i suspect it's because I'm trying to reference them from a collectionconfig-file, which probably runs on the client, while the env-variables are stored on the server
i am setting up live preview, and need the URL for that property to be dynamic based on deployment environment. thought it'd be nice to use an env variable for it, but it's a struggle being able to read it from the client haha
i believe my issue might be related to how Openshift runs my deployment - i suspect my collection runs on the client and my .env-variables are stored on the server, hence them returning "undefined" even though they show up with correct info in my configmap in the openshift deployment itself.
not sure how to check/verify if my suspicion is correct, and if it is correct, how to work around it 🥴
Found a fix! I had placed the env variables after the build in my Dockerfiles, moving them up before it runs the build process fixed it :)
My Dockerfile looks like this:
FROM node:18.8-alpine as base
FROM base as builder
WORKDIR /
COPY package*.json ./
ARG PAYLOAD_PUBLIC_SOME_URL=https://some.site
ENV PAYLOAD_PUBLIC_SOME_URL=$PAYLOAD_PUBLIC_SOME_URL
ARG PAYLOAD_PUBLIC_OTHER_URL=https://some.other.site
ENV PAYLOAD_PUBLIC_OTHER_URL=$PAYLOAD_PUBLIC_OTHER_URL
ARG PAYLOAD_PUBLIC_ANOTHER_URL=https://another.site
ENV PAYLOAD_PUBLIC_ANOTHER_URL=$PAYLOAD_PUBLIC_ANOTHER_URL
COPY . .
RUN npm install
RUN npm run build
FROM base as runtime
ENV NODE_ENV=production
ENV PAYLOAD_CONFIG_PATH=dist/payload.config.js
WORKDIR /
COPY package*.json ./
COPY /src/assets ./
RUN npm install --production
COPY --from=builder /dist ./dist
COPY --from=builder /build ./build
EXPOSE 3030
CMD ["node", "dist/server.js"]
i know next to nothing about Docker, so I dont really know
this just worked for me
Are all of the variables ones you would expect to log in browser?
They are prefixed with PAYLOAD_PUBLIC ?
even the public ones?
Oh i see what you're saying
What kind of env vars would you want to have for the client?
Sounds like you are doing everything correct so I don't know what the problem could be, but maybe the live-preview example can help to spot differences
https://github.com/payloadcms/payload/tree/main/examples/live-preview/payload#quick-start(might also be worth checking that the example works)
how did this get fixed ? I am trying to push payload cms to hetzner through coolify using Dockerfile and I am not getting env variables as it is looking for .env file which isnt there.
I have one question, why does payload requires env variables at build time ? like db url, cloud storage s3 envs ? I dont think it is safe ? or envs in payload config is for only static content at server side ?
Yeah but don’t you think db url and s3 envs at payload config (with PAYLOAD_CONFIG) isn’t a safe option ? May be my understanding is wrong about it.
Gotcha! Good to know about the payload config doesn’t pass to build and I did the test myself. Got little confused about db url as I’ve given as build time var, without build time it doesn’t passes my build, as per your comment I can replace db url at run time, it means I can have different value at run time and build time ? What if I don’t want to declare it as build time var ? Can I give a mock url for webpack to build ?
Hey
@170674545965924352, would you mind sharing an example of this in your Dockerfile? I've run into a similar problem since doing a Turbopack & Docker deployment and wanted to see which one was causing the issue
ahh of course
Would I be correct in thinking you could leave ARG empty seeing as those values will get populated when you run your container
It requires react environment variables at build time. This is actually not unique to Payload. In a server context, the environment variables can be substituted at run time, but with React, Webpack does the variable replacement during build in order to generate a static bundle
Just to be clear, only variables that require PAYLOAD_PUBLIC are required at build time. All other environment variables can be substituted at run time
Your DB url can be set using runtime environment variables. Db is never used in react and thus Webpack does not require it. Only the express side of payload uses the db
The payload config does not reach your frontend code. It is used to configure your express app and Webpack settings for react
theoretically, you can have a normal (not a public one) env variable + an endpoint on the backend that just returns it. Then, on the frontend, you fetch and save it into some provider to have access anywhere
with server components it's even easier and SSR-friendly, just pass it down to a client component (but as value directly, not the whole
process.env
)
public env vars suck really in a dockerized envs , you only need them for the SSG process (because you do that at the build time).
Star
Discord
online
Get dedicated engineering support directly from the Payload team.