Advertisement
badlogic

install.py

Mar 2nd, 2022
214
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
text 18.21 KB | None | 0 0
  1. #!/usr/bin/env python3
  2.  
  3. from __future__ import print_function
  4.  
  5. import os
  6. import sys
  7. import subprocess
  8. import getpass
  9. import json
  10. import multiprocessing
  11. import shutil
  12. import platform
  13. import warnings
  14. import datetime
  15.  
  16.  
  17. tmp_bench_repo = os.path.join('/', 'tmp', '.bench')
  18. tmp_log_folder = os.path.join('/', 'tmp', 'logs')
  19. execution_timestamp = datetime.datetime.utcnow()
  20. execution_day = "{:%Y-%m-%d}".format(execution_timestamp)
  21. execution_time = "{:%H:%M}".format(execution_timestamp)
  22. log_file_name = "easy-install__{0}__{1}.log".format(execution_day, execution_time.replace(':', '-'))
  23. log_path = os.path.join(tmp_log_folder, log_file_name)
  24. log_stream = sys.stdout
  25. distro_required = not ((sys.version_info.major < 3) or (sys.version_info.major == 3 and sys.version_info.minor < 5))
  26.  
  27.  
  28. def log(message, level=0):
  29. levels = {
  30. 0: '\033[94m', # normal
  31. 1: '\033[92m', # success
  32. 2: '\033[91m', # fail
  33. 3: '\033[93m' # warn/suggest
  34. }
  35. start = levels.get(level) or ''
  36. end = '\033[0m'
  37. print(start + message + end)
  38.  
  39.  
  40. def setup_log_stream(args):
  41. global log_stream
  42. sys.stderr = sys.stdout
  43.  
  44. if not args.verbose:
  45. if not os.path.exists(tmp_log_folder):
  46. os.makedirs(tmp_log_folder)
  47. log_stream = open(log_path, 'w')
  48. log("Logs are saved under {0}".format(log_path), level=3)
  49. print("Install script run at {0} on {1}\n\n".format(execution_time, execution_day), file=log_stream)
  50.  
  51.  
  52. def check_environment():
  53. needed_environ_vars = ['LANG', 'LC_ALL']
  54. message = ''
  55.  
  56. for var in needed_environ_vars:
  57. if var not in os.environ:
  58. message += "\nexport {0}=C.UTF-8".format(var)
  59.  
  60. if message:
  61. log("Bench's CLI needs these to be defined!", level=3)
  62. log("Run the following commands in shell: {0}".format(message), level=2)
  63. sys.exit()
  64.  
  65.  
  66. def check_system_package_managers():
  67. if 'Darwin' in os.uname():
  68. if not shutil.which('brew'):
  69. raise Exception('''
  70. Please install brew package manager before proceeding with bench setup. Please run following
  71. to install brew package manager on your machine,
  72.  
  73. /usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
  74. ''')
  75. if 'Linux' in os.uname():
  76. if not any([shutil.which(x) for x in ['apt-get', 'yum']]):
  77. raise Exception('Cannot find any compatible package manager!')
  78.  
  79.  
  80. def check_distribution_compatibility():
  81. dist_name, dist_version = get_distribution_info()
  82. supported_dists = {
  83. 'macos': [10.9, 10.10, 10.11, 10.12],
  84. 'ubuntu': [14, 15, 16, 18, 19, 20],
  85. 'debian': [8, 9, 10],
  86. 'centos': [7]
  87. }
  88.  
  89. log("Checking System Compatibility...")
  90. if dist_name in supported_dists:
  91. if float(dist_version) in supported_dists[dist_name]:
  92. log("{0} {1} is compatible!".format(dist_name, dist_version), level=1)
  93. else:
  94. log("{0} {1} is detected".format(dist_name, dist_version), level=1)
  95. log("Install on {0} {1} instead".format(dist_name, supported_dists[dist_name][-1]), level=3)
  96. else:
  97. log("Sorry, the installer doesn't support {0}. Aborting installation!".format(dist_name), level=2)
  98.  
  99.  
  100. def import_with_install(package):
  101. # copied from https://discuss.erpnext.com/u/nikunj_patel
  102. # https://discuss.erpnext.com/t/easy-install-setup-guide-for-erpnext-installation-on-ubuntu-20-04-lts-with-some-modification-of-course/62375/5
  103. # need to move to top said v13 for fully python3 era
  104. import importlib
  105.  
  106. try:
  107. importlib.import_module(package)
  108. except ImportError:
  109. # caveat : pip3 must be installed
  110.  
  111. import pip
  112.  
  113. pip.main(['install', package])
  114. finally:
  115. globals()[package] = importlib.import_module(package)
  116.  
  117.  
  118. def get_distribution_info():
  119. # return distribution name and major version
  120. if platform.system() == "Linux":
  121. if distro_required:
  122. current_dist = distro.linux_distribution(full_distribution_name=True)
  123. else:
  124. current_dist = platform.dist()
  125.  
  126. return current_dist[0].lower(), current_dist[1].rsplit('.')[0]
  127.  
  128. elif platform.system() == "Darwin":
  129. current_dist = platform.mac_ver()
  130. return "macos", current_dist[0].rsplit('.', 1)[0]
  131.  
  132.  
  133. def run_os_command(command_map):
  134. '''command_map is a dictionary of {'executable': command}. For ex. {'apt-get': 'sudo apt-get install -y python2.7'}'''
  135. success = True
  136.  
  137. for executable, commands in command_map.items():
  138. if shutil.which(executable):
  139. if isinstance(commands, str):
  140. commands = [commands]
  141.  
  142. for command in commands:
  143. returncode = subprocess.check_call(command, shell=True, stdout=log_stream, stderr=sys.stderr)
  144. success = success and (returncode == 0)
  145.  
  146. return success
  147.  
  148.  
  149. def install_prerequisites():
  150. # pre-requisites for bench repo cloning
  151. run_os_command({
  152. 'apt-get': [
  153. 'sudo apt-get update',
  154. 'sudo apt-get install -y git build-essential python3-setuptools python3-dev libffi-dev'
  155. ],
  156. 'yum': [
  157. 'sudo yum groupinstall -y "Development tools"',
  158. 'sudo yum install -y epel-release redhat-lsb-core git python-setuptools python-devel openssl-devel libffi-devel'
  159. ]
  160. })
  161.  
  162. # until psycopg2-binary is available for aarch64 (Arm 64-bit), we'll need libpq and libssl dev packages to build psycopg2 from source
  163. if platform.machine() == 'aarch64':
  164. log("Installing libpq and libssl dev packages to build psycopg2 for aarch64...")
  165. run_os_command({
  166. 'apt-get': ['sudo apt-get install -y libpq-dev libssl-dev'],
  167. 'yum': ['sudo yum install -y libpq-devel openssl-devel']
  168. })
  169.  
  170. install_package('curl')
  171. install_package('wget')
  172. install_package('git')
  173. install_package('pip3', 'python3-pip')
  174.  
  175. run_os_command({
  176. 'python3': "sudo -H python3 -m pip install --upgrade pip setuptools-rust"
  177. })
  178. success = run_os_command({
  179. 'python3': "sudo -H python3 -m pip install --upgrade setuptools wheel cryptography ansible~=2.8.15"
  180. })
  181.  
  182. if not (success or shutil.which('ansible')):
  183. could_not_install('Ansible')
  184.  
  185.  
  186. def could_not_install(package):
  187. raise Exception('Could not install {0}. Please install it manually.'.format(package))
  188.  
  189.  
  190. def is_sudo_user():
  191. return os.geteuid() == 0
  192.  
  193.  
  194. def install_package(package, package_name=None):
  195. if shutil.which(package):
  196. log("{0} already installed!".format(package), level=1)
  197. else:
  198. log("Installing {0}...".format(package))
  199. package_name = package_name or package
  200. success = run_os_command({
  201. 'apt-get': ['sudo apt-get install -y {0}'.format(package_name)],
  202. 'yum': ['sudo yum install -y {0}'.format(package_name)],
  203. 'brew': ['brew install {0}'.format(package_name)]
  204. })
  205. if success:
  206. log("{0} installed!".format(package), level=1)
  207. return success
  208. could_not_install(package)
  209.  
  210.  
  211. def install_bench(args):
  212. # clone bench repo
  213. if not args.run_travis:
  214. clone_bench_repo(args)
  215.  
  216. if not args.user:
  217. if args.production:
  218. args.user = 'frappe'
  219.  
  220. elif 'SUDO_USER' in os.environ:
  221. args.user = os.environ['SUDO_USER']
  222.  
  223. else:
  224. args.user = getpass.getuser()
  225.  
  226. if args.user == 'root':
  227. raise Exception('Please run this script as a non-root user with sudo privileges, but without using sudo or pass --user=USER')
  228.  
  229. # Python executable
  230. dist_name, dist_version = get_distribution_info()
  231. if dist_name=='centos':
  232. args.python = 'python3.6'
  233. else:
  234. args.python = 'python3'
  235.  
  236. # create user if not exists
  237. extra_vars = vars(args)
  238. extra_vars.update(frappe_user=args.user)
  239.  
  240. extra_vars.update(user_directory=get_user_home_directory(args.user))
  241.  
  242. if os.path.exists(tmp_bench_repo):
  243. repo_path = tmp_bench_repo
  244. else:
  245. repo_path = os.path.join(os.path.expanduser('~'), 'bench')
  246.  
  247. extra_vars.update(repo_path=repo_path)
  248. run_playbook('create_user.yml', extra_vars=extra_vars)
  249.  
  250. extra_vars.update(get_passwords(args))
  251. if args.production:
  252. extra_vars.update(max_worker_connections=multiprocessing.cpu_count() * 1024)
  253.  
  254. if args.version <= 10:
  255. frappe_branch = "{0}.x.x".format(args.version)
  256. erpnext_branch = "{0}.x.x".format(args.version)
  257. else:
  258. frappe_branch = "version-{0}".format(args.version)
  259. erpnext_branch = "version-{0}".format(args.version)
  260.  
  261. # Allow override of frappe_branch and erpnext_branch, regardless of args.version (which always has a default set)
  262. if args.frappe_branch:
  263. frappe_branch = args.frappe_branch
  264. if args.erpnext_branch:
  265. erpnext_branch = args.erpnext_branch
  266.  
  267. extra_vars.update(frappe_branch=frappe_branch)
  268. extra_vars.update(erpnext_branch=erpnext_branch)
  269.  
  270. bench_name = 'frappe-bench' if not args.bench_name else args.bench_name
  271. extra_vars.update(bench_name=bench_name)
  272.  
  273. # Will install ERPNext production setup by default
  274. if args.without_erpnext:
  275. log("Initializing bench {bench_name}:\n\tFrappe Branch: {frappe_branch}\n\tERPNext will not be installed due to --without-erpnext".format(bench_name=bench_name, frappe_branch=frappe_branch))
  276. else:
  277. log("Initializing bench {bench_name}:\n\tFrappe Branch: {frappe_branch}\n\tERPNext Branch: {erpnext_branch}".format(bench_name=bench_name, frappe_branch=frappe_branch, erpnext_branch=erpnext_branch))
  278. run_playbook('site.yml', sudo=True, extra_vars=extra_vars)
  279.  
  280. if os.path.exists(tmp_bench_repo):
  281. shutil.rmtree(tmp_bench_repo)
  282.  
  283.  
  284. def clone_bench_repo(args):
  285. '''Clones the bench repository in the user folder'''
  286. branch = args.bench_branch or 'develop'
  287. repo_url = args.repo_url or 'https://github.com/frappe/bench'
  288.  
  289. if os.path.exists(tmp_bench_repo):
  290. log('Not cloning already existing Bench repository at {tmp_bench_repo}'.format(tmp_bench_repo=tmp_bench_repo))
  291. return 0
  292. elif args.without_bench_setup:
  293. clone_path = os.path.join(os.path.expanduser('~'), 'bench')
  294. log('--without-bench-setup specified, clone path is: {clone_path}'.format(clone_path=clone_path))
  295. else:
  296. clone_path = tmp_bench_repo
  297. # Not logging repo_url to avoid accidental credential leak in case credential is embedded in URL
  298. log('Cloning bench repository branch {branch} into {clone_path}'.format(branch=branch, clone_path=clone_path))
  299.  
  300. success = run_os_command(
  301. {'git': 'git clone --quiet {repo_url} {bench_repo} --depth 1 --branch {branch}'.format(
  302. repo_url=repo_url, bench_repo=clone_path, branch=branch)}
  303. )
  304.  
  305. return success
  306.  
  307.  
  308. def passwords_didnt_match(context=''):
  309. log("{} passwords did not match!".format(context), level=3)
  310.  
  311.  
  312. def get_passwords(args):
  313. """
  314. Returns a dict of passwords for further use
  315. and creates passwords.txt in the bench user's home directory
  316. """
  317. log("Input MySQL and Frappe Administrator passwords:")
  318. ignore_prompt = args.run_travis or args.without_bench_setup
  319. mysql_root_password, admin_password = '', ''
  320. passwords_file_path = os.path.join(os.path.expanduser('~' + args.user), 'passwords.txt')
  321.  
  322. if not ignore_prompt:
  323. # set passwords from existing passwords.txt
  324. if os.path.isfile(passwords_file_path):
  325. with open(passwords_file_path, 'r') as f:
  326. passwords = json.load(f)
  327. mysql_root_password, admin_password = passwords['mysql_root_password'], passwords['admin_password']
  328.  
  329. # set passwords from cli args
  330. if args.mysql_root_password:
  331. mysql_root_password = args.mysql_root_password
  332. if args.admin_password:
  333. admin_password = args.admin_password
  334.  
  335. # prompt for passwords
  336. pass_set = True
  337. while pass_set:
  338. # mysql root password
  339. if not mysql_root_password:
  340. mysql_root_password = getpass.unix_getpass(prompt='Please enter mysql root password: ')
  341. conf_mysql_passwd = getpass.unix_getpass(prompt='Re-enter mysql root password: ')
  342.  
  343. if mysql_root_password != conf_mysql_passwd or mysql_root_password == '':
  344. passwords_didnt_match("MySQL")
  345. mysql_root_password = ''
  346. continue
  347.  
  348. # admin password, only needed if we're also creating a site
  349. if not admin_password and not args.without_site:
  350. admin_password = getpass.unix_getpass(prompt='Please enter the default Administrator user password: ')
  351. conf_admin_passswd = getpass.unix_getpass(prompt='Re-enter Administrator password: ')
  352.  
  353. if admin_password != conf_admin_passswd or admin_password == '':
  354. passwords_didnt_match("Administrator")
  355. admin_password = ''
  356. continue
  357. elif args.without_site:
  358. log("Not creating a new site due to --without-site")
  359.  
  360. pass_set = False
  361. else:
  362. mysql_root_password = admin_password = 'travis'
  363.  
  364. passwords = {
  365. 'mysql_root_password': mysql_root_password,
  366. 'admin_password': admin_password
  367. }
  368.  
  369. if not ignore_prompt:
  370. with open(passwords_file_path, 'w') as f:
  371. json.dump(passwords, f, indent=1)
  372.  
  373. log('Passwords saved at ~/passwords.txt')
  374.  
  375. return passwords
  376.  
  377.  
  378. def get_extra_vars_json(extra_args):
  379. # We need to pass production as extra_vars to the playbook to execute conditionals in the
  380. # playbook. Extra variables can passed as json or key=value pair. Here, we will use JSON.
  381. json_path = os.path.join('/', 'tmp', 'extra_vars.json')
  382. extra_vars = dict(list(extra_args.items()))
  383.  
  384. with open(json_path, mode='w') as j:
  385. json.dump(extra_vars, j, indent=1, sort_keys=True)
  386.  
  387. return ('@' + json_path)
  388.  
  389. def get_user_home_directory(user):
  390. # Return home directory /home/USERNAME or anything else defined as home directory in
  391. # passwd for user.
  392. return os.path.expanduser('~'+user)
  393.  
  394.  
  395. def run_playbook(playbook_name, sudo=False, extra_vars=None):
  396. args = ['ansible-playbook', '-c', 'local', playbook_name , '-vvvv']
  397.  
  398. if extra_vars:
  399. args.extend(['-e', get_extra_vars_json(extra_vars)])
  400.  
  401. if sudo:
  402. user = extra_vars.get('user') or getpass.getuser()
  403. args.extend(['--become', '--become-user={0}'.format(user)])
  404.  
  405. if os.path.exists(tmp_bench_repo):
  406. cwd = tmp_bench_repo
  407. else:
  408. cwd = os.path.join(os.path.expanduser('~'), 'bench')
  409.  
  410. playbooks_locations = [os.path.join(cwd, 'bench', 'playbooks'), os.path.join(cwd, 'playbooks')]
  411. playbooks_folder = [x for x in playbooks_locations if os.path.exists(x)][0]
  412.  
  413. success = subprocess.check_call(args, cwd=playbooks_folder, stdout=log_stream, stderr=sys.stderr)
  414. return success
  415.  
  416.  
  417. def setup_script_requirements():
  418. if distro_required:
  419. install_package('pip3', 'python3-pip')
  420. import_with_install('distro')
  421.  
  422.  
  423. def parse_commandline_args():
  424. import argparse
  425.  
  426. parser = argparse.ArgumentParser(description='Frappe Installer')
  427. # Arguments develop and production are mutually exclusive both can't be specified together.
  428. # Hence, we need to create a group for discouraging use of both options at the same time.
  429. args_group = parser.add_mutually_exclusive_group()
  430.  
  431. args_group.add_argument('--develop', dest='develop', action='store_true', default=False, help='Install developer setup')
  432. args_group.add_argument('--production', dest='production', action='store_true', default=False, help='Setup Production environment for bench')
  433. parser.add_argument('--site', dest='site', action='store', default='site1.local', help='Specify name for your first ERPNext site')
  434. parser.add_argument('--without-site', dest='without_site', action='store_true', default=False, help='Do not create a new site')
  435. parser.add_argument('--verbose', dest='verbose', action='store_true', default=False, help='Run the script in verbose mode')
  436. parser.add_argument('--user', dest='user', help='Install frappe-bench for this user')
  437. parser.add_argument('--bench-branch', dest='bench_branch', help='Clone a particular branch of bench repository')
  438. parser.add_argument('--repo-url', dest='repo_url', help='Clone bench from the given url')
  439. 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')
  440. parser.add_argument('--frappe-branch', dest='frappe_branch', action='store', help='Clone a particular branch of frappe')
  441. 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')
  442. parser.add_argument('--erpnext-branch', dest='erpnext_branch', action='store', help='Clone a particular branch of erpnext')
  443. parser.add_argument('--without-erpnext', dest='without_erpnext', action='store_true', default=False, help='Prevent fetching ERPNext')
  444. # direct provision to install versions
  445. parser.add_argument('--version', dest='version', action='store', default=13, type=int, help='Clone particular version of frappe and erpnext')
  446. # To enable testing of script using Travis, this should skip the prompt
  447. parser.add_argument('--run-travis', dest='run_travis', action='store_true', default=False, help=argparse.SUPPRESS)
  448. parser.add_argument('--without-bench-setup', dest='without_bench_setup', action='store_true', default=False, help=argparse.SUPPRESS)
  449. # whether to overwrite an existing bench
  450. parser.add_argument('--overwrite', dest='overwrite', action='store_true', default=False, help='Whether to overwrite an existing bench')
  451. # set passwords
  452. parser.add_argument('--mysql-root-password', dest='mysql_root_password', help='Set mysql root password')
  453. parser.add_argument('--mariadb-version', dest='mariadb_version', default='10.4', help='Specify mariadb version')
  454. parser.add_argument('--admin-password', dest='admin_password', help='Set admin password')
  455. parser.add_argument('--bench-name', dest='bench_name', help='Create bench with specified name. Default name is frappe-bench')
  456. # Python interpreter to be used
  457. parser.add_argument('--python', dest='python', default='python3', help=argparse.SUPPRESS)
  458. # LXC Support
  459. parser.add_argument('--container', dest='container', default=False, action='store_true', help='Use if you\'re creating inside LXC')
  460.  
  461. args = parser.parse_args()
  462.  
  463. return args
  464.  
  465.  
  466. if __name__ == '__main__':
  467. if sys.version[0] == '2':
  468. if not os.environ.get('CI'):
  469. if not raw_input("It is recommended to run this script with Python 3\nDo you still wish to continue? [Y/n]: ").lower() == "y":
  470. sys.exit()
  471.  
  472. try:
  473. from distutils.spawn import find_executable
  474. except ImportError:
  475. try:
  476. subprocess.check_call('pip install --upgrade setuptools')
  477. except subprocess.CalledProcessError:
  478. print("Install distutils or use Python3 to run the script")
  479. sys.exit(1)
  480.  
  481. shutil.which = find_executable
  482.  
  483. if not is_sudo_user():
  484. log("Please run this script as a non-root user with sudo privileges", level=3)
  485. sys.exit()
  486.  
  487. args = parse_commandline_args()
  488.  
  489. with warnings.catch_warnings():
  490. warnings.simplefilter("ignore")
  491. setup_log_stream(args)
  492. install_prerequisites()
  493. setup_script_requirements()
  494. check_distribution_compatibility()
  495. check_system_package_managers()
  496. check_environment()
  497. install_bench(args)
  498.  
  499. log("Bench + Frappe + ERPNext has been successfully installed!")
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement