2021-06-03 11:43:42 +02:00
|
|
|
#!/usr/bin/env python3
|
2022-06-24 13:06:16 +02:00
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
import os
|
2021-02-04 17:31:22 +01:00
|
|
|
import platform
|
2021-10-21 14:57:58 +02:00
|
|
|
import sys
|
2021-10-21 12:48:46 +02:00
|
|
|
|
2022-04-27 10:15:45 +02:00
|
|
|
from PyInstaller.__main__ import run as run_pyinstaller
|
2021-10-21 14:57:58 +02:00
|
|
|
|
2022-07-17 13:11:40 +02:00
|
|
|
OS_NAME, MACHINE, ARCH = sys.platform, platform.machine(), platform.architecture()[0][:2]
|
|
|
|
if MACHINE in ('x86_64', 'AMD64') or ('i' in MACHINE and '86' in MACHINE):
|
|
|
|
# NB: Windows x86 has MACHINE = AMD64 irrespective of bitness
|
|
|
|
MACHINE = 'x86' if ARCH == '32' else ''
|
2021-01-29 18:45:27 +01:00
|
|
|
|
2021-10-21 12:48:46 +02:00
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
def main():
|
|
|
|
opts = parse_options()
|
2022-04-27 10:15:45 +02:00
|
|
|
version = read_version('yt_dlp/version.py')
|
|
|
|
|
|
|
|
onedir = '--onedir' in opts or '-D' in opts
|
|
|
|
if not onedir and '-F' not in opts and '--onefile' not in opts:
|
|
|
|
opts.append('--onefile')
|
2021-10-03 22:55:13 +02:00
|
|
|
|
2022-05-22 13:37:18 +02:00
|
|
|
name, final_file = exe(onedir)
|
2022-07-16 18:22:48 +02:00
|
|
|
print(f'Building yt-dlp v{version} for {OS_NAME} {platform.machine()} with options {opts}')
|
2021-10-22 22:37:20 +02:00
|
|
|
print('Remember to update the version using "devscripts/update-version.py"')
|
2021-10-21 14:57:58 +02:00
|
|
|
if not os.path.isfile('yt_dlp/extractor/lazy_extractors.py'):
|
|
|
|
print('WARNING: Building without lazy_extractors. Run '
|
2021-10-22 22:37:20 +02:00
|
|
|
'"devscripts/make_lazy_extractors.py" to build lazy extractors', file=sys.stderr)
|
2021-10-21 14:57:58 +02:00
|
|
|
print(f'Destination: {final_file}\n')
|
2021-01-29 18:45:27 +01:00
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
opts = [
|
2022-04-28 18:46:23 +02:00
|
|
|
f'--name={name}',
|
2021-10-21 14:57:58 +02:00
|
|
|
'--icon=devscripts/logo.ico',
|
|
|
|
'--upx-exclude=vcruntime140.dll',
|
|
|
|
'--noconfirm',
|
2022-04-27 10:15:45 +02:00
|
|
|
# NB: Modules that are only imported dynamically must be added here.
|
|
|
|
# --collect-submodules may not work correctly if user has a yt-dlp installed via PIP
|
|
|
|
'--hidden-import=yt_dlp.compat._legacy',
|
2021-12-19 16:18:06 +01:00
|
|
|
*dependency_options(),
|
2021-10-21 14:57:58 +02:00
|
|
|
*opts,
|
|
|
|
'yt_dlp/__main__.py',
|
|
|
|
]
|
|
|
|
|
2022-04-27 10:15:45 +02:00
|
|
|
print(f'Running PyInstaller with {opts}')
|
|
|
|
run_pyinstaller(opts)
|
2021-10-21 14:57:58 +02:00
|
|
|
set_version_info(final_file, version)
|
|
|
|
|
|
|
|
|
|
|
|
def parse_options():
|
2022-06-27 02:50:06 +02:00
|
|
|
# Compatibility with older arguments
|
2021-10-21 14:57:58 +02:00
|
|
|
opts = sys.argv[1:]
|
|
|
|
if opts[0:1] in (['32'], ['64']):
|
|
|
|
if ARCH != opts[0]:
|
|
|
|
raise Exception(f'{opts[0]}bit executable cannot be built on a {ARCH}bit system')
|
|
|
|
opts = opts[1:]
|
2022-04-27 10:15:45 +02:00
|
|
|
return opts
|
2021-01-29 18:45:27 +01:00
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
|
2022-04-27 10:15:45 +02:00
|
|
|
# Get the version from yt_dlp/version.py without importing the package
|
|
|
|
def read_version(fname):
|
|
|
|
with open(fname, encoding='utf-8') as f:
|
|
|
|
exec(compile(f.read(), fname, 'exec'))
|
|
|
|
return locals()['__version__']
|
2021-10-21 14:57:58 +02:00
|
|
|
|
|
|
|
|
2022-05-22 13:37:18 +02:00
|
|
|
def exe(onedir):
|
|
|
|
"""@returns (name, path)"""
|
|
|
|
name = '_'.join(filter(None, (
|
|
|
|
'yt-dlp',
|
2022-05-21 21:49:49 +02:00
|
|
|
{'win32': '', 'darwin': 'macos'}.get(OS_NAME, OS_NAME),
|
2022-07-16 18:22:48 +02:00
|
|
|
MACHINE
|
2022-05-22 13:37:18 +02:00
|
|
|
)))
|
|
|
|
return name, ''.join(filter(None, (
|
|
|
|
'dist/',
|
|
|
|
onedir and f'{name}/',
|
|
|
|
name,
|
|
|
|
OS_NAME == 'win32' and '.exe'
|
|
|
|
)))
|
|
|
|
|
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
def version_to_list(version):
|
|
|
|
version_list = version.split('.')
|
|
|
|
return list(map(int, version_list)) + [0] * (4 - len(version_list))
|
|
|
|
|
|
|
|
|
2021-12-19 16:18:06 +01:00
|
|
|
def dependency_options():
|
2022-04-27 10:15:45 +02:00
|
|
|
# Due to the current implementation, these are auto-detected, but explicitly add them just in case
|
|
|
|
dependencies = [pycryptodome_module(), 'mutagen', 'brotli', 'certifi', 'websockets']
|
|
|
|
excluded_modules = ['test', 'ytdlp_plugins', 'youtube_dl', 'youtube_dlc']
|
2021-10-21 14:57:58 +02:00
|
|
|
|
2021-12-19 16:18:06 +01:00
|
|
|
yield from (f'--hidden-import={module}' for module in dependencies)
|
2022-04-27 10:15:45 +02:00
|
|
|
yield '--collect-submodules=websockets'
|
2021-10-21 14:57:58 +02:00
|
|
|
yield from (f'--exclude-module={module}' for module in excluded_modules)
|
2021-01-29 18:45:27 +01:00
|
|
|
|
2021-10-06 03:04:10 +02:00
|
|
|
|
|
|
|
def pycryptodome_module():
|
|
|
|
try:
|
|
|
|
import Cryptodome # noqa: F401
|
|
|
|
except ImportError:
|
|
|
|
try:
|
|
|
|
import Crypto # noqa: F401
|
|
|
|
print('WARNING: Using Crypto since Cryptodome is not available. '
|
|
|
|
'Install with: pip install pycryptodomex', file=sys.stderr)
|
|
|
|
return 'Crypto'
|
|
|
|
except ImportError:
|
|
|
|
pass
|
|
|
|
return 'Cryptodome'
|
|
|
|
|
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
def set_version_info(exe, version):
|
2022-06-03 17:59:03 +02:00
|
|
|
if OS_NAME == 'win32':
|
2021-10-21 14:57:58 +02:00
|
|
|
windows_set_version(exe, version)
|
|
|
|
|
|
|
|
|
|
|
|
def windows_set_version(exe, version):
|
2022-05-22 13:37:18 +02:00
|
|
|
from PyInstaller.utils.win32.versioninfo import (
|
|
|
|
FixedFileInfo,
|
|
|
|
SetVersion,
|
|
|
|
StringFileInfo,
|
|
|
|
StringStruct,
|
|
|
|
StringTable,
|
|
|
|
VarFileInfo,
|
|
|
|
VarStruct,
|
|
|
|
VSVersionInfo,
|
|
|
|
)
|
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
version_list = version_to_list(version)
|
2022-07-16 18:22:48 +02:00
|
|
|
suffix = MACHINE and f'_{MACHINE}'
|
2021-10-21 14:57:58 +02:00
|
|
|
SetVersion(exe, VSVersionInfo(
|
|
|
|
ffi=FixedFileInfo(
|
|
|
|
filevers=version_list,
|
|
|
|
prodvers=version_list,
|
|
|
|
mask=0x3F,
|
|
|
|
flags=0x0,
|
|
|
|
OS=0x4,
|
|
|
|
fileType=0x1,
|
|
|
|
subtype=0x0,
|
|
|
|
date=(0, 0),
|
|
|
|
),
|
|
|
|
kids=[
|
|
|
|
StringFileInfo([StringTable('040904B0', [
|
2022-07-16 18:22:48 +02:00
|
|
|
StringStruct('Comments', 'yt-dlp%s Command Line Interface' % suffix),
|
2021-10-21 14:57:58 +02:00
|
|
|
StringStruct('CompanyName', 'https://github.com/yt-dlp'),
|
2022-07-16 18:22:48 +02:00
|
|
|
StringStruct('FileDescription', 'yt-dlp%s' % (MACHINE and f' ({MACHINE})')),
|
2021-10-21 14:57:58 +02:00
|
|
|
StringStruct('FileVersion', version),
|
|
|
|
StringStruct('InternalName', f'yt-dlp{suffix}'),
|
|
|
|
StringStruct('LegalCopyright', 'pukkandan.ytdlp@gmail.com | UNLICENSE'),
|
|
|
|
StringStruct('OriginalFilename', f'yt-dlp{suffix}.exe'),
|
|
|
|
StringStruct('ProductName', f'yt-dlp{suffix}'),
|
|
|
|
StringStruct(
|
|
|
|
'ProductVersion', f'{version}{suffix} on Python {platform.python_version()}'),
|
|
|
|
])]), VarFileInfo([VarStruct('Translation', [0, 1200])])
|
|
|
|
]
|
|
|
|
))
|
2021-06-21 19:23:17 +02:00
|
|
|
|
2021-10-21 12:48:46 +02:00
|
|
|
|
2021-10-21 14:57:58 +02:00
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|