Merge pull request #698 from thecodingmachine/fix/deploy-cleanup
Adding docker-compose.prod.yaml to ease deployment
This commit is contained in:
commit
e07efbdf28
17 changed files with 221 additions and 90 deletions
2
.dockerignore
Normal file
2
.dockerignore
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
**/node_modules/**
|
||||||
|
**/Dockerfile
|
2
.github/workflows/build-and-deploy.yml
vendored
2
.github/workflows/build-and-deploy.yml
vendored
|
@ -159,5 +159,5 @@ jobs:
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
with:
|
||||||
msg: Environment deployed at https://${{ env.GITHUB_REF_SLUG }}.workadventure.test.thecodingmachine.com
|
msg: Environment deployed at https://play.${{ env.GITHUB_REF_SLUG }}.workadventure.test.thecodingmachine.com
|
||||||
check_for_duplicate_msg: true
|
check_for_duplicate_msg: true
|
||||||
|
|
|
@ -15,7 +15,7 @@ In Work Adventure, you can move around your office and talk to your colleagues (
|
||||||
triggered when you move next to a colleague).
|
triggered when you move next to a colleague).
|
||||||
|
|
||||||
|
|
||||||
## Getting started
|
## Setting up a development environment
|
||||||
|
|
||||||
Install Docker.
|
Install Docker.
|
||||||
|
|
||||||
|
@ -101,5 +101,7 @@ Vagrant destroy
|
||||||
* `Vagrant halt`: stop your VM Vagrant.
|
* `Vagrant halt`: stop your VM Vagrant.
|
||||||
* `Vagrant destroy`: delete your VM Vagrant.
|
* `Vagrant destroy`: delete your VM Vagrant.
|
||||||
|
|
||||||
## Features developed
|
## Setting up a production environment
|
||||||
You have more details of features developed in back [README.md](./back/README.md).
|
|
||||||
|
The way you set up your production environment will highly depend on your servers.
|
||||||
|
We provide a production ready `docker-compose` file that you can use as a good starting point in the [contrib/docker](https://github.com/thecodingmachine/workadventure/tree/master/contrib/docker) directory.
|
||||||
|
|
|
@ -1,16 +1,26 @@
|
||||||
FROM thecodingmachine/workadventure-back-base:latest as builder
|
# protobuf build
|
||||||
WORKDIR /var/www/messages
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76 as builder
|
||||||
COPY --chown=docker:docker messages .
|
WORKDIR /usr/src
|
||||||
|
COPY messages .
|
||||||
RUN yarn install && yarn proto
|
RUN yarn install && yarn proto
|
||||||
|
|
||||||
FROM thecodingmachine/nodejs:12
|
# typescript build
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76 as builder2
|
||||||
COPY --chown=docker:docker back .
|
WORKDIR /usr/src
|
||||||
COPY --from=builder --chown=docker:docker /var/www/messages/generated /usr/src/app/src/Messages/generated
|
COPY back/yarn.lock back/package.json ./
|
||||||
RUN yarn install
|
RUN yarn install
|
||||||
|
COPY back .
|
||||||
|
COPY --from=builder /usr/src/generated src/Messages/generated
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
RUN yarn run tsc
|
RUN yarn run tsc
|
||||||
|
|
||||||
CMD ["yarn", "run", "runprod"]
|
# final production image
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76
|
||||||
|
WORKDIR /usr/src
|
||||||
|
COPY back/yarn.lock back/package.json ./
|
||||||
|
COPY --from=builder2 /usr/src/dist /usr/src/dist
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
RUN yarn install --production
|
||||||
|
|
||||||
|
USER node
|
||||||
|
CMD ["yarn", "run", "runprod"]
|
||||||
|
|
|
@ -1,61 +0,0 @@
|
||||||
# Back Features
|
|
||||||
|
|
||||||
## Login
|
|
||||||
To start your game, you must authenticate on the server back.
|
|
||||||
When you are authenticated, the back server return token and room starting.
|
|
||||||
```
|
|
||||||
POST => /login
|
|
||||||
Params :
|
|
||||||
email: email of user.
|
|
||||||
```
|
|
||||||
|
|
||||||
## Join a room
|
|
||||||
When a user is connected, the user can join a room.
|
|
||||||
So you must send emit `join-room` with information user:
|
|
||||||
```
|
|
||||||
Socket.io => 'join-room'
|
|
||||||
|
|
||||||
userId: user id of gamer
|
|
||||||
roomId: room id when user enter in game
|
|
||||||
position: {
|
|
||||||
x: position x on map
|
|
||||||
y: position y on map
|
|
||||||
}
|
|
||||||
```
|
|
||||||
All data users are stocked on socket client.
|
|
||||||
|
|
||||||
## Send position user
|
|
||||||
When user move on the map, you can share new position on back with event `user-position`.
|
|
||||||
The information sent:
|
|
||||||
```
|
|
||||||
Socket.io => 'user-position'
|
|
||||||
|
|
||||||
userId: user id of gamer
|
|
||||||
roomId: room id when user enter in game
|
|
||||||
position: {
|
|
||||||
x: position x on map
|
|
||||||
y: position y on map
|
|
||||||
}
|
|
||||||
```
|
|
||||||
All data users are updated on socket client.
|
|
||||||
|
|
||||||
## Receive positions of all users
|
|
||||||
The application sends position of all users in each room in every few 10 milliseconds.
|
|
||||||
The data will pushed on event `user-position`:
|
|
||||||
```
|
|
||||||
Socket.io => 'user-position'
|
|
||||||
|
|
||||||
[
|
|
||||||
{
|
|
||||||
userId: user id of gamer
|
|
||||||
roomId: room id when user enter in game
|
|
||||||
position: {
|
|
||||||
x: position x on map
|
|
||||||
y: position y on map
|
|
||||||
}
|
|
||||||
},
|
|
||||||
...
|
|
||||||
]
|
|
||||||
```
|
|
||||||
|
|
||||||
[<<< back](../README.md)
|
|
|
@ -1,4 +1,3 @@
|
||||||
const SECRET_KEY = process.env.SECRET_KEY || "THECODINGMACHINE_SECRET_KEY";
|
|
||||||
const MINIMUM_DISTANCE = process.env.MINIMUM_DISTANCE ? Number(process.env.MINIMUM_DISTANCE) : 64;
|
const MINIMUM_DISTANCE = process.env.MINIMUM_DISTANCE ? Number(process.env.MINIMUM_DISTANCE) : 64;
|
||||||
const GROUP_RADIUS = process.env.GROUP_RADIUS ? Number(process.env.GROUP_RADIUS) : 48;
|
const GROUP_RADIUS = process.env.GROUP_RADIUS ? Number(process.env.GROUP_RADIUS) : 48;
|
||||||
const ALLOW_ARTILLERY = process.env.ALLOW_ARTILLERY ? process.env.ALLOW_ARTILLERY == 'true' : false;
|
const ALLOW_ARTILLERY = process.env.ALLOW_ARTILLERY ? process.env.ALLOW_ARTILLERY == 'true' : false;
|
||||||
|
@ -14,7 +13,6 @@ const GRPC_PORT = parseInt(process.env.GRPC_PORT || '50051') || 50051;
|
||||||
export const SOCKET_IDLE_TIMER = parseInt(process.env.SOCKET_IDLE_TIMER as string) || 30; // maximum time (in second) without activity before a socket is closed
|
export const SOCKET_IDLE_TIMER = parseInt(process.env.SOCKET_IDLE_TIMER as string) || 30; // maximum time (in second) without activity before a socket is closed
|
||||||
|
|
||||||
export {
|
export {
|
||||||
SECRET_KEY,
|
|
||||||
MINIMUM_DISTANCE,
|
MINIMUM_DISTANCE,
|
||||||
ADMIN_API_URL,
|
ADMIN_API_URL,
|
||||||
ADMIN_API_TOKEN,
|
ADMIN_API_TOKEN,
|
||||||
|
|
20
contrib/docker/.env.prod.template
Normal file
20
contrib/docker/.env.prod.template
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
# The base domain
|
||||||
|
DOMAIN=workadventure.localhost
|
||||||
|
|
||||||
|
DEBUG_MODE=false
|
||||||
|
JITSI_URL=meet.jit.si
|
||||||
|
# If your Jitsi environment has authentication set up, you MUST set JITSI_PRIVATE_MODE to "true" and you MUST pass a SECRET_JITSI_KEY to generate the JWT secret
|
||||||
|
JITSI_PRIVATE_MODE=false
|
||||||
|
JITSI_ISS=
|
||||||
|
SECRET_JITSI_KEY=
|
||||||
|
|
||||||
|
# URL of the TURN server (needed to "punch a hole" through some networks for P2P connections)
|
||||||
|
TURN_SERVER=
|
||||||
|
TURN_USER=
|
||||||
|
TURN_PASSWORD=
|
||||||
|
|
||||||
|
# The URL used by default, in the form: "/_/global/map/url.json"
|
||||||
|
START_ROOM_URL=/_/global/maps.workadventu.re/Floor0/floor0.json
|
||||||
|
|
||||||
|
# The email address used by Let's encrypt to send renewal warnings (compulsory)
|
||||||
|
ACME_EMAIL=
|
100
contrib/docker/docker-compose.prod.yaml
Normal file
100
contrib/docker/docker-compose.prod.yaml
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
version: "3.3"
|
||||||
|
services:
|
||||||
|
reverse-proxy:
|
||||||
|
image: traefik:v2.3
|
||||||
|
command:
|
||||||
|
- --log.level=WARN
|
||||||
|
#- --api.insecure=true
|
||||||
|
- --providers.docker
|
||||||
|
- --entryPoints.web.address=:80
|
||||||
|
- --entrypoints.web.http.redirections.entryPoint.to=websecure
|
||||||
|
- --entrypoints.web.http.redirections.entryPoint.scheme=https
|
||||||
|
- --entryPoints.websecure.address=:443
|
||||||
|
- --certificatesresolvers.myresolver.acme.email=d.negrier@thecodingmachine.com
|
||||||
|
- --certificatesresolvers.myresolver.acme.storage=/acme.json
|
||||||
|
# used during the challenge
|
||||||
|
- --certificatesresolvers.myresolver.acme.httpchallenge.entrypoint=web
|
||||||
|
ports:
|
||||||
|
- "80:80"
|
||||||
|
- "443:443"
|
||||||
|
# The Web UI (enabled by --api.insecure=true)
|
||||||
|
#- "8080:8080"
|
||||||
|
depends_on:
|
||||||
|
- pusher
|
||||||
|
- front
|
||||||
|
volumes:
|
||||||
|
- /var/run/docker.sock:/var/run/docker.sock
|
||||||
|
- ./acme.json:/acme.json
|
||||||
|
restart: unless-stopped
|
||||||
|
|
||||||
|
|
||||||
|
front:
|
||||||
|
build:
|
||||||
|
context: ../..
|
||||||
|
dockerfile: front/Dockerfile
|
||||||
|
#image: thecodingmachine/workadventure-front:master
|
||||||
|
environment:
|
||||||
|
DEBUG_MODE: "$DEBUG_MODE"
|
||||||
|
JITSI_URL: $JITSI_URL
|
||||||
|
JITSI_PRIVATE_MODE: "$JITSI_PRIVATE_MODE"
|
||||||
|
API_URL: pusher.${DOMAIN}
|
||||||
|
TURN_SERVER: "${TURN_SERVER}"
|
||||||
|
TURN_USER: "${TURN_USER}"
|
||||||
|
TURN_PASSWORD: "${TURN_PASSWORD}"
|
||||||
|
START_ROOM_URL: "${START_ROOM_URL}"
|
||||||
|
labels:
|
||||||
|
- "traefik.http.routers.front.rule=Host(`play.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.front.entryPoints=web,traefik"
|
||||||
|
- "traefik.http.services.front.loadbalancer.server.port=80"
|
||||||
|
- "traefik.http.routers.front-ssl.rule=Host(`play.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.front-ssl.entryPoints=websecure"
|
||||||
|
- "traefik.http.routers.front-ssl.tls=true"
|
||||||
|
- "traefik.http.routers.front-ssl.service=front"
|
||||||
|
- "traefik.http.routers.front-ssl.tls.certresolver=myresolver"
|
||||||
|
restart: unless-stopped
|
||||||
|
|
||||||
|
pusher:
|
||||||
|
build:
|
||||||
|
context: ../..
|
||||||
|
dockerfile: pusher/Dockerfile
|
||||||
|
#image: thecodingmachine/workadventure-pusher:master
|
||||||
|
command: yarn run runprod
|
||||||
|
environment:
|
||||||
|
SECRET_JITSI_KEY: "$SECRET_JITSI_KEY"
|
||||||
|
SECRET_KEY: yourSecretKey
|
||||||
|
API_URL: back:50051
|
||||||
|
JITSI_URL: $JITSI_URL
|
||||||
|
JITSI_ISS: $JITSI_ISS
|
||||||
|
labels:
|
||||||
|
- "traefik.http.routers.pusher.rule=Host(`pusher.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.pusher.entryPoints=web,traefik"
|
||||||
|
- "traefik.http.services.pusher.loadbalancer.server.port=8080"
|
||||||
|
- "traefik.http.routers.pusher-ssl.rule=Host(`pusher.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.pusher-ssl.entryPoints=websecure"
|
||||||
|
- "traefik.http.routers.pusher-ssl.tls=true"
|
||||||
|
- "traefik.http.routers.pusher-ssl.service=pusher"
|
||||||
|
- "traefik.http.routers.pusher-ssl.tls.certresolver=myresolver"
|
||||||
|
restart: unless-stopped
|
||||||
|
|
||||||
|
back:
|
||||||
|
build:
|
||||||
|
context: ../..
|
||||||
|
dockerfile: back/Dockerfile
|
||||||
|
#image: thecodingmachine/workadventure-back:master
|
||||||
|
command: yarn run runprod
|
||||||
|
environment:
|
||||||
|
SECRET_JITSI_KEY: "$SECRET_JITSI_KEY"
|
||||||
|
ADMIN_API_TOKEN: "$ADMIN_API_TOKEN"
|
||||||
|
ADMIN_API_URL: "$ADMIN_API_URL"
|
||||||
|
JITSI_URL: $JITSI_URL
|
||||||
|
JITSI_ISS: $JITSI_ISS
|
||||||
|
labels:
|
||||||
|
- "traefik.http.routers.back.rule=Host(`api.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.back.entryPoints=web"
|
||||||
|
- "traefik.http.services.back.loadbalancer.server.port=8080"
|
||||||
|
- "traefik.http.routers.back-ssl.rule=Host(`api.${DOMAIN}`)"
|
||||||
|
- "traefik.http.routers.back-ssl.entryPoints=websecure"
|
||||||
|
- "traefik.http.routers.back-ssl.tls=true"
|
||||||
|
- "traefik.http.routers.back-ssl.service=back"
|
||||||
|
- "traefik.http.routers.back-ssl.tls.certresolver=myresolver"
|
||||||
|
restart: unless-stopped
|
|
@ -1,7 +1,7 @@
|
||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
set -x
|
set -x
|
||||||
set -o nounset errexit
|
set -o nounset errexit
|
||||||
template_file_index=dist/index.html.tmpl
|
template_file_index=dist/index.tmpl.html
|
||||||
generated_file_index=dist/index.html
|
generated_file_index=dist/index.html
|
||||||
tmp_trackcodefile=/tmp/trackcode
|
tmp_trackcodefile=/tmp/trackcode
|
||||||
|
|
||||||
|
|
|
@ -39,7 +39,16 @@ module.exports = {
|
||||||
plugins: [
|
plugins: [
|
||||||
new HtmlWebpackPlugin(
|
new HtmlWebpackPlugin(
|
||||||
{
|
{
|
||||||
template: './dist/index.html'
|
template: './dist/index.tmpl.html',
|
||||||
|
minify: {
|
||||||
|
collapseWhitespace: true,
|
||||||
|
keepClosingSlash: true,
|
||||||
|
removeComments: false,
|
||||||
|
removeRedundantAttributes: true,
|
||||||
|
removeScriptTypeAttributes: true,
|
||||||
|
removeStyleLinkTypeAttributes: true,
|
||||||
|
useShortDoctype: true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
),
|
),
|
||||||
new webpack.ProvidePlugin({
|
new webpack.ProvidePlugin({
|
||||||
|
|
4
maps/.dockerignore
Normal file
4
maps/.dockerignore
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
/node_modules/
|
||||||
|
/dist/bundle.js
|
||||||
|
/yarn-error.log
|
||||||
|
/Dockerfile
|
4
messages/.dockerignore
Normal file
4
messages/.dockerignore
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
/node_modules/
|
||||||
|
/dist/bundle.js
|
||||||
|
/yarn-error.log
|
||||||
|
/Dockerfile
|
|
@ -1,15 +1,26 @@
|
||||||
FROM thecodingmachine/workadventure-back-base:latest as builder
|
# protobuf build
|
||||||
WORKDIR /var/www/messages
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76 as builder
|
||||||
COPY --chown=docker:docker messages .
|
WORKDIR /usr/src
|
||||||
|
COPY messages .
|
||||||
RUN yarn install && yarn proto
|
RUN yarn install && yarn proto
|
||||||
|
|
||||||
FROM thecodingmachine/nodejs:12
|
# typescript build
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76 as builder2
|
||||||
COPY --chown=docker:docker pusher .
|
WORKDIR /usr/src
|
||||||
COPY --from=builder --chown=docker:docker /var/www/messages/generated /usr/src/app/src/Messages/generated
|
COPY pusher/yarn.lock pusher/package.json ./
|
||||||
RUN yarn install
|
RUN yarn install
|
||||||
|
COPY pusher .
|
||||||
|
COPY --from=builder /usr/src/generated src/Messages/generated
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
RUN yarn run tsc
|
RUN yarn run tsc
|
||||||
|
|
||||||
|
# final production image
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76
|
||||||
|
WORKDIR /usr/src
|
||||||
|
COPY pusher/yarn.lock pusher/package.json ./
|
||||||
|
COPY --from=builder2 /usr/src/dist /usr/src/dist
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
RUN yarn install --production
|
||||||
|
|
||||||
|
USER node
|
||||||
CMD ["yarn", "run", "runprod"]
|
CMD ["yarn", "run", "runprod"]
|
||||||
|
|
17
pusher/Dockerfile.prod
Normal file
17
pusher/Dockerfile.prod
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
FROM node:12.19.0-slim
|
||||||
|
|
||||||
|
RUN mkdir -p /home/node/app && chown -R node:node /home/node/app
|
||||||
|
WORKDIR /home/node/app
|
||||||
|
|
||||||
|
USER node
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
ENV DEBUG=*
|
||||||
|
|
||||||
|
COPY --chown=node:node package.json yarn.lock ./
|
||||||
|
|
||||||
|
RUN yarn install --prod --frozen-lockfile
|
||||||
|
|
||||||
|
COPY --chown=node:node ./dist/ ./dist/
|
||||||
|
|
||||||
|
EXPOSE 8080
|
||||||
|
CMD ["yarn", "run", "runprod"]
|
|
@ -1,9 +1,19 @@
|
||||||
FROM thecodingmachine/nodejs:12
|
# typescript build
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76 as builder2
|
||||||
COPY --chown=docker:docker uploader .
|
WORKDIR /usr/src
|
||||||
|
COPY uploader/yarn.lock uploader/package.json ./
|
||||||
RUN yarn install
|
RUN yarn install
|
||||||
|
COPY uploader .
|
||||||
ENV NODE_ENV=production
|
ENV NODE_ENV=production
|
||||||
RUN yarn run tsc
|
RUN yarn run tsc
|
||||||
|
|
||||||
|
# final production image
|
||||||
|
FROM node:14.15.4-buster-slim@sha256:cbae886186467bbfd274b82a234a1cdfbbd31201c2a6ee63a6893eefcf3c6e76
|
||||||
|
WORKDIR /usr/src
|
||||||
|
COPY uploader/yarn.lock uploader/package.json ./
|
||||||
|
COPY --from=builder2 /usr/src/dist /usr/src/dist
|
||||||
|
ENV NODE_ENV=production
|
||||||
|
RUN yarn install --production
|
||||||
|
|
||||||
|
USER node
|
||||||
CMD ["yarn", "run", "runprod"]
|
CMD ["yarn", "run", "runprod"]
|
||||||
|
|
5
website/.dockerignore
Normal file
5
website/.dockerignore
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
/dist/
|
||||||
|
/node_modules/
|
||||||
|
/dist/bundle.js
|
||||||
|
/yarn-error.log
|
||||||
|
/Dockerfile
|
Loading…
Add table
Add a link
Reference in a new issue