2020-03-05 16:39:32 +00:00
## Getting Started
2017-07-18 09:26:31 +00:00
2020-03-02 10:59:25 +05:30
The templates in this repository will help deploy Frappe/ERPNext docker in a production environment.
2017-08-03 16:12:46 +05:30
2020-03-02 10:59:25 +05:30
This docker installation takes care of the following:
2017-07-18 09:26:31 +00:00
2020-03-02 10:59:25 +05:30
* Setting up the desired version of Frappe/ERPNext.
* Setting up all the system requirements: eg. MariaDB, Node, Redis.
* [OPTIONAL] Configuring networking for remote access and setting up LetsEncrypt
2017-07-18 09:26:31 +00:00
2020-03-08 20:29:17 +00:00
For docker based development refer to this [README ](development/README.md )
2017-07-24 15:34:28 +05:30
2020-03-05 16:39:32 +00:00
## Deployment
2017-07-24 15:19:32 +05:30
2020-03-05 16:39:32 +00:00
### Setting up Pre-requisites
2017-08-14 17:16:10 +05:30
2020-03-02 10:59:25 +05:30
This repository requires Docker and Git to be setup on the instance to be used.
2017-08-14 17:16:10 +05:30
2020-03-05 16:39:32 +00:00
### Cloning the repository and preliminary steps
2017-07-24 15:19:32 +05:30
2020-03-02 10:59:25 +05:30
Clone this repository somewhere in your system:
2017-07-24 16:41:51 +05:30
2020-03-02 10:59:25 +05:30
```sh
git clone https://github.com/frappe/frappe_docker.git
cd frappe_docker
```
2017-07-24 15:19:32 +05:30
2020-03-02 10:59:25 +05:30
Copy the example docker environment file to `.env` :
2017-07-24 15:30:18 +05:30
2020-03-05 16:39:32 +00:00
```sh
2020-03-02 10:59:25 +05:30
cp installation/env-example installation/.env
```
2017-08-14 17:16:10 +05:30
2020-03-02 10:59:25 +05:30
Make a directory for sites:
2017-07-24 15:19:32 +05:30
2020-03-02 10:59:25 +05:30
```sh
mkdir installation/sites
```
2017-07-24 16:11:20 +05:30
2020-03-05 16:39:32 +00:00
### Setup Environment Variables
2017-07-24 15:19:32 +05:30
2020-03-02 10:59:25 +05:30
To get started, copy the existing `env-example` file to `.env` inside the `installation` directory. By default, the file will contain the following variables:
2017-07-24 15:19:32 +05:30
2020-03-02 10:59:25 +05:30
- `VERSION=edge`
- In this case, `edge` corresponds to `develop` . To setup any other version, you may use the branch name or version specific tags. (eg. version-12, v11.1.15, v11)
- `MYSQL_ROOT_PASSWORD=admin`
2020-03-08 20:29:17 +00:00
- Bootstraps a MariaDB container with this value set as the root password. If a managed MariaDB instance is used, there is no need to set the password here.
2020-03-02 10:59:25 +05:30
- `MARIADB_HOST=mariadb`
2020-03-08 20:29:17 +00:00
- Sets the hostname to `mariadb` . This is required if the database is managed by the containerized MariaDB instance.
- In case of a separately managed database setups, set the value to the database's hostname/IP/domain.
2020-03-02 10:59:25 +05:30
- `SITES=site1.domain.com,site2.domain.com`
2020-03-08 20:29:17 +00:00
- List of sites that are part of the deployment "bench" Each site is separated by a comma(,).
- If LetsEncrypt is being setup, make sure that the DNS for all the site's domains correctly point to the current instance.
2020-03-02 10:59:25 +05:30
- `LETSENCRYPT_EMAIL=your.email@your.domain.com`
2020-03-05 16:39:32 +00:00
- Email for LetsEncrypt expiry notification. This is only required if you are setting up LetsEncrypt.
2020-03-08 20:17:39 +00:00
### Local deployment for testing
2020-03-05 16:39:32 +00:00
2020-03-08 20:17:39 +00:00
For trying out locally or to develop apps using ERPNext REST API port 80 must be published.
2020-03-08 20:29:17 +00:00
The first command will start the containers; the second command will publish the port of the *-nginx container.
2020-03-05 16:39:32 +00:00
2020-03-06 06:45:54 +05:30
For Erpnext:
2020-03-05 16:39:32 +00:00
```sh
2020-03-06 06:45:54 +05:30
# Start services
docker-compose \
2020-03-05 16:39:32 +00:00
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
2020-03-06 06:45:54 +05:30
--project-directory installation up -d
# Publish port
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
--project-directory installation run --publish 80:80 -d erpnext-nginx
2020-03-05 16:39:32 +00:00
```
For Frappe:
2020-03-06 06:45:54 +05:30
2020-03-05 16:39:32 +00:00
```sh
2020-03-06 06:45:54 +05:30
# Start services
docker-compose \
2020-03-05 16:39:32 +00:00
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-frappe.yml \
2020-03-06 06:45:54 +05:30
--project-directory installation up -d
# Publish port
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-frappe.yml \
--project-directory installation run --publish 80:80 -d frappe-nginx
2020-03-05 16:39:32 +00:00
```
2020-03-08 20:17:39 +00:00
Make sure to replace `<project-name>` with the desired name you wish to set for the project.
2020-03-05 16:39:32 +00:00
2020-03-08 20:17:39 +00:00
Notes:
2020-03-08 20:29:17 +00:00
- The local deployment is for testing and REST API development purpose only
- A complete development environment is available [here ](Development/README.md )
- The site names are limited to patterns matching \*.localhost by default
- Additional site name patterns can be added by editing /etc/hosts of your host machine
2020-03-05 16:39:32 +00:00
### Deployment for production
#### Setup Letsencrypt Nginx Proxy Companion
2020-03-06 06:45:54 +05:30
Letsencrypt Nginx Proxy Companion can optionally be setup to provide SSL. This is recommended for instances accessed over the internet.
2020-03-05 16:39:32 +00:00
2020-03-08 20:29:17 +00:00
Your DNS will need to be configured correctly for Letsencrypt to verify your domain.
2020-03-05 16:39:32 +00:00
To setup the proxy companion, run the following commands:
```sh
cd $HOME
git clone https://github.com/evertramos/docker-compose-letsencrypt-nginx-proxy-companion.git
cd docker-compose-letsencrypt-nginx-proxy-companion
cp .env.sample .env
./start.sh
```
2020-03-08 20:17:39 +00:00
For more details, see the [Letsencrypt Nginx Proxy Companion github repo ](https://github.com/evertramos/docker-compose-letsencrypt-nginx-proxy-companion ). Letsencrypt Nginx Proxy Companion github repo works by automatically proxying to containers with the `VIRTUAL_HOST` environmental variable.
2018-05-04 10:54:30 -07:00
2020-03-02 10:59:25 +05:30
#### Start Frappe/ERPNext Services
2017-07-18 09:26:31 +00:00
2020-03-05 16:39:32 +00:00
To start the Frappe/ERPNext services for production, run the following command:
2017-07-18 09:26:31 +00:00
2020-03-02 10:59:25 +05:30
```sh
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
-f installation/docker-compose-networks.yml \
--project-directory installation up -d
```
2017-07-18 09:26:31 +00:00
2020-03-02 10:59:25 +05:30
Make sure to replace `<project-name>` with any desired name you wish to set for the project.
2020-03-05 16:39:32 +00:00
Note: use `docker-compose-frappe.yml` in case you need only Frappe without ERPNext.
### Docker containers
2020-03-08 20:29:17 +00:00
This repository contains the following docker-compose files, each one containing the described images:
2020-03-05 16:39:32 +00:00
* docker-compose-common.yml
* redis-cache
* volume: redis-cache-vol
* redis-queue
* volume: redis-queue-vol
* redis-socketio
* volume: redis-socketio-vol
* mariadb: main database
* volume: mariadb-vol
* docker-compose-erpnext.yml
2020-03-06 06:45:54 +05:30
* erpnext-nginx: serves static assets and proxies web request to the appropriate container, allowing to offer all services on the same port.
2020-03-05 16:39:32 +00:00
* volume: assets
* erpnext-python: main application code
2020-03-06 06:45:54 +05:30
* frappe-socketio: enables realtime communication to the user interface through websockets
2020-03-05 16:39:32 +00:00
* frappe-worker-default: background runner
* frappe-worker-short: background runner for short-running jobs
* frappe-worker-long: background runner for long-running jobs
* frappe-schedule
* docker-compose-frappe.yml
2020-03-06 06:45:54 +05:30
* frappe-nginx: serves static assets and proxies web request to the appropriate container, allowing to offer all services on the same port.
2020-03-05 16:39:32 +00:00
* volume: assets
* erpnext-python: main application code
2020-03-06 06:45:54 +05:30
* frappe-socketio: enables realtime communication to the user interface through websockets
2020-03-05 16:39:32 +00:00
* frappe-worker-default: background runner
* frappe-worker-short: background runner for short-running jobs
* frappe-worker-long: background runner for long-running jobs
* frappe-schedule
2020-03-08 20:29:17 +00:00
* docker-compose-networks.yml: this yaml define the network to communicate with *Letsencrypt Nginx Proxy Companion* .
2020-03-05 16:39:32 +00:00
### Site operations
2017-07-18 09:26:31 +00:00
2020-03-02 10:59:25 +05:30
#### Setup New Sites
Note:
2020-03-08 20:29:17 +00:00
- Wait for the MariaDB service to start before trying to create a new site.
- If new site creation fails, retry after the MariaDB container is up and running.
2020-03-02 10:59:25 +05:30
- If you're using a managed database instance, make sure that the database is running before setting up a new site.
- Use `.env` file or environment variables instead of passing secrets as command arguments.
```sh
# Create ERPNext site
docker exec -it \
-e "SITE_NAME=$SITE_NAME" \
-e "DB_ROOT_USER=$DB_ROOT_USER" \
-e "MYSQL_ROOT_PASSWORD=$MYSQL_ROOT_PASSWORD" \
-e "ADMIN_PASSWORD=$ADMIN_PASSWORD" \
2020-03-09 13:49:35 +05:30
-e "INSTALL_APPS=erpnext" \
2020-03-02 10:59:25 +05:30
< project-name > _erpnext-python_1 docker-entrypoint.sh new
```
Environment Variables needed:
- `SITE_NAME` : name of the new site to create.
2020-03-09 13:49:35 +05:30
- `DB_ROOT_USER` : MariaDB Root user.
2020-03-08 20:29:17 +00:00
- `MYSQL_ROOT_PASSWORD` : In case of the MariaDB docker container use the one set in `MYSQL_ROOT_PASSWORD` in previous steps. In case of a managed database use the appropriate password.
- `ADMIN_PASSWORD` : set the administrator password for the new site.
2020-03-09 13:49:35 +05:30
- `INSTALL_APPS=erpnext` : available only in erpnext-worker and erpnext containers (or other containers with custom apps). Installs ERPNext (and/or the specified apps, comma-delinieated) on this new site.
2020-03-08 20:29:17 +00:00
- `FORCE=1` : optional variable which force installation of the same site.
2020-03-02 10:59:25 +05:30
#### Backup Sites
Environment Variables
2020-03-08 20:29:17 +00:00
- `SITES` is list of sites separated by `:` colon to migrate. e.g. `SITES=site1.domain.com` or `SITES=site1.domain.com:site2.domain.com` By default all sites in bench will be backed up.
- `WITH_FILES` if set to 1, it will backup user-uploaded files.
2020-03-02 10:59:25 +05:30
```sh
docker exec -it \
-e "SITES=site1.domain.com:site2.domain.com" \
-e "WITH_FILES=1" \
< project-name > _erpnext-python_1 docker-entrypoint.sh backup
```
2020-03-08 20:29:17 +00:00
The backup will be available in the `sites` mounted volume.
2020-03-02 10:59:25 +05:30
#### Updating and Migrating Sites
Switch to the root of the `frappe_docker` directory before running the following commands:
```sh
# Update environment variable VERSION
nano .env
# Pull new images
docker-compose \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
pull
# Restart containers
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
-f installation/docker-compose-networks.yml \
--project-directory installation up -d
docker exec -it \
-e "MAINTENANCE_MODE=1" \
< project-name > _erpnext-python_1 docker-entrypoint.sh migrate
```
2020-03-06 06:12:11 +00:00
### Custom apps
2020-03-08 20:29:17 +00:00
To add your own Frappe/ERPNext apps to the image, we'll need to create a custom image with the help of a unique wrapper script
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
> For the sake of simplicity, in this example, we'll be using a place holder called `[custom]`, and we'll be building off the edge image.
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
Create two directories called `[custom]-worker` and `[custom]-nginx` in the `build` directory.
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
```shell
cd frappe_docker
mkdir ./build/[custom]-worker ./build/[custom]-nginx
```
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
Create a `Dockerfile` in `./build/[custom]-worker` with the following content:
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
```Dockerfile
FROM frappe/erpnext-worker:edge
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
RUN install_app [custom] https://github.com/[username]/[custom] [branch]
# Only add the branch if you are using a specific tag or branch.
```
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
Create a `Dockerfile` in `./build/[custom]-nginx` with the following content:
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
```Dockerfile
FROM bitnami/node:12-prod
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
COPY build/[custom]-nginx/install_app.sh /install_app
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
RUN /install_app [custom] https://github.com/[username]/[custom]
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
FROM frappe/erpnext-nginx:edge
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
COPY --from=0 /home/frappe/frappe-bench/sites/ /var/www/html/
COPY --from=0 /rsync /rsync
RUN echo -n "\n[custom]" >> /var/www/html/apps.txt
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
VOLUME [ "/assets" ]
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
ENTRYPOINT ["/docker-entrypoint.sh"]
CMD ["nginx", "-g", "daemon off;"]
```
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
Copy over the `install_app.sh` file from `./build/erpnext-nginx`
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
```shell
cp ./build/erpnext-nginx/install.sh ./build/[custom]-nginx
```
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
Open up `./installation/docker-compose-custom.yml` and replace all instances of `[app]` with the name of your app.
2020-03-06 06:12:11 +00:00
2020-03-08 20:17:39 +00:00
```shell
sed -i "s#\[app\]#[custom]#" ./installation/docker-compose-custom.yml
```
2020-03-06 06:12:11 +00:00
2020-03-08 20:29:17 +00:00
Install like usual, except that when you set the `INSTALL_APPS` variable to `erpnext,[custom]` .
2020-03-06 06:12:11 +00:00
2020-03-05 16:39:32 +00:00
## Troubleshoot
2020-03-02 10:59:25 +05:30
2020-03-08 20:17:39 +00:00
### Failed migration after image upgrade
2020-03-02 10:59:25 +05:30
2020-03-08 20:17:39 +00:00
Issue: After upgrade of the containers, the automatic migration fails.
Solution: Remove containers and volumes, and clear redis cache:
2020-03-02 10:59:25 +05:30
2020-03-08 20:17:39 +00:00
```shell
2020-03-02 10:59:25 +05:30
# change to repo root
cd $HOME/frappe_docker
# Stop all bench containers
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
-f installation/docker-compose-networks.yml \
--project-directory installation stop
# Remove redis containers
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
-f installation/docker-compose-networks.yml \
--project-directory installation rm redis-cache redis-queue redis-socketio
# Clean redis volumes
docker volume rm \
< project-name > _redis-cache-vol \
< project-name > _redis-queue-vol \
< project-name > _redis-socketio-vol
# Restart project
docker-compose \
--project-name < project-name > \
-f installation/docker-compose-common.yml \
-f installation/docker-compose-erpnext.yml \
-f installation/docker-compose-networks.yml \
--project-directory installation up -d
```
2020-03-08 20:17:39 +00:00
### ValueError: There exists an active worker named XXX already
Issue: You have the following error during container restart
2020-03-02 10:59:25 +05:30
```
frappe-worker-short_1 | Traceback (most recent call last):
frappe-worker-short_1 | File "/home/frappe/frappe-bench/commands/worker.py", line 5, in < module >
frappe-worker-short_1 | start_worker(queue, False)
frappe-worker-short_1 | File "/home/frappe/frappe-bench/apps/frappe/frappe/utils/background_jobs.py", line 147, in start_worker
frappe-worker-short_1 | Worker(queues, name=get_worker_name(queue)).work(logging_level = logging_level)
frappe-worker-short_1 | File "/home/frappe/frappe-bench/env/lib/python3.7/site-packages/rq/worker.py", line 474, in work
frappe-worker-short_1 | self.register_birth()
frappe-worker-short_1 | File "/home/frappe/frappe-bench/env/lib/python3.7/site-packages/rq/worker.py", line 261, in register_birth
frappe-worker-short_1 | raise ValueError(msg.format(self.name))
frappe-worker-short_1 | ValueError: There exists an active worker named '8dfe5c234085.10.short' already
```
2020-03-08 20:17:39 +00:00
Solution: Clear redis cache using `docker exec` command (take care of replacing `<project-name>` accordingly):
2020-03-02 10:59:25 +05:30
```sh
# Clear the cache which is causing problem.
docker exec -it < project-name > _redis-cache_1 redis-cli FLUSHALL
docker exec -it < project-name > _redis-queue_1 redis-cli FLUSHALL
docker exec -it < project-name > _redis-socketio_1 redis-cli FLUSHALL
```
2020-03-08 20:29:17 +00:00
Note: Environment variables from `.env` file located at the current working directory will be used.
2020-03-08 20:17:39 +00:00
## Development
This repository includes a complete setup to develop with Frappe/ERPNext and Bench, Including the following features:
- VSCode containers integration
- VSCode Python debugger
- Pre-configured Docker containers for an easy start
A complete Readme is available in [development/README.md ](development/README.md )