# syntax=docker/dockerfile:1-labs # --------------------------------------------- # Overleaf Community Edition (overleaf/overleaf) # --------------------------------------------- #ARG OVERLEAF_BASE_TAG=sharelatex/sharelatex-base:latest ARG OVERLEAF_BASE_TAG=sharelatex/sharelatex-base:5 FROM $OVERLEAF_BASE_TAG WORKDIR /overleaf # Add required source files for yarn install # ------------------------------------------- COPY --parents libraries/*/package.json .yarn/patches/ services/*/package.json tools/migrations/ package.json yarn.lock .yarnrc.yml /overleaf/ COPY server-ce/genScript.js server-ce/services.js /overleaf/ # Pre-install yarn via corepack so it is available at runtime for all users # ------------------------------------------------------------------------- # Corepack setup, shared between all the images. ENV PATH="/overleaf/node_modules/.bin:$PATH" ENV COREPACK_HOME=/opt/corepack #RUN corepack enable && corepack install -g yarn@4.14.1 RUN corepack enable && corepack prepare yarn@4.14.1 --activate ENV COREPACK_ENABLE_NETWORK=0 # Install yarn dependencies # ------------------------- # The git-sourced @replit/codemirror-* deps are prepared with Yarn Classic, # whose cache lives in /usr/local/share/.cache/yarn. We mount that as a *tmpfs* # (fresh every build) rather than a persistent BuildKit cache: when it was # persistent, BuildKit would garbage-collect/evict part of it between builds, # leaving a half-populated cache that Yarn Classic then tripped over (missing # .yarn-tarball.tgz / EEXIST). A clean cache per build is reliable; the cost is # re-fetching that small set of git deps. The valuable Berry cache # (server-ce-yarn-cache) stays persistent. YARN_NETWORK_CONCURRENCY=1 is kept # as cheap insurance against concurrent writes to the fresh cache. # # Preparing those git deps also makes Yarn Classic fetch esbuild's ~10 # per-platform binaries, whose downloads occasionally arrive truncated ("the # file appears to be corrupt" / missing .yarn-tarball.tgz). Since the tmpfs is # fresh each build there is nothing to fall back to, so we wrap the step in a # small retry loop that wipes the classic cache and re-fetches before failing. RUN --mount=type=cache,target=/root/.cache \ --mount=type=cache,target=/root/.yarn/berry/cache,id=server-ce-yarn-cache \ --mount=type=tmpfs,target=/usr/local/share/.cache/yarn \ --mount=type=tmpfs,target=/tmp \ for i in 1 2 3; do \ node genScript install | YARN_NETWORK_CONCURRENCY=1 bash && exit 0; \ echo "==== install attempt $i failed; wiping Yarn Classic cache and retrying ===="; \ rm -rf /usr/local/share/.cache/yarn/* 2>/dev/null || true; \ done; \ exit 1 # Add the actual source files # --------------------------- COPY --parents libraries/ services/ tools/migrations/ /overleaf/ # Syntax-check all server-side ESM modules before the expensive webpack # compile. node --check parses without executing, so it's fast and safe. # Catches things like escaped backticks from sed substitutions that webpack # never sees (it only bundles frontend code). RUN find services/web/app/src services/web/modules -name '*.mjs' | xargs node --check RUN --mount=type=cache,target=/root/.cache \ --mount=type=cache,target=/root/.yarn/berry/cache,id=server-ce-yarn-cache \ --mount=type=tmpfs,target=/usr/local/share/.cache/yarn \ --mount=type=cache,target=/overleaf/services/web/node_modules/.cache,id=server-ce-webpack-cache \ --mount=type=tmpfs,target=/tmp \ for i in 1 2 3; do \ node genScript compile | YARN_NETWORK_CONCURRENCY=1 bash && exit 0; \ echo "==== compile attempt $i failed; wiping Yarn Classic cache and retrying ===="; \ find /tmp -name pack.log -exec cat {} \; 2>/dev/null || true; \ rm -rf /usr/local/share/.cache/yarn/* 2>/dev/null || true; \ done; \ echo "==== PACK LOGS (all attempts failed) ===="; \ find /tmp -name pack.log -exec cat {} \; 2>/dev/null || true; \ exit 1 # Copy runit service startup scripts to its location # -------------------------------------------------- ADD server-ce/runit /etc/service # Copy runit global settings to its location # ------------------------------------------ ADD server-ce/config/env.sh /etc/overleaf/env.sh # Configure nginx # --------------- ADD server-ce/nginx/nginx.conf.template /etc/nginx/templates/nginx.conf.template ADD server-ce/nginx/overleaf.conf /etc/nginx/sites-enabled/overleaf.conf ADD server-ce/nginx/clsi-nginx.conf /etc/nginx/sites-enabled/clsi-nginx.conf # Configure log rotation # ---------------------- ADD server-ce/logrotate/overleaf /etc/logrotate.d/overleaf RUN chmod 644 /etc/logrotate.d/overleaf # Configure cron tasks # ---------------------- ADD server-ce/cron /overleaf/cron ADD server-ce/config/crontab-history /etc/cron.d/crontab-history RUN chmod 600 /etc/cron.d/crontab-history ADD server-ce/config/crontab-deletion /etc/cron.d/crontab-deletion RUN chmod 600 /etc/cron.d/crontab-deletion # Copy Phusion Image startup and shutdown scripts to their locations # ------------------------------------------------------------------ COPY server-ce/init_scripts/ /etc/my_init.d/ COPY server-ce/init_preshutdown_scripts/ /etc/my_init.pre_shutdown.d/ # Copy app settings files # ----------------------- COPY server-ce/config/settings.js /etc/overleaf/settings.js # Copy history-v1 files # ----------------------- COPY server-ce/config/production.json /overleaf/services/history-v1/config/production.json COPY server-ce/config/custom-environment-variables.json /overleaf/services/history-v1/config/custom-environment-variables.json # Copy grunt thin wrapper # ----------------------- ADD server-ce/bin/grunt /usr/local/bin/grunt RUN chmod +x /usr/local/bin/grunt # Copy history helper scripts # --------------------------- ADD server-ce/bin/flush-history-queues /overleaf/bin/flush-history-queues RUN chmod +x /overleaf/bin/flush-history-queues ADD server-ce/bin/force-history-resyncs /overleaf/bin/force-history-resyncs RUN chmod +x /overleaf/bin/force-history-resyncs # Copy Latexmkrc # ----------------------- COPY server-ce/config/latexmkrc /usr/local/share/latexmk/LatexMk # File that controls open|closed status of the site # ------------------------------------------------- ENV SITE_MAINTENANCE_FILE="/etc/overleaf/site_status" RUN touch $SITE_MAINTENANCE_FILE # Set Environment Variables # -------------------------------- ENV OVERLEAF_CONFIG=/etc/overleaf/settings.js ENV WEB_API_USER="overleaf" ENV ADMIN_PRIVILEGE_AVAILABLE="true" ENV OVERLEAF_APP_NAME="Overleaf Community Edition" ENV OPTIMISE_PDF="true" # Phusion Image timeouts before sending SIGKILL to processes # ---------------------------------------------------------- ENV KILL_PROCESS_TIMEOUT=55 ENV KILL_ALL_PROCESSES_TIMEOUT=55 ENV GRACEFUL_SHUTDOWN_DELAY_SECONDS=1 ENV NODE_ENV="production" ENV LOG_LEVEL="info" EXPOSE 80 ENTRYPOINT ["/sbin/my_init"]