1 from __future__
import annotations
13 from dataclasses
import dataclass
14 from zipimport
import zipimporter
16 from .networking
import Request
17 from .networking
.exceptions
import HTTPError
, network_exceptions
28 from .version
import (
38 'stable': 'yt-dlp/yt-dlp',
39 'nightly': 'yt-dlp/yt-dlp-nightly-builds',
40 'master': 'yt-dlp/yt-dlp-master-builds',
42 REPOSITORY
= UPDATE_SOURCES
['stable']
43 _INVERSE_UPDATE_SOURCES
= {value
: key
for key
, value
in UPDATE_SOURCES
.items()}
45 _VERSION_RE
= re
.compile(r
'(\d+\.)*\d+')
46 _HASH_PATTERN
= r
'[\da-f]{40}'
47 _COMMIT_RE
= re
.compile(rf
'Generated from: https://(?:[^/?#]+/){{3}}commit/(?P<hash>{_HASH_PATTERN})')
49 API_BASE_URL
= 'https://api.github.com/repos'
51 # Backwards compatibility variables for the current channel
52 API_URL
= f
'{API_BASE_URL}/{REPOSITORY}/releases'
56 def _get_variant_and_executable_path():
57 """@returns (variant, executable_path)"""
58 if getattr(sys
, 'frozen', False):
60 if not hasattr(sys
, '_MEIPASS'):
62 elif sys
._MEIPASS
== os
.path
.dirname(path
):
63 return f
'{sys.platform}_dir', path
64 elif sys
.platform
== 'darwin':
65 machine
= '_legacy' if version_tuple(platform
.mac_ver()[0]) < (10, 15) else ''
67 machine
= f
'_{platform.machine().lower()}'
68 # Ref: https://en.wikipedia.org/wiki/Uname#Examples
69 if machine
[1:] in ('x86', 'x86_64', 'amd64', 'i386', 'i686'):
70 machine
= '_x86' if platform
.architecture()[0][:2] == '32' else ''
71 # sys.executable returns a /tmp/ path for staticx builds (linux_static)
72 # Ref: https://staticx.readthedocs.io/en/latest/usage.html#run-time-information
73 if static_exe_path
:= os
.getenv('STATICX_PROG_PATH'):
74 path
= static_exe_path
75 return f
'{remove_end(sys.platform, "32")}{machine}_exe', path
77 path
= os
.path
.dirname(__file__
)
78 if isinstance(__loader__
, zipimporter
):
79 return 'zip', os
.path
.join(path
, '..')
80 elif (os
.path
.basename(sys
.argv
[0]) in ('__main__.py', '-m')
81 and os
.path
.exists(os
.path
.join(path
, '../.git/HEAD'))):
83 return 'unknown', path
87 return VARIANT
or _get_variant_and_executable_path()[0]
91 def current_git_head():
92 if detect_variant() != 'source':
94 with contextlib
.suppress(Exception):
95 stdout
, _
, _
= Popen
.run(
96 ['git', 'rev-parse', '--short', 'HEAD'],
97 text
=True, cwd
=os
.path
.dirname(os
.path
.abspath(__file__
)),
98 stdout
=subprocess
.PIPE
, stderr
=subprocess
.PIPE
)
99 if re
.fullmatch('[0-9a-f]+', stdout
.strip()):
100 return stdout
.strip()
106 'win_x86_exe': '_x86.exe',
107 'darwin_exe': '_macos',
108 'darwin_legacy_exe': '_macos_legacy',
109 'linux_exe': '_linux',
110 'linux_aarch64_exe': '_linux_aarch64',
111 'linux_armv7l_exe': '_linux_armv7l',
114 _NON_UPDATEABLE_REASONS
= {
115 **{variant
: None for variant
in _FILE_SUFFIXES
}, # Updatable
116 **{variant
: f
'Auto-update is not supported for unpackaged {name} executable; Re-download the latest release'
117 for variant
, name
in {'win32_dir': 'Windows', 'darwin_dir': 'MacOS', 'linux_dir': 'Linux'}.items()},
118 'py2exe': 'py2exe is no longer supported by yt-dlp; This executable cannot be updated',
119 'source': 'You cannot update when running from source code; Use git to pull the latest changes',
120 'unknown': 'You installed yt-dlp from a manual build or with a package manager; Use that to update',
121 'other': 'You are using an unofficial build of yt-dlp; Build the executable again',
125 def is_non_updateable():
128 return _NON_UPDATEABLE_REASONS
.get(
129 detect_variant(), _NON_UPDATEABLE_REASONS
['unknown' if VARIANT
else 'other'])
132 def _get_binary_name():
133 return format_field(_FILE_SUFFIXES
, detect_variant(), template
='yt-dlp%s', ignore
=None, default
=None)
136 def _get_system_deprecation():
137 MIN_SUPPORTED
, MIN_RECOMMENDED
= (3, 9), (3, 9)
139 if sys
.version_info
> MIN_RECOMMENDED
:
142 major
, minor
= sys
.version_info
[:2]
143 PYTHON_MSG
= f
'Please update to Python {".".join(map(str, MIN_RECOMMENDED))} or above'
145 if sys
.version_info
< MIN_SUPPORTED
:
146 return f
'Python version {major}.{minor} is no longer supported! {PYTHON_MSG}'
148 return f
'Support for Python version {major}.{minor} has been deprecated. {PYTHON_MSG}'
151 def _sha256_file(path
):
153 mv
= memoryview(bytearray(128 * 1024))
154 with
open(os
.path
.realpath(path
), 'rb', buffering
=0) as f
:
155 for n
in iter(lambda: f
.readinto(mv
), 0):
160 def _make_label(origin
, tag
, version
=None):
162 channel
= _INVERSE_UPDATE_SOURCES
.get(origin
, origin
)
165 label
= f
'{channel}@{tag}'
166 if version
and version
!= tag
:
167 label
+= f
' build {version}'
168 if channel
!= origin
:
169 label
+= f
' from {origin}'
176 Update target information
178 Can be created by `query_update()` or manually.
181 tag The release tag that will be updated to. If from query_update,
182 the value is after API resolution and update spec processing.
183 The only property that is required.
184 version The actual numeric version (if available) of the binary to be updated to,
185 after API resolution and update spec processing. (default: None)
186 requested_version Numeric version of the binary being requested (if available),
187 after API resolution only. (default: None)
188 commit Commit hash (if available) of the binary to be updated to,
189 after API resolution and update spec processing. (default: None)
190 This value will only match the RELEASE_GIT_HEAD of prerelease builds.
191 binary_name Filename of the binary to be updated to. (default: current binary name)
192 checksum Expected checksum (if available) of the binary to be
193 updated to. (default: None)
196 version
: str |
None = None
197 requested_version
: str |
None = None
198 commit
: str |
None = None
200 binary_name
: str |
None = _get_binary_name() # noqa: RUF009: Always returns the same value
201 checksum
: str |
None = None
205 # XXX: use class variables to simplify testing
208 _update_sources
= UPDATE_SOURCES
210 def __init__(self
, ydl
, target
: str |
None = None):
212 # For backwards compat, target needs to be treated as if it could be None
213 self
.requested_channel
, sep
, self
.requested_tag
= (target
or self
._channel
).rpartition('@')
214 # Check if requested_tag is actually the requested repo/channel
215 if not sep
and ('/' in self
.requested_tag
or self
.requested_tag
in self
._update
_sources
):
216 self
.requested_channel
= self
.requested_tag
217 self
.requested_tag
: str = None # type: ignore (we set it later)
218 elif not self
.requested_channel
:
219 # User did not specify a channel, so we are requesting the default channel
220 self
.requested_channel
= self
._channel
.partition('@')[0]
222 # --update should not be treated as an exact tag request even if CHANNEL has a @tag
223 self
._exact
= bool(target
) and target
!= self
._channel
224 if not self
.requested_tag
:
225 # User did not specify a tag, so we request 'latest' and track that no exact tag was passed
226 self
.requested_tag
= 'latest'
229 if '/' in self
.requested_channel
:
230 # requested_channel is actually a repository
231 self
.requested_repo
= self
.requested_channel
232 if not self
.requested_repo
.startswith('yt-dlp/') and self
.requested_repo
!= self
._origin
:
233 self
.ydl
.report_warning(
234 f
'You are switching to an {self.ydl._format_err("unofficial", "red")} executable '
235 f
'from {self.ydl._format_err(self.requested_repo, self.ydl.Styles.EMPHASIS)}. '
236 f
'Run {self.ydl._format_err("at your own risk", "light red")}')
237 self
._block
_restart
('Automatically restarting into custom builds is disabled for security reasons')
239 # Check if requested_channel resolves to a known repository or else raise
240 self
.requested_repo
= self
._update
_sources
.get(self
.requested_channel
)
241 if not self
.requested_repo
:
243 f
'Invalid update channel {self.requested_channel!r} requested. '
244 f
'Valid channels are {", ".join(self._update_sources)}', True)
246 self
._identifier
= f
'{detect_variant()} {system_identifier()}'
249 def current_version(self
):
250 """Current version"""
254 def current_commit(self
):
255 """Current commit hash"""
256 return RELEASE_GIT_HEAD
258 def _download_asset(self
, name
, tag
=None):
260 tag
= self
.requested_tag
262 path
= 'latest/download' if tag
== 'latest' else f
'download/{tag}'
263 url
= f
'https://github.com/{self.requested_repo}/releases/{path}/{name}'
264 self
.ydl
.write_debug(f
'Downloading {name} from {url}')
265 return self
.ydl
.urlopen(url
).read()
267 def _call_api(self
, tag
):
268 tag
= f
'tags/{tag}' if tag
!= 'latest' else tag
269 url
= f
'{API_BASE_URL}/{self.requested_repo}/releases/{tag}'
270 self
.ydl
.write_debug(f
'Fetching release info: {url}')
271 return json
.loads(self
.ydl
.urlopen(Request(url
, headers
={
272 'Accept': 'application/vnd.github+json',
273 'User-Agent': 'yt-dlp',
274 'X-GitHub-Api-Version': '2022-11-28',
277 def _get_version_info(self
, tag
: str) -> tuple[str |
None, str |
None]:
278 if _VERSION_RE
.fullmatch(tag
):
281 api_info
= self
._call
_api
(tag
)
284 requested_version
= api_info
['tag_name']
286 match
= re
.search(rf
'\s+(?P<version>{_VERSION_RE.pattern})$', api_info
.get('name', ''))
287 requested_version
= match
.group('version') if match
else None
289 if re
.fullmatch(_HASH_PATTERN
, api_info
.get('target_commitish', '')):
290 target_commitish
= api_info
['target_commitish']
292 match
= _COMMIT_RE
.match(api_info
.get('body', ''))
293 target_commitish
= match
.group('hash') if match
else None
295 if not (requested_version
or target_commitish
):
296 self
._report
_error
('One of either version or commit hash must be available on the release', expected
=True)
298 return requested_version
, target_commitish
300 def _download_update_spec(self
, source_tags
):
301 for tag
in source_tags
:
303 return self
._download
_asset
('_update_spec', tag
=tag
).decode()
304 except network_exceptions
as error
:
305 if isinstance(error
, HTTPError
) and error
.status
== 404:
307 self
._report
_network
_error
(f
'fetch update spec: {error}')
311 f
'The requested tag {self.requested_tag} does not exist for {self.requested_repo}', True)
314 def _process_update_spec(self
, lockfile
: str, resolved_tag
: str):
315 lines
= lockfile
.splitlines()
316 is_version2
= any(line
.startswith('lockV2 ') for line
in lines
)
320 if not line
.startswith(f
'lockV2 {self.requested_repo} '):
322 _
, _
, tag
, pattern
= line
.split(' ', 3)
324 if not line
.startswith('lock '):
326 _
, tag
, pattern
= line
.split(' ', 2)
328 if re
.match(pattern
, self
._identifier
):
329 if _VERSION_RE
.fullmatch(tag
):
332 elif self
._version
_compare
(tag
, resolved_tag
):
334 elif tag
!= resolved_tag
:
338 f
'yt-dlp cannot be updated to {resolved_tag} since you are on an older Python version '
339 'or your operating system is not compatible with the requested build', True)
344 def _version_compare(self
, a
: str, b
: str):
346 Compare two version strings
348 This function SHOULD NOT be called if self._exact == True
350 if _VERSION_RE
.fullmatch(f
'{a}.{b}'):
351 return version_tuple(a
) >= version_tuple(b
)
354 def query_update(self
, *, _output
=False) -> UpdateInfo |
None:
355 """Fetches info about the available update
356 @returns An `UpdateInfo` if there is an update available, else None
358 if not self
.requested_repo
:
359 self
._report
_error
('No target repository could be determined from input')
363 requested_version
, target_commitish
= self
._get
_version
_info
(self
.requested_tag
)
364 except network_exceptions
as e
:
365 self
._report
_network
_error
(f
'obtain version info ({e})', delim
='; Please try again later or')
368 if self
._exact
and self
._origin
!= self
.requested_repo
:
370 elif requested_version
:
372 has_update
= self
.current_version
!= requested_version
374 has_update
= not self
._version
_compare
(self
.current_version
, requested_version
)
375 elif target_commitish
:
376 has_update
= target_commitish
!= self
.current_commit
380 resolved_tag
= requested_version
if self
.requested_tag
== 'latest' else self
.requested_tag
381 current_label
= _make_label(self
._origin
, self
._channel
.partition('@')[2] or self
.current_version
, self
.current_version
)
382 requested_label
= _make_label(self
.requested_repo
, resolved_tag
, requested_version
)
383 latest_or_requested
= f
'{"Latest" if self.requested_tag == "latest" else "Requested"} version: {requested_label}'
386 self
.ydl
.to_screen(f
'{latest_or_requested}\nyt-dlp is up to date ({current_label})')
389 update_spec
= self
._download
_update
_spec
(('latest', None) if requested_version
else (None,))
392 # `result_` prefixed vars == post-_process_update_spec() values
393 result_tag
= self
._process
_update
_spec
(update_spec
, resolved_tag
)
394 if not result_tag
or result_tag
== self
.current_version
:
396 elif result_tag
== resolved_tag
:
397 result_version
= requested_version
398 elif _VERSION_RE
.fullmatch(result_tag
):
399 result_version
= result_tag
400 else: # actual version being updated to is unknown
401 result_version
= None
404 # Non-updateable variants can get update_info but need to skip checksum
405 if not is_non_updateable():
407 hashes
= self
._download
_asset
('SHA2-256SUMS', result_tag
)
408 except network_exceptions
as error
:
409 if not isinstance(error
, HTTPError
) or error
.status
!= 404:
410 self
._report
_network
_error
(f
'fetch checksums: {error}')
412 self
.ydl
.report_warning('No hash information found for the release, skipping verification')
414 for ln
in hashes
.decode().splitlines():
415 if ln
.endswith(_get_binary_name()):
416 checksum
= ln
.split()[0]
419 self
.ydl
.report_warning('The hash could not be found in the checksum file, skipping verification')
422 update_label
= _make_label(self
.requested_repo
, result_tag
, result_version
)
424 f
'Current version: {current_label}\n{latest_or_requested}'
425 + (f
'\nUpgradable to: {update_label}' if update_label
!= requested_label
else ''))
429 version
=result_version
,
430 requested_version
=requested_version
,
431 commit
=target_commitish
if result_tag
== resolved_tag
else None,
434 def update(self
, update_info
=NO_DEFAULT
):
435 """Update yt-dlp executable to the latest version
436 @param update_info `UpdateInfo | None` as returned by query_update()
438 if update_info
is NO_DEFAULT
:
439 update_info
= self
.query_update(_output
=True)
443 err
= is_non_updateable()
445 self
._report
_error
(err
, True)
448 self
.ydl
.to_screen(f
'Current Build Hash: {_sha256_file(self.filename)}')
450 update_label
= _make_label(self
.requested_repo
, update_info
.tag
, update_info
.version
)
451 self
.ydl
.to_screen(f
'Updating to {update_label} ...')
453 directory
= os
.path
.dirname(self
.filename
)
454 if not os
.access(self
.filename
, os
.W_OK
):
455 return self
._report
_permission
_error
(self
.filename
)
456 elif not os
.access(directory
, os
.W_OK
):
457 return self
._report
_permission
_error
(directory
)
459 new_filename
, old_filename
= f
'{self.filename}.new', f
'{self.filename}.old'
460 if detect_variant() == 'zip': # Can be replaced in-place
461 new_filename
, old_filename
= self
.filename
, None
464 if os
.path
.exists(old_filename
or ''):
465 os
.remove(old_filename
)
467 return self
._report
_error
('Unable to remove the old version')
470 newcontent
= self
._download
_asset
(update_info
.binary_name
, update_info
.tag
)
471 except network_exceptions
as e
:
472 if isinstance(e
, HTTPError
) and e
.status
== 404:
473 return self
._report
_error
(
474 f
'The requested tag {self.requested_repo}@{update_info.tag} does not exist', True)
475 return self
._report
_network
_error
(f
'fetch updates: {e}', tag
=update_info
.tag
)
477 if not update_info
.checksum
:
478 self
._block
_restart
('Automatically restarting into unverified builds is disabled for security reasons')
479 elif hashlib
.sha256(newcontent
).hexdigest() != update_info
.checksum
:
480 return self
._report
_network
_error
('verify the new executable', tag
=update_info
.tag
)
483 with
open(new_filename
, 'wb') as outf
:
484 outf
.write(newcontent
)
486 return self
._report
_permission
_error
(new_filename
)
489 mask
= os
.stat(self
.filename
).st_mode
491 os
.rename(self
.filename
, old_filename
)
493 return self
._report
_error
('Unable to move current version')
496 os
.rename(new_filename
, self
.filename
)
498 self
._report
_error
('Unable to overwrite current version')
499 return os
.rename(old_filename
, self
.filename
)
501 variant
= detect_variant()
502 if variant
.startswith('win'):
503 atexit
.register(Popen
, f
'ping 127.0.0.1 -n 5 -w 1000 & del /F "{old_filename}"',
504 shell
=True, stdout
=subprocess
.DEVNULL
, stderr
=subprocess
.DEVNULL
)
507 os
.remove(old_filename
)
509 self
._report
_error
('Unable to remove the old version')
512 os
.chmod(self
.filename
, mask
)
514 return self
._report
_error
(
515 f
'Unable to set permissions. Run: sudo chmod a+rx {shell_quote(self.filename)}')
517 self
.ydl
.to_screen(f
'Updated yt-dlp to {update_label}')
520 @functools.cached_property
522 """Filename of the executable"""
523 return os
.path
.realpath(_get_variant_and_executable_path()[1])
525 @functools.cached_property
527 """The command-line to run the executable, if known"""
528 # There is no sys.orig_argv in py < 3.10. Also, it can be [] when frozen
529 if getattr(sys
, 'orig_argv', None):
531 elif getattr(sys
, 'frozen', False):
535 """Restart the executable"""
536 assert self
.cmd
, 'Must be frozen or Py >= 3.10'
537 self
.ydl
.write_debug(f
'Restarting: {shell_quote(self.cmd)}')
538 _
, _
, returncode
= Popen
.run(self
.cmd
)
541 def _block_restart(self
, msg
):
543 self
._report
_error
(f
'{msg}. Restart yt-dlp to use the updated version', expected
=True)
544 return self
.ydl
._download
_retcode
545 self
.restart
= wrapper
547 def _report_error(self
, msg
, expected
=False):
548 self
.ydl
.report_error(msg
, tb
=False if expected
else None)
549 self
.ydl
._download
_retcode
= 100
551 def _report_permission_error(self
, file):
552 self
._report
_error
(f
'Unable to write to {file}; try running as administrator', True)
554 def _report_network_error(self
, action
, delim
=';', tag
=None):
556 tag
= self
.requested_tag
557 path
= tag
if tag
== 'latest' else f
'tag/{tag}'
559 f
'Unable to {action}{delim} visit '
560 f
'https://github.com/{self.requested_repo}/releases/{path}', True)
564 """Update the program file with the latest version from the repository
565 @returns Whether there was a successful update (No update = False)
568 '"yt_dlp.update.run_update(ydl)" is deprecated and may be removed in a future version. '
569 'Use "yt_dlp.update.Updater(ydl).update()" instead')
570 return Updater(ydl
).update()
573 __all__
= ['Updater']