From 7da62f96f29fbd12d42c0686abbc8adcb05c4049 Mon Sep 17 00:00:00 2001 From: Jared L <48422312+lhjt@users.noreply.github.com> Date: Mon, 25 Jul 2022 10:38:47 +1000 Subject: [PATCH 01/17] fix(nginx): set `proxy_http_version` to `1.1` This sets the `proxy_http_version` to `1.1`. The reason this change should be implemented is that when this is deployed in a Kubernetes environment with Istio, the proxy request fails with HTTP error 426 as the HTTP version is too low for envoy to handle. Setting it to `1.1` will solve the issue. --- bench/config/templates/nginx.conf | 1 + 1 file changed, 1 insertion(+) diff --git a/bench/config/templates/nginx.conf b/bench/config/templates/nginx.conf index 4af72ab4..6cd67d0b 100644 --- a/bench/config/templates/nginx.conf +++ b/bench/config/templates/nginx.conf @@ -91,6 +91,7 @@ server { } location @webserver { + proxy_http_version 1.1; proxy_set_header X-Forwarded-For $remote_addr; proxy_set_header X-Forwarded-Proto $scheme; proxy_set_header X-Frappe-Site-Name {{ site_name }}; From a6f196440ad8c9b65a5bbc31a88d8c80f2ba2ded Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 10:56:46 +0530 Subject: [PATCH 02/17] refactor(cli): Commands Resolution The implementations so far were hacks that worked for the most used commands but broken down when challenged or expected to maintain documented usages [eg: custom app commands]. The current implementation consists of a two step approach: 1. figure out command name that user is trying to execute 2. pass the directive to the app (bench, frappe or other) that consists of the cmd --- For tackling #1, get_cmd_from_sysargv contains exhaustive rules that cover all (that i know and ive come across) combinations of valid frappe commands. For problem #2, a simple check in click's Group object does the trick. Tested with possible known commands with combinations of context flags and params, with bench, frappe & external app's commands --- bench/cli.py | 59 ++++++++++++++++++++++------------------- bench/utils/__init__.py | 46 +++++++++++++++++++++++++------- 2 files changed, 68 insertions(+), 37 deletions(-) diff --git a/bench/cli.py b/bench/cli.py index 9cf946cc..bfe6f4c3 100755 --- a/bench/cli.py +++ b/bench/cli.py @@ -1,6 +1,8 @@ # imports - standard imports import atexit +from contextlib import contextmanager import json +from logging import Logger import os import pwd import sys @@ -25,7 +27,7 @@ from bench.utils import ( is_root, log, setup_logging, - parse_sys_argv, + get_cmd_from_sysargv, ) from bench.utils.bench import get_env_cmd @@ -35,23 +37,41 @@ verbose = False is_envvar_warn_set = None from_command_line = False # set when commands are executed via the CLI bench.LOG_BUFFER = [] -sys_argv = None change_uid_msg = "You should not run this command as root" src = os.path.dirname(__file__) +@contextmanager +def execute_cmd(check_for_update=True, command: str = None, logger: Logger = None): + if check_for_update: + atexit.register(check_latest_version) + + try: + yield + except BaseException as e: + return_code = getattr(e, "code", 1) + + if isinstance(e, Exception): + click.secho(f"ERROR: {e}", fg="red") + + if return_code: + logger.warning(f"{command} executed with exit code {return_code}") + + raise e + + def cli(): - global from_command_line, bench_config, is_envvar_warn_set, verbose, sys_argv + global from_command_line, bench_config, is_envvar_warn_set, verbose from_command_line = True command = " ".join(sys.argv) argv = set(sys.argv) is_envvar_warn_set = not (os.environ.get("BENCH_DEVELOPER") or os.environ.get("CI")) is_cli_command = len(sys.argv) > 1 and not argv.intersection({"src", "--version"}) - sys_argv = parse_sys_argv() + cmd_from_sys = get_cmd_from_sysargv() - if "--verbose" in sys_argv.options: + if "--verbose" in argv: verbose = True change_working_directory() @@ -69,8 +89,8 @@ def cli(): if ( is_envvar_warn_set and is_cli_command - and is_dist_editable(bench.PROJECT_NAME) and not bench_config.get("developer_mode") + and is_dist_editable(bench.PROJECT_NAME) ): log( "bench is installed in editable mode!\n\nThis is not the recommended mode" @@ -95,31 +115,14 @@ def cli(): print(get_frappe_help()) return - if ( - sys_argv.commands.intersection(get_cached_frappe_commands()) - or sys_argv.commands.intersection(get_frappe_commands()) - ): + if cmd_from_sys in bench_command.commands: + with execute_cmd(check_for_update=not is_cli_command, command=command, logger=logger): + bench_command() + elif cmd_from_sys in get_frappe_commands(): frappe_cmd() - - if sys.argv[1] in Bench(".").apps: + else: app_cmd() - if not is_cli_command: - atexit.register(check_latest_version) - - try: - bench_command() - except BaseException as e: - return_code = getattr(e, "code", 1) - - if isinstance(e, Exception): - click.secho(f"ERROR: {e}", fg="red") - - if return_code: - logger.warning(f"{command} executed with exit code {return_code}") - - raise e - def check_uid(): if cmd_requires_root() and not is_root(): diff --git a/bench/utils/__init__.py b/bench/utils/__init__.py index 85746877..cd323ca1 100644 --- a/bench/utils/__init__.py +++ b/bench/utils/__init__.py @@ -165,7 +165,7 @@ def which(executable: str, raise_err: bool = False) -> str: return exec_ -def setup_logging(bench_path=".") -> "logger": +def setup_logging(bench_path=".") -> logging.Logger: LOG_LEVEL = 15 logging.addLevelName(LOG_LEVEL, "LOG") @@ -529,13 +529,41 @@ class _dict(dict): return _dict(dict(self).copy()) -def parse_sys_argv(): - sys_argv = _dict(options=set(), commands=set()) +def get_cmd_from_sysargv(): + """Identify and segregate tokens to options and command - for c in sys.argv[1:]: - if c.startswith("-"): - sys_argv.options.add(c) - else: - sys_argv.commands.add(c) + For Command: `bench --profile --site frappeframework.com migrate --no-backup` + sys.argv: ["/home/frappe/.local/bin/bench", "--profile", "--site", "frappeframework.com", "migrate", "--no-backup"] + Actual command run: migrate - return sys_argv + """ + # context is passed as options to frappe's bench_helper + from bench.bench import Bench + frappe_context = _dict( + params={"--site"}, + flags={"--verbose", "--profile", "--force"} + ) + cmd_from_ctx = None + sys_argv = sys.argv[1:] + skip_next = False + + for arg in sys_argv: + if skip_next: + skip_next = False + continue + + if arg in frappe_context.flags: + continue + + elif arg in frappe_context.params: + skip_next = True + continue + + if sys_argv.index(arg) == 0 and arg in Bench(".").apps: + continue + + cmd_from_ctx = arg + + break + + return cmd_from_ctx From 8a0b78451b3ef33db342c3081ee371bc29e4a82f Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 14:21:44 +0530 Subject: [PATCH 03/17] fix: Add support for options on bench main group Options like --use-feature, --version are tested and support maintained by the changes defined in this commit --- bench/cli.py | 27 +++++++++++++++++++-------- 1 file changed, 19 insertions(+), 8 deletions(-) diff --git a/bench/cli.py b/bench/cli.py index bfe6f4c3..b874532a 100755 --- a/bench/cli.py +++ b/bench/cli.py @@ -109,20 +109,31 @@ def cli(): ): log("Command not being executed in bench directory", level=3) - if in_bench and len(sys.argv) > 1: - if sys.argv[1] == "--help": - print(click.Context(bench_command).get_help()) + if len(sys.argv) == 1 or sys.argv[1] == "--help": + print(click.Context(bench_command).get_help()) + if in_bench: print(get_frappe_help()) - return + return - if cmd_from_sys in bench_command.commands: - with execute_cmd(check_for_update=not is_cli_command, command=command, logger=logger): - bench_command() - elif cmd_from_sys in get_frappe_commands(): + _opts = [x.opts + x.secondary_opts for x in bench_command.params] + opts = {item for sublist in _opts for item in sublist} + + # handle usages like `--use-feature='feat-x'` and `--use-feature 'feat-x'` + if cmd_from_sys and cmd_from_sys.split("=", 1)[0].strip() in opts: + bench_command() + + if cmd_from_sys in bench_command.commands: + with execute_cmd(check_for_update=not is_cli_command, command=command, logger=logger): + bench_command() + + if in_bench: + if cmd_from_sys in get_frappe_commands(): frappe_cmd() else: app_cmd() + bench_command() + def check_uid(): if cmd_requires_root() and not is_root(): From e03f597ac6edec87526e00d765f0dcf17d3653f4 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 14:24:10 +0530 Subject: [PATCH 04/17] fix: get-app on existing apps get-app to replace existing folder would fail due to bad url generation. Changes * Archive old repo instead of overwritting * Resetting flags in App instance --- bench/app.py | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/bench/app.py b/bench/app.py index ed006427..529e942a 100755 --- a/bench/app.py +++ b/bench/app.py @@ -209,6 +209,9 @@ class App(AppMeta): shutil.move(active_app_path, archived_app_path) log(f"App moved from {active_app_path} to {archived_app_path}") + self.from_apps = False + self.on_disk = False + @step(title="Installing App {repo}", success="App {repo} Installed") def install( self, @@ -419,7 +422,7 @@ def get_app( "Do you want to continue and overwrite it?" ) ): - shutil.rmtree(cloned_path) + app.remove() to_clone = True if to_clone: From b1b8d70055868ce1cf6b36789dfebc0e08936ef9 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 11:39:36 +0530 Subject: [PATCH 05/17] feat: Add pre-commit checks * Base pre-commit checks for trailing whitespces, valid files, etc * pyupgrade with PY37+ syntax * Aditya's black fork for codebases with tabs * Flake8 with Frappe's config --- .flake8 | 36 ++++++++++++++++++++++++++++++++++++ .pre-commit-config.yaml | 37 +++++++++++++++++++++++++++++++++++++ 2 files changed, 73 insertions(+) create mode 100644 .flake8 create mode 100644 .pre-commit-config.yaml diff --git a/.flake8 b/.flake8 new file mode 100644 index 00000000..940e8e55 --- /dev/null +++ b/.flake8 @@ -0,0 +1,36 @@ +[flake8] +ignore = + E121, + E126, + E127, + E128, + E203, + E225, + E226, + E231, + E241, + E251, + E261, + E265, + E302, + E303, + E305, + E402, + E501, + E741, + W291, + W292, + W293, + W391, + W503, + W504, + F403, + B007, + B950, + W191, + E124, # closing bracket, irritating while writing QB code + E131, # continuation line unaligned for hanging indent + E123, # closing bracket does not match indentation of opening bracket's line + E101, # ensured by use of black + +max-line-length = 200 diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml new file mode 100644 index 00000000..b40aa088 --- /dev/null +++ b/.pre-commit-config.yaml @@ -0,0 +1,37 @@ +exclude: '.git' +default_stages: [commit] +fail_fast: false + +repos: + - repo: https://github.com/pre-commit/pre-commit-hooks + rev: v4.3.0 + hooks: + - id: trailing-whitespace + files: "frappe.*" + exclude: ".*json$|.*txt$|.*csv|.*md|.*svg" + - id: check-yaml + - id: check-merge-conflict + - id: check-ast + - id: check-json + - id: check-toml + - id: check-yaml + - id: debug-statements + + - repo: https://github.com/asottile/pyupgrade + rev: v2.34.0 + hooks: + - id: pyupgrade + args: ['--py37-plus'] + + - repo: https://github.com/adityahase/black + rev: 9cb0a69f4d0030cdf687eddf314468b39ed54119 + hooks: + - id: black + additional_dependencies: ['click==8.0.4'] + + - repo: https://gitlab.com/pycqa/flake8 + rev: 3.9.2 + hooks: + - id: flake8 + additional_dependencies: ['flake8-bugbear',] + args: ['--config', '.flake8'] From a84239d6ab2e1b3c7fe459cf0896ca624375ed53 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 14:39:22 +0530 Subject: [PATCH 06/17] refactor: Bench * Drop patches of v3 & v4 * Re-write buggy / broken code wrt hints by pre-commit checks * Auto-format and transform code * Remove re-written & mutable function defaults --- .flake8 | 1 + bench/app.py | 77 ++++--- bench/bench.py | 92 ++++---- bench/cli.py | 9 +- bench/commands/__init__.py | 11 +- bench/commands/config.py | 64 +++--- bench/commands/install.py | 95 +++++--- bench/commands/make.py | 16 +- bench/commands/setup.py | 167 +++++++++++--- bench/commands/update.py | 99 ++++++-- bench/commands/utils.py | 142 +++++++----- bench/config/__init__.py | 3 +- bench/config/common_site_config.py | 59 ++--- bench/config/lets_encrypt.py | 99 ++++---- bench/config/nginx.py | 160 +++++++------ bench/config/procfile.py | 29 ++- bench/config/production_setup.py | 92 +++++--- bench/config/redis.py | 54 ++--- bench/config/site_config.py | 84 ++++--- bench/config/supervisor.py | 90 +++++--- bench/config/systemd.py | 255 ++++++++++++++------- bench/exceptions.py | 2 +- bench/patches/__init__.py | 31 ++- bench/patches/v3/__init__.py | 0 bench/patches/v3/celery_to_rq.py | 25 -- bench/patches/v3/deprecate_old_config.py | 38 --- bench/patches/v3/redis_bind_ip.py | 10 - bench/patches/v4/__init__.py | 0 bench/patches/v4/install_yarn.py | 5 - bench/patches/v4/update_node.py | 32 --- bench/patches/v4/update_socketio.py | 4 - bench/patches/v5/fix_backup_cronjob.py | 4 +- bench/patches/v5/fix_user_permissions.py | 6 +- bench/patches/v5/set_live_reload_config.py | 2 +- bench/patches/v5/update_archived_sites.py | 2 +- bench/tests/test_base.py | 57 +++-- bench/tests/test_init.py | 93 +++++--- bench/tests/test_setup_production.py | 106 +++++---- bench/tests/test_utils.py | 31 ++- bench/utils/__init__.py | 34 +-- bench/utils/app.py | 30 ++- bench/utils/bench.py | 56 ++--- bench/utils/render.py | 29 ++- bench/utils/system.py | 4 +- setup.py | 12 +- 45 files changed, 1378 insertions(+), 933 deletions(-) delete mode 100644 bench/patches/v3/__init__.py delete mode 100644 bench/patches/v3/celery_to_rq.py delete mode 100644 bench/patches/v3/deprecate_old_config.py delete mode 100644 bench/patches/v3/redis_bind_ip.py delete mode 100644 bench/patches/v4/__init__.py delete mode 100644 bench/patches/v4/install_yarn.py delete mode 100644 bench/patches/v4/update_node.py delete mode 100644 bench/patches/v4/update_socketio.py diff --git a/.flake8 b/.flake8 index 940e8e55..47c45d7e 100644 --- a/.flake8 +++ b/.flake8 @@ -32,5 +32,6 @@ ignore = E131, # continuation line unaligned for hanging indent E123, # closing bracket does not match indentation of opening bracket's line E101, # ensured by use of black + B009, # allow usage of getattr max-line-length = 200 diff --git a/bench/app.py b/bench/app.py index 529e942a..6406260e 100755 --- a/bench/app.py +++ b/bench/app.py @@ -11,7 +11,6 @@ import typing from collections import OrderedDict from datetime import date from urllib.parse import urlparse -import os # imports - third party imports import click @@ -21,6 +20,7 @@ from git import Repo import bench from bench.exceptions import NotInBenchDirectoryError from bench.utils import ( + UNSET_ARG, fetch_details_from_tag, get_available_folder_name, is_bench_directory, @@ -29,10 +29,7 @@ from bench.utils import ( log, run_frappe_cmd, ) -from bench.utils.bench import ( - build_assets, - install_python_dev_dependencies, -) +from bench.utils.bench import build_assets, install_python_dev_dependencies from bench.utils.render import step if typing.TYPE_CHECKING: @@ -46,18 +43,18 @@ class AppMeta: def __init__(self, name: str, branch: str = None, to_clone: bool = True): """ name (str): This could look something like - 1. https://github.com/frappe/healthcare.git - 2. git@github.com:frappe/healthcare.git - 3. frappe/healthcare@develop - 4. healthcare - 5. healthcare@develop, healthcare@v13.12.1 + 1. https://github.com/frappe/healthcare.git + 2. git@github.com:frappe/healthcare.git + 3. frappe/healthcare@develop + 4. healthcare + 5. healthcare@develop, healthcare@v13.12.1 References for Version Identifiers: * https://www.python.org/dev/peps/pep-0440/#version-specifiers * https://docs.npmjs.com/about-semantic-versioning class Healthcare(AppConfig): - dependencies = [{"frappe/erpnext": "~13.17.0"}] + dependencies = [{"frappe/erpnext": "~13.17.0"}] """ self.name = name.rstrip("/") self.remote_server = "github.com" @@ -76,9 +73,7 @@ class AppMeta: def setup_details(self): # fetch meta from installed apps - if self.bench and os.path.exists( - os.path.join(self.bench.name, "apps", self.name) - ): + if self.bench and os.path.exists(os.path.join(self.bench.name, "apps", self.name)): self.mount_path = os.path.join(self.bench.name, "apps", self.name) self.from_apps = True self._setup_details_from_mounted_disk() @@ -98,9 +93,7 @@ class AppMeta: self._setup_details_from_name_tag() if self.git_repo: - self.app_name = os.path.basename( - os.path.normpath(self.git_repo.working_tree_dir) - ) + self.app_name = os.path.basename(os.path.normpath(self.git_repo.working_tree_dir)) else: self.app_name = self.repo @@ -203,7 +196,9 @@ class App(AppMeta): log(f"App deleted from {active_app_path}") else: archived_path = os.path.join("archived", "apps") - archived_name = get_available_folder_name(f"{self.repo}-{date.today()}", archived_path) + archived_name = get_available_folder_name( + f"{self.repo}-{date.today()}", archived_path + ) archived_app_path = os.path.join(archived_path, archived_name) shutil.move(active_app_path, archived_app_path) @@ -239,7 +234,7 @@ class App(AppMeta): verbose=verbose, skip_assets=skip_assets, restart_bench=restart_bench, - resolution=self.local_resolution + resolution=self.local_resolution, ) @step(title="Cloning and installing {repo}", success="App {repo} Installed") @@ -255,7 +250,7 @@ class App(AppMeta): from bench.utils.app import get_required_deps, required_apps_from_hooks if self.on_disk: - required_deps = os.path.join(self.mount_path, self.repo,'hooks.py') + required_deps = os.path.join(self.mount_path, self.repo, "hooks.py") try: return required_apps_from_hooks(required_deps, local=True) except IndexError: @@ -278,7 +273,6 @@ class App(AppMeta): ) - def make_resolution_plan(app: App, bench: "Bench"): """ decide what apps and versions to install and in what order @@ -303,7 +297,7 @@ def get_excluded_apps(bench_path="."): try: with open(os.path.join(bench_path, "sites", "excluded_apps.txt")) as f: return f.read().strip().split("\n") - except IOError: + except OSError: return [] @@ -366,7 +360,9 @@ def get_app( resolution = make_resolution_plan(app, bench) click.secho("Following apps will be installed", fg="bright_blue") for idx, app in enumerate(reversed(resolution.values()), start=1): - print(f"{idx}. {app.name} {f'(required by {app.required_by})' if app.required_by else ''}") + print( + f"{idx}. {app.name} {f'(required by {app.required_by})' if app.required_by else ''}" + ) if "frappe" in resolution: # Todo: Make frappe a terminal dependency for all frappe apps. @@ -385,7 +381,7 @@ def get_app( init( path=bench_path, frappe_path=frappe_path, - frappe_branch=frappe_branch if frappe_branch else branch, + frappe_branch=frappe_branch or branch, ) os.chdir(bench_path) bench_setup = True @@ -458,22 +454,27 @@ def install_resolved_deps( installed_branch = bench.apps.states[repo_name]["resolution"]["branch"].strip() except Exception: installed_branch = ( - subprocess. - check_output("git rev-parse --abbrev-ref HEAD", shell=True, cwd=path_to_app) + subprocess.check_output( + "git rev-parse --abbrev-ref HEAD", shell=True, cwd=path_to_app + ) .decode("utf-8") .rstrip() - ) + ) try: if app.tag is None: current_remote = ( - subprocess.check_output(f"git config branch.{installed_branch}.remote", shell=True, cwd=path_to_app) + subprocess.check_output( + f"git config branch.{installed_branch}.remote", shell=True, cwd=path_to_app + ) .decode("utf-8") .rstrip() ) default_branch = ( subprocess.check_output( - f"git symbolic-ref refs/remotes/{current_remote}/HEAD", shell=True, cwd=path_to_app + f"git symbolic-ref refs/remotes/{current_remote}/HEAD", + shell=True, + cwd=path_to_app, ) .decode("utf-8") .rsplit("/")[-1] @@ -485,7 +486,7 @@ def install_resolved_deps( except Exception: is_compatible = False - prefix = 'C' if is_compatible else 'Inc' + prefix = "C" if is_compatible else "Inc" click.secho( f"{prefix}ompatible version of {repo_name} is already installed", fg="green" if is_compatible else "red", @@ -503,14 +504,15 @@ def install_resolved_deps( def new_app(app, no_git=None, bench_path="."): if bench.FRAPPE_VERSION in (0, None): - raise NotInBenchDirectoryError(f"{os.path.realpath(bench_path)} is not a valid bench directory.") + raise NotInBenchDirectoryError( + f"{os.path.realpath(bench_path)} is not a valid bench directory." + ) # For backwards compatibility app = app.lower().replace(" ", "_").replace("-", "_") if app[0].isdigit() or "." in app: click.secho( - "App names cannot start with numbers(digits) or have dot(.) in them", - fg="red" + "App names cannot start with numbers(digits) or have dot(.) in them", fg="red" ) return @@ -535,7 +537,7 @@ def install_app( no_cache=False, restart_bench=True, skip_assets=False, - resolution=[] + resolution=UNSET_ARG, ): import bench.cli as bench_cli from bench.bench import Bench @@ -544,6 +546,9 @@ def install_app( click.secho(install_text, fg="yellow") logger.log(install_text) + if resolution == UNSET_ARG: + resolution = [] + bench = Bench(bench_path) conf = bench.conf @@ -553,7 +558,9 @@ def install_app( app_path = os.path.realpath(os.path.join(bench_path, "apps", app)) - bench.run(f"{bench.python} -m pip install {quiet_flag} --upgrade -e {app_path} {cache_flag}") + bench.run( + f"{bench.python} -m pip install {quiet_flag} --upgrade -e {app_path} {cache_flag}" + ) if conf.get("developer_mode"): install_python_dev_dependencies(apps=app, bench_path=bench_path, verbose=verbose) diff --git a/bench/bench.py b/bench/bench.py index 03496ce0..1f79f190 100644 --- a/bench/bench.py +++ b/bench/bench.py @@ -13,6 +13,7 @@ import bench from bench.exceptions import AppNotInstalledError, InvalidRemoteException from bench.config.common_site_config import setup_config from bench.utils import ( + UNSET_ARG, paths_in_bench, exec_cmd, is_bench_directory, @@ -141,8 +142,7 @@ class Bench(Base, Validator): @step(title="Reloading Bench Processes", success="Bench Processes Reloaded") def reload(self, web=False, supervisor=True, systemd=True): - """If web is True, only web workers are restarted - """ + """If web is True, only web workers are restarted""" conf = self.conf if conf.get("developer_mode"): @@ -153,15 +153,17 @@ class Bench(Base, Validator): restart_systemd_processes(bench_path=self.name, web_workers=web) def get_installed_apps(self) -> List: - """Returns list of installed apps on bench, not in excluded_apps.txt - """ + """Returns list of installed apps on bench, not in excluded_apps.txt""" try: installed_packages = get_cmd_output(f"{self.python} -m pip freeze", cwd=self.name) except Exception: installed_packages = [] - is_installed = lambda app: app in installed_packages - return [app for app in self.apps if app not in self.excluded_apps and is_installed(app)] + return [ + app + for app in self.apps + if app not in self.excluded_apps and app in installed_packages + ] class BenchApps(MutableSequence): @@ -174,18 +176,20 @@ class BenchApps(MutableSequence): def set_states(self): try: - with open(self.states_path, "r") as f: + with open(self.states_path) as f: self.states = json.loads(f.read() or "{}") except FileNotFoundError: self.states = {} def update_apps_states( - self, - app_dir: str = None, - app_name: Union[str, None] = None, - branch: Union[str, None] = None, - required: List = [], + self, + app_dir: str = None, + app_name: Union[str, None] = None, + branch: Union[str, None] = None, + required: List = UNSET_ARG, ): + if required == UNSET_ARG: + required = [] if self.apps and not os.path.exists(self.states_path): # idx according to apps listed in apps.txt (backwards compatibility) # Keeping frappe as the first app. @@ -198,13 +202,10 @@ class BenchApps(MutableSequence): print("Found existing apps updating states...") for idx, app in enumerate(self.apps, start=1): self.states[app] = { - "resolution": { - "commit_hash": None, - "branch": None - }, - "required": required, - "idx": idx, - "version": get_current_version(app, self.bench.name), + "resolution": {"commit_hash": None, "branch": None}, + "required": required, + "idx": idx, + "version": get_current_version(app, self.bench.name), } apps_to_remove = [] @@ -224,21 +225,21 @@ class BenchApps(MutableSequence): app_dir = os.path.join(self.apps_path, app_dir) if not branch: branch = ( - subprocess - .check_output("git rev-parse --abbrev-ref HEAD", shell=True, cwd=app_dir) - .decode("utf-8") - .rstrip() - ) + subprocess.check_output("git rev-parse --abbrev-ref HEAD", shell=True, cwd=app_dir) + .decode("utf-8") + .rstrip() + ) - commit_hash = subprocess.check_output(f"git rev-parse {branch}", shell=True, cwd=app_dir).decode("utf-8").rstrip() + commit_hash = ( + subprocess.check_output(f"git rev-parse {branch}", shell=True, cwd=app_dir) + .decode("utf-8") + .rstrip() + ) self.states[app_name] = { - "resolution": { - "commit_hash":commit_hash, - "branch": branch - }, - "required":required, - "idx":len(self.states) + 1, + "resolution": {"commit_hash": commit_hash, "branch": branch}, + "required": required, + "idx": len(self.states) + 1, "version": version, } @@ -250,18 +251,17 @@ class BenchApps(MutableSequence): app_name: Union[str, None] = None, app_dir: Union[str, None] = None, branch: Union[str, None] = None, - required: List = [] + required: List = UNSET_ARG, ): + if required == UNSET_ARG: + required = [] self.initialize_apps() with open(self.bench.apps_txt, "w") as f: f.write("\n".join(self.apps)) self.update_apps_states( - app_name=app_name, - app_dir=app_dir, - branch=branch, - required=required + app_name=app_name, app_dir=app_dir, branch=branch, required=required ) def initialize_apps(self): @@ -277,17 +277,17 @@ class BenchApps(MutableSequence): self.apps = [] def __getitem__(self, key): - """ retrieves an item by its index, key""" + """retrieves an item by its index, key""" return self.apps[key] def __setitem__(self, key, value): - """ set the item at index, key, to value """ + """set the item at index, key, to value""" # should probably not be allowed # self.apps[key] = value raise NotImplementedError def __delitem__(self, key): - """ removes the item at index, key """ + """removes the item at index, key""" # TODO: uninstall and delete app from bench del self.apps[key] @@ -295,7 +295,7 @@ class BenchApps(MutableSequence): return len(self.apps) def insert(self, key, value): - """ add an item, value, at index, key. """ + """add an item, value, at index, key.""" # TODO: fetch and install app to bench self.apps.insert(key, value) @@ -382,8 +382,7 @@ class BenchSetup(Base): @step(title="Updating pip", success="Updated pip") def pip(self, verbose=False): - """Updates env pip; assumes that env is setup - """ + """Updates env pip; assumes that env is setup""" import bench.cli verbose = bench.cli.verbose or verbose @@ -428,8 +427,7 @@ class BenchSetup(Base): @job(title="Setting Up Bench Dependencies", success="Bench Dependencies Set Up") def requirements(self, apps=None): - """Install and upgrade specified / all installed apps on given Bench - """ + """Install and upgrade specified / all installed apps on given Bench""" from bench.app import App apps = apps or self.bench.apps @@ -445,8 +443,7 @@ class BenchSetup(Base): ) def python(self, apps=None): - """Install and upgrade Python dependencies for specified / all installed apps on given Bench - """ + """Install and upgrade Python dependencies for specified / all installed apps on given Bench""" import bench.cli apps = apps or self.bench.apps @@ -461,8 +458,7 @@ class BenchSetup(Base): self.run(f"{self.bench.python} -m pip install {quiet_flag} --upgrade -e {app_path}") def node(self, apps=None): - """Install and upgrade Node dependencies for specified / all apps on given Bench - """ + """Install and upgrade Node dependencies for specified / all apps on given Bench""" from bench.utils.bench import update_node_packages return update_node_packages(bench_path=self.bench.name, apps=apps) diff --git a/bench/cli.py b/bench/cli.py index b874532a..b38aea30 100755 --- a/bench/cli.py +++ b/bench/cli.py @@ -35,7 +35,7 @@ from bench.utils.bench import get_env_cmd dynamic_feed = False verbose = False is_envvar_warn_set = None -from_command_line = False # set when commands are executed via the CLI +from_command_line = False # set when commands are executed via the CLI bench.LOG_BUFFER = [] change_uid_msg = "You should not run this command as root" @@ -104,7 +104,9 @@ def cli(): if ( not in_bench and len(sys.argv) > 1 - and not argv.intersection({"init", "find", "src", "drop", "get", "get-app", "--version"}) + and not argv.intersection( + {"init", "find", "src", "drop", "get", "get-app", "--version"} + ) and not cmd_requires_root() ): log("Command not being executed in bench directory", level=3) @@ -201,7 +203,7 @@ def frappe_cmd(bench_path="."): def get_cached_frappe_commands(): if os.path.exists(bench_cache_file): - command_dump = open(bench_cache_file, "r").read() or "[]" + command_dump = open(bench_cache_file).read() or "[]" return set(json.loads(command_dump)) return set() @@ -238,6 +240,7 @@ def change_working_directory(): def setup_clear_cache(): from copy import copy + f = copy(os.chdir) def _chdir(*args, **kwargs): diff --git a/bench/commands/__init__.py b/bench/commands/__init__.py index f44f08be..bc6805fa 100755 --- a/bench/commands/__init__.py +++ b/bench/commands/__init__.py @@ -19,10 +19,17 @@ from bench.utils.cli import ( expose_value=False, ) @click.option( - "--use-feature", is_eager=True, callback=use_experimental_feature, expose_value=False, + "--use-feature", + is_eager=True, + callback=use_experimental_feature, + expose_value=False, ) @click.option( - "-v", "--verbose", is_flag=True, callback=setup_verbosity, expose_value=False, + "-v", + "--verbose", + is_flag=True, + callback=setup_verbosity, + expose_value=False, ) def bench_command(bench_path="."): import bench diff --git a/bench/commands/config.py b/bench/commands/config.py index c3376f71..056b60cd 100644 --- a/bench/commands/config.py +++ b/bench/commands/config.py @@ -5,55 +5,64 @@ from bench.config.common_site_config import update_config, put_config import click -@click.group(help='Change bench configuration') +@click.group(help="Change bench configuration") def config(): pass -@click.command('restart_supervisor_on_update', help='Enable/Disable auto restart of supervisor processes') -@click.argument('state', type=click.Choice(['on', 'off'])) +@click.command( + "restart_supervisor_on_update", + help="Enable/Disable auto restart of supervisor processes", +) +@click.argument("state", type=click.Choice(["on", "off"])) def config_restart_supervisor_on_update(state): - update_config({'restart_supervisor_on_update': state == 'on'}) + update_config({"restart_supervisor_on_update": state == "on"}) -@click.command('restart_systemd_on_update', help='Enable/Disable auto restart of systemd units') -@click.argument('state', type=click.Choice(['on', 'off'])) +@click.command( + "restart_systemd_on_update", help="Enable/Disable auto restart of systemd units" +) +@click.argument("state", type=click.Choice(["on", "off"])) def config_restart_systemd_on_update(state): - update_config({'restart_systemd_on_update': state == 'on'}) + update_config({"restart_systemd_on_update": state == "on"}) -@click.command('dns_multitenant', help='Enable/Disable bench multitenancy on running bench update') -@click.argument('state', type=click.Choice(['on', 'off'])) +@click.command( + "dns_multitenant", help="Enable/Disable bench multitenancy on running bench update" +) +@click.argument("state", type=click.Choice(["on", "off"])) def config_dns_multitenant(state): - update_config({'dns_multitenant': state == 'on'}) + update_config({"dns_multitenant": state == "on"}) -@click.command('serve_default_site', help='Configure nginx to serve the default site on port 80') -@click.argument('state', type=click.Choice(['on', 'off'])) +@click.command( + "serve_default_site", help="Configure nginx to serve the default site on port 80" +) +@click.argument("state", type=click.Choice(["on", "off"])) def config_serve_default_site(state): - update_config({'serve_default_site': state == 'on'}) + update_config({"serve_default_site": state == "on"}) -@click.command('rebase_on_pull', help='Rebase repositories on pulling') -@click.argument('state', type=click.Choice(['on', 'off'])) +@click.command("rebase_on_pull", help="Rebase repositories on pulling") +@click.argument("state", type=click.Choice(["on", "off"])) def config_rebase_on_pull(state): - update_config({'rebase_on_pull': state == 'on'}) + update_config({"rebase_on_pull": state == "on"}) -@click.command('http_timeout', help='Set HTTP timeout') -@click.argument('seconds', type=int) +@click.command("http_timeout", help="Set HTTP timeout") +@click.argument("seconds", type=int) def config_http_timeout(seconds): - update_config({'http_timeout': seconds}) + update_config({"http_timeout": seconds}) -@click.command('set-common-config', help='Set value in common config') -@click.option('configs', '-c', '--config', multiple=True, type=(str, str)) +@click.command("set-common-config", help="Set value in common config") +@click.option("configs", "-c", "--config", multiple=True, type=(str, str)) def set_common_config(configs): import ast common_site_config = {} for key, value in configs: - if value in ('true', 'false'): + if value in ("true", "false"): value = value.title() try: value = ast.literal_eval(value) @@ -62,14 +71,17 @@ def set_common_config(configs): common_site_config[key] = value - update_config(common_site_config, bench_path='.') + update_config(common_site_config, bench_path=".") -@click.command('remove-common-config', help='Remove specific keys from current bench\'s common config') -@click.argument('keys', nargs=-1) +@click.command( + "remove-common-config", help="Remove specific keys from current bench's common config" +) +@click.argument("keys", nargs=-1) def remove_common_config(keys): from bench.bench import Bench - common_site_config = Bench('.').conf + + common_site_config = Bench(".").conf for key in keys: if key in common_site_config: del common_site_config[key] diff --git a/bench/commands/install.py b/bench/commands/install.py index 132d2255..e86e9429 100644 --- a/bench/commands/install.py +++ b/bench/commands/install.py @@ -6,9 +6,7 @@ from bench.utils.system import setup_sudoers import click -extra_vars = { - "production": True -} +extra_vars = {"production": True} @click.group(help="Install system dependencies for setting up Frappe environment") @@ -16,75 +14,100 @@ def install(): pass -@click.command('prerequisites', help="Installs pre-requisite libraries, essential tools like b2zip, htop, screen, vim, x11-fonts, python libs, cups and Redis") +@click.command( + "prerequisites", + help="Installs pre-requisite libraries, essential tools like b2zip, htop, screen, vim, x11-fonts, python libs, cups and Redis", +) def install_prerequisites(): - run_playbook('site.yml', tag='common, redis') + run_playbook("site.yml", tag="common, redis") -@click.command('mariadb', help="Install and setup MariaDB of specified version and root password") -@click.option('--mysql_root_password', '--mysql-root-password', default="") -@click.option('--version', default="10.3") +@click.command( + "mariadb", help="Install and setup MariaDB of specified version and root password" +) +@click.option("--mysql_root_password", "--mysql-root-password", default="") +@click.option("--version", default="10.3") def install_maridb(mysql_root_password, version): if mysql_root_password: - extra_vars.update({ - "mysql_root_password": mysql_root_password, - }) + extra_vars.update( + { + "mysql_root_password": mysql_root_password, + } + ) - extra_vars.update({ - "mariadb_version": version - }) + extra_vars.update({"mariadb_version": version}) - run_playbook('site.yml', extra_vars=extra_vars, tag='mariadb') + run_playbook("site.yml", extra_vars=extra_vars, tag="mariadb") -@click.command('wkhtmltopdf', help="Installs wkhtmltopdf v0.12.3 for linux") +@click.command("wkhtmltopdf", help="Installs wkhtmltopdf v0.12.3 for linux") def install_wkhtmltopdf(): - run_playbook('site.yml', extra_vars=extra_vars, tag='wkhtmltopdf') + run_playbook("site.yml", extra_vars=extra_vars, tag="wkhtmltopdf") -@click.command('nodejs', help="Installs Node.js v8") +@click.command("nodejs", help="Installs Node.js v8") def install_nodejs(): - run_playbook('site.yml', extra_vars=extra_vars, tag='nodejs') + run_playbook("site.yml", extra_vars=extra_vars, tag="nodejs") -@click.command('psutil', help="Installs psutil via pip") +@click.command("psutil", help="Installs psutil via pip") def install_psutil(): - run_playbook('site.yml', extra_vars=extra_vars, tag='psutil') + run_playbook("site.yml", extra_vars=extra_vars, tag="psutil") -@click.command('supervisor', help="Installs supervisor. If user is specified, sudoers is setup for that user") -@click.option('--user') +@click.command( + "supervisor", + help="Installs supervisor. If user is specified, sudoers is setup for that user", +) +@click.option("--user") def install_supervisor(user=None): - run_playbook('site.yml', extra_vars=extra_vars, tag='supervisor') + run_playbook("site.yml", extra_vars=extra_vars, tag="supervisor") if user: setup_sudoers(user) -@click.command('nginx', help="Installs NGINX. If user is specified, sudoers is setup for that user") -@click.option('--user') +@click.command( + "nginx", help="Installs NGINX. If user is specified, sudoers is setup for that user" +) +@click.option("--user") def install_nginx(user=None): - run_playbook('site.yml', extra_vars=extra_vars, tag='nginx') + run_playbook("site.yml", extra_vars=extra_vars, tag="nginx") if user: setup_sudoers(user) -@click.command('virtualbox', help="Installs supervisor") +@click.command("virtualbox", help="Installs supervisor") def install_virtualbox(): - run_playbook('vm_build.yml', tag='virtualbox') + run_playbook("vm_build.yml", tag="virtualbox") -@click.command('packer', help="Installs Oracle virtualbox and packer 1.2.1") +@click.command("packer", help="Installs Oracle virtualbox and packer 1.2.1") def install_packer(): - run_playbook('vm_build.yml', tag='packer') + run_playbook("vm_build.yml", tag="packer") -@click.command("fail2ban", help="Install fail2ban, an intrusion prevention software framework that protects computer servers from brute-force attacks") -@click.option('--maxretry', default=6, help="Number of matches (i.e. value of the counter) which triggers ban action on the IP.") -@click.option('--bantime', default=600, help="The counter is set to zero if no match is found within 'findtime' seconds.") -@click.option('--findtime', default=600, help='Duration (in seconds) for IP to be banned for. Negative number for "permanent" ban.') +@click.command( + "fail2ban", + help="Install fail2ban, an intrusion prevention software framework that protects computer servers from brute-force attacks", +) +@click.option( + "--maxretry", + default=6, + help="Number of matches (i.e. value of the counter) which triggers ban action on the IP.", +) +@click.option( + "--bantime", + default=600, + help="The counter is set to zero if no match is found within 'findtime' seconds.", +) +@click.option( + "--findtime", + default=600, + help='Duration (in seconds) for IP to be banned for. Negative number for "permanent" ban.', +) def install_failtoban(**kwargs): extra_vars.update(kwargs) - run_playbook('site.yml', extra_vars=extra_vars, tag='fail2ban') + run_playbook("site.yml", extra_vars=extra_vars, tag="fail2ban") install.add_command(install_prerequisites) diff --git a/bench/commands/make.py b/bench/commands/make.py index a3401c99..42ce08cd 100755 --- a/bench/commands/make.py +++ b/bench/commands/make.py @@ -37,9 +37,7 @@ import click help="Skip redis config generation if already specifying the common-site-config file", ) @click.option("--skip-assets", is_flag=True, default=False, help="Do not build assets") -@click.option( - "--install-app", help="Install particular app after initialization" -) +@click.option("--install-app", help="Install particular app after initialization") @click.option("--verbose", is_flag=True, help="Verbose output during install") def init( path, @@ -69,7 +67,7 @@ def init( try: init( path, - apps_path=apps_path, # can be used from --config flag? Maybe config file could have more info? + apps_path=apps_path, # can be used from --config flag? Maybe config file could have more info? no_procfile=no_procfile, no_backups=no_backups, frappe_path=frappe_path, @@ -130,7 +128,12 @@ def drop(path): @click.option("--branch", default=None, help="branch to checkout") @click.option("--overwrite", is_flag=True, default=False) @click.option("--skip-assets", is_flag=True, default=False, help="Do not build assets") -@click.option("--soft-link", is_flag=True, default=False, help="Create a soft link to git repo instead of clone.") +@click.option( + "--soft-link", + is_flag=True, + default=False, + help="Create a soft link to git repo instead of clone.", +) @click.option( "--init-bench", is_flag=True, default=False, help="Initialize Bench if not in one" ) @@ -163,12 +166,13 @@ def get_app( resolve_deps=resolve_deps, ) + @click.command("new-app", help="Create a new Frappe application under apps folder") @click.option( "--no-git", is_flag=True, flag_value="--no-git", - help="Do not initialize git repository for the app (available in Frappe v14+)" + help="Do not initialize git repository for the app (available in Frappe v14+)", ) @click.argument("app-name") def new_app(app_name, no_git=None): diff --git a/bench/commands/setup.py b/bench/commands/setup.py index 42672aa1..4139e8fb 100755 --- a/bench/commands/setup.py +++ b/bench/commands/setup.py @@ -14,15 +14,20 @@ def setup(): pass -@click.command("sudoers", help="Add commands to sudoers list for execution without password") +@click.command( + "sudoers", help="Add commands to sudoers list for execution without password" +) @click.argument("user") def setup_sudoers(user): from bench.utils.system import setup_sudoers + setup_sudoers(user) @click.command("nginx", help="Generate configuration files for NGINX") -@click.option("--yes", help="Yes to regeneration of nginx config file", default=False, is_flag=True) +@click.option( + "--yes", help="Yes to regeneration of nginx config file", default=False, is_flag=True +) def setup_nginx(yes=False): from bench.config.nginx import make_nginx_conf @@ -38,11 +43,18 @@ def reload_nginx(): @click.command("supervisor", help="Generate configuration for supervisor") @click.option("--user", help="optional user argument") -@click.option("--yes", help="Yes to regeneration of supervisor config", is_flag=True, default=False) -@click.option("--skip-redis", help="Skip redis configuration", is_flag=True, default=False) +@click.option( + "--yes", help="Yes to regeneration of supervisor config", is_flag=True, default=False +) +@click.option( + "--skip-redis", help="Skip redis configuration", is_flag=True, default=False +) def setup_supervisor(user=None, yes=False, skip_redis=False): from bench.utils import get_cmd_output - from bench.config.supervisor import update_supervisord_config, generate_supervisor_config + from bench.config.supervisor import ( + update_supervisord_config, + generate_supervisor_config, + ) which("supervisorctl", raise_err=True) @@ -55,33 +67,42 @@ def setup_supervisor(user=None, yes=False, skip_redis=False): @click.command("redis", help="Generates configuration for Redis") def setup_redis(): from bench.config.redis import generate_config + generate_config(".") @click.command("fonts", help="Add Frappe fonts to system") def setup_fonts(): from bench.utils.system import setup_fonts + setup_fonts() -@click.command("production", help="Setup Frappe production environment for specific user") +@click.command( + "production", help="Setup Frappe production environment for specific user" +) @click.argument("user") @click.option("--yes", help="Yes to regeneration config", is_flag=True, default=False) def setup_production(user, yes=False): from bench.config.production_setup import setup_production + setup_production(user=user, yes=yes) @click.command("backups", help="Add cronjob for bench backups") def setup_backups(): from bench.bench import Bench + Bench(".").setup.backups() @click.command("env", help="Setup virtualenv for bench") -@click.option("--python", type = str, default = "python3", help = "Path to Python Executable.") +@click.option( + "--python", type=str, default="python3", help="Path to Python Executable." +) def setup_env(python="python3"): from bench.bench import Bench + return Bench(".").setup.env(python=python) @@ -90,7 +111,10 @@ def setup_env(python="python3"): @click.option("--force") def setup_firewall(ssh_port=None, force=False): if not force: - click.confirm(f"Setting up the firewall will block all ports except 80, 443 and {ssh_port}\nDo you want to continue?", abort=True) + click.confirm( + f"Setting up the firewall will block all ports except 80, 443 and {ssh_port}\nDo you want to continue?", + abort=True, + ) if not ssh_port: ssh_port = 22 @@ -103,7 +127,9 @@ def setup_firewall(ssh_port=None, force=False): @click.option("--force") def set_ssh_port(port, force=False): if not force: - click.confirm(f"This will change your SSH Port to {port}\nDo you want to continue?", abort=True) + click.confirm( + f"This will change your SSH Port to {port}\nDo you want to continue?", abort=True + ) run_playbook("roles/bench/tasks/change_ssh_port.yml", {"ssh_port": port}) @@ -111,35 +137,63 @@ def set_ssh_port(port, force=False): @click.command("lets-encrypt", help="Setup lets-encrypt SSL for site") @click.argument("site") @click.option("--custom-domain") -@click.option('-n', '--non-interactive', default=False, is_flag=True, help="Run command non-interactively. This flag restarts nginx and runs certbot non interactively. Shouldn't be used on 1'st attempt") +@click.option( + "-n", + "--non-interactive", + default=False, + is_flag=True, + help="Run command non-interactively. This flag restarts nginx and runs certbot non interactively. Shouldn't be used on 1'st attempt", +) def setup_letsencrypt(site, custom_domain, non_interactive): from bench.config.lets_encrypt import setup_letsencrypt + setup_letsencrypt(site, custom_domain, bench_path=".", interactive=not non_interactive) -@click.command("wildcard-ssl", help="Setup wildcard SSL certificate for multi-tenant bench") +@click.command( + "wildcard-ssl", help="Setup wildcard SSL certificate for multi-tenant bench" +) @click.argument("domain") @click.option("--email") -@click.option("--exclude-base-domain", default=False, is_flag=True, help="SSL Certificate not applicable for base domain") +@click.option( + "--exclude-base-domain", + default=False, + is_flag=True, + help="SSL Certificate not applicable for base domain", +) def setup_wildcard_ssl(domain, email, exclude_base_domain): from bench.config.lets_encrypt import setup_wildcard_ssl - setup_wildcard_ssl(domain, email, bench_path=".", exclude_base_domain=exclude_base_domain) + + setup_wildcard_ssl( + domain, email, bench_path=".", exclude_base_domain=exclude_base_domain + ) @click.command("procfile", help="Generate Procfile for bench start") def setup_procfile(): from bench.config.procfile import setup_procfile + setup_procfile(".") -@click.command("socketio", help="[DEPRECATED] Setup node dependencies for socketio server") +@click.command( + "socketio", help="[DEPRECATED] Setup node dependencies for socketio server" +) def setup_socketio(): return + @click.command("requirements") @click.option("--node", help="Update only Node packages", default=False, is_flag=True) -@click.option("--python", help="Update only Python packages", default=False, is_flag=True) -@click.option("--dev", help="Install optional python development dependencies", default=False, is_flag=True) +@click.option( + "--python", help="Update only Python packages", default=False, is_flag=True +) +@click.option( + "--dev", + help="Install optional python development dependencies", + default=False, + is_flag=True, +) @click.argument("apps", nargs=-1) def setup_requirements(node=False, python=False, dev=False, apps=None): """ @@ -162,15 +216,26 @@ def setup_requirements(node=False, python=False, dev=False, apps=None): else: from bench.utils.bench import install_python_dev_dependencies + install_python_dev_dependencies(apps=apps) if node: - click.secho("--dev flag only supports python dependencies. All node development dependencies are installed by default.", fg="yellow") + click.secho( + "--dev flag only supports python dependencies. All node development dependencies are installed by default.", + fg="yellow", + ) -@click.command("manager", help="Setup bench-manager.local site with the bench_manager app installed on it") -@click.option("--yes", help="Yes to regeneration of nginx config file", default=False, is_flag=True) -@click.option("--port", help="Port on which you want to run bench manager", default=23624) +@click.command( + "manager", + help="Setup bench-manager.local site with the bench_manager app installed on it", +) +@click.option( + "--yes", help="Yes to regeneration of nginx config file", default=False, is_flag=True +) +@click.option( + "--port", help="Port on which you want to run bench manager", default=23624 +) @click.option("--domain", help="Domain on which you want to run bench manager") def setup_manager(yes=False, port=23624, domain=None): from bench.bench import Bench @@ -194,10 +259,14 @@ def setup_manager(yes=False, port=23624, domain=None): bench_path = "." bench = Bench(bench_path) - if bench.conf.get("restart_supervisor_on_update") or bench.conf.get("restart_systemd_on_update"): + if bench.conf.get("restart_supervisor_on_update") or bench.conf.get( + "restart_systemd_on_update" + ): # implicates a production setup or so I presume if not domain: - print("Please specify the site name on which you want to host bench-manager using the 'domain' flag") + print( + "Please specify the site name on which you want to host bench-manager using the 'domain' flag" + ) sys.exit(1) if domain not in bench.sites: @@ -209,6 +278,7 @@ def setup_manager(yes=False, port=23624, domain=None): @click.command("config", help="Generate or over-write sites/common_site_config.json") def setup_config(): from bench.config.common_site_config import setup_config + setup_config(".") @@ -224,6 +294,7 @@ def add_domain(domain, site=None, ssl_certificate=None, ssl_certificate_key=None sys.exit(1) from bench.config.site_config import add_domain + add_domain(site, domain, ssl_certificate, ssl_certificate_key, bench_path=".") @@ -236,10 +307,14 @@ def remove_domain(domain, site=None): sys.exit(1) from bench.config.site_config import remove_domain + remove_domain(site, domain, bench_path=".") -@click.command("sync-domains", help="Check if there is a change in domains. If yes, updates the domains list.") +@click.command( + "sync-domains", + help="Check if there is a change in domains. If yes, updates the domains list.", +) @click.option("--domain", multiple=True) @click.option("--site", prompt=True) def sync_domains(domain=None, site=None): @@ -254,6 +329,7 @@ def sync_domains(domain=None, site=None): sys.exit(1) from bench.config.site_config import sync_domains + changed = sync_domains(site, domains, bench_path=".") # if changed, success, else failure @@ -275,24 +351,53 @@ def setup_roles(role, **kwargs): run_playbook("site.yml", extra_vars=extra_vars) -@click.command("fail2ban", help="Setup fail2ban, an intrusion prevention software framework that protects computer servers from brute-force attacks") -@click.option("--maxretry", default=6, help="Number of matches (i.e. value of the counter) which triggers ban action on the IP. Default is 6 seconds" ) -@click.option("--bantime", default=600, help="Duration (in seconds) for IP to be banned for. Negative number for 'permanent' ban. Default is 600 seconds") -@click.option("--findtime", default=600, help="The counter is set to zero if match found within 'findtime' seconds doesn't exceed 'maxretry'. Default is 600 seconds") +@click.command( + "fail2ban", + help="Setup fail2ban, an intrusion prevention software framework that protects computer servers from brute-force attacks", +) +@click.option( + "--maxretry", + default=6, + help="Number of matches (i.e. value of the counter) which triggers ban action on the IP. Default is 6 seconds", +) +@click.option( + "--bantime", + default=600, + help="Duration (in seconds) for IP to be banned for. Negative number for 'permanent' ban. Default is 600 seconds", +) +@click.option( + "--findtime", + default=600, + help="The counter is set to zero if match found within 'findtime' seconds doesn't exceed 'maxretry'. Default is 600 seconds", +) def setup_nginx_proxy_jail(**kwargs): run_playbook("roles/fail2ban/tasks/configure_nginx_jail.yml", extra_vars=kwargs) @click.command("systemd", help="Generate configuration for systemd") @click.option("--user", help="Optional user argument") -@click.option("--yes", help="Yes to regeneration of systemd config files", is_flag=True, default=False) +@click.option( + "--yes", + help="Yes to regeneration of systemd config files", + is_flag=True, + default=False, +) @click.option("--stop", help="Stop bench services", is_flag=True, default=False) @click.option("--create-symlinks", help="Create Symlinks", is_flag=True, default=False) @click.option("--delete-symlinks", help="Delete Symlinks", is_flag=True, default=False) -def setup_systemd(user=None, yes=False, stop=False, create_symlinks=False, delete_symlinks=False): +def setup_systemd( + user=None, yes=False, stop=False, create_symlinks=False, delete_symlinks=False +): from bench.config.systemd import generate_systemd_config - generate_systemd_config(bench_path=".", user=user, yes=yes, - stop=stop, create_symlinks=create_symlinks, delete_symlinks=delete_symlinks) + + generate_systemd_config( + bench_path=".", + user=user, + yes=yes, + stop=stop, + create_symlinks=create_symlinks, + delete_symlinks=delete_symlinks, + ) setup.add_command(setup_sudoers) diff --git a/bench/commands/update.py b/bench/commands/update.py index 5d97d1d8..d03c0f86 100755 --- a/bench/commands/update.py +++ b/bench/commands/update.py @@ -6,43 +6,96 @@ from bench.app import pull_apps from bench.utils.bench import post_upgrade, patch_sites, build_assets -@click.command('update', help="Performs an update operation on current bench. Without any flags will backup, pull, setup requirements, build, run patches and restart bench. Using specific flags will only do certain tasks instead of all") -@click.option('--pull', is_flag=True, help="Pull updates for all the apps in bench") -@click.option('--apps', type=str) -@click.option('--patch', is_flag=True, help="Run migrations for all sites in the bench") -@click.option('--build', is_flag=True, help="Build JS and CSS assets for the bench") -@click.option('--requirements', is_flag=True, help="Update requirements. If run alone, equivalent to `bench setup requirements`") -@click.option('--restart-supervisor', is_flag=True, help="Restart supervisor processes after update") -@click.option('--restart-systemd', is_flag=True, help="Restart systemd units after update") -@click.option('--no-backup', is_flag=True, help="If this flag is set, sites won't be backed up prior to updates. Note: This is not recommended in production.") -@click.option('--no-compile', is_flag=True, help="If set, Python bytecode won't be compiled before restarting the processes") -@click.option('--force', is_flag=True, help="Forces major version upgrades") -@click.option('--reset', is_flag=True, help="Hard resets git branch's to their new states overriding any changes and overriding rebase on pull") -def update(pull, apps, patch, build, requirements, restart_supervisor, restart_systemd, no_backup, no_compile, force, reset): +@click.command( + "update", + help="Performs an update operation on current bench. Without any flags will backup, pull, setup requirements, build, run patches and restart bench. Using specific flags will only do certain tasks instead of all", +) +@click.option("--pull", is_flag=True, help="Pull updates for all the apps in bench") +@click.option("--apps", type=str) +@click.option("--patch", is_flag=True, help="Run migrations for all sites in the bench") +@click.option("--build", is_flag=True, help="Build JS and CSS assets for the bench") +@click.option( + "--requirements", + is_flag=True, + help="Update requirements. If run alone, equivalent to `bench setup requirements`", +) +@click.option( + "--restart-supervisor", is_flag=True, help="Restart supervisor processes after update" +) +@click.option( + "--restart-systemd", is_flag=True, help="Restart systemd units after update" +) +@click.option( + "--no-backup", + is_flag=True, + help="If this flag is set, sites won't be backed up prior to updates. Note: This is not recommended in production.", +) +@click.option( + "--no-compile", + is_flag=True, + help="If set, Python bytecode won't be compiled before restarting the processes", +) +@click.option("--force", is_flag=True, help="Forces major version upgrades") +@click.option( + "--reset", + is_flag=True, + help="Hard resets git branch's to their new states overriding any changes and overriding rebase on pull", +) +def update( + pull, + apps, + patch, + build, + requirements, + restart_supervisor, + restart_systemd, + no_backup, + no_compile, + force, + reset, +): from bench.utils.bench import update - update(pull=pull, apps=apps, patch=patch, build=build, requirements=requirements, restart_supervisor=restart_supervisor, restart_systemd=restart_systemd, backup=not no_backup, compile=not no_compile, force=force, reset=reset) + + update( + pull=pull, + apps=apps, + patch=patch, + build=build, + requirements=requirements, + restart_supervisor=restart_supervisor, + restart_systemd=restart_systemd, + backup=not no_backup, + compile=not no_compile, + force=force, + reset=reset, + ) -@click.command('retry-upgrade', help="Retry a failed upgrade") -@click.option('--version', default=5) +@click.command("retry-upgrade", help="Retry a failed upgrade") +@click.option("--version", default=5) def retry_upgrade(version): pull_apps() patch_sites() build_assets() - post_upgrade(version-1, version) + post_upgrade(version - 1, version) -@click.command('switch-to-branch', help="Switch all apps to specified branch, or specify apps separated by space") -@click.argument('branch') -@click.argument('apps', nargs=-1) -@click.option('--upgrade',is_flag=True) +@click.command( + "switch-to-branch", + help="Switch all apps to specified branch, or specify apps separated by space", +) +@click.argument("branch") +@click.argument("apps", nargs=-1) +@click.option("--upgrade", is_flag=True) def switch_to_branch(branch, apps, upgrade=False): from bench.utils.app import switch_to_branch + switch_to_branch(branch=branch, apps=list(apps), upgrade=upgrade) -@click.command('switch-to-develop') +@click.command("switch-to-develop") def switch_to_develop(upgrade=False): "Switch frappe and erpnext to develop branch" from bench.utils.app import switch_to_develop - switch_to_develop(apps=['frappe', 'erpnext']) + + switch_to_develop(apps=["frappe", "erpnext"]) diff --git a/bench/commands/utils.py b/bench/commands/utils.py index 35952b3e..c723b4bb 100644 --- a/bench/commands/utils.py +++ b/bench/commands/utils.py @@ -6,162 +6,200 @@ import sys import click -@click.command('start', help="Start Frappe development processes") -@click.option('--no-dev', is_flag=True, default=False) -@click.option('--no-prefix', is_flag=True, default=False, help="Hide process name from bench start log") -@click.option('--concurrency', '-c', type=str) -@click.option('--procfile', '-p', type=str) -@click.option('--man', '-m', help="Process Manager of your choice ;)") +@click.command("start", help="Start Frappe development processes") +@click.option("--no-dev", is_flag=True, default=False) +@click.option( + "--no-prefix", + is_flag=True, + default=False, + help="Hide process name from bench start log", +) +@click.option("--concurrency", "-c", type=str) +@click.option("--procfile", "-p", type=str) +@click.option("--man", "-m", help="Process Manager of your choice ;)") def start(no_dev, concurrency, procfile, no_prefix, man): from bench.utils.system import start - start(no_dev=no_dev, concurrency=concurrency, procfile=procfile, no_prefix=no_prefix, procman=man) + + start( + no_dev=no_dev, + concurrency=concurrency, + procfile=procfile, + no_prefix=no_prefix, + procman=man, + ) -@click.command('restart', help="Restart supervisor processes or systemd units") -@click.option('--web', is_flag=True, default=False) -@click.option('--supervisor', is_flag=True, default=False) -@click.option('--systemd', is_flag=True, default=False) +@click.command("restart", help="Restart supervisor processes or systemd units") +@click.option("--web", is_flag=True, default=False) +@click.option("--supervisor", is_flag=True, default=False) +@click.option("--systemd", is_flag=True, default=False) def restart(web, supervisor, systemd): from bench.bench import Bench + if not systemd and not web: supervisor = True Bench(".").reload(web, supervisor, systemd) -@click.command('set-nginx-port', help="Set NGINX port for site") -@click.argument('site') -@click.argument('port', type=int) +@click.command("set-nginx-port", help="Set NGINX port for site") +@click.argument("site") +@click.argument("port", type=int) def set_nginx_port(site, port): from bench.config.site_config import set_nginx_port + set_nginx_port(site, port) -@click.command('set-ssl-certificate', help="Set SSL certificate path for site") -@click.argument('site') -@click.argument('ssl-certificate-path') +@click.command("set-ssl-certificate", help="Set SSL certificate path for site") +@click.argument("site") +@click.argument("ssl-certificate-path") def set_ssl_certificate(site, ssl_certificate_path): from bench.config.site_config import set_ssl_certificate + set_ssl_certificate(site, ssl_certificate_path) -@click.command('set-ssl-key', help="Set SSL certificate private key path for site") -@click.argument('site') -@click.argument('ssl-certificate-key-path') +@click.command("set-ssl-key", help="Set SSL certificate private key path for site") +@click.argument("site") +@click.argument("ssl-certificate-key-path") def set_ssl_certificate_key(site, ssl_certificate_key_path): from bench.config.site_config import set_ssl_certificate_key + set_ssl_certificate_key(site, ssl_certificate_key_path) -@click.command('set-url-root', help="Set URL root for site") -@click.argument('site') -@click.argument('url-root') +@click.command("set-url-root", help="Set URL root for site") +@click.argument("site") +@click.argument("url-root") def set_url_root(site, url_root): from bench.config.site_config import set_url_root + set_url_root(site, url_root) -@click.command('set-mariadb-host', help="Set MariaDB host for bench") -@click.argument('host') +@click.command("set-mariadb-host", help="Set MariaDB host for bench") +@click.argument("host") def set_mariadb_host(host): from bench.utils.bench import set_mariadb_host + set_mariadb_host(host) -@click.command('set-redis-cache-host', help="Set Redis cache host for bench") -@click.argument('host') +@click.command("set-redis-cache-host", help="Set Redis cache host for bench") +@click.argument("host") def set_redis_cache_host(host): """ Usage: bench set-redis-cache-host localhost:6379/1 """ from bench.utils.bench import set_redis_cache_host + set_redis_cache_host(host) -@click.command('set-redis-queue-host', help="Set Redis queue host for bench") -@click.argument('host') +@click.command("set-redis-queue-host", help="Set Redis queue host for bench") +@click.argument("host") def set_redis_queue_host(host): """ Usage: bench set-redis-queue-host localhost:6379/2 """ from bench.utils.bench import set_redis_queue_host + set_redis_queue_host(host) -@click.command('set-redis-socketio-host', help="Set Redis socketio host for bench") -@click.argument('host') +@click.command("set-redis-socketio-host", help="Set Redis socketio host for bench") +@click.argument("host") def set_redis_socketio_host(host): """ Usage: bench set-redis-socketio-host localhost:6379/3 """ from bench.utils.bench import set_redis_socketio_host + set_redis_socketio_host(host) - -@click.command('download-translations', help="Download latest translations") +@click.command("download-translations", help="Download latest translations") def download_translations(): from bench.utils.translation import download_translations_p + download_translations_p() -@click.command('renew-lets-encrypt', help="Sets Up latest cron and Renew Let's Encrypt certificate") +@click.command( + "renew-lets-encrypt", help="Sets Up latest cron and Renew Let's Encrypt certificate" +) def renew_lets_encrypt(): from bench.config.lets_encrypt import renew_certs + renew_certs() -@click.command('backup', help="Backup single site") -@click.argument('site') +@click.command("backup", help="Backup single site") +@click.argument("site") def backup_site(site): from bench.bench import Bench from bench.utils.system import backup_site + if site not in Bench(".").sites: - print(f'Site `{site}` not found') + print(f"Site `{site}` not found") sys.exit(1) - backup_site(site, bench_path='.') + backup_site(site, bench_path=".") -@click.command('backup-all-sites', help="Backup all sites in current bench") +@click.command("backup-all-sites", help="Backup all sites in current bench") def backup_all_sites(): from bench.utils.system import backup_all_sites - backup_all_sites(bench_path='.') + + backup_all_sites(bench_path=".") -@click.command('disable-production', help="Disables production environment for the bench.") +@click.command( + "disable-production", help="Disables production environment for the bench." +) def disable_production(): from bench.config.production_setup import disable_production - disable_production(bench_path='.') + + disable_production(bench_path=".") -@click.command('src', help="Prints bench source folder path, which can be used as: cd `bench src`") +@click.command( + "src", help="Prints bench source folder path, which can be used as: cd `bench src`" +) def bench_src(): from bench.cli import src + print(os.path.dirname(src)) -@click.command('find', help="Finds benches recursively from location") -@click.argument('location', default='') +@click.command("find", help="Finds benches recursively from location") +@click.argument("location", default="") def find_benches(location): from bench.utils import find_benches + find_benches(directory=location) -@click.command('migrate-env', help="Migrate Virtual Environment to desired Python Version") -@click.argument('python', type=str) -@click.option('--no-backup', 'backup', is_flag=True, default=True) +@click.command( + "migrate-env", help="Migrate Virtual Environment to desired Python Version" +) +@click.argument("python", type=str) +@click.option("--no-backup", "backup", is_flag=True, default=True) def migrate_env(python, backup=True): from bench.utils.bench import migrate_env + migrate_env(python=python, backup=backup) -@click.command('generate-command-cache', help="Caches Frappe Framework commands") -def generate_command_cache(bench_path='.'): +@click.command("generate-command-cache", help="Caches Frappe Framework commands") +def generate_command_cache(bench_path="."): from bench.utils import generate_command_cache + return generate_command_cache(bench_path=bench_path) -@click.command('clear-command-cache', help="Clears Frappe Framework cached commands") -def clear_command_cache(bench_path='.'): +@click.command("clear-command-cache", help="Clears Frappe Framework cached commands") +def clear_command_cache(bench_path="."): from bench.utils import clear_command_cache + return clear_command_cache(bench_path=bench_path) diff --git a/bench/config/__init__.py b/bench/config/__init__.py index f79be532..64fdcac6 100644 --- a/bench/config/__init__.py +++ b/bench/config/__init__.py @@ -3,4 +3,5 @@ def env(): from jinja2 import Environment, PackageLoader - return Environment(loader=PackageLoader('bench.config')) + + return Environment(loader=PackageLoader("bench.config")) diff --git a/bench/config/common_site_config.py b/bench/config/common_site_config.py index be011d87..d278dedb 100644 --- a/bench/config/common_site_config.py +++ b/bench/config/common_site_config.py @@ -3,20 +3,19 @@ import getpass import json import os - - default_config = { - 'restart_supervisor_on_update': False, - 'restart_systemd_on_update': False, - 'serve_default_site': True, - 'rebase_on_pull': False, - 'frappe_user': getpass.getuser(), - 'shallow_clone': True, - 'background_workers': 1, - 'use_redis_auth': False, - 'live_reload': True + "restart_supervisor_on_update": False, + "restart_systemd_on_update": False, + "serve_default_site": True, + "rebase_on_pull": False, + "frappe_user": getpass.getuser(), + "shallow_clone": True, + "background_workers": 1, + "use_redis_auth": False, + "live_reload": True, } + def setup_config(bench_path): make_pid_folder(bench_path) bench_config = get_config(bench_path) @@ -26,52 +25,55 @@ def setup_config(bench_path): put_config(bench_config, bench_path) + def get_config(bench_path): return get_common_site_config(bench_path) + def get_common_site_config(bench_path): config_path = get_config_path(bench_path) if not os.path.exists(config_path): return {} - with open(config_path, 'r') as f: + with open(config_path) as f: return json.load(f) -def put_config(config, bench_path='.'): + +def put_config(config, bench_path="."): config_path = get_config_path(bench_path) - with open(config_path, 'w') as f: + with open(config_path, "w") as f: return json.dump(config, f, indent=1, sort_keys=True) -def update_config(new_config, bench_path='.'): + +def update_config(new_config, bench_path="."): config = get_config(bench_path=bench_path) config.update(new_config) put_config(config, bench_path=bench_path) + def get_config_path(bench_path): - return os.path.join(bench_path, 'sites', 'common_site_config.json') + return os.path.join(bench_path, "sites", "common_site_config.json") + def get_gunicorn_workers(): - '''This function will return the maximum workers that can be started depending upon - number of cpu's present on the machine''' + """This function will return the maximum workers that can be started depending upon + number of cpu's present on the machine""" import multiprocessing - return { - "gunicorn_workers": multiprocessing.cpu_count() * 2 + 1 - } + return {"gunicorn_workers": multiprocessing.cpu_count() * 2 + 1} + def update_config_for_frappe(config, bench_path): ports = make_ports(bench_path) - for key in ('redis_cache', 'redis_queue', 'redis_socketio'): + for key in ("redis_cache", "redis_queue", "redis_socketio"): if key not in config: config[key] = f"redis://localhost:{ports[key]}" - for key in ('webserver_port', 'socketio_port', 'file_watcher_port'): + for key in ("webserver_port", "socketio_port", "file_watcher_port"): if key not in config: config[key] = ports[key] - # TODO Optionally we need to add the host or domain name in case dns_multitenant is false - def make_ports(bench_path): from urllib.parse import urlparse @@ -83,7 +85,7 @@ def make_ports(bench_path): "file_watcher_port": 6787, "redis_queue": 11000, "redis_socketio": 12000, - "redis_cache": 13000 + "redis_cache": 13000, } # collect all existing ports @@ -96,7 +98,7 @@ def make_ports(bench_path): value = bench_config.get(key) # extract port from redis url - if value and (key in ('redis_cache', 'redis_queue', 'redis_socketio')): + if value and (key in ("redis_cache", "redis_queue", "redis_socketio")): value = urlparse(value).port if value: @@ -113,7 +115,8 @@ def make_ports(bench_path): return ports + def make_pid_folder(bench_path): - pids_path = os.path.join(bench_path, 'config', 'pids') + pids_path = os.path.join(bench_path, "config", "pids") if not os.path.exists(pids_path): os.makedirs(pids_path) diff --git a/bench/config/lets_encrypt.py b/bench/config/lets_encrypt.py index 4919ef73..c1c7298e 100755 --- a/bench/config/lets_encrypt.py +++ b/bench/config/lets_encrypt.py @@ -14,28 +14,31 @@ from bench.utils import exec_cmd, which from bench.utils.bench import update_common_site_config from bench.exceptions import CommandFailedError + def setup_letsencrypt(site, custom_domain, bench_path, interactive): site_path = os.path.join(bench_path, "sites", site, "site_config.json") if not os.path.exists(os.path.dirname(site_path)): - print("No site named "+site) + print("No site named " + site) return if custom_domain: domains = get_domains(site, bench_path) for d in domains: - if (isinstance(d, dict) and d['domain']==custom_domain): + if isinstance(d, dict) and d["domain"] == custom_domain: print(f"SSL for Domain {custom_domain} already exists") return - if not custom_domain in domains: + if custom_domain not in domains: print(f"No custom domain named {custom_domain} set for site") return if interactive: - click.confirm('Running this will stop the nginx service temporarily causing your sites to go offline\n' - 'Do you want to continue?', - abort=True) + click.confirm( + "Running this will stop the nginx service temporarily causing your sites to go offline\n" + "Do you want to continue?", + abort=True, + ) if not Bench(bench_path).conf.get("dns_multitenant"): print("You cannot setup SSL without DNS Multitenancy") @@ -47,56 +50,66 @@ def setup_letsencrypt(site, custom_domain, bench_path, interactive): def create_config(site, custom_domain): - config = bench.config.env().get_template('letsencrypt.cfg').render(domain=custom_domain or site) - config_path = f'/etc/letsencrypt/configs/{custom_domain or site}.cfg' + config = ( + bench.config.env() + .get_template("letsencrypt.cfg") + .render(domain=custom_domain or site) + ) + config_path = f"/etc/letsencrypt/configs/{custom_domain or site}.cfg" create_dir_if_missing(config_path) - with open(config_path, 'w') as f: + with open(config_path, "w") as f: f.write(config) def run_certbot_and_setup_ssl(site, custom_domain, bench_path, interactive=True): - service('nginx', 'stop') + service("nginx", "stop") try: - interactive = '' if interactive else '-n' - exec_cmd(f"{get_certbot_path()} {interactive} --config /etc/letsencrypt/configs/{custom_domain or site}.cfg certonly") + interactive = "" if interactive else "-n" + exec_cmd( + f"{get_certbot_path()} {interactive} --config /etc/letsencrypt/configs/{custom_domain or site}.cfg certonly" + ) except CommandFailedError: - service('nginx', 'start') + service("nginx", "start") print("There was a problem trying to setup SSL for your site") return ssl_path = f"/etc/letsencrypt/live/{custom_domain or site}/" - ssl_config = { "ssl_certificate": os.path.join(ssl_path, "fullchain.pem"), - "ssl_certificate_key": os.path.join(ssl_path, "privkey.pem") } + ssl_config = { + "ssl_certificate": os.path.join(ssl_path, "fullchain.pem"), + "ssl_certificate_key": os.path.join(ssl_path, "privkey.pem"), + } if custom_domain: remove_domain(site, custom_domain, bench_path) domains = get_domains(site, bench_path) - ssl_config['domain'] = custom_domain + ssl_config["domain"] = custom_domain domains.append(ssl_config) - update_site_config(site, { "domains": domains }, bench_path=bench_path) + update_site_config(site, {"domains": domains}, bench_path=bench_path) else: update_site_config(site, ssl_config, bench_path=bench_path) make_nginx_conf(bench_path) - service('nginx', 'start') + service("nginx", "start") def setup_crontab(): from crontab import CronTab - job_command = f'{get_certbot_path()} renew -a nginx --post-hook "systemctl reload nginx"' - job_comment = 'Renew lets-encrypt every month' + job_command = ( + f'{get_certbot_path()} renew -a nginx --post-hook "systemctl reload nginx"' + ) + job_comment = "Renew lets-encrypt every month" print(f"Setting Up cron job to {job_comment}") - system_crontab = CronTab(user='root') + system_crontab = CronTab(user="root") - for job in system_crontab.find_comment(comment=job_comment): # Removes older entries + for job in system_crontab.find_comment(comment=job_comment): # Removes older entries system_crontab.remove(job) job = system_crontab.new(command=job_command, comment=job_comment) - job.setall('0 0 */1 * *') # Run at 00:00 every day-of-month + job.setall("0 0 */1 * *") # Run at 00:00 every day-of-month system_crontab.write() @@ -109,35 +122,39 @@ def get_certbot_path(): try: return which("certbot", raise_err=True) except FileNotFoundError: - raise CommandFailedError("Certbot is not installed on your system. Please visit https://certbot.eff.org/instructions for installation instructions, then try again.") + raise CommandFailedError( + "Certbot is not installed on your system. Please visit https://certbot.eff.org/instructions for installation instructions, then try again." + ) + def renew_certs(): # Needs to be run with sudo - click.confirm('Running this will stop the nginx service temporarily causing your sites to go offline\n' - 'Do you want to continue?', - abort=True) + click.confirm( + "Running this will stop the nginx service temporarily causing your sites to go offline\n" + "Do you want to continue?", + abort=True, + ) setup_crontab() - service('nginx', 'stop') + service("nginx", "stop") exec_cmd(f"{get_certbot_path()} renew") - service('nginx', 'start') + service("nginx", "start") def setup_wildcard_ssl(domain, email, bench_path, exclude_base_domain): - def _get_domains(domain): domain_list = [domain] - if not domain.startswith('*.'): + if not domain.startswith("*."): # add wildcard caracter to domain if missing - domain_list.append(f'*.{domain}') + domain_list.append(f"*.{domain}") else: # include base domain based on flag - domain_list.append(domain.replace('*.', '')) + domain_list.append(domain.replace("*.", "")) if exclude_base_domain: - domain_list.remove(domain.replace('*.', '')) + domain_list.remove(domain.replace("*.", "")) return domain_list @@ -147,14 +164,16 @@ def setup_wildcard_ssl(domain, email, bench_path, exclude_base_domain): domain_list = _get_domains(domain.strip()) - email_param = '' + email_param = "" if email: - email_param = f'--email {email}' + email_param = f"--email {email}" try: - exec_cmd(f"{get_certbot_path()} certonly --manual --preferred-challenges=dns {email_param} \ + exec_cmd( + f"{get_certbot_path()} certonly --manual --preferred-challenges=dns {email_param} \ --server https://acme-v02.api.letsencrypt.org/directory \ - --agree-tos -d {' -d '.join(domain_list)}") + --agree-tos -d {' -d '.join(domain_list)}" + ) except CommandFailedError: print("There was a problem trying to setup SSL") @@ -165,7 +184,7 @@ def setup_wildcard_ssl(domain, email, bench_path, exclude_base_domain): "wildcard": { "domain": domain, "ssl_certificate": os.path.join(ssl_path, "fullchain.pem"), - "ssl_certificate_key": os.path.join(ssl_path, "privkey.pem") + "ssl_certificate_key": os.path.join(ssl_path, "privkey.pem"), } } @@ -174,4 +193,4 @@ def setup_wildcard_ssl(domain, email, bench_path, exclude_base_domain): make_nginx_conf(bench_path) print("Restrting Nginx service") - service('nginx', 'restart') + service("nginx", "restart") diff --git a/bench/config/nginx.py b/bench/config/nginx.py index a3117f9c..a19f6e56 100644 --- a/bench/config/nginx.py +++ b/bench/config/nginx.py @@ -17,10 +17,12 @@ def make_nginx_conf(bench_path, yes=False): conf_path = os.path.join(bench_path, "config", "nginx.conf") if not yes and os.path.exists(conf_path): - if not click.confirm('nginx.conf already exists and this will overwrite it. Do you want to continue?'): + if not click.confirm( + "nginx.conf already exists and this will overwrite it. Do you want to continue?" + ): return - template = bench.config.env().get_template('nginx.conf') + template = bench.config.env().get_template("nginx.conf") bench_path = os.path.abspath(bench_path) sites_path = os.path.join(bench_path, "sites") @@ -28,37 +30,39 @@ def make_nginx_conf(bench_path, yes=False): sites = prepare_sites(config, bench_path) bench_name = get_bench_name(bench_path) - allow_rate_limiting = config.get('allow_rate_limiting', False) + allow_rate_limiting = config.get("allow_rate_limiting", False) template_vars = { "sites_path": sites_path, "http_timeout": config.get("http_timeout"), "sites": sites, - "webserver_port": config.get('webserver_port'), - "socketio_port": config.get('socketio_port'), + "webserver_port": config.get("webserver_port"), + "socketio_port": config.get("socketio_port"), "bench_name": bench_name, "error_pages": get_error_pages(), "allow_rate_limiting": allow_rate_limiting, # for nginx map variable - "random_string": "".join(random.choice(string.ascii_lowercase) for i in range(7)) + "random_string": "".join(random.choice(string.ascii_lowercase) for i in range(7)), } if allow_rate_limiting: - template_vars.update({ - 'bench_name_hash': hashlib.sha256(bench_name).hexdigest()[:16], - 'limit_conn_shared_memory': get_limit_conn_shared_memory() - }) + template_vars.update( + { + "bench_name_hash": hashlib.sha256(bench_name).hexdigest()[:16], + "limit_conn_shared_memory": get_limit_conn_shared_memory(), + } + ) nginx_conf = template.render(**template_vars) - with open(conf_path, "w") as f: f.write(nginx_conf) + def make_bench_manager_nginx_conf(bench_path, yes=False, port=23624, domain=None): from bench.config.site_config import get_site_config - template = bench.config.env().get_template('bench_manager_nginx.conf') + template = bench.config.env().get_template("bench_manager_nginx.conf") bench_path = os.path.abspath(bench_path) sites_path = os.path.join(bench_path, "sites") @@ -72,12 +76,12 @@ def make_bench_manager_nginx_conf(bench_path, yes=False, port=23624, domain=None "bench_manager_site_name": "bench-manager.local", "sites_path": sites_path, "http_timeout": config.get("http_timeout"), - "webserver_port": config.get('webserver_port'), - "socketio_port": config.get('socketio_port'), + "webserver_port": config.get("webserver_port"), + "socketio_port": config.get("socketio_port"), "bench_name": bench_name, "error_pages": get_error_pages(), - "ssl_certificate": site_config.get('ssl_certificate'), - "ssl_certificate_key": site_config.get('ssl_certificate_key') + "ssl_certificate": site_config.get("ssl_certificate"), + "ssl_certificate_key": site_config.get("ssl_certificate_key"), } bench_manager_nginx_conf = template.render(**template_vars) @@ -85,29 +89,31 @@ def make_bench_manager_nginx_conf(bench_path, yes=False, port=23624, domain=None conf_path = os.path.join(bench_path, "config", "nginx.conf") if not yes and os.path.exists(conf_path): - click.confirm('nginx.conf already exists and bench-manager configuration will be appended to it. Do you want to continue?', - abort=True) + click.confirm( + "nginx.conf already exists and bench-manager configuration will be appended to it. Do you want to continue?", + abort=True, + ) with open(conf_path, "a") as myfile: myfile.write(bench_manager_nginx_conf) + def prepare_sites(config, bench_path): sites = { "that_use_port": [], "that_use_dns": [], "that_use_ssl": [], - "that_use_wildcard_ssl": [] + "that_use_wildcard_ssl": [], } domain_map = {} ports_in_use = {} - dns_multitenant = config.get('dns_multitenant') + dns_multitenant = config.get("dns_multitenant") shared_port_exception_found = False sites_configs = get_sites_with_config(bench_path=bench_path) - # preload all preset site ports to avoid conflicts if not dns_multitenant: @@ -119,20 +125,20 @@ def prepare_sites(config, bench_path): for site in sites_configs: if dns_multitenant: - domain = site.get('domain') + domain = site.get("domain") if domain: # when site's folder name is different than domain name - domain_map[domain] = site['name'] + domain_map[domain] = site["name"] - site_name = domain or site['name'] + site_name = domain or site["name"] - if site.get('wildcard'): + if site.get("wildcard"): sites["that_use_wildcard_ssl"].append(site_name) - if not sites.get('wildcard_ssl_certificate'): - sites["wildcard_ssl_certificate"] = site['ssl_certificate'] - sites["wildcard_ssl_certificate_key"] = site['ssl_certificate_key'] + if not sites.get("wildcard_ssl_certificate"): + sites["wildcard_ssl_certificate"] = site["ssl_certificate"] + sites["wildcard_ssl_certificate_key"] = site["ssl_certificate_key"] elif site.get("ssl_certificate") and site.get("ssl_certificate_key"): sites["that_use_ssl"].append(site) @@ -157,7 +163,6 @@ def prepare_sites(config, bench_path): sites["that_use_port"].append(site) - if not dns_multitenant and shared_port_exception_found: message = "Port conflicts found:" port_conflict_index = 0 @@ -176,11 +181,11 @@ def prepare_sites(config, bench_path): print(message) - - sites['domain_map'] = domain_map + sites["domain_map"] = domain_map return sites + def get_sites_with_config(bench_path): from bench.bench import Bench from bench.config.site_config import get_site_config @@ -188,94 +193,105 @@ def get_sites_with_config(bench_path): bench = Bench(bench_path) sites = bench.sites conf = bench.conf - dns_multitenant = conf.get('dns_multitenant') + dns_multitenant = conf.get("dns_multitenant") ret = [] for site in sites: try: site_config = get_site_config(site, bench_path=bench_path) except Exception as e: - strict_nginx = conf.get('strict_nginx') + strict_nginx = conf.get("strict_nginx") if strict_nginx: - print(f"\n\nERROR: The site config for the site {site} is broken.", + print( + f"\n\nERROR: The site config for the site {site} is broken.", "If you want this command to pass, instead of just throwing an error,", "You may remove the 'strict_nginx' flag from common_site_config.json or set it to 0", - "\n\n") + "\n\n", + ) raise e else: - print(f"\n\nWARNING: The site config for the site {site} is broken.", + print( + f"\n\nWARNING: The site config for the site {site} is broken.", "If you want this command to fail, instead of just showing a warning,", "You may add the 'strict_nginx' flag to common_site_config.json and set it to 1", - "\n\n") + "\n\n", + ) continue - ret.append({ - "name": site, - "port": site_config.get('nginx_port'), - "ssl_certificate": site_config.get('ssl_certificate'), - "ssl_certificate_key": site_config.get('ssl_certificate_key') - }) + ret.append( + { + "name": site, + "port": site_config.get("nginx_port"), + "ssl_certificate": site_config.get("ssl_certificate"), + "ssl_certificate_key": site_config.get("ssl_certificate_key"), + } + ) - if dns_multitenant and site_config.get('domains'): - for domain in site_config.get('domains'): + if dns_multitenant and site_config.get("domains"): + for domain in site_config.get("domains"): # domain can be a string or a dict with 'domain', 'ssl_certificate', 'ssl_certificate_key' if isinstance(domain, str): - domain = { 'domain': domain } + domain = {"domain": domain} - domain['name'] = site + domain["name"] = site ret.append(domain) use_wildcard_certificate(bench_path, ret) return ret + def use_wildcard_certificate(bench_path, ret): - ''' - stored in common_site_config.json as: - "wildcard": { - "domain": "*.erpnext.com", - "ssl_certificate": "/path/to/erpnext.com.cert", - "ssl_certificate_key": "/path/to/erpnext.com.key" - } - ''' + """ + stored in common_site_config.json as: + "wildcard": { + "domain": "*.erpnext.com", + "ssl_certificate": "/path/to/erpnext.com.cert", + "ssl_certificate_key": "/path/to/erpnext.com.key" + } + """ from bench.bench import Bench + config = Bench(bench_path).conf - wildcard = config.get('wildcard') + wildcard = config.get("wildcard") if not wildcard: return - domain = wildcard['domain'] - ssl_certificate = wildcard['ssl_certificate'] - ssl_certificate_key = wildcard['ssl_certificate_key'] + domain = wildcard["domain"] + ssl_certificate = wildcard["ssl_certificate"] + ssl_certificate_key = wildcard["ssl_certificate_key"] # If domain is set as "*" all domains will be included - if domain.startswith('*'): + if domain.startswith("*"): domain = domain[1:] else: - domain = '.' + domain + domain = "." + domain for site in ret: - if site.get('ssl_certificate'): + if site.get("ssl_certificate"): continue - if (site.get('domain') or site['name']).endswith(domain): + if (site.get("domain") or site["name"]).endswith(domain): # example: ends with .erpnext.com - site['ssl_certificate'] = ssl_certificate - site['ssl_certificate_key'] = ssl_certificate_key - site['wildcard'] = 1 + site["ssl_certificate"] = ssl_certificate + site["ssl_certificate_key"] = ssl_certificate_key + site["wildcard"] = 1 + def get_error_pages(): import bench - bench_app_path = os.path.abspath(bench.__path__[0]) - templates = os.path.join(bench_app_path, 'config', 'templates') - return { - 502: os.path.join(templates, '502.html') - } + bench_app_path = os.path.abspath(bench.__path__[0]) + templates = os.path.join(bench_app_path, "config", "templates") + + return {502: os.path.join(templates, "502.html")} + def get_limit_conn_shared_memory(): """Allocate 2 percent of total virtual memory as shared memory for nginx limit_conn_zone""" - total_vm = (os.sysconf('SC_PAGE_SIZE') * os.sysconf('SC_PHYS_PAGES')) / (1024 * 1024) # in MB + total_vm = (os.sysconf("SC_PAGE_SIZE") * os.sysconf("SC_PHYS_PAGES")) / ( + 1024 * 1024 + ) # in MB return int(0.02 * total_vm) diff --git a/bench/config/procfile.py b/bench/config/procfile.py index a6bb64ec..38ee5c60 100755 --- a/bench/config/procfile.py +++ b/bench/config/procfile.py @@ -13,18 +13,25 @@ from bench.bench import Bench def setup_procfile(bench_path, yes=False, skip_redis=False): config = Bench(bench_path).conf - procfile_path = os.path.join(bench_path, 'Procfile') + procfile_path = os.path.join(bench_path, "Procfile") if not yes and os.path.exists(procfile_path): - click.confirm('A Procfile already exists and this will overwrite it. Do you want to continue?', - abort=True) + click.confirm( + "A Procfile already exists and this will overwrite it. Do you want to continue?", + abort=True, + ) - procfile = bench.config.env().get_template('Procfile').render( - node=which("node") or which("nodejs"), - use_rq=use_rq(bench_path), - webserver_port=config.get('webserver_port'), - CI=os.environ.get('CI'), - skip_redis=skip_redis, - workers=config.get("workers", {})) + procfile = ( + bench.config.env() + .get_template("Procfile") + .render( + node=which("node") or which("nodejs"), + use_rq=use_rq(bench_path), + webserver_port=config.get("webserver_port"), + CI=os.environ.get("CI"), + skip_redis=skip_redis, + workers=config.get("workers", {}), + ) + ) - with open(procfile_path, 'w') as f: + with open(procfile_path, "w") as f: f.write(procfile) diff --git a/bench/config/production_setup.py b/bench/config/production_setup.py index ad233e31..4d5a4bec 100755 --- a/bench/config/production_setup.py +++ b/bench/config/production_setup.py @@ -6,7 +6,10 @@ import sys # imports - module imports import bench from bench.config.nginx import make_nginx_conf -from bench.config.supervisor import generate_supervisor_config, update_supervisord_config +from bench.config.supervisor import ( + generate_supervisor_config, + update_supervisord_config, +) from bench.config.systemd import generate_systemd_config from bench.bench import Bench from bench.utils import exec_cmd, which, get_bench_name, get_cmd_output, log @@ -28,16 +31,18 @@ def setup_production_prerequisites(): exec_cmd("bench setup role supervisor") -def setup_production(user, bench_path='.', yes=False): +def setup_production(user, bench_path=".", yes=False): print("Setting Up prerequisites...") setup_production_prerequisites() conf = Bench(bench_path).conf - if conf.get('restart_supervisor_on_update') and conf.get('restart_systemd_on_update'): - raise Exception("You cannot use supervisor and systemd at the same time. Modify your common_site_config accordingly." ) + if conf.get("restart_supervisor_on_update") and conf.get("restart_systemd_on_update"): + raise Exception( + "You cannot use supervisor and systemd at the same time. Modify your common_site_config accordingly." + ) - if conf.get('restart_systemd_on_update'): + if conf.get("restart_systemd_on_update"): print("Setting Up systemd...") generate_systemd_config(bench_path=bench_path, user=user, yes=yes) else: @@ -51,45 +56,54 @@ def setup_production(user, bench_path='.', yes=False): remove_default_nginx_configs() bench_name = get_bench_name(bench_path) - nginx_conf = f'/etc/nginx/conf.d/{bench_name}.conf' + nginx_conf = f"/etc/nginx/conf.d/{bench_name}.conf" print("Setting Up symlinks and reloading services...") - if conf.get('restart_supervisor_on_update'): + if conf.get("restart_supervisor_on_update"): supervisor_conf_extn = "ini" if is_centos7() else "conf" - supervisor_conf = os.path.join(get_supervisor_confdir(), f'{bench_name}.{supervisor_conf_extn}') + supervisor_conf = os.path.join( + get_supervisor_confdir(), f"{bench_name}.{supervisor_conf_extn}" + ) # Check if symlink exists, If not then create it. if not os.path.islink(supervisor_conf): - os.symlink(os.path.abspath(os.path.join(bench_path, 'config', 'supervisor.conf')), supervisor_conf) + os.symlink( + os.path.abspath(os.path.join(bench_path, "config", "supervisor.conf")), + supervisor_conf, + ) if not os.path.islink(nginx_conf): - os.symlink(os.path.abspath(os.path.join(bench_path, 'config', 'nginx.conf')), nginx_conf) + os.symlink( + os.path.abspath(os.path.join(bench_path, "config", "nginx.conf")), nginx_conf + ) - if conf.get('restart_supervisor_on_update'): + if conf.get("restart_supervisor_on_update"): reload_supervisor() - if os.environ.get('NO_SERVICE_RESTART'): + if os.environ.get("NO_SERVICE_RESTART"): return reload_nginx() -def disable_production(bench_path='.'): +def disable_production(bench_path="."): bench_name = get_bench_name(bench_path) conf = Bench(bench_path).conf # supervisorctl supervisor_conf_extn = "ini" if is_centos7() else "conf" - supervisor_conf = os.path.join(get_supervisor_confdir(), f'{bench_name}.{supervisor_conf_extn}') + supervisor_conf = os.path.join( + get_supervisor_confdir(), f"{bench_name}.{supervisor_conf_extn}" + ) if os.path.islink(supervisor_conf): os.unlink(supervisor_conf) - if conf.get('restart_supervisor_on_update'): + if conf.get("restart_supervisor_on_update"): reload_supervisor() # nginx - nginx_conf = f'/etc/nginx/conf.d/{bench_name}.conf' + nginx_conf = f"/etc/nginx/conf.d/{bench_name}.conf" if os.path.islink(nginx_conf): os.unlink(nginx_conf) @@ -98,10 +112,10 @@ def disable_production(bench_path='.'): def service(service_name, service_option): - if os.path.basename(which('systemctl') or '') == 'systemctl' and is_running_systemd(): + if os.path.basename(which("systemctl") or "") == "systemctl" and is_running_systemd(): exec_cmd(f"sudo systemctl {service_option} {service_name}") - elif os.path.basename(which('service') or '') == 'service': + elif os.path.basename(which("service") or "") == "service": exec_cmd(f"sudo service {service_name} {service_option}") else: @@ -115,18 +129,29 @@ def service(service_name, service_option): exec_cmd(service_manager_command) else: - log(f"No service manager found: '{service_name} {service_option}' failed to execute", level=2) + log( + f"No service manager found: '{service_name} {service_option}' failed to execute", + level=2, + ) def get_supervisor_confdir(): - possiblities = ('/etc/supervisor/conf.d', '/etc/supervisor.d/', '/etc/supervisord/conf.d', '/etc/supervisord.d') + possiblities = ( + "/etc/supervisor/conf.d", + "/etc/supervisor.d/", + "/etc/supervisord/conf.d", + "/etc/supervisord.d", + ) for possiblity in possiblities: if os.path.exists(possiblity): return possiblity def remove_default_nginx_configs(): - default_nginx_configs = ['/etc/nginx/conf.d/default.conf', '/etc/nginx/sites-enabled/default'] + default_nginx_configs = [ + "/etc/nginx/conf.d/default.conf", + "/etc/nginx/sites-enabled/default", + ] for conf_file in default_nginx_configs: if os.path.exists(conf_file): @@ -134,11 +159,17 @@ def remove_default_nginx_configs(): def is_centos7(): - return os.path.exists('/etc/redhat-release') and get_cmd_output("cat /etc/redhat-release | sed 's/Linux\ //g' | cut -d' ' -f3 | cut -d. -f1").strip() == '7' + return ( + os.path.exists("/etc/redhat-release") + and get_cmd_output( + r"cat /etc/redhat-release | sed 's/Linux\ //g' | cut -d' ' -f3 | cut -d. -f1" + ).strip() + == "7" + ) def is_running_systemd(): - with open('/proc/1/comm') as f: + with open("/proc/1/comm") as f: comm = f.read().strip() if comm == "init": return False @@ -148,41 +179,42 @@ def is_running_systemd(): def reload_supervisor(): - supervisorctl = which('supervisorctl') + supervisorctl = which("supervisorctl") try: # first try reread/update - exec_cmd(f'{supervisorctl} reread') - exec_cmd(f'{supervisorctl} update') + exec_cmd(f"{supervisorctl} reread") + exec_cmd(f"{supervisorctl} update") return except CommandFailedError: pass try: # something is wrong, so try reloading - exec_cmd(f'{supervisorctl} reload') + exec_cmd(f"{supervisorctl} reload") return except CommandFailedError: pass try: # then try restart for centos - service('supervisord', 'restart') + service("supervisord", "restart") return except CommandFailedError: pass try: # else try restart for ubuntu / debian - service('supervisor', 'restart') + service("supervisor", "restart") return except CommandFailedError: pass + def reload_nginx(): try: exec_cmd(f"sudo {which('nginx')} -t") except Exception: raise - service('nginx', 'reload') + service("nginx", "reload") diff --git a/bench/config/redis.py b/bench/config/redis.py index 5d509af3..7e4f5af8 100644 --- a/bench/config/redis.py +++ b/bench/config/redis.py @@ -15,36 +15,33 @@ def generate_config(bench_path): redis_version = get_redis_version() ports = {} - for key in ('redis_cache', 'redis_queue', 'redis_socketio'): + for key in ("redis_cache", "redis_queue", "redis_socketio"): ports[key] = urlparse(config[key]).port write_redis_config( - template_name='redis_queue.conf', + template_name="redis_queue.conf", context={ - "port": ports['redis_queue'], + "port": ports["redis_queue"], "bench_path": os.path.abspath(bench_path), - "redis_version": redis_version + "redis_version": redis_version, }, - bench_path=bench_path + bench_path=bench_path, ) write_redis_config( - template_name='redis_socketio.conf', - context={ - "port": ports['redis_socketio'], - "redis_version": redis_version - }, - bench_path=bench_path + template_name="redis_socketio.conf", + context={"port": ports["redis_socketio"], "redis_version": redis_version}, + bench_path=bench_path, ) write_redis_config( - template_name='redis_cache.conf', + template_name="redis_cache.conf", context={ - "maxmemory": config.get('cache_maxmemory', get_max_redis_memory()), - "port": ports['redis_cache'], - "redis_version": redis_version + "maxmemory": config.get("cache_maxmemory", get_max_redis_memory()), + "port": ports["redis_cache"], + "redis_version": redis_version, }, - bench_path=bench_path + bench_path=bench_path, ) # make pids folder @@ -60,9 +57,10 @@ def generate_config(bench_path): acl_rq_path = os.path.join(bench_path, "config", "redis_queue.acl") acl_redis_cache_path = os.path.join(bench_path, "config", "redis_cache.acl") acl_redis_socketio_path = os.path.join(bench_path, "config", "redis_socketio.acl") - open(acl_rq_path, 'a').close() - open(acl_redis_cache_path, 'a').close() - open(acl_redis_socketio_path, 'a').close() + open(acl_rq_path, "a").close() + open(acl_redis_cache_path, "a").close() + open(acl_redis_socketio_path, "a").close() + def write_redis_config(template_name, context, bench_path): template = bench.config.env().get_template(template_name) @@ -73,25 +71,27 @@ def write_redis_config(template_name, context, bench_path): if "pid_path" not in context: context["pid_path"] = os.path.join(context["config_path"], "pids") - with open(os.path.join(bench_path, 'config', template_name), 'w') as f: + with open(os.path.join(bench_path, "config", template_name), "w") as f: f.write(template.render(**context)) + def get_redis_version(): import semantic_version - version_string = subprocess.check_output('redis-server --version', shell=True) - version_string = version_string.decode('utf-8').strip() + version_string = subprocess.check_output("redis-server --version", shell=True) + version_string = version_string.decode("utf-8").strip() # extract version number from string - version = re.findall("\d+\.\d+", version_string) + version = re.findall(r"\d+\.\d+", version_string) if not version: return None version = semantic_version.Version(version[0], partial=True) - return float(f'{version.major}.{version.minor}') + return float(f"{version.major}.{version.minor}") + def get_max_redis_memory(): try: - max_mem = os.sysconf('SC_PAGE_SIZE') * os.sysconf('SC_PHYS_PAGES') + max_mem = os.sysconf("SC_PAGE_SIZE") * os.sysconf("SC_PHYS_PAGES") except ValueError: - max_mem = int(subprocess.check_output(['sysctl', '-n', 'hw.memsize']).strip()) - return max(50, int((max_mem / (1024. ** 2)) * 0.05)) + max_mem = int(subprocess.check_output(["sysctl", "-n", "hw.memsize"]).strip()) + return max(50, int((max_mem / (1024.0**2)) * 0.05)) diff --git a/bench/config/site_config.py b/bench/config/site_config.py index bb4cea74..b8bc01ca 100644 --- a/bench/config/site_config.py +++ b/bench/config/site_config.py @@ -4,33 +4,51 @@ import os from collections import defaultdict -def get_site_config(site, bench_path='.'): - config_path = os.path.join(bench_path, 'sites', site, 'site_config.json') +def get_site_config(site, bench_path="."): + config_path = os.path.join(bench_path, "sites", site, "site_config.json") if not os.path.exists(config_path): return {} with open(config_path) as f: return json.load(f) -def put_site_config(site, config, bench_path='.'): - config_path = os.path.join(bench_path, 'sites', site, 'site_config.json') - with open(config_path, 'w') as f: + +def put_site_config(site, config, bench_path="."): + config_path = os.path.join(bench_path, "sites", site, "site_config.json") + with open(config_path, "w") as f: return json.dump(config, f, indent=1) -def update_site_config(site, new_config, bench_path='.'): + +def update_site_config(site, new_config, bench_path="."): config = get_site_config(site, bench_path=bench_path) config.update(new_config) put_site_config(site, config, bench_path=bench_path) -def set_nginx_port(site, port, bench_path='.', gen_config=True): - set_site_config_nginx_property(site, {"nginx_port": port}, bench_path=bench_path, gen_config=gen_config) -def set_ssl_certificate(site, ssl_certificate, bench_path='.', gen_config=True): - set_site_config_nginx_property(site, {"ssl_certificate": ssl_certificate}, bench_path=bench_path, gen_config=gen_config) +def set_nginx_port(site, port, bench_path=".", gen_config=True): + set_site_config_nginx_property( + site, {"nginx_port": port}, bench_path=bench_path, gen_config=gen_config + ) -def set_ssl_certificate_key(site, ssl_certificate_key, bench_path='.', gen_config=True): - set_site_config_nginx_property(site, {"ssl_certificate_key": ssl_certificate_key}, bench_path=bench_path, gen_config=gen_config) -def set_site_config_nginx_property(site, config, bench_path='.', gen_config=True): +def set_ssl_certificate(site, ssl_certificate, bench_path=".", gen_config=True): + set_site_config_nginx_property( + site, + {"ssl_certificate": ssl_certificate}, + bench_path=bench_path, + gen_config=gen_config, + ) + + +def set_ssl_certificate_key(site, ssl_certificate_key, bench_path=".", gen_config=True): + set_site_config_nginx_property( + site, + {"ssl_certificate_key": ssl_certificate_key}, + bench_path=bench_path, + gen_config=gen_config, + ) + + +def set_site_config_nginx_property(site, config, bench_path=".", gen_config=True): from bench.config.nginx import make_nginx_conf from bench.bench import Bench @@ -40,36 +58,40 @@ def set_site_config_nginx_property(site, config, bench_path='.', gen_config=True if gen_config: make_nginx_conf(bench_path=bench_path) -def set_url_root(site, url_root, bench_path='.'): + +def set_url_root(site, url_root, bench_path="."): update_site_config(site, {"host_name": url_root}, bench_path=bench_path) -def add_domain(site, domain, ssl_certificate, ssl_certificate_key, bench_path='.'): + +def add_domain(site, domain, ssl_certificate, ssl_certificate_key, bench_path="."): domains = get_domains(site, bench_path) for d in domains: - if (isinstance(d, dict) and d['domain']==domain) or d==domain: + if (isinstance(d, dict) and d["domain"] == domain) or d == domain: print(f"Domain {domain} already exists") return if ssl_certificate_key and ssl_certificate: domain = { - 'domain' : domain, - 'ssl_certificate': ssl_certificate, - 'ssl_certificate_key': ssl_certificate_key + "domain": domain, + "ssl_certificate": ssl_certificate, + "ssl_certificate_key": ssl_certificate_key, } domains.append(domain) - update_site_config(site, { "domains": domains }, bench_path=bench_path) + update_site_config(site, {"domains": domains}, bench_path=bench_path) -def remove_domain(site, domain, bench_path='.'): + +def remove_domain(site, domain, bench_path="."): domains = get_domains(site, bench_path) for i, d in enumerate(domains): - if (isinstance(d, dict) and d['domain']==domain) or d==domain: + if (isinstance(d, dict) and d["domain"] == domain) or d == domain: domains.remove(d) break - update_site_config(site, { 'domains': domains }, bench_path=bench_path) + update_site_config(site, {"domains": domains}, bench_path=bench_path) -def sync_domains(site, domains, bench_path='.'): + +def sync_domains(site, domains, bench_path="."): """Checks if there is a change in domains. If yes, updates the domains list.""" changed = False existing_domains = get_domains_dict(get_domains(site, bench_path)) @@ -80,26 +102,28 @@ def sync_domains(site, domains, bench_path='.'): else: for d in list(existing_domains.values()): - if d != new_domains.get(d['domain']): + if d != new_domains.get(d["domain"]): changed = True break if changed: # replace existing domains with this one - update_site_config(site, { 'domains': domains }, bench_path='.') + update_site_config(site, {"domains": domains}, bench_path=".") return changed -def get_domains(site, bench_path='.'): - return get_site_config(site, bench_path=bench_path).get('domains') or [] + +def get_domains(site, bench_path="."): + return get_site_config(site, bench_path=bench_path).get("domains") or [] + def get_domains_dict(domains): domains_dict = defaultdict(dict) for d in domains: if isinstance(d, str): - domains_dict[d] = { 'domain': d } + domains_dict[d] = {"domain": d} elif isinstance(d, dict): - domains_dict[d['domain']] = d + domains_dict[d["domain"]] = d return domains_dict diff --git a/bench/config/supervisor.py b/bench/config/supervisor.py index 765e9989..d4d6b14e 100644 --- a/bench/config/supervisor.py +++ b/bench/config/supervisor.py @@ -23,44 +23,56 @@ def generate_supervisor_config(bench_path, user=None, yes=False, skip_redis=Fals user = getpass.getuser() config = Bench(bench_path).conf - template = bench.config.env().get_template('supervisor.conf') + template = bench.config.env().get_template("supervisor.conf") bench_dir = os.path.abspath(bench_path) - config = template.render(**{ - "bench_dir": bench_dir, - "sites_dir": os.path.join(bench_dir, 'sites'), - "user": user, - "use_rq": use_rq(bench_path), - "http_timeout": config.get("http_timeout", 120), - "redis_server": which('redis-server'), - "node": which('node') or which('nodejs'), - "redis_cache_config": os.path.join(bench_dir, 'config', 'redis_cache.conf'), - "redis_socketio_config": os.path.join(bench_dir, 'config', 'redis_socketio.conf'), - "redis_queue_config": os.path.join(bench_dir, 'config', 'redis_queue.conf'), - "webserver_port": config.get('webserver_port', 8000), - "gunicorn_workers": config.get('gunicorn_workers', get_gunicorn_workers()["gunicorn_workers"]), - "bench_name": get_bench_name(bench_path), - "background_workers": config.get('background_workers') or 1, - "bench_cmd": which('bench'), - "skip_redis": skip_redis, - "workers": config.get("workers", {}), - }) + config = template.render( + **{ + "bench_dir": bench_dir, + "sites_dir": os.path.join(bench_dir, "sites"), + "user": user, + "use_rq": use_rq(bench_path), + "http_timeout": config.get("http_timeout", 120), + "redis_server": which("redis-server"), + "node": which("node") or which("nodejs"), + "redis_cache_config": os.path.join(bench_dir, "config", "redis_cache.conf"), + "redis_socketio_config": os.path.join(bench_dir, "config", "redis_socketio.conf"), + "redis_queue_config": os.path.join(bench_dir, "config", "redis_queue.conf"), + "webserver_port": config.get("webserver_port", 8000), + "gunicorn_workers": config.get( + "gunicorn_workers", get_gunicorn_workers()["gunicorn_workers"] + ), + "bench_name": get_bench_name(bench_path), + "background_workers": config.get("background_workers") or 1, + "bench_cmd": which("bench"), + "skip_redis": skip_redis, + "workers": config.get("workers", {}), + } + ) - conf_path = os.path.join(bench_path, 'config', 'supervisor.conf') + conf_path = os.path.join(bench_path, "config", "supervisor.conf") if not yes and os.path.exists(conf_path): - click.confirm('supervisor.conf already exists and this will overwrite it. Do you want to continue?', - abort=True) + click.confirm( + "supervisor.conf already exists and this will overwrite it. Do you want to continue?", + abort=True, + ) - with open(conf_path, 'w') as f: + with open(conf_path, "w") as f: f.write(config) - update_config({'restart_supervisor_on_update': True}, bench_path=bench_path) - update_config({'restart_systemd_on_update': False}, bench_path=bench_path) + update_config({"restart_supervisor_on_update": True}, bench_path=bench_path) + update_config({"restart_systemd_on_update": False}, bench_path=bench_path) def get_supervisord_conf(): """Returns path of supervisord config from possible paths""" - possibilities = ("supervisord.conf", "etc/supervisord.conf", "/etc/supervisord.conf", "/etc/supervisor/supervisord.conf", "/etc/supervisord.conf") + possibilities = ( + "supervisord.conf", + "etc/supervisord.conf", + "/etc/supervisord.conf", + "/etc/supervisor/supervisord.conf", + "/etc/supervisord.conf", + ) for possibility in possibilities: if os.path.exists(possibility): @@ -77,10 +89,7 @@ def update_supervisord_config(user=None, yes=False): supervisord_conf = get_supervisord_conf() section = "unix_http_server" - updated_values = { - "chmod": "0760", - "chown": f"{user}:{user}" - } + updated_values = {"chmod": "0760", "chown": f"{user}:{user}"} supervisord_conf_changes = "" if not supervisord_conf: @@ -94,7 +103,7 @@ def update_supervisord_config(user=None, yes=False): config.add_section(section) action = f"Section {section} Added" logger.log(action) - supervisord_conf_changes += '\n' + action + supervisord_conf_changes += "\n" + action for key, value in updated_values.items(): try: @@ -104,18 +113,25 @@ def update_supervisord_config(user=None, yes=False): if current_value.strip() != value: config.set(section, key, value) - action = f"Updated supervisord.conf: '{key}' changed from '{current_value}' to '{value}'" + action = ( + f"Updated supervisord.conf: '{key}' changed from '{current_value}' to '{value}'" + ) logger.log(action) - supervisord_conf_changes += '\n' + action + supervisord_conf_changes += "\n" + action if not supervisord_conf_changes: logger.error("supervisord.conf not updated") contents = "\n".join(f"{x}={y}" for x, y in updated_values.items()) - print(f"Update your {supervisord_conf} with the following values:\n[{section}]\n{contents}") + print( + f"Update your {supervisord_conf} with the following values:\n[{section}]\n{contents}" + ) return if not yes: - click.confirm(f"{supervisord_conf} will be updated with the following values:\n{supervisord_conf_changes}\nDo you want to continue?", abort=True) + click.confirm( + f"{supervisord_conf} will be updated with the following values:\n{supervisord_conf_changes}\nDo you want to continue?", + abort=True, + ) try: with open(supervisord_conf, "w") as f: @@ -125,4 +141,4 @@ def update_supervisord_config(user=None, yes=False): logger.log(f"Updating supervisord.conf failed due to '{e}'") # Reread supervisor configuration, reload supervisord and supervisorctl, restart services that were started - service('supervisor', 'reload') + service("supervisor", "reload") diff --git a/bench/config/systemd.py b/bench/config/systemd.py index 60053e54..d30edfc9 100644 --- a/bench/config/systemd.py +++ b/bench/config/systemd.py @@ -13,9 +13,14 @@ from bench.config.common_site_config import get_gunicorn_workers, update_config from bench.utils import exec_cmd, which, get_bench_name -def generate_systemd_config(bench_path, user=None, yes=False, - stop=False, create_symlinks=False, - delete_symlinks=False): +def generate_systemd_config( + bench_path, + user=None, + yes=False, + stop=False, + create_symlinks=False, + delete_symlinks=False, +): if not user: user = getpass.getuser() @@ -26,7 +31,9 @@ def generate_systemd_config(bench_path, user=None, yes=False, bench_name = get_bench_name(bench_path) if stop: - exec_cmd(f'sudo systemctl stop -- $(systemctl show -p Requires {bench_name}.target | cut -d= -f2)') + exec_cmd( + f"sudo systemctl stop -- $(systemctl show -p Requires {bench_name}.target | cut -d= -f2)" + ) return if create_symlinks: @@ -37,38 +44,48 @@ def generate_systemd_config(bench_path, user=None, yes=False, _delete_symlinks(bench_path) return - number_of_workers = config.get('background_workers') or 1 + number_of_workers = config.get("background_workers") or 1 background_workers = [] for i in range(number_of_workers): - background_workers.append(get_bench_name(bench_path) + "-frappe-default-worker@" + str(i+1) + ".service") + background_workers.append( + get_bench_name(bench_path) + "-frappe-default-worker@" + str(i + 1) + ".service" + ) for i in range(number_of_workers): - background_workers.append(get_bench_name(bench_path) + "-frappe-short-worker@" + str(i+1) + ".service") + background_workers.append( + get_bench_name(bench_path) + "-frappe-short-worker@" + str(i + 1) + ".service" + ) for i in range(number_of_workers): - background_workers.append(get_bench_name(bench_path) + "-frappe-long-worker@" + str(i+1) + ".service") + background_workers.append( + get_bench_name(bench_path) + "-frappe-long-worker@" + str(i + 1) + ".service" + ) bench_info = { "bench_dir": bench_dir, - "sites_dir": os.path.join(bench_dir, 'sites'), + "sites_dir": os.path.join(bench_dir, "sites"), "user": user, "use_rq": use_rq(bench_path), "http_timeout": config.get("http_timeout", 120), - "redis_server": which('redis-server'), - "node": which('node') or which('nodejs'), - "redis_cache_config": os.path.join(bench_dir, 'config', 'redis_cache.conf'), - "redis_socketio_config": os.path.join(bench_dir, 'config', 'redis_socketio.conf'), - "redis_queue_config": os.path.join(bench_dir, 'config', 'redis_queue.conf'), - "webserver_port": config.get('webserver_port', 8000), - "gunicorn_workers": config.get('gunicorn_workers', get_gunicorn_workers()["gunicorn_workers"]), + "redis_server": which("redis-server"), + "node": which("node") or which("nodejs"), + "redis_cache_config": os.path.join(bench_dir, "config", "redis_cache.conf"), + "redis_socketio_config": os.path.join(bench_dir, "config", "redis_socketio.conf"), + "redis_queue_config": os.path.join(bench_dir, "config", "redis_queue.conf"), + "webserver_port": config.get("webserver_port", 8000), + "gunicorn_workers": config.get( + "gunicorn_workers", get_gunicorn_workers()["gunicorn_workers"] + ), "bench_name": get_bench_name(bench_path), "worker_target_wants": " ".join(background_workers), - "bench_cmd": which('bench') + "bench_cmd": which("bench"), } if not yes: - click.confirm('current systemd configuration will be overwritten. Do you want to continue?', - abort=True) + click.confirm( + "current systemd configuration will be overwritten. Do you want to continue?", + abort=True, + ) setup_systemd_directory(bench_path) setup_main_config(bench_info, bench_path) @@ -76,29 +93,44 @@ def generate_systemd_config(bench_path, user=None, yes=False, setup_web_config(bench_info, bench_path) setup_redis_config(bench_info, bench_path) - update_config({'restart_systemd_on_update': True}, bench_path=bench_path) - update_config({'restart_supervisor_on_update': False}, bench_path=bench_path) + update_config({"restart_systemd_on_update": True}, bench_path=bench_path) + update_config({"restart_supervisor_on_update": False}, bench_path=bench_path) + def setup_systemd_directory(bench_path): - if not os.path.exists(os.path.join(bench_path, 'config', 'systemd')): - os.makedirs(os.path.join(bench_path, 'config', 'systemd')) + if not os.path.exists(os.path.join(bench_path, "config", "systemd")): + os.makedirs(os.path.join(bench_path, "config", "systemd")) + def setup_main_config(bench_info, bench_path): # Main config - bench_template = bench.config.env().get_template('systemd/frappe-bench.target') + bench_template = bench.config.env().get_template("systemd/frappe-bench.target") bench_config = bench_template.render(**bench_info) - bench_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '.target') + bench_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + ".target" + ) - with open(bench_config_path, 'w') as f: + with open(bench_config_path, "w") as f: f.write(bench_config) + def setup_workers_config(bench_info, bench_path): # Worker Group - bench_workers_target_template = bench.config.env().get_template('systemd/frappe-bench-workers.target') - bench_default_worker_template = bench.config.env().get_template('systemd/frappe-bench-frappe-default-worker.service') - bench_short_worker_template = bench.config.env().get_template('systemd/frappe-bench-frappe-short-worker.service') - bench_long_worker_template = bench.config.env().get_template('systemd/frappe-bench-frappe-long-worker.service') - bench_schedule_worker_template = bench.config.env().get_template('systemd/frappe-bench-frappe-schedule.service') + bench_workers_target_template = bench.config.env().get_template( + "systemd/frappe-bench-workers.target" + ) + bench_default_worker_template = bench.config.env().get_template( + "systemd/frappe-bench-frappe-default-worker.service" + ) + bench_short_worker_template = bench.config.env().get_template( + "systemd/frappe-bench-frappe-short-worker.service" + ) + bench_long_worker_template = bench.config.env().get_template( + "systemd/frappe-bench-frappe-long-worker.service" + ) + bench_schedule_worker_template = bench.config.env().get_template( + "systemd/frappe-bench-frappe-schedule.service" + ) bench_workers_target_config = bench_workers_target_template.render(**bench_info) bench_default_worker_config = bench_default_worker_template.render(**bench_info) @@ -106,112 +138,175 @@ def setup_workers_config(bench_info, bench_path): bench_long_worker_config = bench_long_worker_template.render(**bench_info) bench_schedule_worker_config = bench_schedule_worker_template.render(**bench_info) - bench_workers_target_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-workers.target') - bench_default_worker_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-frappe-default-worker@.service') - bench_short_worker_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-frappe-short-worker@.service') - bench_long_worker_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-frappe-long-worker@.service') - bench_schedule_worker_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-frappe-schedule.service') + bench_workers_target_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-workers.target" + ) + bench_default_worker_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-frappe-default-worker@.service", + ) + bench_short_worker_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-frappe-short-worker@.service", + ) + bench_long_worker_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-frappe-long-worker@.service", + ) + bench_schedule_worker_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-frappe-schedule.service", + ) - with open(bench_workers_target_config_path, 'w') as f: + with open(bench_workers_target_config_path, "w") as f: f.write(bench_workers_target_config) - with open(bench_default_worker_config_path, 'w') as f: + with open(bench_default_worker_config_path, "w") as f: f.write(bench_default_worker_config) - with open(bench_short_worker_config_path, 'w') as f: + with open(bench_short_worker_config_path, "w") as f: f.write(bench_short_worker_config) - with open(bench_long_worker_config_path, 'w') as f: + with open(bench_long_worker_config_path, "w") as f: f.write(bench_long_worker_config) - with open(bench_schedule_worker_config_path, 'w') as f: + with open(bench_schedule_worker_config_path, "w") as f: f.write(bench_schedule_worker_config) + def setup_web_config(bench_info, bench_path): # Web Group - bench_web_target_template = bench.config.env().get_template('systemd/frappe-bench-web.target') - bench_web_service_template = bench.config.env().get_template('systemd/frappe-bench-frappe-web.service') - bench_node_socketio_template = bench.config.env().get_template('systemd/frappe-bench-node-socketio.service') + bench_web_target_template = bench.config.env().get_template( + "systemd/frappe-bench-web.target" + ) + bench_web_service_template = bench.config.env().get_template( + "systemd/frappe-bench-frappe-web.service" + ) + bench_node_socketio_template = bench.config.env().get_template( + "systemd/frappe-bench-node-socketio.service" + ) bench_web_target_config = bench_web_target_template.render(**bench_info) bench_web_service_config = bench_web_service_template.render(**bench_info) bench_node_socketio_config = bench_node_socketio_template.render(**bench_info) - bench_web_target_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-web.target') - bench_web_service_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-frappe-web.service') - bench_node_socketio_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-node-socketio.service') + bench_web_target_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-web.target" + ) + bench_web_service_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-frappe-web.service" + ) + bench_node_socketio_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-node-socketio.service", + ) - with open(bench_web_target_config_path, 'w') as f: + with open(bench_web_target_config_path, "w") as f: f.write(bench_web_target_config) - with open(bench_web_service_config_path, 'w') as f: + with open(bench_web_service_config_path, "w") as f: f.write(bench_web_service_config) - with open(bench_node_socketio_config_path, 'w') as f: + with open(bench_node_socketio_config_path, "w") as f: f.write(bench_node_socketio_config) + def setup_redis_config(bench_info, bench_path): # Redis Group - bench_redis_target_template = bench.config.env().get_template('systemd/frappe-bench-redis.target') - bench_redis_cache_template = bench.config.env().get_template('systemd/frappe-bench-redis-cache.service') - bench_redis_queue_template = bench.config.env().get_template('systemd/frappe-bench-redis-queue.service') - bench_redis_socketio_template = bench.config.env().get_template('systemd/frappe-bench-redis-socketio.service') + bench_redis_target_template = bench.config.env().get_template( + "systemd/frappe-bench-redis.target" + ) + bench_redis_cache_template = bench.config.env().get_template( + "systemd/frappe-bench-redis-cache.service" + ) + bench_redis_queue_template = bench.config.env().get_template( + "systemd/frappe-bench-redis-queue.service" + ) + bench_redis_socketio_template = bench.config.env().get_template( + "systemd/frappe-bench-redis-socketio.service" + ) bench_redis_target_config = bench_redis_target_template.render(**bench_info) bench_redis_cache_config = bench_redis_cache_template.render(**bench_info) bench_redis_queue_config = bench_redis_queue_template.render(**bench_info) bench_redis_socketio_config = bench_redis_socketio_template.render(**bench_info) - bench_redis_target_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-redis.target') - bench_redis_cache_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-redis-cache.service') - bench_redis_queue_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-redis-queue.service') - bench_redis_socketio_config_path = os.path.join(bench_path, 'config', 'systemd' , bench_info.get("bench_name") + '-redis-socketio.service') + bench_redis_target_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-redis.target" + ) + bench_redis_cache_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-redis-cache.service" + ) + bench_redis_queue_config_path = os.path.join( + bench_path, "config", "systemd", bench_info.get("bench_name") + "-redis-queue.service" + ) + bench_redis_socketio_config_path = os.path.join( + bench_path, + "config", + "systemd", + bench_info.get("bench_name") + "-redis-socketio.service", + ) - with open(bench_redis_target_config_path, 'w') as f: + with open(bench_redis_target_config_path, "w") as f: f.write(bench_redis_target_config) - with open(bench_redis_cache_config_path, 'w') as f: + with open(bench_redis_cache_config_path, "w") as f: f.write(bench_redis_cache_config) - with open(bench_redis_queue_config_path, 'w') as f: + with open(bench_redis_queue_config_path, "w") as f: f.write(bench_redis_queue_config) - with open(bench_redis_socketio_config_path, 'w') as f: + with open(bench_redis_socketio_config_path, "w") as f: f.write(bench_redis_socketio_config) + def _create_symlinks(bench_path): bench_dir = os.path.abspath(bench_path) - etc_systemd_system = os.path.join('/', 'etc', 'systemd', 'system') - config_path = os.path.join(bench_dir, 'config', 'systemd') + etc_systemd_system = os.path.join("/", "etc", "systemd", "system") + config_path = os.path.join(bench_dir, "config", "systemd") unit_files = get_unit_files(bench_dir) for unit_file in unit_files: filename = "".join(unit_file) - exec_cmd(f'sudo ln -s {config_path}/{filename} {etc_systemd_system}/{"".join(unit_file)}') - exec_cmd('sudo systemctl daemon-reload') + exec_cmd( + f'sudo ln -s {config_path}/{filename} {etc_systemd_system}/{"".join(unit_file)}' + ) + exec_cmd("sudo systemctl daemon-reload") + def _delete_symlinks(bench_path): bench_dir = os.path.abspath(bench_path) - etc_systemd_system = os.path.join('/', 'etc', 'systemd', 'system') + etc_systemd_system = os.path.join("/", "etc", "systemd", "system") unit_files = get_unit_files(bench_dir) for unit_file in unit_files: exec_cmd(f'sudo rm {etc_systemd_system}/{"".join(unit_file)}') - exec_cmd('sudo systemctl daemon-reload') + exec_cmd("sudo systemctl daemon-reload") + def get_unit_files(bench_path): bench_name = get_bench_name(bench_path) unit_files = [ [bench_name, ".target"], - [bench_name+"-workers", ".target"], - [bench_name+"-web", ".target"], - [bench_name+"-redis", ".target"], - [bench_name+"-frappe-default-worker@", ".service"], - [bench_name+"-frappe-short-worker@", ".service"], - [bench_name+"-frappe-long-worker@", ".service"], - [bench_name+"-frappe-schedule", ".service"], - [bench_name+"-frappe-web", ".service"], - [bench_name+"-node-socketio", ".service"], - [bench_name+"-redis-cache", ".service"], - [bench_name+"-redis-queue", ".service"], - [bench_name+"-redis-socketio", ".service"], + [bench_name + "-workers", ".target"], + [bench_name + "-web", ".target"], + [bench_name + "-redis", ".target"], + [bench_name + "-frappe-default-worker@", ".service"], + [bench_name + "-frappe-short-worker@", ".service"], + [bench_name + "-frappe-long-worker@", ".service"], + [bench_name + "-frappe-schedule", ".service"], + [bench_name + "-frappe-web", ".service"], + [bench_name + "-node-socketio", ".service"], + [bench_name + "-redis-cache", ".service"], + [bench_name + "-redis-queue", ".service"], + [bench_name + "-redis-socketio", ".service"], ] return unit_files diff --git a/bench/exceptions.py b/bench/exceptions.py index 48cb6668..0465167c 100644 --- a/bench/exceptions.py +++ b/bench/exceptions.py @@ -39,4 +39,4 @@ class NotInBenchDirectoryError(Exception): class VersionNotFound(Exception): - pass \ No newline at end of file + pass diff --git a/bench/patches/__init__.py b/bench/patches/__init__.py index 5a504737..625f94d1 100644 --- a/bench/patches/__init__.py +++ b/bench/patches/__init__.py @@ -1,31 +1,38 @@ -import os, importlib +import os +import importlib + def run(bench_path): - source_patch_file = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'patches.txt') - target_patch_file = os.path.join(os.path.abspath(bench_path), 'patches.txt') + source_patch_file = os.path.join( + os.path.dirname(os.path.abspath(__file__)), "patches.txt" + ) + target_patch_file = os.path.join(os.path.abspath(bench_path), "patches.txt") - with open(source_patch_file, 'r') as f: - patches = [p.strip() for p in f.read().splitlines() - if p.strip() and not p.strip().startswith("#")] + with open(source_patch_file) as f: + patches = [ + p.strip() + for p in f.read().splitlines() + if p.strip() and not p.strip().startswith("#") + ] executed_patches = [] if os.path.exists(target_patch_file): - with open(target_patch_file, 'r') as f: + with open(target_patch_file) as f: executed_patches = f.read().splitlines() try: for patch in patches: if patch not in executed_patches: module = importlib.import_module(patch.split()[0]) - execute = getattr(module, 'execute') + execute = getattr(module, "execute") result = execute(bench_path) - if result != False: + if not result: executed_patches.append(patch) finally: - with open(target_patch_file, 'w') as f: - f.write('\n'.join(executed_patches)) + with open(target_patch_file, "w") as f: + f.write("\n".join(executed_patches)) # end with an empty line - f.write('\n') + f.write("\n") diff --git a/bench/patches/v3/__init__.py b/bench/patches/v3/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/bench/patches/v3/celery_to_rq.py b/bench/patches/v3/celery_to_rq.py deleted file mode 100644 index 29b2af2e..00000000 --- a/bench/patches/v3/celery_to_rq.py +++ /dev/null @@ -1,25 +0,0 @@ -import click, os -from bench.config.procfile import setup_procfile -from bench.config.supervisor import generate_supervisor_config -from bench.utils.app import get_current_frappe_version, get_current_branch - -def execute(bench_path): - frappe_branch = get_current_branch('frappe', bench_path) - frappe_version = get_current_frappe_version(bench_path) - - if not (frappe_branch=='develop' or frappe_version >= 7): - # not version 7+ - # prevent running this patch - return False - - click.confirm('\nThis update will remove Celery config and prepare the bench to use Python RQ.\n' - 'And it will overwrite Procfile and supervisor.conf.\n' - 'If you don\'t know what this means, type Y ;)\n\n' - 'Do you want to continue?', - abort=True) - - setup_procfile(bench_path, yes=True) - - # if production setup - if os.path.exists(os.path.join(bench_path, 'config', 'supervisor.conf')): - generate_supervisor_config(bench_path, yes=True) diff --git a/bench/patches/v3/deprecate_old_config.py b/bench/patches/v3/deprecate_old_config.py deleted file mode 100644 index 4b86d0d2..00000000 --- a/bench/patches/v3/deprecate_old_config.py +++ /dev/null @@ -1,38 +0,0 @@ -import os, json -from bench.config.common_site_config import get_config, put_config, get_common_site_config - -def execute(bench_path): - # deprecate bench config - bench_config_path = os.path.join(bench_path, 'config.json') - if not os.path.exists(bench_config_path): - return - - with open(bench_config_path, "r") as f: - bench_config = json.loads(f.read()) - - common_site_config = get_common_site_config(bench_path) - common_site_config.update(bench_config) - put_config(common_site_config, bench_path) - - # remove bench/config.json - os.remove(bench_config_path) - - # change keys - config = get_config(bench_path) - changed = False - for from_key, to_key, default in ( - ("celery_broker", "redis_queue", "redis://localhost:6379"), - ("async_redis_server", "redis_socketio", "redis://localhost:12311"), - ("cache_redis_server", "redis_cache", "redis://localhost:11311") - ): - if from_key in config: - config[to_key] = config[from_key] - del config[from_key] - changed = True - - elif to_key not in config: - config[to_key] = default - changed = True - - if changed: - put_config(config, bench_path) diff --git a/bench/patches/v3/redis_bind_ip.py b/bench/patches/v3/redis_bind_ip.py deleted file mode 100644 index a5b1196c..00000000 --- a/bench/patches/v3/redis_bind_ip.py +++ /dev/null @@ -1,10 +0,0 @@ -import click -from bench.config.redis import generate_config - -def execute(bench_path): - click.confirm('\nThis update will replace ERPNext\'s Redis configuration files to fix a major security issue.\n' - 'If you don\'t know what this means, type Y ;)\n\n' - 'Do you want to continue?', - abort=True) - - generate_config(bench_path) diff --git a/bench/patches/v4/__init__.py b/bench/patches/v4/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/bench/patches/v4/install_yarn.py b/bench/patches/v4/install_yarn.py deleted file mode 100644 index e29868e9..00000000 --- a/bench/patches/v4/install_yarn.py +++ /dev/null @@ -1,5 +0,0 @@ -import os -from bench.utils import exec_cmd - -def execute(bench_path): - exec_cmd('npm install yarn', os.path.join(bench_path, 'apps/frappe')) diff --git a/bench/patches/v4/update_node.py b/bench/patches/v4/update_node.py deleted file mode 100644 index ffbe280c..00000000 --- a/bench/patches/v4/update_node.py +++ /dev/null @@ -1,32 +0,0 @@ -import click, subprocess, sys -from semantic_version import Version -from distutils.spawn import find_executable - -def execute(bench_path): - expected_node_ver = Version('5.0.0') - node_exec = find_executable('node') or find_executable('nodejs') - - - if node_exec: - result = subprocess.check_output([node_exec, '-v']).decode() - else: - click.echo(''' - No node executable was found on your machine. - Please install latest node version before running "bench update". For installation instructions - please refer "Debian and Ubuntu based Linux distributions" section or "Enterprise Linux and - Fedora" section depending upon your OS on the following link, - "https://nodejs.org/en/download/package-manager/" - ''') - sys.exit(1) - - node_ver = Version(result.rstrip('\n').lstrip('v')) - - if node_ver < expected_node_ver: - click.echo(''' - Please update node to latest version before running "bench update". - Please install latest node version before running "bench update". For installation instructions - please refer "Debian and Ubuntu based Linux distributions" section or "Enterprise Linux and - Fedora" section depending upon your OS on the following link, - "https://nodejs.org/en/download/package-manager/" - ''') - sys.exit(1) diff --git a/bench/patches/v4/update_socketio.py b/bench/patches/v4/update_socketio.py deleted file mode 100644 index e7ffd092..00000000 --- a/bench/patches/v4/update_socketio.py +++ /dev/null @@ -1,4 +0,0 @@ -import subprocess - -def execute(bench_path): - subprocess.check_output(['npm', 'install', 'socket.io']) \ No newline at end of file diff --git a/bench/patches/v5/fix_backup_cronjob.py b/bench/patches/v5/fix_backup_cronjob.py index 15bbfcf3..db253aa2 100644 --- a/bench/patches/v5/fix_backup_cronjob.py +++ b/bench/patches/v5/fix_backup_cronjob.py @@ -4,10 +4,10 @@ from crontab import CronTab def execute(bench_path): """ - This patch fixes a cron job that would backup sites every minute per 6 hours + This patch fixes a cron job that would backup sites every minute per 6 hours """ - user = get_config(bench_path=bench_path).get('frappe_user') + user = get_config(bench_path=bench_path).get("frappe_user") user_crontab = CronTab(user=user) for job in user_crontab.find_comment("bench auto backups set for every 6 hours"): diff --git a/bench/patches/v5/fix_user_permissions.py b/bench/patches/v5/fix_user_permissions.py index 85dea8e6..c780bbc0 100644 --- a/bench/patches/v5/fix_user_permissions.py +++ b/bench/patches/v5/fix_user_permissions.py @@ -34,13 +34,13 @@ def is_production_set(bench_path): bench_name = get_bench_name(bench_path) supervisor_conf_extn = "ini" if is_centos7() else "conf" - supervisor_conf_file_name = f'{bench_name}.{supervisor_conf_extn}' + supervisor_conf_file_name = f"{bench_name}.{supervisor_conf_extn}" supervisor_conf = os.path.join(get_supervisor_confdir(), supervisor_conf_file_name) if os.path.exists(supervisor_conf): production_setup = production_setup or True - nginx_conf = f'/etc/nginx/conf.d/{bench_name}.conf' + nginx_conf = f"/etc/nginx/conf.d/{bench_name}.conf" if os.path.exists(nginx_conf): production_setup = production_setup or True @@ -50,7 +50,7 @@ def is_production_set(bench_path): def execute(bench_path): """This patch checks if bench sudoers is set and regenerate supervisor and sudoers files""" - user = get_config('.').get("frappe_user") or getpass.getuser() + user = get_config(".").get("frappe_user") or getpass.getuser() if is_sudoers_set(): if is_production_set(bench_path): diff --git a/bench/patches/v5/set_live_reload_config.py b/bench/patches/v5/set_live_reload_config.py index 4538a6df..97c5e638 100644 --- a/bench/patches/v5/set_live_reload_config.py +++ b/bench/patches/v5/set_live_reload_config.py @@ -2,4 +2,4 @@ from bench.config.common_site_config import update_config def execute(bench_path): - update_config({'live_reload': True}, bench_path) + update_config({"live_reload": True}, bench_path) diff --git a/bench/patches/v5/update_archived_sites.py b/bench/patches/v5/update_archived_sites.py index 139b65ef..ce24bb61 100644 --- a/bench/patches/v5/update_archived_sites.py +++ b/bench/patches/v5/update_archived_sites.py @@ -17,7 +17,7 @@ from semantic_version import Version def execute(bench_path): - frappe_version = Version(get_current_version('frappe')) + frappe_version = Version(get_current_version("frappe")) if frappe_version.major < 14 or os.name != "posix": # Returning False means patch has been skipped diff --git a/bench/tests/test_base.py b/bench/tests/test_base.py index 4c5b6889..cc910f94 100644 --- a/bench/tests/test_base.py +++ b/bench/tests/test_base.py @@ -9,7 +9,6 @@ import traceback import unittest # imports - module imports -import bench from bench.utils import paths_in_bench, exec_cmd from bench.utils.system import init from bench.bench import Bench @@ -23,6 +22,7 @@ if PYTHON_VER.major == 3: else: FRAPPE_BRANCH = "develop" + class TestBenchBase(unittest.TestCase): def setUp(self): self.benches_path = "." @@ -32,11 +32,26 @@ class TestBenchBase(unittest.TestCase): for bench_name in self.benches: bench_path = os.path.join(self.benches_path, bench_name) bench = Bench(bench_path) - mariadb_password = "travis" if os.environ.get("CI") else getpass.getpass(prompt="Enter MariaDB root Password: ") + mariadb_password = ( + "travis" + if os.environ.get("CI") + else getpass.getpass(prompt="Enter MariaDB root Password: ") + ) if bench.exists: for site in bench.sites: - subprocess.call(["bench", "drop-site", site, "--force", "--no-backup", "--root-password", mariadb_password], cwd=bench_path) + subprocess.call( + [ + "bench", + "drop-site", + site, + "--force", + "--no-backup", + "--root-password", + mariadb_password, + ], + cwd=bench_path, + ) shutil.rmtree(bench_path, ignore_errors=True) def assert_folders(self, bench_name): @@ -55,18 +70,21 @@ class TestBenchBase(unittest.TestCase): for config, search_key in ( ("redis_queue.conf", "redis_queue.rdb"), ("redis_socketio.conf", "redis_socketio.rdb"), - ("redis_cache.conf", "redis_cache.rdb")): + ("redis_cache.conf", "redis_cache.rdb"), + ): self.assert_exists(bench_name, "config", config) - with open(os.path.join(bench_name, "config", config), "r") as f: + with open(os.path.join(bench_name, "config", config)) as f: self.assertTrue(search_key in f.read()) def assert_common_site_config(self, bench_name, expected_config): - common_site_config_path = os.path.join(self.benches_path, bench_name, 'sites', 'common_site_config.json') + common_site_config_path = os.path.join( + self.benches_path, bench_name, "sites", "common_site_config.json" + ) self.assertTrue(os.path.exists(common_site_config_path)) - with open(common_site_config_path, "r") as f: + with open(common_site_config_path) as f: config = json.load(f) for key, value in list(expected_config.items()): @@ -78,7 +96,7 @@ class TestBenchBase(unittest.TestCase): def new_site(self, site_name, bench_name): new_site_cmd = ["bench", "new-site", site_name, "--admin-password", "admin"] - if os.environ.get('CI'): + if os.environ.get("CI"): new_site_cmd.extend(["--mariadb-root-password", "travis"]) subprocess.call(new_site_cmd, cwd=os.path.join(self.benches_path, bench_name)) @@ -88,18 +106,25 @@ class TestBenchBase(unittest.TestCase): frappe_tmp_path = "/tmp/frappe" if not os.path.exists(frappe_tmp_path): - exec_cmd(f"git clone https://github.com/frappe/frappe -b {FRAPPE_BRANCH} --depth 1 --origin upstream {frappe_tmp_path}") + exec_cmd( + f"git clone https://github.com/frappe/frappe -b {FRAPPE_BRANCH} --depth 1 --origin upstream {frappe_tmp_path}" + ) - kwargs.update(dict( - python=sys.executable, - no_procfile=True, - no_backups=True, - frappe_path=frappe_tmp_path - )) + kwargs.update( + dict( + python=sys.executable, + no_procfile=True, + no_backups=True, + frappe_path=frappe_tmp_path, + ) + ) if not os.path.exists(os.path.join(self.benches_path, bench_name)): init(bench_name, **kwargs) - exec_cmd("git remote set-url upstream https://github.com/frappe/frappe", cwd=os.path.join(self.benches_path, bench_name, "apps", "frappe")) + exec_cmd( + "git remote set-url upstream https://github.com/frappe/frappe", + cwd=os.path.join(self.benches_path, bench_name, "apps", "frappe"), + ) def file_exists(self, path): if os.environ.get("CI"): diff --git a/bench/tests/test_init.py b/bench/tests/test_init.py index 978f63eb..685dcf40 100755 --- a/bench/tests/test_init.py +++ b/bench/tests/test_init.py @@ -19,6 +19,7 @@ from bench.bench import Bench # for longer since docs.erpnext.com is powered by it ;) TEST_FRAPPE_APP = "frappe_docs" + class TestBenchInit(TestBenchBase): def test_utils(self): self.assertEqual(subprocess.call("bench"), 0) @@ -27,9 +28,9 @@ class TestBenchInit(TestBenchBase): self.init_bench(bench_name, **kwargs) app = App("file:///tmp/frappe") self.assertTupleEqual( - (app.mount_path, app.url, app.repo, app.org), - ("/tmp/frappe", "file:///tmp/frappe", "frappe", "frappe"), - ) + (app.mount_path, app.url, app.repo, app.org), + ("/tmp/frappe", "file:///tmp/frappe", "frappe", "frappe"), + ) self.assert_folders(bench_name) self.assert_virtual_env(bench_name) self.assert_config(bench_name) @@ -43,30 +44,33 @@ class TestBenchInit(TestBenchBase): except Exception: print(self.get_traceback()) - def test_multiple_benches(self): for bench_name in ("test-bench-1", "test-bench-2"): self.init_bench(bench_name) - self.assert_common_site_config("test-bench-1", { - "webserver_port": 8000, - "socketio_port": 9000, - "file_watcher_port": 6787, - "redis_queue": "redis://localhost:11000", - "redis_socketio": "redis://localhost:12000", - "redis_cache": "redis://localhost:13000" - }) - - self.assert_common_site_config("test-bench-2", { - "webserver_port": 8001, - "socketio_port": 9001, - "file_watcher_port": 6788, - "redis_queue": "redis://localhost:11001", - "redis_socketio": "redis://localhost:12001", - "redis_cache": "redis://localhost:13001" - }) - + self.assert_common_site_config( + "test-bench-1", + { + "webserver_port": 8000, + "socketio_port": 9000, + "file_watcher_port": 6787, + "redis_queue": "redis://localhost:11000", + "redis_socketio": "redis://localhost:12000", + "redis_cache": "redis://localhost:13000", + }, + ) + self.assert_common_site_config( + "test-bench-2", + { + "webserver_port": 8001, + "socketio_port": 9001, + "file_watcher_port": 6788, + "redis_queue": "redis://localhost:11001", + "redis_socketio": "redis://localhost:12001", + "redis_cache": "redis://localhost:13001", + }, + ) def test_new_site(self): bench_name = "test-bench" @@ -85,7 +89,7 @@ class TestBenchInit(TestBenchBase): self.assertTrue(os.path.exists(os.path.join(site_path, "public", "files"))) self.assertTrue(os.path.exists(site_config_path)) - with open(site_config_path, "r") as f: + with open(site_config_path) as f: site_config = json.loads(f.read()) for key in ("db_name", "db_password"): @@ -97,7 +101,9 @@ class TestBenchInit(TestBenchBase): bench_path = os.path.join(self.benches_path, "test-bench") exec_cmd(f"bench get-app {TEST_FRAPPE_APP}", cwd=bench_path) self.assertTrue(os.path.exists(os.path.join(bench_path, "apps", TEST_FRAPPE_APP))) - app_installed_in_env = TEST_FRAPPE_APP in subprocess.check_output(["bench", "pip", "freeze"], cwd=bench_path).decode('utf8') + app_installed_in_env = TEST_FRAPPE_APP in subprocess.check_output( + ["bench", "pip", "freeze"], cwd=bench_path + ).decode("utf8") self.assertTrue(app_installed_in_env) def test_get_app_resolve_deps(self): @@ -108,12 +114,12 @@ class TestBenchInit(TestBenchBase): self.assertTrue(os.path.exists(os.path.join(bench_path, "apps", FRAPPE_APP))) states_path = os.path.join(bench_path, "sites", "apps.json") - self.assert_(os.path.exists(states_path)) + self.assertTrue(os.path.exists(states_path)) - with open(states_path, "r") as f: + with open(states_path) as f: states = json.load(f) - self.assert_(FRAPPE_APP in states) + self.assertTrue(FRAPPE_APP in states) def test_install_app(self): bench_name = "test-bench" @@ -128,33 +134,42 @@ class TestBenchInit(TestBenchBase): self.assertTrue(os.path.exists(os.path.join(bench_path, "apps", TEST_FRAPPE_APP))) # check if app is installed - app_installed_in_env = TEST_FRAPPE_APP in subprocess.check_output(["bench", "pip", "freeze"], cwd=bench_path).decode('utf8') + app_installed_in_env = TEST_FRAPPE_APP in subprocess.check_output( + ["bench", "pip", "freeze"], cwd=bench_path + ).decode("utf8") self.assertTrue(app_installed_in_env) # create and install app on site self.new_site(site_name, bench_name) - installed_app = not exec_cmd(f"bench --site {site_name} install-app {TEST_FRAPPE_APP}", cwd=bench_path) + installed_app = not exec_cmd( + f"bench --site {site_name} install-app {TEST_FRAPPE_APP}", cwd=bench_path + ) - app_installed_on_site = subprocess.check_output(["bench", "--site", site_name, "list-apps"], cwd=bench_path).decode('utf8') + app_installed_on_site = subprocess.check_output( + ["bench", "--site", site_name, "list-apps"], cwd=bench_path + ).decode("utf8") if installed_app: self.assertTrue(TEST_FRAPPE_APP in app_installed_on_site) - def test_remove_app(self): self.init_bench("test-bench") bench_path = os.path.join(self.benches_path, "test-bench") exec_cmd("bench setup requirements --node", cwd=bench_path) - exec_cmd(f"bench get-app {TEST_FRAPPE_APP} --branch master --overwrite", cwd=bench_path) + exec_cmd( + f"bench get-app {TEST_FRAPPE_APP} --branch master --overwrite", cwd=bench_path + ) exec_cmd(f"bench remove-app {TEST_FRAPPE_APP}", cwd=bench_path) with open(os.path.join(bench_path, "sites", "apps.txt")) as f: self.assertFalse(TEST_FRAPPE_APP in f.read()) - self.assertFalse(TEST_FRAPPE_APP in subprocess.check_output(["bench", "pip", "freeze"], cwd=bench_path).decode('utf8')) + self.assertFalse( + TEST_FRAPPE_APP + in subprocess.check_output(["bench", "pip", "freeze"], cwd=bench_path).decode("utf8") + ) self.assertFalse(os.path.exists(os.path.join(bench_path, "apps", TEST_FRAPPE_APP))) - def test_switch_to_branch(self): self.init_bench("test-bench") bench_path = os.path.join(self.benches_path, "test-bench") @@ -166,16 +181,20 @@ class TestBenchInit(TestBenchBase): # assuming we follow `version-#` prevoius_branch = f"version-{int(FRAPPE_BRANCH.split('-')[1]) - 1}" - successful_switch = not exec_cmd(f"bench switch-to-branch {prevoius_branch} frappe --upgrade", cwd=bench_path) + successful_switch = not exec_cmd( + f"bench switch-to-branch {prevoius_branch} frappe --upgrade", cwd=bench_path + ) app_branch_after_switch = str(git.Repo(path=app_path).active_branch) if successful_switch: self.assertEqual(prevoius_branch, app_branch_after_switch) - successful_switch = not exec_cmd(f"bench switch-to-branch {FRAPPE_BRANCH} frappe --upgrade", cwd=bench_path) + successful_switch = not exec_cmd( + f"bench switch-to-branch {FRAPPE_BRANCH} frappe --upgrade", cwd=bench_path + ) app_branch_after_second_switch = str(git.Repo(path=app_path).active_branch) if successful_switch: self.assertEqual(FRAPPE_BRANCH, app_branch_after_second_switch) -if __name__ == '__main__': +if __name__ == "__main__": unittest.main() diff --git a/bench/tests/test_setup_production.py b/bench/tests/test_setup_production.py index af9c0b5b..8c7a9100 100644 --- a/bench/tests/test_setup_production.py +++ b/bench/tests/test_setup_production.py @@ -32,16 +32,16 @@ class TestSetupProduction(TestBenchBase): bench_path = os.path.join(os.path.abspath(self.benches_path), bench_name) exec_cmd("sudo bench disable-production", cwd=bench_path) - def production(self): try: self.test_setup_production() except Exception: print(self.get_traceback()) - def assert_nginx_config(self, bench_name): - conf_src = os.path.join(os.path.abspath(self.benches_path), bench_name, 'config', 'nginx.conf') + conf_src = os.path.join( + os.path.abspath(self.benches_path), bench_name, "config", "nginx.conf" + ) conf_dest = f"/etc/nginx/conf.d/{bench_name}.conf" self.assertTrue(self.file_exists(conf_src)) @@ -51,23 +51,23 @@ class TestSetupProduction(TestBenchBase): self.assertEqual(os.path.realpath(conf_dest), conf_src) # file content - with open(conf_src, "r") as f: + with open(conf_src) as f: f = f.read() for key in ( - f"upstream {bench_name}-frappe", - f"upstream {bench_name}-socketio-server" - ): + f"upstream {bench_name}-frappe", + f"upstream {bench_name}-socketio-server", + ): self.assertTrue(key in f) - def assert_nginx_process(self): out = get_cmd_output("sudo nginx -t 2>&1") - self.assertTrue("nginx: configuration file /etc/nginx/nginx.conf test is successful" in out) - + self.assertTrue( + "nginx: configuration file /etc/nginx/nginx.conf test is successful" in out + ) def assert_sudoers(self, user): - sudoers_file = '/etc/sudoers.d/frappe' + sudoers_file = "/etc/sudoers.d/frappe" service = which("service") nginx = which("nginx") @@ -76,15 +76,16 @@ class TestSetupProduction(TestBenchBase): if os.environ.get("CI"): sudoers = subprocess.check_output(["sudo", "cat", sudoers_file]).decode("utf-8") else: - with open(sudoers_file, 'r') as f: + with open(sudoers_file) as f: sudoers = f.read() - self.assertTrue(f'{user} ALL = (root) NOPASSWD: {service} nginx *' in sudoers) - self.assertTrue(f'{user} ALL = (root) NOPASSWD: {nginx}' in sudoers) - + self.assertTrue(f"{user} ALL = (root) NOPASSWD: {service} nginx *" in sudoers) + self.assertTrue(f"{user} ALL = (root) NOPASSWD: {nginx}" in sudoers) def assert_supervisor_config(self, bench_name, use_rq=True): - conf_src = os.path.join(os.path.abspath(self.benches_path), bench_name, 'config', 'supervisor.conf') + conf_src = os.path.join( + os.path.abspath(self.benches_path), bench_name, "config", "supervisor.conf" + ) supervisor_conf_dir = get_supervisor_confdir() conf_dest = f"{supervisor_conf_dir}/{bench_name}.conf" @@ -96,7 +97,7 @@ class TestSetupProduction(TestBenchBase): self.assertEqual(os.path.realpath(conf_dest), conf_src) # file content - with open(conf_src, "r") as f: + with open(conf_src) as f: f = f.read() tests = [ @@ -106,65 +107,72 @@ class TestSetupProduction(TestBenchBase): f"program:{bench_name}-redis-socketio", f"group:{bench_name}-web", f"group:{bench_name}-workers", - f"group:{bench_name}-redis" + f"group:{bench_name}-redis", ] if not os.environ.get("CI"): tests.append(f"program:{bench_name}-node-socketio") if use_rq: - tests.extend([ - f"program:{bench_name}-frappe-schedule", - f"program:{bench_name}-frappe-default-worker", - f"program:{bench_name}-frappe-short-worker", - f"program:{bench_name}-frappe-long-worker" - ]) + tests.extend( + [ + f"program:{bench_name}-frappe-schedule", + f"program:{bench_name}-frappe-default-worker", + f"program:{bench_name}-frappe-short-worker", + f"program:{bench_name}-frappe-long-worker", + ] + ) else: - tests.extend([ - f"program:{bench_name}-frappe-workerbeat", - f"program:{bench_name}-frappe-worker", - f"program:{bench_name}-frappe-longjob-worker", - f"program:{bench_name}-frappe-async-worker" - ]) + tests.extend( + [ + f"program:{bench_name}-frappe-workerbeat", + f"program:{bench_name}-frappe-worker", + f"program:{bench_name}-frappe-longjob-worker", + f"program:{bench_name}-frappe-async-worker", + ] + ) for key in tests: self.assertTrue(key in f) - def assert_supervisor_process(self, bench_name, use_rq=True, disable_production=False): out = get_cmd_output("supervisorctl status") while "STARTING" in out: - print ("Waiting for all processes to start...") + print("Waiting for all processes to start...") time.sleep(10) out = get_cmd_output("supervisorctl status") tests = [ - "{bench_name}-web:{bench_name}-frappe-web[\s]+RUNNING", + r"{bench_name}-web:{bench_name}-frappe-web[\s]+RUNNING", # Have commented for the time being. Needs to be uncommented later on. Bench is failing on travis because of this. # It works on one bench and fails on another.giving FATAL or BACKOFF (Exited too quickly (process log may have details)) # "{bench_name}-web:{bench_name}-node-socketio[\s]+RUNNING", - "{bench_name}-redis:{bench_name}-redis-cache[\s]+RUNNING", - "{bench_name}-redis:{bench_name}-redis-queue[\s]+RUNNING", - "{bench_name}-redis:{bench_name}-redis-socketio[\s]+RUNNING" + r"{bench_name}-redis:{bench_name}-redis-cache[\s]+RUNNING", + r"{bench_name}-redis:{bench_name}-redis-queue[\s]+RUNNING", + r"{bench_name}-redis:{bench_name}-redis-socketio[\s]+RUNNING", ] if use_rq: - tests.extend([ - "{bench_name}-workers:{bench_name}-frappe-schedule[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-default-worker-0[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-short-worker-0[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-long-worker-0[\s]+RUNNING" - ]) + tests.extend( + [ + r"{bench_name}-workers:{bench_name}-frappe-schedule[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-default-worker-0[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-short-worker-0[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-long-worker-0[\s]+RUNNING", + ] + ) else: - tests.extend([ - "{bench_name}-workers:{bench_name}-frappe-workerbeat[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-worker[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-longjob-worker[\s]+RUNNING", - "{bench_name}-workers:{bench_name}-frappe-async-worker[\s]+RUNNING" - ]) + tests.extend( + [ + r"{bench_name}-workers:{bench_name}-frappe-workerbeat[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-worker[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-longjob-worker[\s]+RUNNING", + r"{bench_name}-workers:{bench_name}-frappe-async-worker[\s]+RUNNING", + ] + ) for key in tests: if disable_production: @@ -173,5 +181,5 @@ class TestSetupProduction(TestBenchBase): self.assertTrue(re.search(key, out)) -if __name__ == '__main__': +if __name__ == "__main__": unittest.main() diff --git a/bench/tests/test_utils.py b/bench/tests/test_utils.py index f34d161b..db830e1b 100644 --- a/bench/tests/test_utils.py +++ b/bench/tests/test_utils.py @@ -20,8 +20,8 @@ class TestUtils(unittest.TestCase): app.name == git_url, app.branch == branch, app.tag == branch, - app.is_url == True, - app.on_disk == False, + app.is_url is True, + app.on_disk is False, app.org == "frappe", app.url == git_url, ] @@ -30,11 +30,19 @@ class TestUtils(unittest.TestCase): def test_is_valid_frappe_branch(self): with self.assertRaises(InvalidRemoteException): - is_valid_frappe_branch("https://github.com/frappe/frappe.git", frappe_branch="random-branch") - is_valid_frappe_branch("https://github.com/random/random.git", frappe_branch="random-branch") + is_valid_frappe_branch( + "https://github.com/frappe/frappe.git", frappe_branch="random-branch" + ) + is_valid_frappe_branch( + "https://github.com/random/random.git", frappe_branch="random-branch" + ) - is_valid_frappe_branch("https://github.com/frappe/frappe.git", frappe_branch="develop") - is_valid_frappe_branch("https://github.com/frappe/frappe.git", frappe_branch="v13.29.0") + is_valid_frappe_branch( + "https://github.com/frappe/frappe.git", frappe_branch="develop" + ) + is_valid_frappe_branch( + "https://github.com/frappe/frappe.git", frappe_branch="v13.29.0" + ) def test_app_states(self): bench_dir = "./sandbox" @@ -48,7 +56,10 @@ class TestUtils(unittest.TestCase): self.assertTrue(hasattr(fake_bench.apps, "states")) fake_bench.apps.states = { - "frappe": {"resolution": {"branch": "develop", "commit_hash": "234rwefd"}, "version": "14.0.0-dev"} + "frappe": { + "resolution": {"branch": "develop", "commit_hash": "234rwefd"}, + "version": "14.0.0-dev", + } } fake_bench.apps.update_apps_states() @@ -64,7 +75,9 @@ class TestUtils(unittest.TestCase): f.write("__version__ = '11.0'") subprocess.run(["git", "add", "."], cwd=frappe_path, capture_output=True, check=True) - subprocess.run(["git", "commit", "-m", "temp"], cwd=frappe_path, capture_output=True, check=True) + subprocess.run( + ["git", "commit", "-m", "temp"], cwd=frappe_path, capture_output=True, check=True + ) fake_bench.apps.update_apps_states(app_name="frappe") @@ -76,4 +89,4 @@ class TestUtils(unittest.TestCase): def test_ssh_ports(self): app = App("git@github.com:22:frappe/frappe") - self.assertEqual((app.use_ssh, app.org, app.repo), (True, "frappe", "frappe")) \ No newline at end of file + self.assertEqual((app.use_ssh, app.org, app.repo), (True, "frappe", "frappe")) diff --git a/bench/utils/__init__.py b/bench/utils/__init__.py index cd323ca1..952ca2a4 100644 --- a/bench/utils/__init__.py +++ b/bench/utils/__init__.py @@ -2,29 +2,31 @@ import json import logging import os -import subprocess import re +import subprocess import sys +from functools import lru_cache from glob import glob from shlex import split from typing import List, Tuple -from functools import lru_cache # imports - third party imports import click -import requests # imports - module imports from bench import PROJECT_NAME, VERSION - -from bench.exceptions import CommandFailedError, InvalidRemoteException, AppNotInstalledError - +from bench.exceptions import ( + AppNotInstalledError, + CommandFailedError, + InvalidRemoteException, +) logger = logging.getLogger(PROJECT_NAME) bench_cache_file = ".bench.cmd" paths_in_app = ("hooks.py", "modules.txt", "patches.txt") paths_in_bench = ("apps", "sites", "config", "logs", "config/pids") sudoers_file = "/etc/sudoers.d/frappe" +UNSET_ARG = object() def is_bench_directory(directory=os.path.curdir): @@ -51,7 +53,7 @@ def is_frappe_app(directory: str) -> bool: @lru_cache(maxsize=None) -def is_valid_frappe_branch(frappe_path:str, frappe_branch:str): +def is_valid_frappe_branch(frappe_path: str, frappe_branch: str): """Check if a branch exists in a repo. Throws InvalidRemoteException if branch is not found Uses native git command to check for branches on a remote. @@ -209,7 +211,9 @@ def get_git_version() -> float: def get_cmd_output(cmd, cwd=".", _raise=True): output = "" try: - output = subprocess.check_output(cmd, cwd=cwd, shell=True, stderr=subprocess.PIPE, encoding="utf-8").strip() + output = subprocess.check_output( + cmd, cwd=cwd, shell=True, stderr=subprocess.PIPE, encoding="utf-8" + ).strip() except subprocess.CalledProcessError as e: if e.output: output = e.output @@ -508,6 +512,7 @@ def get_traceback() -> str: class _dict(dict): """dict like object that exposes keys as attributes""" + # bench port of frappe._dict def __getattr__(self, key): ret = self.get(key) @@ -515,16 +520,21 @@ class _dict(dict): if not ret and key.startswith("__") and key != "__deepcopy__": raise AttributeError() return ret + def __setattr__(self, key, value): self[key] = value + def __getstate__(self): return self + def __setstate__(self, d): self.update(d) + def update(self, d): """update and return self -- the missing dict feature in python""" - super(_dict, self).update(d) + super().update(d) return self + def copy(self): return _dict(dict(self).copy()) @@ -539,10 +549,8 @@ def get_cmd_from_sysargv(): """ # context is passed as options to frappe's bench_helper from bench.bench import Bench - frappe_context = _dict( - params={"--site"}, - flags={"--verbose", "--profile", "--force"} - ) + + frappe_context = _dict(params={"--site"}, flags={"--verbose", "--profile", "--force"}) cmd_from_ctx = None sys_argv = sys.argv[1:] skip_next = False diff --git a/bench/utils/app.py b/bench/utils/app.py index 11eab676..51ef2c6a 100644 --- a/bench/utils/app.py +++ b/bench/utils/app.py @@ -1,7 +1,13 @@ +# imports - standard imports import os +import pathlib import re import sys import subprocess +from typing import List +from functools import lru_cache + +# imports - module imports from bench.exceptions import ( InvalidRemoteException, InvalidBranchException, @@ -9,7 +15,6 @@ from bench.exceptions import ( VersionNotFound, ) from bench.app import get_repo_dir -from functools import lru_cache def is_version_upgrade(app="frappe", bench_path=".", branch=None): @@ -108,7 +113,9 @@ def switch_to_develop(apps=None, bench_path=".", upgrade=True): def get_version_from_string(contents, field="__version__"): - match = re.search(r"^(\s*%s\s*=\s*['\\\"])(.+?)(['\"])" % field, contents, flags=(re.S | re.M)) + match = re.search( + r"^(\s*%s\s*=\s*['\\\"])(.+?)(['\"])" % field, contents, flags=(re.S | re.M) + ) if not match: raise VersionNotFound(f"{contents} is not a valid version") return match.group(2) @@ -157,7 +164,7 @@ def get_upstream_version(app, branch=None, bench_path="."): def get_current_frappe_version(bench_path="."): try: return get_major_version(get_current_version("frappe", bench_path=bench_path)) - except IOError: + except OSError: return 0 @@ -184,13 +191,17 @@ def get_required_deps(org, name, branch, deps="hooks.py"): return base64.decodebytes(res["content"].encode()).decode() -def required_apps_from_hooks(required_deps, local=False): +def required_apps_from_hooks(required_deps: str, local: bool = False) -> List: + import ast + + required_apps_re = re.compile(r"required_apps\s+=\s+(.*)") + if local: - with open(required_deps) as f: - required_deps = f.read() - lines = [x for x in required_deps.split("\n") if x.strip().startswith("required_apps")] - required_apps = eval(lines[0].strip("required_apps").strip().lstrip("=").strip()) - return required_apps + required_deps = pathlib.Path(required_deps).read_text() + + _req_apps_tag = required_apps_re.search(required_deps) + req_apps_tag = _req_apps_tag[1] + return ast.literal_eval(req_apps_tag) def get_remote(app, bench_path="."): @@ -247,6 +258,7 @@ def get_app_name(bench_path: str, folder_name: str) -> str: return folder_name + def check_existing_dir(bench_path, repo_name): cloned_path = os.path.join(bench_path, "apps", repo_name) dir_already_exists = os.path.isdir(cloned_path) diff --git a/bench/utils/bench.py b/bench/utils/bench.py index 72653dbc..761e45fc 100644 --- a/bench/utils/bench.py +++ b/bench/utils/bench.py @@ -6,25 +6,14 @@ import re import subprocess import sys from json.decoder import JSONDecodeError -import typing # imports - third party imports import click -import bench # imports - module imports -from bench.utils import ( - which, - log, - exec_cmd, - get_bench_name, - get_cmd_output, -) +import bench from bench.exceptions import PatchError, ValidationError - - -if typing.TYPE_CHECKING: - from bench.bench import Bench +from bench.utils import exec_cmd, get_bench_name, get_cmd_output, log, which logger = logging.getLogger(bench.PROJECT_NAME) @@ -56,9 +45,10 @@ def get_venv_path(verbose=False): def update_node_packages(bench_path=".", apps=None): print("Updating node packages...") - from bench.utils.app import get_develop_version from distutils.version import LooseVersion + from bench.utils.app import get_develop_version + v = LooseVersion(get_develop_version("frappe", bench_path=bench_path)) # After rollup was merged, frappe_version = 10.1 @@ -95,7 +85,9 @@ def install_python_dev_dependencies(bench_path=".", apps=None, verbose=False): bench.run(f"{bench.python} -m pip install {quiet_flag} --upgrade {pyproject_deps}") if not pyproject_deps and os.path.exists(dev_requirements_path): - bench.run(f"{bench.python} -m pip install {quiet_flag} --upgrade -r {dev_requirements_path}") + bench.run( + f"{bench.python} -m pip install {quiet_flag} --upgrade -r {dev_requirements_path}" + ) def _generate_dev_deps_pattern(pyproject_path): @@ -108,7 +100,7 @@ def _generate_dev_deps_pattern(pyproject_path): pyroject_config = loads(open(pyproject_path).read()) try: - for pkg, version in pyroject_config['tool']['bench']['dev-dependencies'].items(): + for pkg, version in pyroject_config["tool"]["bench"]["dev-dependencies"].items(): op = "==" if "=" not in version else "" requirements_pattern += f"{pkg}{op}{version} " except KeyError: @@ -120,9 +112,7 @@ def update_yarn_packages(bench_path=".", apps=None): from bench.bench import Bench bench = Bench(bench_path) - apps = apps or bench.apps - apps_dir = os.path.join(bench.name, "apps") # TODO: Check for stuff like this early on only?? @@ -149,7 +139,7 @@ def update_npm_packages(bench_path=".", apps=None): package_json_path = os.path.join(apps_dir, app, "package.json") if os.path.exists(package_json_path): - with open(package_json_path, "r") as f: + with open(package_json_path) as f: app_package_json = json.loads(f.read()) # package.json is usually a dict in a dict for key, value in app_package_json.items(): @@ -164,7 +154,7 @@ def update_npm_packages(bench_path=".", apps=None): package_json[key] = value if package_json is {}: - with open(os.path.join(os.path.dirname(__file__), "package.json"), "r") as f: + with open(os.path.join(os.path.dirname(__file__), "package.json")) as f: package_json = json.loads(f.read()) with open(os.path.join(bench_path, "package.json"), "w") as f: @@ -176,6 +166,7 @@ def update_npm_packages(bench_path=".", apps=None): def migrate_env(python, backup=False): import shutil from urllib.parse import urlparse + from bench.bench import Bench bench = Bench(".") @@ -237,10 +228,10 @@ def validate_upgrade(from_ver, to_ver, bench_path="."): def post_upgrade(from_ver, to_ver, bench_path="."): - from bench.config import redis - from bench.config.supervisor import generate_supervisor_config - from bench.config.nginx import make_nginx_conf from bench.bench import Bench + from bench.config import redis + from bench.config.nginx import make_nginx_conf + from bench.config.supervisor import generate_supervisor_config conf = Bench(bench_path).conf print("-" * 80 + f"Your bench was upgraded to version {to_ver}") @@ -323,9 +314,7 @@ def restart_systemd_processes(bench_path=".", web_workers=False): def restart_process_manager(bench_path=".", web_workers=False): # only overmind has the restart feature, not sure other supported procmans do - if which("overmind") and os.path.exists( - os.path.join(bench_path, ".overmind.sock") - ): + if which("overmind") and os.path.exists(os.path.join(bench_path, ".overmind.sock")): worker = "web" if web_workers else "" exec_cmd(f"overmind restart {worker}", cwd=bench_path) @@ -338,7 +327,7 @@ def build_assets(bench_path=".", app=None): def handle_version_upgrade(version_upgrade, bench_path, force, reset, conf): - from bench.utils import pause_exec, log + from bench.utils import log, pause_exec if version_upgrade[0]: if force: @@ -386,13 +375,12 @@ def update( ): """command: bench update""" import re - from bench import patches + from bench import patches from bench.app import pull_apps from bench.bench import Bench from bench.config.common_site_config import update_config from bench.exceptions import CannotUpdateReleaseBench - from bench.utils import clear_command_cache from bench.utils.app import is_version_upgrade from bench.utils.system import backup_all_sites @@ -459,8 +447,7 @@ def update( update_config(conf, bench_path=bench_path) print( - "_" * 80 - + "\nBench: Deployment tool for Frappe and Frappe Applications" + "_" * 80 + "\nBench: Deployment tool for Frappe and Frappe Applications" " (https://frappe.io/bench).\nOpen source depends on your contributions, so do" " give back by submitting bug reports, patches and fixes and be a part of the" " community :)" @@ -500,7 +487,7 @@ def clone_apps_from(bench_path, clone_from, update_app=True): install_app(app, bench_path, restart_bench=False) - with open(os.path.join(clone_from, "sites", "apps.txt"), "r") as f: + with open(os.path.join(clone_from, "sites", "apps.txt")) as f: apps = f.read().splitlines() for app in apps: @@ -509,6 +496,7 @@ def clone_apps_from(bench_path, clone_from, update_app=True): def remove_backups_crontab(bench_path="."): from crontab import CronTab + from bench.bench import Bench logger.log("removing backup cronjob") @@ -543,7 +531,7 @@ def update_common_site_config(ddict, bench_path="."): filename = os.path.join(bench_path, "sites", "common_site_config.json") if os.path.exists(filename): - with open(filename, "r") as f: + with open(filename) as f: content = json.load(f) else: @@ -605,7 +593,7 @@ def validate_branch(): apps = Bench(".").apps installed_apps = set(apps) - check_apps = set(["frappe", "erpnext"]) + check_apps = {"frappe", "erpnext"} intersection_apps = installed_apps.intersection(check_apps) for app in intersection_apps: diff --git a/bench/utils/render.py b/bench/utils/render.py index b0ca6a19..46a95cbf 100644 --- a/bench/utils/render.py +++ b/bench/utils/render.py @@ -14,7 +14,7 @@ class Capturing(list): Util to consume the stdout encompassed in it and push it to a list with Capturing() as output: - subprocess.check_output("ls", shell=True) + subprocess.check_output("ls", shell=True) print(output) # ["b'Applications\\nDesktop\\nDocuments\\nDownloads\\n'"] @@ -53,20 +53,25 @@ class Rendering: if not self.dynamic_feed: return - _prefix = click.style('⏼', fg='bright_yellow') + _prefix = click.style("⏼", fg="bright_yellow") _hierarchy = " " if not self.is_parent else "" self._title = self.title.format(**self.kw) click.secho(f"{_hierarchy}{_prefix} {self._title}") bench.LOG_BUFFER.append( - {"message": self._title, "prefix": _prefix, "color": None, "is_parent": self.is_parent} + { + "message": self._title, + "prefix": _prefix, + "color": None, + "is_parent": self.is_parent, + } ) def __exit__(self, *args, **kwargs): if not self.dynamic_feed: return - self._prefix = click.style('✔', fg='green') + self._prefix = click.style("✔", fg="green") self._success = self.success.format(**self.kw) self.render_screen() @@ -87,14 +92,20 @@ def job(title: str = None, success: str = None): For instance, the `get-app` command consists of two jobs: `initializing bench` and `fetching and installing app`. """ + def innfn(fn): def wrapper_fn(*args, **kwargs): with Rendering( - success=success, title=title, is_parent=True, args=args, kwargs=kwargs, + success=success, + title=title, + is_parent=True, + args=args, + kwargs=kwargs, ): return fn(*args, **kwargs) return wrapper_fn + return innfn @@ -102,12 +113,18 @@ def step(title: str = None, success: str = None): """Supposed to be wrapped around the smallest possible atomic step in a given operation. For instance, `building assets` is a step in the update operation. """ + def innfn(fn): def wrapper_fn(*args, **kwargs): with Rendering( - success=success, title=title, is_parent=False, args=args, kwargs=kwargs, + success=success, + title=title, + is_parent=False, + args=args, + kwargs=kwargs, ): return fn(*args, **kwargs) return wrapper_fn + return innfn diff --git a/bench/utils/system.py b/bench/utils/system.py index 131db2a9..22eb2454 100644 --- a/bench/utils/system.py +++ b/bench/utils/system.py @@ -44,8 +44,8 @@ def init( * setup config (dir/pids/redis/procfile) for the bench * setup patches.txt for bench * clone & install frappe - * install python & node dependencies - * build assets + * install python & node dependencies + * build assets * setup backups crontab """ diff --git a/setup.py b/setup.py index fade655f..1e648184 100644 --- a/setup.py +++ b/setup.py @@ -1,11 +1,11 @@ +import pathlib + from setuptools import find_packages, setup + from bench import PROJECT_NAME, VERSION -with open("requirements.txt") as f: - install_requires = f.read().strip().split("\n") - -with open("README.md") as f: - long_description = f.read() +install_requires = pathlib.Path("requirements.txt").read_text().strip().split("\n") +long_description = pathlib.Path("README.md").read_text() setup( name=PROJECT_NAME, @@ -34,7 +34,7 @@ setup( "Topic :: System :: Installation/Setup", ], packages=find_packages(), - python_requires="~=3.6", + python_requires=">=3.7", zip_safe=False, include_package_data=True, install_requires=install_requires, From af46fedc2869caf43318796e8cd9b667ea6b4db5 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 14:29:43 +0530 Subject: [PATCH 07/17] refactor(minor): Use more readable alternatives --- bench/utils/bench.py | 10 ++++------ bench/utils/render.py | 4 ++-- bench/utils/system.py | 11 ++--------- bench/utils/translation.py | 2 +- 4 files changed, 9 insertions(+), 18 deletions(-) diff --git a/bench/utils/bench.py b/bench/utils/bench.py index 761e45fc..f0157fa8 100644 --- a/bench/utils/bench.py +++ b/bench/utils/bench.py @@ -1,4 +1,5 @@ # imports - standard imports +import contextlib import json import logging import os @@ -99,12 +100,10 @@ def _generate_dev_deps_pattern(pyproject_path): requirements_pattern = "" pyroject_config = loads(open(pyproject_path).read()) - try: + with contextlib.suppress(KeyError): for pkg, version in pyroject_config["tool"]["bench"]["dev-dependencies"].items(): op = "==" if "=" not in version else "" requirements_pattern += f"{pkg}{op}{version} " - except KeyError: - pass return requirements_pattern @@ -222,9 +221,8 @@ def migrate_env(python, backup=False): def validate_upgrade(from_ver, to_ver, bench_path="."): - if to_ver >= 6: - if not which("npm") and not (which("node") or which("nodejs")): - raise Exception("Please install nodejs and npm") + if to_ver >= 6 and not which("npm") and not which("node") and not which("nodejs"): + raise Exception("Please install nodejs and npm") def post_upgrade(from_ver, to_ver, bench_path="."): diff --git a/bench/utils/render.py b/bench/utils/render.py index 46a95cbf..155aa75e 100644 --- a/bench/utils/render.py +++ b/bench/utils/render.py @@ -54,7 +54,7 @@ class Rendering: return _prefix = click.style("⏼", fg="bright_yellow") - _hierarchy = " " if not self.is_parent else "" + _hierarchy = "" if self.is_parent else " " self._title = self.title.format(**self.kw) click.secho(f"{_hierarchy}{_prefix} {self._title}") @@ -83,7 +83,7 @@ class Rendering: if l["message"] == self._title: l["prefix"] = self._prefix l["message"] = self._success - _hierarchy = " " if not l["is_parent"] else "" + _hierarchy = "" if l.get("is_parent") else " " click.secho(f'{_hierarchy}{l["prefix"]} {l["message"]}', fg=l["color"]) diff --git a/bench/utils/system.py b/bench/utils/system.py index 22eb2454..63a301b9 100644 --- a/bench/utils/system.py +++ b/bench/utils/system.py @@ -113,10 +113,7 @@ def setup_sudoers(user): if not os.path.exists("/etc/sudoers.d"): os.makedirs("/etc/sudoers.d") - set_permissions = False - if not os.path.exists("/etc/sudoers"): - set_permissions = True - + set_permissions = not os.path.exists("/etc/sudoers") with open("/etc/sudoers", "a") as f: f.write("\n#includedir /etc/sudoers.d\n") @@ -142,11 +139,7 @@ def setup_sudoers(user): def start(no_dev=False, concurrency=None, procfile=None, no_prefix=False, procman=None): - if procman: - program = which(procman) - else: - program = get_process_manager() - + program = which(procman) if procman else get_process_manager() if not program: raise Exception("No process manager found") diff --git a/bench/utils/translation.py b/bench/utils/translation.py index ad3a202d..e0dee804 100644 --- a/bench/utils/translation.py +++ b/bench/utils/translation.py @@ -43,7 +43,7 @@ def update_translations(app, lang): import requests translations_dir = os.path.join("apps", app, app, "translations") - csv_file = os.path.join(translations_dir, lang + ".csv") + csv_file = os.path.join(translations_dir, f"{lang}.csv") url = f"https://translate.erpnext.com/files/{app}-{lang}.csv" r = requests.get(url, stream=True) r.raise_for_status() From 94a25d3da7fa9552d94f362c0b3016b316399473 Mon Sep 17 00:00:00 2001 From: Orsiris de Jong Date: Wed, 27 Jul 2022 11:28:53 +0200 Subject: [PATCH 08/17] fix(config): Fix fail2ban filter (#1308) * Add missing nginx-proxy filter * Add all nginx log files to fail2ban filter, regardless of frappe site * Adds per site nginx logs in /var/log --- bench/config/templates/nginx.conf | 5 +++++ bench/playbooks/roles/fail2ban/defaults/main.yml | 4 ++-- .../playbooks/roles/fail2ban/tasks/configure_nginx_jail.yml | 5 ++++- 3 files changed, 11 insertions(+), 3 deletions(-) diff --git a/bench/config/templates/nginx.conf b/bench/config/templates/nginx.conf index 4af72ab4..666af1bf 100644 --- a/bench/config/templates/nginx.conf +++ b/bench/config/templates/nginx.conf @@ -113,6 +113,11 @@ server { {% endfor -%} + # logs in var + access_log /var/log/nginx/{{ site_name }}_access.log main; + error_log /var/log/nginx/{{ site_name }}_error.log; + + # optimizations sendfile on; keepalive_timeout 15; diff --git a/bench/playbooks/roles/fail2ban/defaults/main.yml b/bench/playbooks/roles/fail2ban/defaults/main.yml index 5aae2800..81019070 100644 --- a/bench/playbooks/roles/fail2ban/defaults/main.yml +++ b/bench/playbooks/roles/fail2ban/defaults/main.yml @@ -1,5 +1,5 @@ --- -fail2ban_nginx_access_log: /var/log/nginx/access.log +fail2ban_nginx_access_log: /var/log/nginx/*access.log maxretry: 6 bantime: 600 -findtime: 600 \ No newline at end of file +findtime: 600 diff --git a/bench/playbooks/roles/fail2ban/tasks/configure_nginx_jail.yml b/bench/playbooks/roles/fail2ban/tasks/configure_nginx_jail.yml index 4d2ede07..b9ced995 100644 --- a/bench/playbooks/roles/fail2ban/tasks/configure_nginx_jail.yml +++ b/bench/playbooks/roles/fail2ban/tasks/configure_nginx_jail.yml @@ -5,7 +5,10 @@ vars_files: - ../defaults/main.yml tasks: + + - name: Setup filter + template: src="../templates/nginx-proxy-filter.conf.j2" dest="/etc/fail2ban/filter.d/nginx-proxy.conf" - name: Setup jail template: src="../templates/nginx-proxy-jail.conf.j2" dest="/etc/fail2ban/jail.d/nginx-proxy.conf" - name: restart service - service: name=fail2ban state=restarted \ No newline at end of file + service: name=fail2ban state=restarted From 1badfa8da4c9088b564cf2f42d595bdde5364d06 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 16:16:54 +0530 Subject: [PATCH 09/17] ci: Update conditions for FRAPPE_BRANCH --- .travis.yml | 15 ++++----------- bench/tests/test_base.py | 9 ++++----- bench/tests/test_init.py | 4 ++-- bench/tests/test_setup_production.py | 5 ++--- requirements.txt | 2 +- 5 files changed, 13 insertions(+), 22 deletions(-) diff --git a/.travis.yml b/.travis.yml index f1d8b83f..a3d88ab3 100644 --- a/.travis.yml +++ b/.travis.yml @@ -5,11 +5,6 @@ sudo: true git: depth: 1 -cache: - - pip - - npm - - yarn - addons: mariadb: '10.3' @@ -61,20 +56,18 @@ matrix: script: python -m unittest -v bench.tests.test_utils && python -m unittest -v bench.tests.test_init install: - - pip3 install urllib3 pyOpenSSL ndg-httpsclient pyasn1 + - python -m pip install urllib3 pyOpenSSL ndg-httpsclient pyasn1 - if [ $TEST == "bench" ];then - wget -q -O /tmp/wkhtmltox.tar.xz https://github.com/frappe/wkhtmltopdf/raw/master/wkhtmltox-0.12.3_linux-generic-amd64.tar.xz; - tar -xf /tmp/wkhtmltox.tar.xz -C /tmp; - sudo mv /tmp/wkhtmltox/bin/wkhtmltopdf /usr/local/bin/wkhtmltopdf; - sudo chmod o+x /usr/local/bin/wkhtmltopdf; + wget https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb; + sudo apt install ./wkhtmltox_0.12.6-1.focal_amd64.deb; nvm install 14; nvm use 14; mkdir -p ~/.bench; cp -r $TRAVIS_BUILD_DIR/* ~/.bench; - pip3 install -q -U -e ~/.bench; + python -m pip install -U -e ~/.bench; mysql -u root -e "SET GLOBAL character_set_server = 'utf8mb4'"; mysql -u root -e "SET GLOBAL collation_server = 'utf8mb4_unicode_ci'"; diff --git a/bench/tests/test_base.py b/bench/tests/test_base.py index cc910f94..9ca23de4 100644 --- a/bench/tests/test_base.py +++ b/bench/tests/test_base.py @@ -17,10 +17,10 @@ PYTHON_VER = sys.version_info FRAPPE_BRANCH = "version-12" if PYTHON_VER.major == 3: - if PYTHON_VER.minor in [6, 7]: - FRAPPE_BRANCH = "version-13" - else: + if PYTHON_VER.minor >= 10: FRAPPE_BRANCH = "develop" + if 7 >= PYTHON_VER.minor >= 9: + FRAPPE_BRANCH = "version-13" class TestBenchBase(unittest.TestCase): @@ -134,5 +134,4 @@ class TestBenchBase(unittest.TestCase): def get_traceback(self): exc_type, exc_value, exc_tb = sys.exc_info() trace_list = traceback.format_exception(exc_type, exc_value, exc_tb) - body = "".join(str(t) for t in trace_list) - return body + return "".join(str(t) for t in trace_list) diff --git a/bench/tests/test_init.py b/bench/tests/test_init.py index 685dcf40..61150c83 100755 --- a/bench/tests/test_init.py +++ b/bench/tests/test_init.py @@ -184,15 +184,15 @@ class TestBenchInit(TestBenchBase): successful_switch = not exec_cmd( f"bench switch-to-branch {prevoius_branch} frappe --upgrade", cwd=bench_path ) - app_branch_after_switch = str(git.Repo(path=app_path).active_branch) if successful_switch: + app_branch_after_switch = str(git.Repo(path=app_path).active_branch) self.assertEqual(prevoius_branch, app_branch_after_switch) successful_switch = not exec_cmd( f"bench switch-to-branch {FRAPPE_BRANCH} frappe --upgrade", cwd=bench_path ) - app_branch_after_second_switch = str(git.Repo(path=app_path).active_branch) if successful_switch: + app_branch_after_second_switch = str(git.Repo(path=app_path).active_branch) self.assertEqual(FRAPPE_BRANCH, app_branch_after_second_switch) diff --git a/bench/tests/test_setup_production.py b/bench/tests/test_setup_production.py index 8c7a9100..4dae93c0 100644 --- a/bench/tests/test_setup_production.py +++ b/bench/tests/test_setup_production.py @@ -1,6 +1,7 @@ # imports - standard imports import getpass import os +import pathlib import re import subprocess import time @@ -76,9 +77,7 @@ class TestSetupProduction(TestBenchBase): if os.environ.get("CI"): sudoers = subprocess.check_output(["sudo", "cat", sudoers_file]).decode("utf-8") else: - with open(sudoers_file) as f: - sudoers = f.read() - + sudoers = pathlib.Path(sudoers_file).read_text() self.assertTrue(f"{user} ALL = (root) NOPASSWD: {service} nginx *" in sudoers) self.assertTrue(f"{user} ALL = (root) NOPASSWD: {nginx}" in sudoers) diff --git a/requirements.txt b/requirements.txt index 664c9639..d5b26797 100644 --- a/requirements.txt +++ b/requirements.txt @@ -5,5 +5,5 @@ Jinja2~=3.0.3 python-crontab~=2.4.0 requests semantic-version~=2.8.2 -setuptools +setuptools>60.0.0 tomli;python_version<"3.11" \ No newline at end of file From 86c3c90fd9d260b001edf1c798df9fbd2af64ab1 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Wed, 27 Jul 2022 18:40:17 +0530 Subject: [PATCH 10/17] build: Retire setup.py + reqs.txt to use pyproject.toml --- pyproject.toml | 57 ++++++++++++++++++++++++++++++++++++++++++++++++ requirements.txt | 9 -------- setup.py | 42 ----------------------------------- 3 files changed, 57 insertions(+), 51 deletions(-) create mode 100644 pyproject.toml delete mode 100644 requirements.txt delete mode 100644 setup.py diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 00000000..190689ea --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,57 @@ +[project] +name = "frappe-bench" +description = "CLI to manage Multi-tenant deployments for Frappe apps" +readme = "README.md" +license = "GPL-3.0-only" +requires-python = ">=3.7" +authors = [ + { name = "Frappe Technologies Pvt Ltd", email = "developers@frappe.io" }, +] +classifiers = [ + "Development Status :: 5 - Production/Stable", + "Environment :: Console", + "License :: OSI Approved :: GNU Affero General Public License v3", + "Natural Language :: English", + "Operating System :: MacOS", + "Operating System :: OS Independent", + "Topic :: Software Development :: Build Tools", + "Topic :: Software Development :: User Interfaces", + "Topic :: System :: Installation/Setup", +] +dependencies = [ + "Click>=7.0", + "GitPython~=2.1.15", + "honcho", + "Jinja2~=3.0.3", + "python-crontab~=2.4.0", + "requests", + "semantic-version~=2.8.2", + "setuptools>40.9.0", + "tomli;python_version<'3.11'", +] +dynamic = [ + "version", +] + +[project.scripts] +bench = "bench.cli:cli" + +[project.urls] +Changelog = "https://github.com/frappe/bench/releases" +Documentation = "https://frappeframework.com/docs/user/en/bench" +Homepage = "https://frappe.io/bench" +Source = "https://github.com/frappe/bench" + +[build-system] +requires = [ + "hatchling>=1.6.0", +] +build-backend = "hatchling.build" + +[tool.hatch.version] +path = "bench/__init__.py" + +[tool.hatch.build.targets.sdist] +include = [ + "/bench", +] diff --git a/requirements.txt b/requirements.txt deleted file mode 100644 index d5b26797..00000000 --- a/requirements.txt +++ /dev/null @@ -1,9 +0,0 @@ -Click>=7.0 -GitPython~=2.1.15 -honcho -Jinja2~=3.0.3 -python-crontab~=2.4.0 -requests -semantic-version~=2.8.2 -setuptools>60.0.0 -tomli;python_version<"3.11" \ No newline at end of file diff --git a/setup.py b/setup.py deleted file mode 100644 index 1e648184..00000000 --- a/setup.py +++ /dev/null @@ -1,42 +0,0 @@ -import pathlib - -from setuptools import find_packages, setup - -from bench import PROJECT_NAME, VERSION - -install_requires = pathlib.Path("requirements.txt").read_text().strip().split("\n") -long_description = pathlib.Path("README.md").read_text() - -setup( - name=PROJECT_NAME, - description="CLI to manage Multi-tenant deployments for Frappe apps", - long_description=long_description, - long_description_content_type="text/markdown", - version=VERSION, - license="GPLv3", - author="Frappe Technologies Pvt Ltd", - author_email="developers@frappe.io", - url="https://frappe.io/bench", - project_urls={ - "Documentation": "https://frappeframework.com/docs/user/en/bench", - "Source": "https://github.com/frappe/bench", - "Changelog": "https://github.com/frappe/bench/releases", - }, - classifiers=[ - "Development Status :: 5 - Production/Stable", - "Environment :: Console", - "License :: OSI Approved :: GNU Affero General Public License v3", - "Natural Language :: English", - "Operating System :: MacOS", - "Operating System :: OS Independent", - "Topic :: Software Development :: Build Tools", - "Topic :: Software Development :: User Interfaces", - "Topic :: System :: Installation/Setup", - ], - packages=find_packages(), - python_requires=">=3.7", - zip_safe=False, - include_package_data=True, - install_requires=install_requires, - entry_points={"console_scripts": ["bench=bench.cli:cli"]}, -) From 38fa45607dee5c3e2bd1315c67cec7197817eac7 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Thu, 28 Jul 2022 11:24:29 +0530 Subject: [PATCH 11/17] build: Bump python-crontab dependency Package v2.4.2 leads to a build error on PY 3.7, 3.8 ref: https://app.travis-ci.com/github/frappe/bench/jobs/577932789 --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index 190689ea..55f40a05 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -23,7 +23,7 @@ dependencies = [ "GitPython~=2.1.15", "honcho", "Jinja2~=3.0.3", - "python-crontab~=2.4.0", + "python-crontab~=2.6.0", "requests", "semantic-version~=2.8.2", "setuptools>40.9.0", From 5515b0f4ca9d1bc6e7a28cf9ecfe194e70eb6e2d Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Thu, 28 Jul 2022 11:37:16 +0530 Subject: [PATCH 12/17] ci: Update pip, wheel, setuptools explicitly in install --- .travis.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index a3d88ab3..780847ae 100644 --- a/.travis.yml +++ b/.travis.yml @@ -56,7 +56,7 @@ matrix: script: python -m unittest -v bench.tests.test_utils && python -m unittest -v bench.tests.test_init install: - - python -m pip install urllib3 pyOpenSSL ndg-httpsclient pyasn1 + - python -m pip install -U --no-cache-dir --force-reinstall urllib3 pyOpenSSL ndg-httpsclient pyasn1 wheel setuptools pip - if [ $TEST == "bench" ];then wget https://github.com/wkhtmltopdf/packaging/releases/download/0.12.6-1/wkhtmltox_0.12.6-1.focal_amd64.deb; From 22294fce5bf5949ff9f6401e8708af80b7b3b438 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Thu, 28 Jul 2022 11:50:10 +0530 Subject: [PATCH 13/17] ci: Add PY3.10 to runner matrix Reduce existing runners based on lesser variance to breakages --- .travis.yml | 23 +++++++++-------------- 1 file changed, 9 insertions(+), 14 deletions(-) diff --git a/.travis.yml b/.travis.yml index 780847ae..76618d5a 100644 --- a/.travis.yml +++ b/.travis.yml @@ -25,18 +25,18 @@ matrix: env: TEST=bench script: python bench/tests/test_init.py TestBenchInit.basic + - name: "Python 3.10 Basic Setup" + python: "3.10" + env: TEST=bench + script: python bench/tests/test_init.py TestBenchInit.basic + - name: "Python 3.7 Production Setup" python: 3.7 env: TEST=bench script: python bench/tests/test_setup_production.py TestSetupProduction.production - - name: "Python 3.8 Production Setup" - python: 3.8 - env: TEST=bench - script: python bench/tests/test_setup_production.py TestSetupProduction.production - - - name: "Python 3.9 Production Setup" - python: 3.9 + - name: "Python 3.10 Production Setup" + python: "3.10" env: TEST=bench script: python bench/tests/test_setup_production.py TestSetupProduction.production @@ -45,13 +45,8 @@ matrix: env: TEST=bench script: python -m unittest -v bench.tests.test_utils && python -m unittest -v bench.tests.test_init - - name: "Python 3.8 Tests" - python: 3.8 - env: TEST=bench - script: python -m unittest -v bench.tests.test_utils && python -m unittest -v bench.tests.test_init - - - name: "Python 3.9 Tests" - python: 3.9 + - name: "Python 3.10 Tests" + python: "3.10" env: TEST=bench script: python -m unittest -v bench.tests.test_utils && python -m unittest -v bench.tests.test_init From 203f41e405cf7f37ea4dd07b852a988494c2286f Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Thu, 28 Jul 2022 13:35:04 +0530 Subject: [PATCH 14/17] test: Skip deps resolution for non develop branches --- bench/tests/test_init.py | 1 + 1 file changed, 1 insertion(+) diff --git a/bench/tests/test_init.py b/bench/tests/test_init.py index 61150c83..1d4e1027 100755 --- a/bench/tests/test_init.py +++ b/bench/tests/test_init.py @@ -106,6 +106,7 @@ class TestBenchInit(TestBenchBase): ).decode("utf8") self.assertTrue(app_installed_in_env) + @unittest.skipIf(FRAPPE_BRANCH != "develop", "only for develop branch") def test_get_app_resolve_deps(self): FRAPPE_APP = "healthcare" self.init_bench("test-bench") From df84c2772d8eb7334f37ec7380bed3ac424b0cc4 Mon Sep 17 00:00:00 2001 From: barredterra <14891507+barredterra@users.noreply.github.com> Date: Thu, 28 Jul 2022 14:19:09 +0200 Subject: [PATCH 15/17] fix: use specified python for venv --- bench/bench.py | 2 +- bench/utils/bench.py | 7 +++---- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/bench/bench.py b/bench/bench.py index 1f79f190..d94c9f70 100644 --- a/bench/bench.py +++ b/bench/bench.py @@ -354,7 +354,7 @@ class BenchSetup(Base): if virtualenv: self.run(f"{virtualenv} {quiet_flag} env -p {python}") else: - venv = get_venv_path(verbose=verbose) + venv = get_venv_path(verbose=verbose, python=python) self.run(f"{venv} env") self.pip() diff --git a/bench/utils/bench.py b/bench/utils/bench.py index f0157fa8..c0263657 100644 --- a/bench/utils/bench.py +++ b/bench/utils/bench.py @@ -32,14 +32,13 @@ def get_virtualenv_path(verbose=False): return virtualenv_path -def get_venv_path(verbose=False): - current_python = sys.executable +def get_venv_path(verbose=False, python="python3"): with open(os.devnull, "wb") as devnull: is_venv_installed = not subprocess.call( - [current_python, "-m", "venv", "--help"], stdout=devnull + [python, "-m", "venv", "--help"], stdout=devnull ) if is_venv_installed: - return f"{current_python} -m venv" + return f"{python} -m venv" else: log("virtualenv cannot be found", level=2) From f68ff595bfceb124c057577e90744156a36468d6 Mon Sep 17 00:00:00 2001 From: gavin Date: Fri, 29 Jul 2022 00:03:59 +0530 Subject: [PATCH 16/17] chore: Add logo for Dark mode --- README.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 4d5c0f40..37bde87f 100755 --- a/README.md +++ b/README.md @@ -1,5 +1,8 @@
- + + + +

Bench

From 060e6b7ceaa40b32c2879b2232242b5f9a367a72 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Fri, 29 Jul 2022 19:17:22 +0530 Subject: [PATCH 17/17] ci: Use hatch to build package --- .github/workflows/release.yml | 18 +++++++++++------- .releaserc | 2 +- 2 files changed, 12 insertions(+), 8 deletions(-) diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index fbb96207..c16bba90 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -6,20 +6,24 @@ on: jobs: release: name: Release - runs-on: ubuntu-18.04 + runs-on: ubuntu-latest steps: - - name: Checkout Entire Repository - uses: actions/checkout@v2 + - uses: actions/checkout@v2 with: fetch-depth: 0 - - name: Setup Node.js v12 - uses: actions/setup-node@v1 + - uses: actions/setup-node@v2 with: - node-version: 12 + node-version: 14 + - uses: actions/setup-python@v4 + with: + python-version: '3.x' + - name: Setup dependencies run: | npm install @semantic-release/git @semantic-release/exec --no-save - pip install wheel twine + python3 -m pip install wheel twine + python3 -m pip install git+https://github.com/pypa/hatch + - name: Create Release env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} diff --git a/.releaserc b/.releaserc index 43cc0109..3d33a1da 100644 --- a/.releaserc +++ b/.releaserc @@ -10,7 +10,7 @@ ], [ "@semantic-release/exec", { - "prepareCmd": "python setup.py bdist_wheel --universal" + "prepareCmd": "hatch build -t sdist -t wheel" } ], [