From cd0fccfdb49f24a4383ca0a6e4e5849f67637fc0 Mon Sep 17 00:00:00 2001 From: Dany Robert Date: Mon, 6 Jun 2022 17:18:54 +0530 Subject: [PATCH 1/5] fix: Bump Nodejs v12 => v14 --- bench/playbooks/roles/nodejs/defaults/main.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/bench/playbooks/roles/nodejs/defaults/main.yml b/bench/playbooks/roles/nodejs/defaults/main.yml index a09f08cb..33b3dfc0 100644 --- a/bench/playbooks/roles/nodejs/defaults/main.yml +++ b/bench/playbooks/roles/nodejs/defaults/main.yml @@ -1,3 +1,3 @@ --- -node_version: 12 -... \ No newline at end of file +node_version: 14 +... From 0f64446c176f15fc69373a1729d5bf4c6a5c5dd3 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Tue, 14 Jun 2022 15:36:20 +0530 Subject: [PATCH 2/5] chore: Drop vm project VM project has been unmaintained and out of date for a long long time now. The auto-build and publish pipeline has been dead for years and has been de-listed too from erpnext.org. Nows a better time than ever to get rid of this unmaintained piece of the project. --- vm/Readme.md | 37 ----- vm/Vagrantfile | 73 ---------- vm/build.py | 128 ------------------ vm/http/preseed.cfg | 41 ------ vm/scripts/debian_family/cleanup.sh | 13 -- vm/scripts/debian_family/install_ansible.sh | 10 -- .../debian_family/install_erpnext_develop.sh | 5 - .../install_erpnext_production.sh | 5 - .../debian_family/install_prerequisites.sh | 4 - vm/scripts/debian_family/setup.sh | 8 -- vm/scripts/restart_supervisor.sh | 14 -- vm/scripts/set_message_develop.sh | 18 --- vm/scripts/set_message_production.sh | 18 --- vm/vm-develop.json | 101 -------------- vm/vm-production.json | 95 ------------- 15 files changed, 570 deletions(-) delete mode 100644 vm/Readme.md delete mode 100644 vm/Vagrantfile delete mode 100644 vm/build.py delete mode 100644 vm/http/preseed.cfg delete mode 100644 vm/scripts/debian_family/cleanup.sh delete mode 100644 vm/scripts/debian_family/install_ansible.sh delete mode 100644 vm/scripts/debian_family/install_erpnext_develop.sh delete mode 100644 vm/scripts/debian_family/install_erpnext_production.sh delete mode 100644 vm/scripts/debian_family/install_prerequisites.sh delete mode 100644 vm/scripts/debian_family/setup.sh delete mode 100644 vm/scripts/restart_supervisor.sh delete mode 100644 vm/scripts/set_message_develop.sh delete mode 100644 vm/scripts/set_message_production.sh delete mode 100644 vm/vm-develop.json delete mode 100644 vm/vm-production.json diff --git a/vm/Readme.md b/vm/Readme.md deleted file mode 100644 index 316b3c33..00000000 --- a/vm/Readme.md +++ /dev/null @@ -1,37 +0,0 @@ -## ERPNext VM Builder - - -### Steps to build a VM Image - -* `python build.py` builds a new Production VM, a Dev VM and a Dev Vagrant Box - - -### Requirements - -* Bench should be installed -* Ansible should be installed - - -### How it works - -Apart from the above the rest is handled by bench: - -* Install prerequisites if not already installed - - virtualbox - - packer -* Cleanup - - Clean the required directories -* Build the VM using packer - - Packer downloads the mentioned Ubuntu iso, boots a virtual machine and preceeds the preseed.cfg file into it in order to setup a clean Ubuntu OS - - Then packer uses ssh to enter the virtual machine to execute the required commands - - `scripts/debian_family/install_ansible.sh` sets up ansible on the vm. - - Depending on the VM being built, the `vm-develop.json` or the `vm-production.json` is used - - `scripts/set_message.sh` sets welcome message (with update instructions) in the vm. - - `scripts/cleanup.sh` writes zeros to all the free space in the disk, it shrinks the disk image -* Set the correct permissions for the built Vagrant and Virtual Appliance Images -* Cleanup - - Delete the generated files from the required directories -* Restart nginx - - -Running the `build.py` script builds the VMs and puts them in `~/Public`. It also creates the md5 hashes for the same, and puts them in the same folder. diff --git a/vm/Vagrantfile b/vm/Vagrantfile deleted file mode 100644 index 19fe518f..00000000 --- a/vm/Vagrantfile +++ /dev/null @@ -1,73 +0,0 @@ -# -*- mode: ruby -*- -# vi: set ft=ruby : - -# All Vagrant configuration is done below. The "2" in Vagrant.configure -# configures the configuration version (we support older styles for -# backwards compatibility). Please don't change it unless you know what -# you're doing. -Vagrant.configure(2) do |config| - # The most common configuration options are documented and commented below. - # For a complete reference, please see the online documentation at - # https://docs.vagrantup.com. - - # Every Vagrant development environment requires a box. You can search for - # boxes at https://atlas.hashicorp.com/search. - config.vm.box = "erpnext" - config.ssh.username = "frappe" - config.ssh.password = "frappe" - - # Disable automatic box update checking. If you disable this, then - # boxes will only be checked for updates when the user runs - # `vagrant box outdated`. This is not recommended. - # config.vm.box_check_update = false - - # Create a forwarded port mapping which allows access to a specific port - # within the machine from a port on the host machine. In the example below, - # accessing "localhost:8080" will access port 80 on the guest machine. - # config.vm.network "forwarded_port", guest: 80, host: 8080 - - # Create a private network, which allows host-only access to the machine - # using a specific IP. - # config.vm.network "private_network", ip: "192.168.33.10" - - # Create a public network, which generally matched to bridged network. - # Bridged networks make the machine appear as another physical device on - # your network. - # config.vm.network "public_network" - - # Share an additional folder to the guest VM. The first argument is - # the path on the host to the actual folder. The second argument is - # the path on the guest to mount the folder. And the optional third - # argument is a set of non-required options. - # config.vm.synced_folder "../data", "/vagrant_data" - - # Provider-specific configuration so you can fine-tune various - # backing providers for Vagrant. These expose provider-specific options. - # Example for VirtualBox: - # - # config.vm.provider "virtualbox" do |vb| - # # Display the VirtualBox GUI when booting the machine - # vb.gui = true - # - # # Customize the amount of memory on the VM: - # vb.memory = "1024" - # end - # - # View the documentation for the provider you are using for more - # information on available options. - - # Define a Vagrant Push strategy for pushing to Atlas. Other push strategies - # such as FTP and Heroku are also available. See the documentation at - # https://docs.vagrantup.com/v2/push/atlas.html for more information. - # config.push.define "atlas" do |push| - # push.app = "YOUR_ATLAS_USERNAME/YOUR_APPLICATION_NAME" - # end - - # Enable provisioning with a shell script. Additional provisioners such as - # Puppet, Chef, Ansible, Salt, and Docker are also available. Please see the - # documentation for more information about their specific syntax and use. - # config.vm.provision "shell", inline: <<-SHELL - # sudo apt-get update - # sudo apt-get install -y apache2 - # SHELL -end diff --git a/vm/build.py b/vm/build.py deleted file mode 100644 index 0fc0d6d0..00000000 --- a/vm/build.py +++ /dev/null @@ -1,128 +0,0 @@ -""" -Builds a vm and puts it in ~/public with a latest.json that has its filename and md5sum -""" - -# imports - standard imports -import os -import json -import stat -import errno -from shutil import rmtree -from distutils import spawn -from subprocess import check_output - -NEW_FILES = [] -BUILDS = ['Production', 'Developer'] -PUBLIC_DIR = os.path.join(os.path.expanduser('~'), 'Public') -SYMLINKS = ['ERPNext-Production.ova', 'ERPNext-Dev.ova', 'ERPNext-Vagrant.box', - 'ERPNext-Production.ova.md5', 'ERPNext-Dev.ova.md5', 'ERPNext-Vagrant.box.md5'] - -def main(): - install_virtualbox() - install_packer() - cleanup() - build_vm() - generate_md5_hashes() - generate_symlinks() - delete_old_vms() - move_current_vms() - cleanup() - -def install_virtualbox(): - if not spawn.find_executable("virtualbox"): - check_output(['bench', 'install', 'virtualbox']) - -def install_packer(): - if not spawn.find_executable("packer") and not os.path.exists(os.path.join('/', 'opt', 'packer')): - check_output(['bench', 'install', 'packer']) - -def silent_remove(name, is_dir=False): - ''' - Method to safely remove a file or directory, - without throwing error if file doesn't exist - - By default takes in file as input, for directory: - is_dir = True - ''' - try: - if is_dir: - rmtree(name) - else: - os.remove(name) - except OSError as e: - if e.errno != errno.ENOENT: # errno.ENOENT = no such file or directory - raise # re-raise exception if a different error occurred - -def cleanup(): - silent_remove("Production Builds", is_dir=True) - silent_remove("Developer Builds", is_dir=True) - silent_remove("packer_virtualbox-iso_virtualbox-iso_md5.checksum") - -def build_vm(): - check_output(["packer", "build", "vm-production.json"]) - check_output(["packer", "build", "vm-develop.json"]) - -def md5(build, file): - return check_output("md5sum '{} Builds/{}'".format(build, file), shell=True).split()[0] - -def move_to_public(build, file): - NEW_FILES.append(file) - src = os.path.join('{} Builds/{}'.format(build, file)) - dest = os.path.join(PUBLIC_DIR, file) - os.rename(src, dest) - # Make Public folder readable by others - st = os.stat(dest) - os.chmod(dest, st.st_mode | stat.S_IROTH) - -def generate_md5_hashes(): - for build in BUILDS: - for file in os.listdir('{} Builds'.format(build)): - if file.endswith(".ova") or file.endswith(".box"): - with open('{} Builds/{}.md5'.format(build, file), 'w') as f: - f.write(md5(build, file)) - move_to_public(build, file) - move_to_public(build, '{}.md5'.format(file)) - -def generate_symlinks(): - for file in NEW_FILES: - if 'md5' in file: - if 'Vagrant' in file: - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Vagrant.box.md5')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Vagrant.box.md5')) - elif 'Production' in file: - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Production.ova.md5')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Production.ova.md5')) - else: # Develop - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Dev.ova.md5')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Dev.ova.md5')) - else: # ova/box files - if 'Vagrant' in file: - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Vagrant.box')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Vagrant.box')) - elif 'Production' in file: - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Production.ova')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Production.ova')) - else: # Develop - silent_remove(os.path.join(PUBLIC_DIR, 'ERPNext-Dev.ova')) - os.symlink(os.path.join(PUBLIC_DIR, file), - os.path.join(PUBLIC_DIR, 'ERPNext-Dev.ova')) - -def delete_old_vms(): - silent_remove(os.path.join(PUBLIC_DIR, 'BACKUPS'), is_dir=True) - -def move_current_vms(): - os.mkdir(os.path.join(PUBLIC_DIR, 'BACKUPS')) - for file in os.listdir(PUBLIC_DIR): - if file in NEW_FILES or file in SYMLINKS or file == 'BACKUPS': - continue - src = os.path.join(PUBLIC_DIR, '{}'.format(file)) - dest = os.path.join(PUBLIC_DIR, 'BACKUPS/{}'.format(file)) - os.rename(src, dest) - -if __name__ == "__main__": - main() diff --git a/vm/http/preseed.cfg b/vm/http/preseed.cfg deleted file mode 100644 index 87c4622e..00000000 --- a/vm/http/preseed.cfg +++ /dev/null @@ -1,41 +0,0 @@ -choose-mirror-bin mirror/http/proxy string -choose-mirror-bin mirror/http/proxy string -d-i base-installer/kernel/override-image string linux-server -d-i clock-setup/utc boolean true -d-i clock-setup/utc-auto boolean true -d-i finish-install/reboot_in_progress note -d-i grub-installer/only_debian boolean true -d-i grub-installer/with_other_os boolean true -d-i partman-auto/disk string /dev/sda -d-i partman-auto-lvm/guided_size string max -d-i partman-auto/choose_recipe select atomic -d-i partman-auto/method string lvm -d-i partman-lvm/confirm boolean true -d-i partman-lvm/confirm boolean true -d-i partman-lvm/confirm_nooverwrite boolean true -d-i partman-lvm/device_remove_lvm boolean true -d-i partman/choose_partition select finish -d-i partman/confirm boolean true -d-i partman/confirm_nooverwrite boolean true -d-i partman/confirm_write_new_label boolean true -d-i pkgsel/include string openssh-server cryptsetup build-essential libssl-dev libreadline-dev zlib1g-dev linux-source dkms nfs-common -d-i pkgsel/install-language-support boolean false -d-i pkgsel/update-policy select none -d-i pkgsel/upgrade select full-upgrade -d-i time/zone string UTC -tasksel tasksel/first multiselect standard, ubuntu-server - -d-i console-setup/ask_detect boolean false -d-i keyboard-configuration/layoutcode string us -d-i keyboard-configuration/modelcode string pc105 -d-i debian-installer/locale string en_US - -# Create frappe user account. -d-i passwd/user-fullname string frappe -d-i passwd/username string frappe -d-i passwd/user-password password frappe -d-i passwd/user-password-again password frappe -d-i user-setup/allow-password-weak boolean true -d-i user-setup/encrypt-home boolean false -d-i passwd/user-default-groups frappe sudo -d-i passwd/user-uid string 900 diff --git a/vm/scripts/debian_family/cleanup.sh b/vm/scripts/debian_family/cleanup.sh deleted file mode 100644 index d70c7a09..00000000 --- a/vm/scripts/debian_family/cleanup.sh +++ /dev/null @@ -1,13 +0,0 @@ -#!/bin/bash -eux - -# Apt cleanup -apt-get autoremove -apt-get clean -apt-get update - -# Zero out the rest of the free space using dd, then delete the written file. -dd if=/dev/zero of=/EMPTY bs=1M -rm -f /EMPTY - -# Add `sync` so Packer doesn't quit too early, before the large file is deleted. -sync \ No newline at end of file diff --git a/vm/scripts/debian_family/install_ansible.sh b/vm/scripts/debian_family/install_ansible.sh deleted file mode 100644 index c173dd94..00000000 --- a/vm/scripts/debian_family/install_ansible.sh +++ /dev/null @@ -1,10 +0,0 @@ -#!/bin/bash -eux - -# Install Ansible repository. -apt -y update && apt-get -y upgrade -apt -y install software-properties-common -apt-add-repository ppa:ansible/ansible - -# Install Ansible. -apt -y update -apt -y install ansible diff --git a/vm/scripts/debian_family/install_erpnext_develop.sh b/vm/scripts/debian_family/install_erpnext_develop.sh deleted file mode 100644 index 19b1b705..00000000 --- a/vm/scripts/debian_family/install_erpnext_develop.sh +++ /dev/null @@ -1,5 +0,0 @@ -#!/bin/bash -eux - -# Install ERPNext -wget https://raw.githubusercontent.com/frappe/bench/develop/install.py -python install.py --develop --user frappe --mysql-root-password frappe --admin-password admin \ No newline at end of file diff --git a/vm/scripts/debian_family/install_erpnext_production.sh b/vm/scripts/debian_family/install_erpnext_production.sh deleted file mode 100644 index b8493f1e..00000000 --- a/vm/scripts/debian_family/install_erpnext_production.sh +++ /dev/null @@ -1,5 +0,0 @@ -#!/bin/bash -eux - -# Install ERPNext -wget https://raw.githubusercontent.com/frappe/bench/develop/install.py -python install.py --production --user frappe --mysql-root-password frappe --admin-password admin \ No newline at end of file diff --git a/vm/scripts/debian_family/install_prerequisites.sh b/vm/scripts/debian_family/install_prerequisites.sh deleted file mode 100644 index 0aed12fd..00000000 --- a/vm/scripts/debian_family/install_prerequisites.sh +++ /dev/null @@ -1,4 +0,0 @@ -#!/bin/bash -eux - -# Install base requirements. -apt-get install -y curl git wget vim python-dev gcc diff --git a/vm/scripts/debian_family/setup.sh b/vm/scripts/debian_family/setup.sh deleted file mode 100644 index 743300e1..00000000 --- a/vm/scripts/debian_family/setup.sh +++ /dev/null @@ -1,8 +0,0 @@ -#!/bin/bash -eux - -# Add frappe user to sudoers. -echo "frappe ALL=(ALL) NOPASSWD: ALL" >> /etc/sudoers -sed -i "s/^.*requiretty/#Defaults requiretty/" /etc/sudoers - -# Disable daily apt unattended updates. -echo 'APT::Periodic::Enable "0";' >> /etc/apt/apt.conf.d/10periodic diff --git a/vm/scripts/restart_supervisor.sh b/vm/scripts/restart_supervisor.sh deleted file mode 100644 index 0afad870..00000000 --- a/vm/scripts/restart_supervisor.sh +++ /dev/null @@ -1,14 +0,0 @@ -#! /bin/bash - - -# Write out current crontab -crontab -l > current_cron - -# Echo new cron into cron file -echo "@reboot sleep 20 && systemctl restart supervisor" >> current_cron - -# Install new cron file -crontab current_cron - -# Delete the temporary cron file -rm current_cron \ No newline at end of file diff --git a/vm/scripts/set_message_develop.sh b/vm/scripts/set_message_develop.sh deleted file mode 100644 index a1e94353..00000000 --- a/vm/scripts/set_message_develop.sh +++ /dev/null @@ -1,18 +0,0 @@ -#! /bin/bash - -message=" - ERPNext Evaluation VM (built on `date +\"%B %d, %Y\"`) - - Please access ERPNext by going to http://localhost:8000 on the host system. - The username is \"Administrator\" and password is \"admin\" - - Do consider donating at https://frappe.io/buy - - To update, login as - username: frappe - password: frappe - cd frappe-bench - bench update -" -echo "$message" | sudo tee -a /etc/issue -echo "$message" | sudo tee -a /etc/motd diff --git a/vm/scripts/set_message_production.sh b/vm/scripts/set_message_production.sh deleted file mode 100644 index 2fddcb77..00000000 --- a/vm/scripts/set_message_production.sh +++ /dev/null @@ -1,18 +0,0 @@ -#! /bin/bash - -message=" - ERPNext VM (built on `date +\"%B %d, %Y\"`) - - Please access ERPNext by going to http://localhost:8080 on the host system. - The username is \"Administrator\" and password is \"admin\" - - Consider buying professional support from us at https://erpnext.com/support - - To update, login as - username: frappe - password: frappe - cd frappe-bench - bench update -" -echo "$message" | sudo tee -a /etc/issue -echo "$message" | sudo tee -a /etc/motd diff --git a/vm/vm-develop.json b/vm/vm-develop.json deleted file mode 100644 index 1a70c44a..00000000 --- a/vm/vm-develop.json +++ /dev/null @@ -1,101 +0,0 @@ -{ - "builders": [{ - "vm_name": "ERPNext-Develop-{{isotime \"20060102150405\"}}", - "output_directory": "Developer Builds", - "type": "virtualbox-iso", - "boot_command": [ - "", - "", - "", - "", - "/install/vmlinuz", - " auto", - " console-setup/ask_detect=false", - " console-setup/layoutcode=us", - " console-setup/modelcode=pc105", - " debconf/frontend=noninteractive", - " debian-installer=en_US", - " fb=false", - " initrd=/install/initrd.gz", - " kbd-chooser/method=us", - " keyboard-configuration/layout=USA", - " keyboard-configuration/variant=USA", - " locale=en_US", - " netcfg/get_domain=vm", - " netcfg/get_hostname=ubuntu", - " grub-installer/bootdev=/dev/sda", - " noapic", - " preseed/url=http://{{ .HTTPIP }}:{{ .HTTPPort }}/preseed.cfg", - " -- ", - "" - ], - "boot_wait": "10s", - "format": "ova", - "guest_os_type": "Ubuntu_64", - "headless": true, - "iso_url": "http://releases.ubuntu.com/16.04/ubuntu-16.04.6-server-amd64.iso", - "iso_checksum": "ac8a79a86a905ebdc3ef3f5dd16b7360", - "iso_checksum_type": "md5", - "ssh_username": "frappe", - "ssh_password": "frappe", - "ssh_port": 22, - "ssh_wait_timeout": "10000s", - "http_directory": "http", - "guest_additions_mode": "disable", - "virtualbox_version_file": ".vbox_version", - "guest_additions_path": "VBoxGuestAdditions_{{.Version}}.iso", - "export_opts": [ - "--vsys", "0", - "--product", "ERPNext", - "--producturl", "https://erpnext.com", - "--vendor", "Frappe Techonologies", - "--vendorurl", "https://frappe.io", - "--description", "ERPNext Evaluation VM" - ], - "shutdown_command": "echo 'frappe'|sudo -S shutdown -P now", - "vboxmanage": [ - [ "modifyvm", "{{.Name}}", "--memory", "1024" ], - [ "modifyvm", "{{.Name}}", "--cpus", "1" ], - [ "modifyvm", "{{.Name}}", "--audio", "none" ], - [ "modifyvm", "{{.Name}}", "--natpf1", "vm_ssh,tcp,,3022,,22" ], - [ "modifyvm", "{{.Name}}", "--natpf1", "vm_http,tcp,,8080,,80" ], - [ "modifyvm", "{{.Name}}", "--natpf1", "vm_http2,tcp,,8000,,8000" ] - ] - }], - "provisioners": [{ - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/install_ansible.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/setup.sh" - },{ - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/install_prerequisites.sh" - }, { - "type": "shell", - "script": "scripts/debian_family/install_erpnext_develop.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/cleanup.sh" - }, { - "type": "shell", - "script": "scripts/set_message_develop.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/restart_supervisor.sh" - }], - "post-processors": [{ - "type": "checksum", - "checksum_types": ["md5"] - }, { - "type": "vagrant", - "keep_input_artifact": true, - "output": "Developer Builds/ERPNext-Vagrant-Develop-{{isotime \"20060102150405\"}}.box", - "vagrantfile_template": "Vagrantfile" - }] -} diff --git a/vm/vm-production.json b/vm/vm-production.json deleted file mode 100644 index c2883d54..00000000 --- a/vm/vm-production.json +++ /dev/null @@ -1,95 +0,0 @@ -{ - "builders": [{ - "vm_name": "ERPNext-Production-{{isotime \"20060102150405\"}}", - "output_directory": "Production Builds", - "type": "virtualbox-iso", - "boot_command": [ - "", - "", - "", - "", - "/install/vmlinuz", - " auto", - " console-setup/ask_detect=false", - " console-setup/layoutcode=us", - " console-setup/modelcode=pc105", - " debconf/frontend=noninteractive", - " debian-installer=en_US", - " fb=false", - " initrd=/install/initrd.gz", - " kbd-chooser/method=us", - " keyboard-configuration/layout=USA", - " keyboard-configuration/variant=USA", - " locale=en_US", - " netcfg/get_domain=vm", - " netcfg/get_hostname=ubuntu", - " grub-installer/bootdev=/dev/sda", - " noapic", - " preseed/url=http://{{ .HTTPIP }}:{{ .HTTPPort }}/preseed.cfg", - " -- ", - "" - ], - "boot_wait": "10s", - "format": "ova", - "guest_os_type": "Ubuntu_64", - "headless": true, - "iso_url": "http://releases.ubuntu.com/16.04/ubuntu-16.04.6-server-amd64.iso", - "iso_checksum": "ac8a79a86a905ebdc3ef3f5dd16b7360", - "iso_checksum_type": "md5", - "ssh_username": "frappe", - "ssh_password": "frappe", - "ssh_port": 22, - "ssh_wait_timeout": "10000s", - "http_directory": "http", - "guest_additions_mode": "disable", - "virtualbox_version_file": ".vbox_version", - "guest_additions_path": "VBoxGuestAdditions_{{.Version}}.iso", - "export_opts": [ - "--vsys", "0", - "--product", "ERPNext", - "--producturl", "https://erpnext.com", - "--vendor", "Frappe Techonologies", - "--vendorurl", "https://frappe.io", - "--description", "ERPNext Evaluation VM" - ], - "shutdown_command": "echo 'frappe'|sudo -S shutdown -P now", - "vboxmanage": [ - [ "modifyvm", "{{.Name}}", "--memory", "1024" ], - [ "modifyvm", "{{.Name}}", "--cpus", "1" ], - [ "modifyvm", "{{.Name}}", "--audio", "none" ], - [ "modifyvm", "{{.Name}}", "--natpf1", "vm_ssh,tcp,,3022,,22" ], - [ "modifyvm", "{{.Name}}", "--natpf1", "vm_http,tcp,,8080,,80" ] - ] - }], - "provisioners": [{ - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/install_ansible.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/setup.sh" - },{ - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/install_prerequisites.sh" - }, { - "type": "shell", - "script": "scripts/debian_family/install_erpnext_production.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/debian_family/cleanup.sh" - }, { - "type": "shell", - "script": "scripts/set_message_production.sh" - }, { - "type": "shell", - "execute_command": "echo 'frappe' | {{.Vars}} sudo -S -E bash '{{.Path}}'", - "script": "scripts/restart_supervisor.sh" - }], - "post-processors": [{ - "type": "checksum", - "checksum_types": ["md5"] - }] -} From 4fcda9ae22a9d7435f6d43c95f33fc455a84aed8 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Tue, 14 Jun 2022 15:41:22 +0530 Subject: [PATCH 3/5] chore: Drop release project Release is yet another dead / unmaintained project in bench. It used to be in action - the tool to release ERPNext & Frappe for years but we've moved to other automated pipelines (that have been around for a while too). Time to say good-bye to these commands too :wave: --- bench/commands/__init__.py | 4 - bench/commands/utils.py | 23 --- bench/prepare_beta_release.py | 118 ----------- bench/release.py | 367 ---------------------------------- 4 files changed, 512 deletions(-) delete mode 100644 bench/prepare_beta_release.py delete mode 100755 bench/release.py diff --git a/bench/commands/__init__.py b/bench/commands/__init__.py index 08a493b9..f44f08be 100755 --- a/bench/commands/__init__.py +++ b/bench/commands/__init__.py @@ -74,8 +74,6 @@ from bench.commands.utils import ( find_benches, generate_command_cache, migrate_env, - prepare_beta_release, - release, renew_lets_encrypt, restart, set_mariadb_host, @@ -102,11 +100,9 @@ bench_command.add_command(set_redis_socketio_host) bench_command.add_command(download_translations) bench_command.add_command(backup_site) bench_command.add_command(backup_all_sites) -bench_command.add_command(release) bench_command.add_command(renew_lets_encrypt) bench_command.add_command(disable_production) bench_command.add_command(bench_src) -bench_command.add_command(prepare_beta_release) bench_command.add_command(find_benches) bench_command.add_command(migrate_env) bench_command.add_command(generate_command_cache) diff --git a/bench/commands/utils.py b/bench/commands/utils.py index b2d8a301..35952b3e 100644 --- a/bench/commands/utils.py +++ b/bench/commands/utils.py @@ -128,29 +128,6 @@ def backup_all_sites(): backup_all_sites(bench_path='.') -@click.command('release', help="Release a Frappe app (internal to the Frappe team)") -@click.argument('app') -@click.argument('bump-type', type=click.Choice(['major', 'minor', 'patch', 'stable', 'prerelease'])) -@click.option('--from-branch', default='develop') -@click.option('--to-branch', default='master') -@click.option('--remote', default='upstream') -@click.option('--owner', default='frappe') -@click.option('--repo-name') -@click.option('--dont-frontport', is_flag=True, default=False, help='Front port fixes to new branches, example merging hotfix(v10) into staging-fixes(v11)') -def release(app, bump_type, from_branch, to_branch, owner, repo_name, remote, dont_frontport): - from bench.release import release - frontport = not dont_frontport - release(bench_path='.', app=app, bump_type=bump_type, from_branch=from_branch, to_branch=to_branch, remote=remote, owner=owner, repo_name=repo_name, frontport=frontport) - - -@click.command('prepare-beta-release', help="Prepare major beta release from develop branch") -@click.argument('app') -@click.option('--owner', default='frappe') -def prepare_beta_release(app, owner): - from bench.prepare_beta_release import prepare_beta_release - prepare_beta_release(bench_path='.', app=app, owner=owner) - - @click.command('disable-production', help="Disables production environment for the bench.") def disable_production(): from bench.config.production_setup import disable_production diff --git a/bench/prepare_beta_release.py b/bench/prepare_beta_release.py deleted file mode 100644 index 52054b9c..00000000 --- a/bench/prepare_beta_release.py +++ /dev/null @@ -1,118 +0,0 @@ -#! env python -import os -import git -import click -from .config.common_site_config import get_config -import semantic_version - -github_username = None -github_password = None - -def prepare_beta_release(bench_path, app, owner='frappe', remote='upstream'): - from .release import get_release_message - - beta_hotfix = '' - beta_master = click.prompt('Branch name for beta release', type=str) - - if click.confirm("Do you want to setup hotfix for beta ?"): - beta_hotfix = click.prompt(f'Branch name for beta hotfix ({beta_master}_hotifx)', type=str) - - validate(bench_path) - repo_path = os.path.join(bench_path, 'apps', app) - version = get_bummped_version(repo_path) - - update_branch(repo_path, remote) - prepare_beta_master(repo_path, beta_master, version, remote) - - if beta_hotfix: - prepare_beta_hotfix(repo_path, beta_hotfix, remote) - - tag_name = merge_beta_release_to_develop(repo_path, beta_master, remote, version) - push_branches(repo_path, beta_master, beta_hotfix, remote) - create_github_release(repo_path, tag_name, '', owner, remote) - -def validate(bench_path): - from .release import validate - - config = get_config(bench_path) - validate(bench_path, config) - -def get_bummped_version(repo_path): - from .release import get_current_version - current_version = get_current_version(repo_path, 'master') - - v = semantic_version.Version(current_version) - - if v.major: - v.major += 1 - v.minor = 0 - v.patch = 0 - v.prerelease = ['staging'] - - return str(v) - -def update_branch(repo_path, remote): - from .release import update_branch - update_branch(repo_path, 'develop', remote) - -def prepare_beta_master(repo_path, beta_master, version, remote): - g = git.Repo(repo_path).git - g.checkout(b=beta_master) - - set_beta_version(repo_path, version) - -def set_beta_version(repo_path, version): - from .release import set_filename_version - set_filename_version(os.path.join(repo_path, os.path.basename(repo_path),'hooks.py'), version, 'staging_version') - - repo = git.Repo(repo_path) - app_name = os.path.basename(repo_path) - repo.index.add([os.path.join(app_name, 'hooks.py')]) - repo.index.commit(f'bumped to version {version}') - - -def prepare_beta_hotfix(repo_path, beta_hotfix, remote): - g = git.Repo(repo_path).git - g.checkout(b=beta_hotfix) - - -def merge_beta_release_to_develop(repo_path, beta_master, remote, version): - from .release import handle_merge_error - - repo = git.Repo(repo_path) - g = repo.git - - tag_name = 'v' + version - repo.create_tag(tag_name, message=f'Release {version}') - - g.checkout('develop') - - try: - g.merge(beta_master) - except git.exc.GitCommandError as e: - handle_merge_error(e, source=beta_master, target='develop') - - return tag_name - -def push_branches(repo_path, beta_master, beta_hotfix, remote): - repo = git.Repo(repo_path) - g = repo.git - - args = [ - 'develop:develop', - f'{beta_master}:{beta_master}', - ] - - if beta_hotfix: - args.append(f'{beta_hotfix}:{beta_hotfix}') - - args.append('--tags') - - print("Pushing branches") - print(g.push(remote, *args)) - -def create_github_release(repo_path, tag_name, message, owner, remote): - from .release import create_github_release - - create_github_release(repo_path, tag_name, message, remote=remote, owner=owner, - repo_name=None, gh_username=github_username, gh_password=github_password) \ No newline at end of file diff --git a/bench/release.py b/bench/release.py deleted file mode 100755 index 303795d8..00000000 --- a/bench/release.py +++ /dev/null @@ -1,367 +0,0 @@ -#! env python -import json -import os -import sys -import semantic_version -import git -import getpass -import re -from time import sleep - -from bench.exceptions import ValidationError -from .config.common_site_config import get_config -import click - -branches_to_update = { - 'develop': [], - 'version-11-hotfix': [], - 'version-12-hotfix': [], -} - -releasable_branches = ['master'] - -github_username = None -github_password = None - -def release(bench_path, app, bump_type, from_branch, to_branch, - remote='upstream', owner='frappe', repo_name=None, frontport=True): - - confirm_testing() - config = get_config(bench_path) - - if not config.get('release_bench'): - print('bench not configured to release') - sys.exit(1) - - - if config.get('branches_to_update'): - branches_to_update.update(config.get('branches_to_update')) - - if config.get('releasable_branches'): - releasable_branches.extend(config.get('releasable_branches',[])) - - validate(bench_path, config) - - bump(bench_path, app, bump_type, from_branch=from_branch, to_branch=to_branch, owner=owner, - repo_name=repo_name, remote=remote, frontport=frontport) - -def validate(bench_path, config): - import requests - from requests.auth import HTTPBasicAuth - - global github_username, github_password - - github_username = config.get('github_username') - github_password = config.get('github_password') - - if not github_username: - github_username = click.prompt('Username', type=str) - - if not github_password: - github_password = getpass.getpass() - - r = requests.get('https://api.github.com/user', auth=HTTPBasicAuth(github_username, github_password)) - r.raise_for_status() - -def confirm_testing(): - print('') - print('================ CAUTION ==================') - print('Never miss this, even if it is a really small release!!') - print('Manual Testing Checklisk: https://github.com/frappe/bench/wiki/Testing-Checklist') - print('') - print('') - click.confirm('Is manual testing done?', abort = True) - click.confirm('Have you added the change log?', abort = True) - -def bump(bench_path, app, bump_type, from_branch, to_branch, remote, owner, repo_name=None, frontport=True): - assert bump_type in ['minor', 'major', 'patch', 'stable', 'prerelease'] - - repo_path = os.path.join(bench_path, 'apps', app) - push_branch_for_old_major_version(bench_path, bump_type, app, repo_path, from_branch, to_branch, remote, owner) - update_branches_and_check_for_changelog(repo_path, from_branch, to_branch, remote=remote) - message = get_release_message(repo_path, from_branch=from_branch, to_branch=to_branch, remote=remote) - - if not message: - print('No commits to release') - return - - print(message) - - click.confirm('Do you want to continue?', abort=True) - - new_version = bump_repo(repo_path, bump_type, from_branch=from_branch, to_branch=to_branch) - commit_changes(repo_path, new_version, to_branch) - tag_name = create_release(repo_path, new_version, from_branch=from_branch, to_branch=to_branch, frontport=frontport) - push_release(repo_path, from_branch=from_branch, to_branch=to_branch, remote=remote) - prerelease = True if 'beta' in new_version else False - create_github_release(repo_path, tag_name, message, remote=remote, owner=owner, repo_name=repo_name, prerelease=prerelease) - print(f'Released {tag_name} for {repo_path}') - -def update_branches_and_check_for_changelog(repo_path, from_branch, to_branch, remote='upstream'): - - update_branch(repo_path, to_branch, remote=remote) - update_branch(repo_path, from_branch, remote=remote) - - for branch in branches_to_update[from_branch]: - update_branch(repo_path, branch, remote=remote) - - git.Repo(repo_path).git.checkout(from_branch) - check_for_unmerged_changelog(repo_path) - -def update_branch(repo_path, branch, remote): - print("updating local branch of", repo_path, 'using', remote + '/' + branch) - - repo = git.Repo(repo_path) - g = repo.git - g.fetch(remote) - g.checkout(branch) - g.reset('--hard', remote+'/'+branch) - -def check_for_unmerged_changelog(repo_path): - current = os.path.join(repo_path, os.path.basename(repo_path), 'change_log', 'current') - if os.path.exists(current) and [f for f in os.listdir(current) if f != "readme.md"]: - raise Exception("Unmerged change log! in " + repo_path) - -def get_release_message(repo_path, from_branch, to_branch, remote='upstream'): - print('getting release message for', repo_path, 'comparing', to_branch, '...', from_branch) - - repo = git.Repo(repo_path) - g = repo.git - log = g.log(f'{remote}/{to_branch}..{remote}/{from_branch}', '--format=format:%s', '--no-merges') - - if log: - return "* " + log.replace('\n', '\n* ') - -def bump_repo(repo_path, bump_type, from_branch, to_branch): - current_version = get_current_version(repo_path, to_branch) - new_version = get_bumped_version(current_version, bump_type) - - print('bumping version from', current_version, 'to', new_version) - - set_version(repo_path, new_version, to_branch) - return new_version - -def get_current_version(repo_path, to_branch): - # TODO clean this up! - version_key = '__version__' - - if to_branch.lower() in releasable_branches: - filename = os.path.join(repo_path, os.path.basename(repo_path), '__init__.py') - else: - filename = os.path.join(repo_path, os.path.basename(repo_path), 'hooks.py') - version_key = 'staging_version' - - with open(filename) as f: - contents = f.read() - match = re.search(r"^(\s*%s\s*=\s*['\\\"])(.+?)(['\"])(?sm)" % version_key, - contents) - return match.group(2) - -def get_bumped_version(version, bump_type): - v = semantic_version.Version(version) - if bump_type == 'major': - v.major += 1 - v.minor = 0 - v.patch = 0 - v.prerelease = None - - elif bump_type == 'minor': - v.minor += 1 - v.patch = 0 - v.prerelease = None - - elif bump_type == 'patch': - if v.prerelease == (): - v.patch += 1 - v.prerelease = None - - elif len(v.prerelease) == 2: - v.prerelease = () - - elif bump_type == 'stable': - # remove pre-release tag - v.prerelease = None - - elif bump_type == 'prerelease': - if v.prerelease == (): - v.patch += 1 - v.prerelease = ('beta', '1') - - elif len(v.prerelease) == 2: - v.prerelease = ('beta', str(int(v.prerelease[1]) + 1)) - - else: - raise ValidationError("Something wen't wrong while doing a prerelease") - - else: - raise ValidationError("bump_type not amongst [major, minor, patch, prerelease]") - - return str(v) - -def set_version(repo_path, version, to_branch): - if to_branch.lower() in releasable_branches: - set_filename_version(os.path.join(repo_path, os.path.basename(repo_path),'__init__.py'), version, '__version__') - else: - set_filename_version(os.path.join(repo_path, os.path.basename(repo_path),'hooks.py'), version, 'staging_version') - - # TODO fix this - # set_setuppy_version(repo_path, version) - # set_versionpy_version(repo_path, version) - # set_hooks_version(repo_path, version) - -# def set_setuppy_version(repo_path, version): -# set_filename_version(os.path.join(repo_path, 'setup.py'), version, 'version') -# -# def set_versionpy_version(repo_path, version): -# set_filename_version(os.path.join(repo_path, os.path.basename(repo_path),'__version__.py'), version, '__version__') -# -# def set_hooks_version(repo_path, version): -# set_filename_version(os.path.join(repo_path, os.path.basename(repo_path),'hooks.py'), version, 'app_version') - -def set_filename_version(filename, version_number, pattern): - changed = [] - - def inject_version(match): - before, old, after = match.groups() - changed.append(True) - return before + version_number + after - - with open(filename) as f: - contents = re.sub(r"^(\s*%s\s*=\s*['\\\"])(.+?)(['\"])(?sm)" % pattern, - inject_version, f.read()) - - if not changed: - raise Exception('Could not find %s in %s', pattern, filename) - - with open(filename, 'w') as f: - f.write(contents) - -def commit_changes(repo_path, new_version, to_branch): - print('committing version change to', repo_path) - - repo = git.Repo(repo_path) - app_name = os.path.basename(repo_path) - - if to_branch.lower() in releasable_branches: - repo.index.add([os.path.join(app_name, '__init__.py')]) - else: - repo.index.add([os.path.join(app_name, 'hooks.py')]) - - repo.index.commit(f'bumped to version {new_version}') - -def create_release(repo_path, new_version, from_branch, to_branch, frontport=True): - print('creating release for version', new_version) - repo = git.Repo(repo_path) - g = repo.git - g.checkout(to_branch) - try: - g.merge(from_branch, '--no-ff') - except git.exc.GitCommandError as e: - handle_merge_error(e, source=from_branch, target=to_branch) - - tag_name = 'v' + new_version - repo.create_tag(tag_name, message=f'Release {new_version}') - g.checkout(from_branch) - - try: - g.merge(to_branch) - except git.exc.GitCommandError as e: - handle_merge_error(e, source=to_branch, target=from_branch) - - if frontport: - for branch in branches_to_update[from_branch]: - print (f"Front porting changes to {branch}") - print(f'merging {to_branch} into', branch) - g.checkout(branch) - try: - g.merge(to_branch) - except git.exc.GitCommandError as e: - handle_merge_error(e, source=to_branch, target=branch) - - return tag_name - -def handle_merge_error(e, source, target): - print('-'*80) - print(f'Error when merging {source} into {target}') - print(e) - print('You can open a new terminal, try to manually resolve the conflict/error and continue') - print('-'*80) - click.confirm('Have you manually resolved the error?', abort=True) - -def push_release(repo_path, from_branch, to_branch, remote='upstream'): - print('pushing branches', to_branch, from_branch, 'of', repo_path) - repo = git.Repo(repo_path) - g = repo.git - args = [ - f'{to_branch}:{to_branch}', - f'{from_branch}:{from_branch}' - ] - - for branch in branches_to_update[from_branch]: - print(f'pushing {branch} branch of', repo_path) - args.append(f'{branch}:{branch}') - - args.append('--tags') - - print(g.push(remote, *args)) - -def create_github_release(repo_path, tag_name, message, remote='upstream', owner='frappe', repo_name=None, - gh_username=None, gh_password=None, prerelease=False): - import requests - import requests.exceptions - from requests.auth import HTTPBasicAuth - - print('creating release on github') - - global github_username, github_password - if not (gh_username and gh_password): - if not (github_username and github_password): - raise Exception("No credentials") - gh_username = github_username - gh_password = github_password - - repo_name = repo_name or os.path.basename(repo_path) - data = { - 'tag_name': tag_name, - 'target_commitish': 'master', - 'name': 'Release ' + tag_name, - 'body': message, - 'draft': False, - 'prerelease': prerelease - } - for i in range(3): - try: - r = requests.post(f'https://api.github.com/repos/{owner}/{repo_name}/releases', - auth=HTTPBasicAuth(gh_username, gh_password), data=json.dumps(data)) - r.raise_for_status() - break - except requests.exceptions.HTTPError: - print('request failed, retrying....') - sleep(3*i + 1) - if i !=2: - continue - else: - print(r.json()) - raise - return r - -def push_branch_for_old_major_version(bench_path, bump_type, app, repo_path, from_branch, to_branch, remote, owner): - if bump_type != 'major': - return - - current_version = get_current_version(repo_path) - old_major_version_branch = f"v{current_version.split('.')[0]}.x.x" - - click.confirm(f'Do you want to push {old_major_version_branch}?', abort=True) - - update_branch(repo_path, to_branch, remote=remote) - - g = git.Repo(repo_path).git - g.checkout(b=old_major_version_branch) - - args = [ - f'{old_major_version_branch}:{old_major_version_branch}', - ] - - print(f"Pushing {old_major_version_branch} ") - print(g.push(remote, *args)) From 21eceae6ac2994b40791e20d0b4626e41599dacb Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Tue, 14 Jun 2022 15:48:56 +0530 Subject: [PATCH 4/5] chore: Drop dead code tx [vulture](https://pypi.org/project/vulture/) --- bench/app.py | 27 --------------------------- bench/tests/test_utils.py | 1 - 2 files changed, 28 deletions(-) diff --git a/bench/app.py b/bench/app.py index f2117385..1dd0b8aa 100755 --- a/bench/app.py +++ b/bench/app.py @@ -13,8 +13,6 @@ from datetime import date from urllib.parse import urlparse import os -from markupsafe import soft_str - # imports - third party imports import click from git import Repo @@ -285,31 +283,6 @@ def make_resolution_plan(app: App, bench: "Bench"): return resolution -def add_to_appstxt(app, bench_path="."): - from bench.bench import Bench - - apps = Bench(bench_path).apps - - if app not in apps: - apps.append(app) - return write_appstxt(apps, bench_path=bench_path) - - -def remove_from_appstxt(app, bench_path="."): - from bench.bench import Bench - - apps = Bench(bench_path).apps - - if app in apps: - apps.remove(app) - return write_appstxt(apps, bench_path=bench_path) - - -def write_appstxt(apps, bench_path="."): - with open(os.path.join(bench_path, "sites", "apps.txt"), "w") as f: - return f.write("\n".join(apps)) - - def get_excluded_apps(bench_path="."): try: with open(os.path.join(bench_path, "sites", "excluded_apps.txt")) as f: diff --git a/bench/tests/test_utils.py b/bench/tests/test_utils.py index 0575fa9e..f34d161b 100644 --- a/bench/tests/test_utils.py +++ b/bench/tests/test_utils.py @@ -2,7 +2,6 @@ import os import shutil import subprocess import unittest -from tabnanny import check from bench.app import App from bench.bench import Bench From 24e2dd37fe7139a39c6437c10b207496fecebb36 Mon Sep 17 00:00:00 2001 From: Gavin D'souza Date: Tue, 14 Jun 2022 16:00:50 +0530 Subject: [PATCH 5/5] chore: Remove release related tests --- bench/tests/test_init.py | 13 ------------- 1 file changed, 13 deletions(-) diff --git a/bench/tests/test_init.py b/bench/tests/test_init.py index e9e9b8b4..978f63eb 100755 --- a/bench/tests/test_init.py +++ b/bench/tests/test_init.py @@ -9,7 +9,6 @@ import git # imports - module imports from bench.utils import exec_cmd -from bench.release import get_bumped_version from bench.app import App from bench.tests.test_base import FRAPPE_BRANCH, TestBenchBase from bench.bench import Bench @@ -21,18 +20,6 @@ from bench.bench import Bench TEST_FRAPPE_APP = "frappe_docs" class TestBenchInit(TestBenchBase): - def test_semantic_version(self): - self.assertEqual( get_bumped_version('11.0.4', 'major'), '12.0.0' ) - self.assertEqual( get_bumped_version('11.0.4', 'minor'), '11.1.0' ) - self.assertEqual( get_bumped_version('11.0.4', 'patch'), '11.0.5' ) - self.assertEqual( get_bumped_version('11.0.4', 'prerelease'), '11.0.5-beta.1' ) - - self.assertEqual( get_bumped_version('11.0.5-beta.22', 'major'), '12.0.0' ) - self.assertEqual( get_bumped_version('11.0.5-beta.22', 'minor'), '11.1.0' ) - self.assertEqual( get_bumped_version('11.0.5-beta.22', 'patch'), '11.0.5' ) - self.assertEqual( get_bumped_version('11.0.5-beta.22', 'prerelease'), '11.0.5-beta.23' ) - - def test_utils(self): self.assertEqual(subprocess.call("bench"), 0)