├── tests
├── __init__.py
├── test_template.py
├── test_run.py
├── utils.py
└── test_config.py
├── helpers
├── __init__.py
├── utils.py
├── singleton.py
├── updater.py
├── aws_validation.py
├── cli.py
├── setup.py
├── network.py
├── upgrading.py
├── template.py
└── command.py
├── .github
├── FUNDING.yml
├── workflows
│ └── pytest.yml
└── ISSUE_TEMPLATE.md
├── .gitignore
├── tox.ini
├── requirements_tests.txt
├── templates
├── kobo-env
│ ├── envfiles
│ │ ├── external_services.txt.tpl
│ │ ├── smtp.txt.tpl
│ │ ├── domains.txt.tpl
│ │ ├── django.txt.tpl
│ │ ├── aws.txt.tpl
│ │ └── databases.txt.tpl
│ ├── postgres
│ │ └── conf
│ │ │ └── postgres.conf.tpl
│ └── enketo_express
│ │ └── config.json.tpl
├── kobo-docker
│ ├── docker-compose.maintenance.override.yml.tpl
│ ├── docker-compose.backend.override.yml.tpl
│ └── docker-compose.frontend.override.yml.tpl
└── nginx-certbot
│ ├── docker-compose.yml.tpl
│ ├── data
│ └── nginx
│ │ └── app.conf.tpl
│ └── init-letsencrypt.sh.tpl
├── setup.py
├── conftest.py
├── run.py
└── readme.md
/tests/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/helpers/__init__.py:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/.github/FUNDING.yml:
--------------------------------------------------------------------------------
1 | custom: ['https://kobotoolbox.org/donate']
2 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .idea
2 | .run.conf
3 | .run.conf.*
4 | *.pyc
5 | __pycache__
6 | .pytest_cache/
7 | .tox
8 | *.egg-info
9 |
--------------------------------------------------------------------------------
/helpers/utils.py:
--------------------------------------------------------------------------------
1 | from __future__ import annotations
2 |
3 |
4 | def run_docker_compose(config_dict: dict, command: list[str]) -> list[str]:
5 |
6 | return ['docker', 'compose'] + command
7 |
--------------------------------------------------------------------------------
/tox.ini:
--------------------------------------------------------------------------------
1 | # content of: tox.ini , put in same dir as setup.py
2 | [tox]
3 | skipsdist=True
4 | envlist = py38,py310,py312
5 |
6 | [testenv]
7 | deps = -rrequirements_tests.txt
8 | commands =
9 | pytest -vv {posargs} --disable-pytest-warnings
10 |
--------------------------------------------------------------------------------
/requirements_tests.txt:
--------------------------------------------------------------------------------
1 | atomicwrites==1.4.0
2 | attrs==21.4.0
3 | iniconfig==1.1.1
4 | more-itertools==8.12.0
5 | netifaces==0.11.0
6 | packaging==21.3
7 | pathlib2==2.3.2
8 | pluggy==1.0.0
9 | py==1.11.0
10 | pyparsing==3.0.8
11 | pytest==7.1.1
12 | six==1.16.0
13 | tomli==2.0.1
14 |
--------------------------------------------------------------------------------
/helpers/singleton.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | class Singleton(type):
3 | _instances = {}
4 |
5 | def __call__(cls, *args, **kwargs):
6 | if cls not in cls._instances:
7 | cls._instances[cls] = super(Singleton, cls).__call__(*args, **kwargs)
8 |
9 | return cls._instances[cls]
10 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/external_services.txt.tpl:
--------------------------------------------------------------------------------
1 | ############################################################################
2 | # GOOGLE_ANALYTICS_TOKEN must be changed in enketo_express/config.json too #
3 | ############################################################################
4 | GOOGLE_ANALYTICS_TOKEN=${GOOGLE_UA}
5 |
6 | SENTRY_DSN=${KPI_RAVEN_DSN}
7 | SENTRY_JS_DSN=${KPI_RAVEN_JS_DSN}
8 |
--------------------------------------------------------------------------------
/setup.py:
--------------------------------------------------------------------------------
1 | from distutils.core import setup
2 | from setuptools import find_packages
3 | from helpers.config import Config
4 |
5 | setup(
6 | name='kobo-install',
7 | version=Config.KOBO_INSTALL_VERSION,
8 | # Include all the python modules except `tests`,
9 | packages=find_packages(exclude=['tests']),
10 | url='https://github.com/kobotoolbox/kobo-install/',
11 | license='',
12 | author='KoboToolbox',
13 | author_email='',
14 | description='Installer for KoboToolbox'
15 | )
16 |
--------------------------------------------------------------------------------
/templates/kobo-env/postgres/conf/postgres.conf.tpl:
--------------------------------------------------------------------------------
1 | #------------------------------------------------------------------------------------
2 | # TUNING
3 | #------------------------------------------------------------------------------------
4 | # These settings are based on server configuration
5 | # https://www.pgconfig.org/#/tuning
6 | # DB Version: 14
7 | # OS Type: linux
8 | # App profile: ${POSTGRES_APP_PROFILE}
9 | # Hard-drive: SSD
10 | # Total Memory (RAM): ${POSTGRES_RAM}GB
11 |
12 | ${POSTGRES_SETTINGS}
13 |
--------------------------------------------------------------------------------
/templates/kobo-docker/docker-compose.maintenance.override.yml.tpl:
--------------------------------------------------------------------------------
1 | # For public, HTTPS servers.
2 |
3 | services:
4 |
5 | maintenance:
6 | environment:
7 | - ETA=${MAINTENANCE_ETA}
8 | - DATE_STR=${MAINTENANCE_DATE_STR}
9 | - DATE_ISO=${MAINTENANCE_DATE_ISO}
10 | - EMAIL=${MAINTENANCE_EMAIL}
11 | ${USE_LETSENSCRYPT}ports:
12 | ${USE_LETSENSCRYPT} - ${NGINX_EXPOSED_PORT}:80
13 | networks:
14 | kobo-fe-network:
15 | aliases:
16 | - nginx
17 | - nginx.internal
18 |
19 | networks:
20 | kobo-fe-network:
21 | name: ${DOCKER_NETWORK_FRONTEND_PREFIX}_kobo-fe-network
22 |
--------------------------------------------------------------------------------
/conftest.py:
--------------------------------------------------------------------------------
1 | # coding: utf-8
2 | import os
3 | import pytest
4 |
5 |
6 | def clean_up():
7 | """
8 | Removes files created by tests
9 | """
10 | files = ['.uniqid',
11 | 'upsert_db_users']
12 | for file_ in files:
13 | try:
14 | os.remove(os.path.join('/tmp', file_))
15 | except FileNotFoundError:
16 | pass
17 |
18 |
19 | @pytest.fixture(scope="session", autouse=True)
20 | def setup(request):
21 | # Clean up before tests begin in case of orphan files.
22 | clean_up()
23 | request.addfinalizer(_tear_down)
24 |
25 |
26 | def _tear_down():
27 | clean_up()
28 | pass
29 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/smtp.txt.tpl:
--------------------------------------------------------------------------------
1 | ##################################
2 | # For sending e-mail using SMTP. #
3 | ##################################
4 |
5 | # NOTE: To send from GMail, the sending account must enable "Allowing less secure apps to access your account" (https://support.google.com/accounts/answer/6010255).
6 | # NOTE: To send from AWS EC2 instances, SNS must be used instead of SMTP. These and the SNS e-mail settings from `envfiles/aws.txt` are mutually exclusive; do not use both.
7 | # See https://docs.djangoproject.com/en/1.8/topics/email/#smtp-backend.
8 |
9 | EMAIL_BACKEND=django.core.mail.backends.smtp.EmailBackend
10 | EMAIL_HOST=${SMTP_HOST}
11 | EMAIL_PORT=${SMTP_PORT}
12 | EMAIL_HOST_USER=${SMTP_USER}
13 | EMAIL_HOST_PASSWORD=${SMTP_PASSWORD}
14 | EMAIL_USE_TLS=${SMTP_USE_TLS}
15 | DEFAULT_FROM_EMAIL=${DEFAULT_FROM_EMAIL}
--------------------------------------------------------------------------------
/.github/workflows/pytest.yml:
--------------------------------------------------------------------------------
1 | # For more information see: https://help.github.com/actions/language-and-framework-guides/using-python-with-github-actions
2 |
3 | name: pytest
4 |
5 | on:
6 | push:
7 | branches: [ master ]
8 | pull_request:
9 | branches: [ master ]
10 |
11 | jobs:
12 | build:
13 |
14 | runs-on: ubuntu-24.04
15 | strategy:
16 | matrix:
17 | python-version: ['3.10']
18 |
19 | steps:
20 | - uses: actions/checkout@v2
21 | - name: Set up Python ${{ matrix.python-version }}
22 | uses: actions/setup-python@v2
23 | with:
24 | python-version: ${{ matrix.python-version }}
25 | - name: Upgrade pip
26 | run: python -m pip install --upgrade pip
27 | - name: Install Python dependencies
28 | run: pip install -r requirements_tests.txt
29 | - name: Run pytest
30 | run: pytest -vv -rf
31 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/domains.txt.tpl:
--------------------------------------------------------------------------------
1 | # Choose between http or https
2 | PUBLIC_REQUEST_SCHEME=${PUBLIC_REQUEST_SCHEME}
3 | # The publicly-accessible domain where your KoBo Toolbox instance will be reached (e.g. example.com).
4 | PUBLIC_DOMAIN_NAME=${PUBLIC_DOMAIN_NAME}
5 | # The private domain used in docker network. Useful for communication between containers without passing through
6 | # a load balancer. No need to be resolved by a public DNS.
7 | INTERNAL_DOMAIN_NAME=${INTERNAL_DOMAIN_NAME}
8 | # The publicly-accessible subdomain for the KoBoForm form building and management interface (e.g. koboform).
9 | KOBOFORM_PUBLIC_SUBDOMAIN=${KOBOFORM_SUBDOMAIN}
10 | # The publicly-accessible subdomain for the KoBoCAT data collection and project management interface (e.g.kobocat).
11 | KOBOCAT_PUBLIC_SUBDOMAIN=${KOBOCAT_SUBDOMAIN}
12 | # The publicly-accessible subdomain for the Enketo Express web forms (e.g. enketo).
13 | ENKETO_EXPRESS_PUBLIC_SUBDOMAIN=${ENKETO_SUBDOMAIN}
14 |
--------------------------------------------------------------------------------
/templates/nginx-certbot/docker-compose.yml.tpl:
--------------------------------------------------------------------------------
1 | services:
2 | nginx_ssl_proxy:
3 | image: nginx:1.26-alpine
4 | restart: unless-stopped
5 | volumes:
6 | - ./data/nginx:/etc/nginx/conf.d
7 | - ./data/certbot/conf:/etc/letsencrypt
8 | - ./data/certbot/www:/var/www/certbot
9 | ports:
10 | - "80:80"
11 | - "443:443"
12 | command: "/bin/sh -c 'while :; do sleep 6h & wait $$$${!}; nginx -s reload; done & nginx -g \"daemon off;\"'"
13 | networks:
14 | kobo-fe-network:
15 | aliases:
16 | - nginx_ssl_proxy
17 | certbot:
18 | image: certbot/certbot
19 | restart: unless-stopped
20 | volumes:
21 | - ./data/certbot/conf:/etc/letsencrypt
22 | - ./data/certbot/www:/var/www/certbot
23 | entrypoint: "/bin/sh -c 'trap exit TERM; while :; do certbot renew; sleep 12h & wait $$$${!}; done;'"
24 |
25 | networks:
26 | kobo-fe-network:
27 | name: ${DOCKER_NETWORK_FRONTEND_PREFIX}_kobo-fe-network
28 | external: true
29 |
--------------------------------------------------------------------------------
/.github/ISSUE_TEMPLATE.md:
--------------------------------------------------------------------------------
1 |
4 |
5 |
6 | **Description**
7 |
8 |
9 | **Steps to Reproduce**
10 |
11 |
12 | **Expected behavior**
13 |
14 |
15 | **Desktop**
16 |
17 |
18 | - OS:
19 | - Python Version:
20 | - Docker Version:
21 | - Docker Compose Version:
22 |
23 | **Additional context**
24 |
25 |
--------------------------------------------------------------------------------
/templates/nginx-certbot/data/nginx/app.conf.tpl:
--------------------------------------------------------------------------------
1 | server {
2 | listen 80;
3 | server_name ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME};
4 | server_tokens off;
5 |
6 | location /.well-known/acme-challenge/ {
7 | root /var/www/certbot;
8 | }
9 |
10 | location / {
11 | return 301 https://$$host$$request_uri;
12 | }
13 | }
14 |
15 | server {
16 | listen 443 ssl;
17 | server_name ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME};
18 | server_tokens off;
19 |
20 | ssl_certificate /etc/letsencrypt/live/${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}/fullchain.pem;
21 | ssl_certificate_key /etc/letsencrypt/live/${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}/privkey.pem;
22 | include /etc/letsencrypt/options-ssl-nginx.conf;
23 | ssl_dhparam /etc/letsencrypt/ssl-dhparams.pem;
24 |
25 | # Allow 100M upload
26 | client_max_body_size 100M;
27 |
28 | location / {
29 | proxy_pass http://nginx;
30 | proxy_set_header Host $$http_host;
31 | proxy_set_header X-Real-IP $$remote_addr;
32 | proxy_set_header X-Forwarded-For $$proxy_add_x_forwarded_for;
33 | proxy_set_header X-Forwarded-Proto https;
34 | }
35 | }
36 |
--------------------------------------------------------------------------------
/templates/kobo-docker/docker-compose.backend.override.yml.tpl:
--------------------------------------------------------------------------------
1 | # Override for primary back-end server
2 |
3 | services:
4 |
5 | postgres:
6 | volumes:
7 | - ../kobo-env/postgres/conf/postgres.conf:/kobo-docker-scripts/conf/postgres.conf
8 | ${EXPOSE_BACKEND_PORTS}ports:
9 | ${EXPOSE_BACKEND_PORTS} - ${POSTGRES_PORT}:5432
10 | ${USE_BACKEND_NETWORK}networks:
11 | ${USE_BACKEND_NETWORK} kobo-be-network:
12 | ${USE_BACKEND_NETWORK} aliases:
13 | ${USE_BACKEND_NETWORK} - postgres.${PRIVATE_DOMAIN_NAME}
14 |
15 | mongo:
16 | ${EXPOSE_BACKEND_PORTS}ports:
17 | ${EXPOSE_BACKEND_PORTS} - ${MONGO_PORT}:27017
18 | ${USE_BACKEND_NETWORK}networks:
19 | ${USE_BACKEND_NETWORK} kobo-be-network:
20 | ${USE_BACKEND_NETWORK} aliases:
21 | ${USE_BACKEND_NETWORK} - mongo.${PRIVATE_DOMAIN_NAME}
22 |
23 | redis_main:
24 | ${EXPOSE_BACKEND_PORTS}ports:
25 | ${EXPOSE_BACKEND_PORTS} - ${REDIS_MAIN_PORT}:6379
26 | ${USE_BACKEND_NETWORK}networks:
27 | ${USE_BACKEND_NETWORK} kobo-be-network:
28 | ${USE_BACKEND_NETWORK} aliases:
29 | ${USE_BACKEND_NETWORK} - redis-main.${PRIVATE_DOMAIN_NAME}
30 |
31 | redis_cache:
32 | ${EXPOSE_BACKEND_PORTS}ports:
33 | ${EXPOSE_BACKEND_PORTS} - ${REDIS_CACHE_PORT}:6380
34 | ${USE_BACKEND_NETWORK}networks:
35 | ${USE_BACKEND_NETWORK} kobo-be-network:
36 | ${USE_BACKEND_NETWORK} aliases:
37 | ${USE_BACKEND_NETWORK} - redis-cache.${PRIVATE_DOMAIN_NAME}
38 |
39 | ${USE_BACKEND_NETWORK}networks:
40 | ${USE_BACKEND_NETWORK} kobo-be-network:
41 | ${USE_BACKEND_NETWORK} driver: bridge
42 |
--------------------------------------------------------------------------------
/helpers/updater.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import os
3 | import sys
4 |
5 | from helpers.cli import CLI
6 | from helpers.setup import Setup
7 |
8 |
9 | class Updater:
10 | """
11 | Updates kobo-install (this utility), restarts this script, and updates
12 | kobo-docker
13 | """
14 | NO_UPDATE_SELF_OPTION = '--no-update-self'
15 |
16 | @classmethod
17 | def run(cls, version=None, cron=False, update_self=True):
18 | # Validate kobo-docker already exists and is valid
19 | Setup.validate_already_run()
20 |
21 | if version is None:
22 | git_commit_version_command = [
23 | 'git',
24 | 'rev-parse',
25 | '--abbrev-ref',
26 | 'HEAD',
27 | ]
28 | version = CLI.run_command(git_commit_version_command).strip()
29 |
30 | if update_self:
31 | # Update kobo-install first
32 | Setup.update_koboinstall(version)
33 | CLI.colored_print('kobo-install has been updated',
34 | CLI.COLOR_SUCCESS)
35 |
36 | # Reload this script to use `version`.
37 | # NB:`argv[0]` does not automatically get set to the executable
38 | # path as it usually would, so we have to do it manually--hence the
39 | # double `sys.executable`
40 | sys.argv.append(cls.NO_UPDATE_SELF_OPTION)
41 | os.execl(sys.executable, sys.executable, *sys.argv)
42 |
43 | # Update kobo-docker
44 | Setup.update_kobodocker()
45 | CLI.colored_print('kobo-docker has been updated', CLI.COLOR_SUCCESS)
46 | Setup.post_update(cron)
47 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/django.txt.tpl:
--------------------------------------------------------------------------------
1 | DJANGO_DEBUG=${DEBUG}
2 | TEMPLATE_DEBUG=${DEBUG}
3 | ${USE_X_FORWARDED_HOST}USE_X_FORWARDED_HOST=True
4 |
5 | DJANGO_SECRET_KEY=${DJANGO_SECRET_KEY}
6 | DJANGO_SESSION_COOKIE_AGE=${DJANGO_SESSION_COOKIE_AGE}
7 | DJANGO_ALLOWED_HOSTS=.${PUBLIC_DOMAIN_NAME} .${INTERNAL_DOMAIN_NAME}
8 | KPI_PREFIX=/
9 |
10 | SESSION_COOKIE_DOMAIN=".${PUBLIC_DOMAIN_NAME}"
11 |
12 | CELERY_BROKER_URL=redis://{% if REDIS_PASSWORD %}:${REDIS_PASSWORD}@{% endif REDIS_PASSWORD %}redis-main.${PRIVATE_DOMAIN_NAME}:${REDIS_MAIN_PORT}/1
13 | CELERY_AUTOSCALE_MIN=2
14 | CELERY_AUTOSCALE_MAX=6
15 |
16 | # See "api key" here: https://github.com/kobotoolbox/enketo-express/tree/master/config#linked-form-and-data-server.
17 | ENKETO_API_KEY=${ENKETO_API_KEY}
18 |
19 | # The initial superuser's username.
20 | KOBO_SUPERUSER_USERNAME=${KOBO_SUPERUSER_USERNAME}
21 | # The initial superuser's password.
22 | KOBO_SUPERUSER_PASSWORD=${KOBO_SUPERUSER_PASSWORD}
23 | # The e-mail address where your users can contact you.
24 | KOBO_SUPPORT_EMAIL=${DEFAULT_FROM_EMAIL}
25 |
26 | KOBOFORM_URL=${PUBLIC_REQUEST_SCHEME}://${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}${NGINX_PUBLIC_PORT}
27 | KOBOFORM_INTERNAL_URL=http://${KOBOFORM_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME} # Always use HTTP internally.
28 | ENKETO_URL=${PUBLIC_REQUEST_SCHEME}://${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}${NGINX_PUBLIC_PORT}
29 | ENKETO_INTERNAL_URL=http://${ENKETO_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME} # Always use HTTP internally.
30 | KOBOCAT_URL=${PUBLIC_REQUEST_SCHEME}://${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}${NGINX_PUBLIC_PORT}
31 | KOBOCAT_INTERNAL_URL=http://${KOBOCAT_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME} # Always use HTTP internally.
32 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/aws.txt.tpl:
--------------------------------------------------------------------------------
1 | ####################
2 | # Account settings #
3 | ####################
4 |
5 | ${USE_AWS}AWS_ACCESS_KEY_ID=${AWS_ACCESS_KEY_ID}
6 | ${USE_AWS}AWS_SECRET_ACCESS_KEY=${AWS_SECRET_ACCESS_KEY}
7 |
8 | ####################
9 | # Storage settings #
10 | ####################
11 |
12 | # To use S3, the specified buckets must already exist and the owner of your `AWS_ACCESS_KEY_ID` must have the appropriate S3 permissions.
13 |
14 | ${USE_AWS}KOBOCAT_DEFAULT_FILE_STORAGE=storages.backends.s3boto3.S3Boto3Storage
15 | ${USE_AWS}KOBOCAT_AWS_STORAGE_BUCKET_NAME=${AWS_BUCKET_NAME}
16 |
17 | ${USE_AWS}KPI_DEFAULT_FILE_STORAGE=storages.backends.s3boto3.S3Boto3Storage
18 | ${USE_AWS}KPI_AWS_STORAGE_BUCKET_NAME=${AWS_BUCKET_NAME}
19 |
20 | ${USE_AWS}AWS_S3_REGION_NAME=${AWS_S3_REGION_NAME}
21 |
22 | ###################
23 | # Backup settings #
24 | ###################
25 |
26 | ${USE_AWS_BACKUP}BACKUP_AWS_STORAGE_BUCKET_NAME=${AWS_BACKUP_BUCKET_NAME}
27 | #Backups files deletion is handled by bucket rules when True
28 | ${USE_AWS_BACKUP}AWS_BACKUP_BUCKET_DELETION_RULE_ENABLED=${AWS_BACKUP_BUCKET_DELETION_RULE_ENABLED}
29 | ${USE_AWS_BACKUP}AWS_BACKUP_YEARLY_RETENTION=${AWS_BACKUP_YEARLY_RETENTION}
30 | ${USE_AWS_BACKUP}AWS_BACKUP_MONTHLY_RETENTION=${AWS_BACKUP_MONTHLY_RETENTION}
31 | ${USE_AWS_BACKUP}AWS_BACKUP_WEEKLY_RETENTION=${AWS_BACKUP_WEEKLY_RETENTION}
32 | ${USE_AWS_BACKUP}AWS_BACKUP_DAILY_RETENTION=${AWS_BACKUP_DAILY_RETENTION}
33 |
34 | # In MB
35 | ${USE_AWS_BACKUP}AWS_MONGO_BACKUP_MINIMUM_SIZE=${AWS_MONGO_BACKUP_MINIMUM_SIZE}
36 | ${USE_AWS_BACKUP}AWS_POSTGRES_BACKUP_MINIMUM_SIZE=${AWS_POSTGRES_BACKUP_MINIMUM_SIZE}
37 | ${USE_AWS_BACKUP}AWS_REDIS_BACKUP_MINIMUM_SIZE=${AWS_REDIS_BACKUP_MINIMUM_SIZE}
38 | ${USE_AWS_BACKUP}AWS_BACKUP_UPLOAD_CHUNK_SIZE=${AWS_BACKUP_UPLOAD_CHUNK_SIZE}
39 |
40 |
--------------------------------------------------------------------------------
/tests/test_template.py:
--------------------------------------------------------------------------------
1 | import os
2 | import shutil
3 | from unittest.mock import patch, MagicMock
4 |
5 | from helpers.template import Template
6 | from .utils import mock_read_config as read_config
7 |
8 |
9 | WORK_DIR = '/tmp/kobo-install-tests'
10 |
11 | @patch(
12 | 'helpers.template.Template._Template__read_unique_id',
13 | MagicMock(return_value='123456789')
14 | )
15 | @patch(
16 | 'helpers.template.Template._Template__write_unique_id',
17 | MagicMock(return_value='123456789')
18 | )
19 | @patch(
20 | 'helpers.template.Template._get_templates_path_parent',
21 | MagicMock(return_value=f'{WORK_DIR}/templates/')
22 | )
23 | @patch(
24 | 'helpers.config.Config.get_env_files_path',
25 | MagicMock(return_value=f'{WORK_DIR}/kobo-env/')
26 | )
27 | @patch(
28 | 'helpers.config.Config.get_letsencrypt_repo_path',
29 | MagicMock(return_value=f'{WORK_DIR}/nginx-certbot/')
30 | )
31 | def test_render_templates():
32 | config = read_config()
33 | config._Config__dict['unique_id'] = '123456789'
34 | config._Config__dict['kobodocker_path'] = f'{WORK_DIR}/kobo-docker/'
35 | try:
36 | _copy_templates()
37 | assert not os.path.exists(
38 | f'{WORK_DIR}/kobo-docker/docker-compose.frontend.override.yml'
39 | )
40 | assert not os.path.exists(
41 | f'{WORK_DIR}/kobo-docker/docker-compose.backend.override.yml'
42 | )
43 | assert not os.path.exists(f'{WORK_DIR}/kobo-env/envfiles/django.txt')
44 | Template.render(config)
45 | assert os.path.exists(
46 | f'{WORK_DIR}/kobo-docker/docker-compose.frontend.override.yml'
47 | )
48 | assert os.path.exists(
49 | f'{WORK_DIR}/kobo-docker/docker-compose.backend.override.yml'
50 | )
51 | assert os.path.exists(f'{WORK_DIR}/kobo-env/envfiles/django.txt')
52 | finally:
53 | shutil.rmtree(WORK_DIR)
54 |
55 |
56 | def _copy_templates(src: str = None, dst: str = None):
57 | if not src:
58 | src = os.path.dirname(os.path.realpath(__file__)) + '/../templates/'
59 | if not dst:
60 | dst = f'{WORK_DIR}/templates/'
61 |
62 | # Create the destination directory if needed
63 | os.makedirs(dst, exist_ok=True)
64 |
65 | for entry in os.listdir(src):
66 | src_path = os.path.join(src, entry)
67 | dst_path = os.path.join(dst, entry)
68 |
69 | if os.path.isdir(src_path):
70 | # Recursively copy subdirectories
71 | _copy_templates(src_path, dst_path)
72 | else:
73 | # Copy files (overwrite if exists)
74 | shutil.copy2(src_path, dst_path)
75 |
--------------------------------------------------------------------------------
/templates/nginx-certbot/init-letsencrypt.sh.tpl:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | function join_by { local d=$$1; shift; echo -n "$$1"; shift; printf "%s" "$${@/#/$$d}"; }
4 |
5 | DOMAINS=(${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME} ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME})
6 | DOMAINS_CSV=$$(join_by , "$${DOMAINS[@]}")
7 | RSA_KEY_SIZE=4096
8 | DATA_PATH="./data/certbot"
9 | EMAIL="" # Adding a valid address is strongly recommended
10 | STAGING=0 # Set to 1 if you're testing your setup to avoid hitting request limits
11 | MKDIR_CMD=$$(which mkdir)
12 | DOCKER_COMPOSE_CMD="$$(which ${DOCKER_COMPOSE_CMD})"
13 | CURL_CMD=$$(which curl)
14 |
15 |
16 | if [ -d "$$DATA_PATH/conf/live/$$DOMAINS" ]; then
17 | read -p "Existing data found for $$DOMAINS_CSV. Continue and replace existing certificate? (y/N) " decision
18 | if [ "$$decision" != "Y" ] && [ "$$decision" != "y" ]; then
19 | exit
20 | fi
21 | fi
22 |
23 | if [ ! -e "$$DATA_PATH/conf/options-ssl-nginx.conf" ] || [ ! -e "$$DATA_PATH/conf/ssl-dhparams.pem" ]; then
24 | echo "### Downloading recommended TLS parameters ..."
25 | $$MKDIR_CMD -p "$$DATA_PATH/conf"
26 | $$CURL_CMD -s https://raw.githubusercontent.com/kobotoolbox/nginx-certbot/master/certbot/options-ssl-nginx.conf > "$$DATA_PATH/conf/options-ssl-nginx.conf"
27 | $$CURL_CMD -s https://raw.githubusercontent.com/kobotoolbox/nginx-certbot/master/certbot/ssl-dhparams.pem > "$$DATA_PATH/conf/ssl-dhparams.pem"
28 | echo
29 | fi
30 |
31 | echo "### Creating dummy certificate for $${DOMAINS_CSV} ..."
32 | DOMAINS_PATH="/etc/letsencrypt/live/$$DOMAINS"
33 | $$MKDIR_CMD -p "$$DATA_PATH/conf/live/$$DOMAINS"
34 | $$DOCKER_COMPOSE_CMD ${DOCKER_COMPOSE_SUFFIX} run --rm --entrypoint "\
35 | openssl req -x509 -nodes -newkey rsa:2048 -days 1\
36 | -keyout '$$DOMAINS_PATH/privkey.pem' \
37 | -out '$$DOMAINS_PATH/fullchain.pem' \
38 | -subj '/CN=localhost'" certbot
39 | echo
40 |
41 |
42 | echo "### Starting nginx ..."
43 | $$DOCKER_COMPOSE_CMD ${DOCKER_COMPOSE_SUFFIX} up --force-recreate -d nginx_ssl_proxy
44 | echo
45 |
46 | echo "### Deleting dummy certificate for $${DOMAINS_CSV} ..."
47 | $$DOCKER_COMPOSE_CMD ${DOCKER_COMPOSE_SUFFIX} run --rm --entrypoint "\
48 | rm -Rf /etc/letsencrypt/live/$$DOMAINS && \
49 | rm -Rf /etc/letsencrypt/archive/$$DOMAINS && \
50 | rm -Rf /etc/letsencrypt/renewal/$$DOMAINS.conf" certbot
51 | echo
52 |
53 |
54 | echo "### Requesting Let's Encrypt certificate for $${DOMAINS_CSV} ..."
55 | #Join $$DOMAINS to -d args
56 | DOMAIN_ARGS=""
57 | for DOMAIN in "$${DOMAINS[@]}"; do
58 | DOMAIN_ARGS="$$DOMAIN_ARGS -d $$DOMAIN"
59 | done
60 |
61 | # Select appropriate EMAIL arg
62 | case "$$EMAIL" in
63 | "") EMAIL_ARG="--register-unsafely-without-email" ;;
64 | *) EMAIL_ARG="--email $$EMAIL" ;;
65 | esac
66 |
67 | # Enable staging mode if needed
68 | if [ $$STAGING != "0" ]; then STAGING_ARG="--staging"; fi
69 |
70 | $$DOCKER_COMPOSE_CMD ${DOCKER_COMPOSE_SUFFIX} run --rm --entrypoint "\
71 | certbot certonly --webroot -w /var/www/certbot \
72 | $$STAGING_ARG \
73 | $$EMAIL_ARG \
74 | $$DOMAIN_ARGS \
75 | --rsa-key-size $$RSA_KEY_SIZE \
76 | --agree-tos \
77 | --force-renewal" certbot
78 | echo
79 |
80 | echo "### Reloading nginx ..."
81 | $$DOCKER_COMPOSE_CMD ${DOCKER_COMPOSE_SUFFIX} exec nginx_ssl_proxy nginx -s reload
82 |
--------------------------------------------------------------------------------
/templates/kobo-env/envfiles/databases.txt.tpl:
--------------------------------------------------------------------------------
1 | #--------------------------------------------------------------------------------
2 | # MONGO
3 | #--------------------------------------------------------------------------------
4 | # These `KOBO_MONGO_` settings only affect the mongo container itself and the
5 | # `wait_for_mongo.bash` init script that runs within the kpi and kobocat.
6 | # Please see kobocat.txt to set container variables
7 | KOBO_MONGO_PORT=${MONGO_PORT}
8 | KOBO_MONGO_HOST=mongo.${PRIVATE_DOMAIN_NAME}
9 | MONGO_INITDB_ROOT_USERNAME=${MONGO_ROOT_USERNAME}
10 | MONGO_INITDB_ROOT_PASSWORD=${MONGO_ROOT_PASSWORD}
11 | MONGO_INITDB_DATABASE=formhub
12 | KOBO_MONGO_USERNAME=${MONGO_USER_USERNAME}
13 | KOBO_MONGO_PASSWORD=${MONGO_USER_PASSWORD}
14 | MONGO_DB_NAME=formhub
15 | MONGO_DB_URL=mongodb://${MONGO_USER_USERNAME}:${MONGO_USER_PASSWORD}@mongo.${PRIVATE_DOMAIN_NAME}:${MONGO_PORT}/formhub
16 |
17 | # Default MongoDB backup schedule is weekly at 01:00 AM UTC on Sunday.
18 | ${USE_BACKUP}MONGO_BACKUP_SCHEDULE=${MONGO_BACKUP_SCHEDULE}
19 |
20 | #--------------------------------------------------------------------------------
21 | # POSTGRES
22 | #--------------------------------------------------------------------------------
23 |
24 | # These `KOBO_POSTGRES_` settings only affect the postgres container itself and the
25 | # `wait_for_postgres.bash` init script that runs within the kpi and kobocat
26 | # containers. To control Django database connections, please see the
27 | # `DATABASE_URL` environment variable.
28 | POSTGRES_PORT=${POSTGRES_PORT}
29 | POSTGRES_HOST=postgres.${PRIVATE_DOMAIN_NAME}
30 | POSTGRES_USER=${POSTGRES_USER}
31 | POSTGRES_PASSWORD=${POSTGRES_PASSWORD}
32 | KC_POSTGRES_DB=${KC_POSTGRES_DB}
33 | KPI_POSTGRES_DB=${KPI_POSTGRES_DB}
34 |
35 | # Postgres database used by kpi and kobocat Django apps
36 | KC_DATABASE_URL=postgis://${POSTGRES_USER}:${POSTGRES_PASSWORD}@postgres.${PRIVATE_DOMAIN_NAME}:${POSTGRES_PORT}/${KC_POSTGRES_DB}
37 | KPI_DATABASE_URL=postgis://${POSTGRES_USER}:${POSTGRES_PASSWORD}@postgres.${PRIVATE_DOMAIN_NAME}:${POSTGRES_PORT}/${KPI_POSTGRES_DB}
38 | DATABASE_URL=postgis://${POSTGRES_USER}:${POSTGRES_PASSWORD}@postgres.${PRIVATE_DOMAIN_NAME}:${POSTGRES_PORT}/${KPI_POSTGRES_DB}
39 |
40 | # Default Postgres backup schedule is weekly at 02:00 AM UTC on Sunday.
41 | ${USE_BACKUP}POSTGRES_BACKUP_SCHEDULE=${POSTGRES_BACKUP_SCHEDULE}
42 |
43 | #--------------------------------------------------------------------------------
44 | # REDIS
45 | #--------------------------------------------------------------------------------
46 |
47 | # Default Redis backup schedule is weekly at 03:00 AM UTC on Sunday.
48 | ${USE_BACKUP}REDIS_BACKUP_SCHEDULE=${REDIS_BACKUP_SCHEDULE}
49 |
50 | REDIS_SESSION_URL=redis://{% if REDIS_PASSWORD %}:${REDIS_PASSWORD}@{% endif REDIS_PASSWORD %}redis-cache.${PRIVATE_DOMAIN_NAME}:${REDIS_CACHE_PORT}/2
51 | REDIS_PASSWORD=${REDIS_PASSWORD}
52 | CACHE_URL=redis://{% if REDIS_PASSWORD %}:${REDIS_PASSWORD}@{% endif REDIS_PASSWORD %}redis-cache.${PRIVATE_DOMAIN_NAME}:${REDIS_CACHE_PORT}/5
53 | REDIS_CACHE_MAX_MEMORY=${REDIS_CACHE_MAX_MEMORY}
54 | SERVICE_ACCOUNT_BACKEND_URL=redis://{% if REDIS_PASSWORD %}:${REDIS_PASSWORD}@{% endif REDIS_PASSWORD %}redis-cache.${PRIVATE_DOMAIN_NAME}:${REDIS_CACHE_PORT}/6
55 | ENKETO_REDIS_MAIN_URL=redis://{% if REDIS_PASSWORD %}:${REDIS_PASSWORD}@{% endif REDIS_PASSWORD %}redis-main.${PRIVATE_DOMAIN_NAME}:${REDIS_MAIN_PORT}/0
56 |
--------------------------------------------------------------------------------
/helpers/aws_validation.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import datetime
3 | import hashlib
4 | import hmac
5 | from urllib.error import HTTPError
6 | from urllib.request import Request, urlopen
7 |
8 |
9 | class AWSValidation:
10 | """
11 | A class to validate AWS credentials without using boto3 as a dependency.
12 |
13 | The structure and methods have been adapted from the AWS documentation:
14 | http://docs.aws.amazon.com/general/latest/gr/signature-v4-examples.html#signature-v4-examples-python
15 | """
16 |
17 | METHOD = 'POST'
18 | SERVICE = 'sts'
19 | REGION = 'us-east-1'
20 | HOST = 'sts.amazonaws.com'
21 | ENDPOINT = 'https://sts.amazonaws.com'
22 | REQUEST_PARAMETERS = 'Action=GetCallerIdentity&Version=2011-06-15'
23 | CANONICAL_URI = '/'
24 | SIGNED_HEADERS = 'host;x-amz-date'
25 | PAYLOAD_HASH = hashlib.sha256(''.encode()).hexdigest()
26 | ALGORITHM = 'AWS4-HMAC-SHA256'
27 |
28 | def __init__(self, aws_access_key_id, aws_secret_access_key):
29 | self.access_key = aws_access_key_id
30 | self.secret_key = aws_secret_access_key
31 |
32 | @staticmethod
33 | def _sign(key, msg):
34 | return hmac.new(key, msg.encode(), hashlib.sha256).digest()
35 |
36 | @classmethod
37 | def _get_signature_key(cls, key, date_stamp, region_name, service_name):
38 | k_date = cls._sign(('AWS4' + key).encode(), date_stamp)
39 | k_region = cls._sign(k_date, region_name)
40 | k_service = cls._sign(k_region, service_name)
41 | return cls._sign(k_service, 'aws4_request')
42 |
43 | def _get_request_url_and_headers(self):
44 | t = datetime.datetime.utcnow()
45 | amzdate = t.strftime('%Y%m%dT%H%M%SZ')
46 | datestamp = t.strftime('%Y%m%d')
47 |
48 | canonical_querystring = self.REQUEST_PARAMETERS
49 |
50 | canonical_headers = '\n'.join(
51 | [
52 | 'host:{host}'.format(host=self.HOST),
53 | 'x-amz-date:{amzdate}'.format(amzdate=amzdate),
54 | '',
55 | ]
56 | )
57 |
58 | canonical_request = '\n'.join(
59 | [
60 | self.METHOD,
61 | self.CANONICAL_URI,
62 | canonical_querystring,
63 | canonical_headers,
64 | self.SIGNED_HEADERS,
65 | self.PAYLOAD_HASH,
66 | ]
67 | )
68 |
69 | credential_scope = '/'.join(
70 | [datestamp, self.REGION, self.SERVICE, 'aws4_request']
71 | )
72 |
73 | string_to_sign = '\n'.join(
74 | [
75 | self.ALGORITHM,
76 | amzdate,
77 | credential_scope,
78 | hashlib.sha256(canonical_request.encode()).hexdigest(),
79 | ]
80 | )
81 |
82 | signing_key = self._get_signature_key(
83 | self.secret_key, datestamp, self.REGION, self.SERVICE
84 | )
85 |
86 | signature = hmac.new(
87 | signing_key, string_to_sign.encode(), hashlib.sha256
88 | ).hexdigest()
89 |
90 | authorization_header = (
91 | '{} Credential={}/{}, SignedHeaders={}, Signature={}'.format(
92 | self.ALGORITHM,
93 | self.access_key,
94 | credential_scope,
95 | self.SIGNED_HEADERS,
96 | signature,
97 | )
98 | )
99 |
100 | headers = {'x-amz-date': amzdate, 'Authorization': authorization_header}
101 | request_url = '?'.join([self.ENDPOINT, canonical_querystring])
102 |
103 | return request_url, headers
104 |
105 | def validate_credentials(self):
106 | request_url, headers = self._get_request_url_and_headers()
107 | req = Request(request_url, headers=headers, method=self.METHOD)
108 |
109 | try:
110 | with urlopen(req) as res:
111 | if res.status == 200:
112 | return True
113 | else:
114 | return False
115 | except HTTPError as e:
116 | return False
117 |
118 |
--------------------------------------------------------------------------------
/run.py:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env python3
2 | # -*- coding: utf-8 -*-
3 | import platform
4 | import sys
5 |
6 | if (
7 | sys.version_info[0] == 2
8 | or (sys.version_info[0] == 3 and sys.version_info[1] <= 5)
9 | ):
10 | # Do not import any classes because they can contain not supported syntax
11 | # for older python versions. (i.e. avoid SyntaxError on imports)
12 | message = (
13 | '╔══════════════════════════════════════════════════════╗\n'
14 | '║ ║\n'
15 | '║ Your Python version has reached the end of its life. ║\n'
16 | '║ Please upgrade it as it is not maintained anymore. ║\n'
17 | '║ ║\n'
18 | '╚══════════════════════════════════════════════════════╝'
19 | )
20 | print("\033[1;31m" + message + '\033[0;0m')
21 | sys.exit(1)
22 |
23 | from helpers.cli import CLI
24 | from helpers.command import Command
25 | from helpers.config import Config
26 | from helpers.setup import Setup
27 | from helpers.template import Template
28 | from helpers.updater import Updater
29 |
30 |
31 | def run(force_setup=False):
32 |
33 | if not platform.system() in ['Linux', 'Darwin']:
34 | CLI.colored_print('Not compatible with this OS', CLI.COLOR_ERROR)
35 | else:
36 | config = Config()
37 | dict_ = config.get_dict()
38 | if config.first_time:
39 | force_setup = True
40 |
41 | if force_setup:
42 | dict_ = config.build()
43 | Setup.clone_kobodocker(config)
44 | Template.render(config)
45 | Setup.update_hosts(dict_)
46 | else:
47 | if config.auto_detect_network():
48 | Template.render(config)
49 | Setup.update_hosts(dict_)
50 |
51 | config.validate_passwords()
52 | Command.start(force_setup=force_setup)
53 |
54 |
55 | if __name__ == '__main__':
56 | try:
57 |
58 | # avoid infinite self-updating loops
59 | update_self = Updater.NO_UPDATE_SELF_OPTION not in sys.argv
60 | while True:
61 | try:
62 | sys.argv.remove(Updater.NO_UPDATE_SELF_OPTION)
63 | except ValueError:
64 | break
65 |
66 | if len(sys.argv) > 2:
67 | if sys.argv[1] == '-cf' or sys.argv[1] == '--compose-frontend':
68 | Command.compose_frontend(sys.argv[2:])
69 | elif sys.argv[1] == '-cb' or sys.argv[1] == '--compose-backend':
70 | Command.compose_backend(sys.argv[2:])
71 | elif sys.argv[1] == '-u' or sys.argv[1] == '--update':
72 | Updater.run(sys.argv[2], update_self=update_self)
73 | elif sys.argv[1] == '--upgrade':
74 | Updater.run(sys.argv[2], update_self=update_self)
75 | elif sys.argv[1] == '--auto-update':
76 | Updater.run(sys.argv[2], cron=True, update_self=update_self)
77 | else:
78 | CLI.colored_print("Bad syntax. Try 'run.py --help'",
79 | CLI.COLOR_ERROR)
80 | elif len(sys.argv) == 2:
81 | if sys.argv[1] == '-h' or sys.argv[1] == '--help':
82 | Command.help()
83 | elif sys.argv[1] == '-u' or sys.argv[1] == '--update':
84 | Updater.run(update_self=update_self)
85 | elif sys.argv[1] == '--upgrade':
86 | # 'update' was called 'upgrade' in a previous release; accept
87 | # either 'update' or 'upgrade' here to ease the transition
88 | Updater.run(update_self=update_self)
89 | elif sys.argv[1] == '--auto-update':
90 | Updater.run(cron=True, update_self=update_self)
91 | elif sys.argv[1] == '-i' or sys.argv[1] == '--info':
92 | Command.info(0)
93 | elif sys.argv[1] == '-s' or sys.argv[1] == '--setup':
94 | run(force_setup=True)
95 | elif sys.argv[1] == '-S' or sys.argv[1] == '--stop':
96 | Command.stop()
97 | elif sys.argv[1] == '-l' or sys.argv[1] == '--logs':
98 | Command.logs()
99 | elif sys.argv[1] == '-b' or sys.argv[1] == '--build':
100 | Command.build()
101 | elif sys.argv[1] == '-v' or sys.argv[1] == '--version':
102 | Command.version()
103 | elif sys.argv[1] == '-m' or sys.argv[1] == '--maintenance':
104 | Command.configure_maintenance()
105 | elif sys.argv[1] == '-sm' or sys.argv[1] == '--stop-maintenance':
106 | Command.stop_maintenance()
107 | else:
108 | CLI.colored_print("Bad syntax. Try 'run.py --help'",
109 | CLI.COLOR_ERROR)
110 | else:
111 | run()
112 |
113 | except KeyboardInterrupt:
114 | CLI.colored_print('\nUser interrupted execution', CLI.COLOR_INFO)
115 |
--------------------------------------------------------------------------------
/tests/test_run.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | from unittest.mock import patch, MagicMock
3 |
4 | from helpers.command import Command
5 | from .utils import (
6 | mock_read_config as read_config,
7 | MockCommand,
8 | MockDocker,
9 | MockUpgrading,
10 | )
11 |
12 |
13 | @patch('helpers.network.Network.is_port_open',
14 | MagicMock(return_value=False))
15 | @patch('helpers.command.Upgrading.migrate_single_to_two_databases',
16 | new=MockUpgrading.migrate_single_to_two_databases)
17 | @patch('helpers.command.Command.info',
18 | MagicMock(return_value=True))
19 | @patch('helpers.cli.CLI.run_command',
20 | new=MockCommand.run_command)
21 | def test_toggle_trivial():
22 | config = read_config()
23 | Command.start()
24 | mock_docker = MockDocker()
25 | expected_containers = MockDocker.FRONTEND_CONTAINERS + \
26 | MockDocker.BACKEND_CONTAINERS + \
27 | MockDocker.LETSENCRYPT
28 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
29 |
30 | Command.stop()
31 | assert len(mock_docker.ps()) == 0
32 | del mock_docker
33 |
34 |
35 | @patch('helpers.network.Network.is_port_open',
36 | MagicMock(return_value=False))
37 | @patch('helpers.command.Upgrading.migrate_single_to_two_databases',
38 | new=MockUpgrading.migrate_single_to_two_databases)
39 | @patch('helpers.command.Command.info',
40 | MagicMock(return_value=True))
41 | @patch('helpers.cli.CLI.run_command',
42 | new=MockCommand.run_command)
43 | def test_toggle_no_letsencrypt():
44 | config_object = read_config()
45 | config_object._Config__dict['use_letsencrypt'] = False
46 | Command.start()
47 | mock_docker = MockDocker()
48 | expected_containers = (
49 | MockDocker.FRONTEND_CONTAINERS + MockDocker.BACKEND_CONTAINERS
50 | )
51 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
52 |
53 | Command.stop()
54 | assert len(mock_docker.ps()) == 0
55 | del mock_docker
56 |
57 |
58 | @patch('helpers.network.Network.is_port_open',
59 | MagicMock(return_value=False))
60 | @patch('helpers.command.Upgrading.migrate_single_to_two_databases',
61 | new=MockUpgrading.migrate_single_to_two_databases)
62 | @patch('helpers.command.Command.info',
63 | MagicMock(return_value=True))
64 | @patch('helpers.cli.CLI.run_command',
65 | new=MockCommand.run_command)
66 | def test_toggle_frontend():
67 | config_object = read_config()
68 | Command.start(frontend_only=True)
69 | mock_docker = MockDocker()
70 | expected_containers = MockDocker.FRONTEND_CONTAINERS + \
71 | MockDocker.LETSENCRYPT
72 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
73 |
74 | Command.stop()
75 | assert len(mock_docker.ps()) == 0
76 | del mock_docker
77 |
78 |
79 | @patch('helpers.network.Network.is_port_open',
80 | MagicMock(return_value=False))
81 | @patch('helpers.command.Upgrading.migrate_single_to_two_databases',
82 | new=MockUpgrading.migrate_single_to_two_databases)
83 | @patch('helpers.command.Command.info',
84 | MagicMock(return_value=True))
85 | @patch('helpers.cli.CLI.run_command',
86 | new=MockCommand.run_command)
87 | def test_toggle_backend():
88 | config_object = read_config()
89 | config_object._Config__dict['server_role'] = 'backend'
90 | config_object._Config__dict['multi'] = True
91 |
92 | Command.start()
93 | mock_docker = MockDocker()
94 | expected_containers = MockDocker.BACKEND_CONTAINERS
95 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
96 |
97 | Command.stop()
98 | assert len(mock_docker.ps()) == 0
99 | del mock_docker
100 |
101 |
102 | @patch('helpers.network.Network.is_port_open',
103 | MagicMock(return_value=False))
104 | @patch('helpers.command.Upgrading.migrate_single_to_two_databases',
105 | new=MockUpgrading.migrate_single_to_two_databases)
106 | @patch('helpers.command.Command.info',
107 | MagicMock(return_value=True))
108 | @patch('helpers.cli.CLI.run_command',
109 | new=MockCommand.run_command)
110 | def test_toggle_maintenance():
111 | config_object = read_config()
112 | mock_docker = MockDocker()
113 | Command.start()
114 | expected_containers = (
115 | MockDocker.FRONTEND_CONTAINERS
116 | + MockDocker.BACKEND_CONTAINERS
117 | + MockDocker.LETSENCRYPT
118 | )
119 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
120 |
121 | config_object._Config__dict['maintenance_enabled'] = True
122 | Command.start()
123 | maintenance_containers = (
124 | MockDocker.BACKEND_CONTAINERS
125 | + MockDocker.MAINTENANCE_CONTAINERS
126 | + MockDocker.LETSENCRYPT
127 | )
128 | assert sorted(mock_docker.ps()) == sorted(maintenance_containers)
129 | config_object._Config__dict['maintenance_enabled'] = False
130 | Command.start()
131 | assert sorted(mock_docker.ps()) == sorted(expected_containers)
132 | Command.stop()
133 | assert len(mock_docker.ps()) == 0
134 | del mock_docker
135 |
--------------------------------------------------------------------------------
/tests/utils.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import json
3 | from unittest.mock import patch, mock_open
4 |
5 | from helpers.config import Config
6 | from helpers.singleton import Singleton
7 |
8 |
9 | def mock_read_config(overrides=None):
10 |
11 | config_dict = dict(Config.get_template())
12 | config_dict['kobodocker_path'] = '/tmp'
13 | if overrides is not None:
14 | config_dict.update(overrides)
15 |
16 | str_config = json.dumps(config_dict)
17 | # `Config()` constructor calls `read_config()` internally
18 | # We need to mock `open()` twice.
19 | # - Once to read kobo-install config file (i.e. `.run.conf`)
20 | # - Once to read value of `unique_id` (i.e. `/tmp/.uniqid`)
21 | with patch('builtins.open', spec=open) as mock_file:
22 | mock_file.side_effect = iter([
23 | mock_open(read_data=str_config).return_value,
24 | mock_open(read_data='').return_value,
25 | ])
26 | config = Config()
27 |
28 | # We call `read_config()` another time to be sure to reset the config
29 | # before each test. Thanks to `mock_open`, `Config.get_dict()` always
30 | # returns `config_dict`.
31 | with patch('builtins.open', spec=open) as mock_file:
32 | mock_file.side_effect = iter([
33 | mock_open(read_data=str_config).return_value,
34 | mock_open(read_data='').return_value,
35 | ])
36 | config.read_config()
37 |
38 | dict_ = config.get_dict()
39 | assert config_dict['kobodocker_path'] == dict_['kobodocker_path']
40 |
41 | return config
42 |
43 | def mock_reset_config(config):
44 |
45 | dict_ = dict(Config.get_template())
46 | dict_['kobodocker_path'] = '/tmp'
47 | config.__dict = dict_
48 |
49 |
50 | def mock_write_trigger_upsert_db_users(*args):
51 |
52 | content = args[1]
53 | with open('/tmp/upsert_db_users', 'w') as f:
54 | f.write(content)
55 |
56 |
57 | class MockCommand:
58 | """
59 | Create a mock class for Python2 retro compatibility.
60 | Python2 does not pass the class as the first argument explicitly when
61 | `run_command` (as a standalone method) is used as a mock.
62 | """
63 | @classmethod
64 | def run_command(cls, command, cwd=None, polling=False):
65 | if not ('docker' == command[0] and len(command) > 1 and command[1] == 'compose'):
66 | message = f'Command: `{command[0]}` is not implemented!'
67 | raise Exception(message)
68 |
69 | mock_docker = MockDocker()
70 | return mock_docker.compose(command, cwd)
71 |
72 |
73 | class MockDocker(metaclass=Singleton):
74 |
75 | BACKEND_CONTAINERS = [
76 | 'primary_postgres',
77 | 'mongo',
78 | 'redis_main',
79 | 'redis_cache',
80 | ]
81 | FRONTEND_CONTAINERS = ['nginx', 'kpi', 'enketo_express']
82 | MAINTENANCE_CONTAINERS = ['maintenance', 'kpi', 'enketo_express']
83 | LETSENCRYPT = ['letsencrypt_nginx', 'certbot']
84 |
85 | def __init__(self):
86 | self.__containers = []
87 |
88 | def ps(self):
89 | return self.__containers
90 |
91 | def compose(self, command, cwd):
92 | config_object = Config()
93 | letsencrypt = cwd == config_object.get_letsencrypt_repo_path()
94 |
95 | if command[-2] == 'config':
96 | return '\n'.join([c
97 | for c in self.FRONTEND_CONTAINERS
98 | if c != 'nginx'])
99 | if command[-2] == 'up':
100 | if letsencrypt:
101 | self.__containers += self.LETSENCRYPT
102 | elif 'backend' in command[3]:
103 | self.__containers += self.BACKEND_CONTAINERS
104 | elif 'maintenance' in command[3]:
105 | self.__containers += self.MAINTENANCE_CONTAINERS
106 | elif 'frontend' in command[3]:
107 | self.__containers += self.FRONTEND_CONTAINERS
108 | elif command[-1] == 'down':
109 | try:
110 | if letsencrypt:
111 | for container in self.LETSENCRYPT:
112 | self.__containers.remove(container)
113 | elif 'backend' in command[3]:
114 | for container in self.BACKEND_CONTAINERS:
115 | self.__containers.remove(container)
116 | elif 'maintenance' in command[3]:
117 | for container in self.MAINTENANCE_CONTAINERS:
118 | self.__containers.remove(container)
119 | elif 'frontend' in command[3]:
120 | for container in self.FRONTEND_CONTAINERS:
121 | self.__containers.remove(container)
122 | except ValueError:
123 | # Try to take a container down but was not up before.
124 | pass
125 |
126 | return True
127 |
128 |
129 | class MockUpgrading:
130 |
131 | @staticmethod
132 | def migrate_single_to_two_databases(config):
133 | pass
134 |
135 |
136 | class MockAWSValidation:
137 |
138 | def validate_credentials(self):
139 | if (
140 | self.access_key == 'test_access_key'
141 | and self.secret_key == 'test_secret_key'
142 | ):
143 | return True
144 | else:
145 | return False
146 |
--------------------------------------------------------------------------------
/helpers/cli.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import subprocess
3 | import sys
4 | import re
5 | import textwrap
6 |
7 |
8 | class CLI:
9 |
10 | NO_COLOR = '\033[0;0m'
11 | COLOR_ERROR = '\033[0;31m' # dark red
12 | COLOR_SUCCESS = '\033[0;32m' # dark green
13 | COLOR_INFO = '\033[1;34m' # blue
14 | COLOR_WARNING = '\033[1;31m' # red
15 | COLOR_QUESTION = '\033[1;33m' # dark yellow
16 | COLOR_DEFAULT = '\033[1;37m' # white
17 |
18 | EMPTY_CHARACTER = '-'
19 |
20 | DEFAULT_CHOICES = {
21 | '1': True,
22 | '2': False,
23 | }
24 | # We need an inverted dict version of `DEFAULT_CHOICES` to be able to
25 | # retrieve keys from the values
26 | DEFAULT_RESPONSES = dict(zip(DEFAULT_CHOICES.values(),
27 | DEFAULT_CHOICES.keys()))
28 |
29 | @classmethod
30 | def colored_input(cls, message, color=NO_COLOR, default=None):
31 | text = cls.get_message_with_default(message, default)
32 | input_ = input(cls.colorize(text, color))
33 |
34 | # User wants to delete value previously entered.
35 | if input_ == '-':
36 | default = ''
37 | input_ = ''
38 |
39 | return input_ if input_ is not None and input_ != '' else default
40 |
41 | @classmethod
42 | def colored_print(cls, message, color=NO_COLOR):
43 | print(cls.colorize(message, color))
44 |
45 | @classmethod
46 | def colorize(cls, message, color=NO_COLOR):
47 | return f'{color}{message}{cls.NO_COLOR}'
48 |
49 | @classmethod
50 | def framed_print(cls, message, color=COLOR_WARNING, columns=70):
51 | border = '═' * (columns - 2)
52 | blank_line = ' ' * (columns - 2)
53 | framed_message = [
54 | f'╔{border}╗',
55 | f'║{blank_line}║',
56 | ]
57 |
58 | if not isinstance(message, list):
59 | paragraphs = message.split('\n')
60 | else:
61 | paragraphs = ''.join(message).split('\n')
62 |
63 | for paragraph in paragraphs:
64 | if paragraph == '':
65 | framed_message.append(
66 | f'║{blank_line}║'
67 | )
68 | continue
69 |
70 | for line in textwrap.wrap(paragraph, columns - 4):
71 | message_length = len(line)
72 | spacer = ' ' * (columns - 4 - message_length)
73 | framed_message.append(
74 | f'║ {line}{spacer} ║'
75 | )
76 |
77 | framed_message.append(f'║{blank_line}║')
78 | framed_message.append(f'╚{border}╝')
79 | cls.colored_print('\n'.join(framed_message), color=color)
80 |
81 | @classmethod
82 | def get_response(cls, validators=None, default='', to_lower=True,
83 | error_msg="Sorry, I didn't understand that!"):
84 |
85 | use_default = False
86 | # If not validators are provided, let's use default validation
87 | # "Yes/No", where "Yes" equals 1, and "No" equals 2
88 | # Example:
89 | # Are you sure?
90 | # 1) Yes
91 | # 2) No
92 | if validators is None:
93 | use_default = True
94 | default = cls.DEFAULT_RESPONSES[default]
95 | validators = cls.DEFAULT_CHOICES.keys()
96 |
97 | while True:
98 | try:
99 | response = cls.colored_input('', cls.COLOR_QUESTION, default)
100 |
101 | if (
102 | response.lower() in map(lambda x: x.lower(), validators)
103 | or validators is None
104 | or (
105 | isinstance(validators, str)
106 | and validators.startswith('~')
107 | and re.match(validators[1:], response)
108 | )
109 | ):
110 | break
111 | else:
112 | cls.colored_print(error_msg,
113 | cls.COLOR_ERROR)
114 | except ValueError:
115 | cls.colored_print("Sorry, I didn't understand that.",
116 | cls.COLOR_ERROR)
117 |
118 | if use_default:
119 | return cls.DEFAULT_CHOICES[response]
120 |
121 | return response.lower() if to_lower else response
122 |
123 | @classmethod
124 | def get_message_with_default(cls, message, default):
125 | message = f'{message} ' if message else ''
126 |
127 | if default is None:
128 | default = ''
129 | else:
130 | default = '{white}[{off}{default}{white}]{off}: '.format(
131 | white=cls.COLOR_DEFAULT,
132 | off=cls.NO_COLOR,
133 | default=default
134 | )
135 |
136 | if message:
137 | message = f'{message.strip()}: ' if not default else message
138 |
139 | return f'{message}{default}'
140 |
141 | @classmethod
142 | def run_command(cls, command, cwd=None, polling=False):
143 | if polling:
144 | process = subprocess.Popen(command, stdout=subprocess.PIPE, cwd=cwd)
145 | while True:
146 | output = process.stdout.readline()
147 | if output == '' and process.poll() is not None:
148 | break
149 | if output:
150 | print(output.decode().strip())
151 | return process.poll()
152 | else:
153 | try:
154 | stdout = subprocess.check_output(command,
155 | universal_newlines=True,
156 | cwd=cwd)
157 | except subprocess.CalledProcessError as cpe:
158 | # Error will be display by above command.
159 | # ^^^ this doesn't seem to be true? let's write it explicitly
160 | # see https://docs.python.org/3/library/subprocess.html#subprocess.check_output
161 | sys.stderr.write(cpe.output)
162 | cls.colored_print('An error has occurred', CLI.COLOR_ERROR)
163 | sys.exit(1)
164 | return stdout
165 |
166 | @classmethod
167 | def yes_no_question(cls, question, default=True,
168 | labels=['Yes', 'No']):
169 | cls.colored_print(question, color=cls.COLOR_QUESTION)
170 | for index, label in enumerate(labels):
171 | choice_number = index + 1
172 | cls.colored_print(f'\t{choice_number}) {label}')
173 | return cls.get_response(default=default)
174 |
--------------------------------------------------------------------------------
/helpers/setup.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import os
3 | import shutil
4 | import sys
5 | import tempfile
6 |
7 | from helpers.cli import CLI
8 | from helpers.command import Command
9 | from helpers.config import Config
10 | from helpers.template import Template
11 |
12 |
13 | class Setup:
14 |
15 | @classmethod
16 | def clone_kobodocker(cls, config):
17 | """
18 | Args:
19 | config (helpers.config.Config)
20 | """
21 | dict_ = config.get_dict()
22 | do_update = config.first_time
23 |
24 | if not os.path.isdir(os.path.join(dict_['kobodocker_path'], '.git')):
25 | # Move unique id file to /tmp in order to clone without errors
26 | # (e.g. not empty directory)
27 | tmp_dirpath = tempfile.mkdtemp()
28 | shutil.move(os.path.join(dict_['kobodocker_path'],
29 | Config.UNIQUE_ID_FILE),
30 | os.path.join(tmp_dirpath, Config.UNIQUE_ID_FILE))
31 |
32 | # clone project
33 | git_command = [
34 | 'git', 'clone', 'https://github.com/kobotoolbox/kobo-docker',
35 | dict_['kobodocker_path']
36 | ]
37 | CLI.run_command(git_command, cwd=os.path.dirname(
38 | dict_['kobodocker_path']))
39 |
40 | shutil.move(os.path.join(tmp_dirpath, Config.UNIQUE_ID_FILE),
41 | os.path.join(dict_['kobodocker_path'],
42 | Config.UNIQUE_ID_FILE))
43 | shutil.rmtree(tmp_dirpath)
44 | do_update = True # Force update
45 |
46 | if do_update:
47 | cls.update_kobodocker(dict_)
48 |
49 | @classmethod
50 | def post_update(cls, cron):
51 |
52 | config = Config()
53 |
54 | # When `cron` is True, we want to bypass question and just recreate
55 | # YML and environment files from new templates
56 | if cron is True:
57 | current_dict = config.get_upgraded_dict()
58 | config.set_config(current_dict)
59 | config.write_config()
60 | Template.render(config, force=True)
61 | sys.exit(0)
62 |
63 | message = (
64 | 'After an update, it is strongly recommended to run\n'
65 | '`python3 run.py --setup` to regenerate environment files.'
66 | )
67 | CLI.framed_print(message, color=CLI.COLOR_INFO)
68 | response = CLI.yes_no_question('Do you want to proceed?')
69 | if response is True:
70 | current_dict = config.build()
71 | Template.render(config)
72 | Setup.update_hosts(current_dict)
73 | question = 'Do you want to (re)start containers?'
74 | response = CLI.yes_no_question(question)
75 | if response is True:
76 | Command.start(force_setup=True)
77 |
78 | @staticmethod
79 | def update_kobodocker(dict_=None):
80 | """
81 | Args:
82 | dict_ (dict): Dictionary provided by `Config.get_dict()`
83 | """
84 | if not dict_:
85 | config = Config()
86 | dict_ = config.get_dict()
87 |
88 | # fetch new tags and prune
89 | git_command = ['git', 'fetch', '-p']
90 | CLI.run_command(git_command, cwd=dict_['kobodocker_path'])
91 |
92 | # checkout branch
93 | git_command = ['git', 'checkout', '--force', Config.KOBO_DOCKER_BRANCH]
94 | CLI.run_command(git_command, cwd=dict_['kobodocker_path'])
95 |
96 | # update code
97 | git_command = ['git', 'pull', 'origin', Config.KOBO_DOCKER_BRANCH]
98 | CLI.run_command(git_command, cwd=dict_['kobodocker_path'])
99 |
100 | @staticmethod
101 | def update_koboinstall(version):
102 | # fetch new tags and prune
103 | git_fetch_prune_command = ['git', 'fetch', '-p']
104 | CLI.run_command(git_fetch_prune_command)
105 |
106 | # checkout branch
107 | git_command = ['git', 'checkout', '--force', version]
108 | CLI.run_command(git_command)
109 |
110 | # update code
111 | git_command = ['git', 'pull', 'origin', version]
112 | CLI.run_command(git_command)
113 |
114 | @classmethod
115 | def update_hosts(cls, dict_):
116 | """
117 |
118 | Args:
119 | dict_ (dict): Dictionary provided by `Config.get_dict()`
120 | """
121 | if dict_['local_installation']:
122 | start_sentence = '### (BEGIN) KoboToolbox local routes'
123 | end_sentence = '### (END) KoboToolbox local routes'
124 |
125 | _, tmp_file_path = tempfile.mkstemp()
126 |
127 | with open('/etc/hosts', 'r') as f:
128 | tmp_host = f.read()
129 |
130 | start_position = tmp_host.lower().find(start_sentence.lower())
131 | end_position = tmp_host.lower().find(end_sentence.lower())
132 |
133 | if start_position > -1:
134 | tmp_host = tmp_host[0: start_position] \
135 | + tmp_host[end_position + len(end_sentence) + 1:]
136 |
137 | public_domain_name = dict_['public_domain_name']
138 | routes = (
139 | f"{dict_['local_interface_ip']} "
140 | f"{dict_['kpi_subdomain']}.{public_domain_name} "
141 | f"{dict_['kc_subdomain']}.{public_domain_name} "
142 | f"{dict_['ee_subdomain']}.{public_domain_name}"
143 | )
144 |
145 | bof = tmp_host.strip()
146 | tmp_host = (
147 | f'{bof}'
148 | f'\n{start_sentence}'
149 | f'\n{routes}'
150 | f'\n{end_sentence}'
151 | )
152 |
153 | with open(tmp_file_path, 'w') as f:
154 | f.write(tmp_host)
155 |
156 | message = (
157 | 'Privileges escalation is required to update '
158 | 'your `/etc/hosts`.'
159 | )
160 | CLI.framed_print(message, color=CLI.COLOR_INFO)
161 | dict_['review_host'] = CLI.yes_no_question(
162 | 'Do you want to review your /etc/hosts file '
163 | 'before overwriting it?',
164 | default=dict_['review_host']
165 | )
166 | if dict_['review_host']:
167 | print(tmp_host)
168 | CLI.colored_input('Press any keys when ready')
169 |
170 | # Save 'review_host'
171 | config = Config()
172 | config.write_config()
173 |
174 | cmd = (
175 | 'sudo cp /etc/hosts /etc/hosts.old '
176 | '&& sudo cp {tmp_file_path} /etc/hosts'
177 | ).format(tmp_file_path=tmp_file_path)
178 |
179 | return_value = os.system(cmd)
180 |
181 | os.unlink(tmp_file_path)
182 |
183 | if return_value != 0:
184 | sys.exit(1)
185 |
186 | @staticmethod
187 | def validate_already_run():
188 | """
189 | Validates that Setup has been run at least once and kobo-docker has been
190 | pulled and checked out before going further.
191 | """
192 |
193 | config = Config()
194 | dict_ = config.get_dict()
195 |
196 | def display_error_message(message):
197 | message += '\nPlease run `python3 run.py --setup` first.'
198 | CLI.framed_print(message, color=CLI.COLOR_ERROR)
199 | sys.exit(1)
200 |
201 | try:
202 | dict_['kobodocker_path']
203 | except KeyError:
204 | display_error_message('No configuration file found.')
205 |
206 | if not os.path.isdir(os.path.join(dict_['kobodocker_path'], '.git')):
207 | display_error_message('`kobo-docker` repository is missing!')
208 |
--------------------------------------------------------------------------------
/helpers/network.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import array
3 | import fcntl
4 | import platform
5 | import socket
6 | import struct
7 | import sys
8 | from http import client as httplib
9 | from urllib.request import urlopen
10 |
11 | from helpers.cli import CLI
12 |
13 |
14 | class Network:
15 |
16 | STATUS_OK_200 = 200
17 |
18 | @staticmethod
19 | def get_local_interfaces(all_=False):
20 | """
21 | Returns a dictionary of name:ip key value pairs.
22 | Linux Only!
23 | Source: https://gist.github.com/bubthegreat/24c0c43ad159d8dfed1a5d3f6ca99f9b
24 |
25 | Args:
26 | all_ (bool): If False, filter virtual interfaces such VMWare,
27 | Docker etc...
28 | Returns:
29 | dict
30 | """
31 | ip_dict = {}
32 | excluded_interfaces = ('lo', 'docker', 'br-', 'veth', 'vmnet')
33 |
34 | if platform.system() == 'Linux':
35 | # Max possible bytes for interface result.
36 | # Will truncate if more than 4096 characters to describe interfaces.
37 | MAX_BYTES = 4096
38 |
39 | # We're going to make a blank byte array to operate on.
40 | # This is our fill char.
41 | FILL_CHAR = b'\0'
42 |
43 | # Command defined in ioctl.h for the system operation for get iface
44 | # list.
45 | # Defined at https://code.woboq.org/qt5/include/bits/ioctls.h.html
46 | # under /* Socket configuration controls. */ section.
47 | SIOCGIFCONF = 0x8912
48 |
49 | # Make a dgram socket to use as our file descriptor that we'll
50 | # operate on.
51 | sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
52 |
53 | # Make a byte array with our fill character.
54 | names = array.array('B', MAX_BYTES * FILL_CHAR)
55 |
56 | # Get the address of our names byte array for use in our struct.
57 | names_address, names_length = names.buffer_info()
58 |
59 | # Create a mutable byte buffer to store the data in
60 | mutable_byte_buffer = struct.pack('iL', MAX_BYTES, names_address)
61 |
62 | # mutate our mutable_byte_buffer with the results of get_iface_list.
63 | # NOTE: mutated_byte_buffer is just a reference to
64 | # mutable_byte_buffer - for the sake of clarity we've defined
65 | # them as separate variables, however they are the same address
66 | # space - that's how fcntl.ioctl() works since the mutate_flag=True
67 | # by default.
68 | mutated_byte_buffer = fcntl.ioctl(sock.fileno(),
69 | SIOCGIFCONF,
70 | mutable_byte_buffer)
71 |
72 | # Get our max_bytes of our mutated byte buffer
73 | # that points to the names variable address space.
74 | max_bytes_out, names_address_out = struct.unpack(
75 | 'iL',
76 | mutated_byte_buffer)
77 |
78 | # Convert names to a bytes array - keep in mind we've mutated the
79 | # names array, so now our bytes out should represent the bytes
80 | # results of the get iface list ioctl command.
81 | namestr = names.tobytes()
82 |
83 | namestr[:max_bytes_out]
84 |
85 | bytes_out = namestr[:max_bytes_out]
86 |
87 | # Each entry is 40 bytes long. The first 16 bytes are the
88 | # name string. The 20-24th bytes are IP address octet strings in
89 | # byte form - one for each byte.
90 | # Don't know what 17-19 are, or bytes 25:40.
91 |
92 | for i in range(0, max_bytes_out, 40):
93 | name = namestr[i: i + 16].split(FILL_CHAR, 1)[0]
94 | name = name.decode()
95 | ip_bytes = namestr[i + 20:i + 24]
96 | full_addr = []
97 | for netaddr in ip_bytes:
98 | if isinstance(netaddr, int):
99 | full_addr.append(str(netaddr))
100 | elif isinstance(netaddr, str):
101 | full_addr.append(str(ord(netaddr)))
102 | if not name.startswith(excluded_interfaces) or all_:
103 | ip_dict[name] = '.'.join(full_addr)
104 | else:
105 | try:
106 | import netifaces
107 | except ImportError:
108 | CLI.colored_print('You must install netinfaces first! Please '
109 | 'type `pip install netifaces --user`',
110 | CLI.COLOR_ERROR)
111 | sys.exit(1)
112 |
113 | for interface in netifaces.interfaces():
114 | if not interface.startswith(excluded_interfaces) or all_:
115 | ifaddresses = netifaces.ifaddresses(interface)
116 | if (
117 | ifaddresses.get(netifaces.AF_INET)
118 | and ifaddresses.get(netifaces.AF_INET)[0].get('addr')
119 | ):
120 | addresses = ifaddresses.get(netifaces.AF_INET)
121 | ip_dict[interface] = addresses[0].get('addr')
122 | for i in range(1, len(addresses)):
123 | virtual_interface = '{interface}:{idx}'.format(
124 | interface=interface,
125 | idx=i
126 | )
127 | ip_dict[virtual_interface] = addresses[i]['addr']
128 |
129 | return ip_dict
130 |
131 | @staticmethod
132 | def get_primary_ip():
133 | """
134 | https://stackoverflow.com/a/28950776/1141214
135 | :return:
136 | """
137 | s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
138 | try:
139 | # doesn't even have to be reachable
140 | # …but it can't be a broadcast address, or you'll get
141 | # `Permission denied`. See recent comments on the same SO answer:
142 | # https://stackoverflow.com/questions/166506/finding-local-ip-addresses-using-pythons-stdlib/28950776#comment128390746_28950776
143 | s.connect(('10.255.255.254', 1))
144 | ip_address = s.getsockname()[0]
145 | except:
146 | ip_address = None
147 | finally:
148 | s.close()
149 | return ip_address
150 |
151 | @classmethod
152 | def get_primary_interface(cls):
153 | """
154 | :return: string
155 | """
156 | primary_ip = cls.get_primary_ip()
157 | local_interfaces = cls.get_local_interfaces()
158 |
159 | for interface, ip_address in local_interfaces.items():
160 | if ip_address == primary_ip:
161 | return interface
162 |
163 | return 'eth0'
164 |
165 | @staticmethod
166 | def status_check(hostname, endpoint, port=80, secure=False):
167 | try:
168 | if secure:
169 | conn = httplib.HTTPSConnection(
170 | f'{hostname}:{port}',
171 | timeout=10)
172 | else:
173 | conn = httplib.HTTPConnection(
174 | f'{hostname}:{port}',
175 | timeout=10)
176 | conn.request('GET', endpoint)
177 | response = conn.getresponse()
178 | return response.status
179 | except:
180 | pass
181 |
182 | return
183 |
184 | @staticmethod
185 | def is_port_open(port):
186 | sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
187 | result = sock.connect_ex(('127.0.0.1', int(port)))
188 | return result == 0
189 |
190 | @staticmethod
191 | def curl(url):
192 | try:
193 | response = urlopen(url)
194 | data = response.read()
195 | if isinstance(data, str):
196 | # Python 2
197 | return data
198 | else:
199 | # Python 3
200 | return data.decode(response.headers.get_content_charset())
201 | except Exception as e:
202 | pass
203 | return
204 |
--------------------------------------------------------------------------------
/templates/kobo-env/enketo_express/config.json.tpl:
--------------------------------------------------------------------------------
1 | {
2 | "app name": "Enketo Express for KoboToolbox",
3 | "linked form and data server": {
4 | "name": "KoboToolbox",
5 | "server url": "",
6 | "api key": "${ENKETO_API_KEY}"
7 | },
8 | "ip filtering": {
9 | "allowPrivateIPAddress": ${ENKETO_ALLOW_PRIVATE_IP_ADDRESS},
10 | "allowMetaIPAddress": false,
11 | "allowIPAddressList": [],
12 | "denyAddressList": []
13 | },
14 | "encryption key": "${ENKETO_ENCRYPTION_KEY}",
15 | "less secure encryption key": "${ENKETO_LESS_SECURE_ENCRYPTION_KEY}",
16 | "support": {
17 | "email": "${DEFAULT_FROM_EMAIL}"
18 | },
19 | "widgets": [
20 | "note",
21 | "select-desktop",
22 | "select-mobile",
23 | "autocomplete",
24 | "geo",
25 | "textarea",
26 | "url",
27 | "table",
28 | "radio",
29 | "date",
30 | "time",
31 | "datetime",
32 | "select-media",
33 | "file",
34 | "draw",
35 | "rank",
36 | "likert",
37 | "range",
38 | "columns",
39 | "image-view",
40 | "comment",
41 | "image-map",
42 | "date-native",
43 | "date-native-ios",
44 | "date-mobile",
45 | "text-max",
46 | "text-print",
47 | "rating",
48 | "thousands-sep",
49 | "integer",
50 | "decimal",
51 | "../../../node_modules/enketo-image-customization-widget/image-customization",
52 | "../../../node_modules/enketo-literacy-test-widget/literacywidget"
53 | ],
54 | "redis": {
55 | "cache": {
56 | "host": "redis-cache.${PRIVATE_DOMAIN_NAME}",
57 | "port": "${REDIS_CACHE_PORT}"{% if REDIS_PASSWORD %},{% endif REDIS_PASSWORD %}
58 | {% if REDIS_PASSWORD %}
59 | "password": ${REDIS_PASSWORD_JS_ENCODED}
60 | {% endif REDIS_PASSWORD %}
61 | },
62 | "main": {
63 | "host": "redis-main.${PRIVATE_DOMAIN_NAME}",
64 | "port": "${REDIS_MAIN_PORT}"{% if REDIS_PASSWORD %},{% endif REDIS_PASSWORD %}
65 | {% if REDIS_PASSWORD %}
66 | "password": ${REDIS_PASSWORD_JS_ENCODED}
67 | {% endif REDIS_PASSWORD %}
68 | }
69 | },
70 | "google": {
71 | "api key": "${GOOGLE_API_KEY}",
72 | "analytics": {
73 | "ua": "${GOOGLE_UA}",
74 | "domain": "${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}"
75 | }
76 | },
77 | "logo": {
78 | "source": "",
79 | "href": ""
80 | },
81 | "payload limit": "1mb",
82 | "text field character limit": 1000000,
83 | "maps": [
84 | {
85 | "name": "humanitarian",
86 | "tiles": [ "https://{s}.tile.openstreetmap.fr/hot/{z}/{x}/{y}.png" ],
87 | "attribution": "© OpenStreetMap & Yohan Boniface & Humanitarian OpenStreetMap Team | Terms"
88 | }, {
89 | "name": "satellite",
90 | "tiles": [ "https://server.arcgisonline.com/ArcGIS/rest/services/World_Imagery/MapServer/tile/{z}/{y}/{x}" ],
91 | "attribution": "Tiles © Esri — Source: Esri, i-cubed, USDA, USGS, AEX, GeoEye, Getmapping, Aerogrid, IGN, IGP, UPR-EGP, and the GIS User Community"
92 | }, {
93 | "name": "terrain",
94 | "tiles": [ "https://{s}.tile.opentopomap.org/{z}/{x}/{y}.png" ],
95 | "attribution": "© OpenStreetMap | Terms"
96 | }, {
97 | "name": "streets",
98 | "tiles": [ "https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png" ],
99 | "attribution": "© OpenStreetMap | Terms"
100 | }
101 | ]
102 | }
103 |
--------------------------------------------------------------------------------
/helpers/upgrading.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | from __future__ import annotations
3 |
4 | import subprocess
5 | import sys
6 | from shutil import which
7 |
8 | from helpers.cli import CLI
9 | from helpers.utils import run_docker_compose
10 |
11 |
12 | class Upgrading:
13 |
14 | @staticmethod
15 | def migrate_single_to_two_databases(config: 'helpers.Config'):
16 | """
17 | Check the contents of the databases. If KPI's is empty or doesn't exist
18 | while KoboCAT's has user data, then we are migrating from a
19 | single-database setup
20 |
21 | Args
22 | config (helpers.config.Config)
23 | """
24 | dict_ = config.get_dict()
25 |
26 | def _kpi_db_alias_kludge(command):
27 | """
28 | Sorry, this is not very nice. See
29 | https://github.com/kobotoolbox/kobo-docker/issues/264.
30 | """
31 | set_env = 'DATABASE_URL="${KPI_DATABASE_URL}"'
32 | return ['bash', '-c', f'{set_env} {command}']
33 |
34 | kpi_run_command = run_docker_compose(dict_, [
35 | '-f', 'docker-compose.frontend.yml',
36 | '-f', 'docker-compose.frontend.override.yml',
37 | '-p', config.get_prefix('frontend'),
38 | 'run', '--rm', 'kpi'
39 | ])
40 |
41 | # Make sure Postgres is running
42 | # We add this message to users because when AWS backups are activated,
43 | # it takes a long time to install the virtualenv in PostgreSQL
44 | # container, so the `wait_for_database` below sits there a while.
45 | # It makes us think kobo-install is frozen.
46 | CLI.colored_print(
47 | 'Waiting for PostgreSQL database to be up & running...',
48 | CLI.COLOR_INFO)
49 | frontend_command = kpi_run_command + _kpi_db_alias_kludge(' '.join([
50 | 'python', 'manage.py',
51 | 'wait_for_database', '--retries', '45'
52 | ]))
53 | CLI.run_command(frontend_command, dict_['kobodocker_path'])
54 | CLI.colored_print('The PostgreSQL database is running!',
55 | CLI.COLOR_SUCCESS)
56 |
57 | frontend_command = kpi_run_command + _kpi_db_alias_kludge(' '.join([
58 | 'python', 'manage.py',
59 | 'is_database_empty', 'kpi', 'kobocat'
60 | ]))
61 | output = CLI.run_command(frontend_command, dict_['kobodocker_path'])
62 | # TODO: read only stdout and don't consider stderr unless the exit code
63 | # is non-zero. Currently, `output` combines both stdout and stderr
64 | kpi_kc_db_empty = output.strip().split('\n')[-1]
65 |
66 | if kpi_kc_db_empty == 'True\tFalse':
67 | # KPI empty but KC is not: run the two-database upgrade script
68 | CLI.colored_print(
69 | 'Upgrading from single-database setup to separate databases '
70 | 'for KPI and KoboCAT',
71 | CLI.COLOR_INFO
72 | )
73 | message = (
74 | 'Upgrading to separate databases is required to run the latest '
75 | 'release of KoboToolbox, but it may be a slow process if you '
76 | 'have a lot of data. Expect at least one minute of downtime '
77 | 'for every 1,500 KPI assets. Assets are surveys and library '
78 | 'items: questions, blocks, and templates.\n'
79 | '\n'
80 | 'To postpone this process, downgrade to the last '
81 | 'single-database release by stopping this script and executing '
82 | 'the following commands:\n'
83 | '\n'
84 | ' python3 run.py --stop\n'
85 | ' git fetch\n'
86 | ' git checkout shared-database-obsolete\n'
87 | ' python3 run.py --update\n'
88 | ' python3 run.py --setup\n'
89 | )
90 | CLI.framed_print(message)
91 | message = (
92 | 'For help, visit https://community.kobotoolbox.org/t/upgrading-'
93 | 'to-separate-databases-for-kpi-and-kobocat/7202.'
94 | )
95 | CLI.colored_print(message, CLI.COLOR_WARNING)
96 | response = CLI.yes_no_question(
97 | 'Do you want to proceed?',
98 | default=False
99 | )
100 | if response is False:
101 | sys.exit(0)
102 |
103 | backend_command = run_docker_compose(dict_, [
104 | '-f', f'docker-compose.backend.yml',
105 | '-f', f'docker-compose.backend.override.yml',
106 | '-p', config.get_prefix('backend'),
107 | 'exec', 'postgres', 'bash',
108 | '/kobo-docker-scripts/scripts/clone_data_from_kc_to_kpi.sh',
109 | '--noinput'
110 | ])
111 | try:
112 | subprocess.check_call(
113 | backend_command, cwd=dict_['kobodocker_path']
114 | )
115 | except subprocess.CalledProcessError:
116 | CLI.colored_print('An error has occurred', CLI.COLOR_ERROR)
117 | sys.exit(1)
118 |
119 | elif kpi_kc_db_empty not in [
120 | 'True\tTrue',
121 | 'False\tTrue',
122 | 'False\tFalse',
123 | ]:
124 | # The output was invalid
125 | CLI.colored_print('An error has occurred', CLI.COLOR_ERROR)
126 | sys.stderr.write(kpi_kc_db_empty)
127 | sys.exit(1)
128 |
129 | @staticmethod
130 | def two_databases(upgraded_dict: dict, current_dict: dict) -> dict:
131 | """
132 | If the configuration came from a previous version that had a single
133 | Postgres database, we need to make sure the new `kc_postgres_db` is
134 | set to the name of that single database, *not* the default from
135 | `Config.get_template()`
136 |
137 | Args:
138 | upgraded_dict (dict): Configuration values to be upgraded
139 | current_dict (dict): Current configuration values
140 | (i.e. `Config.get_dict()`)
141 | Returns:
142 | dict
143 |
144 | """
145 |
146 | try:
147 | current_dict['postgres_db']
148 | except KeyError:
149 | # Install has been made with two databases.
150 | return upgraded_dict
151 |
152 | try:
153 | current_dict['kc_postgres_db']
154 | except KeyError:
155 | # Configuration does not have names of KPI and KoboCAT databases.
156 | # Let's copy old single database name to KoboCAT database name
157 | upgraded_dict['kc_postgres_db'] = current_dict['postgres_db']
158 |
159 | # Force this property to False. It helps to detect whether the
160 | # database names have changed in `Config.__questions_postgres()`
161 | upgraded_dict['two_databases'] = False
162 |
163 | return upgraded_dict
164 |
165 | @staticmethod
166 | def use_booleans(upgraded_dict: dict) -> dict:
167 | """
168 | Until version 3.x, two constants (`Config.TRUE` and `Config.FALSE`) were
169 | used to store "Yes/No" users' responses. It made the code more
170 | complex than it should have been.
171 | This method converts these values to boolean.
172 | - `Config.TRUE` -> `True`
173 | - `Config.FALSE` -> False`
174 | Args:
175 | upgraded_dict (dict): Configuration values to be upgraded
176 |
177 | Returns:
178 | dict
179 | """
180 | try:
181 | upgraded_dict['use_booleans_v4']
182 | except KeyError:
183 | pass
184 | else:
185 | return upgraded_dict
186 |
187 | boolean_properties = [
188 | 'advanced',
189 | 'aws_backup_bucket_deletion_rule_enabled',
190 | 'backup_from_primary',
191 | 'block_common_http_ports',
192 | 'custom_secret_keys',
193 | 'customized_ports',
194 | 'debug',
195 | 'dev_mode',
196 | 'expose_backend_ports',
197 | 'https',
198 | 'local_installation',
199 | 'multi',
200 | 'npm_container',
201 | 'postgres_settings',
202 | 'proxy',
203 | 'raven_settings',
204 | 'review_host',
205 | 'smtp_use_tls',
206 | 'staging_mode',
207 | 'two_databases',
208 | 'use_aws',
209 | 'use_backup',
210 | 'use_letsencrypt',
211 | 'use_private_dns',
212 | 'uwsgi_settings',
213 | ]
214 | for property_ in boolean_properties:
215 | try:
216 | if isinstance(upgraded_dict[property_], bool):
217 | continue
218 | except KeyError:
219 | pass
220 | else:
221 | upgraded_dict[property_] = True \
222 | if upgraded_dict[property_] == '1' else False
223 |
224 | upgraded_dict['use_booleans_v4'] = True
225 |
226 | return upgraded_dict
227 |
--------------------------------------------------------------------------------
/readme.md:
--------------------------------------------------------------------------------
1 | The purpose of the script is to install KoboToolbox in minutes without messing with configuration files.
2 | It prompts the user to answer some questions to create configuration files automatically and to start docker containers based on [`kobo-docker`](https://github.com/kobotoolbox/kobo-docker "").
3 |
4 | ## :warning: You _must observe_ the following when upgrading:
5 |
6 | ### …from any release older than [`2.022.44`](https://github.com/kobotoolbox/kobo-install/releases/tag/2.022.44) (November 2022)
7 |
8 | If you have already installed KoboToolbox between March 2019 and November 2022, you **must** complete [a manual upgrade process](https://github.com/kobotoolbox/kobo-docker/blob/master/doc/November-2022-Upgrade.md) before trying to upgrade. **If you do not, `kobo-install` will not be able to start.**
9 |
10 | ### …from any release older than [`2.020.18`](https://github.com/kobotoolbox/kobo-install/releases/tag/2.020.18) (May 2020)
11 |
12 | Prior to release [`2.020.18`](https://github.com/kobotoolbox/kobo-install/releases/tag/2.020.18), [KPI](https://github.com/kobotoolbox/kpi) and [KoBoCAT](https://github.com/kobotoolbox/kobocat) both shared a common Postgres database. They now each have their own. **If you are upgrading an existing single-database installation, you must follow [these instructions](https://community.kobotoolbox.org/t/upgrading-to-separate-databases-for-kpi-and-kobocat/7202)** to migrate the KPI tables to a new database and adjust your configuration appropriately.
13 |
14 | If you do not want to upgrade at this time, please use the [`shared-database-obsolete`](https://github.com/kobotoolbox/kobo-install/tree/shared-database-obsolete) branch instead.
15 |
16 | ### …installations made prior to March 2019
17 |
18 | If you have already installed KoboToolbox with `kobo-docker` prior March 2019,
19 | you **must** complete [a manual upgrade process](https://github.com/kobotoolbox/kobo-docker/#important-notice-when-upgrading-from-commit-5c2ef02-march-4-2019-or-earlier)
20 | before using this repository. **If you do not, `kobo-install` will not be able to start.**
21 |
22 | ## Versions
23 |
24 | Release branches `release/*` (e.g. `release/2.024.36`) are the recommended branches to use with `kobo-install` on your production environment. From the `kpi` folder run `git branch -rl 'origin/release/*'` to list release branches and then switch to a release branch of your choice.
25 |
26 | Branch `main` is a pre-release of the next version. It contains new features and bug fixes.
27 |
28 | Other branches are for development purposes.
29 |
30 | ## Usage
31 |
32 | `$kobo-install> python3 run.py`
33 |
34 | First time the command is executed, setup will be launched.
35 | Subsequent executions will launch docker containers directly.
36 |
37 | Rebuild configuration:
38 | `$kobo-install> python3 run.py --setup`
39 |
40 | Get info:
41 | `$kobo-install> python3 run.py --info`
42 |
43 | Get docker logs:
44 | `$kobo-install> python3 run.py --logs`
45 |
46 | Update KoboToolbox:
47 | `$kobo-install> python3 run.py --update [branch or tag]`
48 |
49 | By default, fetch the latest version of `master` branch
50 |
51 |
52 | Stop KoboToolbox:
53 | `$kobo-install> python3 run.py --stop`
54 |
55 | Get help:
56 | `$kobo-install> python3 run.py --help`
57 |
58 | Get version:
59 | `$kobo-install> python3 run.py --version`
60 |
61 | Build kpi and kobocat (dev mode):
62 | `$kobo-install> python3 run.py --build`
63 |
64 | Run docker commands on front-end containers:
65 | `$kobo-install> python3 run.py --compose-frontend [docker-compose arguments]`
66 |
67 | Run docker commands on back-end containers:
68 | `$kobo-install> python3 run.py --compose-backend [docker-compose arguments]`
69 |
70 | Start maintenance mode:
71 | `$kobo-install> python3 run.py --maintenance`
72 |
73 | Stop maintenance mode:
74 | `$kobo-install> python3 run.py --stop-maintenance`
75 |
76 |
77 | ## Build the configuration
78 | User can choose between 2 types of installations:
79 |
80 | - `Workstation`: KoboToolbox doesn't need to be accessible from anywhere except the computer where it's installed. No DNS needed
81 | - `Server`: KoboToolbox needs to be accessible from the local network or from the Internet. DNS are needed
82 |
83 | ### Options
84 |
85 | |Option|Default|Workstation|Server
86 | |---|---|---|---|
87 | |Installation directory| **../kobo-docker** | ✓ | ✓ |
88 | |SMTP information| | ✓ | ✓ (front end only) |
89 | |Public domain name| **kobo.local** | | ✓ (front end only) |
90 | |Subdomain names| **kf, kc, ee** | | ✓ (front end only) |
91 | |Use HTTPS1| **False** (Workstation)
**True** (Server) | | ✓ (front end only) |
92 | |Super user's username| **super_admin** | ✓ | ✓ (front end only) |
93 | |Super user's password| **Random string** | ✓ | ✓ (front end only) |
94 | |Activate backups2| **False** | ✓ | ✓ (back end only) |
95 |
96 | ### Advanced Options
97 |
98 | | Option |Default|Workstation|Server
99 | |-------------------------------------------------|---|---|---|
100 | | Webserver port | **80** | ✓ | |
101 | | Reverse proxy internal port | **8080** | | ✓ (front end only) |
102 | | Network interface | **Autodetected** | ✓ | ✓ (front end only) |
103 | | Use separate servers | **No** | | ✓ |
104 | | Use DNS for private routes | **No** | | ✓ (front end only) |
105 | | Back-end server IP _(if previous answer is no)_ | **Local IP** | | ✓ (front end only) |
106 | | PostgreSQL DB | **kobo** | ✓ | ✓ |
107 | | PostgreSQL user's username | **kobo** | ✓ | ✓ |
108 | | PostgreSQL user's password | **Autogenerate** | ✓ | ✓ |
109 | | PostgreSQL number of connections3 | **100** | ✓ | ✓ (back end only) |
110 | | PostgreSQL RAM3 | **2** | ✓ | ✓ (back end only) |
111 | | PostgreSQL Application Profile3 | **Mixed** | ✓ | ✓ (back end only) |
112 | | PostgreSQL Storage3 | **HDD** | ✓ | ✓ (back end only) |
113 | | MongoDB super user's username | **root** | ✓ | ✓ |
114 | | MongoDB super user's password | **Autogenerate** | ✓ | ✓ |
115 | | MongoDB user's username | **kobo** | ✓ | ✓ |
116 | | MongoDB user's password | **Autogenerate** | ✓ | ✓ |
117 | | Redis password4 | **Autogenerate** | ✓ | ✓ |
118 | | Use AWS storage5 | **No** | ✓ | ✓ |
119 | | uWGI workers | **start: 2, max: 4** | ✓ | ✓ (front end only) |
120 | | uWGI memory limit | **128 MB** | ✓ | ✓ (front end only) |
121 | | uWGI harakiri timeout | **120s** | ✓ | ✓ (front end only) |
122 | | uWGI worker reload timeout | **120s** | ✓ | ✓ (front end only) |
123 | | Google UA | | ✓ | ✓ (front end only) |
124 | | Google API Key | | ✓ | ✓ (front end only) |
125 | | Sentry tokens | | ✓ | ✓ (front end only) |
126 | | Debug | **False** | ✓ | |
127 | | Developer mode | **False** | ✓ | |
128 | | Staging mode | **False** | | ✓ (front end only) |
129 |
130 | 1) _HTTPS certificates must be installed on a Reverse Proxy.
131 | `kobo-install` can install one and use `Let's Encrypt` to generate certificates
132 | thanks
133 | to [nginx-certbot project](https://github.com/wmnnd/nginx-certbot "")_
134 |
135 | 2) _If AWS credentials are provided, backups are sent to configured bucket_
136 |
137 | 3) _Custom settings are provided by [PostgreSQL Configuration Tool API](https://github.com/sebastianwebber/pgconfig-api "")_
138 |
139 | 4) _Redis password is optional but **strongly** recommended_
140 |
141 | 5) _If AWS storage is selected, credentials must be provided if backups are activated_
142 |
143 | ## Requirements
144 |
145 | - Linux 5 / macOS 6
146 | - Python 3.10+
147 | - [Docker](https://www.docker.com/get-started "") 7
148 | - Available TCP Ports: 8
149 |
150 | 1. 80 NGINX
151 | 1. 443 NGINX (if you use kobo-install with LetsEncrypt proxy)
152 | 2. Additional ports when `expose ports` advanced option has been selected
153 | 1. 5432 PostgreSQL
154 | 3. 6379-6380 redis
155 | 4. 27017 MongoDB
156 |
157 | _**WARNING:**_
158 |
159 | - _If you use a firewall, be sure to open traffic publicly on NGINX port, otherwise kobo-install cannot work_
160 | - _By default, additional ports are not exposed except when using multi servers configuration. If you choose to expose them, **be sure to not expose them publicly** (e.g. use a firewall and allow traffic between front-end and back-end containers only. NGINX port still has to stay publicly opened though)._
161 |
162 | 5) _It has been tested with Ubuntu 20.04, 22.04 and 24.04_
163 |
164 | 6) _Docker on macOS is slow. First boot usually takes a while to be ready. You may have to answer `Yes` once or twice to question `Wait for another 600 seconds?` when prompted_
165 |
166 | 7) _Compose V1 is **NOT** supported anymore. It has reached its EOL from July 2023_
167 |
168 | 8) _These are defaults but can be customized with advanced options_
169 |
170 |
171 | ## Development
172 |
173 | ### React files: Hot Module Reload (HMR) by Webpack
174 | For frontend file changes to take effect, run watch in terminal and open/refresh http://kf.kobo.local to see your changes hot reloaded (don’t worry about first timeout error, it’s still building):
175 |
176 | ```shell
177 | ./run.py -cf run --rm --publish 3000:3000 kpi npm run watch && ./run.py -cf restart kpi
178 | ```
179 |
180 | The script creates a new docker container for frontend in `npm run watch` mode within the same docker network with the same (internal) port.
181 | Using the same port will overshadow the original kpi container’s ports and nginx will instantly serve the new container instead.
182 | Unfortunately the port overshadowing doesn’t nicely undo itself and a restart of `kpi` is required.
183 | Once the container exits (hit CTRL+C **once**), the container will automatically remove itself and initiate kpi restart which will take up to few minutes.
184 |
185 | It should as well handle dependency changes, maybe except for webpack itself.
186 |
187 | You can also [this gist](https://gist.github.com/jnm/dd323e0ff5be0d79e12e76bb9dfb7aed) to refresh front-end files without rebuilding the container.
188 |
189 | ### Tests
190 |
191 | Tests can be run with `tox`.
192 | Be sure it is installed before running the tests.
193 |
194 | ```
195 | $kobo-install> sudo apt install python3-pip
196 | $kobo-install> pip3 install tox
197 | $kobo-install> tox
198 | ```
199 | or
200 |
201 | ```
202 | $kobo-install> sudo apt install tox
203 | $kobo-install> tox
204 | ```
205 |
--------------------------------------------------------------------------------
/templates/kobo-docker/docker-compose.frontend.override.yml.tpl:
--------------------------------------------------------------------------------
1 | # For public, HTTPS servers.
2 |
3 | services:
4 | kpi:
5 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
6 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
7 | ${USE_KPI_DEV_MODE} volumes:
8 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
9 | environment:
10 | - UWSGI_WORKERS_COUNT=${UWSGI_WORKERS_MAX}
11 | - UWSGI_CHEAPER_WORKERS_COUNT=${UWSGI_WORKERS_START}
12 | - UWSGI_MAX_REQUESTS=${UWSGI_MAX_REQUESTS}
13 | - UWSGI_CHEAPER_RSS_LIMIT_SOFT=${UWSGI_SOFT_LIMIT}
14 | - UWSGI_HARAKIRI=${UWSGI_HARAKIRI}
15 | - UWSGI_WORKER_RELOAD_MERCY=${UWSGI_WORKER_RELOAD_MERCY}
16 | - WSGI=${WSGI}
17 | ${USE_CELERY} - SKIP_CELERY=True
18 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
19 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
20 | ${USE_NPM_FROM_HOST} - FRONTEND_DEV_MODE=host
21 | ${USE_EXTRA_HOSTS}extra_hosts:
22 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
23 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
24 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
25 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
26 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
27 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
28 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
29 | ${USE_BACKEND_NETWORK}networks:
30 | ${USE_BACKEND_NETWORK} kobo-be-network:
31 | ${USE_BACKEND_NETWORK} aliases:
32 | ${USE_BACKEND_NETWORK} - kpi
33 | ${USE_BACKEND_NETWORK} - kpi.docker.container
34 |
35 | worker:
36 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
37 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
38 | ${USE_KPI_DEV_MODE} volumes:
39 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
40 | environment:
41 | - WSGI=${WSGI}
42 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
43 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
44 | ${USE_EXTRA_HOSTS}extra_hosts:
45 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
46 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
47 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
48 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
49 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
50 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
51 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
52 | ${USE_BACKEND_NETWORK}networks:
53 | ${USE_BACKEND_NETWORK} kobo-be-network:
54 | ${USE_BACKEND_NETWORK} aliases:
55 | ${USE_BACKEND_NETWORK} - worker
56 | ${USE_BACKEND_NETWORK} - worker.docker.container
57 |
58 | worker_kobocat:
59 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
60 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
61 | ${USE_KPI_DEV_MODE} volumes:
62 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
63 | environment:
64 | - WSGI=${WSGI}
65 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
66 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
67 | ${USE_EXTRA_HOSTS}extra_hosts:
68 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
69 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
70 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
71 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
72 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
73 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
74 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
75 | ${USE_BACKEND_NETWORK}networks:
76 | ${USE_BACKEND_NETWORK} kobo-be-network:
77 | ${USE_BACKEND_NETWORK} aliases:
78 | ${USE_BACKEND_NETWORK} - worker_kobocat
79 | ${USE_BACKEND_NETWORK} - worker_kobocat.docker.container
80 |
81 | worker_low_priority:
82 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
83 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
84 | ${USE_KPI_DEV_MODE} volumes:
85 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
86 | environment:
87 | - WSGI=${WSGI}
88 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
89 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
90 | ${USE_EXTRA_HOSTS}extra_hosts:
91 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
92 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
93 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
94 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
95 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
96 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
97 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
98 | ${USE_BACKEND_NETWORK}networks:
99 | ${USE_BACKEND_NETWORK} kobo-be-network:
100 | ${USE_BACKEND_NETWORK} aliases:
101 | ${USE_BACKEND_NETWORK} - worker_low_priority
102 | ${USE_BACKEND_NETWORK} - worker_low_priority.docker.container
103 |
104 | worker_long_running_tasks:
105 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
106 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
107 | ${USE_KPI_DEV_MODE} volumes:
108 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
109 | environment:
110 | - WSGI=${WSGI}
111 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
112 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
113 | ${USE_EXTRA_HOSTS}extra_hosts:
114 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
115 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
116 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
117 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
118 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
119 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
120 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
121 | ${USE_BACKEND_NETWORK}networks:
122 | ${USE_BACKEND_NETWORK} kobo-be-network:
123 | ${USE_BACKEND_NETWORK} aliases:
124 | ${USE_BACKEND_NETWORK} - worker_long_running_tasks
125 | ${USE_BACKEND_NETWORK} - worker_long_running_tasks.docker.container
126 |
127 | beat:
128 | ${USE_KPI_DEV_MODE} build: ${KPI_PATH}
129 | ${USE_KPI_DEV_MODE} image: kpi:dev.${KPI_DEV_BUILD_ID}
130 | ${USE_KPI_DEV_MODE} volumes:
131 | ${USE_KPI_DEV_MODE} - ${KPI_PATH}:/srv/src/kpi
132 | environment:
133 | - WSGI=${WSGI}
134 | ${USE_DEV_MODE} - DJANGO_SETTINGS_MODULE=kobo.settings.dev
135 | ${USE_HTTPS} - SECURE_PROXY_SSL_HEADER=HTTP_X_FORWARDED_PROTO,https
136 | ${USE_EXTRA_HOSTS}extra_hosts:
137 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
138 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
139 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
140 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
141 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
142 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
143 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
144 | ${USE_BACKEND_NETWORK}networks:
145 | ${USE_BACKEND_NETWORK} kobo-be-network:
146 | ${USE_BACKEND_NETWORK} aliases:
147 | ${USE_BACKEND_NETWORK} - beat
148 | ${USE_BACKEND_NETWORK} - beat.docker.container
149 |
150 | nginx:
151 | environment:
152 | - NGINX_PUBLIC_PORT=${NGINX_PUBLIC_PORT}
153 | - UWSGI_PASS_TIMEOUT=${UWSGI_PASS_TIMEOUT}
154 | - WSGI=${WSGI}
155 | ${USE_LETSENSCRYPT}ports:
156 | ${USE_LETSENSCRYPT} - ${NGINX_EXPOSED_PORT}:80
157 | ${USE_EXTRA_HOSTS}extra_hosts:
158 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
159 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
160 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
161 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
162 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
163 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
164 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
165 | networks:
166 | kobo-fe-network:
167 | aliases:
168 | - ${KOBOFORM_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME}
169 | - ${KOBOCAT_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME}
170 | - ${ENKETO_SUBDOMAIN}.${INTERNAL_DOMAIN_NAME}
171 |
172 | enketo_express:
173 | # `DUMMY_ENV` is only there to avoid extra complex condition to override
174 | # `enketo_express` section or not. It allows to always this section whatever
175 | # `USE_EXTRA_HOSTS` and `USE_BACKEND_NETWORK` values are.
176 | environment:
177 | - DUMMY_ENV=True
178 | ${USE_EXTRA_HOSTS}extra_hosts:
179 | ${USE_FAKE_DNS} - ${KOBOFORM_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
180 | ${USE_FAKE_DNS} - ${KOBOCAT_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
181 | ${USE_FAKE_DNS} - ${ENKETO_SUBDOMAIN}.${PUBLIC_DOMAIN_NAME}:${LOCAL_INTERFACE_IP}
182 | ${ADD_BACKEND_EXTRA_HOSTS} - postgres.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
183 | ${ADD_BACKEND_EXTRA_HOSTS} - mongo.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
184 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-main.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
185 | ${ADD_BACKEND_EXTRA_HOSTS} - redis-cache.${PRIVATE_DOMAIN_NAME}:${PRIMARY_BACKEND_IP}
186 | ${USE_BACKEND_NETWORK}networks:
187 | ${USE_BACKEND_NETWORK} kobo-be-network:
188 | ${USE_BACKEND_NETWORK} aliases:
189 | ${USE_BACKEND_NETWORK} - enketo_express
190 |
191 | ${USE_BACKEND_NETWORK}networks:
192 | ${USE_BACKEND_NETWORK} kobo-be-network:
193 | ${USE_BACKEND_NETWORK} name: ${DOCKER_NETWORK_BACKEND_PREFIX}_kobo-be-network
194 | ${USE_BACKEND_NETWORK} external: true
195 |
--------------------------------------------------------------------------------
/helpers/template.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import fnmatch
3 | import json
4 | import os
5 | import re
6 | import stat
7 | import sys
8 | from string import Template as PyTemplate
9 |
10 | from helpers.cli import CLI
11 | from helpers.config import Config
12 |
13 |
14 | class Template:
15 | UNIQUE_ID_FILE = '.uniqid'
16 |
17 | @classmethod
18 | def render(cls, config, force=False):
19 | """
20 | Write configuration files based on `config`
21 |
22 | Args:
23 | config (helpers.config.Config)
24 | force (bool)
25 | """
26 |
27 | dict_ = config.get_dict()
28 | template_variables = cls.__get_template_variables(config)
29 |
30 | environment_directory = config.get_env_files_path()
31 | unique_id = cls.__read_unique_id(environment_directory)
32 |
33 | if (
34 | not force and unique_id
35 | and str(dict_.get('unique_id', '')) != str(unique_id)
36 | ):
37 | message = (
38 | 'WARNING!\n\n'
39 | 'Existing environment files are detected. Files will be '
40 | 'overwritten.'
41 | )
42 | CLI.framed_print(message)
43 | response = CLI.yes_no_question(
44 | 'Do you want to continue?',
45 | default=False
46 | )
47 | if not response:
48 | sys.exit(0)
49 |
50 | cls.__write_unique_id(environment_directory, dict_['unique_id'])
51 |
52 | # Environment
53 | templates_path_parent = cls._get_templates_path_parent()
54 | templates_path = os.path.join(
55 | templates_path_parent, Config.ENV_FILES_DIR, ''
56 | )
57 | for root, dirnames, filenames in os.walk(templates_path):
58 | destination_directory = cls.__create_directory(
59 | environment_directory,
60 | root,
61 | templates_path
62 | )
63 | cls.__write_templates(
64 | template_variables, root, destination_directory, filenames
65 | )
66 |
67 | # kobo-docker
68 | templates_path = os.path.join(templates_path_parent, 'kobo-docker')
69 | for root, dirnames, filenames in os.walk(templates_path):
70 | destination_directory = cls.__create_directory(dict_['kobodocker_path'])
71 | cls.__write_templates(
72 | template_variables, root, destination_directory, filenames
73 | )
74 |
75 | # nginx-certbox
76 | if config.use_letsencrypt:
77 | templates_path = os.path.join(
78 | templates_path_parent, Config.LETSENCRYPT_DOCKER_DIR, ''
79 | )
80 | for root, dirnames, filenames in os.walk(templates_path):
81 | destination_directory = cls.__create_directory(
82 | config.get_letsencrypt_repo_path(),
83 | root,
84 | templates_path)
85 | cls.__write_templates(template_variables,
86 | root,
87 | destination_directory,
88 | filenames)
89 |
90 | @classmethod
91 | def render_maintenance(cls, config):
92 |
93 | dict_ = config.get_dict()
94 | template_variables = cls.__get_template_variables(config)
95 |
96 | base_dir = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
97 | templates_path_parent = os.path.join(base_dir, 'templates')
98 |
99 | # kobo-docker
100 | templates_path = os.path.join(templates_path_parent, 'kobo-docker')
101 | for root, dirnames, filenames in os.walk(templates_path):
102 | filenames = [filename
103 | for filename in filenames if 'maintenance' in filename]
104 | destination_directory = dict_['kobodocker_path']
105 | cls.__write_templates(template_variables,
106 | root,
107 | destination_directory,
108 | filenames)
109 |
110 | @classmethod
111 | def __create_directory(cls, template_root_directory, path='', base_dir=''):
112 |
113 | # Handle case when path is root and equals ''.
114 | path = os.path.join(path, '')
115 |
116 | destination_directory = os.path.realpath(os.path.join(
117 | template_root_directory,
118 | path.replace(base_dir, '')
119 | ))
120 |
121 | if not os.path.isdir(destination_directory):
122 | try:
123 | os.makedirs(destination_directory)
124 | except OSError:
125 | CLI.colored_print(
126 | f'Cannot create {destination_directory}. '
127 | 'Please verify permissions!',
128 | CLI.COLOR_ERROR)
129 | sys.exit(1)
130 |
131 | return destination_directory
132 |
133 | @staticmethod
134 | def __get_template_variables(config):
135 | """
136 | Write configuration files based on `config`
137 |
138 | Args:
139 | config (helpers.config.Config)
140 | """
141 |
142 | dict_ = config.get_dict()
143 |
144 | def _get_value(property_, true_value='', false_value='#',
145 | comparison_value=True):
146 | return (
147 | true_value
148 | if dict_[property_] == comparison_value
149 | else false_value
150 | )
151 |
152 | if config.proxy:
153 | nginx_port = dict_['nginx_proxy_port']
154 | else:
155 | nginx_port = dict_['exposed_nginx_docker_port']
156 |
157 | return {
158 | 'PUBLIC_REQUEST_SCHEME': _get_value('https', 'https', 'http'),
159 | 'USE_HTTPS': _get_value('https'),
160 | 'USE_AWS': _get_value('use_aws'),
161 | 'AWS_ACCESS_KEY_ID': dict_['aws_access_key'],
162 | 'AWS_SECRET_ACCESS_KEY': dict_['aws_secret_key'],
163 | 'AWS_BUCKET_NAME': dict_['aws_bucket_name'],
164 | 'AWS_S3_REGION_NAME': dict_['aws_s3_region_name'],
165 | 'GOOGLE_UA': dict_['google_ua'],
166 | 'GOOGLE_API_KEY': dict_['google_api_key'],
167 | 'INTERNAL_DOMAIN_NAME': dict_['internal_domain_name'],
168 | 'PRIVATE_DOMAIN_NAME': dict_['private_domain_name'],
169 | 'PUBLIC_DOMAIN_NAME': dict_['public_domain_name'],
170 | 'KOBOFORM_SUBDOMAIN': dict_['kpi_subdomain'],
171 | 'KOBOCAT_SUBDOMAIN': dict_['kc_subdomain'],
172 | 'ENKETO_SUBDOMAIN': dict_['ee_subdomain'],
173 | 'KOBO_SUPERUSER_USERNAME': dict_['super_user_username'],
174 | 'KOBO_SUPERUSER_PASSWORD': dict_['super_user_password'],
175 | 'ENKETO_API_KEY': dict_['enketo_api_token'],
176 | 'DJANGO_SECRET_KEY': dict_['django_secret_key'],
177 | 'DJANGO_SESSION_COOKIE_AGE': dict_['django_session_cookie_age'],
178 | 'ENKETO_ENCRYPTION_KEY': dict_['enketo_encryption_key'],
179 | 'ENKETO_LESS_SECURE_ENCRYPTION_KEY': dict_[
180 | 'enketo_less_secure_encryption_key'
181 | ],
182 | 'KOBOCAT_RAVEN_DSN': dict_['kobocat_raven'],
183 | 'KPI_RAVEN_DSN': dict_['kpi_raven'],
184 | 'KPI_RAVEN_JS_DSN': dict_['kpi_raven_js'],
185 | 'KC_POSTGRES_DB': dict_['kc_postgres_db'],
186 | 'KPI_POSTGRES_DB': dict_['kpi_postgres_db'],
187 | 'POSTGRES_USER': dict_['postgres_user'],
188 | 'POSTGRES_PASSWORD': dict_['postgres_password'],
189 | 'DEBUG': dict_['debug'],
190 | 'SMTP_HOST': dict_['smtp_host'],
191 | 'SMTP_PORT': dict_['smtp_port'],
192 | 'SMTP_USER': dict_['smtp_user'],
193 | 'SMTP_PASSWORD': dict_['smtp_password'],
194 | 'SMTP_USE_TLS': dict_['smtp_use_tls'],
195 | 'DEFAULT_FROM_EMAIL': dict_['default_from_email'],
196 | 'PRIMARY_BACKEND_IP': dict_['primary_backend_ip'],
197 | 'LOCAL_INTERFACE_IP': dict_['local_interface_ip'],
198 | 'KPI_PATH': dict_['kpi_path'],
199 | 'USE_KPI_DEV_MODE': _get_value(
200 | 'kpi_path', true_value='#', false_value='', comparison_value=''
201 | ),
202 | 'KPI_DEV_BUILD_ID': dict_['kpi_dev_build_id'],
203 | 'NGINX_PUBLIC_PORT': (
204 | ''
205 | if dict_['exposed_nginx_docker_port'] == '80'
206 | else f":{dict_['exposed_nginx_docker_port']}"
207 | ),
208 | 'NGINX_EXPOSED_PORT': nginx_port,
209 | 'UWSGI_WORKERS_MAX': dict_['uwsgi_workers_max'],
210 | # Deactivate cheaper algorithm if defaults are 1 worker to start and
211 | # 2 maximum.
212 | 'UWSGI_WORKERS_START': (
213 | ''
214 | if dict_['uwsgi_workers_start'] == '1'
215 | and dict_['uwsgi_workers_max'] == '2'
216 | else dict_['uwsgi_workers_start']
217 | ),
218 | 'UWSGI_MAX_REQUESTS': dict_['uwsgi_max_requests'],
219 | 'UWSGI_SOFT_LIMIT': int(dict_['uwsgi_soft_limit']) * 1024 * 1024,
220 | 'UWSGI_HARAKIRI': dict_['uwsgi_harakiri'],
221 | 'UWSGI_WORKER_RELOAD_MERCY': dict_['uwsgi_worker_reload_mercy'],
222 | 'UWSGI_PASS_TIMEOUT': int(dict_['uwsgi_harakiri']) + 10,
223 | 'POSTGRES_REPLICATION_PASSWORD': dict_[
224 | 'postgres_replication_password'
225 | ],
226 | 'WSGI': 'runserver_plus' if config.dev_mode else 'uWSGI',
227 | 'USE_X_FORWARDED_HOST': '' if config.dev_mode else '#',
228 | 'OVERRIDE_POSTGRES_SETTINGS': _get_value('postgres_settings'),
229 | 'POSTGRES_APP_PROFILE': dict_['postgres_profile'],
230 | 'POSTGRES_RAM': dict_['postgres_ram'],
231 | 'POSTGRES_SETTINGS': dict_['postgres_settings_content'],
232 | 'POSTGRES_PORT': dict_['postgresql_port'],
233 | 'MONGO_PORT': dict_['mongo_port'],
234 | 'REDIS_MAIN_PORT': dict_['redis_main_port'],
235 | 'REDIS_CACHE_PORT': dict_['redis_cache_port'],
236 | 'REDIS_CACHE_MAX_MEMORY': dict_['redis_cache_max_memory'],
237 | 'USE_BACKUP': '' if dict_['use_backup'] else '#',
238 | 'USE_AWS_BACKUP': (
239 | ''
240 | if (
241 | config.aws
242 | and dict_['aws_backup_bucket_name'] != ''
243 | and dict_['use_backup']
244 | )
245 | else '#'
246 | ),
247 | 'USE_MEDIA_BACKUP': (
248 | '' if (not config.aws and dict_['use_backup']) else '#'
249 | ),
250 | 'KOBOCAT_MEDIA_BACKUP_SCHEDULE': dict_[
251 | 'kobocat_media_backup_schedule'
252 | ],
253 | 'MONGO_BACKUP_SCHEDULE': dict_['mongo_backup_schedule'],
254 | 'POSTGRES_BACKUP_SCHEDULE': dict_['postgres_backup_schedule'],
255 | 'REDIS_BACKUP_SCHEDULE': dict_['redis_backup_schedule'],
256 | 'AWS_BACKUP_BUCKET_NAME': dict_['aws_backup_bucket_name'],
257 | 'AWS_BACKUP_YEARLY_RETENTION': dict_['aws_backup_yearly_retention'],
258 | 'AWS_BACKUP_MONTHLY_RETENTION': dict_[
259 | 'aws_backup_monthly_retention'
260 | ],
261 | 'AWS_BACKUP_WEEKLY_RETENTION': dict_['aws_backup_weekly_retention'],
262 | 'AWS_BACKUP_DAILY_RETENTION': dict_['aws_backup_daily_retention'],
263 | 'AWS_MONGO_BACKUP_MINIMUM_SIZE': dict_[
264 | 'aws_mongo_backup_minimum_size'
265 | ],
266 | 'AWS_POSTGRES_BACKUP_MINIMUM_SIZE': dict_[
267 | 'aws_postgres_backup_minimum_size'
268 | ],
269 | 'AWS_REDIS_BACKUP_MINIMUM_SIZE': dict_[
270 | 'aws_redis_backup_minimum_size'
271 | ],
272 | 'AWS_BACKUP_UPLOAD_CHUNK_SIZE': dict_[
273 | 'aws_backup_upload_chunk_size'
274 | ],
275 | 'AWS_BACKUP_BUCKET_DELETION_RULE_ENABLED': _get_value(
276 | 'aws_backup_bucket_deletion_rule_enabled', 'True', 'False'
277 | ),
278 | 'LETSENCRYPT_EMAIL': dict_['letsencrypt_email'],
279 | 'MAINTENANCE_ETA': dict_['maintenance_eta'],
280 | 'MAINTENANCE_DATE_ISO': dict_['maintenance_date_iso'],
281 | 'MAINTENANCE_DATE_STR': dict_['maintenance_date_str'],
282 | 'MAINTENANCE_EMAIL': dict_['maintenance_email'],
283 | 'USE_NPM_FROM_HOST': (
284 | '' if (config.dev_mode and not dict_['npm_container']) else '#'
285 | ),
286 | 'DOCKER_NETWORK_BACKEND_PREFIX': config.get_prefix('backend'),
287 | 'DOCKER_NETWORK_FRONTEND_PREFIX': config.get_prefix('frontend'),
288 | 'USE_BACKEND_NETWORK': _get_value(
289 | 'expose_backend_ports', comparison_value=False
290 | ),
291 | 'EXPOSE_BACKEND_PORTS': _get_value('expose_backend_ports'),
292 | 'USE_FAKE_DNS': _get_value('local_installation'),
293 | 'ADD_BACKEND_EXTRA_HOSTS': (
294 | ''
295 | if (config.expose_backend_ports and not config.use_private_dns)
296 | else '#'
297 | ),
298 | 'USE_EXTRA_HOSTS': (
299 | ''
300 | if (
301 | config.local_install
302 | or config.expose_backend_ports
303 | and not config.use_private_dns
304 | )
305 | else '#'
306 | ),
307 | 'MONGO_ROOT_USERNAME': dict_['mongo_root_username'],
308 | 'MONGO_ROOT_PASSWORD': dict_['mongo_root_password'],
309 | 'MONGO_USER_USERNAME': dict_['mongo_user_username'],
310 | 'MONGO_USER_PASSWORD': dict_['mongo_user_password'],
311 | 'REDIS_PASSWORD': dict_['redis_password'],
312 | 'REDIS_PASSWORD_JS_ENCODED': json.dumps(dict_['redis_password']),
313 | 'USE_DEV_MODE': _get_value('dev_mode'),
314 | 'USE_CELERY': _get_value('use_celery', comparison_value=False),
315 | 'ENKETO_ALLOW_PRIVATE_IP_ADDRESS': _get_value(
316 | 'local_installation', true_value='true', false_value='false'
317 | ),
318 | 'USE_REDIS_CACHE_MAX_MEMORY': _get_value(
319 | 'redis_cache_max_memory',
320 | true_value='#',
321 | false_value='',
322 | comparison_value='',
323 | ),
324 | 'USE_LETSENSCRYPT': '#' if config.use_letsencrypt else '',
325 | 'DOCKER_COMPOSE_CMD': 'docker',
326 | # Keep leading space in front of suffix if any
327 | 'DOCKER_COMPOSE_SUFFIX': ' compose'
328 | }
329 |
330 | @staticmethod
331 | def _get_templates_path_parent():
332 | base_dir = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
333 | templates_path_parent = os.path.join(base_dir, 'templates')
334 | return templates_path_parent
335 |
336 | @staticmethod
337 | def __read_unique_id(destination_directory):
338 | """
339 | Reads unique id from file `Template.UNIQUE_ID_FILE`
340 | :return: str
341 | """
342 | unique_id = ''
343 |
344 | if os.path.isdir(destination_directory):
345 | try:
346 | unique_id_file = os.path.join(destination_directory,
347 | Template.UNIQUE_ID_FILE)
348 | with open(unique_id_file, 'r') as f:
349 | unique_id = f.read().strip()
350 | except IOError:
351 | pass
352 | else:
353 | unique_id = None
354 |
355 | return unique_id
356 |
357 | @staticmethod
358 | def __write_templates(
359 | template_variables_, root_, destination_directory_, filenames_
360 | ):
361 | for filename in fnmatch.filter(filenames_, '*.tpl'):
362 | with open(os.path.join(root_, filename), 'r') as template:
363 | t = ExtendedPyTemplate(template.read(), template_variables_)
364 | with open(
365 | os.path.join(destination_directory_, filename[:-4]), 'w'
366 | ) as f:
367 | f.write(t.substitute(template_variables_))
368 |
369 | @classmethod
370 | def __write_unique_id(cls, destination_directory, unique_id):
371 | try:
372 | unique_id_file = os.path.join(destination_directory,
373 | Template.UNIQUE_ID_FILE)
374 | # Ensure kobo-deployment is created.
375 | cls.__create_directory(destination_directory)
376 |
377 | with open(unique_id_file, 'w') as f:
378 | f.write(str(unique_id))
379 |
380 | os.chmod(unique_id_file, stat.S_IWRITE | stat.S_IREAD)
381 |
382 | except (IOError, OSError):
383 | CLI.colored_print('Could not write unique_id file', CLI.COLOR_ERROR)
384 | return False
385 |
386 | return True
387 |
388 |
389 | class ExtendedPyTemplate(PyTemplate):
390 | """
391 | Basic class to add conditional substitution to `string.Template`
392 |
393 | Usage example:
394 | ```
395 | {
396 | 'host': 'redis-cache.kobo.local',
397 | 'port': '6379'{% if REDIS_PASSWORD %},{% endif REDIS_PASSWORD %}
398 | {% if REDIS_PASSWORD %}
399 | 'password': ${REDIS_PASSWORD}
400 | {% endif REDIS_PASSWORD %}
401 | }
402 | ```
403 |
404 | If `REDIS_PASSWORD` equals '123456', output would be:
405 | ```
406 | {
407 | 'host': 'redis-cache.kobo.local',
408 | 'port': '6379',
409 | 'password': '123456'
410 | }
411 | ```
412 |
413 | If `REDIS_PASSWORD` equals '' (or `False` or `None`), output would be:
414 | ```
415 | {
416 | 'host': 'redis-cache.kobo.local',
417 | 'port': '6379'
418 |
419 | }
420 | ```
421 |
422 | """
423 | IF_PATTERN = '{{% if {} %}}'
424 | ENDIF_PATTERN = '{{% endif {} %}}'
425 |
426 | def __init__(self, template, template_variables_):
427 | for key, value in template_variables_.items():
428 | if self.IF_PATTERN.format(key) in template:
429 | if value:
430 | if_pattern = r'{}\s*'.format(self.IF_PATTERN.format(key))
431 | endif_pattern = r'\s*{}'.format(
432 | self.ENDIF_PATTERN.format(key))
433 | template = re.sub(if_pattern, '', template)
434 | template = re.sub(endif_pattern, '', template)
435 | else:
436 | pattern = r'{}(.|\s)*?{}'.format(
437 | self.IF_PATTERN.format(key),
438 | self.ENDIF_PATTERN.format(key))
439 | template = re.sub(pattern, '', template)
440 | super(ExtendedPyTemplate, self).__init__(template)
441 |
--------------------------------------------------------------------------------
/helpers/command.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import os
3 | import sys
4 | import time
5 | import subprocess
6 |
7 | from helpers.cli import CLI
8 | from helpers.config import Config
9 | from helpers.network import Network
10 | from helpers.template import Template
11 | from helpers.upgrading import Upgrading
12 | from helpers.utils import run_docker_compose
13 |
14 |
15 | class Command:
16 |
17 | @staticmethod
18 | def help():
19 | output = [
20 | 'Usage: python3 run.py [options]',
21 | '',
22 | ' Options:',
23 | ' -i, --info',
24 | ' Show KoboToolbox Url and super user credentials',
25 | ' -l, --logs',
26 | ' Display docker logs',
27 | ' -b, --build',
28 | ' Build django (kpi) container (only on dev/staging mode)',
29 | ' -s, --setup',
30 | ' Prompt questions to (re)write configuration files',
31 | ' -S, --stop',
32 | ' Stop KoboToolbox',
33 | ' -u, --update, --upgrade [branch or tag]',
34 | ' Update KoboToolbox',
35 | ' -cf, --compose-frontend [docker-compose arguments]',
36 | ' Run a docker-compose command in the front-end '
37 | 'environment',
38 | ' -cb, --compose-backend [docker-compose arguments]',
39 | ' Run a docker-compose command in the back-end '
40 | 'environment',
41 | ' -m, --maintenance',
42 | ' Activate maintenance mode. All traffic is '
43 | 'redirected to maintenance page',
44 | ' -sm, --stop-maintenance',
45 | ' Stop maintenance mode',
46 | ' -v, --version',
47 | ' Display current version',
48 | ''
49 | ]
50 | print('\n'.join(output))
51 |
52 | @classmethod
53 | def build(cls):
54 | """
55 | Builds kpi image with `--no-caches` option
56 | """
57 | config = Config()
58 | dict_ = config.get_dict()
59 |
60 | if config.dev_mode or config.staging_mode:
61 |
62 | prefix = config.get_prefix('frontend')
63 | timestamp = int(time.time())
64 | dict_['kpi_dev_build_id'] = f'{prefix}{timestamp}'
65 | config.write_config()
66 | Template.render(config)
67 | frontend_command = run_docker_compose(dict_, [
68 | '-f', 'docker-compose.frontend.yml',
69 | '-f', 'docker-compose.frontend.override.yml',
70 | '-p', config.get_prefix('frontend'),
71 | 'build', '--force-rm', '--no-cache', 'kpi'
72 | ])
73 | CLI.run_command(frontend_command, dict_['kobodocker_path'])
74 |
75 | @classmethod
76 | def compose_frontend(cls, args):
77 | config = Config()
78 | dict_ = config.get_dict()
79 | command = run_docker_compose(dict_, [
80 | '-f', 'docker-compose.frontend.yml',
81 | '-f', 'docker-compose.frontend.override.yml',
82 | '-p', config.get_prefix('frontend')
83 | ])
84 |
85 | cls.__validate_custom_yml(config, command)
86 | command.extend(args)
87 | subprocess.call(command, cwd=dict_['kobodocker_path'])
88 |
89 | @classmethod
90 | def compose_backend(cls, args):
91 | config = Config()
92 | dict_ = config.get_dict()
93 | command = run_docker_compose(dict_, [
94 | '-f', f'docker-compose.backend.yml',
95 | '-f', f'docker-compose.backend.override.yml',
96 | '-p', config.get_prefix('backend')
97 | ])
98 | cls.__validate_custom_yml(config, command)
99 | command.extend(args)
100 | subprocess.call(command, cwd=dict_['kobodocker_path'])
101 |
102 | @classmethod
103 | def info(cls, timeout=600):
104 | config = Config()
105 | dict_ = config.get_dict()
106 |
107 | nginx_port = dict_['exposed_nginx_docker_port']
108 |
109 | main_url = '{}://{}.{}{}'.format(
110 | 'https' if dict_['https'] else 'http',
111 | dict_['kpi_subdomain'],
112 | dict_['public_domain_name'],
113 | ':{}'.format(nginx_port) if (
114 | nginx_port and
115 | str(nginx_port) != Config.DEFAULT_NGINX_PORT
116 | ) else ''
117 | )
118 |
119 | stop = False
120 | start = int(time.time())
121 | success = False
122 | hostname = f"{dict_['kpi_subdomain']}.{dict_['public_domain_name']}"
123 | https = dict_['https']
124 | nginx_port = int(Config.DEFAULT_NGINX_HTTPS_PORT) \
125 | if https else int(dict_['exposed_nginx_docker_port'])
126 | already_retried = False
127 | while not stop:
128 | if Network.status_check(hostname,
129 | '/service_health/',
130 | nginx_port, https) == Network.STATUS_OK_200:
131 | stop = True
132 | success = True
133 | elif int(time.time()) - start >= timeout:
134 | if timeout > 0:
135 | CLI.colored_print(
136 | '\n`KoboToolbox` has not started yet. '
137 | 'This can happen with low CPU/RAM computers.\n',
138 | CLI.COLOR_INFO)
139 | question = f'Wait for another {timeout} seconds?'
140 | response = CLI.yes_no_question(question)
141 | if response:
142 | start = int(time.time())
143 | continue
144 | else:
145 | if not already_retried:
146 | already_retried = True
147 | CLI.colored_print(
148 | '\nSometimes front-end containers cannot '
149 | 'communicate with back-end containers.\n'
150 | 'Restarting the front-end containers usually '
151 | 'fixes it.\n', CLI.COLOR_INFO)
152 | question = 'Would you like to try?'
153 | response = CLI.yes_no_question(question)
154 | if response:
155 | start = int(time.time())
156 | cls.restart_frontend()
157 | continue
158 | stop = True
159 | else:
160 | sys.stdout.write('.')
161 | sys.stdout.flush()
162 | time.sleep(10)
163 |
164 | # Create a new line
165 | print('')
166 |
167 | if success:
168 | username = dict_['super_user_username']
169 | password = dict_['super_user_password']
170 |
171 | message = (
172 | 'Ready\n'
173 | f'URL: {main_url}\n'
174 | f'User: {username}\n'
175 | f'Password: {password}'
176 | )
177 | CLI.framed_print(message,
178 | color=CLI.COLOR_SUCCESS)
179 |
180 | else:
181 | message = (
182 | 'KoboToolbox could not start!\n'
183 | 'Please try `python3 run.py --logs` to see the logs.'
184 | )
185 | CLI.framed_print(message, color=CLI.COLOR_ERROR)
186 |
187 | return success
188 |
189 | @classmethod
190 | def logs(cls):
191 | config = Config()
192 | dict_ = config.get_dict()
193 |
194 | if config.backend:
195 | backend_command = run_docker_compose(dict_, [
196 | '-f', f'docker-compose.backend.yml',
197 | '-f', f'docker-compose.backend.override.yml',
198 | '-p', config.get_prefix('backend'),
199 | 'logs', '-f'
200 | ])
201 | cls.__validate_custom_yml(config, backend_command)
202 | CLI.run_command(backend_command, dict_['kobodocker_path'], True)
203 |
204 | if config.frontend:
205 | frontend_command = run_docker_compose(dict_, [
206 | '-f', 'docker-compose.frontend.yml',
207 | '-f', 'docker-compose.frontend.override.yml',
208 | '-p', config.get_prefix('frontend'),
209 | 'logs', '-f',
210 | ])
211 |
212 | cls.__validate_custom_yml(config, frontend_command)
213 | CLI.run_command(frontend_command, dict_['kobodocker_path'], True)
214 |
215 | @classmethod
216 | def configure_maintenance(cls):
217 | config = Config()
218 | dict_ = config.get_dict()
219 |
220 | if not config.multi_servers or config.frontend:
221 |
222 | config.maintenance()
223 | Template.render_maintenance(config)
224 | dict_['maintenance_enabled'] = True
225 | config.write_config()
226 | cls.stop_nginx()
227 | cls.start_maintenance()
228 |
229 | @classmethod
230 | def stop_nginx(cls):
231 | config = Config()
232 | dict_ = config.get_dict()
233 |
234 | nginx_stop_command = run_docker_compose(dict_, [
235 | '-f', 'docker-compose.frontend.yml',
236 | '-f', 'docker-compose.frontend.override.yml',
237 | '-p', config.get_prefix('frontend'),
238 | 'stop', 'nginx',
239 | ])
240 |
241 | cls.__validate_custom_yml(config, nginx_stop_command)
242 | CLI.run_command(nginx_stop_command, dict_['kobodocker_path'])
243 |
244 | @classmethod
245 | def start_maintenance(cls):
246 | config = Config()
247 | dict_ = config.get_dict()
248 |
249 | frontend_command = run_docker_compose(dict_, [
250 | '-f', 'docker-compose.maintenance.yml',
251 | '-f', 'docker-compose.maintenance.override.yml',
252 | '-p', config.get_prefix('maintenance'),
253 | 'up', '-d',
254 | ])
255 |
256 | CLI.run_command(frontend_command, dict_['kobodocker_path'])
257 | CLI.colored_print('Maintenance mode has been started',
258 | CLI.COLOR_SUCCESS)
259 |
260 | @classmethod
261 | def restart_frontend(cls):
262 | cls.start(frontend_only=True)
263 |
264 | @classmethod
265 | def start(cls, frontend_only=False, force_setup=False):
266 | config = Config()
267 | dict_ = config.get_dict()
268 |
269 | cls.stop(output=False, frontend_only=frontend_only)
270 | if frontend_only:
271 | CLI.colored_print('Launching front-end containers', CLI.COLOR_INFO)
272 | else:
273 | CLI.colored_print('Launching environment', CLI.COLOR_INFO)
274 |
275 | # Test if ports are available
276 | ports = []
277 | if config.proxy:
278 | nginx_port = int(dict_['nginx_proxy_port'])
279 | else:
280 | nginx_port = int(dict_['exposed_nginx_docker_port'])
281 |
282 | if frontend_only or config.frontend or not config.multi_servers:
283 | ports.append(nginx_port)
284 |
285 | if not frontend_only and config.expose_backend_ports and config.backend:
286 | ports.append(dict_['postgresql_port'])
287 | ports.append(dict_['mongo_port'])
288 | ports.append(dict_['redis_main_port'])
289 | ports.append(dict_['redis_cache_port'])
290 |
291 | for port in ports:
292 | if Network.is_port_open(port):
293 | CLI.colored_print(f'Port {port} is already open. '
294 | 'KoboToolbox cannot start',
295 | CLI.COLOR_ERROR)
296 | sys.exit(1)
297 |
298 | # Start the back-end containers
299 | if not frontend_only and config.backend:
300 |
301 | backend_command = run_docker_compose(dict_, [
302 | '-f', f'docker-compose.backend.yml',
303 | '-f', f'docker-compose.backend.override.yml',
304 | '-p', config.get_prefix('backend'),
305 | 'up', '-d'
306 | ])
307 |
308 | cls.__validate_custom_yml(config, backend_command)
309 | CLI.run_command(backend_command, dict_['kobodocker_path'])
310 |
311 | # Start the front-end containers
312 | if config.frontend:
313 |
314 | # If this was previously a shared-database setup, migrate to
315 | # separate databases for KPI and KoboCAT
316 | Upgrading.migrate_single_to_two_databases(config)
317 |
318 | frontend_command = run_docker_compose(dict_, [
319 | '-f', 'docker-compose.frontend.yml',
320 | '-f', 'docker-compose.frontend.override.yml',
321 | '-p', config.get_prefix('frontend'),
322 | 'up', '-d',
323 | ])
324 |
325 | if dict_['maintenance_enabled']:
326 | cls.start_maintenance()
327 | # Start all front-end services except the non-maintenance NGINX
328 | frontend_command.extend([
329 | s for s in config.get_service_names() if s != 'nginx'
330 | ])
331 |
332 | cls.__validate_custom_yml(config, frontend_command)
333 | CLI.run_command(frontend_command, dict_['kobodocker_path'])
334 |
335 | # Start reverse proxy if user uses it.
336 | if config.use_letsencrypt:
337 | if force_setup:
338 | # Let's Encrypt NGINX container needs kobo-docker NGINX
339 | # container to be started first
340 | config.init_letsencrypt()
341 |
342 | proxy_command = run_docker_compose(dict_, ['up', '-d'])
343 | CLI.run_command(
344 | proxy_command, config.get_letsencrypt_repo_path()
345 | )
346 |
347 | if dict_['maintenance_enabled']:
348 | CLI.colored_print(
349 | 'Maintenance mode is enabled. To resume '
350 | 'normal operation, use `--stop-maintenance`',
351 | CLI.COLOR_INFO,
352 | )
353 | elif not frontend_only:
354 | if not config.multi_servers or config.frontend:
355 | CLI.colored_print('Waiting for environment to be ready. '
356 | 'It can take a few minutes.', CLI.COLOR_INFO)
357 | cls.info()
358 | else:
359 | CLI.colored_print(
360 | (f'Back-end server is starting up '
361 | 'and should be up & running soon!\nPlease look at docker '
362 | 'logs for further information: '
363 | '`python3 run.py -cb logs -f`'),
364 | CLI.COLOR_WARNING)
365 |
366 | @classmethod
367 | def stop(cls, output=True, frontend_only=False):
368 | """
369 | Stop containers.
370 | Because containers share the same network, containers must be stopped
371 | first, then "down-ed" to remove any attached internal networks.
372 | The order must respected to avoid removing networks with active endpoints.
373 | """
374 | config = Config()
375 |
376 | if not config.multi_servers or config.frontend:
377 | # Stop maintenance container in case it's up&running
378 | cls.stop_containers('maintenance')
379 |
380 | # Stop reverse proxy if user uses it.
381 | if config.use_letsencrypt:
382 | cls.stop_containers('certbot')
383 |
384 | # Stop down front-end containers
385 | cls.stop_containers('frontend')
386 |
387 | # Clean maintenance services
388 | cls.stop_containers('maintenance', down=True)
389 |
390 | # Clean certbot services if user uses it.
391 | if config.use_letsencrypt:
392 | cls.stop_containers('certbot', down=True)
393 |
394 | if not frontend_only and config.backend:
395 | cls.stop_containers('backend', down=True)
396 |
397 | # Clean front-end services
398 | if not config.multi_servers or config.frontend:
399 | cls.stop_containers('frontend', down=True)
400 |
401 | if output:
402 | CLI.colored_print('KoboToolbox has been stopped', CLI.COLOR_SUCCESS)
403 |
404 | @classmethod
405 | def stop_containers(cls, group: str, down: bool = False):
406 |
407 | config = Config()
408 | dict_ = config.get_dict()
409 |
410 | if group not in ['frontend', 'backend', 'certbot', 'maintenance']:
411 | raise Exception('Unknown group')
412 |
413 | group_docker_maps = {
414 | 'frontend': {
415 | 'options': [
416 | '-f', 'docker-compose.frontend.yml',
417 | '-f', 'docker-compose.frontend.override.yml',
418 | '-p', config.get_prefix('frontend'),
419 | ],
420 | 'custom_yml': True,
421 | },
422 | 'backend': {
423 | 'options': [
424 | '-f', f'docker-compose.backend.yml',
425 | '-f', f'docker-compose.backend.override.yml',
426 | '-p', config.get_prefix('backend'),
427 | ],
428 | 'custom_yml': True,
429 | },
430 | 'certbot': {
431 | 'options': [],
432 | 'custom_yml': False,
433 | 'path': config.get_letsencrypt_repo_path(),
434 | },
435 | 'maintenance': {
436 | 'options': [
437 | '-f', 'docker-compose.maintenance.yml',
438 | '-f', 'docker-compose.maintenance.override.yml',
439 | '-p', config.get_prefix('maintenance'),
440 | ],
441 | 'custom_yml': False,
442 | }
443 | }
444 |
445 | path = group_docker_maps[group].get('path', dict_['kobodocker_path'])
446 | mode = 'stop' if not down else 'down'
447 | options = group_docker_maps[group]['options']
448 | command = run_docker_compose(dict_, options + [mode])
449 | if group_docker_maps[group]['custom_yml']:
450 | cls.__validate_custom_yml(config, command)
451 |
452 | CLI.run_command(command, path)
453 |
454 | @classmethod
455 | def stop_maintenance(cls):
456 | """
457 | Stop maintenance mode
458 | """
459 | config = Config()
460 | dict_ = config.get_dict()
461 |
462 | if not config.multi_servers or config.frontend:
463 | # Stop maintenance container in case it's up&running
464 | cls.stop_containers('maintenance')
465 |
466 | # Create and start NGINX container
467 | frontend_command = run_docker_compose(dict_, [
468 | '-f', 'docker-compose.frontend.yml',
469 | '-f', 'docker-compose.frontend.override.yml',
470 | '-p', config.get_prefix('frontend'),
471 | 'up', '-d',
472 | 'nginx',
473 | ])
474 |
475 | cls.__validate_custom_yml(config, frontend_command)
476 | CLI.run_command(frontend_command, dict_['kobodocker_path'])
477 |
478 | CLI.colored_print('Maintenance mode has been stopped',
479 | CLI.COLOR_SUCCESS)
480 |
481 | dict_['maintenance_enabled'] = False
482 | config.write_config()
483 |
484 | @classmethod
485 | def version(cls):
486 | git_commit_version_command = ['git', 'rev-parse', 'HEAD']
487 | stdout = CLI.run_command(git_commit_version_command)
488 | build = stdout.strip()[0:7]
489 | version = Config.KOBO_INSTALL_VERSION
490 | CLI.colored_print(
491 | f'kobo-install Version: {version} (build {build})',
492 | CLI.COLOR_SUCCESS,
493 | )
494 |
495 | @staticmethod
496 | def __validate_custom_yml(config, command):
497 | """
498 | Validate whether docker-compose must start the containers with a
499 | custom YML file in addition to the default. If the file does not yet exist,
500 | kobo-install is paused until the user creates it and resumes the setup manually.
501 |
502 | If user has chosen to use a custom YML file, it is injected into `command`
503 | before being executed.
504 | """
505 | dict_ = config.get_dict()
506 | frontend_command = True
507 | # Detect if it's a front-end command or back-end command
508 | for part in command:
509 | if 'backend' in part:
510 | frontend_command = False
511 | break
512 |
513 | start_index = 6 # len of command `docker` + extra space
514 | if frontend_command and dict_['use_frontend_custom_yml']:
515 | custom_file = '{}/docker-compose.frontend.custom.yml'.format(
516 | dict_['kobodocker_path']
517 | )
518 |
519 | does_custom_file_exist = os.path.exists(custom_file)
520 | while not does_custom_file_exist:
521 | message = (
522 | 'Please create your custom configuration in\n'
523 | '`{custom_file}`.'
524 | ).format(custom_file=custom_file)
525 | CLI.framed_print(message, color=CLI.COLOR_INFO, columns=90)
526 | input('Press any key when it is done...')
527 | does_custom_file_exist = os.path.exists(custom_file)
528 |
529 | # Add custom file to docker-compose command
530 | command.insert(start_index, '-f')
531 | command.insert(start_index + 1, 'docker-compose.frontend.custom.yml')
532 |
533 | if not frontend_command and dict_['use_backend_custom_yml']:
534 | custom_file = '{}/docker-compose.backend.custom.yml'.format(
535 | dict_['kobodocker_path'],
536 | )
537 |
538 | does_custom_file_exist = os.path.exists(custom_file)
539 | while not does_custom_file_exist:
540 | message = (
541 | 'Please create your custom configuration in\n'
542 | '`{custom_file}`.'
543 | ).format(custom_file=custom_file)
544 | CLI.framed_print(message, color=CLI.COLOR_INFO, columns=90)
545 | input('Press any key when it is done...')
546 | does_custom_file_exist = os.path.exists(custom_file)
547 |
548 | # Add custom file to docker-compose command
549 | command.insert(start_index, '-f')
550 | command.insert(
551 | start_index + 1,
552 | 'docker-compose.backend.custom.yml',
553 | )
554 |
--------------------------------------------------------------------------------
/tests/test_config.py:
--------------------------------------------------------------------------------
1 | # -*- coding: utf-8 -*-
2 | import os
3 | import pytest
4 | import random
5 | import shutil
6 | import tempfile
7 | import time
8 | from unittest.mock import patch, MagicMock
9 |
10 | from helpers.cli import CLI
11 | from helpers.config import Config
12 | from .utils import (
13 | mock_read_config as read_config,
14 | mock_write_trigger_upsert_db_users,
15 | MockAWSValidation
16 | )
17 |
18 | CHOICE_YES = '1'
19 | CHOICE_NO = '2'
20 |
21 |
22 | def test_read_config():
23 | read_config()
24 |
25 |
26 | def test_advanced_options():
27 | config = read_config()
28 | with patch.object(CLI, 'colored_input',
29 | return_value=CHOICE_YES) as mock_ci:
30 | config._Config__questions_advanced_options()
31 | assert config.advanced_options
32 |
33 | with patch.object(CLI, 'colored_input',
34 | return_value=CHOICE_NO) as mock_ci:
35 | config._Config__questions_advanced_options()
36 | assert not config.advanced_options
37 |
38 |
39 | def test_installation():
40 | config = read_config()
41 | with patch.object(CLI, 'colored_input',
42 | return_value=CHOICE_NO) as mock_ci:
43 | config._Config__questions_installation_type()
44 | assert not config.local_install
45 |
46 | with patch.object(CLI, 'colored_input',
47 | return_value=CHOICE_YES) as mock_ci:
48 | config._Config__questions_installation_type()
49 | assert config.local_install
50 | assert not config.multi_servers
51 | assert not config.use_letsencrypt
52 |
53 | return config
54 |
55 |
56 | @patch('helpers.config.Config._Config__clone_repo',
57 | MagicMock(return_value=True))
58 | def test_staging_mode():
59 | config = read_config()
60 | kpi_repo_path = tempfile.mkdtemp()
61 |
62 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
63 | mock_colored_input.side_effect = iter([CHOICE_YES, kpi_repo_path])
64 | config._Config__questions_dev_mode()
65 | dict_ = config.get_dict()
66 | assert not config.dev_mode
67 | assert config.staging_mode
68 | assert dict_['kpi_path'] == kpi_repo_path
69 | shutil.rmtree(kpi_repo_path)
70 |
71 |
72 | @patch('helpers.config.Config._Config__clone_repo', MagicMock(return_value=True))
73 | def test_dev_mode():
74 | config = test_installation()
75 |
76 | kpi_repo_path = tempfile.mkdtemp()
77 |
78 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
79 | mock_colored_input.side_effect = iter(
80 | [
81 | '8080',
82 | CHOICE_YES,
83 | CHOICE_NO,
84 | kpi_repo_path,
85 | CHOICE_YES,
86 | CHOICE_NO,
87 | ]
88 | )
89 |
90 | config._Config__questions_dev_mode()
91 | dict_ = config.get_dict()
92 | assert config.dev_mode
93 | assert not config.staging_mode
94 | assert config.get_dict().get('exposed_nginx_docker_port') == '8080'
95 | assert dict_['kpi_path'] == kpi_repo_path
96 | assert dict_['npm_container'] is False
97 | assert dict_['use_celery'] is False
98 |
99 | shutil.rmtree(kpi_repo_path)
100 |
101 | with patch.object(CLI, 'colored_input', return_value=CHOICE_NO) as mock_ci:
102 | config._Config__questions_dev_mode()
103 | dict_ = config.get_dict()
104 | assert not config.dev_mode
105 | assert dict_['kpi_path'] == ''
106 |
107 |
108 | def test_server_roles_questions():
109 | config = read_config()
110 | assert config.frontend
111 | assert config.backend
112 |
113 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
114 | mock_colored_input.side_effect = iter(
115 | [CHOICE_YES, 'frontend', 'backend'])
116 |
117 | config._Config__questions_multi_servers()
118 |
119 | config._Config__questions_roles()
120 | assert config.frontend
121 | assert not config.backend
122 |
123 | config._Config__questions_roles()
124 | assert not config.frontend
125 | assert config.backend
126 |
127 |
128 | def test_session_cookies():
129 | config = read_config()
130 |
131 | assert config._Config__dict['django_session_cookie_age'] == 604800
132 |
133 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input_1:
134 | mock_colored_input_1.side_effect = iter([
135 | 'None', # Wrong, should ask again
136 | '', # Wrong, should ask again
137 | '1' # Correct, will continue
138 | ])
139 | config._Config__questions_session_cookies()
140 | assert config._Config__dict['django_session_cookie_age'] == 3600
141 |
142 |
143 | def test_use_https():
144 | config = read_config()
145 |
146 | assert config.is_secure
147 |
148 | with patch.object(CLI, 'colored_input',
149 | return_value=CHOICE_YES) as mock_ci:
150 | config._Config__questions_https()
151 | assert not config.local_install
152 | assert config.is_secure
153 |
154 | with patch.object(CLI, 'colored_input',
155 | return_value=CHOICE_YES) as mock_ci:
156 | config._Config__questions_installation_type()
157 | assert config.local_install
158 | assert not config.is_secure
159 |
160 |
161 | def _aws_validation_setup():
162 | config = read_config()
163 |
164 | assert not config._Config__dict['use_aws']
165 | assert not config._Config__dict['aws_credentials_valid']
166 |
167 | return config
168 |
169 |
170 | def test_aws_credentials_invalid_with_no_configuration():
171 | config = _aws_validation_setup()
172 |
173 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
174 | mock_colored_input.side_effect = CHOICE_NO
175 | assert not config._Config__dict['use_aws']
176 | assert not config._Config__dict['aws_credentials_valid']
177 |
178 |
179 | def test_aws_validation_fails_with_system_exit():
180 | config = _aws_validation_setup()
181 |
182 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
183 | mock_colored_input.side_effect = iter(
184 | [
185 | CHOICE_YES, # Yes, Use AWS Storage
186 | '', # Empty Access Key
187 | '', # Empty Secret Key
188 | '', # Empty Bucket Name
189 | '', # Empty Region Name
190 | CHOICE_YES, # Yes, validate AWS credentials
191 | '', # Empty Access Key
192 | '', # Empty Secret Key
193 | '', # Empty Bucket Name
194 | '', # Empty Region Name
195 | # it failed, let's try one more time
196 | '', # Empty Access Key
197 | '', # Empty Secret Key
198 | '', # Empty Bucket Name
199 | '', # Empty Region Name
200 | ]
201 | )
202 | try:
203 | config._Config__questions_aws()
204 | except SystemExit:
205 | pass
206 | assert not config._Config__dict['aws_credentials_valid']
207 |
208 |
209 | def test_aws_invalid_credentials_continue_without_validation():
210 | config = _aws_validation_setup()
211 |
212 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
213 | mock_colored_input.side_effect = iter([CHOICE_YES, '', '', '', '', CHOICE_NO])
214 | config._Config__questions_aws()
215 | assert not config._Config__dict['aws_credentials_valid']
216 |
217 |
218 | @patch('helpers.aws_validation.AWSValidation.validate_credentials',
219 | new=MockAWSValidation.validate_credentials)
220 | def test_aws_validation_passes_with_valid_credentials():
221 | config = _aws_validation_setup()
222 |
223 | # correct keys, no validation, should continue without issue
224 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
225 | mock_colored_input.side_effect = iter(
226 | [
227 | CHOICE_YES,
228 | 'test_access_key',
229 | 'test_secret_key',
230 | 'test_bucket_name',
231 | 'test_region_name',
232 | CHOICE_NO,
233 | ]
234 | )
235 | config._Config__questions_aws()
236 | assert not config._Config__dict['aws_credentials_valid']
237 |
238 | # correct keys in first attempt, choose to validate, continue
239 | # without issue
240 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
241 | config._Config__dict['aws_credentials_valid'] = False
242 | mock_colored_input.side_effect = iter(
243 | [
244 | CHOICE_YES,
245 | 'test_access_key',
246 | 'test_secret_key',
247 | 'test_bucket_name',
248 | 'test_region_name',
249 | CHOICE_YES,
250 | ]
251 | )
252 | config._Config__questions_aws()
253 | assert config._Config__dict['aws_credentials_valid']
254 |
255 | # correct keys in second attempt, choose to validate, continue
256 | # without issue
257 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
258 | config._Config__dict['aws_credentials_valid'] = False
259 | mock_colored_input.side_effect = iter(
260 | [
261 | CHOICE_YES,
262 | '',
263 | '',
264 | '',
265 | '',
266 | CHOICE_YES,
267 | 'test_access_key',
268 | 'test_secret_key',
269 | 'test_bucket_name',
270 | 'test_region_name',
271 | ]
272 | )
273 | config._Config__questions_aws()
274 | assert config._Config__dict['aws_credentials_valid']
275 |
276 | # correct keys in third attempt, choose to validate, continue
277 | # without issue
278 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
279 | config._Config__dict['aws_credentials_valid'] = False
280 | mock_colored_input.side_effect = iter(
281 | [
282 | CHOICE_YES,
283 | '',
284 | '',
285 | '',
286 | '',
287 | CHOICE_YES,
288 | '',
289 | '',
290 | '',
291 | '',
292 | 'test_access_key',
293 | 'test_secret_key',
294 | 'test_bucket_name',
295 | 'test_region_name',
296 | ]
297 | )
298 | config._Config__questions_aws()
299 | assert config._Config__dict['aws_credentials_valid']
300 |
301 |
302 | @patch('helpers.config.Config._Config__clone_repo',
303 | MagicMock(return_value=True))
304 | def test_proxy_letsencrypt():
305 | config = read_config()
306 |
307 | assert config.proxy
308 | assert config.use_letsencrypt
309 |
310 | # Force custom exposed port
311 | config._Config__dict['exposed_nginx_docker_port'] = '8088'
312 |
313 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
314 | # Use default options
315 | mock_colored_input.side_effect = iter(
316 | [CHOICE_YES, 'test@test.com', CHOICE_YES, Config.DEFAULT_NGINX_PORT]
317 | )
318 | config._Config__questions_reverse_proxy()
319 | dict_ = config.get_dict()
320 | assert config.proxy
321 | assert config.use_letsencrypt
322 | assert config.block_common_http_ports
323 | assert dict_['nginx_proxy_port'] == Config.DEFAULT_PROXY_PORT
324 | assert dict_['exposed_nginx_docker_port'] == Config.DEFAULT_NGINX_PORT
325 |
326 |
327 | def test_proxy_no_letsencrypt_advanced():
328 | config = read_config()
329 | # Force advanced options
330 | config._Config__dict['advanced'] = True
331 | assert config.advanced_options
332 | assert config.proxy
333 | assert config.use_letsencrypt
334 | proxy_port = Config.DEFAULT_NGINX_PORT
335 |
336 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
337 | mock_colored_input.side_effect = iter(
338 | [CHOICE_NO, CHOICE_NO, proxy_port])
339 | config._Config__questions_reverse_proxy()
340 | dict_ = config.get_dict()
341 | assert config.proxy
342 | assert not config.use_letsencrypt
343 | assert not config.block_common_http_ports
344 | assert dict_['nginx_proxy_port'] == proxy_port
345 |
346 |
347 | def test_proxy_no_letsencrypt():
348 | config = read_config()
349 |
350 | assert config.proxy
351 | assert config.use_letsencrypt
352 |
353 | with patch.object(CLI, 'colored_input',
354 | return_value=CHOICE_NO) as mock_ci:
355 | config._Config__questions_reverse_proxy()
356 | dict_ = config.get_dict()
357 | assert config.proxy
358 | assert not config.use_letsencrypt
359 | assert config.block_common_http_ports
360 | assert dict_['nginx_proxy_port'] == Config.DEFAULT_PROXY_PORT
361 |
362 |
363 | def test_proxy_no_letsencrypt_retains_custom_nginx_proxy_port():
364 | custom_proxy_port = 9090
365 | config = read_config(overrides={
366 | 'advanced': True,
367 | 'use_letsencrypt': False,
368 | 'nginx_proxy_port': str(custom_proxy_port),
369 | })
370 | with patch.object(
371 | CLI, 'colored_input',
372 | new=classmethod(lambda cls, message, color, default: default)
373 | ) as mock_ci:
374 | config._Config__questions_reverse_proxy()
375 | dict_ = config.get_dict()
376 | assert dict_['nginx_proxy_port'] == str(custom_proxy_port)
377 |
378 |
379 | def test_no_proxy_no_ssl():
380 | config = read_config()
381 | dict_ = config.get_dict()
382 | assert config.is_secure
383 | assert dict_['nginx_proxy_port'] == Config.DEFAULT_PROXY_PORT
384 |
385 | proxy_port = Config.DEFAULT_NGINX_PORT
386 |
387 | with patch.object(CLI, 'colored_input',
388 | return_value=CHOICE_NO) as mock_ci:
389 | config._Config__questions_https()
390 | assert not config.is_secure
391 |
392 | with patch.object(CLI, 'colored_input',
393 | return_value=CHOICE_NO) as mock_ci_2:
394 | config._Config__questions_reverse_proxy()
395 | dict_ = config.get_dict()
396 | assert not config.proxy
397 | assert not config.use_letsencrypt
398 | assert not config.block_common_http_ports
399 | assert dict_['nginx_proxy_port'] == proxy_port
400 |
401 |
402 | def test_proxy_no_ssl_advanced():
403 | config = read_config()
404 | # Force advanced options
405 | config._Config__dict['advanced'] = True
406 | assert config.advanced_options
407 | assert config.is_secure
408 |
409 | with patch.object(CLI, 'colored_input',
410 | return_value=CHOICE_NO) as mock_ci:
411 | config._Config__questions_https()
412 | assert not config.is_secure
413 |
414 | # Proxy - not on the same server
415 | proxy_port = Config.DEFAULT_NGINX_PORT
416 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input_1:
417 | mock_colored_input_1.side_effect = iter(
418 | [CHOICE_YES, CHOICE_NO, proxy_port])
419 | config._Config__questions_reverse_proxy()
420 | dict_ = config.get_dict()
421 | assert config.proxy
422 | assert not config.use_letsencrypt
423 | assert not config.block_common_http_ports
424 | assert dict_['nginx_proxy_port'] == proxy_port
425 |
426 | # Proxy - on the same server
427 | proxy_port = Config.DEFAULT_PROXY_PORT
428 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input_2:
429 | mock_colored_input_2.side_effect = iter(
430 | [CHOICE_YES, CHOICE_YES, proxy_port])
431 | config._Config__questions_reverse_proxy()
432 | dict_ = config.get_dict()
433 | assert config.proxy
434 | assert not config.use_letsencrypt
435 | assert config.block_common_http_ports
436 | assert dict_['nginx_proxy_port'] == proxy_port
437 |
438 |
439 | def test_port_allowed():
440 | config = read_config()
441 | # Use let's encrypt by default
442 | assert not config._Config__is_port_allowed(Config.DEFAULT_NGINX_PORT)
443 | assert not config._Config__is_port_allowed('443')
444 | assert config._Config__is_port_allowed(Config.DEFAULT_PROXY_PORT)
445 |
446 | # Don't use let's encrypt
447 | config._Config__dict['use_letsencrypt'] = False
448 | config._Config__dict['block_common_http_ports'] = False
449 | assert config._Config__is_port_allowed(Config.DEFAULT_NGINX_PORT)
450 | assert config._Config__is_port_allowed('443')
451 |
452 |
453 | def test_create_directory():
454 | config = read_config()
455 | destination_path = tempfile.mkdtemp()
456 |
457 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
458 | mock_colored_input.side_effect = iter([destination_path, CHOICE_YES])
459 | config._Config__create_directory()
460 | dict_ = config.get_dict()
461 | assert dict_['kobodocker_path'] == destination_path
462 |
463 | shutil.rmtree(destination_path)
464 |
465 |
466 | @patch('helpers.config.Config.write_config', new=lambda *a, **k: None)
467 | def test_maintenance():
468 | config = read_config()
469 |
470 | # First time
471 | with pytest.raises(SystemExit) as pytest_wrapped_e:
472 | config.maintenance()
473 | assert pytest_wrapped_e.type == SystemExit
474 | assert pytest_wrapped_e.value.code == 1
475 |
476 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input_1:
477 | mock_colored_input_1.side_effect = iter([
478 | '2hours', # Wrong value, it should ask again
479 | '2 hours', # OK
480 | '', # Wrong value, it should ask again
481 | '20190101T0200', # OK
482 | 'email@example.com'
483 | ])
484 | config._Config__dict['date_created'] = time.time()
485 | config._Config__first_time = False
486 | config.maintenance()
487 | dict_ = config.get_dict()
488 | expected_str = 'Tuesday, January 01 ' \
489 | 'at 02:00 GMT'
490 | assert dict_['maintenance_date_str'] == expected_str
491 |
492 |
493 | def test_exposed_ports():
494 | config = read_config()
495 | with patch.object(CLI, 'colored_input',
496 | return_value=CHOICE_YES) as mock_ci:
497 | # Choose multi servers options
498 | config._Config__questions_multi_servers()
499 |
500 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
501 | # Choose to customize ports
502 | mock_ci.side_effect = iter(
503 | [CHOICE_YES, '5532', '27117', '6479', '6480'])
504 | config._Config__questions_ports()
505 |
506 | assert config._Config__dict['postgresql_port'] == '5532'
507 | assert config._Config__dict['mongo_port'] == '27117'
508 | assert config._Config__dict['redis_main_port'] == '6479'
509 | assert config._Config__dict['redis_cache_port'] == '6480'
510 | assert config.expose_backend_ports
511 |
512 | with patch.object(CLI, 'colored_input',
513 | return_value=CHOICE_NO) as mock_ci_1:
514 | # Choose to single server
515 | config._Config__questions_multi_servers()
516 |
517 | with patch.object(CLI, 'colored_input',
518 | return_value=CHOICE_NO) as mock_ci_2:
519 | # Choose to not expose ports
520 | config._Config__questions_ports()
521 |
522 | assert config._Config__dict['postgresql_port'] == '5432'
523 | assert config._Config__dict['mongo_port'] == '27017'
524 | assert config._Config__dict['redis_main_port'] == '6379'
525 | assert config._Config__dict['redis_cache_port'] == '6380'
526 | assert not config.expose_backend_ports
527 |
528 |
529 | @patch('helpers.config.Config.write_config', new=lambda *a, **k: None)
530 | def test_force_secure_mongo():
531 | config = read_config()
532 | dict_ = config.get_dict()
533 |
534 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
535 | # We need to run it like if user has already run the setup once to
536 | # force MongoDB to 'upsert' users.
537 | config._Config__first_time = False
538 | # Run with no advanced options
539 |
540 | mock_ci.side_effect = iter([
541 | dict_['kobodocker_path'],
542 | CHOICE_YES, # Confirm path
543 | CHOICE_NO,
544 | CHOICE_NO,
545 | dict_['public_domain_name'],
546 | dict_['kpi_subdomain'],
547 | dict_['kc_subdomain'],
548 | dict_['ee_subdomain'],
549 | CHOICE_NO, # Do you want to use HTTPS?
550 | dict_['smtp_host'],
551 | dict_['smtp_port'],
552 | dict_['smtp_user'],
553 | 'test@test.com',
554 | dict_['super_user_username'],
555 | dict_['super_user_password'],
556 | CHOICE_NO,
557 | ])
558 | new_config = config.build()
559 | assert new_config['mongo_secured'] is True
560 |
561 |
562 | @patch('helpers.config.Config._Config__write_upsert_db_users_trigger_file',
563 | new=mock_write_trigger_upsert_db_users)
564 | def test_secure_mongo_advanced_options():
565 | config = read_config()
566 | config._Config__dict['advanced'] = True
567 |
568 | # Try when setup is run for the first time.
569 | config._Config__first_time = True
570 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
571 | mock_ci.side_effect = iter([
572 | 'root',
573 | 'rootpassword',
574 | 'mongo_kobo_user',
575 | 'mongopassword',
576 | ])
577 | config._Config__questions_mongo()
578 | assert not os.path.exists('/tmp/upsert_db_users')
579 |
580 | # Try when setup has been already run once
581 | # If it's an upgrade, users should not see:
582 | # ╔══════════════════════════════════════════════════════╗
583 | # ║ MongoDB root's and/or user's usernames have changed! ║
584 | # ╚══════════════════════════════════════════════════════╝
585 | config._Config__first_time = False
586 | config._Config__dict['mongo_secured'] = False
587 |
588 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
589 | mock_ci.side_effect = iter([
590 | 'root',
591 | 'rootPassword',
592 | 'mongo_kobo_user',
593 | 'mongoPassword',
594 | ])
595 | config._Config__questions_mongo()
596 | assert os.path.exists('/tmp/upsert_db_users')
597 | assert os.path.getsize('/tmp/upsert_db_users') == 0
598 | os.remove('/tmp/upsert_db_users')
599 |
600 | # Try when setup has been already run once
601 | # If it's NOT an upgrade, Users should see:
602 | # ╔══════════════════════════════════════════════════════╗
603 | # ║ MongoDB root's and/or user's usernames have changed! ║
604 | # ╚══════════════════════════════════════════════════════╝
605 | config._Config__dict['mongo_secured'] = True
606 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
607 | mock_ci.side_effect = iter([
608 | 'root',
609 | 'rootPassw0rd',
610 | 'kobo_user',
611 | 'mongoPassword',
612 | CHOICE_YES,
613 | ])
614 | config._Config__questions_mongo()
615 | assert os.path.exists('/tmp/upsert_db_users')
616 | assert os.path.getsize('/tmp/upsert_db_users') != 0
617 | os.remove('/tmp/upsert_db_users')
618 |
619 |
620 | @patch('helpers.config.Config._Config__write_upsert_db_users_trigger_file',
621 | new=mock_write_trigger_upsert_db_users)
622 | def test_update_mongo_passwords():
623 | config = read_config()
624 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
625 | config._Config__first_time = False
626 | # Test with unsecured MongoDB is covered in test_secure_mongo_advanced_options
627 | config._Config__dict['mongo_secured'] = True
628 | config._Config__dict['mongo_root_username'] = 'root'
629 | config._Config__dict['mongo_user_username'] = 'user'
630 | mock_ci.side_effect = iter([
631 | 'root',
632 | 'rootPassword',
633 | 'user',
634 | 'mongoPassword'
635 | ])
636 | config._Config__questions_mongo()
637 | assert os.path.exists('/tmp/upsert_db_users')
638 | assert os.path.getsize('/tmp/upsert_db_users') == 0
639 | os.remove('/tmp/upsert_db_users')
640 |
641 |
642 | @patch('helpers.config.Config._Config__write_upsert_db_users_trigger_file',
643 | new=mock_write_trigger_upsert_db_users)
644 | def test_update_mongo_usernames():
645 | config = read_config()
646 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
647 | config._Config__first_time = False
648 | config._Config__dict['mongo_root_username'] = 'root'
649 | config._Config__dict['mongo_user_username'] = 'user'
650 | mock_ci.side_effect = iter([
651 | 'admin',
652 | 'rootPassword',
653 | 'another_user',
654 | 'mongoPassword',
655 | CHOICE_YES # Delete users
656 | ])
657 | config._Config__questions_mongo()
658 | assert os.path.exists('/tmp/upsert_db_users')
659 | with open('/tmp/upsert_db_users', 'r') as f:
660 | content = f.read()
661 | expected_content = 'user\tformhub\nroot\tadmin'
662 | assert content == expected_content
663 | os.remove('/tmp/upsert_db_users')
664 |
665 |
666 | @patch('helpers.config.Config._Config__write_upsert_db_users_trigger_file',
667 | new=mock_write_trigger_upsert_db_users)
668 | def test_update_postgres_password():
669 | """
670 | Does **NOT** test if user is updated in PostgreSQL but the file creation
671 | (and its content) used to trigger the action by PostgreSQL container.
672 |
673 | When password changes, file must contain ``
674 | Users should not be deleted if they already exist.
675 | """
676 | config = read_config()
677 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
678 | config._Config__first_time = False
679 | config._Config__dict['postgres_user'] = 'user'
680 | config._Config__dict['postgres_password'] = 'password'
681 | mock_ci.side_effect = iter([
682 | 'kobocat',
683 | 'koboform',
684 | 'user',
685 | 'userPassw0rd',
686 | CHOICE_NO, # Tweak settings
687 | ])
688 | config._Config__questions_postgres()
689 | assert os.path.exists('/tmp/upsert_db_users')
690 | with open('/tmp/upsert_db_users', 'r') as f:
691 | content = f.read()
692 | expected_content = 'user\tfalse'
693 | assert content == expected_content
694 | os.remove('/tmp/upsert_db_users')
695 |
696 |
697 | @patch('helpers.config.Config._Config__write_upsert_db_users_trigger_file',
698 | new=mock_write_trigger_upsert_db_users)
699 | def test_update_postgres_username():
700 | """
701 | Does **NOT** test if user is updated in PostgreSQL but the file creation
702 | (and its content) used to trigger the action by PostgreSQL container.
703 |
704 | When username changes, file must contain ``
705 | """
706 | config = read_config()
707 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
708 | config._Config__first_time = False
709 | config._Config__dict['postgres_user'] = 'user'
710 | config._Config__dict['postgres_password'] = 'password'
711 | mock_ci.side_effect = iter([
712 | 'kobocat',
713 | 'koboform',
714 | 'another_user',
715 | 'password',
716 | CHOICE_YES, # Delete user
717 | CHOICE_NO, # Tweak settings
718 | ])
719 | config._Config__questions_postgres()
720 | assert os.path.exists('/tmp/upsert_db_users')
721 | with open('/tmp/upsert_db_users', 'r') as f:
722 | content = f.read()
723 | expected_content = 'user\ttrue'
724 | assert content == expected_content
725 | os.remove('/tmp/upsert_db_users')
726 |
727 |
728 | def test_update_postgres_db_name_from_single_database():
729 | """
730 | Simulate upgrade from single database to two databases.
731 | With two databases, KoboCat has its own database. We ensure that
732 | `kc_postgres_db` gets `postgres_db` value.
733 | """
734 | config = read_config()
735 | dict_ = config.get_dict()
736 | old_db_name = 'postgres_db_kobo'
737 | config._Config__dict['postgres_db'] = old_db_name
738 | del config._Config__dict['kc_postgres_db']
739 | assert 'postgres_db' in dict_
740 | assert 'kc_postgres_db' not in dict_
741 | dict_ = config.get_upgraded_dict()
742 | assert dict_['kc_postgres_db'] == old_db_name
743 |
744 |
745 | def test_use_boolean():
746 | """
747 | Ensure config uses booleans instead of '1' or '2'
748 | """
749 | config = read_config()
750 | boolean_properties = [
751 | 'advanced',
752 | 'aws_backup_bucket_deletion_rule_enabled',
753 | 'backup_from_primary',
754 | 'block_common_http_ports',
755 | 'custom_secret_keys',
756 | 'customized_ports',
757 | 'debug',
758 | 'dev_mode',
759 | 'expose_backend_ports',
760 | 'https',
761 | 'local_installation',
762 | 'multi',
763 | 'npm_container',
764 | 'postgres_settings',
765 | 'proxy',
766 | 'raven_settings',
767 | 'review_host',
768 | 'smtp_use_tls',
769 | 'staging_mode',
770 | 'two_databases',
771 | 'use_aws',
772 | 'use_backup',
773 | 'use_letsencrypt',
774 | 'use_private_dns',
775 | 'uwsgi_settings',
776 | ]
777 | expected_dict = {}
778 | for property_ in boolean_properties:
779 | old_value = str(random.randint(1, 2))
780 | expected_dict[property_] = True if old_value == '1' else False
781 | config._Config__dict[property_] = old_value
782 |
783 | dict_ = config.get_upgraded_dict()
784 |
785 | for property_ in boolean_properties:
786 | assert dict_[property_] == expected_dict[property_]
787 |
788 |
789 | def test_backup_schedules_from_single_instance():
790 | config = read_config()
791 | # Force advanced options and single instance
792 | config._Config__dict['advanced'] = True
793 | config._Config__dict['multi'] = False
794 |
795 | assert config._Config__dict['kobocat_media_backup_schedule'] == '0 0 * * 0'
796 | assert config._Config__dict['mongo_backup_schedule'] == '0 1 * * 0'
797 | assert config._Config__dict['postgres_backup_schedule'] == '0 2 * * 0'
798 | assert config._Config__dict['redis_backup_schedule'] == '0 3 * * 0'
799 |
800 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
801 | mock_ci.side_effect = iter([
802 | CHOICE_YES, # Activate backup
803 | '1 1 1 1 1', # KoBoCAT media
804 | '2 2 2 2 2', # PostgreSQL
805 | '3 3 3 3 3', # Mongo
806 | '4 4 4 4 4', # Redis
807 | ])
808 | config._Config__questions_backup()
809 | assert config._Config__dict['kobocat_media_backup_schedule'] == '1 1 1 1 1'
810 | assert config._Config__dict['postgres_backup_schedule'] == '2 2 2 2 2'
811 | assert config._Config__dict['mongo_backup_schedule'] == '3 3 3 3 3'
812 | assert config._Config__dict['redis_backup_schedule'] == '4 4 4 4 4'
813 |
814 |
815 | def test_backup_schedules_from_frontend_instance():
816 | config = read_config()
817 | # Force advanced options
818 | config._Config__dict['advanced'] = True
819 |
820 | assert config._Config__dict['kobocat_media_backup_schedule'] == '0 0 * * 0'
821 |
822 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
823 | mock_colored_input.side_effect = iter(
824 | [CHOICE_YES, 'frontend']
825 | )
826 | config._Config__questions_multi_servers()
827 | config._Config__questions_roles()
828 |
829 | assert config.frontend
830 |
831 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
832 | mock_ci.side_effect = iter([
833 | CHOICE_YES, # Activate backup
834 | '1 1 1 1 1', # KoBoCAT media
835 | ])
836 | config._Config__questions_backup()
837 | assert config._Config__dict['kobocat_media_backup_schedule'] == '1 1 1 1 1'
838 |
839 |
840 | def test_backup_schedules_from_backend():
841 | config = read_config()
842 | # Force advanced options
843 | config._Config__dict['advanced'] = True
844 |
845 | assert config._Config__dict['mongo_backup_schedule'] == '0 1 * * 0'
846 | assert config._Config__dict['postgres_backup_schedule'] == '0 2 * * 0'
847 | assert config._Config__dict['redis_backup_schedule'] == '0 3 * * 0'
848 |
849 | with patch('helpers.cli.CLI.colored_input') as mock_colored_input:
850 | mock_colored_input.side_effect = iter([CHOICE_YES, 'backend'])
851 | config._Config__questions_multi_servers()
852 | config._Config__questions_roles()
853 | assert config.backend
854 |
855 | with patch('helpers.cli.CLI.colored_input') as mock_ci:
856 | mock_ci.side_effect = iter([
857 | CHOICE_YES, # Activate backup
858 | CHOICE_NO, # Choose AWS
859 | '1 1 1 1 1', # PostgreSQL
860 | '3 3 3 3 3', # Mongo
861 | '4 4 4 4 4', # Redis
862 | ])
863 | config._Config__questions_backup()
864 |
865 | assert config._Config__dict['postgres_backup_schedule'] == '1 1 1 1 1'
866 | assert config._Config__dict['mongo_backup_schedule'] == '3 3 3 3 3'
867 | assert config._Config__dict['redis_backup_schedule'] == '4 4 4 4 4'
868 |
869 |
870 | def test_activate_only_postgres_backup():
871 | config = read_config()
872 | # Force advanced options and single instance
873 | config._Config__dict['advanced'] = True
874 | config._Config__dict['multi'] = False
875 | # Force `False` to validate it becomes `True` at the end
876 | config._Config__dict['backup_from_primary'] = False
877 |
878 | assert config._Config__dict['kobocat_media_backup_schedule'] == '0 0 * * 0'
879 | assert config._Config__dict['mongo_backup_schedule'] == '0 1 * * 0'
880 | assert config._Config__dict['postgres_backup_schedule'] == '0 2 * * 0'
881 | assert config._Config__dict['redis_backup_schedule'] == '0 3 * * 0'
882 |
883 | with patch('builtins.input') as mock_input:
884 | mock_input.side_effect = iter([
885 | CHOICE_YES, # Activate backup
886 | '-', # Deactivate KoBoCAT media
887 | '2 2 2 2 2', # Modify PostgreSQL
888 | '-', # Deactivate Mongo
889 | '-', # Deactivate Redis
890 | ])
891 | config._Config__questions_backup()
892 | assert config._Config__dict['kobocat_media_backup_schedule'] == ''
893 | assert config._Config__dict['postgres_backup_schedule'] == '2 2 2 2 2'
894 | assert config._Config__dict['mongo_backup_schedule'] == ''
895 | assert config._Config__dict['redis_backup_schedule'] == ''
896 |
--------------------------------------------------------------------------------