2019-12-18 09:03:21 +00:00
#!/usr/bin/env python3
import os , sys , subprocess , getpass , json , multiprocessing , shutil , platform , warnings
2016-03-15 06:54:17 +00:00
2019-12-19 08:31:24 +00:00
2019-12-27 07:44:33 +00:00
tmp_bench_repo = os . path . join ( ' / ' , ' tmp ' , ' .bench ' )
tmp_log_folder = os . path . join ( ' / ' , ' tmp ' , ' logs ' )
2019-12-18 09:03:21 +00:00
log_path = os . path . join ( tmp_log_folder , ' install_bench.log ' )
log_stream = sys . stdout
2019-12-19 09:08:46 +00:00
def log ( message , level = 0 ) :
2019-12-18 09:03:21 +00:00
levels = {
0 : ' \033 [94m ' , # normal
1 : ' \033 [92m ' , # success
2 : ' \033 [91m ' , # fail
3 : ' \033 [93m ' # warn/suggest
}
start = levels . get ( level ) or ' '
end = ' \033 [0m '
2019-12-19 09:08:46 +00:00
print ( start + message + end )
2019-12-18 09:03:21 +00:00
2019-12-27 07:44:33 +00:00
def setup_log_stream ( args ) :
global log_stream
sys . stderr = sys . stdout
if not args . verbose :
if not os . path . exists ( tmp_log_folder ) :
os . makedirs ( tmp_log_folder )
log_stream = open ( log_path , ' w ' )
log ( " Logs are saved under {0} " . format ( log_path ) , level = 3 )
2019-12-19 08:31:24 +00:00
def check_environment ( ) :
needed_environ_vars = [ ' LANG ' , ' LC_ALL ' ]
message = ' '
for var in needed_environ_vars :
if var not in os . environ :
2019-12-27 07:44:33 +00:00
message + = " \n export {0} =C.UTF-8 " . format ( var )
2019-12-19 08:31:24 +00:00
if message :
2019-12-19 09:08:46 +00:00
log ( " Bench ' s CLI needs these to be defined! " , level = 3 )
2019-12-27 07:44:33 +00:00
log ( " Run the following commands in shell: {0} " . format ( message ) , level = 2 )
2019-12-19 08:31:24 +00:00
sys . exit ( )
2019-12-27 07:44:33 +00:00
def check_system_package_managers ( ) :
if ' Darwin ' in os . uname ( ) :
if not shutil . which ( ' brew ' ) :
raise Exception ( '''
Please install brew package manager before proceeding with bench setup . Please run following
to install brew package manager on your machine ,
/ usr / bin / ruby - e " $(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install) "
''' )
if ' Linux ' in os . uname ( ) :
if not any ( [ shutil . which ( x ) for x in [ ' apt-get ' , ' yum ' ] ] ) :
raise Exception ( ' Cannot find any compatible package manager! ' )
def check_distribution_compatibility ( ) :
dist_name , dist_version = get_distribution_info ( )
supported_dists = {
' macos ' : [ 10.9 , 10.10 , 10.11 , 10.12 ] ,
' ubuntu ' : [ 14 , 15 , 16 , 18 , 19 ] ,
' debian ' : [ 8 , 9 ] ,
' centos ' : [ 7 ]
}
log ( " Checking System Compatibility... " )
if dist_name in supported_dists :
if float ( dist_version ) in supported_dists [ dist_name ] :
log ( " {0} {1} is compatible! " . format ( dist_name , dist_version ) , level = 1 )
else :
log ( " Install on {0} {1} instead " . format ( dist_name , supported_dists [ dist_name ] [ - 1 ] ) , level = 3 )
else :
log ( " Sorry, the installer doesn ' t support {0} . Aborting installation! " . format ( dist_name ) , level = 2 )
def get_distribution_info ( ) :
# return distribution name and major version
if platform . system ( ) == " Linux " :
current_dist = platform . dist ( )
return current_dist [ 0 ] . lower ( ) , current_dist [ 1 ] . rsplit ( ' . ' ) [ 0 ]
elif platform . system ( ) == " Darwin " :
current_dist = platform . mac_ver ( )
return " macos " , current_dist [ 0 ] . rsplit ( ' . ' , 1 ) [ 0 ]
2019-12-18 09:03:21 +00:00
def run_os_command ( command_map ) :
''' command_map is a dictionary of { ' executable ' : command}. For ex. { ' apt-get ' : ' sudo apt-get install -y python2.7 ' } '''
success = True
for executable , commands in command_map . items ( ) :
2019-12-20 09:15:56 +00:00
if shutil . which ( executable ) :
2019-12-18 09:03:21 +00:00
if isinstance ( commands , str ) :
commands = [ commands ]
for command in commands :
returncode = subprocess . check_call ( command , shell = True , stdout = log_stream , stderr = sys . stderr )
success = success and ( returncode == 0 )
return success
2019-12-27 07:44:33 +00:00
def install_pip ( ) :
2019-12-20 09:15:56 +00:00
if shutil . which ( ' pip ' ) :
2016-03-22 07:44:31 +00:00
run_os_command ( {
2019-12-18 09:03:21 +00:00
' pip ' : ' sudo -H pip install --upgrade setuptools cryptography pip '
2016-03-22 07:44:31 +00:00
} )
2016-06-28 06:41:06 +00:00
else :
if not os . path . exists ( " get-pip.py " ) :
run_os_command ( {
2019-12-18 09:03:21 +00:00
' wget ' : ' wget -q https://bootstrap.pypa.io/get-pip.py '
2016-06-28 06:41:06 +00:00
} )
2016-05-04 07:53:33 +00:00
2016-06-28 06:41:06 +00:00
success = run_os_command ( {
2019-06-21 03:10:13 +00:00
' python3 ' : ' sudo python3 get-pip.py --force-reinstall '
2016-05-30 11:12:55 +00:00
} )
2016-06-28 09:36:12 +00:00
if success :
2019-05-16 13:55:45 +00:00
dist_name , dist_version = get_distribution_info ( )
if dist_name == ' centos ' :
run_os_command ( {
2019-12-18 09:03:21 +00:00
' pip ' : ' sudo -H pip install --upgrade --ignore-installed requests '
2019-05-16 13:55:45 +00:00
} )
else :
run_os_command ( {
2019-12-18 09:03:21 +00:00
' pip ' : ' sudo -H pip install --upgrade requests '
2019-05-16 13:55:45 +00:00
} )
2016-06-28 09:36:12 +00:00
2019-12-27 07:44:33 +00:00
log ( " pip installed! " , level = 1 )
def install_prerequisites ( ) :
# pre-requisites for bench repo cloning
run_os_command ( {
' apt-get ' : [
' sudo apt-get update ' ,
' sudo apt-get install -y git build-essential python3-setuptools python3-dev libffi-dev '
] ,
' yum ' : [
' sudo yum groupinstall -y " Development tools " ' ,
' sudo yum install -y epel-release redhat-lsb-core git python-setuptools python-devel openssl-devel libffi-devel '
]
} )
install_package ( ' curl ' )
install_package ( ' wget ' )
install_package ( ' git ' )
install_pip ( )
2016-03-17 09:50:21 +00:00
success = run_os_command ( {
2019-12-18 09:03:21 +00:00
' pip ' : " sudo -H pip install --upgrade setuptools cryptography ansible==2.8.5 pip "
2016-03-17 09:50:21 +00:00
} )
2016-03-15 06:54:17 +00:00
2016-03-22 07:44:31 +00:00
if not success :
could_not_install ( ' Ansible ' )
2019-12-27 07:44:33 +00:00
def could_not_install ( package ) :
raise Exception ( ' Could not install {0} . Please install it manually. ' . format ( package ) )
def is_sudo_user ( ) :
return os . geteuid ( ) == 0
def install_package ( package ) :
if shutil . which ( package ) :
log ( " {0} already installed! " . format ( package ) , level = 1 )
else :
log ( " Installing {0} ... " . format ( package ) )
success = run_os_command ( {
' apt-get ' : [ ' sudo apt-get install -y {0} ' . format ( package ) ] ,
' yum ' : [ ' sudo yum install -y {0} ' . format ( package ) ] ,
' brew ' : [ ' brew install {0} ' . format ( package ) ]
} )
if success :
log ( " {0} installed! " . format ( package ) , level = 1 )
return success
could_not_install ( package )
def install_bench ( args ) :
2016-03-15 06:54:17 +00:00
# clone bench repo
2016-06-28 06:41:06 +00:00
if not args . run_travis :
clone_bench_repo ( args )
2016-05-25 10:47:48 +00:00
2016-06-28 09:36:12 +00:00
if not args . user :
if args . production :
args . user = ' frappe '
2016-06-30 11:10:07 +00:00
2017-04-17 11:58:18 +00:00
elif ' SUDO_USER ' in os . environ :
2016-06-30 11:10:07 +00:00
args . user = os . environ [ ' SUDO_USER ' ]
2016-06-28 09:36:12 +00:00
else :
args . user = getpass . getuser ( )
2016-03-15 06:54:17 +00:00
2016-06-28 09:36:12 +00:00
if args . user == ' root ' :
2016-06-30 11:10:07 +00:00
raise Exception ( ' Please run this script as a non-root user with sudo privileges, but without using sudo or pass --user=USER ' )
2016-05-27 06:55:12 +00:00
2018-03-26 11:25:31 +00:00
# Python executable
2019-06-21 03:10:13 +00:00
dist_name , dist_version = get_distribution_info ( )
if dist_name == ' centos ' :
args . python = ' python3.6 '
else :
args . python = ' python3 '
2018-03-26 11:25:31 +00:00
2016-05-27 06:55:12 +00:00
# create user if not exists
2016-05-30 11:12:55 +00:00
extra_vars = vars ( args )
2016-06-28 09:36:12 +00:00
extra_vars . update ( frappe_user = args . user )
2016-05-30 11:12:55 +00:00
2016-08-01 06:39:54 +00:00
if os . path . exists ( tmp_bench_repo ) :
repo_path = tmp_bench_repo
else :
repo_path = os . path . join ( os . path . expanduser ( ' ~ ' ) , ' bench ' )
extra_vars . update ( repo_path = repo_path )
2018-02-05 09:37:14 +00:00
run_playbook ( ' create_user.yml ' , extra_vars = extra_vars )
2016-05-27 06:55:12 +00:00
2017-08-03 11:03:57 +00:00
extra_vars . update ( get_passwords ( args ) )
2016-06-28 09:36:12 +00:00
if args . production :
extra_vars . update ( max_worker_connections = multiprocessing . cpu_count ( ) * 1024 )
2019-07-23 06:34:47 +00:00
frappe_branch = ' version-12 '
erpnext_branch = ' version-12 '
2019-07-22 10:31:36 +00:00
if args . version :
if args . version < = 10 :
frappe_branch = " {0} .x.x " . format ( args . version )
erpnext_branch = " {0} .x.x " . format ( args . version )
else :
frappe_branch = " version- {0} " . format ( args . version )
erpnext_branch = " version- {0} " . format ( args . version )
2018-04-14 23:30:16 +00:00
else :
2019-07-22 10:31:36 +00:00
if args . frappe_branch :
frappe_branch = args . frappe_branch
2018-04-14 23:30:16 +00:00
2019-07-22 10:31:36 +00:00
if args . erpnext_branch :
erpnext_branch = args . erpnext_branch
2019-07-23 06:34:47 +00:00
extra_vars . update ( frappe_branch = frappe_branch )
extra_vars . update ( erpnext_branch = erpnext_branch )
2019-05-07 04:17:35 +00:00
2017-08-28 13:40:38 +00:00
bench_name = ' frappe-bench ' if not args . bench_name else args . bench_name
extra_vars . update ( bench_name = bench_name )
2016-06-28 09:36:12 +00:00
2018-02-05 09:37:14 +00:00
# Will install ERPNext production setup by default
run_playbook ( ' site.yml ' , sudo = True , extra_vars = extra_vars )
2016-05-25 10:47:48 +00:00
2016-06-28 06:41:06 +00:00
if os . path . exists ( tmp_bench_repo ) :
shutil . rmtree ( tmp_bench_repo )
2016-03-15 06:54:17 +00:00
2016-08-02 06:28:28 +00:00
2016-05-25 10:47:48 +00:00
def clone_bench_repo ( args ) :
2016-03-22 07:44:31 +00:00
''' Clones the bench repository in the user folder '''
2019-12-18 09:03:21 +00:00
branch = args . bench_branch or ' master '
repo_url = args . repo_url or ' https://github.com/frappe/bench '
2016-05-25 10:47:48 +00:00
if os . path . exists ( tmp_bench_repo ) :
2016-03-15 06:54:17 +00:00
return 0
2016-08-01 06:39:54 +00:00
elif args . without_bench_setup :
clone_path = os . path . join ( os . path . expanduser ( ' ~ ' ) , ' bench ' )
else :
clone_path = tmp_bench_repo
2016-03-15 06:54:17 +00:00
success = run_os_command (
2019-12-18 09:03:21 +00:00
{ ' git ' : ' git clone --quiet {repo_url} {bench_repo} --depth 1 --branch {branch} ' . format (
2016-08-01 06:39:54 +00:00
repo_url = repo_url , bench_repo = clone_path , branch = branch ) }
2016-03-15 06:54:17 +00:00
)
return success
2016-03-22 07:44:31 +00:00
2017-08-03 11:03:57 +00:00
def get_passwords ( args ) :
"""
Returns a dict of passwords for further use
and creates passwords . txt in the bench user ' s home directory
"""
2019-12-27 07:44:33 +00:00
log ( " Input mySQL and Frappe Administrator passwords: " )
2017-08-03 11:03:57 +00:00
ignore_prompt = args . run_travis or args . without_bench_setup
mysql_root_password , admin_password = ' ' , ' '
passwords_file_path = os . path . join ( os . path . expanduser ( ' ~ ' + args . user ) , ' passwords.txt ' )
2016-08-01 05:44:01 +00:00
if not ignore_prompt :
2017-08-03 11:03:57 +00:00
# set passwords from existing passwords.txt
if os . path . isfile ( passwords_file_path ) :
with open ( passwords_file_path , ' r ' ) as f :
passwords = json . load ( f )
mysql_root_password , admin_password = passwords [ ' mysql_root_password ' ] , passwords [ ' admin_password ' ]
# set passwords from cli args
if args . mysql_root_password :
mysql_root_password = args . mysql_root_password
if args . admin_password :
admin_password = args . admin_password
# prompt for passwords
2016-05-04 07:53:33 +00:00
pass_set = True
while pass_set :
# mysql root password
if not mysql_root_password :
mysql_root_password = getpass . unix_getpass ( prompt = ' Please enter mysql root password: ' )
conf_mysql_passwd = getpass . unix_getpass ( prompt = ' Re-enter mysql root password: ' )
2018-04-09 10:11:01 +00:00
if mysql_root_password != conf_mysql_passwd or mysql_root_password == ' ' :
2016-05-04 07:53:33 +00:00
mysql_root_password = ' '
continue
# admin password
if not admin_password :
2016-10-13 07:38:12 +00:00
admin_password = getpass . unix_getpass ( prompt = ' Please enter the default Administrator user password: ' )
2016-05-04 07:53:33 +00:00
conf_admin_passswd = getpass . unix_getpass ( prompt = ' Re-enter Administrator password: ' )
2018-04-09 10:11:01 +00:00
if admin_password != conf_admin_passswd or admin_password == ' ' :
2016-05-04 07:53:33 +00:00
admin_password = ' '
continue
pass_set = False
else :
mysql_root_password = admin_password = ' travis '
2016-09-22 09:23:36 +00:00
passwords = {
2016-05-04 07:53:33 +00:00
' mysql_root_password ' : mysql_root_password ,
' admin_password ' : admin_password
}
2016-09-22 09:23:36 +00:00
if not ignore_prompt :
with open ( passwords_file_path , ' w ' ) as f :
json . dump ( passwords , f , indent = 1 )
2019-12-19 09:08:46 +00:00
log ( ' Passwords saved at ~/passwords.txt ' )
2016-10-13 07:38:12 +00:00
2016-09-22 09:23:36 +00:00
return passwords
2017-08-03 11:03:57 +00:00
2016-06-28 09:36:12 +00:00
def get_extra_vars_json ( extra_args ) :
2016-05-30 11:12:55 +00:00
# We need to pass production as extra_vars to the playbook to execute conditionals in the
# playbook. Extra variables can passed as json or key=value pair. Here, we will use JSON.
2019-12-27 07:44:33 +00:00
json_path = os . path . join ( ' / ' , ' tmp ' , ' extra_vars.json ' )
2017-04-17 11:58:18 +00:00
extra_vars = dict ( list ( extra_args . items ( ) ) )
2019-12-27 07:44:33 +00:00
2016-05-30 11:12:55 +00:00
with open ( json_path , mode = ' w ' ) as j :
json . dump ( extra_vars , j , indent = 1 , sort_keys = True )
return ( ' @ ' + json_path )
2016-05-27 06:55:12 +00:00
2019-12-27 07:44:33 +00:00
2016-05-27 06:55:12 +00:00
def run_playbook ( playbook_name , sudo = False , extra_vars = None ) :
2019-07-23 14:20:07 +00:00
args = [ ' ansible-playbook ' , ' -c ' , ' local ' , playbook_name , ' -vvvv ' ]
2016-05-27 06:55:12 +00:00
if extra_vars :
2016-05-30 11:12:55 +00:00
args . extend ( [ ' -e ' , get_extra_vars_json ( extra_vars ) ] )
2016-05-27 06:55:12 +00:00
2016-03-15 06:54:17 +00:00
if sudo :
2016-05-30 11:12:55 +00:00
user = extra_vars . get ( ' user ' ) or getpass . getuser ( )
2016-05-25 10:47:48 +00:00
args . extend ( [ ' --become ' , ' --become-user= {0} ' . format ( user ) ] )
2016-05-04 07:53:33 +00:00
2016-08-01 06:39:54 +00:00
if os . path . exists ( tmp_bench_repo ) :
cwd = tmp_bench_repo
else :
cwd = os . path . join ( os . path . expanduser ( ' ~ ' ) , ' bench ' )
2019-12-18 09:03:21 +00:00
success = subprocess . check_call ( args , cwd = os . path . join ( cwd , ' playbooks ' ) , stdout = log_stream , stderr = sys . stderr )
2016-03-15 06:54:17 +00:00
return success
2019-12-18 09:03:21 +00:00
2016-03-15 06:54:17 +00:00
def parse_commandline_args ( ) :
import argparse
parser = argparse . ArgumentParser ( description = ' Frappe Installer ' )
2016-05-25 10:47:48 +00:00
# Arguments develop and production are mutually exclusive both can't be specified together.
2016-05-04 07:53:33 +00:00
# Hence, we need to create a group for discouraging use of both options at the same time.
args_group = parser . add_mutually_exclusive_group ( )
2019-12-18 09:03:21 +00:00
args_group . add_argument ( ' --develop ' , dest = ' develop ' , action = ' store_true ' , default = False , help = ' Install developer setup ' )
args_group . add_argument ( ' --production ' , dest = ' production ' , action = ' store_true ' , default = False , help = ' Setup Production environment for bench ' )
parser . add_argument ( ' --site ' , dest = ' site ' , action = ' store ' , default = ' site1.local ' , help = ' Specifiy name for your first ERPNext site ' )
parser . add_argument ( ' --without-site ' , dest = ' without_site ' , action = ' store_true ' , default = False )
parser . add_argument ( ' --verbose ' , dest = ' verbose ' , action = ' store_true ' , default = False , help = ' Run the script in verbose mode ' )
2016-05-25 10:47:48 +00:00
parser . add_argument ( ' --user ' , dest = ' user ' , help = ' Install frappe-bench for this user ' )
parser . add_argument ( ' --bench-branch ' , dest = ' bench_branch ' , help = ' Clone a particular branch of bench repository ' )
2016-05-30 11:12:55 +00:00
parser . add_argument ( ' --repo-url ' , dest = ' repo_url ' , help = ' Clone bench from the given url ' )
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --frappe-repo-url ' , dest = ' frappe_repo_url ' , action = ' store ' , default = ' https://github.com/frappe/frappe ' , help = ' Clone frappe from the given url ' )
parser . add_argument ( ' --frappe-branch ' , dest = ' frappe_branch ' , action = ' store ' , help = ' Clone a particular branch of frappe ' )
parser . add_argument ( ' --erpnext-repo-url ' , dest = ' erpnext_repo_url ' , action = ' store ' , default = ' https://github.com/frappe/erpnext ' , help = ' Clone erpnext from the given url ' )
parser . add_argument ( ' --erpnext-branch ' , dest = ' erpnext_branch ' , action = ' store ' , help = ' Clone a particular branch of erpnext ' )
parser . add_argument ( ' --without-erpnext ' , dest = ' without_erpnext ' , action = ' store_true ' , default = False , help = ' Prevent fetching ERPNext ' )
2019-07-22 10:31:36 +00:00
# direct provision to install versions
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --version ' , dest = ' version ' , action = ' store ' , default = ' 12 ' , type = int , help = ' Clone particular version of frappe and erpnext ' )
2016-05-04 07:53:33 +00:00
# To enable testing of script using Travis, this should skip the prompt
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --run-travis ' , dest = ' run_travis ' , action = ' store_true ' , default = False , help = argparse . SUPPRESS )
parser . add_argument ( ' --without-bench-setup ' , dest = ' without_bench_setup ' , action = ' store_true ' , default = False , help = argparse . SUPPRESS )
2017-05-15 15:43:37 +00:00
# whether to overwrite an existing bench
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --overwrite ' , dest = ' overwrite ' , action = ' store_true ' , default = False , help = ' Whether to overwrite an existing bench ' )
2017-08-03 11:03:57 +00:00
# set passwords
parser . add_argument ( ' --mysql-root-password ' , dest = ' mysql_root_password ' , help = ' Set mysql root password ' )
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --mariadb-version ' , dest = ' mariadb_version ' , default = ' 10.2 ' , help = ' Specify mariadb version ' )
2017-08-03 11:03:57 +00:00
parser . add_argument ( ' --admin-password ' , dest = ' admin_password ' , help = ' Set admin password ' )
2017-08-28 13:40:38 +00:00
parser . add_argument ( ' --bench-name ' , dest = ' bench_name ' , help = ' Create bench with specified name. Default name is frappe-bench ' )
2018-03-26 11:25:31 +00:00
# Python interpreter to be used
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --python ' , dest = ' python ' , default = ' python3 ' , help = argparse . SUPPRESS )
2018-12-11 17:56:15 +00:00
# LXC Support
2019-12-18 09:03:21 +00:00
parser . add_argument ( ' --container ' , dest = ' container ' , default = False , action = ' store_true ' , help = ' Use if you \' re creating inside LXC ' )
2016-03-15 06:54:17 +00:00
args = parser . parse_args ( )
return args
2016-03-22 07:44:31 +00:00
if __name__ == ' __main__ ' :
2019-12-19 08:31:24 +00:00
if not is_sudo_user ( ) :
2019-12-19 09:08:46 +00:00
log ( " Please run this script as a non-root user with sudo privileges " , level = 3 )
2019-12-19 08:31:24 +00:00
sys . exit ( )
2016-03-15 06:54:17 +00:00
args = parse_commandline_args ( )
2019-12-27 07:44:33 +00:00
2019-12-18 09:03:21 +00:00
with warnings . catch_warnings ( ) :
warnings . simplefilter ( " ignore " )
2019-12-27 07:44:33 +00:00
setup_log_stream ( args )
check_distribution_compatibility ( )
check_system_package_managers ( )
2019-12-19 08:31:24 +00:00
check_environment ( )
2019-12-27 07:44:33 +00:00
install_prerequisites ( )
2019-12-18 09:03:21 +00:00
install_bench ( args )
2019-12-19 08:31:24 +00:00
2019-12-19 09:08:46 +00:00
log ( " Bench + Frappe + ERPNext has been successfully installed! " )