# # Licensed to the Apache Software Foundation (ASF) under one or more # contributor license agreements. See the NOTICE file distributed with # this work for additional information regarding copyright ownership. # The ASF licenses this file to You under the Apache License, Version 2.0 # (the "License"); you may not use this file except in compliance with # the License. You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # ###################################################################### # Node stage to deal with static asset construction ###################################################################### ARG PY_VER=3.10-slim-bookworm # if BUILDPLATFORM is null, set it to 'amd64' (or leave as is otherwise). ARG BUILDPLATFORM=${BUILDPLATFORM:-amd64} FROM --platform=${BUILDPLATFORM} node:20-bullseye-slim AS superset-node ARG NPM_BUILD_CMD="build" # Include translations in the final build. The default supports en only to # reduce complexity and weight for those only using en ARG BUILD_TRANSLATIONS="false" # Used by docker-compose to skip the frontend build, # in dev we mount the repo and build the frontend inside docker ARG DEV_MODE="false" # Include headless browsers? Allows for alerts, reports & thumbnails, but bloats the images ARG INCLUDE_CHROMIUM="true" ARG INCLUDE_FIREFOX="false" # Somehow we need python3 + build-essential on this side of the house to install node-gyp RUN apt-get update -qq \ && apt-get install \ -yqq --no-install-recommends \ build-essential \ python3 \ zstd ENV BUILD_CMD=${NPM_BUILD_CMD} \ PUPPETEER_SKIP_CHROMIUM_DOWNLOAD=true # NPM ci first, as to NOT invalidate previous steps except for when package.json changes RUN --mount=type=bind,target=/frontend-mem-nag.sh,src=./docker/frontend-mem-nag.sh \ /frontend-mem-nag.sh WORKDIR /app/superset-frontend # Creating empty folders to avoid errors when running COPY later on RUN mkdir -p /app/superset/static/assets RUN --mount=type=bind,target=./package.json,src=./superset-frontend/package.json \ --mount=type=bind,target=./package-lock.json,src=./superset-frontend/package-lock.json \ if [ "$DEV_MODE" = "false" ]; then \ npm ci; \ else \ echo "Skipping 'npm ci' in dev mode"; \ fi # Runs the webpack build process COPY superset-frontend /app/superset-frontend # This copies the .po files needed for translation RUN mkdir -p /app/superset/translations COPY superset/translations /app/superset/translations RUN if [ "$DEV_MODE" = "false" ]; then \ BUILD_TRANSLATIONS=$BUILD_TRANSLATIONS npm run ${BUILD_CMD}; \ else \ echo "Skipping 'npm run ${BUILD_CMD}' in dev mode"; \ fi # Compiles .json files from the .po files, then deletes the .po files RUN if [ "$BUILD_TRANSLATIONS" = "true" ]; then \ npm run build-translation; \ else \ echo "Skipping translations as requested by build flag"; \ fi RUN rm /app/superset/translations/*/LC_MESSAGES/*.po RUN rm /app/superset/translations/messages.pot ###################################################################### # Final lean image... ###################################################################### FROM python:${PY_VER} AS lean # Include translations in the final build. The default supports en only to # reduce complexity and weight for those only using en ARG BUILD_TRANSLATIONS="false" WORKDIR /app ENV LANG=C.UTF-8 \ LC_ALL=C.UTF-8 \ SUPERSET_ENV=production \ FLASK_APP="superset.app:create_app()" \ PYTHONPATH="/app/pythonpath" \ SUPERSET_HOME="/app/superset_home" \ SUPERSET_PORT=8088 RUN mkdir -p ${PYTHONPATH} superset/static requirements superset-frontend apache_superset.egg-info requirements \ && useradd --user-group -d ${SUPERSET_HOME} -m --no-log-init --shell /bin/bash superset \ && apt-get update -qq && apt-get install -yqq --no-install-recommends \ curl \ libsasl2-dev \ libsasl2-modules-gssapi-mit \ libpq-dev \ libecpg-dev \ libldap2-dev \ && touch superset/static/version_info.json \ && chown -R superset:superset ./* \ && rm -rf /var/lib/apt/lists/* COPY --chown=superset:superset pyproject.toml setup.py MANIFEST.in README.md ./ # setup.py uses the version information in package.json COPY --chown=superset:superset superset-frontend/package.json superset-frontend/ COPY --chown=superset:superset requirements/base.txt requirements/ RUN --mount=type=cache,target=/root/.cache/pip \ apt-get update -qq && apt-get install -yqq --no-install-recommends \ build-essential \ && pip install --no-cache-dir --upgrade setuptools pip \ && pip install --no-cache-dir -r requirements/base.txt \ && apt-get autoremove -yqq --purge build-essential \ && rm -rf /var/lib/apt/lists/* # Copy the compiled frontend assets COPY --chown=superset:superset --from=superset-node /app/superset/static/assets superset/static/assets ## Lastly, let's install superset itself COPY --chown=superset:superset superset superset RUN --mount=type=cache,target=/root/.cache/pip \ pip install --no-cache-dir -e . # Copy the .json translations from the frontend layer COPY --chown=superset:superset --from=superset-node /app/superset/translations superset/translations # Compile translations for the backend - this generates .mo files, then deletes the .po files COPY ./scripts/translations/generate_mo_files.sh ./scripts/translations/ RUN if [ "$BUILD_TRANSLATIONS" = "true" ]; then \ ./scripts/translations/generate_mo_files.sh \ && chown -R superset:superset superset/translations \ && rm superset/translations/messages.pot \ && rm superset/translations/*/LC_MESSAGES/*.po; \ else \ echo "Skipping translations as requested by build flag"; \ fi COPY --chmod=755 ./docker/run-server.sh /usr/bin/ USER superset HEALTHCHECK CMD curl -f "http://localhost:${SUPERSET_PORT}/health" EXPOSE ${SUPERSET_PORT} CMD ["/usr/bin/run-server.sh"] ###################################################################### # Dev image... ###################################################################### FROM lean AS dev USER root RUN apt-get update -qq \ && apt-get install -yqq --no-install-recommends \ libnss3 \ libdbus-glib-1-2 \ libgtk-3-0 \ libx11-xcb1 \ libasound2 \ libxtst6 \ git \ pkg-config \ && rm -rf /var/lib/apt/lists/* RUN --mount=type=cache,target=/root/.cache/pip \ pip install --no-cache-dir playwright RUN playwright install-deps RUN if [ "$INCLUDE_CHROMIUM" = "true" ]; then \ playwright install chromium; \ else \ echo "Skipping translations in dev mode"; \ fi # Install GeckoDriver WebDriver ARG GECKODRIVER_VERSION=v0.34.0 \ FIREFOX_VERSION=125.0.3 RUN if [ "$INCLUDE_FIREFOX" = "true" ]; then \ apt-get update -qq \ && apt-get install -yqq --no-install-recommends wget bzip2 \ && wget -q https://github.com/mozilla/geckodriver/releases/download/${GECKODRIVER_VERSION}/geckodriver-${GECKODRIVER_VERSION}-linux64.tar.gz -O - | tar xfz - -C /usr/local/bin \ && wget -q https://download-installer.cdn.mozilla.net/pub/firefox/releases/${FIREFOX_VERSION}/linux-x86_64/en-US/firefox-${FIREFOX_VERSION}.tar.bz2 -O - | tar xfj - -C /opt \ && ln -s /opt/firefox/firefox /usr/local/bin/firefox \ && apt-get autoremove -yqq --purge wget bzip2 && rm -rf /var/[log,tmp]/* /tmp/* /var/lib/apt/lists/*; \ fi # Installing mysql client os-level dependencies in dev image only because GPL RUN apt-get install -yqq --no-install-recommends \ default-libmysqlclient-dev \ && rm -rf /var/lib/apt/lists/* COPY --chown=superset:superset requirements/development.txt requirements/ RUN --mount=type=cache,target=/root/.cache/pip \ apt-get update -qq && apt-get install -yqq --no-install-recommends \ build-essential \ && pip install --no-cache-dir -r requirements/development.txt \ && apt-get autoremove -yqq --purge build-essential \ && rm -rf /var/lib/apt/lists/* USER superset ###################################################################### # CI image... ###################################################################### FROM lean AS ci COPY --chown=superset:superset --chmod=755 ./docker/*.sh /app/docker/ CMD ["/app/docker/docker-ci.sh"]