| 
									
										
										
										
											2015-06-03 15:10:18 +00:00
										 |  |  | from __future__ import unicode_literals | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  | import itertools | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  | from .fragment import FragmentFD | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  | from ..compat import compat_urllib_error | 
					
						
							| 
									
										
										
										
											2018-07-08 01:22:56 +00:00
										 |  |  | from ..utils import ( | 
					
						
							|  |  |  |     DownloadError, | 
					
						
							|  |  |  |     urljoin, | 
					
						
							|  |  |  | ) | 
					
						
							| 
									
										
										
										
											2015-06-04 14:12:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-03 15:10:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  | class DashSegmentsFD(FragmentFD): | 
					
						
							| 
									
										
										
										
											2015-06-03 15:10:18 +00:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     Download segments in a DASH manifest | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |     FD_NAME = 'dashsegments' | 
					
						
							| 
									
										
										
										
											2015-06-10 06:45:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |     def real_download(self, filename, info_dict): | 
					
						
							| 
									
										
										
										
											2017-08-04 23:57:19 +00:00
										 |  |  |         fragment_base_url = info_dict.get('fragment_base_url') | 
					
						
							|  |  |  |         fragments = info_dict['fragments'][:1] if self.params.get( | 
					
						
							| 
									
										
										
										
											2016-09-17 13:35:22 +00:00
										 |  |  |             'test', False) else info_dict['fragments'] | 
					
						
							| 
									
										
										
										
											2015-06-10 06:45:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |         ctx = { | 
					
						
							|  |  |  |             'filename': filename, | 
					
						
							| 
									
										
										
										
											2017-08-04 23:57:19 +00:00
										 |  |  |             'total_frags': len(fragments), | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-06-10 06:45:54 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |         self._prepare_and_start_frag_download(ctx) | 
					
						
							| 
									
										
										
										
											2015-06-03 15:10:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  |         fragment_retries = self.params.get('fragment_retries', 0) | 
					
						
							| 
									
										
										
										
											2016-08-26 21:55:55 +00:00
										 |  |  |         skip_unavailable_fragments = self.params.get('skip_unavailable_fragments', True) | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  |         for frag_index, fragment in enumerate(fragments, 1): | 
					
						
							| 
									
										
										
										
											2017-04-22 15:42:24 +00:00
										 |  |  |             if frag_index <= ctx['fragment_index']: | 
					
						
							| 
									
										
										
										
											2016-06-28 17:07:50 +00:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2024-01-27 00:07:14 +00:00
										 |  |  |             success = False | 
					
						
							| 
									
										
										
										
											2016-09-17 13:35:22 +00:00
										 |  |  |             # In DASH, the first segment contains necessary headers to | 
					
						
							|  |  |  |             # generate a valid MP4 file, so always abort for the first segment | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  |             fatal = frag_index == 1 or not skip_unavailable_fragments | 
					
						
							| 
									
										
										
										
											2023-04-12 23:15:07 +00:00
										 |  |  |             fragment_url = fragment.get('url') | 
					
						
							|  |  |  |             if not fragment_url: | 
					
						
							|  |  |  |                 assert fragment_base_url | 
					
						
							|  |  |  |                 fragment_url = urljoin(fragment_base_url, fragment['path']) | 
					
						
							| 
									
										
										
										
											2024-01-27 00:07:14 +00:00
										 |  |  |             headers = info_dict.get('http_headers') | 
					
						
							|  |  |  |             fragment_range = fragment.get('range') | 
					
						
							|  |  |  |             if fragment_range: | 
					
						
							|  |  |  |                 headers = headers.copy() if headers else {} | 
					
						
							|  |  |  |                 headers['Range'] = 'bytes=%s' % (fragment_range,) | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  |             for count in itertools.count(): | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  |                 try: | 
					
						
							| 
									
										
										
										
											2024-01-27 00:07:14 +00:00
										 |  |  |                     success, frag_content = self._download_fragment(ctx, fragment_url, info_dict, headers) | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  |                     if not success: | 
					
						
							|  |  |  |                         return False | 
					
						
							| 
									
										
										
										
											2016-06-28 17:07:50 +00:00
										 |  |  |                     self._append_fragment(ctx, frag_content) | 
					
						
							| 
									
										
										
										
											2016-08-26 21:57:59 +00:00
										 |  |  |                 except compat_urllib_error.HTTPError as err: | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  |                     # YouTube may often return 404 HTTP error for a fragment causing the | 
					
						
							|  |  |  |                     # whole download to fail. However if the same fragment is immediately | 
					
						
							| 
									
										
										
										
											2019-07-26 15:30:18 +00:00
										 |  |  |                     # retried with the same request data this usually succeeds (1-2 attempts | 
					
						
							| 
									
										
										
										
											2016-03-19 14:42:23 +00:00
										 |  |  |                     # is usually enough) thus allowing to download the whole file successfully. | 
					
						
							| 
									
										
										
										
											2016-08-26 21:55:55 +00:00
										 |  |  |                     # To be future-proof we will retry all fragments that fail with any | 
					
						
							|  |  |  |                     # HTTP error. | 
					
						
							| 
									
										
										
										
											2023-03-11 12:17:00 +00:00
										 |  |  |                     if count < fragment_retries: | 
					
						
							|  |  |  |                         self.report_retry_fragment(err, frag_index, count + 1, fragment_retries) | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  |                         continue | 
					
						
							| 
									
										
										
										
											2018-07-08 01:22:56 +00:00
										 |  |  |                 except DownloadError: | 
					
						
							|  |  |  |                     # Don't retry fragment if error occurred during HTTP downloading | 
					
						
							| 
									
										
										
										
											2023-04-05 17:29:24 +00:00
										 |  |  |                     # itself since it has its own retry settings | 
					
						
							|  |  |  |                     if fatal: | 
					
						
							|  |  |  |                         raise | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2018-07-08 01:22:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-12 23:15:07 +00:00
										 |  |  |             if not success: | 
					
						
							| 
									
										
										
										
											2016-09-03 15:00:52 +00:00
										 |  |  |                 if not fatal: | 
					
						
							| 
									
										
										
										
											2016-06-28 17:07:50 +00:00
										 |  |  |                     self.report_skip_fragment(frag_index) | 
					
						
							|  |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2023-04-12 23:15:07 +00:00
										 |  |  |                 self.report_error('giving up after %s fragment retries' % count) | 
					
						
							| 
									
										
										
										
											2016-02-09 16:25:02 +00:00
										 |  |  |                 return False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._finish_frag_download(ctx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-03 15:10:18 +00:00
										 |  |  |         return True |