mirror of
https://github.com/freeCodeCamp/freeCodeCamp.git
synced 2025-12-19 18:18:27 -05:00
feat: create Docker images for api-server (#51142)
This commit is contained in:
committed by
GitHub
parent
05d19b8b42
commit
ade2092e1f
91
.github/workflows/temporary-container-checks.yml
vendored
91
.github/workflows/temporary-container-checks.yml
vendored
@@ -28,23 +28,6 @@ jobs:
|
|||||||
- name: Checkout Source Files
|
- name: Checkout Source Files
|
||||||
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
||||||
|
|
||||||
- name: Checkout client-config
|
|
||||||
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
|
||||||
with:
|
|
||||||
repository: freeCodeCamp/client-config
|
|
||||||
path: client-config
|
|
||||||
|
|
||||||
- name: Setup pnpm
|
|
||||||
uses: pnpm/action-setup@v2
|
|
||||||
with:
|
|
||||||
version: 8
|
|
||||||
|
|
||||||
- name: Use Node.js ${{ matrix.node-version }}
|
|
||||||
uses: actions/setup-node@5e21ff4d9bc1a8cf6de233a3057d20ec6b3fb69d # v3
|
|
||||||
with:
|
|
||||||
node-version: ${{ matrix.node-version }}
|
|
||||||
cache: pnpm
|
|
||||||
|
|
||||||
- name: Create Image
|
- name: Create Image
|
||||||
run: |
|
run: |
|
||||||
docker build \
|
docker build \
|
||||||
@@ -88,6 +71,32 @@ jobs:
|
|||||||
# name: webpack-stats
|
# name: webpack-stats
|
||||||
# path: client/public/stats.json
|
# path: client/public/stats.json
|
||||||
|
|
||||||
|
build-api:
|
||||||
|
name: Build Api (Container)
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
node-version: [18.x]
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout Source Files
|
||||||
|
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
||||||
|
|
||||||
|
- name: Create Image
|
||||||
|
run: |
|
||||||
|
docker build \
|
||||||
|
-t fcc-api \
|
||||||
|
-f docker/api/Dockerfile .
|
||||||
|
|
||||||
|
- name: Save Image
|
||||||
|
run: docker save fcc-api > api-artifact.tar
|
||||||
|
|
||||||
|
- name: Upload Api Artifact
|
||||||
|
uses: actions/upload-artifact@0b7f8abb1508181956e8e162db84b466c27e18ce # v3
|
||||||
|
with:
|
||||||
|
name: api-artifact
|
||||||
|
path: api-artifact.tar
|
||||||
|
|
||||||
build-new-api:
|
build-new-api:
|
||||||
name: Build New Api (Container)
|
name: Build New Api (Container)
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
@@ -108,7 +117,7 @@ jobs:
|
|||||||
cypress-run:
|
cypress-run:
|
||||||
name: Test
|
name: Test
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
needs: build-client
|
needs: [build-client, build-api]
|
||||||
strategy:
|
strategy:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
@@ -116,17 +125,6 @@ jobs:
|
|||||||
browsers: [chrome]
|
browsers: [chrome]
|
||||||
node-version: [18.x]
|
node-version: [18.x]
|
||||||
|
|
||||||
services:
|
|
||||||
mongodb:
|
|
||||||
image: mongo:4.4
|
|
||||||
ports:
|
|
||||||
- 27017:27017
|
|
||||||
# We need mailhog to catch any emails the api tries to send.
|
|
||||||
mailhog:
|
|
||||||
image: mailhog/mailhog
|
|
||||||
ports:
|
|
||||||
- 1025:1025
|
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Set Action Environment Variables
|
- name: Set Action Environment Variables
|
||||||
run: |
|
run: |
|
||||||
@@ -137,13 +135,8 @@ jobs:
|
|||||||
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
uses: actions/checkout@f43a0e5ff2bd294095638e18286ca9a3d1956744 # v3
|
||||||
|
|
||||||
- uses: actions/download-artifact@9bc31d5ccc31df68ecc42ccf4149144866c47d8a # v3
|
- uses: actions/download-artifact@9bc31d5ccc31df68ecc42ccf4149144866c47d8a # v3
|
||||||
with:
|
|
||||||
name: client-artifact
|
|
||||||
|
|
||||||
- name: Load Client Image
|
|
||||||
run: |
|
|
||||||
docker load < client-artifact.tar
|
|
||||||
|
|
||||||
|
# Cypress calls some pnpm scripts, so we need to install pnpm.
|
||||||
- name: Setup pnpm
|
- name: Setup pnpm
|
||||||
uses: pnpm/action-setup@v2
|
uses: pnpm/action-setup@v2
|
||||||
with:
|
with:
|
||||||
@@ -153,34 +146,24 @@ jobs:
|
|||||||
uses: actions/setup-node@5e21ff4d9bc1a8cf6de233a3057d20ec6b3fb69d # v3
|
uses: actions/setup-node@5e21ff4d9bc1a8cf6de233a3057d20ec6b3fb69d # v3
|
||||||
with:
|
with:
|
||||||
node-version: ${{ matrix.node-version }}
|
node-version: ${{ matrix.node-version }}
|
||||||
# cypress-io/github-action caches the store, so we should not cache it
|
|
||||||
# here.
|
|
||||||
|
|
||||||
- name: Set freeCodeCamp Environment Variables
|
- name: Install Dependencies
|
||||||
run: cp sample.env .env
|
run: pnpm install
|
||||||
|
|
||||||
- name: Install and Build
|
- name: Load Images
|
||||||
run: |
|
run: |
|
||||||
pnpm install
|
docker load < client-artifact/client-artifact.tar
|
||||||
pnpm run create:shared
|
docker load < api-artifact/api-artifact.tar
|
||||||
pnpm run build:curriculum
|
|
||||||
pnpm run build:server
|
|
||||||
|
|
||||||
- name: Seed Database
|
- name: Set freeCodeCamp Environment Variables (needed by api)
|
||||||
run: pnpm run seed
|
run: cp sample.env .env
|
||||||
|
|
||||||
# start-ci uses pm2, so it needs to be installed globally
|
|
||||||
- name: Install pm2
|
|
||||||
run: npm i -g pm2
|
|
||||||
|
|
||||||
- name: Cypress run
|
- name: Cypress run
|
||||||
uses: cypress-io/github-action@v4
|
uses: cypress-io/github-action@v4
|
||||||
with:
|
with:
|
||||||
record: ${{ env.CYPRESS_RECORD_KEY != 0 }}
|
record: ${{ env.CYPRESS_RECORD_KEY != 0 }}
|
||||||
start: |
|
start: docker compose up -d
|
||||||
pnpm start:server
|
wait-on: http://localhost:8000, http://localhost:3000
|
||||||
docker compose up -d
|
|
||||||
wait-on: http://localhost:8000
|
|
||||||
wait-on-timeout: 1200
|
wait-on-timeout: 1200
|
||||||
config: baseUrl=http://localhost:8000
|
config: baseUrl=http://localhost:8000
|
||||||
browser: ${{ matrix.browsers }}
|
browser: ${{ matrix.browsers }}
|
||||||
|
|||||||
@@ -4,7 +4,15 @@ const path = require('path');
|
|||||||
const dotenv = require('dotenv');
|
const dotenv = require('dotenv');
|
||||||
|
|
||||||
const filePath = path.resolve(__dirname, '..', '.env');
|
const filePath = path.resolve(__dirname, '..', '.env');
|
||||||
const env = dotenv.parse(fs.readFileSync(filePath));
|
let env = {};
|
||||||
|
try {
|
||||||
|
env = dotenv.parse(fs.readFileSync(filePath));
|
||||||
|
} catch (e) {
|
||||||
|
console.log(
|
||||||
|
"If you're setting the env vars in the shell, it should be fine (you can probably ignore the error)."
|
||||||
|
);
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
// without this, loopback cannot find strong-error-handler. Node can, so we know
|
// without this, loopback cannot find strong-error-handler. Node can, so we know
|
||||||
// there's no _real_ issue, but loopback is not able to find it.
|
// there's no _real_ issue, but loopback is not able to find it.
|
||||||
const loopbackModuleResolutionHack = path.resolve(
|
const loopbackModuleResolutionHack = path.resolve(
|
||||||
|
|||||||
@@ -20,7 +20,10 @@
|
|||||||
"main": "none",
|
"main": "none",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"babel-dev-server": "babel-node --inspect=0.0.0.0 ./src/server/index.js",
|
"babel-dev-server": "babel-node --inspect=0.0.0.0 ./src/server/index.js",
|
||||||
|
"prebuild": "pnpm common-setup",
|
||||||
"build": "babel src --out-dir lib --ignore '/**/*.test.js' --copy-files --no-copy-ignored",
|
"build": "babel src --out-dir lib --ignore '/**/*.test.js' --copy-files --no-copy-ignored",
|
||||||
|
"common-setup": "pnpm -w run create:shared",
|
||||||
|
"predevelop": "pnpm common-setup",
|
||||||
"develop": "node src/development-start.js",
|
"develop": "node src/development-start.js",
|
||||||
"start": "cross-env DEBUG=fcc* node lib/production-start.js"
|
"start": "cross-env DEBUG=fcc* node lib/production-start.js"
|
||||||
},
|
},
|
||||||
@@ -47,6 +50,8 @@
|
|||||||
"express-validator": "6.14.1",
|
"express-validator": "6.14.1",
|
||||||
"helmet": "3.23.3",
|
"helmet": "3.23.3",
|
||||||
"helmet-csp": "2.10.0",
|
"helmet-csp": "2.10.0",
|
||||||
|
"joi": "17.9.2",
|
||||||
|
"joi-objectid": "3.0.1",
|
||||||
"jsonwebtoken": "8.5.1",
|
"jsonwebtoken": "8.5.1",
|
||||||
"lodash": "4.17.21",
|
"lodash": "4.17.21",
|
||||||
"loopback": "3.28.0",
|
"loopback": "3.28.0",
|
||||||
@@ -82,8 +87,6 @@
|
|||||||
"@babel/plugin-proposal-optional-chaining": "7.17.12",
|
"@babel/plugin-proposal-optional-chaining": "7.17.12",
|
||||||
"@babel/preset-env": "7.18.0",
|
"@babel/preset-env": "7.18.0",
|
||||||
"@babel/register": "7.17.7",
|
"@babel/register": "7.17.7",
|
||||||
"joi": "17.9.2",
|
|
||||||
"joi-objectid": "3.0.1",
|
|
||||||
"loopback-component-explorer": "6.4.0",
|
"loopback-component-explorer": "6.4.0",
|
||||||
"nodemon": "2.0.16",
|
"nodemon": "2.0.16",
|
||||||
"smee-client": "1.2.3"
|
"smee-client": "1.2.3"
|
||||||
|
|||||||
@@ -20,6 +20,7 @@ module.exports = (on, config) => {
|
|||||||
// `on` is used to hook into various events Cypress emits
|
// `on` is used to hook into various events Cypress emits
|
||||||
// `config` is the resolved Cypress config
|
// `config` is the resolved Cypress config
|
||||||
config.env = config.env || {};
|
config.env = config.env || {};
|
||||||
|
// TODO: once we've containerized the API, we can remove this.
|
||||||
on('before:run', () => {
|
on('before:run', () => {
|
||||||
if (!existsSync('../../shared/config/curriculum.json')) {
|
if (!existsSync('../../shared/config/curriculum.json')) {
|
||||||
execSync('pnpm run build:curriculum');
|
execSync('pnpm run build:curriculum');
|
||||||
|
|||||||
@@ -1,7 +1,32 @@
|
|||||||
services:
|
services:
|
||||||
|
mongo:
|
||||||
|
image: mongo
|
||||||
|
ports:
|
||||||
|
- '27017:27017'
|
||||||
|
mailhog:
|
||||||
|
restart: unless-stopped
|
||||||
|
image: mailhog/mailhog
|
||||||
|
ports:
|
||||||
|
- '1025:1025'
|
||||||
|
- '8025:8025'
|
||||||
|
api:
|
||||||
|
depends_on:
|
||||||
|
- mongo
|
||||||
|
- mailhog
|
||||||
|
image: fcc-api
|
||||||
|
env_file:
|
||||||
|
- .env
|
||||||
|
environment:
|
||||||
|
# The api cannot connect to mongodb or mailhog via localhost from inside the
|
||||||
|
# container, so we have to override these variables.
|
||||||
|
- MONGOHQ_URL=mongodb://mongo:27017/freecodecamp?directConnection=true
|
||||||
|
- MAILHOG_HOST=mailhog
|
||||||
|
ports:
|
||||||
|
# PORT is used by the new api, so we use the less generic API_PORT to
|
||||||
|
# avoid conflicts.
|
||||||
|
- '${API_PORT:-3000}:3000'
|
||||||
client:
|
client:
|
||||||
image: fcc-client
|
image: fcc-client
|
||||||
ports:
|
ports:
|
||||||
# PORT is used by the new api, so we use the less generic HOST_PORT to avoid
|
# Same principle as above (avoiding conflicts)
|
||||||
# conflicts.
|
- '${CLIENT_PORT:-8000}:8000'
|
||||||
- '${HOST_PORT:-8000}:8000'
|
|
||||||
|
|||||||
@@ -1,35 +1,61 @@
|
|||||||
FROM node:16-buster AS builder
|
# bookworm was only released on 10-6-2023, so is a little too new.
|
||||||
# Install doppler CLI
|
FROM node:18-bullseye AS builder
|
||||||
RUN (curl -Ls --tlsv1.2 --proto "=https" --retry 3 https://cli.doppler.com/install.sh) | sh -s -- --verify-signature
|
ENV PNPM_HOME="/pnpm"
|
||||||
|
ENV PATH="$PNPM_HOME:$PATH"
|
||||||
|
# global installs need root permissions, so have to happen before we switch to
|
||||||
|
# the node user
|
||||||
|
RUN npm i -g pnpm@8
|
||||||
# node images create a non-root user that we can use
|
# node images create a non-root user that we can use
|
||||||
USER node
|
USER node
|
||||||
WORKDIR /home/node/build
|
WORKDIR /home/node/build
|
||||||
COPY --chown=node:node . .
|
COPY --chown=node:node . .
|
||||||
# Pass `DOPPLER_TOKEN` at build time to create an encrypted snapshot for high-availability
|
|
||||||
ARG DOPPLER_TOKEN
|
|
||||||
RUN \
|
|
||||||
doppler secrets download doppler.encrypted.json &&\
|
|
||||||
pnpm install --no-progress --ignore-scripts &&\
|
|
||||||
doppler run --fallback=doppler.encrypted.json --command="npm run create:config" &&\
|
|
||||||
doppler run --fallback=doppler.encrypted.json --command="npm run build:curriculum" &&\
|
|
||||||
doppler run --fallback=doppler.encrypted.json --command="npm run build:server"
|
|
||||||
|
|
||||||
FROM node:16-alpine as depends
|
# TODO: figure out why the cache is getting invalidated. Is it in part because
|
||||||
USER node
|
# we're not ignoring THIS file? Or do we need corepack?
|
||||||
WORKDIR /home/node/depends
|
|
||||||
COPY --chown=node:node . .
|
|
||||||
RUN pnpm install --production --workspace=api-server --no-progress --ignore-scripts
|
|
||||||
|
|
||||||
FROM node:16-alpine
|
# We have to prevent pnpm from deduping peer dependencies because otherwise it
|
||||||
|
# will install all of the packages, not just api-server. Also, pnpm deploy is
|
||||||
|
# not useful since we need to install more than one package.
|
||||||
|
|
||||||
|
RUN pnpm config set dedupe-peer-dependents false
|
||||||
|
RUN pnpm -F=api-server -F=tools/scripts/build -F=challenge-parser -F=curriculum -F=shared \
|
||||||
|
install --frozen-lockfile --ignore-scripts
|
||||||
|
|
||||||
|
# The api needs to source curriculum.json and build:curriculum relies on the
|
||||||
|
# following env vars.
|
||||||
|
ARG SHOW_UPCOMING_CHANGES=false
|
||||||
|
ENV SHOW_UPCOMING_CHANGES=$SHOW_UPCOMING_CHANGES
|
||||||
|
ARG SHOW_NEW_CURRICULUM=true
|
||||||
|
ENV SHOW_NEW_CURRICULUM=$SHOW_NEW_CURRICULUM
|
||||||
|
RUN pnpm build:curriculum
|
||||||
|
|
||||||
|
RUN pnpm build:server
|
||||||
|
|
||||||
|
FROM node:18-bullseye AS deps
|
||||||
|
|
||||||
|
WORKDIR /home/node/build
|
||||||
|
COPY --chown=node:node pnpm*.yaml .
|
||||||
|
COPY --chown=node:node api-server/package.json api-server/package.json
|
||||||
|
COPY --chown=node:node shared/package.json shared/package.json
|
||||||
|
|
||||||
|
RUN npm i -g pnpm@8
|
||||||
|
# Prevent pnpm installing unnecessary packages (see above)
|
||||||
|
RUN pnpm config set dedupe-peer-dependents false
|
||||||
|
RUN pnpm -F=api-server -F=shared install --prod --ignore-scripts
|
||||||
|
|
||||||
|
FROM node:18-alpine
|
||||||
RUN npm i -g pm2@4
|
RUN npm i -g pm2@4
|
||||||
USER node
|
USER node
|
||||||
WORKDIR /home/node/api
|
WORKDIR /home/node/fcc
|
||||||
|
COPY --from=builder --chown=node:node /home/node/build/api-server/config/ api-server/config/
|
||||||
COPY --from=builder --chown=node:node /home/node/build/api-server/lib/ api-server/lib/
|
COPY --from=builder --chown=node:node /home/node/build/api-server/lib/ api-server/lib/
|
||||||
COPY --from=builder --chown=node:node /home/node/build/utils/ utils/
|
COPY --from=builder --chown=node:node /home/node/build/api-server/ecosystem.config.js api-server/ecosystem.config.js
|
||||||
COPY --from=builder --chown=node:node /home/node/build/config/ config/
|
COPY --from=builder --chown=node:node /home/node/build/api-server/package.json api-server/package.json
|
||||||
COPY --from=depends --chown=node:node /home/node/depends/api-server/node_modules/ api-server/node_modules/
|
COPY --from=builder --chown=node:node /home/node/build/shared/ shared/
|
||||||
COPY --from=depends --chown=node:node /home/node/depends/node_modules/ node_modules/
|
COPY --from=builder --chown=node:node /home/node/build/package.json package.json
|
||||||
WORKDIR /home/node/api/api-server
|
COPY --from=deps --chown=node:node /home/node/build/node_modules/ node_modules/
|
||||||
CMD ["pm2-runtime", "./lib/production-start.js"]
|
COPY --from=deps --chown=node:node /home/node/build/api-server/node_modules/ api-server/node_modules/
|
||||||
|
COPY --from=deps --chown=node:node /home/node/build/shared/node_modules/ shared/node_modules/
|
||||||
|
|
||||||
|
CMD ["pm2-runtime", "start", "api-server/ecosystem.config.js"]
|
||||||
|
|
||||||
# TODO: don't copy mocks/fixtures
|
|
||||||
|
|||||||
333
pnpm-lock.yaml
generated
333
pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user