electron/script/update-external-binaries.py

141 lines
3.7 KiB
Python
Raw Normal View History

#!/usr/bin/env python
import argparse
2014-05-18 15:03:46 +00:00
import errno
import hashlib
import json
import os
import tarfile
2018-08-21 19:40:06 +00:00
from lib.config import PLATFORM, get_target_arch
from lib.util import add_exec_bit, download, extract_zip, rm_rf, \
safe_mkdir, tempdir
2013-07-01 14:27:14 +00:00
SOURCE_ROOT = os.path.abspath(os.path.dirname(os.path.dirname(__file__)))
CONFIG_PATH = os.path.join(SOURCE_ROOT, 'script', 'external-binaries.json')
2014-04-29 02:03:03 +00:00
def parse_args():
parser = argparse.ArgumentParser(
description='Download binaries for Electron build')
parser.add_argument('--base-url', required=False,
help="Base URL for all downloads")
2020-10-19 11:55:27 +00:00
parser.add_argument('--force',
action='store_true',
default=False,
required=False)
return parser.parse_args()
def parse_config():
with open(CONFIG_PATH, 'r') as config_file:
config = json.load(config_file)
return config
def main():
args = parse_args()
config = parse_config()
base_url = args.base_url if args.base_url is not None else config['baseUrl']
output_dir = os.path.join(SOURCE_ROOT, 'external_binaries')
config_hash_path = os.path.join(output_dir, '.hash')
if (not is_updated(config_hash_path) and not args.force):
return
rm_rf(output_dir)
safe_mkdir(output_dir)
for binary in config['files']:
if not binary_should_be_downloaded(binary):
continue
temp_path = download_binary(base_url, binary['sha'], binary['name'])
if temp_path.endswith('.zip'):
extract_zip(temp_path, output_dir)
else:
tar = tarfile.open(temp_path, "r:gz")
tar.extractall(output_dir)
tar.close()
# Hack alert. Set exec bit for sccache binaries.
# https://bugs.python.org/issue15795
if 'sccache' in binary['name']:
add_exec_bit_to_sccache_binary(output_dir)
2018-08-21 19:40:06 +00:00
with open(config_hash_path, 'w') as f:
f.write(sha256(CONFIG_PATH))
def is_updated(config_hash_path):
existing_hash = ''
try:
with open(config_hash_path, 'r') as f:
existing_hash = f.readline().strip()
except IOError as e:
if e.errno != errno.ENOENT:
raise
return not sha256(CONFIG_PATH) == existing_hash
def binary_should_be_downloaded(binary):
if 'platform' in binary and binary['platform'] != PLATFORM:
return False
if 'targetArch' in binary and binary['targetArch'] != get_target_arch():
return False
return True
def sha256(file_path):
hash_256 = hashlib.sha256()
with open(file_path, "rb") as f:
for chunk in iter(lambda: f.read(4096), b""):
hash_256.update(chunk)
return hash_256.hexdigest()
def download_binary(base_url, sha, binary_name, attempt=3):
full_url = '{0}/{1}/{2}'.format(base_url, sha, binary_name)
try:
temp_path = download_to_temp_dir(full_url, filename=binary_name, sha=sha)
return temp_path
except Exception as e:
if attempt == 1:
raise e
return download_binary(base_url, sha, binary_name, attempt - 1)
def validate_sha(file_path, sha):
downloaded_sha = sha256(file_path)
if downloaded_sha != sha:
2020-10-19 11:55:27 +00:00
raise Exception("SHA for external binary file {} does not match "
"expected '{}' != '{}'".format(
file_path, downloaded_sha, sha))
def download_to_temp_dir(url, filename, sha):
2015-04-12 14:10:46 +00:00
download_dir = tempdir(prefix='electron-')
file_path = os.path.join(download_dir, filename)
download(text='Download ' + filename, url=url, path=file_path)
validate_sha(file_path, sha)
return file_path
def add_exec_bit_to_sccache_binary(binary_dir):
binary_name = 'sccache'
if PLATFORM == 'win32':
binary_name += '.exe'
binary_path = os.path.join(binary_dir, binary_name)
add_exec_bit(binary_path)
if __name__ == '__main__':
main()