Compare commits
	
		
			1430 Commits
		
	
	
		
			2015.02.09
			...
			2015.06.15
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 4b9f9010b0 | ||
|  | 67d95f177c | ||
|  | 44773ad125 | ||
|  | 5774ef35c4 | ||
|  | b95cfa9170 | ||
|  | afa1ded425 | ||
|  | 00ac23e6e0 | ||
|  | 7d0c934a3e | ||
|  | 8f75761f24 | ||
|  | 9fd24e3a22 | ||
|  | 755a9d3d1a | ||
|  | ac499cb61c | ||
|  | 180940e02d | ||
|  | 976b03c56b | ||
|  | 450d89ddc1 | ||
|  | 463b2e5542 | ||
|  | 70a2002399 | ||
|  | a617b10075 | ||
|  | 0029071adb | ||
|  | ad49fe7c8f | ||
|  | 49bc802f81 | ||
|  | af9cdee9cb | ||
|  | b4e1576aee | ||
|  | 78e2b74bb9 | ||
|  | 65d161c480 | ||
|  | 9fcbd5db2a | ||
|  | 4f3bf679f5 | ||
|  | 8b6c896c4b | ||
|  | 185dbc4974 | ||
|  | 3d535e0471 | ||
|  | 9872d3110c | ||
|  | b859971873 | ||
|  | e5095f1198 | ||
|  | 499a077761 | ||
|  | 5da7177729 | ||
|  | 3507766bd0 | ||
|  | f37bdbe537 | ||
|  | 2da09ff8b0 | ||
|  | 5ccddb7ecf | ||
|  | 954c1d0529 | ||
|  | 494f20cbdc | ||
|  | 29902c8ec0 | ||
|  | 9f15bdabc8 | ||
|  | fff3455f58 | ||
|  | 87446dc618 | ||
|  | 99ac0390f5 | ||
|  | ff0f0b9172 | ||
|  | 97b570a94c | ||
|  | a9d56c6843 | ||
|  | f98470df69 | ||
|  | eb8be1fe76 | ||
|  | 7ebd5376fe | ||
|  | 70219b0f43 | ||
|  | bd5bc0cd5a | ||
|  | 6e054aacca | ||
|  | 9d581f3d52 | ||
|  | 9bf99891d0 | ||
|  | d9cf48e81e | ||
|  | e1b9322b09 | ||
|  | 627b964825 | ||
|  | a55e36f48d | ||
|  | 01e21b89ee | ||
|  | 788be3313d | ||
|  | e1ec93304d | ||
|  | edb99d4c18 | ||
|  | 68477c3dab | ||
|  | 65ba8b23f4 | ||
|  | 621ed9f5f4 | ||
|  | b26733ba7f | ||
|  | 9836cfb8d6 | ||
|  | 665b6c1236 | ||
|  | 9414338a48 | ||
|  | de390ea077 | ||
|  | 717b0239fd | ||
|  | d00735a0c5 | ||
|  | c23d5ce926 | ||
|  | b5a3c7f109 | ||
|  | 9c5f685ef1 | ||
|  | 08bb8ef201 | ||
|  | 865ab62f43 | ||
|  | 9948113590 | ||
|  | c4ee87022b | ||
|  | ffba4edb06 | ||
|  | 958d0b659b | ||
|  | aacda28b28 | ||
|  | 29e7e0781b | ||
|  | 7012620e2b | ||
|  | f1da861018 | ||
|  | 05aa9c82d9 | ||
|  | a9e58ecd3f | ||
|  | 223544552f | ||
|  | 3d8e9573a4 | ||
|  | 54eb81a087 | ||
|  | c33c547d66 | ||
|  | dfe7dd9bdb | ||
|  | 63ccf6474d | ||
|  | e8ac61e840 | ||
|  | f00a650705 | ||
|  | 4bde5ce992 | ||
|  | d31573fa37 | ||
|  | 8b8cde2140 | ||
|  | 0e805e782b | ||
|  | f5c78d118b | ||
|  | 9d4f213f90 | ||
|  | 168db222c6 | ||
|  | 3d6388e34e | ||
|  | 3ce9bc712a | ||
|  | e52c0bd0eb | ||
|  | 56c837ccb7 | ||
|  | 55e5841f14 | ||
|  | ed15e9ba02 | ||
|  | eedda32e6b | ||
|  | 4c8fea92f3 | ||
|  | d073055dcd | ||
|  | e4ac7bb1e5 | ||
|  | 9bac8c57e3 | ||
|  | 3153a2c98d | ||
|  | 15b74b94be | ||
|  | 687cb3ad35 | ||
|  | 8f94784124 | ||
|  | 23dd1fc74c | ||
|  | fa971259e6 | ||
|  | b0cda32f72 | ||
|  | 08b7968e28 | ||
|  | 4b5fe1349f | ||
|  | d23da75b32 | ||
|  | 06e027992d | ||
|  | b5597738d4 | ||
|  | bc03e58565 | ||
|  | a00234f1c5 | ||
|  | 34c0f95db2 | ||
|  | fcb04bcaca | ||
|  | 9464a194db | ||
|  | 9f4b9118cc | ||
|  | 60158217ef | ||
|  | 923e79e2e4 | ||
|  | 866b296d0f | ||
|  | 4053ee9104 | ||
|  | 47fd8c2f76 | ||
|  | 96b9690985 | ||
|  | df15ef8dab | ||
|  | 002c0fb511 | ||
|  | 7584e38ce4 | ||
|  | eb47569f8a | ||
|  | d2a9de78df | ||
|  | c5138a7ce4 | ||
|  | c5fa81fe81 | ||
|  | a074e92296 | ||
|  | 1ddb9456c4 | ||
|  | 58bde34a23 | ||
|  | 339516072b | ||
|  | 931bc3c3a7 | ||
|  | db1e9ee771 | ||
|  | a2d971309b | ||
|  | d05a1dbe70 | ||
|  | a80601f8d9 | ||
|  | 1c22238756 | ||
|  | 9ff811c5cd | ||
|  | 1ebc05df91 | ||
|  | 386bdfa698 | ||
|  | 1ae7ff771b | ||
|  | 5196b98897 | ||
|  | e6e63e91a7 | ||
|  | b4dd98358f | ||
|  | 181c7053e3 | ||
|  | 4d454c5e4b | ||
|  | 5c2191a605 | ||
|  | bba5bfc890 | ||
|  | 1a5b77dc21 | ||
|  | b2cf6543b2 | ||
|  | 0385d64223 | ||
|  | 6ebdfe43e4 | ||
|  | fafec39d41 | ||
|  | 670861bd20 | ||
|  | 605ec701b7 | ||
|  | d6aa68ce75 | ||
|  | eb6cb9fbe9 | ||
|  | 84e1e036c2 | ||
|  | 806598b94d | ||
|  | e26be70bca | ||
|  | 9e0b579128 | ||
|  | ff4a1279f2 | ||
|  | 9b254aa177 | ||
|  | 703d78bbf5 | ||
|  | d9446c7319 | ||
|  | b25b645d51 | ||
|  | d90b3854ca | ||
|  | bf24c3d017 | ||
|  | f0bfaa2d7d | ||
|  | f9f3e3df9a | ||
|  | f8d5e1cfb5 | ||
|  | c23848b3c5 | ||
|  | 6d00a2dcd1 | ||
|  | b535170b21 | ||
|  | 1434184c57 | ||
|  | 7a372b64df | ||
|  | 5406af92bc | ||
|  | 7d65242dc3 | ||
|  | 544a8693b7 | ||
|  | 35a4f24a37 | ||
|  | ff305edd64 | ||
|  | efec4358b9 | ||
|  | db3ca36403 | ||
|  | 42833b44b5 | ||
|  | 5d0a33eebc | ||
|  | ba2df04b41 | ||
|  | c6bbdadd79 | ||
|  | b885bae634 | ||
|  | d41ebe146b | ||
|  | 4b4e1af059 | ||
|  | 80240b347e | ||
|  | 04b3b3df05 | ||
|  | 2ad5708c43 | ||
|  | 63f3cab4ae | ||
|  | 8cdf03a7a2 | ||
|  | d78c834ead | ||
|  | 05a976cd99 | ||
|  | 34fb7e46ad | ||
|  | abac15f3c6 | ||
|  | b700055ba4 | ||
|  | 23905927e1 | ||
|  | 56be5f1567 | ||
|  | 1807ae22dd | ||
|  | 71646e4653 | ||
|  | 1335c3aca8 | ||
|  | 30455ce255 | ||
|  | 9bf87ae3aa | ||
|  | abca34cbc0 | ||
|  | d386878af9 | ||
|  | 685c74d315 | ||
|  | 69e0f1b445 | ||
|  | 79979c6897 | ||
|  | ba64547616 | ||
|  | ed5a637d62 | ||
|  | 8a278a1d7e | ||
|  | 77d9cb2f04 | ||
|  | 0459432d96 | ||
|  | 43150d7ac3 | ||
|  | afe8b594be | ||
|  | 878563c847 | ||
|  | 06947add03 | ||
|  | 5cd47a5e4f | ||
|  | 53de95da5e | ||
|  | 663004ac2b | ||
|  | 6ad9cb224a | ||
|  | e7752cd578 | ||
|  | 4d2f42361e | ||
|  | 4d8ee01389 | ||
|  | d01924f488 | ||
|  | bc56355ec6 | ||
|  | ac20d95f97 | ||
|  | 1a83c731bd | ||
|  | ca57a59883 | ||
|  | b0d619fde2 | ||
|  | cc7051efd7 | ||
|  | 5137adb94d | ||
|  | 0b9f7cd074 | ||
|  | 2632941f32 | ||
|  | 137597b0ea | ||
|  | 051df9ad99 | ||
|  | f670ef1c8e | ||
|  | d9d747a06a | ||
|  | b813d8caf1 | ||
|  | ecee572411 | ||
|  | 1b0427e6c4 | ||
|  | 2aa64b89b3 | ||
|  | 484c9d2d5b | ||
|  | 5d8dcb5342 | ||
|  | 2328f2fe68 | ||
|  | 4f514c7e88 | ||
|  | 5bdc520cf1 | ||
|  | fc6e75dd57 | ||
|  | 4a5a898a8f | ||
|  | ba9d16291b | ||
|  | 725652e924 | ||
|  | 8da0e0e946 | ||
|  | 588b82bbf8 | ||
|  | bc0f937b55 | ||
|  | baa43cbaf0 | ||
|  | adb6b1b316 | ||
|  | 1c18de0019 | ||
|  | 4d52f2eb7f | ||
|  | 363cf58645 | ||
|  | 7e760fc188 | ||
|  | ef2dcbe4ad | ||
|  | 9354a5fad4 | ||
|  | 1c97b0a777 | ||
|  | 2f3bdab2b9 | ||
|  | 0d7f036429 | ||
|  | 2cda13213d | ||
|  | 70d0d43b5e | ||
|  | 25c3a7348f | ||
|  | 9123d64592 | ||
|  | b827a6015c | ||
|  | d40a3b5b55 | ||
|  | ef28a6cb26 | ||
|  | 1436a6835e | ||
|  | e8cfacae37 | ||
|  | 3a7382950b | ||
|  | eeb23eb7ea | ||
|  | 34fe5a94ba | ||
|  | 6181864290 | ||
|  | e9ca615a98 | ||
|  | 62c95fd5fc | ||
|  | 25f14e9f93 | ||
|  | ae670a6ed8 | ||
|  | a7b8467ac0 | ||
|  | 15da7ce7fb | ||
|  | e9eaf3fbcf | ||
|  | 3884dcf313 | ||
|  | c4fc559f45 | ||
|  | 2bc4330303 | ||
|  | 12675275a1 | ||
|  | 3a105f7b20 | ||
|  | 1ae72fb23d | ||
|  | 7ec676bb3d | ||
|  | 29ea57283e | ||
|  | 5488973961 | ||
|  | 96d45a5489 | ||
|  | 7a012d5a16 | ||
|  | fa6a16996e | ||
|  | 82245a6de7 | ||
|  | ff28ede2d1 | ||
|  | 98b8ec8616 | ||
|  | 88f9d8748c | ||
|  | 7d57d2e18b | ||
|  | 38caa00d18 | ||
|  | c827d4cfdb | ||
|  | 509c630db8 | ||
|  | fbff30d2db | ||
|  | 86c7fdb17c | ||
|  | 62bd6589c7 | ||
|  | 2cc6d13547 | ||
|  | bb8ca1d112 | ||
|  | 8e59539752 | ||
|  | 372744c544 | ||
|  | 83880949a1 | ||
|  | 3749e36e9f | ||
|  | 0b4253fa37 | ||
|  | 86ec1e487c | ||
|  | fd4eefed39 | ||
|  | b480e7874b | ||
|  | 41333b97b9 | ||
|  | c1c924abfe | ||
|  | 1c7e2e64f6 | ||
|  | 7dff03636a | ||
|  | 5332fd91bf | ||
|  | d4b963d0a6 | ||
|  | 6d3f5935e5 | ||
|  | 968ee17677 | ||
|  | 81ed3bb9c0 | ||
|  | 5115652828 | ||
|  | 1f92865494 | ||
|  | e41f450f28 | ||
|  | 97fcf1bbd0 | ||
|  | 13763ce599 | ||
|  | 7fcb605b82 | ||
|  | 70484b9f8a | ||
|  | 69b46b3d95 | ||
|  | 95c5534f8e | ||
|  | 370b39e8ec | ||
|  | 3da8038918 | ||
|  | a6762c4a22 | ||
|  | 98c2c0febc | ||
|  | 63cbd19f50 | ||
|  | 1934f3a0ea | ||
|  | a909e6ad43 | ||
|  | 1dcb52188d | ||
|  | 28ebef0b1b | ||
|  | f03a8a3c4e | ||
|  | 03f760b1c0 | ||
|  | f900dc3fb9 | ||
|  | 95eb1adda8 | ||
|  | c6ddbdb66c | ||
|  | 3800b908b1 | ||
|  | 69fe3a5f09 | ||
|  | 754270313a | ||
|  | 057ebeaca3 | ||
|  | 480065172d | ||
|  | f2e0056579 | ||
|  | 32fffff2cc | ||
|  | 3c47824d6b | ||
|  | 0892090a56 | ||
|  | d592b42f5c | ||
|  | 3b5f65a64c | ||
|  | 5c0b2c16a8 | ||
|  | d39e0f05db | ||
|  | 6d14d08e06 | ||
|  | 32060c6d6b | ||
|  | 3dbec410a0 | ||
|  | de765f6c31 | ||
|  | dc455a5f88 | ||
|  | bab19a8e91 | ||
|  | 322915014f | ||
|  | 79998cd5af | ||
|  | 50b9013064 | ||
|  | bb03fdae0d | ||
|  | 4384cf9e7d | ||
|  | d47e980d0d | ||
|  | fe373287eb | ||
|  | cbe443362f | ||
|  | 2c0c9dc46c | ||
|  | 0ceab84749 | ||
|  | 34e7dc81a9 | ||
|  | 4e6e9d21bd | ||
|  | d1feb30811 | ||
|  | 43837189c1 | ||
|  | 249962ffa2 | ||
|  | 541168039d | ||
|  | 7ef00afe9d | ||
|  | 156fc83a55 | ||
|  | 46be82b811 | ||
|  | 09b412dafa | ||
|  | 5268a05e47 | ||
|  | 406224be52 | ||
|  | 3799834dcf | ||
|  | 553e412bda | ||
|  | f22834a372 | ||
|  | bd349a8704 | ||
|  | bc08873cff | ||
|  | aafe273990 | ||
|  | c09593c04e | ||
|  | 84bf31aaf8 | ||
|  | 05d5392cda | ||
|  | d9a743d917 | ||
|  | ac6c358c2a | ||
|  | ad0c0ad3b4 | ||
|  | 1ed34f3dd6 | ||
|  | 6a8f9cd22e | ||
|  | e8b9ab8957 | ||
|  | 74f728249f | ||
|  | d6a1738892 | ||
|  | b326b07adc | ||
|  | 07d2921c6d | ||
|  | 22e462c97a | ||
|  | dcf8077906 | ||
|  | 3408f6e64a | ||
|  | e10dc0e1f0 | ||
|  | ce5c1ae517 | ||
|  | bbe718c97f | ||
|  | 01e4b1ee14 | ||
|  | 815ac0293e | ||
|  | 6568382d6f | ||
|  | f943b7ddce | ||
|  | ff9d68e7be | ||
|  | 7212560f4d | ||
|  | 1aa43d77c0 | ||
|  | e038d5c4e3 | ||
|  | dfad3aac98 | ||
|  | df8418ffcf | ||
|  | 50aa43b3ae | ||
|  | a90552663e | ||
|  | 883340c107 | ||
|  | 0fe2ff78e6 | ||
|  | dc1eed93be | ||
|  | b2f82360d7 | ||
|  | 782e0568ef | ||
|  | 90b4b0eabe | ||
|  | cec04ef3a6 | ||
|  | 71fa56b887 | ||
|  | b9b3ab45ea | ||
|  | 957b794c26 | ||
|  | 8001607e90 | ||
|  | 3e7202c1bc | ||
|  | 848edeab89 | ||
|  | 1748d67aea | ||
|  | 5477ca8239 | ||
|  | d0fd305023 | ||
|  | 8dab1e9072 | ||
|  | 963aea5279 | ||
|  | 0a64aa7355 | ||
|  | 0669c89c55 | ||
|  | 2699da8041 | ||
|  | 98727e123f | ||
|  | b29e0000e6 | ||
|  | b3ed15b760 | ||
|  | 666a9a2b95 | ||
|  | a4bcaad773 | ||
|  | e65e4c8874 | ||
|  | 21f6330274 | ||
|  | 38c6902b90 | ||
|  | 2ddcd88129 | ||
|  | dd8920653c | ||
|  | c938c35f95 | ||
|  | 2eb0192155 | ||
|  | d948e09b61 | ||
|  | 89966a5aea | ||
|  | 8e3df9dfee | ||
|  | 5890eef6b0 | ||
|  | 083c1bb960 | ||
|  | 861e65eb05 | ||
|  | 650cfd0cb0 | ||
|  | e68ae99a41 | ||
|  | 8683b4d8d9 | ||
|  | 1dbd717eb4 | ||
|  | 6a8422b942 | ||
|  | cb202fd286 | ||
|  | 67fc8ecd53 | ||
|  | df8301fef5 | ||
|  | 4070b458ec | ||
|  | ffbc3901d2 | ||
|  | 7a03280df4 | ||
|  | 482a1258de | ||
|  | cd298882cd | ||
|  | e01c56f9e1 | ||
|  | 4d72df4031 | ||
|  | f7f1df1d82 | ||
|  | c4a21bc9db | ||
|  | 621ffe7bf4 | ||
|  | 8dd5418803 | ||
|  | 965cb8d530 | ||
|  | b2e8e7dab5 | ||
|  | 59d814f793 | ||
|  | bb865f3a5e | ||
|  | 9ee53a49f0 | ||
|  | 79adb09baa | ||
|  | cf0649f8b7 | ||
|  | f8690631e2 | ||
|  | 5456d78f0c | ||
|  | cbbece96a2 | ||
|  | 9d8ba307ef | ||
|  | ec7c1e85e0 | ||
|  | e70c7568c0 | ||
|  | 39b62db116 | ||
|  | 2edce52584 | ||
|  | 10831b5ec9 | ||
|  | 3a0f0c263a | ||
|  | 2419a376b9 | ||
|  | e206740fd7 | ||
|  | 290a5a8d85 | ||
|  | e2dc351d25 | ||
|  | c86b61428b | ||
|  | 40b96352c9 | ||
|  | 189ba90996 | ||
|  | c8183e661d | ||
|  | 053c94f1b3 | ||
|  | b9d76a9571 | ||
|  | a01cfc2951 | ||
|  | 4eb5c65bee | ||
|  | 06d07c4000 | ||
|  | 74f8654a53 | ||
|  | 9e105a858c | ||
|  | cd8a07a764 | ||
|  | aa49acd15a | ||
|  | 642f23bd81 | ||
|  | 2e24e6bd17 | ||
|  | 2a09c1b8ab | ||
|  | a5ebf77d87 | ||
|  | b874495b1f | ||
|  | b860f5dfd4 | ||
|  | b19fc36c81 | ||
|  | d2d8248f68 | ||
|  | f54bab4d67 | ||
|  | bf6427d2fb | ||
|  | 672f1bd849 | ||
|  | 529d26c3e1 | ||
|  | 857f00ed94 | ||
|  | e4a5e772f2 | ||
|  | a542e372ab | ||
|  | 0d1bd5d62f | ||
|  | 9f3fa89f7c | ||
|  | 92995e6265 | ||
|  | a4196c3ea5 | ||
|  | db37e0c273 | ||
|  | d0aefec99a | ||
|  | 66be4b89d7 | ||
|  | 870744ce8f | ||
|  | 2ad978532b | ||
|  | 5090d93f2c | ||
|  | c8ff645766 | ||
|  | 25f7d1beba | ||
|  | 09aa111918 | ||
|  | 10fb7710e8 | ||
|  | c0ea8ebb9b | ||
|  | 31fd9c7601 | ||
|  | ddbed36455 | ||
|  | a9b0d4e1f4 | ||
|  | 4d6a3ff411 | ||
|  | 7fb993e1f4 | ||
|  | 02f502f435 | ||
|  | 4515cb43ca | ||
|  | d740333224 | ||
|  | c610f38ba9 | ||
|  | 6447353f52 | ||
|  | b46ed49996 | ||
|  | cd9fdccde0 | ||
|  | 2a8137272d | ||
|  | 762155cc90 | ||
|  | f8610ba1ca | ||
|  | c99f4098c4 | ||
|  | 3eec9fef30 | ||
|  | 8c8826176d | ||
|  | 14a2d6789f | ||
|  | 7513f298b0 | ||
|  | c04c3e334c | ||
|  | f8e51f60b3 | ||
|  | 33b066bda0 | ||
|  | 14f41bc2fb | ||
|  | 008bee0f50 | ||
|  | 29492f3332 | ||
|  | bc94bd510b | ||
|  | 9dd8e46a2d | ||
|  | 8be2bdfabd | ||
|  | b4c0806963 | ||
|  | cc38fa6cfb | ||
|  | 6de5dbafee | ||
|  | 60bf45c80d | ||
|  | eb0f3e7ec0 | ||
|  | ed553379df | ||
|  | 5c1e6f69c4 | ||
|  | 757cda0a96 | ||
|  | e94443de80 | ||
|  | 0954cd8aa4 | ||
|  | da55dac047 | ||
|  | 13a11b195f | ||
|  | 92dcba1e1c | ||
|  | 2fe1b5bd2a | ||
|  | f91e1a8739 | ||
|  | 24e21613b6 | ||
|  | c6391cd587 | ||
|  | 006ce15a0c | ||
|  | edf4216119 | ||
|  | ae8953409e | ||
|  | bda44f31a1 | ||
|  | 6621ca39a3 | ||
|  | 14f7abfa71 | ||
|  | 0f0b5736da | ||
|  | 6728187ac0 | ||
|  | 17c8675853 | ||
|  | cfbee8a431 | ||
|  | 736785ab63 | ||
|  | 3ded7bac16 | ||
|  | b524a001d6 | ||
|  | 7b071e317b | ||
|  | a380509259 | ||
|  | c0dea0a782 | ||
|  | 70947ea7b1 | ||
|  | 81cd954a51 | ||
|  | feccf29c87 | ||
|  | 5b5fbc0867 | ||
|  | f158799bbe | ||
|  | 8b0e8990c2 | ||
|  | 880ee801cf | ||
|  | 163965d861 | ||
|  | 6e218b3f9a | ||
|  | cc9b9df0b6 | ||
|  | 31f224008e | ||
|  | f32cb5cb14 | ||
|  | fec2d97ca2 | ||
|  | f2eeafb061 | ||
|  | 8f4e8bf280 | ||
|  | cc36e2295a | ||
|  | d47aeb2252 | ||
|  | 14523ed969 | ||
|  | 592e97e855 | ||
|  | 53faa3ca5f | ||
|  | c62566971f | ||
|  | 902be27cf9 | ||
|  | bf12cbe07c | ||
|  | f52e66505a | ||
|  | ca75235d3d | ||
|  | ecc6bd1341 | ||
|  | ce81b1411d | ||
|  | 7691a7a3bd | ||
|  | 214e74bf6f | ||
|  | c5826a491b | ||
|  | d8e7ef04dc | ||
|  | 08f2a92c9c | ||
|  | 3220c50f9a | ||
|  | 024ebb2706 | ||
|  | 954352c4c0 | ||
|  | 4aec95f3c9 | ||
|  | be531ef1ec | ||
|  | 65c1a750f5 | ||
|  | 5141249c59 | ||
|  | 6225984681 | ||
|  | 5cb91ceaa5 | ||
|  | 89c09e2a08 | ||
|  | fbbb219409 | ||
|  | 820b064804 | ||
|  | 355c524bfa | ||
|  | c052ce6cde | ||
|  | c9a779695d | ||
|  | a685ae511a | ||
|  | 5edea45fab | ||
|  | 8afff9f849 | ||
|  | a2043572aa | ||
|  | 5d98908b26 | ||
|  | d6fd958c5f | ||
|  | d0eb724e22 | ||
|  | afe4a8c769 | ||
|  | 9fc03aa87c | ||
|  | c798f15b98 | ||
|  | 2dcc114f84 | ||
|  | 0dfe9bc9d2 | ||
|  | 4d1cdb5bfe | ||
|  | 9c5335a027 | ||
|  | ae849ca170 | ||
|  | 94c1255782 | ||
|  | 476e1095fa | ||
|  | 8da1bb0418 | ||
|  | 01c58f8473 | ||
|  | edfcf7abe2 | ||
|  | 37b44fe7c1 | ||
|  | 8f02ad4f12 | ||
|  | 51f1244600 | ||
|  | 7bd930368c | ||
|  | fb69240ca0 | ||
|  | 830d53bfae | ||
|  | c36a959549 | ||
|  | e91b2d14e3 | ||
|  | ac58e68bc3 | ||
|  | 504c1cedfe | ||
|  | 9a4d8fae82 | ||
|  | 7d2ba6394c | ||
|  | b04b94da5f | ||
|  | 9933857f67 | ||
|  | ed5641e249 | ||
|  | a4257017ef | ||
|  | 18153f1b32 | ||
|  | 7a91d1fc43 | ||
|  | af14ded75e | ||
|  | 65939effb5 | ||
|  | 66ee7b3234 | ||
|  | cd47a628fc | ||
|  | d7c78decb0 | ||
|  | 8749477ed0 | ||
|  | 7088f5b5fa | ||
|  | 5bb6328cb9 | ||
|  | ce9f47de99 | ||
|  | 4c4780c25e | ||
|  | 64f1aba8f1 | ||
|  | 3359fb661f | ||
|  | 58a9f1b864 | ||
|  | 6ac41a4ef5 | ||
|  | aa2af7ba74 | ||
|  | ce73839fe4 | ||
|  | 1dc2726f8d | ||
|  | af76e8174d | ||
|  | 402a3efc92 | ||
|  | 372f08c990 | ||
|  | dd29eb7f81 | ||
|  | bca788ab1d | ||
|  | aef8fdba11 | ||
|  | 0a1603634b | ||
|  | a662163fd5 | ||
|  | bd7a6478a2 | ||
|  | 4a20c9f628 | ||
|  | 418c5cc3fc | ||
|  | cc55d08832 | ||
|  | de5c545648 | ||
|  | a35099bd33 | ||
|  | 5f4b5cf044 | ||
|  | beb10f843f | ||
|  | 29713e4268 | ||
|  | 8e4b83b96b | ||
|  | ae603b500e | ||
|  | d97aae7572 | ||
|  | a55e2f04a0 | ||
|  | 6e53c91608 | ||
|  | d2272fcf6e | ||
|  | c7ac5dce8c | ||
|  | 5c1d459ae9 | ||
|  | 2e7daef502 | ||
|  | 6410229681 | ||
|  | e40bd5f06b | ||
|  | 06b491eb7b | ||
|  | 3a9fadd6df | ||
|  | 0de9312a7e | ||
|  | 27fe5e3473 | ||
|  | f67dcc09f5 | ||
|  | fefc9d121d | ||
|  | a319c33d8b | ||
|  | 218d6bcc05 | ||
|  | 7d25463972 | ||
|  | aff84bec07 | ||
|  | ac651e974e | ||
|  | e21a55abcc | ||
|  | bc03228ab5 | ||
|  | f05d0e73c6 | ||
|  | aed2d4b31e | ||
|  | 184a197441 | ||
|  | ed676e8c0a | ||
|  | 8e1f937473 | ||
|  | 1a68d39211 | ||
|  | 4ba7d5b14c | ||
|  | 1a48181a9f | ||
|  | 6b70a4eb7d | ||
|  | f01855813b | ||
|  | 4a3cdf81af | ||
|  | f777397aca | ||
|  | 8fb2e5a4f5 | ||
|  | 4e8cc1e973 | ||
|  | ff02a228e3 | ||
|  | 424266abb1 | ||
|  | 3fde134791 | ||
|  | 7c39a65543 | ||
|  | 8cf70de428 | ||
|  | 15ac8413c7 | ||
|  | 79c21abba7 | ||
|  | d5c418f29f | ||
|  | 536b94e56f | ||
|  | 5c29dbd0c7 | ||
|  | ba9e68f402 | ||
|  | e9f65f8749 | ||
|  | ae0dd4b298 | ||
|  | f1ce35af1a | ||
|  | 6e617ed0b6 | ||
|  | 7cf97daf77 | ||
|  | 3d24d997ae | ||
|  | 115c281672 | ||
|  | cce23e43a9 | ||
|  | ff556f5c09 | ||
|  | 16fa01291b | ||
|  | 01534bf54f | ||
|  | cd341b6e06 | ||
|  | 185a7e25e7 | ||
|  | e81a474603 | ||
|  | ff2be6e180 | ||
|  | 3da4b31359 | ||
|  | 4bbeb19fc7 | ||
|  | a9cbab1735 | ||
|  | 6b7556a554 | ||
|  | a3c7019e06 | ||
|  | 416b9c29f7 | ||
|  | 2ec8e04cac | ||
|  | e03bfb30ce | ||
|  | f5b669113f | ||
|  | d08225edf4 | ||
|  | 8075d4f99d | ||
|  | 1a944d8a2a | ||
|  | 7cf02b6619 | ||
|  | 55cde6ef3c | ||
|  | 69c3af567d | ||
|  | 60e1fe0079 | ||
|  | 4669393070 | ||
|  | ce3bfe5d57 | ||
|  | 2a0c2ca2b8 | ||
|  | c89fbfb385 | ||
|  | facecb84a1 | ||
|  | ed06e9949b | ||
|  | e15307a612 | ||
|  | 5cbb2699ee | ||
|  | a2edf2e7ff | ||
|  | 1d31e7a2fc | ||
|  | a2a4d5fa31 | ||
|  | a28ccbabc6 | ||
|  | edd7344820 | ||
|  | c808ef81bb | ||
|  | fd203fe357 | ||
|  | 5bb7ab9928 | ||
|  | 87270c8416 | ||
|  | ebc2f7a2db | ||
|  | 7700207ec7 | ||
|  | 4d5d14f5cf | ||
|  | 72b249bf1f | ||
|  | 9b4774b21b | ||
|  | 2ddf083588 | ||
|  | 8343a03357 | ||
|  | ad320e9b83 | ||
|  | ecb750a446 | ||
|  | 5f88e02818 | ||
|  | 616af2f4b9 | ||
|  | 5a3b315b5f | ||
|  | b7a2268e7b | ||
|  | 20d729228c | ||
|  | af8c93086c | ||
|  | 79fd11ab8e | ||
|  | cb88671e37 | ||
|  | ff79552f13 | ||
|  | 643fe72717 | ||
|  | 4747e2183a | ||
|  | c59e701e35 | ||
|  | 8e678af4ba | ||
|  | 70a1165b32 | ||
|  | af14000215 | ||
|  | 998e6cdba0 | ||
|  | 2315fb5e5f | ||
|  | 157e9e5aa5 | ||
|  | c496ec0848 | ||
|  | 15b67a268a | ||
|  | 31c4809827 | ||
|  | ac0df2350a | ||
|  | 223b27f46c | ||
|  | 425142be60 | ||
|  | 7e17ec8c71 | ||
|  | 448830ce7b | ||
|  | 8896b614a9 | ||
|  | a7fce980ad | ||
|  | 91757b0f37 | ||
|  | fbfcc2972b | ||
|  | db40364b87 | ||
|  | 094ce39c45 | ||
|  | ae67d082fe | ||
|  | 8f76df7f37 | ||
|  | 5c19d18cbf | ||
|  | 838b93405b | ||
|  | 2676caf344 | ||
|  | 17941321ab | ||
|  | 48c971e073 | ||
|  | f5e2efbbf0 | ||
|  | 5d1f0e607b | ||
|  | b0872c19ea | ||
|  | 9f790b9901 | ||
|  | c41a2ec4af | ||
|  | 575dad3c98 | ||
|  | 32d687f55e | ||
|  | 93f787070f | ||
|  | f9544f6e8f | ||
|  | 336d19044c | ||
|  | 7866c9e173 | ||
|  | 1a4123de04 | ||
|  | cf2e2eb1c0 | ||
|  | 2051acdeb2 | ||
|  | cefdf970cc | ||
|  | a1d0aa7b88 | ||
|  | 49aeedb8cb | ||
|  | ef249a2cd7 | ||
|  | a09141548a | ||
|  | 5379a2d40d | ||
|  | c9450c7ab1 | ||
|  | faa1b5c292 | ||
|  | 393d9fc6d2 | ||
|  | 4e6a228689 | ||
|  | 179d6678b1 | ||
|  | 85698c5086 | ||
|  | a7d9ded45d | ||
|  | 531980d89c | ||
|  | 1887ecd4d6 | ||
|  | cd32c2caba | ||
|  | 1c9a1457fc | ||
|  | 038b0eb1da | ||
|  | f20bf146e2 | ||
|  | 01218f919b | ||
|  | 2684871bc1 | ||
|  | ccf3960eec | ||
|  | eecc0685c9 | ||
|  | 2ed849eccf | ||
|  | 3378d67a18 | ||
|  | f3c0c667a6 | ||
|  | 0ae8bbac2d | ||
|  | cbc3cfcab4 | ||
|  | b30ef07c6c | ||
|  | 73900846b1 | ||
|  | d1dc7e3991 | ||
|  | 3073a6d5e9 | ||
|  | aae53774f2 | ||
|  | 7a757b7194 | ||
|  | fa8ce26904 | ||
|  | 2c2c06e359 | ||
|  | ee580538fa | ||
|  | c3c5c31517 | ||
|  | ed9a25dd61 | ||
|  | 9ef4f12b53 | ||
|  | 84f8101606 | ||
|  | b1337948eb | ||
|  | 98f02fdde2 | ||
|  | 048fdc2292 | ||
|  | 2ca1c5aa9f | ||
|  | 674fb0fcc5 | ||
|  | 00bfe40e4d | ||
|  | cd459b1d49 | ||
|  | 92a4793b3c | ||
|  | dc03a42537 | ||
|  | 219da6bb68 | ||
|  | 0499cd866e | ||
|  | 13047f4135 | ||
|  | af69cab21d | ||
|  | d41a3fa1b4 | ||
|  | 733be371af | ||
|  | 576904bce6 | ||
|  | cf47794f09 | ||
|  | c06a9f8730 | ||
|  | 2e90dff2c2 | ||
|  | 90183a46d8 | ||
|  | b68eedba23 | ||
|  | d5b559393b | ||
|  | 1de4ac1385 | ||
|  | 39aa42ffbb | ||
|  | ec1b9577ba | ||
|  | 3b4444f99a | ||
|  | 613b2d9dc6 | ||
|  | 8f4cc22455 | ||
|  | 7c42327e0e | ||
|  | 873383e9bd | ||
|  | 8508557e77 | ||
|  | 4d1652484f | ||
|  | 88cf6fb368 | ||
|  | e7db87f700 | ||
|  | 2cb434e53e | ||
|  | cd65491c30 | ||
|  | 082b1155a3 | ||
|  | 9202b1b787 | ||
|  | a7e01c438d | ||
|  | 05be67e77d | ||
|  | 85741b9986 | ||
|  | f247a199fe | ||
|  | 29171bc2d2 | ||
|  | 7be5a62ed7 | ||
|  | 3647136f24 | ||
|  | 13598940e3 | ||
|  | 0eb365868e | ||
|  | 28c6411e49 | ||
|  | bba3fc7960 | ||
|  | fcd877013e | ||
|  | ba1d4c0488 | ||
|  | 517bcca299 | ||
|  | 1b53778175 | ||
|  | b7a0304d92 | ||
|  | 545315a985 | ||
|  | 3f4327520c | ||
|  | 4a34f69ea6 | ||
|  | fb7e68833c | ||
|  | 486dd09e0b | ||
|  | 054b99a330 | ||
|  | 65c5e044c7 | ||
|  | 11984c7467 | ||
|  | 3946864c8a | ||
|  | b84037013e | ||
|  | 1dbfc62d75 | ||
|  | d7d79106c7 | ||
|  | 1138491631 | ||
|  | 71705fa70d | ||
|  | 602814adab | ||
|  | 3a77719c5a | ||
|  | 7e195d0e92 | ||
|  | e04793401d | ||
|  | a3fbd18824 | ||
|  | c6052b8c14 | ||
|  | c792b5011f | ||
|  | 32aaeca775 | ||
|  | 1593194c63 | ||
|  | 614a7e1e23 | ||
|  | 2ebfeacabc | ||
|  | f5d8f58a17 | ||
|  | 937daef4a7 | ||
|  | dd77f14c64 | ||
|  | c36cbe5a8a | ||
|  | 41b2194f86 | ||
|  | d1e2e8f583 | ||
|  | 47fe42e1ab | ||
|  | 4c60393854 | ||
|  | f848215dfc | ||
|  | dcca581967 | ||
|  | d475b3384c | ||
|  | dd7831fe94 | ||
|  | cc08b11d16 | ||
|  | 8bba753cca | ||
|  | 43d6280d0a | ||
|  | e5a11a2293 | ||
|  | f18ef2d144 | ||
|  | 1bb5c511a5 | ||
|  | d55de57b67 | ||
|  | a2aaf4dbc6 | ||
|  | bdf6eee0ae | ||
|  | 8b910bda0c | ||
|  | 24993e3b39 | ||
|  | 11101076a1 | ||
|  | f838875726 | ||
|  | 28778d6bae | ||
|  | 1132eae56d | ||
|  | d34e79492d | ||
|  | ab205b9dc8 | ||
|  | 7dcad95d4f | ||
|  | 8a48223a7b | ||
|  | d47ae7f620 | ||
|  | 135c9c42bf | ||
|  | 0bf79ac455 | ||
|  | 98998cded6 | ||
|  | 14137b5781 | ||
|  | a172d96292 | ||
|  | 23ba76bc0e | ||
|  | 61e00a9775 | ||
|  | d1508cd68d | ||
|  | 9c85b5376d | ||
|  | 3c6f245083 | ||
|  | f207019ce5 | ||
|  | bd05aa4e24 | ||
|  | 8dc9d361c2 | ||
|  | d0e958c71c | ||
|  | a0bb7c5593 | ||
|  | 7feddd9fc7 | ||
|  | 55969016e9 | ||
|  | 9609f02e3c | ||
|  | 5c7495a194 | ||
|  | 5ee6fc974e | ||
|  | c2ebea6580 | ||
|  | 12a129ec6d | ||
|  | f28fe66970 | ||
|  | 123397317c | ||
|  | dc570c4951 | ||
|  | 22d3628319 | ||
|  | 50c9949d7a | ||
|  | 376817c6d4 | ||
|  | 63fc800057 | ||
|  | e0d0572b73 | ||
|  | 7fde87c77d | ||
|  | 938c3f65b6 | ||
|  | 2461f79d2a | ||
|  | 499bfcbfd0 | ||
|  | 07490f8017 | ||
|  | 91410c9bfa | ||
|  | a7440261c5 | ||
|  | 76c73715fb | ||
|  | c75f0b361a | ||
|  | 295df4edb9 | ||
|  | 562ceab13d | ||
|  | 2f0f6578c3 | ||
|  | 30cbd4e0d6 | ||
|  | 549e58069c | ||
|  | 7594be85ff | ||
|  | 3630034609 | ||
|  | 4e01501bbf | ||
|  | 1aa5172f56 | ||
|  | f7e2ee8fa6 | ||
|  | 66dc9a3701 | ||
|  | 31bd39256b | ||
|  | 003c69a84b | ||
|  | 0134901108 | ||
|  | eee6293d57 | ||
|  | 8237bec4f0 | ||
|  | 29cad7ad13 | ||
|  | 0d103de3b0 | ||
|  | a0090691d0 | ||
|  | 6c87c2eea8 | ||
|  | 58c2ec6ab3 | ||
|  | df5ae3eb16 | ||
|  | efda2d7854 | ||
|  | e143f5dae9 | ||
|  | 48218cdb97 | ||
|  | e9fade72f3 | ||
|  | 0f2c0d335b | ||
|  | 40b077bc7e | ||
|  | a931092cb3 | ||
|  | bd3749ed69 | ||
|  | 4ffbf77886 | ||
|  | 781a7ef60a | ||
|  | 5b2949ee0b | ||
|  | a0d646135a | ||
|  | 7862ad88b7 | ||
|  | f3bff94cf9 | ||
|  | 0eba1e1782 | ||
|  | e3216b82bf | ||
|  | da419e2332 | ||
|  | 0d97ef43be | ||
|  | 1a2313a6f2 | ||
|  | 250a9bdfe2 | ||
|  | 6317a3e9da | ||
|  | 7ab7c9e932 | ||
|  | e129c5bc0d | ||
|  | 2e241242a3 | ||
|  | 9724e5d336 | ||
|  | 63a562f95e | ||
|  | 5c340b0387 | ||
|  | 1c6510f57a | ||
|  | 2a15a98a6a | ||
|  | 72a406e7aa | ||
|  | feccc3ff37 | ||
|  | 265bfa2c79 | ||
|  | 8faf9b9b41 | ||
|  | 84be7c230c | ||
|  | 3e675fabe0 | ||
|  | cd5b4b0bc2 | ||
|  | 7ef822021b | ||
|  | 9a48926a57 | ||
|  | 13cd97f3df | ||
|  | 183139340b | ||
|  | 1c69bca258 | ||
|  | c10ea454dc | ||
|  | 9504fc21b5 | ||
|  | 13d8fbef30 | ||
|  | b8988b63a6 | ||
|  | 5eaaeb7c31 | ||
|  | c4f8c453ae | ||
|  | 6f4ba54079 | ||
|  | 637570326b | ||
|  | 37f885650c | ||
|  | c8c34ccb20 | ||
|  | e765ed3a9c | ||
|  | 677063594e | ||
|  | 59c7cbd482 | ||
|  | 570311610e | ||
|  | 41b264e77c | ||
|  | df4bd0d53f | ||
|  | 7f09a662a0 | ||
|  | 4f3b21e1c7 | ||
|  | 54233c9080 | ||
|  | db8e13ef71 | ||
|  | 5a42414b9c | ||
|  | 9c665ab72e | ||
|  | b665ba6aa6 | ||
|  | ec5913b5cd | ||
|  | 25ac63ed71 | ||
|  | 99209c2916 | ||
|  | 1fbaa0a521 | ||
|  | 3037b91e05 | ||
|  | ffdf972b91 | ||
|  | 459e5fbd5f | ||
|  | bfc993cc91 | ||
|  | 4432db35d9 | ||
|  | 591ab1dff9 | ||
|  | 5bca2424bc | ||
|  | bd61a9e770 | ||
|  | 3438e7acd2 | ||
|  | 09c200acf2 | ||
|  | 716889cab1 | ||
|  | 409693984f | ||
|  | 04e8c11080 | ||
|  | 80af2b73ab | ||
|  | 3cc57f9645 | ||
|  | a65d4e7f14 | ||
|  | b531cfc019 | ||
|  | 543ec2136b | ||
|  | 93b5071f73 | ||
|  | ddc369f073 | ||
|  | 1b40dc92eb | ||
|  | fcc3e6138b | ||
|  | 9fe6ef7ab2 | ||
|  | c010af6f19 | ||
|  | 35b7982303 | ||
|  | f311cfa231 | ||
|  | 80970e531b | ||
|  | b7bb76df05 | ||
|  | 98c70d6fc7 | ||
|  | ab84349b16 | ||
|  | 03091e372f | ||
|  | 4d17184817 | ||
|  | e086e0eb6c | ||
|  | 314368c822 | ||
|  | c5181ab410 | ||
|  | ea5152cae1 | ||
|  | 255fca5eea | ||
|  | 4aeccadf4e | ||
|  | 93540ee10e | ||
|  | 8fb3ac3649 | ||
|  | 77b2986b5b | ||
|  | 62b013df0d | ||
|  | fad6768bd1 | ||
|  | a78125f925 | ||
|  | a00a8bcc8a | ||
|  | 1e9a9e167d | ||
|  | 3da0db62e6 | ||
|  | e14ced7918 | ||
|  | ab9d02f53b | ||
|  | a461a11989 | ||
|  | 1bd838608f | ||
|  | 365577f567 | ||
|  | 50efb383f0 | ||
|  | 5da6bd0083 | ||
|  | 5e9a033e6e | ||
|  | fb7cb6823e | ||
|  | dd0a58f5f0 | ||
|  | a21420389e | ||
|  | 6140baf4e1 | ||
|  | 8fc642eb5b | ||
|  | e66e1a0046 | ||
|  | d5c69f1da4 | ||
|  | f13b1e7d7f | ||
|  | 5c8a3f862a | ||
|  | 8807f1277f | ||
|  | a3b9157f49 | ||
|  | b88ba05356 | ||
|  | b74d505577 | ||
|  | 9e2d7dca87 | ||
|  | d236b37ac9 | ||
|  | e880c66bd8 | ||
|  | 383456aa29 | ||
|  | 1a13940c8d | ||
|  | 3d54788495 | ||
|  | 71d53ace2f | ||
|  | f37e3f99f0 | ||
|  | bd03ffc16e | ||
|  | 1ac1af9b47 | ||
|  | 3bf5705316 | ||
|  | 1c2528c8a3 | ||
|  | 7bd15b1a03 | ||
|  | 6b961a85fd | ||
|  | 7707004043 | ||
|  | a025d3c5a5 | ||
|  | c460bdd56b | ||
|  | b81a359eb6 | ||
|  | d61aefb24c | ||
|  | d305dd73a3 | ||
|  | 93a16ba238 | ||
|  | 4f7cea6c53 | ||
|  | afbdd3acc3 | ||
|  | 85d5866177 | ||
|  | 9789d7535d | ||
|  | d8443cd3f7 | ||
|  | d47c26e168 | ||
|  | 01561da142 | ||
|  | 0af25f784b | ||
|  | b9b42f2ea0 | ||
|  | 311c393838 | ||
|  | 18c1c42405 | ||
|  | 37dd5d4629 | ||
|  | 81975f4693 | ||
|  | b8b928d5cb | ||
|  | 3eff81fbf7 | ||
|  | 785521bf4f | ||
|  | 6d1a55a521 | ||
|  | 9cad27008b | ||
|  | 11e611a7fa | ||
|  | 72c1f8de06 | ||
|  | 6e99868e4c | ||
|  | 4d278fde64 | ||
|  | f21e915fb9 | ||
|  | 6f53c63df6 | ||
|  | 1def5f359e | ||
|  | 15ec669374 | ||
|  | a3fa5da496 | ||
|  | 30965ac66a | ||
|  | 09ab40b7d1 | ||
|  | edab9dbf4d | ||
|  | 9868ea4936 | ||
|  | 85920dd01d | ||
|  | fa15607773 | ||
|  | a91a2c1a83 | ||
|  | 16e7711e22 | ||
|  | 5cda4eda72 | ||
|  | 98f000409f | ||
|  | bd7fe0cf66 | ||
|  | 48246541da | ||
|  | 4a8d4a53b1 | ||
|  | 4cd95bcbc3 | ||
|  | be24c8697f | ||
|  | 0d93378887 | ||
|  | 4069766c52 | ||
|  | 7010577720 | ||
|  | 8ac27a68e6 | ||
|  | 46312e0b46 | ||
|  | f9216ed6ad | ||
|  | 65bf37ef83 | ||
|  | f740fae2a4 | ||
|  | fbc503d696 | ||
|  | 662435f728 | ||
|  | 163d966707 | ||
|  | 85729c51af | ||
|  | 360e1ca5cc | ||
|  | a1f2a06b34 | ||
|  | c84dd8a90d | ||
|  | 65469a7f8b | ||
|  | 6b597516c1 | ||
|  | b5857f62e2 | ||
|  | a504ced097 | ||
|  | 1db5fbcfe3 | ||
|  | 59b8ab5834 | ||
|  | a568180441 | ||
|  | 85e80f71cd | ||
|  | bfa6bdcd8b | ||
|  | 03cd72b007 | ||
|  | 5bfd430f81 | ||
|  | 73fac4e911 | ||
|  | 8fb474fb17 | ||
|  | f813928e4b | ||
|  | b9c7a97318 | ||
|  | 9fb2f1cd6d | ||
|  | 6ca7732d5e | ||
|  | b0ab0fac49 | ||
|  | a294bce82f | ||
|  | 76d1466b08 | ||
|  | 1888d3f7b3 | ||
|  | c2787701cc | ||
|  | 52e1d0ccc4 | ||
|  | 10e3c4c221 | ||
|  | 68f2d273bf | ||
|  | 7c86c21662 | ||
|  | ae1580d790 | ||
|  | 3215c50f25 | ||
|  | 36f73e8044 | ||
|  | a4f3d779db | ||
|  | d9aa2b784d | ||
|  | cffcbc02de | ||
|  | 9347fddbfc | ||
|  | 037e9437e4 | ||
|  | 36e7a4ca2e | ||
|  | ae6423d704 | ||
|  | 7105440cec | ||
|  | c80b9cd280 | ||
|  | 171ca612af | ||
|  | c3d64fc1b3 | ||
|  | 7c24ce225d | ||
|  | 08b38d5401 | ||
|  | 024c53694d | ||
|  | 7e6011101f | ||
|  | c40feaba77 | ||
|  | 5277f09dfc | ||
|  | 2d30521ab9 | ||
|  | 050fa43561 | ||
|  | f36f92f4da | ||
|  | 124f3bc67d | ||
|  | d304209a85 | ||
|  | 8367d3f3cb | ||
|  | c56d7d899d | ||
|  | ea5db8469e | ||
|  | 3811c567e7 | ||
|  | 8708d76425 | ||
|  | 054fe3cc40 | ||
|  | af0d11f244 | ||
|  | 9650885be9 | ||
|  | 596ac6e31f | ||
|  | 612ee37365 | ||
|  | 442c37b7a9 | ||
|  | 04bbe41330 | ||
|  | 8f84f57183 | ||
|  | 6a78740211 | ||
|  | c0e1a415fd | ||
|  | bf8f082a90 | ||
|  | 2f543a2142 | ||
|  | 7e5db8c930 | ||
|  | f7a211dcc8 | ||
|  | 845734773d | ||
|  | 347de4931c | ||
|  | 8829650513 | ||
|  | c73fae1e2e | ||
|  | 834bf069d2 | ||
|  | c06a9fa34f | ||
|  | 753fad4adc | ||
|  | 34814eb66e | ||
|  | 3a5bcd0326 | ||
|  | 99c2398bc6 | ||
|  | 28f1272870 | ||
|  | f18e3a2fc0 | ||
|  | c4c5dc27cb | ||
|  | 2caf182f37 | ||
|  | 43f244b6d5 | ||
|  | 1309b396d0 | ||
|  | ba61796458 | ||
|  | 3255fe7141 | ||
|  | e98b8e79ea | 
| @@ -2,6 +2,7 @@ language: python | ||||
| python: | ||||
|   - "2.6" | ||||
|   - "2.7" | ||||
|   - "3.2" | ||||
|   - "3.3" | ||||
|   - "3.4" | ||||
| before_install: | ||||
|   | ||||
							
								
								
									
										17
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -110,3 +110,20 @@ Shaya Goldberg | ||||
| Paul Hartmann | ||||
| Frans de Jonge | ||||
| Robin de Rooij | ||||
| Ryan Schmidt | ||||
| Leslie P. Polzer | ||||
| Duncan Keall | ||||
| Alexander Mamay | ||||
| Devin J. Pohly | ||||
| Eduardo Ferro Aldama | ||||
| Jeff Buchbinder | ||||
| Amish Bhadeshia | ||||
| Joram Schrijver | ||||
| Will W. | ||||
| Mohammad Teimori Pabandi | ||||
| Roman Le Négrate | ||||
| Matthias Küch | ||||
| Julian Richen | ||||
| Ping O. | ||||
| Mister Hat | ||||
| Peter Ding | ||||
|   | ||||
| @@ -18,7 +18,9 @@ If your report is shorter than two lines, it is almost certainly missing some of | ||||
|  | ||||
| For bug reports, this means that your report should contain the *complete* output of youtube-dl when called with the -v flag. The error message you get for (most) bugs even says so, but you would not believe how many of our bug reports do not contain this information. | ||||
|  | ||||
| Site support requests **must contain an example URL**. An example URL is a URL you might want to download, like http://www.youtube.com/watch?v=BaW_jenozKc . There should be an obvious video present. Except under very special circumstances, the main page of a video service (e.g. http://www.youtube.com/ ) is *not* an example URL. | ||||
| If your server has multiple IPs or you suspect censorship, adding --call-home may be a good idea to get more diagnostics. If the error is `ERROR: Unable to extract ...` and you cannot reproduce it from multiple countries, add `--dump-pages` (warning: this will yield a rather large output, redirect it to the file `log.txt` by adding `>log.txt 2>&1` to your command-line) or upload the `.dump` files you get when you add `--write-pages` [somewhere](https://gist.github.com/). | ||||
|  | ||||
| **Site support requests must contain an example URL**. An example URL is a URL you might want to download, like http://www.youtube.com/watch?v=BaW_jenozKc . There should be an obvious video present. Except under very special circumstances, the main page of a video service (e.g. http://www.youtube.com/ ) is *not* an example URL. | ||||
|  | ||||
| ###  Are you using the latest version? | ||||
|  | ||||
|   | ||||
							
								
								
									
										5
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,7 +1,8 @@ | ||||
| all: youtube-dl README.md CONTRIBUTING.md README.txt youtube-dl.1 youtube-dl.bash-completion youtube-dl.zsh youtube-dl.fish supportedsites | ||||
|  | ||||
| clean: | ||||
| 	rm -rf youtube-dl.1.temp.md youtube-dl.1 youtube-dl.bash-completion README.txt MANIFEST build/ dist/ .coverage cover/ youtube-dl.tar.gz youtube-dl.zsh youtube-dl.fish *.dump *.part *.info.json *.mp4 *.flv *.mp3 CONTRIBUTING.md.tmp youtube-dl youtube-dl.exe | ||||
| 	rm -rf youtube-dl.1.temp.md youtube-dl.1 youtube-dl.bash-completion README.txt MANIFEST build/ dist/ .coverage cover/ youtube-dl.tar.gz youtube-dl.zsh youtube-dl.fish *.dump *.part *.info.json *.mp4 *.flv *.mp3 *.avi CONTRIBUTING.md.tmp youtube-dl youtube-dl.exe | ||||
| 	find . -name "*.pyc" -delete | ||||
|  | ||||
| PREFIX ?= /usr/local | ||||
| BINDIR ?= $(PREFIX)/bin | ||||
| @@ -43,7 +44,7 @@ test: | ||||
| ot: offlinetest | ||||
|  | ||||
| offlinetest: codetest | ||||
| 	nosetests --verbose test --exclude test_download --exclude test_age_restriction --exclude test_subtitles --exclude test_write_annotations --exclude test_youtube_lists | ||||
| 	nosetests --verbose test --exclude test_download.py --exclude test_age_restriction.py --exclude test_subtitles.py --exclude test_write_annotations.py --exclude test_youtube_lists.py | ||||
|  | ||||
| tar: youtube-dl.tar.gz | ||||
|  | ||||
|   | ||||
							
								
								
									
										531
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										531
									
								
								README.md
									
									
									
									
									
								
							| @@ -5,6 +5,7 @@ youtube-dl - download videos from youtube.com or other video platforms | ||||
| - [OPTIONS](#options) | ||||
| - [CONFIGURATION](#configuration) | ||||
| - [OUTPUT TEMPLATE](#output-template) | ||||
| - [FORMAT SELECTION](#format-selection) | ||||
| - [VIDEO SELECTION](#video-selection) | ||||
| - [FAQ](#faq) | ||||
| - [DEVELOPER INSTRUCTIONS](#developer-instructions) | ||||
| @@ -16,12 +17,12 @@ youtube-dl - download videos from youtube.com or other video platforms | ||||
| To install it right away for all UNIX users (Linux, OS X, etc.), type: | ||||
|  | ||||
|     sudo curl https://yt-dl.org/latest/youtube-dl -o /usr/local/bin/youtube-dl | ||||
|     sudo chmod a+x /usr/local/bin/youtube-dl | ||||
|     sudo chmod a+rx /usr/local/bin/youtube-dl | ||||
|  | ||||
| If you do not have curl, you can alternatively use a recent wget: | ||||
|  | ||||
|     sudo wget https://yt-dl.org/downloads/latest/youtube-dl -O /usr/local/bin/youtube-dl | ||||
|     sudo chmod a+x /usr/local/bin/youtube-dl | ||||
|     sudo chmod a+rx /usr/local/bin/youtube-dl | ||||
|  | ||||
| Windows users can [download a .exe file](https://yt-dl.org/latest/youtube-dl.exe) and place it in their home directory or any other location on their [PATH](http://en.wikipedia.org/wiki/PATH_%28variable%29). | ||||
|  | ||||
| @@ -45,344 +46,191 @@ which means you can modify it, redistribute it or use it however you like. | ||||
|     youtube-dl [OPTIONS] URL [URL...] | ||||
|  | ||||
| # OPTIONS | ||||
|     -h, --help                       print this help text and exit | ||||
|     --version                        print program version and exit | ||||
|     -U, --update                     update this program to latest version. Make | ||||
|                                      sure that you have sufficient permissions | ||||
|                                      (run with sudo if needed) | ||||
|     -i, --ignore-errors              continue on download errors, for example to | ||||
|                                      skip unavailable videos in a playlist | ||||
|     --abort-on-error                 Abort downloading of further videos (in the | ||||
|                                      playlist or the command line) if an error | ||||
|                                      occurs | ||||
|     --dump-user-agent                display the current browser identification | ||||
|     --list-extractors                List all supported extractors and the URLs | ||||
|                                      they would handle | ||||
|     --extractor-descriptions         Output descriptions of all supported | ||||
|                                      extractors | ||||
|     --default-search PREFIX          Use this prefix for unqualified URLs. For | ||||
|                                      example "gvsearch2:" downloads two videos | ||||
|                                      from google videos for  youtube-dl "large | ||||
|                                      apple". Use the value "auto" to let | ||||
|                                      youtube-dl guess ("auto_warning" to emit a | ||||
|                                      warning when guessing). "error" just throws | ||||
|                                      an error. The default value "fixup_error" | ||||
|                                      repairs broken URLs, but emits an error if | ||||
|                                      this is not possible instead of searching. | ||||
|     --ignore-config                  Do not read configuration files. When given | ||||
|                                      in the global configuration file /etc | ||||
|                                      /youtube-dl.conf: Do not read the user | ||||
|                                      configuration in ~/.config/youtube- | ||||
|                                      dl/config (%APPDATA%/youtube-dl/config.txt | ||||
|                                      on Windows) | ||||
|     --flat-playlist                  Do not extract the videos of a playlist, | ||||
|                                      only list them. | ||||
|     -h, --help                       Print this help text and exit | ||||
|     --version                        Print program version and exit | ||||
|     -U, --update                     Update this program to latest version. Make sure that you have sufficient permissions (run with sudo if needed) | ||||
|     -i, --ignore-errors              Continue on download errors, for example to skip unavailable videos in a playlist | ||||
|     --abort-on-error                 Abort downloading of further videos (in the playlist or the command line) if an error occurs | ||||
|     --dump-user-agent                Display the current browser identification | ||||
|     --list-extractors                List all supported extractors | ||||
|     --extractor-descriptions         Output descriptions of all supported extractors | ||||
|     --default-search PREFIX          Use this prefix for unqualified URLs. For example "gvsearch2:" downloads two videos from google videos for youtube-dl "large apple". | ||||
|                                      Use the value "auto" to let youtube-dl guess ("auto_warning" to emit a warning when guessing). "error" just throws an error. The | ||||
|                                      default value "fixup_error" repairs broken URLs, but emits an error if this is not possible instead of searching. | ||||
|     --ignore-config                  Do not read configuration files. When given in the global configuration file /etc/youtube-dl.conf: Do not read the user configuration | ||||
|                                      in ~/.config/youtube-dl/config (%APPDATA%/youtube-dl/config.txt on Windows) | ||||
|     --flat-playlist                  Do not extract the videos of a playlist, only list them. | ||||
|     --no-color                       Do not emit color codes in output | ||||
|  | ||||
| ## Network Options: | ||||
|     --proxy URL                      Use the specified HTTP/HTTPS proxy. Pass in | ||||
|                                      an empty string (--proxy "") for direct | ||||
|                                      connection | ||||
|     --proxy URL                      Use the specified HTTP/HTTPS proxy. Pass in an empty string (--proxy "") for direct connection | ||||
|     --socket-timeout SECONDS         Time to wait before giving up, in seconds | ||||
|     --source-address IP              Client-side IP address to bind to | ||||
|                                      (experimental) | ||||
|     -4, --force-ipv4                 Make all connections via IPv4 | ||||
|                                      (experimental) | ||||
|     -6, --force-ipv6                 Make all connections via IPv6 | ||||
|                                      (experimental) | ||||
|     --source-address IP              Client-side IP address to bind to (experimental) | ||||
|     -4, --force-ipv4                 Make all connections via IPv4 (experimental) | ||||
|     -6, --force-ipv6                 Make all connections via IPv6 (experimental) | ||||
|     --cn-verification-proxy URL      Use this proxy to verify the IP address for some Chinese sites. The default proxy specified by --proxy (or none, if the options is | ||||
|                                      not present) is used for the actual downloading. (experimental) | ||||
|  | ||||
| ## Video Selection: | ||||
|     --playlist-start NUMBER          playlist video to start at (default is 1) | ||||
|     --playlist-end NUMBER            playlist video to end at (default is last) | ||||
|     --playlist-items ITEM_SPEC       playlist video items to download. Specify | ||||
|                                      indices of the videos in the playlist | ||||
|                                      seperated by commas like: "--playlist-items | ||||
|                                      1,2,5,8" if you want to download videos | ||||
|                                      indexed 1, 2, 5, 8 in the playlist. You can | ||||
|                                      specify range: "--playlist-items | ||||
|                                      1-3,7,10-13", it will download the videos | ||||
|                                      at index 1, 2, 3, 7, 10, 11, 12 and 13. | ||||
|     --match-title REGEX              download only matching titles (regex or | ||||
|                                      caseless sub-string) | ||||
|     --reject-title REGEX             skip download for matching titles (regex or | ||||
|                                      caseless sub-string) | ||||
|     --playlist-start NUMBER          Playlist video to start at (default is 1) | ||||
|     --playlist-end NUMBER            Playlist video to end at (default is last) | ||||
|     --playlist-items ITEM_SPEC       Playlist video items to download. Specify indices of the videos in the playlist seperated by commas like: "--playlist-items 1,2,5,8" | ||||
|                                      if you want to download videos indexed 1, 2, 5, 8 in the playlist. You can specify range: "--playlist-items 1-3,7,10-13", it will | ||||
|                                      download the videos at index 1, 2, 3, 7, 10, 11, 12 and 13. | ||||
|     --match-title REGEX              Download only matching titles (regex or caseless sub-string) | ||||
|     --reject-title REGEX             Skip download for matching titles (regex or caseless sub-string) | ||||
|     --max-downloads NUMBER           Abort after downloading NUMBER files | ||||
|     --min-filesize SIZE              Do not download any videos smaller than | ||||
|                                      SIZE (e.g. 50k or 44.6m) | ||||
|     --max-filesize SIZE              Do not download any videos larger than SIZE | ||||
|                                      (e.g. 50k or 44.6m) | ||||
|     --date DATE                      download only videos uploaded in this date | ||||
|     --datebefore DATE                download only videos uploaded on or before | ||||
|                                      this date (i.e. inclusive) | ||||
|     --dateafter DATE                 download only videos uploaded on or after | ||||
|                                      this date (i.e. inclusive) | ||||
|     --min-views COUNT                Do not download any videos with less than | ||||
|                                      COUNT views | ||||
|     --max-views COUNT                Do not download any videos with more than | ||||
|                                      COUNT views | ||||
|     --no-playlist                    If the URL refers to a video and a | ||||
|                                      playlist, download only the video. | ||||
|     --age-limit YEARS                download only videos suitable for the given | ||||
|                                      age | ||||
|     --download-archive FILE          Download only videos not listed in the | ||||
|                                      archive file. Record the IDs of all | ||||
|                                      downloaded videos in it. | ||||
|     --include-ads                    Download advertisements as well | ||||
|                                      (experimental) | ||||
|     --min-filesize SIZE              Do not download any videos smaller than SIZE (e.g. 50k or 44.6m) | ||||
|     --max-filesize SIZE              Do not download any videos larger than SIZE (e.g. 50k or 44.6m) | ||||
|     --date DATE                      Download only videos uploaded in this date | ||||
|     --datebefore DATE                Download only videos uploaded on or before this date (i.e. inclusive) | ||||
|     --dateafter DATE                 Download only videos uploaded on or after this date (i.e. inclusive) | ||||
|     --min-views COUNT                Do not download any videos with less than COUNT views | ||||
|     --max-views COUNT                Do not download any videos with more than COUNT views | ||||
|     --match-filter FILTER            Generic video filter (experimental). Specify any key (see help for -o for a list of available keys) to match if the key is present, | ||||
|                                      !key to check if the key is not present,key > NUMBER (like "comment_count > 12", also works with >=, <, <=, !=, =) to compare against | ||||
|                                      a number, and & to require multiple matches. Values which are not known are excluded unless you put a question mark (?) after the | ||||
|                                      operator.For example, to only match videos that have been liked more than 100 times and disliked less than 50 times (or the dislike | ||||
|                                      functionality is not available at the given service), but who also have a description, use  --match-filter "like_count > 100 & | ||||
|                                      dislike_count <? 50 & description" . | ||||
|     --no-playlist                    Download only the video, if the URL refers to a video and a playlist. | ||||
|     --yes-playlist                   Download the playlist, if the URL refers to a video and a playlist. | ||||
|     --age-limit YEARS                Download only videos suitable for the given age | ||||
|     --download-archive FILE          Download only videos not listed in the archive file. Record the IDs of all downloaded videos in it. | ||||
|     --include-ads                    Download advertisements as well (experimental) | ||||
|  | ||||
| ## Download Options: | ||||
|     -r, --rate-limit LIMIT           maximum download rate in bytes per second | ||||
|                                      (e.g. 50K or 4.2M) | ||||
|     -R, --retries RETRIES            number of retries (default is 10), or | ||||
|                                      "infinite". | ||||
|     --buffer-size SIZE               size of download buffer (e.g. 1024 or 16K) | ||||
|                                      (default is 1024) | ||||
|     --no-resize-buffer               do not automatically adjust the buffer | ||||
|                                      size. By default, the buffer size is | ||||
|                                      automatically resized from an initial value | ||||
|                                      of SIZE. | ||||
|     -r, --rate-limit LIMIT           Maximum download rate in bytes per second (e.g. 50K or 4.2M) | ||||
|     -R, --retries RETRIES            Number of retries (default is 10), or "infinite". | ||||
|     --buffer-size SIZE               Size of download buffer (e.g. 1024 or 16K) (default is 1024) | ||||
|     --no-resize-buffer               Do not automatically adjust the buffer size. By default, the buffer size is automatically resized from an initial value of SIZE. | ||||
|     --playlist-reverse               Download playlist videos in reverse order | ||||
|     --xattr-set-filesize             (experimental) set file xattribute | ||||
|                                      ytdl.filesize with expected filesize | ||||
|     --external-downloader COMMAND    (experimental) Use the specified external | ||||
|                                      downloader. Currently supports | ||||
|                                      aria2c,curl,wget | ||||
|     --xattr-set-filesize             Set file xattribute ytdl.filesize with expected filesize (experimental) | ||||
|     --hls-prefer-native              Use the native HLS downloader instead of ffmpeg (experimental) | ||||
|     --external-downloader COMMAND    Use the specified external downloader. Currently supports aria2c,curl,wget | ||||
|     --external-downloader-args ARGS  Give these arguments to the external downloader | ||||
|  | ||||
| ## Filesystem Options: | ||||
|     -a, --batch-file FILE            file containing URLs to download ('-' for | ||||
|                                      stdin) | ||||
|     --id                             use only video ID in file name | ||||
|     -o, --output TEMPLATE            output filename template. Use %(title)s to | ||||
|                                      get the title, %(uploader)s for the | ||||
|                                      uploader name, %(uploader_id)s for the | ||||
|                                      uploader nickname if different, | ||||
|                                      %(autonumber)s to get an automatically | ||||
|                                      incremented number, %(ext)s for the | ||||
|                                      filename extension, %(format)s for the | ||||
|                                      format description (like "22 - 1280x720" or | ||||
|                                      "HD"), %(format_id)s for the unique id of | ||||
|                                      the format (like Youtube's itags: "137"), | ||||
|                                      %(upload_date)s for the upload date | ||||
|                                      (YYYYMMDD), %(extractor)s for the provider | ||||
|                                      (youtube, metacafe, etc), %(id)s for the | ||||
|                                      video id, %(playlist_title)s, | ||||
|                                      %(playlist_id)s, or %(playlist)s (=title if | ||||
|                                      present, ID otherwise) for the playlist the | ||||
|                                      video is in, %(playlist_index)s for the | ||||
|                                      position in the playlist. %(height)s and | ||||
|                                      %(width)s for the width and height of the | ||||
|                                      video format. %(resolution)s for a textual | ||||
|                                      description of the resolution of the video | ||||
|                                      format. %% for a literal percent. Use - to | ||||
|                                      output to stdout. Can also be used to | ||||
|                                      download to a different directory, for | ||||
|                                      example with -o '/my/downloads/%(uploader)s | ||||
|                                      /%(title)s-%(id)s.%(ext)s' . | ||||
|     --autonumber-size NUMBER         Specifies the number of digits in | ||||
|                                      %(autonumber)s when it is present in output | ||||
|                                      filename template or --auto-number option | ||||
|                                      is given | ||||
|     --restrict-filenames             Restrict filenames to only ASCII | ||||
|                                      characters, and avoid "&" and spaces in | ||||
|                                      filenames | ||||
|     -A, --auto-number                [deprecated; use  -o | ||||
|                                      "%(autonumber)s-%(title)s.%(ext)s" ] number | ||||
|                                      downloaded files starting from 00000 | ||||
|     -t, --title                      [deprecated] use title in file name | ||||
|                                      (default) | ||||
|     -l, --literal                    [deprecated] alias of --title | ||||
|     -w, --no-overwrites              do not overwrite files | ||||
|     -c, --continue                   force resume of partially downloaded files. | ||||
|                                      By default, youtube-dl will resume | ||||
|                                      downloads if possible. | ||||
|     --no-continue                    do not resume partially downloaded files | ||||
|                                      (restart from beginning) | ||||
|     --no-part                        do not use .part files - write directly | ||||
|                                      into output file | ||||
|     --no-mtime                       do not use the Last-modified header to set | ||||
|                                      the file modification time | ||||
|     --write-description              write video description to a .description | ||||
|                                      file | ||||
|     --write-info-json                write video metadata to a .info.json file | ||||
|     --write-annotations              write video annotations to a .annotation | ||||
|                                      file | ||||
|     --load-info FILE                 json file containing the video information | ||||
|                                      (created with the "--write-json" option) | ||||
|     --cookies FILE                   file to read cookies from and dump cookie | ||||
|                                      jar in | ||||
|     --cache-dir DIR                  Location in the filesystem where youtube-dl | ||||
|                                      can store some downloaded information | ||||
|                                      permanently. By default $XDG_CACHE_HOME | ||||
|                                      /youtube-dl or ~/.cache/youtube-dl . At the | ||||
|                                      moment, only YouTube player files (for | ||||
|                                      videos with obfuscated signatures) are | ||||
|                                      cached, but that may change. | ||||
|     -a, --batch-file FILE            File containing URLs to download ('-' for stdin) | ||||
|     --id                             Use only video ID in file name | ||||
|     -o, --output TEMPLATE            Output filename template. Use %(title)s to get the title, %(uploader)s for the uploader name, %(uploader_id)s for the uploader | ||||
|                                      nickname if different, %(autonumber)s to get an automatically incremented number, %(ext)s for the filename extension, %(format)s for | ||||
|                                      the format description (like "22 - 1280x720" or "HD"), %(format_id)s for the unique id of the format (like YouTube's itags: "137"), | ||||
|                                      %(upload_date)s for the upload date (YYYYMMDD), %(extractor)s for the provider (youtube, metacafe, etc), %(id)s for the video id, | ||||
|                                      %(playlist_title)s, %(playlist_id)s, or %(playlist)s (=title if present, ID otherwise) for the playlist the video is in, | ||||
|                                      %(playlist_index)s for the position in the playlist. %(height)s and %(width)s for the width and height of the video format. | ||||
|                                      %(resolution)s for a textual description of the resolution of the video format. %% for a literal percent. Use - to output to stdout. | ||||
|                                      Can also be used to download to a different directory, for example with -o '/my/downloads/%(uploader)s/%(title)s-%(id)s.%(ext)s' . | ||||
|     --autonumber-size NUMBER         Specify the number of digits in %(autonumber)s when it is present in output filename template or --auto-number option is given | ||||
|     --restrict-filenames             Restrict filenames to only ASCII characters, and avoid "&" and spaces in filenames | ||||
|     -A, --auto-number                [deprecated; use  -o "%(autonumber)s-%(title)s.%(ext)s" ] Number downloaded files starting from 00000 | ||||
|     -t, --title                      [deprecated] Use title in file name (default) | ||||
|     -l, --literal                    [deprecated] Alias of --title | ||||
|     -w, --no-overwrites              Do not overwrite files | ||||
|     -c, --continue                   Force resume of partially downloaded files. By default, youtube-dl will resume downloads if possible. | ||||
|     --no-continue                    Do not resume partially downloaded files (restart from beginning) | ||||
|     --no-part                        Do not use .part files - write directly into output file | ||||
|     --no-mtime                       Do not use the Last-modified header to set the file modification time | ||||
|     --write-description              Write video description to a .description file | ||||
|     --write-info-json                Write video metadata to a .info.json file | ||||
|     --write-annotations              Write video annotations to a .annotations.xml file | ||||
|     --load-info FILE                 JSON file containing the video information (created with the "--write-info-json" option) | ||||
|     --cookies FILE                   File to read cookies from and dump cookie jar in | ||||
|     --cache-dir DIR                  Location in the filesystem where youtube-dl can store some downloaded information permanently. By default $XDG_CACHE_HOME/youtube-dl | ||||
|                                      or ~/.cache/youtube-dl . At the moment, only YouTube player files (for videos with obfuscated signatures) are cached, but that may | ||||
|                                      change. | ||||
|     --no-cache-dir                   Disable filesystem caching | ||||
|     --rm-cache-dir                   Delete all filesystem cache files | ||||
|  | ||||
| ## Thumbnail images: | ||||
|     --write-thumbnail                write thumbnail image to disk | ||||
|     --write-all-thumbnails           write all thumbnail image formats to disk | ||||
|     --list-thumbnails                Simulate and list all available thumbnail | ||||
|                                      formats | ||||
|     --write-thumbnail                Write thumbnail image to disk | ||||
|     --write-all-thumbnails           Write all thumbnail image formats to disk | ||||
|     --list-thumbnails                Simulate and list all available thumbnail formats | ||||
|  | ||||
| ## Verbosity / Simulation Options: | ||||
|     -q, --quiet                      activates quiet mode | ||||
|     -q, --quiet                      Activate quiet mode | ||||
|     --no-warnings                    Ignore warnings | ||||
|     -s, --simulate                   do not download the video and do not write | ||||
|                                      anything to disk | ||||
|     --skip-download                  do not download the video | ||||
|     -g, --get-url                    simulate, quiet but print URL | ||||
|     -e, --get-title                  simulate, quiet but print title | ||||
|     --get-id                         simulate, quiet but print id | ||||
|     --get-thumbnail                  simulate, quiet but print thumbnail URL | ||||
|     --get-description                simulate, quiet but print video description | ||||
|     --get-duration                   simulate, quiet but print video length | ||||
|     --get-filename                   simulate, quiet but print output filename | ||||
|     --get-format                     simulate, quiet but print output format | ||||
|     -j, --dump-json                  simulate, quiet but print JSON information. | ||||
|                                      See --output for a description of available | ||||
|                                      keys. | ||||
|     -J, --dump-single-json           simulate, quiet but print JSON information | ||||
|                                      for each command-line argument. If the URL | ||||
|                                      refers to a playlist, dump the whole | ||||
|                                      playlist information in a single line. | ||||
|     --print-json                     Be quiet and print the video information as | ||||
|                                      JSON (video is still being downloaded). | ||||
|     --newline                        output progress bar as new lines | ||||
|     --no-progress                    do not print progress bar | ||||
|     --console-title                  display progress in console titlebar | ||||
|     -v, --verbose                    print various debugging information | ||||
|     --dump-intermediate-pages        print downloaded pages to debug problems | ||||
|                                      (very verbose) | ||||
|     --write-pages                    Write downloaded intermediary pages to | ||||
|                                      files in the current directory to debug | ||||
|                                      problems | ||||
|     -s, --simulate                   Do not download the video and do not write anything to disk | ||||
|     --skip-download                  Do not download the video | ||||
|     -g, --get-url                    Simulate, quiet but print URL | ||||
|     -e, --get-title                  Simulate, quiet but print title | ||||
|     --get-id                         Simulate, quiet but print id | ||||
|     --get-thumbnail                  Simulate, quiet but print thumbnail URL | ||||
|     --get-description                Simulate, quiet but print video description | ||||
|     --get-duration                   Simulate, quiet but print video length | ||||
|     --get-filename                   Simulate, quiet but print output filename | ||||
|     --get-format                     Simulate, quiet but print output format | ||||
|     -j, --dump-json                  Simulate, quiet but print JSON information. See --output for a description of available keys. | ||||
|     -J, --dump-single-json           Simulate, quiet but print JSON information for each command-line argument. If the URL refers to a playlist, dump the whole playlist | ||||
|                                      information in a single line. | ||||
|     --print-json                     Be quiet and print the video information as JSON (video is still being downloaded). | ||||
|     --newline                        Output progress bar as new lines | ||||
|     --no-progress                    Do not print progress bar | ||||
|     --console-title                  Display progress in console titlebar | ||||
|     -v, --verbose                    Print various debugging information | ||||
|     --dump-pages                     Print downloaded pages encoded using base64 to debug problems (very verbose) | ||||
|     --write-pages                    Write downloaded intermediary pages to files in the current directory to debug problems | ||||
|     --print-traffic                  Display sent and read HTTP traffic | ||||
|     -C, --call-home                  Contact the youtube-dl server for | ||||
|                                      debugging. | ||||
|     --no-call-home                   Do NOT contact the youtube-dl server for | ||||
|                                      debugging. | ||||
|     -C, --call-home                  Contact the youtube-dl server for debugging | ||||
|     --no-call-home                   Do NOT contact the youtube-dl server for debugging | ||||
|  | ||||
| ## Workarounds: | ||||
|     --encoding ENCODING              Force the specified encoding (experimental) | ||||
|     --no-check-certificate           Suppress HTTPS certificate validation. | ||||
|     --prefer-insecure                Use an unencrypted connection to retrieve | ||||
|                                      information about the video. (Currently | ||||
|                                      supported only for YouTube) | ||||
|     --user-agent UA                  specify a custom user agent | ||||
|     --referer URL                    specify a custom referer, use if the video | ||||
|                                      access is restricted to one domain | ||||
|     --add-header FIELD:VALUE         specify a custom HTTP header and its value, | ||||
|                                      separated by a colon ':'. You can use this | ||||
|                                      option multiple times | ||||
|     --bidi-workaround                Work around terminals that lack | ||||
|                                      bidirectional text support. Requires bidiv | ||||
|                                      or fribidi executable in PATH | ||||
|     --sleep-interval SECONDS         Number of seconds to sleep before each | ||||
|                                      download. | ||||
|     --no-check-certificate           Suppress HTTPS certificate validation | ||||
|     --prefer-insecure                Use an unencrypted connection to retrieve information about the video. (Currently supported only for YouTube) | ||||
|     --user-agent UA                  Specify a custom user agent | ||||
|     --referer URL                    Specify a custom referer, use if the video access is restricted to one domain | ||||
|     --add-header FIELD:VALUE         Specify a custom HTTP header and its value, separated by a colon ':'. You can use this option multiple times | ||||
|     --bidi-workaround                Work around terminals that lack bidirectional text support. Requires bidiv or fribidi executable in PATH | ||||
|     --sleep-interval SECONDS         Number of seconds to sleep before each download. | ||||
|  | ||||
| ## Video Format Options: | ||||
|     -f, --format FORMAT              video format code, specify the order of | ||||
|                                      preference using slashes, as in -f 22/17/18 | ||||
|                                      .  Instead of format codes, you can select | ||||
|                                      by extension for the extensions aac, m4a, | ||||
|                                      mp3, mp4, ogg, wav, webm. You can also use | ||||
|                                      the special names "best", "bestvideo", | ||||
|                                      "bestaudio", "worst".  You can filter the | ||||
|                                      video results by putting a condition in | ||||
|                                      brackets, as in -f "best[height=720]" (or | ||||
|                                      -f "[filesize>10M]").  This works for | ||||
|                                      filesize, height, width, tbr, abr, vbr, | ||||
|                                      asr, and fps and the comparisons <, <=, >, | ||||
|                                      >=, =, != and for ext, acodec, vcodec, | ||||
|                                      container, and protocol and the comparisons | ||||
|                                      =, != . Formats for which the value is not | ||||
|                                      known are excluded unless you put a | ||||
|                                      question mark (?) after the operator. You | ||||
|                                      can combine format filters, so  -f "[height | ||||
|                                      <=? 720][tbr>500]" selects up to 720p | ||||
|                                      videos (or videos where the height is not | ||||
|                                      known) with a bitrate of at least 500 | ||||
|                                      KBit/s. By default, youtube-dl will pick | ||||
|                                      the best quality. Use commas to download | ||||
|                                      multiple audio formats, such as -f | ||||
|                                      136/137/mp4/bestvideo,140/m4a/bestaudio. | ||||
|                                      You can merge the video and audio of two | ||||
|                                      formats into a single file using -f <video- | ||||
|                                      format>+<audio-format> (requires ffmpeg or | ||||
|                                      avconv), for example -f | ||||
|                                      bestvideo+bestaudio. | ||||
|     --all-formats                    download all available video formats | ||||
|     --prefer-free-formats            prefer free video formats unless a specific | ||||
|                                      one is requested | ||||
|     --max-quality FORMAT             highest quality format to download | ||||
|     -F, --list-formats               list all available formats | ||||
|     --youtube-skip-dash-manifest     Do not download the DASH manifest on | ||||
|                                      YouTube videos | ||||
|     --merge-output-format FORMAT     If a merge is required (e.g. | ||||
|                                      bestvideo+bestaudio), output to given | ||||
|                                      container format. One of mkv, mp4, ogg, | ||||
|                                      webm, flv.Ignored if no merge is required | ||||
|     -f, --format FORMAT              Video format code, see the "FORMAT SELECTION" for all the info | ||||
|     --all-formats                    Download all available video formats | ||||
|     --prefer-free-formats            Prefer free video formats unless a specific one is requested | ||||
|     -F, --list-formats               List all available formats | ||||
|     --youtube-skip-dash-manifest     Do not download the DASH manifest on YouTube videos | ||||
|     --merge-output-format FORMAT     If a merge is required (e.g. bestvideo+bestaudio), output to given container format. One of mkv, mp4, ogg, webm, flv.Ignored if no | ||||
|                                      merge is required | ||||
|  | ||||
| ## Subtitle Options: | ||||
|     --write-sub                      write subtitle file | ||||
|     --write-auto-sub                 write automatic subtitle file (youtube | ||||
|                                      only) | ||||
|     --all-subs                       downloads all the available subtitles of | ||||
|                                      the video | ||||
|     --list-subs                      lists all available subtitles for the video | ||||
|     --sub-format FORMAT              subtitle format (default=srt) ([sbv/vtt] | ||||
|                                      youtube only) | ||||
|     --sub-lang LANGS                 languages of the subtitles to download | ||||
|                                      (optional) separated by commas, use IETF | ||||
|                                      language tags like 'en,pt' | ||||
|     --write-sub                      Write subtitle file | ||||
|     --write-auto-sub                 Write automatic subtitle file (YouTube only) | ||||
|     --all-subs                       Download all the available subtitles of the video | ||||
|     --list-subs                      List all available subtitles for the video | ||||
|     --sub-format FORMAT              Subtitle format, accepts formats preference, for example: "srt" or "ass/srt/best" | ||||
|     --sub-lang LANGS                 Languages of the subtitles to download (optional) separated by commas, use IETF language tags like 'en,pt' | ||||
|  | ||||
| ## Authentication Options: | ||||
|     -u, --username USERNAME          login with this account ID | ||||
|     -p, --password PASSWORD          account password. If this option is left | ||||
|                                      out, youtube-dl will ask interactively. | ||||
|     -2, --twofactor TWOFACTOR        two-factor auth code | ||||
|     -n, --netrc                      use .netrc authentication data | ||||
|     --video-password PASSWORD        video password (vimeo, smotri) | ||||
|     -u, --username USERNAME          Login with this account ID | ||||
|     -p, --password PASSWORD          Account password. If this option is left out, youtube-dl will ask interactively. | ||||
|     -2, --twofactor TWOFACTOR        Two-factor auth code | ||||
|     -n, --netrc                      Use .netrc authentication data | ||||
|     --video-password PASSWORD        Video password (vimeo, smotri) | ||||
|  | ||||
| ## Post-processing Options: | ||||
|     -x, --extract-audio              convert video files to audio-only files | ||||
|                                      (requires ffmpeg or avconv and ffprobe or | ||||
|                                      avprobe) | ||||
|     --audio-format FORMAT            "best", "aac", "vorbis", "mp3", "m4a", | ||||
|                                      "opus", or "wav"; "best" by default | ||||
|     --audio-quality QUALITY          ffmpeg/avconv audio quality specification, | ||||
|                                      insert a value between 0 (better) and 9 | ||||
|                                      (worse) for VBR or a specific bitrate like | ||||
|                                      128K (default 5) | ||||
|     --recode-video FORMAT            Encode the video to another format if | ||||
|                                      necessary (currently supported: | ||||
|                                      mp4|flv|ogg|webm|mkv) | ||||
|     -k, --keep-video                 keeps the video file on disk after the | ||||
|                                      post-processing; the video is erased by | ||||
|                                      default | ||||
|     --no-post-overwrites             do not overwrite post-processed files; the | ||||
|                                      post-processed files are overwritten by | ||||
|                                      default | ||||
|     --embed-subs                     embed subtitles in the video (only for mp4 | ||||
|                                      videos) | ||||
|     --embed-thumbnail                embed thumbnail in the audio as cover art | ||||
|     --add-metadata                   write metadata to the video file | ||||
|     --xattrs                         write metadata to the video file's xattrs | ||||
|                                      (using dublin core and xdg standards) | ||||
|     --fixup POLICY                   Automatically correct known faults of the | ||||
|                                      file. One of never (do nothing), warn (only | ||||
|                                      emit a warning), detect_or_warn(the | ||||
|                                      default; fix file if we can, warn | ||||
|                                      otherwise) | ||||
|     --prefer-avconv                  Prefer avconv over ffmpeg for running the | ||||
|                                      postprocessors (default) | ||||
|     --prefer-ffmpeg                  Prefer ffmpeg over avconv for running the | ||||
|                                      postprocessors | ||||
|     --exec CMD                       Execute a command on the file after | ||||
|                                      downloading, similar to find's -exec | ||||
|                                      syntax. Example: --exec 'adb push {} | ||||
|                                      /sdcard/Music/ && rm {}' | ||||
|     -x, --extract-audio              Convert video files to audio-only files (requires ffmpeg or avconv and ffprobe or avprobe) | ||||
|     --audio-format FORMAT            Specify audio format: "best", "aac", "vorbis", "mp3", "m4a", "opus", or "wav"; "best" by default | ||||
|     --audio-quality QUALITY          Specify ffmpeg/avconv audio quality, insert a value between 0 (better) and 9 (worse) for VBR or a specific bitrate like 128K (default | ||||
|                                      5) | ||||
|     --recode-video FORMAT            Encode the video to another format if necessary (currently supported: mp4|flv|ogg|webm|mkv) | ||||
|     -k, --keep-video                 Keep the video file on disk after the post-processing; the video is erased by default | ||||
|     --no-post-overwrites             Do not overwrite post-processed files; the post-processed files are overwritten by default | ||||
|     --embed-subs                     Embed subtitles in the video (only for mkv and mp4 videos) | ||||
|     --embed-thumbnail                Embed thumbnail in the audio as cover art | ||||
|     --add-metadata                   Write metadata to the video file | ||||
|     --metadata-from-title FORMAT     Parse additional metadata like song title / artist from the video title. The format syntax is the same as --output, the parsed | ||||
|                                      parameters replace existing values. Additional templates: %(album)s, %(artist)s. Example: --metadata-from-title "%(artist)s - | ||||
|                                      %(title)s" matches a title like "Coldplay - Paradise" | ||||
|     --xattrs                         Write metadata to the video file's xattrs (using dublin core and xdg standards) | ||||
|     --fixup POLICY                   Automatically correct known faults of the file. One of never (do nothing), warn (only emit a warning), detect_or_warn (the default; | ||||
|                                      fix file if we can, warn otherwise) | ||||
|     --prefer-avconv                  Prefer avconv over ffmpeg for running the postprocessors (default) | ||||
|     --prefer-ffmpeg                  Prefer ffmpeg over avconv for running the postprocessors | ||||
|     --ffmpeg-location PATH           Location of the ffmpeg/avconv binary; either the path to the binary or its containing directory. | ||||
|     --exec CMD                       Execute a command on the file after downloading, similar to find's -exec syntax. Example: --exec 'adb push {} /sdcard/Music/ && rm | ||||
|                                      {}' | ||||
|     --convert-subtitles FORMAT       Convert the subtitles to other format (currently supported: srt|ass|vtt) | ||||
|  | ||||
| # CONFIGURATION | ||||
|  | ||||
| @@ -414,6 +262,17 @@ $ youtube-dl --get-filename -o "%(title)s.%(ext)s" BaW_jenozKc --restrict-filena | ||||
| youtube-dl_test_video_.mp4          # A simple file name | ||||
| ``` | ||||
|  | ||||
| # FORMAT SELECTION | ||||
|  | ||||
| By default youtube-dl tries to download the best quality, but sometimes you may want to download other format. | ||||
| The simplest case is requesting a specific format, for example `-f 22`. You can get the list of available formats using `--list-formats`, you can also use a file extension (currently it supports aac, m4a, mp3, mp4, ogg, wav, webm) or the special names `best`, `bestvideo`, `bestaudio` and `worst`. | ||||
|  | ||||
| If you want to download multiple videos and they don't have the same formats available, you can specify the order of preference using slashes, as in `-f 22/17/18`. You can also filter the video results by putting a condition in brackets, as in `-f "best[height=720]"` (or `-f "[filesize>10M]"`).  This works for filesize, height, width, tbr, abr, vbr, asr, and fps and the comparisons <, <=, >, >=, =, != and for ext, acodec, vcodec, container, and protocol and the comparisons =, != . Formats for which the value is not known are excluded unless you put a question mark (?) after the operator. You can combine format filters, so  `-f "[height <=? 720][tbr>500]"` selects up to 720p videos (or videos where the height is not known) with a bitrate of at least 500 KBit/s. Use commas to download multiple formats, such as `-f 136/137/mp4/bestvideo,140/m4a/bestaudio`. You can merge the video and audio of two formats into a single file using `-f <video-format>+<audio-format>` (requires ffmpeg or avconv), for example `-f bestvideo+bestaudio`. | ||||
|  | ||||
| Since the end of April 2015 and version 2015.04.26 youtube-dl uses `-f bestvideo+bestaudio/best` as default format selection (see #5447, #5456). If ffmpeg or avconv are installed this results in downloading `bestvideo` and `bestaudio` separately and muxing them together into a single file giving the best overall quality available. Otherwise it falls back to `best` and results in downloading best available quality served as a single file. `best` is also needed for videos that don't come from YouTube because they don't provide the audio and video in two different files. If you want to only download some dash formats (for example if you are not interested in getting videos with a resolution higher than 1080p), you can add `-f bestvideo[height<=?1080]+bestaudio/best` to your configuration file. Note that if you use youtube-dl to stream to `stdout` (and most likely to pipe it to your media player then), i.e. you explicitly specify output template as `-o -`, youtube-dl still uses `-f best` format selection in order to start content delivery immediately to your player and not to wait until `bestvideo` and `bestaudio` are downloaded and muxed. | ||||
|  | ||||
| If you want to preserve the old format selection behavior (prior to youtube-dl 2015.04.26), i.e. you want to download best available quality media served as a single file, you should explicitly specify your choice with `-f best`. You may want to add it to the [configuration file](#configuration) in order not to type it every time you run youtube-dl. | ||||
|  | ||||
| # VIDEO SELECTION | ||||
|  | ||||
| Videos can be filtered by their upload date using the options `--date`, `--datebefore` or `--dateafter`, they accept dates in two formats: | ||||
| @@ -464,9 +323,9 @@ YouTube changed their playlist format in March 2014 and later on, so you'll need | ||||
|  | ||||
| If you have installed youtube-dl with a package manager, pip, setup.py or a tarball, please use that to update. Note that Ubuntu packages do not seem to get updated anymore. Since we are not affiliated with Ubuntu, there is little we can do. Feel free to [report bugs](https://bugs.launchpad.net/ubuntu/+source/youtube-dl/+filebug) to the [Ubuntu packaging guys](mailto:ubuntu-motu@lists.ubuntu.com?subject=outdated%20version%20of%20youtube-dl) - all they have to do is update the package to a somewhat recent version. See above for a way to update. | ||||
|  | ||||
| ### Do I always have to pass in `--max-quality FORMAT`, or `-citw`? | ||||
| ### Do I always have to pass `-citw`? | ||||
|  | ||||
| By default, youtube-dl intends to have the best options (incidentally, if you have a convincing case that these should be different, [please file an issue where you explain that](https://yt-dl.org/bug)). Therefore, it is unnecessary and sometimes harmful to copy long option strings from webpages. In particular, `--max-quality` *limits* the video quality (so if you want the best quality, do NOT pass it in), and the only option out of `-citw` that is regularly useful is `-i`. | ||||
| By default, youtube-dl intends to have the best options (incidentally, if you have a convincing case that these should be different, [please file an issue where you explain that](https://yt-dl.org/bug)). Therefore, it is unnecessary and sometimes harmful to copy long option strings from webpages. In particular, the only option out of `-citw` that is regularly useful is `-i`. | ||||
|  | ||||
| ### Can you please put the -b option back? | ||||
|  | ||||
| @@ -492,11 +351,35 @@ If you want to play the video on a machine that is not running youtube-dl, you c | ||||
|  | ||||
| ### ERROR: no fmt_url_map or conn information found in video info | ||||
|  | ||||
| youtube has switched to a new video info format in July 2011 which is not supported by old versions of youtube-dl. You can update youtube-dl with `sudo youtube-dl --update`. | ||||
| YouTube has switched to a new video info format in July 2011 which is not supported by old versions of youtube-dl. See [above](#how-do-i-update-youtube-dl) for how to update youtube-dl. | ||||
|  | ||||
| ### ERROR: unable to download video ### | ||||
|  | ||||
| youtube requires an additional signature since September 2012 which is not supported by old versions of youtube-dl. You can update youtube-dl with `sudo youtube-dl --update`. | ||||
| YouTube requires an additional signature since September 2012 which is not supported by old versions of youtube-dl. See [above](#how-do-i-update-youtube-dl) for how to update youtube-dl. | ||||
|  | ||||
| ### Video URL contains an ampersand and I'm getting some strange output `[1] 2839` or `'v' is not recognized as an internal or external command` ### | ||||
|  | ||||
| That's actually the output from your shell. Since ampersand is one of the special shell characters it's interpreted by shell preventing you from passing the whole URL to youtube-dl. To disable your shell from interpreting the ampersands (or any other special characters) you have to either put the whole URL in quotes or escape them with a backslash (which approach will work depends on your shell). | ||||
|  | ||||
| For example if your URL is https://www.youtube.com/watch?t=4&v=BaW_jenozKc you should end up with following command: | ||||
|  | ||||
| ```youtube-dl 'https://www.youtube.com/watch?t=4&v=BaW_jenozKc'``` | ||||
|  | ||||
| or | ||||
|  | ||||
| ```youtube-dl https://www.youtube.com/watch?t=4\&v=BaW_jenozKc``` | ||||
|  | ||||
| For Windows you have to use the double quotes: | ||||
|  | ||||
| ```youtube-dl "https://www.youtube.com/watch?t=4&v=BaW_jenozKc"``` | ||||
|  | ||||
| ### ExtractorError: Could not find JS function u'OF' | ||||
|  | ||||
| In February 2015, the new YouTube player contained a character sequence in a string that was misinterpreted by old versions of youtube-dl. See [above](#how-do-i-update-youtube-dl) for how to update youtube-dl. | ||||
|  | ||||
| ### HTTP Error 429: Too Many Requests or 402: Payment Required | ||||
|  | ||||
| These two error codes indicate that the service is blocking your IP address because of overuse. Contact the service and ask them to unblock your IP address, or - if you have acquired a whitelisted IP address already - use the [`--proxy` or `--network-address` options](#network-options) to select another IP address. | ||||
|  | ||||
| ### SyntaxError: Non-ASCII character ### | ||||
|  | ||||
| @@ -542,9 +425,21 @@ A note on the service that they don't host the infringing content, but just link | ||||
|  | ||||
| Support requests for services that **do** purchase the rights to distribute their content are perfectly fine though. If in doubt, you can simply include a source that mentions the legitimate purchase of content. | ||||
|  | ||||
| ### How can I speed up work on my issue? | ||||
|  | ||||
| (Also known as: Help, my important issue not being solved!) The youtube-dl core developer team is quite small. While we do our best to solve as many issues as possible, sometimes that can take quite a while. To speed up your issue, here's what you can do: | ||||
|  | ||||
| First of all, please do report the issue [at our issue tracker](https://yt-dl.org/bugs). That allows us to coordinate all efforts by users and developers, and serves as a unified point. Unfortunately, the youtube-dl project has grown too large to use personal email as an effective communication channel. | ||||
|  | ||||
| Please read the [bug reporting instructions](#bugs) below. A lot of bugs lack all the necessary information. If you can, offer proxy, VPN, or shell access to the youtube-dl developers. If you are able to, test the issue from multiple computers in multiple countries to exclude local censorship or misconfiguration issues. | ||||
|  | ||||
| If nobody is interested in solving your issue, you are welcome to take matters into your own hands and submit a pull request (or coerce/pay somebody else to do so). | ||||
|  | ||||
| Feel free to bump the issue from time to time by writing a small comment ("Issue is still present in youtube-dl version ...from France, but fixed from Belgium"), but please not more than once a month. Please do not declare your issue as `important` or `urgent`. | ||||
|  | ||||
| ### How can I detect whether a given URL is supported by youtube-dl? | ||||
|  | ||||
| For one, have a look at the [list of supported sites](docs/supportedsites.md). Note that it can sometimes happen that the site changes its URL scheme (say, from http://example.com/v/1234567 to http://example.com/v/1234567 ) and youtube-dl reports an URL of a service in that list as unsupported. In that case, simply report a bug. | ||||
| For one, have a look at the [list of supported sites](docs/supportedsites.md). Note that it can sometimes happen that the site changes its URL scheme (say, from http://example.com/video/1234567 to http://example.com/v/1234567 ) and youtube-dl reports an URL of a service in that list as unsupported. In that case, simply report a bug. | ||||
|  | ||||
| It is *not* possible to detect whether a URL is supported or not. That's because youtube-dl contains a generic extractor which matches **all** URLs. You may be tempted to disable, exclude, or remove the generic extractor, but the generic extractor not only allows users to extract videos from lots of websites that embed a video from another service, but may also be used to extract video from a service that it's hosting itself. Therefore, we neither recommend nor support disabling, excluding, or removing the generic extractor. | ||||
|  | ||||
| @@ -641,6 +536,7 @@ youtube-dl makes the best effort to be a good command-line program, and thus sho | ||||
| From a Python program, you can embed youtube-dl in a more powerful fashion, like this: | ||||
|  | ||||
| ```python | ||||
| from __future__ import unicode_literals | ||||
| import youtube_dl | ||||
|  | ||||
| ydl_opts = {} | ||||
| @@ -653,6 +549,7 @@ Most likely, you'll want to use various options. For a list of what can be done, | ||||
| Here's a more complete example of a program that outputs only errors (and a short message after the download is finished), and downloads/converts the video to an mp3 file: | ||||
|  | ||||
| ```python | ||||
| from __future__ import unicode_literals | ||||
| import youtube_dl | ||||
|  | ||||
|  | ||||
| @@ -710,7 +607,9 @@ If your report is shorter than two lines, it is almost certainly missing some of | ||||
|  | ||||
| For bug reports, this means that your report should contain the *complete* output of youtube-dl when called with the -v flag. The error message you get for (most) bugs even says so, but you would not believe how many of our bug reports do not contain this information. | ||||
|  | ||||
| Site support requests **must contain an example URL**. An example URL is a URL you might want to download, like http://www.youtube.com/watch?v=BaW_jenozKc . There should be an obvious video present. Except under very special circumstances, the main page of a video service (e.g. http://www.youtube.com/ ) is *not* an example URL. | ||||
| If your server has multiple IPs or you suspect censorship, adding --call-home may be a good idea to get more diagnostics. If the error is `ERROR: Unable to extract ...` and you cannot reproduce it from multiple countries, add `--dump-pages` (warning: this will yield a rather large output, redirect it to the file `log.txt` by adding `>log.txt 2>&1` to your command-line) or upload the `.dump` files you get when you add `--write-pages` [somewhere](https://gist.github.com/). | ||||
|  | ||||
| **Site support requests must contain an example URL**. An example URL is a URL you might want to download, like http://www.youtube.com/watch?v=BaW_jenozKc . There should be an obvious video present. Except under very special circumstances, the main page of a video service (e.g. http://www.youtube.com/ ) is *not* an example URL. | ||||
|  | ||||
| ###  Are you using the latest version? | ||||
|  | ||||
|   | ||||
| @@ -28,7 +28,7 @@ for test in get_testcases(): | ||||
|     if METHOD == 'EURISTIC': | ||||
|         try: | ||||
|             webpage = compat_urllib_request.urlopen(test['url'], timeout=10).read() | ||||
|         except: | ||||
|         except Exception: | ||||
|             print('\nFail: {0}'.format(test['name'])) | ||||
|             continue | ||||
|  | ||||
| @@ -45,12 +45,12 @@ for test in get_testcases(): | ||||
|  | ||||
|         RESULT = ('.' + domain + '\n' in LIST or '\n' + domain + '\n' in LIST) | ||||
|  | ||||
|     if RESULT and ('info_dict' not in test or 'age_limit' not in test['info_dict'] | ||||
|                    or test['info_dict']['age_limit'] != 18): | ||||
|     if RESULT and ('info_dict' not in test or 'age_limit' not in test['info_dict'] or | ||||
|                    test['info_dict']['age_limit'] != 18): | ||||
|         print('\nPotential missing age_limit check: {0}'.format(test['name'])) | ||||
|  | ||||
|     elif not RESULT and ('info_dict' in test and 'age_limit' in test['info_dict'] | ||||
|                          and test['info_dict']['age_limit'] == 18): | ||||
|     elif not RESULT and ('info_dict' in test and 'age_limit' in test['info_dict'] and | ||||
|                          test['info_dict']['age_limit'] == 18): | ||||
|         print('\nPotential false negative: {0}'.format(test['name'])) | ||||
|  | ||||
|     else: | ||||
|   | ||||
							
								
								
									
										42
									
								
								devscripts/generate_aes_testdata.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								devscripts/generate_aes_testdata.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import codecs | ||||
| import subprocess | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from youtube_dl.utils import intlist_to_bytes | ||||
| from youtube_dl.aes import aes_encrypt, key_expansion | ||||
|  | ||||
| secret_msg = b'Secret message goes here' | ||||
|  | ||||
|  | ||||
| def hex_str(int_list): | ||||
|     return codecs.encode(intlist_to_bytes(int_list), 'hex') | ||||
|  | ||||
|  | ||||
| def openssl_encode(algo, key, iv): | ||||
|     cmd = ['openssl', 'enc', '-e', '-' + algo, '-K', hex_str(key), '-iv', hex_str(iv)] | ||||
|     prog = subprocess.Popen(cmd, stdin=subprocess.PIPE, stdout=subprocess.PIPE) | ||||
|     out, _ = prog.communicate(secret_msg) | ||||
|     return out | ||||
|  | ||||
| iv = key = [0x20, 0x15] + 14 * [0] | ||||
|  | ||||
| r = openssl_encode('aes-128-cbc', key, iv) | ||||
| print('aes_cbc_decrypt') | ||||
| print(repr(r)) | ||||
|  | ||||
| password = key | ||||
| new_key = aes_encrypt(password, key_expansion(password)) | ||||
| r = openssl_encode('aes-128-ctr', new_key, iv) | ||||
| print('aes_decrypt_text 16') | ||||
| print(repr(r)) | ||||
|  | ||||
| password = key + 16 * [0] | ||||
| new_key = aes_encrypt(password, key_expansion(password)) * (32 // 16) | ||||
| r = openssl_encode('aes-256-ctr', new_key, iv) | ||||
| print('aes_decrypt_text 32') | ||||
| print(repr(r)) | ||||
| @@ -1,12 +1,16 @@ | ||||
| # Supported sites | ||||
|  - **1tv**: Первый канал | ||||
|  - **1up.com** | ||||
|  - **220.ro** | ||||
|  - **22tracks:genre** | ||||
|  - **22tracks:track** | ||||
|  - **24video** | ||||
|  - **3sat** | ||||
|  - **4tube** | ||||
|  - **56.com** | ||||
|  - **5min** | ||||
|  - **8tracks** | ||||
|  - **91porn** | ||||
|  - **9gag** | ||||
|  - **abc.net.au** | ||||
|  - **Abc7News** | ||||
| @@ -16,14 +20,14 @@ | ||||
|  - **AdultSwim** | ||||
|  - **Aftenposten** | ||||
|  - **Aftonbladet** | ||||
|  - **AirMozilla** | ||||
|  - **AlJazeera** | ||||
|  - **Allocine** | ||||
|  - **AlphaPorno** | ||||
|  - **anitube.se** | ||||
|  - **AnySex** | ||||
|  - **Aparat** | ||||
|  - **AppleDailyAnimationNews** | ||||
|  - **AppleDailyRealtimeNews** | ||||
|  - **AppleDaily** | ||||
|  - **AppleTrailers** | ||||
|  - **archive.org**: archive.org videos | ||||
|  - **ARD** | ||||
| @@ -40,11 +44,13 @@ | ||||
|  - **audiomack** | ||||
|  - **audiomack:album** | ||||
|  - **Azubu** | ||||
|  - **BaiduVideo** | ||||
|  - **bambuser** | ||||
|  - **bambuser:channel** | ||||
|  - **Bandcamp** | ||||
|  - **Bandcamp:album** | ||||
|  - **bbc.co.uk**: BBC iPlayer | ||||
|  - **BeatportPro** | ||||
|  - **Beeg** | ||||
|  - **BehindKink** | ||||
|  - **Bet** | ||||
| @@ -58,16 +64,23 @@ | ||||
|  - **BR**: Bayerischer Rundfunk Mediathek | ||||
|  - **Break** | ||||
|  - **Brightcove** | ||||
|  - **bt:article**: Bergens Tidende Articles | ||||
|  - **bt:vestlendingen**: Bergens Tidende - Vestlendingen | ||||
|  - **BuzzFeed** | ||||
|  - **BYUtv** | ||||
|  - **Camdemy** | ||||
|  - **CamdemyFolder** | ||||
|  - **Canal13cl** | ||||
|  - **canalc2.tv** | ||||
|  - **Canalplus**: canalplus.fr, piwiplus.fr and d8.tv | ||||
|  - **CBS** | ||||
|  - **CBSNews**: CBS News | ||||
|  - **CBSSports** | ||||
|  - **CeskaTelevize** | ||||
|  - **channel9**: Channel 9 | ||||
|  - **Chilloutzone** | ||||
|  - **chirbit** | ||||
|  - **chirbit:profile** | ||||
|  - **Cinchcast** | ||||
|  - **Cinemassacre** | ||||
|  - **clipfish** | ||||
| @@ -88,6 +101,7 @@ | ||||
|  - **CondeNast**: Condé Nast media group: Condé Nast, GQ, Glamour, Vanity Fair, Vogue, W Magazine, WIRED | ||||
|  - **Cracked** | ||||
|  - **Criterion** | ||||
|  - **CrooksAndLiars** | ||||
|  - **Crunchyroll** | ||||
|  - **crunchyroll:playlist** | ||||
|  - **CSpan**: C-SPAN | ||||
| @@ -101,15 +115,21 @@ | ||||
|  - **DctpTv** | ||||
|  - **DeezerPlaylist** | ||||
|  - **defense.gouv.fr** | ||||
|  - **DHM**: Filmarchiv - Deutsches Historisches Museum | ||||
|  - **Discovery** | ||||
|  - **divxstage**: DivxStage | ||||
|  - **Dotsub** | ||||
|  - **DouyuTV** | ||||
|  - **dramafever** | ||||
|  - **dramafever:series** | ||||
|  - **DRBonanza** | ||||
|  - **Dropbox** | ||||
|  - **DrTuber** | ||||
|  - **DRTV** | ||||
|  - **Dump** | ||||
|  - **Dumpert** | ||||
|  - **dvtv**: http://video.aktualne.cz/ | ||||
|  - **EaglePlatform** | ||||
|  - **EbaumsWorld** | ||||
|  - **EchoMsk** | ||||
|  - **eHow** | ||||
| @@ -118,11 +138,13 @@ | ||||
|  - **EllenTV** | ||||
|  - **EllenTV:clips** | ||||
|  - **ElPais**: El País | ||||
|  - **Embedly** | ||||
|  - **EMPFlix** | ||||
|  - **Engadget** | ||||
|  - **Eporner** | ||||
|  - **EroProfile** | ||||
|  - **Escapist** | ||||
|  - **ESPN** (Currently broken) | ||||
|  - **EveryonesMixtape** | ||||
|  - **exfm**: ex.fm | ||||
|  - **ExpoTV** | ||||
| @@ -132,13 +154,14 @@ | ||||
|  - **fc2** | ||||
|  - **fernsehkritik.tv** | ||||
|  - **fernsehkritik.tv:postecke** | ||||
|  - **Firedrive** | ||||
|  - **Firstpost** | ||||
|  - **firsttv**: Видеоархив - Первый канал | ||||
|  - **FiveTV** | ||||
|  - **Flickr** | ||||
|  - **Folketinget**: Folketinget (ft.dk; Danish parliament) | ||||
|  - **FootyRoom** | ||||
|  - **Foxgay** | ||||
|  - **FoxNews** | ||||
|  - **FoxSports** | ||||
|  - **france2.fr:generation-quoi** | ||||
|  - **FranceCulture** | ||||
|  - **FranceInter** | ||||
| @@ -151,11 +174,14 @@ | ||||
|  - **Gamekings** | ||||
|  - **GameOne** | ||||
|  - **gameone:playlist** | ||||
|  - **Gamersyde** | ||||
|  - **GameSpot** | ||||
|  - **GameStar** | ||||
|  - **Gametrailers** | ||||
|  - **Gazeta** | ||||
|  - **GDCVault** | ||||
|  - **generic**: Generic downloader that works on some sites | ||||
|  - **Gfycat** | ||||
|  - **GiantBomb** | ||||
|  - **Giga** | ||||
|  - **Glide**: Glide mobile video messages (glide.me) | ||||
| @@ -163,9 +189,8 @@ | ||||
|  - **GodTube** | ||||
|  - **GoldenMoustache** | ||||
|  - **Golem** | ||||
|  - **GorillaVid**: GorillaVid.in, daclips.in, movpod.in and fastvideo.in | ||||
|  - **GorillaVid**: GorillaVid.in, daclips.in, movpod.in, fastvideo.in and realvid.net | ||||
|  - **Goshgay** | ||||
|  - **Grooveshark** | ||||
|  - **Groupon** | ||||
|  - **Hark** | ||||
|  - **HearThisAt** | ||||
| @@ -174,6 +199,7 @@ | ||||
|  - **Helsinki**: helsinki.fi | ||||
|  - **HentaiStigma** | ||||
|  - **HistoricFilms** | ||||
|  - **History** | ||||
|  - **hitbox** | ||||
|  - **hitbox:live** | ||||
|  - **HornBunny** | ||||
| @@ -187,12 +213,14 @@ | ||||
|  - **ign.com** | ||||
|  - **imdb**: Internet Movie Database trailers | ||||
|  - **imdb:list**: Internet Movie Database lists | ||||
|  - **Imgur** | ||||
|  - **Ina** | ||||
|  - **InfoQ** | ||||
|  - **Instagram** | ||||
|  - **instagram:user**: Instagram user profile | ||||
|  - **InternetVideoArchive** | ||||
|  - **IPrima** | ||||
|  - **iqiyi** | ||||
|  - **ivi**: ivi.ru | ||||
|  - **ivi:compilation**: ivi.ru compilations | ||||
|  - **Izlesene** | ||||
| @@ -201,8 +229,11 @@ | ||||
|  - **Jove** | ||||
|  - **jpopsuki.tv** | ||||
|  - **Jukebox** | ||||
|  - **Kaltura** | ||||
|  - **KanalPlay**: Kanal 5/9/11 Play | ||||
|  - **Kankan** | ||||
|  - **Karaoketv** | ||||
|  - **KarriereVideos** | ||||
|  - **keek** | ||||
|  - **KeezMovies** | ||||
|  - **KhanAcademy** | ||||
| @@ -212,6 +243,11 @@ | ||||
|  - **Ku6** | ||||
|  - **la7.tv** | ||||
|  - **Laola1Tv** | ||||
|  - **Letv** | ||||
|  - **LetvPlaylist** | ||||
|  - **LetvTv** | ||||
|  - **Libsyn** | ||||
|  - **life:embed** | ||||
|  - **lifenews**: LIFE | NEWS | ||||
|  - **LiveLeak** | ||||
|  - **livestream** | ||||
| @@ -225,11 +261,14 @@ | ||||
|  - **mailru**: Видео@Mail.Ru | ||||
|  - **Malemotion** | ||||
|  - **MDR** | ||||
|  - **media.ccc.de** | ||||
|  - **MegaVideoz** | ||||
|  - **metacafe** | ||||
|  - **Metacritic** | ||||
|  - **Mgoon** | ||||
|  - **Minhateca** | ||||
|  - **MinistryGrid** | ||||
|  - **miomio.tv** | ||||
|  - **mitele.es** | ||||
|  - **mixcloud** | ||||
|  - **MLB** | ||||
| @@ -257,11 +296,15 @@ | ||||
|  - **MySpass** | ||||
|  - **myvideo** | ||||
|  - **MyVidster** | ||||
|  - **N-JOY** | ||||
|  - **n-tv.de** | ||||
|  - **NationalGeographic** | ||||
|  - **Naver** | ||||
|  - **NBA** | ||||
|  - **NBC** | ||||
|  - **NBCNews** | ||||
|  - **NBCSports** | ||||
|  - **NBCSportsVPlayer** | ||||
|  - **ndr**: NDR.de - Mediathek | ||||
|  - **NDTV** | ||||
|  - **NerdCubedFeed** | ||||
| @@ -281,47 +324,69 @@ | ||||
|  - **Noco** | ||||
|  - **Normalboots** | ||||
|  - **NosVideo** | ||||
|  - **Nova**: TN.cz, Prásk.tv, Nova.cz, Novaplus.cz, FANDA.tv, Krásná.cz and Doma.cz | ||||
|  - **novamov**: NovaMov | ||||
|  - **Nowness** | ||||
|  - **NowTV** | ||||
|  - **nowvideo**: NowVideo | ||||
|  - **npo.nl** | ||||
|  - **npo.nl:live** | ||||
|  - **npo.nl:radio** | ||||
|  - **npo.nl:radio:fragment** | ||||
|  - **NRK** | ||||
|  - **NRKPlaylist** | ||||
|  - **NRKTV** | ||||
|  - **ntv.ru** | ||||
|  - **Nuvid** | ||||
|  - **NYTimes** | ||||
|  - **NYTimesArticle** | ||||
|  - **ocw.mit.edu** | ||||
|  - **Odnoklassniki** | ||||
|  - **OktoberfestTV** | ||||
|  - **on.aol.com** | ||||
|  - **Ooyala** | ||||
|  - **OoyalaExternal** | ||||
|  - **OpenFilm** | ||||
|  - **orf:fm4**: radio FM4 | ||||
|  - **orf:iptv**: iptv.ORF.at | ||||
|  - **orf:oe1**: Radio Österreich 1 | ||||
|  - **orf:tvthek**: ORF TVthek | ||||
|  - **parliamentlive.tv**: UK parliament videos | ||||
|  - **Patreon** | ||||
|  - **PBS** | ||||
|  - **PhilharmonieDeParis**: Philharmonie de Paris | ||||
|  - **Phoenix** | ||||
|  - **Photobucket** | ||||
|  - **Pladform** | ||||
|  - **PlanetaPlay** | ||||
|  - **play.fm** | ||||
|  - **played.to** | ||||
|  - **Playvid** | ||||
|  - **Playwire** | ||||
|  - **plus.google**: Google Plus | ||||
|  - **pluzz.francetv.fr** | ||||
|  - **podomatic** | ||||
|  - **PornHd** | ||||
|  - **PornHub** | ||||
|  - **PornHubPlaylist** | ||||
|  - **Pornotube** | ||||
|  - **PornoVoisines** | ||||
|  - **PornoXO** | ||||
|  - **PrimeShareTV** | ||||
|  - **PromptFile** | ||||
|  - **prosiebensat1**: ProSiebenSat.1 Digital | ||||
|  - **Puls4** | ||||
|  - **Pyvideo** | ||||
|  - **qqmusic** | ||||
|  - **qqmusic:album** | ||||
|  - **qqmusic:singer** | ||||
|  - **qqmusic:toplist** | ||||
|  - **QuickVid** | ||||
|  - **R7** | ||||
|  - **radio.de** | ||||
|  - **radiobremen** | ||||
|  - **radiofrance** | ||||
|  - **RadioJavan** | ||||
|  - **Rai** | ||||
|  - **RBMARadio** | ||||
|  - **RedTube** | ||||
| @@ -332,12 +397,12 @@ | ||||
|  - **Roxwel** | ||||
|  - **RTBF** | ||||
|  - **Rte** | ||||
|  - **rtl.nl**: rtl.nl and rtlxl.nl | ||||
|  - **RTL2** | ||||
|  - **RTLnow** | ||||
|  - **rtlxl.nl** | ||||
|  - **RTP** | ||||
|  - **RTS**: RTS.ch | ||||
|  - **rtve.es:alacarta**: RTVE a la carta | ||||
|  - **rtve.es:infantil**: RTVE infantil | ||||
|  - **rtve.es:live**: RTVE.es live streams | ||||
|  - **RUHD** | ||||
|  - **rutube**: Rutube videos | ||||
| @@ -346,6 +411,10 @@ | ||||
|  - **rutube:movie**: Rutube movies | ||||
|  - **rutube:person**: Rutube person videos | ||||
|  - **RUTV**: RUTV.RU | ||||
|  - **Ruutu** | ||||
|  - **safari**: safaribooksonline.com online video | ||||
|  - **safari:course**: safaribooksonline.com online courses | ||||
|  - **Sandia**: Sandia National Laboratories | ||||
|  - **Sapo**: SAPO Vídeos | ||||
|  - **savefrom.net** | ||||
|  - **SBS**: sbs.com.au | ||||
| @@ -354,6 +423,7 @@ | ||||
|  - **Screencast** | ||||
|  - **ScreencastOMatic** | ||||
|  - **ScreenwaveMedia** | ||||
|  - **SenateISVP** | ||||
|  - **ServingSys** | ||||
|  - **Sexu** | ||||
|  - **SexyKarma**: Sexy Karma and Watch Indian Porn | ||||
| @@ -367,16 +437,22 @@ | ||||
|  - **smotri:community**: Smotri.com community videos | ||||
|  - **smotri:user**: Smotri.com user videos | ||||
|  - **Snotr** | ||||
|  - **Sockshare** | ||||
|  - **Sohu** | ||||
|  - **soompi** | ||||
|  - **soompi:show** | ||||
|  - **soundcloud** | ||||
|  - **soundcloud:playlist** | ||||
|  - **soundcloud:set** | ||||
|  - **soundcloud:user** | ||||
|  - **Soundgasm** | ||||
|  - **soundgasm** | ||||
|  - **soundgasm:profile** | ||||
|  - **southpark.cc.com** | ||||
|  - **southpark.cc.com:español** | ||||
|  - **southpark.de** | ||||
|  - **southpark.nl** | ||||
|  - **southparkstudios.dk** | ||||
|  - **Space** | ||||
|  - **SpankBang** | ||||
|  - **Spankwire** | ||||
|  - **Spiegel** | ||||
|  - **Spiegel:Article**: Articles on spiegel.de | ||||
| @@ -384,14 +460,19 @@ | ||||
|  - **Spike** | ||||
|  - **Sport5** | ||||
|  - **SportBox** | ||||
|  - **SportBoxEmbed** | ||||
|  - **SportDeutschland** | ||||
|  - **Srf** | ||||
|  - **SRMediathek**: Saarländischer Rundfunk | ||||
|  - **SSA** | ||||
|  - **stanfordoc**: Stanford Open ClassRoom | ||||
|  - **Steam** | ||||
|  - **streamcloud.eu** | ||||
|  - **StreamCZ** | ||||
|  - **StreetVoice** | ||||
|  - **SunPorno** | ||||
|  - **SVT** | ||||
|  - **SVTPlay**: SVT Play and Öppet arkiv | ||||
|  - **SWRMediathek** | ||||
|  - **Syfy** | ||||
|  - **SztvHu** | ||||
| @@ -405,7 +486,7 @@ | ||||
|  - **TeamFour** | ||||
|  - **TechTalks** | ||||
|  - **techtv.mit.edu** | ||||
|  - **TED** | ||||
|  - **ted** | ||||
|  - **tegenlicht.vpro.nl** | ||||
|  - **TeleBruxelles** | ||||
|  - **telecinco.es** | ||||
| @@ -424,6 +505,7 @@ | ||||
|  - **tlc.com** | ||||
|  - **tlc.de** | ||||
|  - **TMZ** | ||||
|  - **TMZArticle** | ||||
|  - **TNAFlix** | ||||
|  - **tou.tv** | ||||
|  - **Toypics**: Toypics user profile | ||||
| @@ -432,12 +514,18 @@ | ||||
|  - **Trilulilu** | ||||
|  - **TruTube** | ||||
|  - **Tube8** | ||||
|  - **TubiTv** | ||||
|  - **Tudou** | ||||
|  - **Tumblr** | ||||
|  - **TuneIn** | ||||
|  - **Turbo** | ||||
|  - **Tutv** | ||||
|  - **tv.dfb.de** | ||||
|  - **TV2** | ||||
|  - **TV2Article** | ||||
|  - **TV4**: tv4.se and tv4play.se | ||||
|  - **TVC** | ||||
|  - **TVCArticle** | ||||
|  - **tvigle**: Интернет-телевидение Tvigle.ru | ||||
|  - **tvp.pl** | ||||
|  - **tvp.pl:Series** | ||||
| @@ -453,16 +541,20 @@ | ||||
|  - **Ubu** | ||||
|  - **udemy** | ||||
|  - **udemy:course** | ||||
|  - **UDNEmbed** | ||||
|  - **Ultimedia** | ||||
|  - **Unistra** | ||||
|  - **Urort**: NRK P3 Urørt | ||||
|  - **ustream** | ||||
|  - **ustream:channel** | ||||
|  - **Varzesh3** | ||||
|  - **Vbox7** | ||||
|  - **VeeHD** | ||||
|  - **Veoh** | ||||
|  - **Vessel** | ||||
|  - **Vesti**: Вести.Ru | ||||
|  - **Vevo** | ||||
|  - **VGTV** | ||||
|  - **VGTV**: VGTV and BTTV | ||||
|  - **vh1.com** | ||||
|  - **Vice** | ||||
|  - **Viddler** | ||||
| @@ -480,7 +572,9 @@ | ||||
|  - **Vidzi** | ||||
|  - **vier** | ||||
|  - **vier:videos** | ||||
|  - **Viewster** | ||||
|  - **viki** | ||||
|  - **viki:channel** | ||||
|  - **vimeo** | ||||
|  - **vimeo:album** | ||||
|  - **vimeo:channel** | ||||
| @@ -489,12 +583,13 @@ | ||||
|  - **vimeo:review**: Review pages on vimeo | ||||
|  - **vimeo:user** | ||||
|  - **vimeo:watchlater**: Vimeo watch later list, "vimeowatchlater" keyword (requires authentication) | ||||
|  - **Vimple**: Vimple.ru | ||||
|  - **Vimple**: Vimple - one-click video hosting | ||||
|  - **Vine** | ||||
|  - **vine:user** | ||||
|  - **vk.com** | ||||
|  - **vk.com:user-videos**: vk.com:All of a user's videos | ||||
|  - **Vodlocker** | ||||
|  - **VoiceRepublic** | ||||
|  - **Vporn** | ||||
|  - **VRT** | ||||
|  - **vube**: Vube.com | ||||
| @@ -519,12 +614,17 @@ | ||||
|  - **XHamster** | ||||
|  - **XMinus** | ||||
|  - **XNXX** | ||||
|  - **Xstream** | ||||
|  - **XTube** | ||||
|  - **XTubeUser**: XTube user profile | ||||
|  - **Xuite** | ||||
|  - **XVideos** | ||||
|  - **XXXYMovies** | ||||
|  - **Yahoo**: Yahoo screen and movies | ||||
|  - **Yam** | ||||
|  - **yandexmusic:album**: Яндекс.Музыка - Альбом | ||||
|  - **yandexmusic:playlist**: Яндекс.Музыка - Плейлист | ||||
|  - **yandexmusic:track**: Яндекс.Музыка - Трек | ||||
|  - **YesJapan** | ||||
|  - **Ynet** | ||||
|  - **YouJizz** | ||||
| @@ -543,7 +643,8 @@ | ||||
|  - **youtube:show**: YouTube.com (multi-season) shows | ||||
|  - **youtube:subscriptions**: YouTube.com subscriptions feed, "ytsubs" keyword (requires authentication) | ||||
|  - **youtube:user**: YouTube.com user videos (URL or "ytuser" keyword) | ||||
|  - **youtube:watch_later**: Youtube watch later list, ":ytwatchlater" for short (requires authentication) | ||||
|  - **youtube:watchlater**: Youtube watch later list, ":ytwatchlater" for short (requires authentication) | ||||
|  - **Zapiks** | ||||
|  - **ZDF** | ||||
|  - **ZDFChannel** | ||||
|  - **zingmp3:album**: mp3.zing.vn albums | ||||
|   | ||||
| @@ -3,4 +3,4 @@ universal = True | ||||
|  | ||||
| [flake8] | ||||
| exclude = youtube_dl/extractor/__init__.py,devscripts/buildserver.py,setup.py,build,.git | ||||
| ignore = E501 | ||||
| ignore = E402,E501,E731 | ||||
|   | ||||
| @@ -113,6 +113,16 @@ def expect_info_dict(self, got_dict, expected_dict): | ||||
|             self.assertTrue( | ||||
|                 got.startswith(start_str), | ||||
|                 'field %s (value: %r) should start with %r' % (info_field, got, start_str)) | ||||
|         elif isinstance(expected, compat_str) and expected.startswith('contains:'): | ||||
|             got = got_dict.get(info_field) | ||||
|             contains_str = expected[len('contains:'):] | ||||
|             self.assertTrue( | ||||
|                 isinstance(got, compat_str), | ||||
|                 'Expected a %s object, but got %s for field %s' % ( | ||||
|                     compat_str.__name__, type(got).__name__, info_field)) | ||||
|             self.assertTrue( | ||||
|                 contains_str in got, | ||||
|                 'field %s (value: %r) should contain %r' % (info_field, got, contains_str)) | ||||
|         elif isinstance(expected, type): | ||||
|             got = got_dict.get(info_field) | ||||
|             self.assertTrue(isinstance(got, expected), | ||||
| @@ -140,7 +150,7 @@ def expect_info_dict(self, got_dict, expected_dict): | ||||
|                              'invalid value for field %s, expected %r, got %r' % (info_field, expected, got)) | ||||
|  | ||||
|     # Check for the presence of mandatory fields | ||||
|     if got_dict.get('_type') != 'playlist': | ||||
|     if got_dict.get('_type') not in ('playlist', 'multi_video'): | ||||
|         for key in ('id', 'url', 'title', 'ext'): | ||||
|             self.assertTrue(got_dict.get(key), 'Missing mandatory field %s' % key) | ||||
|     # Check for mandatory fields that are automatically set by YoutubeDL | ||||
| @@ -163,12 +173,14 @@ def expect_info_dict(self, got_dict, expected_dict): | ||||
|             info_dict_str += ''.join( | ||||
|                 '    %s: %s,\n' % (_repr(k), _repr(v)) | ||||
|                 for k, v in test_info_dict.items() if k not in missing_keys) | ||||
|             info_dict_str += '\n' | ||||
|  | ||||
|             if info_dict_str: | ||||
|                 info_dict_str += '\n' | ||||
|         info_dict_str += ''.join( | ||||
|             '    %s: %s,\n' % (_repr(k), _repr(test_info_dict[k])) | ||||
|             for k in missing_keys) | ||||
|         write_string( | ||||
|             '\n\'info_dict\': {\n' + info_dict_str + '}\n', out=sys.stderr) | ||||
|             '\n\'info_dict\': {\n' + info_dict_str + '},\n', out=sys.stderr) | ||||
|         self.assertFalse( | ||||
|             missing_keys, | ||||
|             'Missing keys in test definition: %s' % ( | ||||
|   | ||||
| @@ -7,8 +7,7 @@ | ||||
|     "forcethumbnail": false,  | ||||
|     "forcetitle": false,  | ||||
|     "forceurl": false,  | ||||
|     "format": null,  | ||||
|     "format_limit": null,  | ||||
|     "format": "best", | ||||
|     "ignoreerrors": false,  | ||||
|     "listformats": null,  | ||||
|     "logtostderr": false,  | ||||
| @@ -28,7 +27,7 @@ | ||||
|     "retries": 10,  | ||||
|     "simulate": false,  | ||||
|     "subtitleslang": null,  | ||||
|     "subtitlesformat": "srt", | ||||
|     "subtitlesformat": "best", | ||||
|     "test": true,  | ||||
|     "updatetime": true,  | ||||
|     "usenetrc": false,  | ||||
| @@ -39,5 +38,6 @@ | ||||
|     "writesubtitles": false, | ||||
|     "allsubtitles": false, | ||||
|     "listssubtitles": false, | ||||
|     "socket_timeout": 20 | ||||
|     "socket_timeout": 20, | ||||
|     "fixup": "never" | ||||
| } | ||||
|   | ||||
| @@ -12,8 +12,12 @@ import copy | ||||
|  | ||||
| from test.helper import FakeYDL, assertRegexpMatches | ||||
| from youtube_dl import YoutubeDL | ||||
| from youtube_dl.compat import compat_str | ||||
| from youtube_dl.extractor import YoutubeIE | ||||
| from youtube_dl.postprocessor.common import PostProcessor | ||||
| from youtube_dl.utils import match_filter_func | ||||
|  | ||||
| TEST_URL = 'http://localhost/sample.mp4' | ||||
|  | ||||
|  | ||||
| class YDL(FakeYDL): | ||||
| @@ -46,8 +50,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = True | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 460, 'url': 'x'}, | ||||
|             {'ext': 'mp4', 'height': 460, 'url': 'y'}, | ||||
|             {'ext': 'webm', 'height': 460, 'url': TEST_URL}, | ||||
|             {'ext': 'mp4', 'height': 460, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -60,8 +64,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = True | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 720, 'url': 'a'}, | ||||
|             {'ext': 'mp4', 'height': 1080, 'url': 'b'}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': TEST_URL}, | ||||
|             {'ext': 'mp4', 'height': 1080, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -74,9 +78,9 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = False | ||||
|         formats = [ | ||||
|             {'ext': 'webm', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'mp4', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'flv', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': TEST_URL}, | ||||
|             {'ext': 'mp4', 'height': 720, 'url': TEST_URL}, | ||||
|             {'ext': 'flv', 'height': 720, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -88,8 +92,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         ydl = YDL() | ||||
|         ydl.params['prefer_free_formats'] = False | ||||
|         formats = [ | ||||
|             {'ext': 'flv', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': '_'}, | ||||
|             {'ext': 'flv', 'height': 720, 'url': TEST_URL}, | ||||
|             {'ext': 'webm', 'height': 720, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict['formats'] = formats | ||||
|         yie = YoutubeIE(ydl) | ||||
| @@ -98,45 +102,12 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['ext'], 'flv') | ||||
|  | ||||
|     def test_format_limit(self): | ||||
|         formats = [ | ||||
|             {'format_id': 'meh', 'url': 'http://example.com/meh', 'preference': 1}, | ||||
|             {'format_id': 'good', 'url': 'http://example.com/good', 'preference': 2}, | ||||
|             {'format_id': 'great', 'url': 'http://example.com/great', 'preference': 3}, | ||||
|             {'format_id': 'excellent', 'url': 'http://example.com/exc', 'preference': 4}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
|         ydl = YDL() | ||||
|         ydl.process_ie_result(info_dict) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], 'excellent') | ||||
|  | ||||
|         ydl = YDL({'format_limit': 'good'}) | ||||
|         assert ydl.params['format_limit'] == 'good' | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], 'good') | ||||
|  | ||||
|         ydl = YDL({'format_limit': 'great', 'format': 'all'}) | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
|         self.assertEqual(ydl.downloaded_info_dicts[0]['format_id'], 'meh') | ||||
|         self.assertEqual(ydl.downloaded_info_dicts[1]['format_id'], 'good') | ||||
|         self.assertEqual(ydl.downloaded_info_dicts[2]['format_id'], 'great') | ||||
|         self.assertTrue('3' in ydl.msgs[0]) | ||||
|  | ||||
|         ydl = YDL() | ||||
|         ydl.params['format_limit'] = 'excellent' | ||||
|         ydl.process_ie_result(info_dict.copy()) | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], 'excellent') | ||||
|  | ||||
|     def test_format_selection(self): | ||||
|         formats = [ | ||||
|             {'format_id': '35', 'ext': 'mp4', 'preference': 1, 'url': '_'}, | ||||
|             {'format_id': '45', 'ext': 'webm', 'preference': 2, 'url': '_'}, | ||||
|             {'format_id': '47', 'ext': 'webm', 'preference': 3, 'url': '_'}, | ||||
|             {'format_id': '2', 'ext': 'flv', 'preference': 4, 'url': '_'}, | ||||
|             {'format_id': '35', 'ext': 'mp4', 'preference': 1, 'url': TEST_URL}, | ||||
|             {'format_id': '45', 'ext': 'webm', 'preference': 2, 'url': TEST_URL}, | ||||
|             {'format_id': '47', 'ext': 'webm', 'preference': 3, 'url': TEST_URL}, | ||||
|             {'format_id': '2', 'ext': 'flv', 'preference': 4, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
| @@ -167,10 +138,10 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_selection_audio(self): | ||||
|         formats = [ | ||||
|             {'format_id': 'audio-low', 'ext': 'webm', 'preference': 1, 'vcodec': 'none', 'url': '_'}, | ||||
|             {'format_id': 'audio-mid', 'ext': 'webm', 'preference': 2, 'vcodec': 'none', 'url': '_'}, | ||||
|             {'format_id': 'audio-high', 'ext': 'flv', 'preference': 3, 'vcodec': 'none', 'url': '_'}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 4, 'url': '_'}, | ||||
|             {'format_id': 'audio-low', 'ext': 'webm', 'preference': 1, 'vcodec': 'none', 'url': TEST_URL}, | ||||
|             {'format_id': 'audio-mid', 'ext': 'webm', 'preference': 2, 'vcodec': 'none', 'url': TEST_URL}, | ||||
|             {'format_id': 'audio-high', 'ext': 'flv', 'preference': 3, 'vcodec': 'none', 'url': TEST_URL}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 4, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
| @@ -185,8 +156,8 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         self.assertEqual(downloaded['format_id'], 'audio-low') | ||||
|  | ||||
|         formats = [ | ||||
|             {'format_id': 'vid-low', 'ext': 'mp4', 'preference': 1, 'url': '_'}, | ||||
|             {'format_id': 'vid-high', 'ext': 'mp4', 'preference': 2, 'url': '_'}, | ||||
|             {'format_id': 'vid-low', 'ext': 'mp4', 'preference': 1, 'url': TEST_URL}, | ||||
|             {'format_id': 'vid-high', 'ext': 'mp4', 'preference': 2, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
| @@ -228,9 +199,9 @@ class TestFormatSelection(unittest.TestCase): | ||||
|  | ||||
|     def test_format_selection_video(self): | ||||
|         formats = [ | ||||
|             {'format_id': 'dash-video-low', 'ext': 'mp4', 'preference': 1, 'acodec': 'none', 'url': '_'}, | ||||
|             {'format_id': 'dash-video-high', 'ext': 'mp4', 'preference': 2, 'acodec': 'none', 'url': '_'}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 3, 'url': '_'}, | ||||
|             {'format_id': 'dash-video-low', 'ext': 'mp4', 'preference': 1, 'acodec': 'none', 'url': TEST_URL}, | ||||
|             {'format_id': 'dash-video-high', 'ext': 'mp4', 'preference': 2, 'acodec': 'none', 'url': TEST_URL}, | ||||
|             {'format_id': 'vid', 'ext': 'mp4', 'preference': 3, 'url': TEST_URL}, | ||||
|         ] | ||||
|         info_dict = _make_result(formats) | ||||
|  | ||||
| @@ -267,7 +238,7 @@ class TestFormatSelection(unittest.TestCase): | ||||
|             f2['url'] = 'url:' + f2id | ||||
|  | ||||
|             info_dict = _make_result([f1, f2], extractor='youtube') | ||||
|             ydl = YDL() | ||||
|             ydl = YDL({'format': 'best/bestvideo'}) | ||||
|             yie = YoutubeIE(ydl) | ||||
|             yie._sort_formats(info_dict['formats']) | ||||
|             ydl.process_ie_result(info_dict) | ||||
| @@ -275,7 +246,7 @@ class TestFormatSelection(unittest.TestCase): | ||||
|             self.assertEqual(downloaded['format_id'], f1id) | ||||
|  | ||||
|             info_dict = _make_result([f2, f1], extractor='youtube') | ||||
|             ydl = YDL() | ||||
|             ydl = YDL({'format': 'best/bestvideo'}) | ||||
|             yie = YoutubeIE(ydl) | ||||
|             yie._sort_formats(info_dict['formats']) | ||||
|             ydl.process_ie_result(info_dict) | ||||
| @@ -337,6 +308,67 @@ class TestFormatSelection(unittest.TestCase): | ||||
|         downloaded = ydl.downloaded_info_dicts[0] | ||||
|         self.assertEqual(downloaded['format_id'], 'G') | ||||
|  | ||||
|  | ||||
| class TestYoutubeDL(unittest.TestCase): | ||||
|     def test_subtitles(self): | ||||
|         def s_formats(lang, autocaption=False): | ||||
|             return [{ | ||||
|                 'ext': ext, | ||||
|                 'url': 'http://localhost/video.%s.%s' % (lang, ext), | ||||
|                 '_auto': autocaption, | ||||
|             } for ext in ['vtt', 'srt', 'ass']] | ||||
|         subtitles = dict((l, s_formats(l)) for l in ['en', 'fr', 'es']) | ||||
|         auto_captions = dict((l, s_formats(l, True)) for l in ['it', 'pt', 'es']) | ||||
|         info_dict = { | ||||
|             'id': 'test', | ||||
|             'title': 'Test', | ||||
|             'url': 'http://localhost/video.mp4', | ||||
|             'subtitles': subtitles, | ||||
|             'automatic_captions': auto_captions, | ||||
|             'extractor': 'TEST', | ||||
|         } | ||||
|  | ||||
|         def get_info(params={}): | ||||
|             params.setdefault('simulate', True) | ||||
|             ydl = YDL(params) | ||||
|             ydl.report_warning = lambda *args, **kargs: None | ||||
|             return ydl.process_video_result(info_dict, download=False) | ||||
|  | ||||
|         result = get_info() | ||||
|         self.assertFalse(result.get('requested_subtitles')) | ||||
|         self.assertEqual(result['subtitles'], subtitles) | ||||
|         self.assertEqual(result['automatic_captions'], auto_captions) | ||||
|  | ||||
|         result = get_info({'writesubtitles': True}) | ||||
|         subs = result['requested_subtitles'] | ||||
|         self.assertTrue(subs) | ||||
|         self.assertEqual(set(subs.keys()), set(['en'])) | ||||
|         self.assertTrue(subs['en'].get('data') is None) | ||||
|         self.assertEqual(subs['en']['ext'], 'ass') | ||||
|  | ||||
|         result = get_info({'writesubtitles': True, 'subtitlesformat': 'foo/srt'}) | ||||
|         subs = result['requested_subtitles'] | ||||
|         self.assertEqual(subs['en']['ext'], 'srt') | ||||
|  | ||||
|         result = get_info({'writesubtitles': True, 'subtitleslangs': ['es', 'fr', 'it']}) | ||||
|         subs = result['requested_subtitles'] | ||||
|         self.assertTrue(subs) | ||||
|         self.assertEqual(set(subs.keys()), set(['es', 'fr'])) | ||||
|  | ||||
|         result = get_info({'writesubtitles': True, 'writeautomaticsub': True, 'subtitleslangs': ['es', 'pt']}) | ||||
|         subs = result['requested_subtitles'] | ||||
|         self.assertTrue(subs) | ||||
|         self.assertEqual(set(subs.keys()), set(['es', 'pt'])) | ||||
|         self.assertFalse(subs['es']['_auto']) | ||||
|         self.assertTrue(subs['pt']['_auto']) | ||||
|  | ||||
|         result = get_info({'writeautomaticsub': True, 'subtitleslangs': ['es', 'pt']}) | ||||
|         subs = result['requested_subtitles'] | ||||
|         self.assertTrue(subs) | ||||
|         self.assertEqual(set(subs.keys()), set(['es', 'pt'])) | ||||
|         self.assertTrue(subs['es']['_auto']) | ||||
|         self.assertTrue(subs['pt']['_auto']) | ||||
|  | ||||
|     def test_add_extra_info(self): | ||||
|         test_dict = { | ||||
|             'extractor': 'Foo', | ||||
| @@ -379,27 +411,148 @@ class TestFormatSelection(unittest.TestCase): | ||||
|             def run(self, info): | ||||
|                 with open(audiofile, 'wt') as f: | ||||
|                     f.write('EXAMPLE') | ||||
|                 info['filepath'] | ||||
|                 return False, info | ||||
|                 return [info['filepath']], info | ||||
|  | ||||
|         def run_pp(params): | ||||
|         def run_pp(params, PP): | ||||
|             with open(filename, 'wt') as f: | ||||
|                 f.write('EXAMPLE') | ||||
|             ydl = YoutubeDL(params) | ||||
|             ydl.add_post_processor(SimplePP()) | ||||
|             ydl.add_post_processor(PP()) | ||||
|             ydl.post_process(filename, {'filepath': filename}) | ||||
|  | ||||
|         run_pp({'keepvideo': True}) | ||||
|         run_pp({'keepvideo': True}, SimplePP) | ||||
|         self.assertTrue(os.path.exists(filename), '%s doesn\'t exist' % filename) | ||||
|         self.assertTrue(os.path.exists(audiofile), '%s doesn\'t exist' % audiofile) | ||||
|         os.unlink(filename) | ||||
|         os.unlink(audiofile) | ||||
|  | ||||
|         run_pp({'keepvideo': False}) | ||||
|         run_pp({'keepvideo': False}, SimplePP) | ||||
|         self.assertFalse(os.path.exists(filename), '%s exists' % filename) | ||||
|         self.assertTrue(os.path.exists(audiofile), '%s doesn\'t exist' % audiofile) | ||||
|         os.unlink(audiofile) | ||||
|  | ||||
|         class ModifierPP(PostProcessor): | ||||
|             def run(self, info): | ||||
|                 with open(info['filepath'], 'wt') as f: | ||||
|                     f.write('MODIFIED') | ||||
|                 return [], info | ||||
|  | ||||
|         run_pp({'keepvideo': False}, ModifierPP) | ||||
|         self.assertTrue(os.path.exists(filename), '%s doesn\'t exist' % filename) | ||||
|         os.unlink(filename) | ||||
|  | ||||
|     def test_match_filter(self): | ||||
|         class FilterYDL(YDL): | ||||
|             def __init__(self, *args, **kwargs): | ||||
|                 super(FilterYDL, self).__init__(*args, **kwargs) | ||||
|                 self.params['simulate'] = True | ||||
|  | ||||
|             def process_info(self, info_dict): | ||||
|                 super(YDL, self).process_info(info_dict) | ||||
|  | ||||
|             def _match_entry(self, info_dict, incomplete): | ||||
|                 res = super(FilterYDL, self)._match_entry(info_dict, incomplete) | ||||
|                 if res is None: | ||||
|                     self.downloaded_info_dicts.append(info_dict) | ||||
|                 return res | ||||
|  | ||||
|         first = { | ||||
|             'id': '1', | ||||
|             'url': TEST_URL, | ||||
|             'title': 'one', | ||||
|             'extractor': 'TEST', | ||||
|             'duration': 30, | ||||
|             'filesize': 10 * 1024, | ||||
|         } | ||||
|         second = { | ||||
|             'id': '2', | ||||
|             'url': TEST_URL, | ||||
|             'title': 'two', | ||||
|             'extractor': 'TEST', | ||||
|             'duration': 10, | ||||
|             'description': 'foo', | ||||
|             'filesize': 5 * 1024, | ||||
|         } | ||||
|         videos = [first, second] | ||||
|  | ||||
|         def get_videos(filter_=None): | ||||
|             ydl = FilterYDL({'match_filter': filter_}) | ||||
|             for v in videos: | ||||
|                 ydl.process_ie_result(v, download=True) | ||||
|             return [v['id'] for v in ydl.downloaded_info_dicts] | ||||
|  | ||||
|         res = get_videos() | ||||
|         self.assertEqual(res, ['1', '2']) | ||||
|  | ||||
|         def f(v): | ||||
|             if v['id'] == '1': | ||||
|                 return None | ||||
|             else: | ||||
|                 return 'Video id is not 1' | ||||
|         res = get_videos(f) | ||||
|         self.assertEqual(res, ['1']) | ||||
|  | ||||
|         f = match_filter_func('duration < 30') | ||||
|         res = get_videos(f) | ||||
|         self.assertEqual(res, ['2']) | ||||
|  | ||||
|         f = match_filter_func('description = foo') | ||||
|         res = get_videos(f) | ||||
|         self.assertEqual(res, ['2']) | ||||
|  | ||||
|         f = match_filter_func('description =? foo') | ||||
|         res = get_videos(f) | ||||
|         self.assertEqual(res, ['1', '2']) | ||||
|  | ||||
|         f = match_filter_func('filesize > 5KiB') | ||||
|         res = get_videos(f) | ||||
|         self.assertEqual(res, ['1']) | ||||
|  | ||||
|     def test_playlist_items_selection(self): | ||||
|         entries = [{ | ||||
|             'id': compat_str(i), | ||||
|             'title': compat_str(i), | ||||
|             'url': TEST_URL, | ||||
|         } for i in range(1, 5)] | ||||
|         playlist = { | ||||
|             '_type': 'playlist', | ||||
|             'id': 'test', | ||||
|             'entries': entries, | ||||
|             'extractor': 'test:playlist', | ||||
|             'extractor_key': 'test:playlist', | ||||
|             'webpage_url': 'http://example.com', | ||||
|         } | ||||
|  | ||||
|         def get_ids(params): | ||||
|             ydl = YDL(params) | ||||
|             # make a copy because the dictionary can be modified | ||||
|             ydl.process_ie_result(playlist.copy()) | ||||
|             return [int(v['id']) for v in ydl.downloaded_info_dicts] | ||||
|  | ||||
|         result = get_ids({}) | ||||
|         self.assertEqual(result, [1, 2, 3, 4]) | ||||
|  | ||||
|         result = get_ids({'playlistend': 10}) | ||||
|         self.assertEqual(result, [1, 2, 3, 4]) | ||||
|  | ||||
|         result = get_ids({'playlistend': 2}) | ||||
|         self.assertEqual(result, [1, 2]) | ||||
|  | ||||
|         result = get_ids({'playliststart': 10}) | ||||
|         self.assertEqual(result, []) | ||||
|  | ||||
|         result = get_ids({'playliststart': 2}) | ||||
|         self.assertEqual(result, [2, 3, 4]) | ||||
|  | ||||
|         result = get_ids({'playlist_items': '2-4'}) | ||||
|         self.assertEqual(result, [2, 3, 4]) | ||||
|  | ||||
|         result = get_ids({'playlist_items': '2,4'}) | ||||
|         self.assertEqual(result, [2, 4]) | ||||
|  | ||||
|         result = get_ids({'playlist_items': '10'}) | ||||
|         self.assertEqual(result, []) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
							
								
								
									
										55
									
								
								test/test_aes.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								test/test_aes.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from youtube_dl.aes import aes_decrypt, aes_encrypt, aes_cbc_decrypt, aes_decrypt_text | ||||
| from youtube_dl.utils import bytes_to_intlist, intlist_to_bytes | ||||
| import base64 | ||||
|  | ||||
| # the encrypted data can be generate with 'devscripts/generate_aes_testdata.py' | ||||
|  | ||||
|  | ||||
| class TestAES(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.key = self.iv = [0x20, 0x15] + 14 * [0] | ||||
|         self.secret_msg = b'Secret message goes here' | ||||
|  | ||||
|     def test_encrypt(self): | ||||
|         msg = b'message' | ||||
|         key = list(range(16)) | ||||
|         encrypted = aes_encrypt(bytes_to_intlist(msg), key) | ||||
|         decrypted = intlist_to_bytes(aes_decrypt(encrypted, key)) | ||||
|         self.assertEqual(decrypted, msg) | ||||
|  | ||||
|     def test_cbc_decrypt(self): | ||||
|         data = bytes_to_intlist( | ||||
|             b"\x97\x92+\xe5\x0b\xc3\x18\x91ky9m&\xb3\xb5@\xe6'\xc2\x96.\xc8u\x88\xab9-[\x9e|\xf1\xcd" | ||||
|         ) | ||||
|         decrypted = intlist_to_bytes(aes_cbc_decrypt(data, self.key, self.iv)) | ||||
|         self.assertEqual(decrypted.rstrip(b'\x08'), self.secret_msg) | ||||
|  | ||||
|     def test_decrypt_text(self): | ||||
|         password = intlist_to_bytes(self.key).decode('utf-8') | ||||
|         encrypted = base64.b64encode( | ||||
|             intlist_to_bytes(self.iv[:8]) + | ||||
|             b'\x17\x15\x93\xab\x8d\x80V\xcdV\xe0\t\xcdo\xc2\xa5\xd8ksM\r\xe27N\xae' | ||||
|         ).decode('utf-8') | ||||
|         decrypted = (aes_decrypt_text(encrypted, password, 16)) | ||||
|         self.assertEqual(decrypted, self.secret_msg) | ||||
|  | ||||
|         password = intlist_to_bytes(self.key).decode('utf-8') | ||||
|         encrypted = base64.b64encode( | ||||
|             intlist_to_bytes(self.iv[:8]) + | ||||
|             b'\x0b\xe6\xa4\xd9z\x0e\xb8\xb9\xd0\xd4i_\x85\x1d\x99\x98_\xe5\x80\xe7.\xbf\xa5\x83' | ||||
|         ).decode('utf-8') | ||||
|         decrypted = (aes_decrypt_text(encrypted, password, 32)) | ||||
|         self.assertEqual(decrypted, self.secret_msg) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
| @@ -59,7 +59,7 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|         self.assertMatch('www.youtube.com/NASAgovVideo/videos', ['youtube:user']) | ||||
|  | ||||
|     def test_youtube_feeds(self): | ||||
|         self.assertMatch('https://www.youtube.com/feed/watch_later', ['youtube:watch_later']) | ||||
|         self.assertMatch('https://www.youtube.com/feed/watch_later', ['youtube:watchlater']) | ||||
|         self.assertMatch('https://www.youtube.com/feed/subscriptions', ['youtube:subscriptions']) | ||||
|         self.assertMatch('https://www.youtube.com/feed/recommended', ['youtube:recommended']) | ||||
|         self.assertMatch('https://www.youtube.com/my_favorites', ['youtube:favorites']) | ||||
| @@ -104,11 +104,11 @@ class TestAllURLsMatching(unittest.TestCase): | ||||
|         self.assertMatch(':tds', ['ComedyCentralShows']) | ||||
|  | ||||
|     def test_vimeo_matching(self): | ||||
|         self.assertMatch('http://vimeo.com/channels/tributes', ['vimeo:channel']) | ||||
|         self.assertMatch('http://vimeo.com/channels/31259', ['vimeo:channel']) | ||||
|         self.assertMatch('http://vimeo.com/channels/31259/53576664', ['vimeo']) | ||||
|         self.assertMatch('http://vimeo.com/user7108434', ['vimeo:user']) | ||||
|         self.assertMatch('http://vimeo.com/user7108434/videos', ['vimeo:user']) | ||||
|         self.assertMatch('https://vimeo.com/channels/tributes', ['vimeo:channel']) | ||||
|         self.assertMatch('https://vimeo.com/channels/31259', ['vimeo:channel']) | ||||
|         self.assertMatch('https://vimeo.com/channels/31259/53576664', ['vimeo']) | ||||
|         self.assertMatch('https://vimeo.com/user7108434', ['vimeo:user']) | ||||
|         self.assertMatch('https://vimeo.com/user7108434/videos', ['vimeo:user']) | ||||
|         self.assertMatch('https://vimeo.com/user21297594/review/75524534/3c257a1b5d', ['vimeo:review']) | ||||
|  | ||||
|     # https://github.com/rg3/youtube-dl/issues/1930 | ||||
|   | ||||
| @@ -153,7 +153,7 @@ def generator(test_case): | ||||
|                     break | ||||
|  | ||||
|             if is_playlist: | ||||
|                 self.assertEqual(res_dict['_type'], 'playlist') | ||||
|                 self.assertTrue(res_dict['_type'] in ['playlist', 'multi_video']) | ||||
|                 self.assertTrue('entries' in res_dict) | ||||
|                 expect_info_dict(self, res_dict, test_case.get('info_dict', {})) | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,6 @@ | ||||
| #!/usr/bin/env python | ||||
| # coding: utf-8 | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import unittest | ||||
| @@ -6,6 +8,9 @@ import unittest | ||||
| import sys | ||||
| import os | ||||
| import subprocess | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from youtube_dl.utils import encodeArgument | ||||
|  | ||||
| rootDir = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) | ||||
|  | ||||
| @@ -27,5 +32,12 @@ class TestExecution(unittest.TestCase): | ||||
|     def test_main_exec(self): | ||||
|         subprocess.check_call([sys.executable, 'youtube_dl/__main__.py', '--version'], cwd=rootDir, stdout=_DEV_NULL) | ||||
|  | ||||
|     def test_cmdline_umlauts(self): | ||||
|         p = subprocess.Popen( | ||||
|             [sys.executable, 'youtube_dl/__main__.py', encodeArgument('ä'), '--version'], | ||||
|             cwd=rootDir, stdout=_DEV_NULL, stderr=subprocess.PIPE) | ||||
|         _, stderr = p.communicate() | ||||
|         self.assertFalse(stderr) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -8,7 +8,7 @@ import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from youtube_dl import YoutubeDL | ||||
| from youtube_dl.compat import compat_http_server | ||||
| from youtube_dl.compat import compat_http_server, compat_urllib_request | ||||
| import ssl | ||||
| import threading | ||||
|  | ||||
| @@ -68,5 +68,52 @@ class TestHTTP(unittest.TestCase): | ||||
|         r = ydl.extract_info('https://localhost:%d/video.html' % self.port) | ||||
|         self.assertEqual(r['url'], 'https://localhost:%d/vid.mp4' % self.port) | ||||
|  | ||||
|  | ||||
| def _build_proxy_handler(name): | ||||
|     class HTTPTestRequestHandler(compat_http_server.BaseHTTPRequestHandler): | ||||
|         proxy_name = name | ||||
|  | ||||
|         def log_message(self, format, *args): | ||||
|             pass | ||||
|  | ||||
|         def do_GET(self): | ||||
|             self.send_response(200) | ||||
|             self.send_header('Content-Type', 'text/plain; charset=utf-8') | ||||
|             self.end_headers() | ||||
|             self.wfile.write('{self.proxy_name}: {self.path}'.format(self=self).encode('utf-8')) | ||||
|     return HTTPTestRequestHandler | ||||
|  | ||||
|  | ||||
| class TestProxy(unittest.TestCase): | ||||
|     def setUp(self): | ||||
|         self.proxy = compat_http_server.HTTPServer( | ||||
|             ('localhost', 0), _build_proxy_handler('normal')) | ||||
|         self.port = self.proxy.socket.getsockname()[1] | ||||
|         self.proxy_thread = threading.Thread(target=self.proxy.serve_forever) | ||||
|         self.proxy_thread.daemon = True | ||||
|         self.proxy_thread.start() | ||||
|  | ||||
|         self.cn_proxy = compat_http_server.HTTPServer( | ||||
|             ('localhost', 0), _build_proxy_handler('cn')) | ||||
|         self.cn_port = self.cn_proxy.socket.getsockname()[1] | ||||
|         self.cn_proxy_thread = threading.Thread(target=self.cn_proxy.serve_forever) | ||||
|         self.cn_proxy_thread.daemon = True | ||||
|         self.cn_proxy_thread.start() | ||||
|  | ||||
|     def test_proxy(self): | ||||
|         cn_proxy = 'localhost:{0}'.format(self.cn_port) | ||||
|         ydl = YoutubeDL({ | ||||
|             'proxy': 'localhost:{0}'.format(self.port), | ||||
|             'cn_verification_proxy': cn_proxy, | ||||
|         }) | ||||
|         url = 'http://foo.com/bar' | ||||
|         response = ydl.urlopen(url).read().decode('utf-8') | ||||
|         self.assertEqual(response, 'normal: {0}'.format(url)) | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Ytdl-request-proxy', cn_proxy) | ||||
|         response = ydl.urlopen(req).read().decode('utf-8') | ||||
|         self.assertEqual(response, 'cn: {0}'.format(url)) | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -70,6 +70,8 @@ class TestJSInterpreter(unittest.TestCase): | ||||
|         self.assertEqual(jsi.call_function('f'), -11) | ||||
|  | ||||
|     def test_comments(self): | ||||
|         'Skipping: Not yet fully implemented' | ||||
|         return | ||||
|         jsi = JSInterpreter(''' | ||||
|         function x() { | ||||
|             var x = /* 1 + */ 2; | ||||
| @@ -80,6 +82,15 @@ class TestJSInterpreter(unittest.TestCase): | ||||
|         ''') | ||||
|         self.assertEqual(jsi.call_function('x'), 52) | ||||
|  | ||||
|         jsi = JSInterpreter(''' | ||||
|         function f() { | ||||
|             var x = "/*"; | ||||
|             var y = 1 /* comment */ + 2; | ||||
|             return y; | ||||
|         } | ||||
|         ''') | ||||
|         self.assertEqual(jsi.call_function('f'), 3) | ||||
|  | ||||
|     def test_precedence(self): | ||||
|         jsi = JSInterpreter(''' | ||||
|         function x() { | ||||
|   | ||||
							
								
								
									
										26
									
								
								test/test_netrc.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								test/test_netrc.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
|  | ||||
| from youtube_dl.extractor import ( | ||||
|     gen_extractors, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class TestNetRc(unittest.TestCase): | ||||
|     def test_netrc_present(self): | ||||
|         for ie in gen_extractors(): | ||||
|             if not hasattr(ie, '_login'): | ||||
|                 continue | ||||
|             self.assertTrue( | ||||
|                 hasattr(ie, '_NETRC_MACHINE'), | ||||
|                 'Extractor %s supports login, but is missing a _NETRC_MACHINE property' % ie.IE_NAME) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
							
								
								
									
										17
									
								
								test/test_postprocessors.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								test/test_postprocessors.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| #!/usr/bin/env python | ||||
|  | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| # Allow direct execution | ||||
| import os | ||||
| import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
| from youtube_dl.postprocessor import MetadataFromTitlePP | ||||
|  | ||||
|  | ||||
| class TestMetadataFromTitle(unittest.TestCase): | ||||
|     def test_format_to_regex(self): | ||||
|         pp = MetadataFromTitlePP(None, '%(title)s - %(artist)s') | ||||
|         self.assertEqual(pp._titleregex, '(?P<title>.+)\ \-\ (?P<artist>.+)') | ||||
| @@ -18,6 +18,15 @@ from youtube_dl.extractor import ( | ||||
|     VimeoIE, | ||||
|     WallaIE, | ||||
|     CeskaTelevizeIE, | ||||
|     LyndaIE, | ||||
|     NPOIE, | ||||
|     ComedyCentralIE, | ||||
|     NRKTVIE, | ||||
|     RaiIE, | ||||
|     VikiIE, | ||||
|     ThePlatformIE, | ||||
|     RTVEALaCartaIE, | ||||
|     FunnyOrDieIE, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -27,42 +36,38 @@ class BaseTestSubtitles(unittest.TestCase): | ||||
|  | ||||
|     def setUp(self): | ||||
|         self.DL = FakeYDL() | ||||
|         self.ie = self.IE(self.DL) | ||||
|         self.ie = self.IE() | ||||
|         self.DL.add_info_extractor(self.ie) | ||||
|  | ||||
|     def getInfoDict(self): | ||||
|         info_dict = self.ie.extract(self.url) | ||||
|         info_dict = self.DL.extract_info(self.url, download=False) | ||||
|         return info_dict | ||||
|  | ||||
|     def getSubtitles(self): | ||||
|         info_dict = self.getInfoDict() | ||||
|         return info_dict['subtitles'] | ||||
|         subtitles = info_dict['requested_subtitles'] | ||||
|         if not subtitles: | ||||
|             return subtitles | ||||
|         for sub_info in subtitles.values(): | ||||
|             if sub_info.get('data') is None: | ||||
|                 uf = self.DL.urlopen(sub_info['url']) | ||||
|                 sub_info['data'] = uf.read().decode('utf-8') | ||||
|         return dict((l, sub_info['data']) for l, sub_info in subtitles.items()) | ||||
|  | ||||
|  | ||||
| class TestYoutubeSubtitles(BaseTestSubtitles): | ||||
|     url = 'QRS8MkLhQmM' | ||||
|     IE = YoutubeIE | ||||
|  | ||||
|     def test_youtube_no_writesubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = False | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_youtube_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260') | ||||
|  | ||||
|     def test_youtube_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['it'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d') | ||||
|  | ||||
|     def test_youtube_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 13) | ||||
|         self.assertEqual(md5(subtitles['en']), '4cd9278a35ba2305f47354ee13472260') | ||||
|         self.assertEqual(md5(subtitles['it']), '164a51f16f260476a05b50fe4c2f161d') | ||||
|         for lang in ['it', 'fr', 'de']: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
|     def test_youtube_subtitles_sbv_format(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
| @@ -76,12 +81,6 @@ class TestYoutubeSubtitles(BaseTestSubtitles): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '3cb210999d3e021bd6c7f0ea751eab06') | ||||
|  | ||||
|     def test_youtube_list_subtitles(self): | ||||
|         self.DL.expect_warning('Video doesn\'t have automatic captions') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_youtube_automatic_captions(self): | ||||
|         self.url = '8YoUxe5ncPo' | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
| @@ -103,55 +102,22 @@ class TestYoutubeSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_youtube_multiple_langs(self): | ||||
|         self.url = 'QRS8MkLhQmM' | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['it', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|         self.assertFalse(subtitles) | ||||
|  | ||||
|  | ||||
| class TestDailymotionSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.dailymotion.com/video/xczg00' | ||||
|     IE = DailymotionIE | ||||
|  | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f') | ||||
|  | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792') | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles.keys()), 5) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|         self.assertTrue(len(subtitles.keys()) >= 6) | ||||
|         self.assertEqual(md5(subtitles['en']), '976553874490cba125086bbfea3ff76f') | ||||
|         self.assertEqual(md5(subtitles['fr']), '594564ec7d588942e384e920e5341792') | ||||
|         for lang in ['es', 'fr', 'de']: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
|     def test_nosubtitles(self): | ||||
|         self.DL.expect_warning('video doesn\'t have subtitles') | ||||
| @@ -159,61 +125,21 @@ class TestDailymotionSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|         self.assertFalse(subtitles) | ||||
|  | ||||
|  | ||||
| class TestTedSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.ted.com/talks/dan_dennett_on_our_consciousness.html' | ||||
|     IE = TEDIE | ||||
|  | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '4262c1665ff928a2dada178f62cb8d14') | ||||
|  | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), '66a63f7f42c97a50f8c0e90bc7797bb5') | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) >= 28) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|  | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|         self.assertEqual(md5(subtitles['en']), '4262c1665ff928a2dada178f62cb8d14') | ||||
|         self.assertEqual(md5(subtitles['fr']), '66a63f7f42c97a50f8c0e90bc7797bb5') | ||||
|         for lang in ['es', 'fr', 'de']: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|  | ||||
|  | ||||
| @@ -221,14 +147,7 @@ class TestBlipTVSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://blip.tv/a/a-6603250' | ||||
|     IE = BlipTVIE | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
| @@ -240,39 +159,13 @@ class TestVimeoSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://vimeo.com/76979871' | ||||
|     IE = VimeoIE | ||||
|  | ||||
|     def test_no_writesubtitles(self): | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(subtitles, None) | ||||
|  | ||||
|     def test_subtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['en']), '26399116d23ae3cf2c087cea94bc43b4') | ||||
|  | ||||
|     def test_subtitles_lang(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['subtitleslangs'] = ['fr'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(md5(subtitles['fr']), 'b6191146a6c5d3a452244d853fde6dc8') | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['de', 'en', 'es', 'fr'])) | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_automatic_captions(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writeautomaticsub'] = True | ||||
|         self.DL.params['subtitleslang'] = ['en'] | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertTrue(len(subtitles.keys()) == 0) | ||||
|         self.assertEqual(md5(subtitles['en']), '8062383cf4dec168fc40a088aa6d5888') | ||||
|         self.assertEqual(md5(subtitles['fr']), 'b6191146a6c5d3a452244d853fde6dc8') | ||||
|  | ||||
|     def test_nosubtitles(self): | ||||
|         self.DL.expect_warning('video doesn\'t have subtitles') | ||||
| @@ -280,27 +173,13 @@ class TestVimeoSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|  | ||||
|     def test_multiple_langs(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         langs = ['es', 'fr', 'de'] | ||||
|         self.DL.params['subtitleslangs'] = langs | ||||
|         subtitles = self.getSubtitles() | ||||
|         for lang in langs: | ||||
|             self.assertTrue(subtitles.get(lang) is not None, 'Subtitles for \'%s\' not extracted' % lang) | ||||
|         self.assertFalse(subtitles) | ||||
|  | ||||
|  | ||||
| class TestWallaSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://vod.walla.co.il/movie/2705958/the-yes-men' | ||||
|     IE = WallaIE | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writesubtitles'] = True | ||||
| @@ -315,26 +194,20 @@ class TestWallaSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|         self.assertFalse(subtitles) | ||||
|  | ||||
|  | ||||
| class TestCeskaTelevizeSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.ceskatelevize.cz/ivysilani/10600540290-u6-uzasny-svet-techniky' | ||||
|     IE = CeskaTelevizeIE | ||||
|  | ||||
|     def test_list_subtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['listsubtitles'] = True | ||||
|         info_dict = self.getInfoDict() | ||||
|         self.assertEqual(info_dict, None) | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.expect_warning('Automatic Captions not supported by this server') | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['cs'])) | ||||
|         self.assertEqual(md5(subtitles['cs']), '9bf52d9549533c32c427e264bf0847d4') | ||||
|         self.assertTrue(len(subtitles['cs']) > 20000) | ||||
|  | ||||
|     def test_nosubtitles(self): | ||||
|         self.DL.expect_warning('video doesn\'t have subtitles') | ||||
| @@ -342,7 +215,122 @@ class TestCeskaTelevizeSubtitles(BaseTestSubtitles): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(len(subtitles), 0) | ||||
|         self.assertFalse(subtitles) | ||||
|  | ||||
|  | ||||
| class TestLyndaSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.lynda.com/Bootstrap-tutorials/Using-exercise-files/110885/114408-4.html' | ||||
|     IE = LyndaIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['en'])) | ||||
|         self.assertEqual(md5(subtitles['en']), '09bbe67222259bed60deaa26997d73a7') | ||||
|  | ||||
|  | ||||
| class TestNPOSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.npo.nl/nos-journaal/28-08-2014/POW_00722860' | ||||
|     IE = NPOIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['nl'])) | ||||
|         self.assertEqual(md5(subtitles['nl']), 'fc6435027572b63fb4ab143abd5ad3f4') | ||||
|  | ||||
|  | ||||
| class TestMTVSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.cc.com/video-clips/kllhuv/stand-up-greg-fitzsimmons--uncensored---too-good-of-a-mother' | ||||
|     IE = ComedyCentralIE | ||||
|  | ||||
|     def getInfoDict(self): | ||||
|         return super(TestMTVSubtitles, self).getInfoDict()['entries'][0] | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['en'])) | ||||
|         self.assertEqual(md5(subtitles['en']), 'b9f6ca22a6acf597ec76f61749765e65') | ||||
|  | ||||
|  | ||||
| class TestNRKSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://tv.nrk.no/serie/ikke-gjoer-dette-hjemme/DMPV73000411/sesong-2/episode-1' | ||||
|     IE = NRKTVIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['no'])) | ||||
|         self.assertEqual(md5(subtitles['no']), '544fa917d3197fcbee64634559221cc2') | ||||
|  | ||||
|  | ||||
| class TestRaiSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.rai.tv/dl/RaiTV/programmi/media/ContentItem-cb27157f-9dd0-4aee-b788-b1f67643a391.html' | ||||
|     IE = RaiIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['it'])) | ||||
|         self.assertEqual(md5(subtitles['it']), 'b1d90a98755126b61e667567a1f6680a') | ||||
|  | ||||
|  | ||||
| class TestVikiSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.viki.com/videos/1060846v-punch-episode-18' | ||||
|     IE = VikiIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['en'])) | ||||
|         self.assertEqual(md5(subtitles['en']), '53cb083a5914b2d84ef1ab67b880d18a') | ||||
|  | ||||
|  | ||||
| class TestThePlatformSubtitles(BaseTestSubtitles): | ||||
|     # from http://www.3playmedia.com/services-features/tools/integrations/theplatform/ | ||||
|     # (see http://theplatform.com/about/partners/type/subtitles-closed-captioning/) | ||||
|     url = 'theplatform:JFUjUE1_ehvq' | ||||
|     IE = ThePlatformIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['en'])) | ||||
|         self.assertEqual(md5(subtitles['en']), '97e7670cbae3c4d26ae8bcc7fdd78d4b') | ||||
|  | ||||
|  | ||||
| class TestRtveSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.rtve.es/alacarta/videos/los-misterios-de-laura/misterios-laura-capitulo-32-misterio-del-numero-17-2-parte/2428621/' | ||||
|     IE = RTVEALaCartaIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         print('Skipping, only available from Spain') | ||||
|         return | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['es'])) | ||||
|         self.assertEqual(md5(subtitles['es']), '69e70cae2d40574fb7316f31d6eb7fca') | ||||
|  | ||||
|  | ||||
| class TestFunnyOrDieSubtitles(BaseTestSubtitles): | ||||
|     url = 'http://www.funnyordie.com/videos/224829ff6d/judd-apatow-will-direct-your-vine' | ||||
|     IE = FunnyOrDieIE | ||||
|  | ||||
|     def test_allsubtitles(self): | ||||
|         self.DL.params['writesubtitles'] = True | ||||
|         self.DL.params['allsubtitles'] = True | ||||
|         subtitles = self.getSubtitles() | ||||
|         self.assertEqual(set(subtitles.keys()), set(['en'])) | ||||
|         self.assertEqual(md5(subtitles['en']), 'c5593c193eacd353596c11c2d4f9ecc4') | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|   | ||||
| @@ -34,8 +34,8 @@ def _make_testfunc(testfile): | ||||
|     def test_func(self): | ||||
|         as_file = os.path.join(TEST_DIR, testfile) | ||||
|         swf_file = os.path.join(TEST_DIR, test_id + '.swf') | ||||
|         if ((not os.path.exists(swf_file)) | ||||
|                 or os.path.getmtime(swf_file) < os.path.getmtime(as_file)): | ||||
|         if ((not os.path.exists(swf_file)) or | ||||
|                 os.path.getmtime(swf_file) < os.path.getmtime(as_file)): | ||||
|             # Recompile | ||||
|             try: | ||||
|                 subprocess.check_call([ | ||||
|   | ||||
| @@ -17,13 +17,22 @@ IGNORED_FILES = [ | ||||
|     'buildserver.py', | ||||
| ] | ||||
|  | ||||
| IGNORED_DIRS = [ | ||||
|     '.git', | ||||
|     '.tox', | ||||
| ] | ||||
|  | ||||
| from test.helper import assertRegexpMatches | ||||
|  | ||||
|  | ||||
| class TestUnicodeLiterals(unittest.TestCase): | ||||
|     def test_all_files(self): | ||||
|         for dirpath, _, filenames in os.walk(rootDir): | ||||
|         for dirpath, dirnames, filenames in os.walk(rootDir): | ||||
|             for ignore_dir in IGNORED_DIRS: | ||||
|                 if ignore_dir in dirnames: | ||||
|                     # If we remove the directory from dirnames os.walk won't | ||||
|                     # recurse into it | ||||
|                     dirnames.remove(ignore_dir) | ||||
|             for basename in filenames: | ||||
|                 if not basename.endswith('.py'): | ||||
|                     continue | ||||
|   | ||||
| @@ -24,6 +24,7 @@ from youtube_dl.utils import ( | ||||
|     encodeFilename, | ||||
|     escape_rfc3986, | ||||
|     escape_url, | ||||
|     ExtractorError, | ||||
|     find_xpath_attr, | ||||
|     fix_xml_ampersands, | ||||
|     InAdvancePagedList, | ||||
| @@ -38,6 +39,9 @@ from youtube_dl.utils import ( | ||||
|     parse_iso8601, | ||||
|     read_batch_urls, | ||||
|     sanitize_filename, | ||||
|     sanitize_path, | ||||
|     prepend_extension, | ||||
|     replace_extension, | ||||
|     shell_quote, | ||||
|     smuggle_url, | ||||
|     str_to_int, | ||||
| @@ -48,11 +52,16 @@ from youtube_dl.utils import ( | ||||
|     unified_strdate, | ||||
|     unsmuggle_url, | ||||
|     uppercase_escape, | ||||
|     lowercase_escape, | ||||
|     url_basename, | ||||
|     urlencode_postdata, | ||||
|     version_tuple, | ||||
|     xpath_with_ns, | ||||
|     xpath_text, | ||||
|     render_table, | ||||
|     match_str, | ||||
|     parse_dfxp_time_expr, | ||||
|     dfxp2srt, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -85,6 +94,11 @@ class TestUtil(unittest.TestCase): | ||||
|             sanitize_filename('New World record at 0:12:34'), | ||||
|             'New World record at 0_12_34') | ||||
|  | ||||
|         self.assertEqual(sanitize_filename('--gasdgf'), '_-gasdgf') | ||||
|         self.assertEqual(sanitize_filename('--gasdgf', is_id=True), '--gasdgf') | ||||
|         self.assertEqual(sanitize_filename('.gasdgf'), 'gasdgf') | ||||
|         self.assertEqual(sanitize_filename('.gasdgf', is_id=True), '.gasdgf') | ||||
|  | ||||
|         forbidden = '"\0\\/' | ||||
|         for fc in forbidden: | ||||
|             for fbc in forbidden: | ||||
| @@ -125,6 +139,58 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual(sanitize_filename('_BD_eEpuzXw', is_id=True), '_BD_eEpuzXw') | ||||
|         self.assertEqual(sanitize_filename('N0Y__7-UOdI', is_id=True), 'N0Y__7-UOdI') | ||||
|  | ||||
|     def test_sanitize_path(self): | ||||
|         if sys.platform != 'win32': | ||||
|             return | ||||
|  | ||||
|         self.assertEqual(sanitize_path('abc'), 'abc') | ||||
|         self.assertEqual(sanitize_path('abc/def'), 'abc\\def') | ||||
|         self.assertEqual(sanitize_path('abc\\def'), 'abc\\def') | ||||
|         self.assertEqual(sanitize_path('abc|def'), 'abc#def') | ||||
|         self.assertEqual(sanitize_path('<>:"|?*'), '#######') | ||||
|         self.assertEqual(sanitize_path('C:/abc/def'), 'C:\\abc\\def') | ||||
|         self.assertEqual(sanitize_path('C?:/abc/def'), 'C##\\abc\\def') | ||||
|  | ||||
|         self.assertEqual(sanitize_path('\\\\?\\UNC\\ComputerName\\abc'), '\\\\?\\UNC\\ComputerName\\abc') | ||||
|         self.assertEqual(sanitize_path('\\\\?\\UNC/ComputerName/abc'), '\\\\?\\UNC\\ComputerName\\abc') | ||||
|  | ||||
|         self.assertEqual(sanitize_path('\\\\?\\C:\\abc'), '\\\\?\\C:\\abc') | ||||
|         self.assertEqual(sanitize_path('\\\\?\\C:/abc'), '\\\\?\\C:\\abc') | ||||
|         self.assertEqual(sanitize_path('\\\\?\\C:\\ab?c\\de:f'), '\\\\?\\C:\\ab#c\\de#f') | ||||
|         self.assertEqual(sanitize_path('\\\\?\\C:\\abc'), '\\\\?\\C:\\abc') | ||||
|  | ||||
|         self.assertEqual( | ||||
|             sanitize_path('youtube/%(uploader)s/%(autonumber)s-%(title)s-%(upload_date)s.%(ext)s'), | ||||
|             'youtube\\%(uploader)s\\%(autonumber)s-%(title)s-%(upload_date)s.%(ext)s') | ||||
|  | ||||
|         self.assertEqual( | ||||
|             sanitize_path('youtube/TheWreckingYard ./00001-Not bad, Especially for Free! (1987 Yamaha 700)-20141116.mp4.part'), | ||||
|             'youtube\\TheWreckingYard #\\00001-Not bad, Especially for Free! (1987 Yamaha 700)-20141116.mp4.part') | ||||
|         self.assertEqual(sanitize_path('abc/def...'), 'abc\\def..#') | ||||
|         self.assertEqual(sanitize_path('abc.../def'), 'abc..#\\def') | ||||
|         self.assertEqual(sanitize_path('abc.../def...'), 'abc..#\\def..#') | ||||
|  | ||||
|         self.assertEqual(sanitize_path('../abc'), '..\\abc') | ||||
|         self.assertEqual(sanitize_path('../../abc'), '..\\..\\abc') | ||||
|         self.assertEqual(sanitize_path('./abc'), 'abc') | ||||
|         self.assertEqual(sanitize_path('./../abc'), '..\\abc') | ||||
|  | ||||
|     def test_prepend_extension(self): | ||||
|         self.assertEqual(prepend_extension('abc.ext', 'temp'), 'abc.temp.ext') | ||||
|         self.assertEqual(prepend_extension('abc.ext', 'temp', 'ext'), 'abc.temp.ext') | ||||
|         self.assertEqual(prepend_extension('abc.unexpected_ext', 'temp', 'ext'), 'abc.unexpected_ext.temp') | ||||
|         self.assertEqual(prepend_extension('abc', 'temp'), 'abc.temp') | ||||
|         self.assertEqual(prepend_extension('.abc', 'temp'), '.abc.temp') | ||||
|         self.assertEqual(prepend_extension('.abc.ext', 'temp'), '.abc.temp.ext') | ||||
|  | ||||
|     def test_replace_extension(self): | ||||
|         self.assertEqual(replace_extension('abc.ext', 'temp'), 'abc.temp') | ||||
|         self.assertEqual(replace_extension('abc.ext', 'temp', 'ext'), 'abc.temp') | ||||
|         self.assertEqual(replace_extension('abc.unexpected_ext', 'temp', 'ext'), 'abc.unexpected_ext.temp') | ||||
|         self.assertEqual(replace_extension('abc', 'temp'), 'abc.temp') | ||||
|         self.assertEqual(replace_extension('.abc', 'temp'), '.abc.temp') | ||||
|         self.assertEqual(replace_extension('.abc.ext', 'temp'), '.abc.temp') | ||||
|  | ||||
|     def test_ordered_set(self): | ||||
|         self.assertEqual(orderedSet([1, 1, 2, 3, 4, 4, 5, 6, 7, 3, 5]), [1, 2, 3, 4, 5, 6, 7]) | ||||
|         self.assertEqual(orderedSet([]), []) | ||||
| @@ -134,6 +200,8 @@ class TestUtil(unittest.TestCase): | ||||
|  | ||||
|     def test_unescape_html(self): | ||||
|         self.assertEqual(unescapeHTML('%20;'), '%20;') | ||||
|         self.assertEqual(unescapeHTML('/'), '/') | ||||
|         self.assertEqual(unescapeHTML('/'), '/') | ||||
|         self.assertEqual( | ||||
|             unescapeHTML('é'), 'é') | ||||
|  | ||||
| @@ -159,6 +227,7 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual( | ||||
|             unified_strdate('2/2/2015 6:47:40 PM', day_first=False), | ||||
|             '20150202') | ||||
|         self.assertEqual(unified_strdate('25-09-2014'), '20140925') | ||||
|  | ||||
|     def test_find_xpath_attr(self): | ||||
|         testxml = '''<root> | ||||
| @@ -186,6 +255,17 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual(find('media:song/media:author').text, 'The Author') | ||||
|         self.assertEqual(find('media:song/url').text, 'http://server.com/download.mp3') | ||||
|  | ||||
|     def test_xpath_text(self): | ||||
|         testxml = '''<root> | ||||
|             <div> | ||||
|                 <p>Foo</p> | ||||
|             </div> | ||||
|         </root>''' | ||||
|         doc = xml.etree.ElementTree.fromstring(testxml) | ||||
|         self.assertEqual(xpath_text(doc, 'div/p'), 'Foo') | ||||
|         self.assertTrue(xpath_text(doc, 'div/bar') is None) | ||||
|         self.assertRaises(ExtractorError, xpath_text, doc, 'div/bar', fatal=True) | ||||
|  | ||||
|     def test_smuggle_url(self): | ||||
|         data = {"ö": "ö", "abc": [3]} | ||||
|         url = 'https://foo.bar/baz?x=y#a' | ||||
| @@ -243,6 +323,7 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual(parse_duration('2.5 hours'), 9000) | ||||
|         self.assertEqual(parse_duration('02:03:04'), 7384) | ||||
|         self.assertEqual(parse_duration('01:02:03:04'), 93784) | ||||
|         self.assertEqual(parse_duration('1 hour 3 minutes'), 3780) | ||||
|  | ||||
|     def test_fix_xml_ampersands(self): | ||||
|         self.assertEqual( | ||||
| @@ -317,6 +398,10 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual(uppercase_escape('aä'), 'aä') | ||||
|         self.assertEqual(uppercase_escape('\\U0001d550'), '𝕐') | ||||
|  | ||||
|     def test_lowercase_escape(self): | ||||
|         self.assertEqual(lowercase_escape('aä'), 'aä') | ||||
|         self.assertEqual(lowercase_escape('\\u0026'), '&') | ||||
|  | ||||
|     def test_limit_length(self): | ||||
|         self.assertEqual(limit_length(None, 12), None) | ||||
|         self.assertEqual(limit_length('foo', 12), 'foo') | ||||
| @@ -369,6 +454,10 @@ class TestUtil(unittest.TestCase): | ||||
|             "playlist":[{"controls":{"all":null}}] | ||||
|         }''') | ||||
|  | ||||
|         inp = '"SAND Number: SAND 2013-7800P\\nPresenter: Tom Russo\\nHabanero Software Training - Xyce Software\\nXyce, Sandia\\u0027s"' | ||||
|         json_code = js_to_json(inp) | ||||
|         self.assertEqual(json.loads(json_code), json.loads(inp)) | ||||
|  | ||||
|     def test_js_to_json_edgecases(self): | ||||
|         on = js_to_json("{abc_def:'1\\'\\\\2\\\\\\'3\"4'}") | ||||
|         self.assertEqual(json.loads(on), {"abc_def": "1'\\2\\'3\"4"}) | ||||
| @@ -386,6 +475,12 @@ class TestUtil(unittest.TestCase): | ||||
|         self.assertEqual(d['x'], 1) | ||||
|         self.assertEqual(d['y'], 'a') | ||||
|  | ||||
|         on = js_to_json('["abc", "def",]') | ||||
|         self.assertEqual(json.loads(on), ['abc', 'def']) | ||||
|  | ||||
|         on = js_to_json('{"abc": "def",}') | ||||
|         self.assertEqual(json.loads(on), {'abc': 'def'}) | ||||
|  | ||||
|     def test_clean_html(self): | ||||
|         self.assertEqual(clean_html('a:\nb'), 'a: b') | ||||
|         self.assertEqual(clean_html('a:\n   "b"'), 'a:    "b"') | ||||
| @@ -459,6 +554,88 @@ ffmpeg version 2.4.4 Copyright (c) 2000-2014 the FFmpeg ...'''), '2.4.4') | ||||
|             '123  4\n' | ||||
|             '9999 51') | ||||
|  | ||||
|     def test_match_str(self): | ||||
|         self.assertRaises(ValueError, match_str, 'xy>foobar', {}) | ||||
|         self.assertFalse(match_str('xy', {'x': 1200})) | ||||
|         self.assertTrue(match_str('!xy', {'x': 1200})) | ||||
|         self.assertTrue(match_str('x', {'x': 1200})) | ||||
|         self.assertFalse(match_str('!x', {'x': 1200})) | ||||
|         self.assertTrue(match_str('x', {'x': 0})) | ||||
|         self.assertFalse(match_str('x>0', {'x': 0})) | ||||
|         self.assertFalse(match_str('x>0', {})) | ||||
|         self.assertTrue(match_str('x>?0', {})) | ||||
|         self.assertTrue(match_str('x>1K', {'x': 1200})) | ||||
|         self.assertFalse(match_str('x>2K', {'x': 1200})) | ||||
|         self.assertTrue(match_str('x>=1200 & x < 1300', {'x': 1200})) | ||||
|         self.assertFalse(match_str('x>=1100 & x < 1200', {'x': 1200})) | ||||
|         self.assertFalse(match_str('y=a212', {'y': 'foobar42'})) | ||||
|         self.assertTrue(match_str('y=foobar42', {'y': 'foobar42'})) | ||||
|         self.assertFalse(match_str('y!=foobar42', {'y': 'foobar42'})) | ||||
|         self.assertTrue(match_str('y!=foobar2', {'y': 'foobar42'})) | ||||
|         self.assertFalse(match_str( | ||||
|             'like_count > 100 & dislike_count <? 50 & description', | ||||
|             {'like_count': 90, 'description': 'foo'})) | ||||
|         self.assertTrue(match_str( | ||||
|             'like_count > 100 & dislike_count <? 50 & description', | ||||
|             {'like_count': 190, 'description': 'foo'})) | ||||
|         self.assertFalse(match_str( | ||||
|             'like_count > 100 & dislike_count <? 50 & description', | ||||
|             {'like_count': 190, 'dislike_count': 60, 'description': 'foo'})) | ||||
|         self.assertFalse(match_str( | ||||
|             'like_count > 100 & dislike_count <? 50 & description', | ||||
|             {'like_count': 190, 'dislike_count': 10})) | ||||
|  | ||||
|     def test_parse_dfxp_time_expr(self): | ||||
|         self.assertEqual(parse_dfxp_time_expr(None), 0.0) | ||||
|         self.assertEqual(parse_dfxp_time_expr(''), 0.0) | ||||
|         self.assertEqual(parse_dfxp_time_expr('0.1'), 0.1) | ||||
|         self.assertEqual(parse_dfxp_time_expr('0.1s'), 0.1) | ||||
|         self.assertEqual(parse_dfxp_time_expr('00:00:01'), 1.0) | ||||
|         self.assertEqual(parse_dfxp_time_expr('00:00:01.100'), 1.1) | ||||
|  | ||||
|     def test_dfxp2srt(self): | ||||
|         dfxp_data = '''<?xml version="1.0" encoding="UTF-8"?> | ||||
|             <tt xmlns="http://www.w3.org/ns/ttml" xml:lang="en" xmlns:tts="http://www.w3.org/ns/ttml#parameter"> | ||||
|             <body> | ||||
|                 <div xml:lang="en"> | ||||
|                     <p begin="0" end="1">The following line contains Chinese characters and special symbols</p> | ||||
|                     <p begin="1" end="2">第二行<br/>♪♪</p> | ||||
|                     <p begin="2" dur="1"><span>Third<br/>Line</span></p> | ||||
|                 </div> | ||||
|             </body> | ||||
|             </tt>''' | ||||
|         srt_data = '''1 | ||||
| 00:00:00,000 --> 00:00:01,000 | ||||
| The following line contains Chinese characters and special symbols | ||||
|  | ||||
| 2 | ||||
| 00:00:01,000 --> 00:00:02,000 | ||||
| 第二行 | ||||
| ♪♪ | ||||
|  | ||||
| 3 | ||||
| 00:00:02,000 --> 00:00:03,000 | ||||
| Third | ||||
| Line | ||||
|  | ||||
| ''' | ||||
|         self.assertEqual(dfxp2srt(dfxp_data), srt_data) | ||||
|  | ||||
|         dfxp_data_no_default_namespace = '''<?xml version="1.0" encoding="UTF-8"?> | ||||
|             <tt xml:lang="en" xmlns:tts="http://www.w3.org/ns/ttml#parameter"> | ||||
|             <body> | ||||
|                 <div xml:lang="en"> | ||||
|                     <p begin="0" end="1">The first line</p> | ||||
|                 </div> | ||||
|             </body> | ||||
|             </tt>''' | ||||
|         srt_data = '''1 | ||||
| 00:00:00,000 --> 00:00:01,000 | ||||
| The first line | ||||
|  | ||||
| ''' | ||||
|         self.assertEqual(dfxp2srt(dfxp_data_no_default_namespace), srt_data) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -8,11 +8,11 @@ import sys | ||||
| import unittest | ||||
| sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__)))) | ||||
|  | ||||
|  | ||||
| import io | ||||
| import re | ||||
| import string | ||||
|  | ||||
| from test.helper import FakeYDL | ||||
| from youtube_dl.extractor import YoutubeIE | ||||
| from youtube_dl.compat import compat_str, compat_urlretrieve | ||||
|  | ||||
| @@ -64,6 +64,12 @@ _TESTS = [ | ||||
|         'js', | ||||
|         '4646B5181C6C3020DF1D9C7FCFEA.AD80ABF70C39BD369CCCAE780AFBB98FA6B6CB42766249D9488C288', | ||||
|         '82C8849D94266724DC6B6AF89BBFA087EACCD963.B93C07FBA084ACAEFCF7C9D1FD0203C6C1815B6B' | ||||
|     ), | ||||
|     ( | ||||
|         'https://s.ytimg.com/yts/jsbin/html5player-en_US-vflKjOTVq/html5player.js', | ||||
|         'js', | ||||
|         '312AA52209E3623129A412D56A40F11CB0AF14AE.3EE09501CB14E3BCDC3B2AE808BF3F1D14E7FBF12', | ||||
|         '112AA5220913623229A412D56A40F11CB0AF14AE.3EE0950FCB14EEBCDC3B2AE808BF331D14E7FBF3', | ||||
|     ) | ||||
| ] | ||||
|  | ||||
| @@ -88,7 +94,8 @@ def make_tfunc(url, stype, sig_input, expected_sig): | ||||
|         if not os.path.exists(fn): | ||||
|             compat_urlretrieve(url, fn) | ||||
|  | ||||
|         ie = YoutubeIE() | ||||
|         ydl = FakeYDL() | ||||
|         ie = YoutubeIE(ydl) | ||||
|         if stype == 'js': | ||||
|             with io.open(fn, encoding='utf-8') as testf: | ||||
|                 jscode = testf.read() | ||||
|   | ||||
							
								
								
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										9
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,8 +1,13 @@ | ||||
| [tox] | ||||
| envlist = py26,py27,py33 | ||||
| envlist = py26,py27,py33,py34 | ||||
| [testenv] | ||||
| deps = | ||||
|    nose | ||||
|    coverage | ||||
| commands = nosetests --verbose {posargs:test}  # --with-coverage --cover-package=youtube_dl --cover-html | ||||
| # We need a valid $HOME for test_compat_expanduser | ||||
| passenv = HOME | ||||
| defaultargs = test --exclude test_download.py --exclude test_age_restriction.py | ||||
|     --exclude test_subtitles.py --exclude test_write_annotations.py | ||||
|     --exclude test_youtube_lists.py | ||||
| commands = nosetests --verbose {posargs:{[testenv]defaultargs}}  # --with-coverage --cover-package=youtube_dl --cover-html | ||||
|                                                # test.test_download:TestDownload.test_NowVideo | ||||
|   | ||||
| @@ -4,8 +4,10 @@ | ||||
| from __future__ import absolute_import, unicode_literals | ||||
|  | ||||
| import collections | ||||
| import contextlib | ||||
| import datetime | ||||
| import errno | ||||
| import fileinput | ||||
| import io | ||||
| import itertools | ||||
| import json | ||||
| @@ -28,6 +30,7 @@ from .compat import ( | ||||
|     compat_basestring, | ||||
|     compat_cookiejar, | ||||
|     compat_expanduser, | ||||
|     compat_get_terminal_size, | ||||
|     compat_http_client, | ||||
|     compat_kwargs, | ||||
|     compat_str, | ||||
| @@ -46,21 +49,22 @@ from .utils import ( | ||||
|     ExtractorError, | ||||
|     format_bytes, | ||||
|     formatSeconds, | ||||
|     get_term_width, | ||||
|     HEADRequest, | ||||
|     locked_file, | ||||
|     make_HTTPS_handler, | ||||
|     MaxDownloadsReached, | ||||
|     PagedList, | ||||
|     parse_filesize, | ||||
|     PerRequestProxyHandler, | ||||
|     PostProcessingError, | ||||
|     platform_name, | ||||
|     preferredencoding, | ||||
|     render_table, | ||||
|     SameFileError, | ||||
|     sanitize_filename, | ||||
|     sanitize_path, | ||||
|     std_headers, | ||||
|     subtitles_filename, | ||||
|     takewhile_inclusive, | ||||
|     UnavailableVideoError, | ||||
|     url_basename, | ||||
|     version_tuple, | ||||
| @@ -68,6 +72,7 @@ from .utils import ( | ||||
|     write_string, | ||||
|     YoutubeDLHandler, | ||||
|     prepend_extension, | ||||
|     replace_extension, | ||||
|     args_to_str, | ||||
|     age_restricted, | ||||
| ) | ||||
| @@ -114,7 +119,7 @@ class YoutubeDL(object): | ||||
|  | ||||
|     username:          Username for authentication purposes. | ||||
|     password:          Password for authentication purposes. | ||||
|     videopassword:     Password for acces a video. | ||||
|     videopassword:     Password for accessing a video. | ||||
|     usenetrc:          Use netrc for authentication instead. | ||||
|     verbose:           Print additional info to stdout. | ||||
|     quiet:             Do not print messages to stdout. | ||||
| @@ -131,7 +136,6 @@ class YoutubeDL(object): | ||||
|                        (or video) as a single JSON line. | ||||
|     simulate:          Do not download the video files. | ||||
|     format:            Video format code. See options.py for more information. | ||||
|     format_limit:      Highest quality format to try. | ||||
|     outtmpl:           Template for output names. | ||||
|     restrictfilenames: Do not allow "&" and spaces in file names | ||||
|     ignoreerrors:      Do not stop on download errors. | ||||
| @@ -154,7 +158,7 @@ class YoutubeDL(object): | ||||
|     allsubtitles:      Downloads all the subtitles of the video | ||||
|                        (requires writesubtitles or writeautomaticsub) | ||||
|     listsubtitles:     Lists all available subtitles for the video | ||||
|     subtitlesformat:   Subtitle format [srt/sbv/vtt] (default=srt) | ||||
|     subtitlesformat:   The format code for subtitles | ||||
|     subtitleslangs:    List of languages of the subtitles to download | ||||
|     keepvideo:         Keep the video file after post-processing | ||||
|     daterange:         A DateRange object, download only if the upload_date is in the range. | ||||
| @@ -181,6 +185,8 @@ class YoutubeDL(object): | ||||
|     prefer_insecure:   Use HTTP instead of HTTPS to retrieve information. | ||||
|                        At the moment, this is only supported by YouTube. | ||||
|     proxy:             URL of the proxy server to use | ||||
|     cn_verification_proxy:  URL of the proxy to use for IP address verification | ||||
|                        on Chinese sites. (Experimental) | ||||
|     socket_timeout:    Time to wait for unresponsive hosts, in seconds | ||||
|     bidi_workaround:   Work around buggy terminals without bidirectional text | ||||
|                        support, using fridibi | ||||
| @@ -199,18 +205,25 @@ class YoutubeDL(object): | ||||
|                        postprocessor. | ||||
|     progress_hooks:    A list of functions that get called on download | ||||
|                        progress, with a dictionary with the entries | ||||
|                        * status: One of "downloading" and "finished". | ||||
|                        * status: One of "downloading", "error", or "finished". | ||||
|                                  Check this first and ignore unknown values. | ||||
|  | ||||
|                        If status is one of "downloading" or "finished", the | ||||
|                        If status is one of "downloading", or "finished", the | ||||
|                        following properties may also be present: | ||||
|                        * filename: The final filename (always present) | ||||
|                        * tmpfilename: The filename we're currently writing to | ||||
|                        * downloaded_bytes: Bytes on disk | ||||
|                        * total_bytes: Size of the whole file, None if unknown | ||||
|                        * tmpfilename: The filename we're currently writing to | ||||
|                        * total_bytes_estimate: Guess of the eventual file size, | ||||
|                                                None if unavailable. | ||||
|                        * elapsed: The number of seconds since download started. | ||||
|                        * eta: The estimated time in seconds, None if unknown | ||||
|                        * speed: The download speed in bytes/second, None if | ||||
|                                 unknown | ||||
|                        * fragment_index: The counter of the currently | ||||
|                                          downloaded video fragment. | ||||
|                        * fragment_count: The number of fragments (= individual | ||||
|                                          files that will be merged) | ||||
|  | ||||
|                        Progress hooks are guaranteed to be called at least once | ||||
|                        (with status "finished") if the download is successful. | ||||
| @@ -225,21 +238,29 @@ class YoutubeDL(object): | ||||
|     call_home:         Boolean, true iff we are allowed to contact the | ||||
|                        youtube-dl servers for debugging. | ||||
|     sleep_interval:    Number of seconds to sleep before each download. | ||||
|     external_downloader:  Executable of the external downloader to call. | ||||
|     listformats:       Print an overview of available video formats and exit. | ||||
|     list_thumbnails:   Print a table of all thumbnails and exit. | ||||
|     match_filter:      A function that gets called with the info_dict of | ||||
|                        every video. | ||||
|                        If it returns a message, the video is ignored. | ||||
|                        If it returns None, the video is downloaded. | ||||
|                        match_filter_func in utils.py is one example for this. | ||||
|     no_color:          Do not emit color codes in output. | ||||
|  | ||||
|     The following options determine which downloader is picked: | ||||
|     external_downloader: Executable of the external downloader to call. | ||||
|                        None or unset for standard (built-in) downloader. | ||||
|     hls_prefer_native: Use the native HLS downloader instead of ffmpeg/avconv. | ||||
|  | ||||
|     The following parameters are not used by YoutubeDL itself, they are used by | ||||
|     the FileDownloader: | ||||
|     the downloader (see youtube_dl/downloader/common.py): | ||||
|     nopart, updatetime, buffersize, ratelimit, min_filesize, max_filesize, test, | ||||
|     noresizebuffer, retries, continuedl, noprogress, consoletitle, | ||||
|     xattr_set_filesize. | ||||
|     xattr_set_filesize, external_downloader_args. | ||||
|  | ||||
|     The following options are used by the post processors: | ||||
|     prefer_ffmpeg:     If True, use ffmpeg instead of avconv if both are available, | ||||
|                        otherwise prefer avconv. | ||||
|     exec_cmd:          Arbitrary command to run after downloading | ||||
|     """ | ||||
|  | ||||
|     params = None | ||||
| @@ -268,7 +289,7 @@ class YoutubeDL(object): | ||||
|             try: | ||||
|                 import pty | ||||
|                 master, slave = pty.openpty() | ||||
|                 width = get_term_width() | ||||
|                 width = compat_get_terminal_size().columns | ||||
|                 if width is None: | ||||
|                     width_args = [] | ||||
|                 else: | ||||
| @@ -292,8 +313,8 @@ class YoutubeDL(object): | ||||
|                     raise | ||||
|  | ||||
|         if (sys.version_info >= (3,) and sys.platform != 'win32' and | ||||
|                 sys.getfilesystemencoding() in ['ascii', 'ANSI_X3.4-1968'] | ||||
|                 and not params.get('restrictfilenames', False)): | ||||
|                 sys.getfilesystemencoding() in ['ascii', 'ANSI_X3.4-1968'] and | ||||
|                 not params.get('restrictfilenames', False)): | ||||
|             # On Python 3, the Unicode filesystem API will throw errors (#1474) | ||||
|             self.report_warning( | ||||
|                 'Assuming --restrict-filenames since file system encoding ' | ||||
| @@ -301,8 +322,10 @@ class YoutubeDL(object): | ||||
|                 'Set the LC_ALL environment variable to fix this.') | ||||
|             self.params['restrictfilenames'] = True | ||||
|  | ||||
|         if '%(stitle)s' in self.params.get('outtmpl', ''): | ||||
|             self.report_warning('%(stitle)s is deprecated. Use the %(title)s and the --restrict-filenames flag(which also secures %(uploader)s et al) instead.') | ||||
|         if isinstance(params.get('outtmpl'), bytes): | ||||
|             self.report_warning( | ||||
|                 'Parameter outtmpl is bytes, but should be a unicode string. ' | ||||
|                 'Put  from __future__ import unicode_literals  at the top of your code file or consider switching to Python 3.x.') | ||||
|  | ||||
|         self._setup_opener() | ||||
|  | ||||
| @@ -485,7 +508,7 @@ class YoutubeDL(object): | ||||
|         else: | ||||
|             if self.params.get('no_warnings'): | ||||
|                 return | ||||
|             if self._err_file.isatty() and os.name != 'nt': | ||||
|             if not self.params.get('no_color') and self._err_file.isatty() and os.name != 'nt': | ||||
|                 _msg_header = '\033[0;33mWARNING:\033[0m' | ||||
|             else: | ||||
|                 _msg_header = 'WARNING:' | ||||
| @@ -497,7 +520,7 @@ class YoutubeDL(object): | ||||
|         Do the same as trouble, but prefixes the message with 'ERROR:', colored | ||||
|         in red if stderr is a tty file. | ||||
|         ''' | ||||
|         if self._err_file.isatty() and os.name != 'nt': | ||||
|         if not self.params.get('no_color') and self._err_file.isatty() and os.name != 'nt': | ||||
|             _msg_header = '\033[0;31mERROR:\033[0m' | ||||
|         else: | ||||
|             _msg_header = 'ERROR:' | ||||
| @@ -541,7 +564,7 @@ class YoutubeDL(object): | ||||
|                                  if v is not None) | ||||
|             template_dict = collections.defaultdict(lambda: 'NA', template_dict) | ||||
|  | ||||
|             outtmpl = self.params.get('outtmpl', DEFAULT_OUTTMPL) | ||||
|             outtmpl = sanitize_path(self.params.get('outtmpl', DEFAULT_OUTTMPL)) | ||||
|             tmpl = compat_expanduser(outtmpl) | ||||
|             filename = tmpl % template_dict | ||||
|             # Temporary fix for #4787 | ||||
| @@ -554,7 +577,7 @@ class YoutubeDL(object): | ||||
|             self.report_error('Error in output template: ' + str(err) + ' (encoding: ' + repr(preferredencoding()) + ')') | ||||
|             return None | ||||
|  | ||||
|     def _match_entry(self, info_dict): | ||||
|     def _match_entry(self, info_dict, incomplete): | ||||
|         """ Returns None iff the file should be downloaded """ | ||||
|  | ||||
|         video_title = info_dict.get('title', info_dict.get('id', 'video')) | ||||
| @@ -583,9 +606,17 @@ class YoutubeDL(object): | ||||
|             if max_views is not None and view_count > max_views: | ||||
|                 return 'Skipping %s, because it has exceeded the maximum view count (%d/%d)' % (video_title, view_count, max_views) | ||||
|         if age_restricted(info_dict.get('age_limit'), self.params.get('age_limit')): | ||||
|             return 'Skipping "%s" because it is age restricted' % title | ||||
|             return 'Skipping "%s" because it is age restricted' % video_title | ||||
|         if self.in_download_archive(info_dict): | ||||
|             return '%s has already been recorded in archive' % video_title | ||||
|  | ||||
|         if not incomplete: | ||||
|             match_filter = self.params.get('match_filter') | ||||
|             if match_filter is not None: | ||||
|                 ret = match_filter(info_dict) | ||||
|                 if ret is not None: | ||||
|                     return ret | ||||
|  | ||||
|         return None | ||||
|  | ||||
|     @staticmethod | ||||
| @@ -600,7 +631,7 @@ class YoutubeDL(object): | ||||
|         Returns a list with a dictionary for each video we find. | ||||
|         If 'download', also downloads the videos. | ||||
|         extra_info is a dict containing the extra values to add to each result | ||||
|          ''' | ||||
|         ''' | ||||
|  | ||||
|         if ie_key: | ||||
|             ies = [self.get_info_extractor(ie_key)] | ||||
| @@ -729,7 +760,9 @@ class YoutubeDL(object): | ||||
|             if isinstance(ie_entries, list): | ||||
|                 n_all_entries = len(ie_entries) | ||||
|                 if playlistitems: | ||||
|                     entries = [ie_entries[i - 1] for i in playlistitems] | ||||
|                     entries = [ | ||||
|                         ie_entries[i - 1] for i in playlistitems | ||||
|                         if -n_all_entries <= i - 1 < n_all_entries] | ||||
|                 else: | ||||
|                     entries = ie_entries[playliststart:playlistend] | ||||
|                 n_entries = len(entries) | ||||
| @@ -779,7 +812,7 @@ class YoutubeDL(object): | ||||
|                     'extractor_key': ie_result['extractor_key'], | ||||
|                 } | ||||
|  | ||||
|                 reason = self._match_entry(entry) | ||||
|                 reason = self._match_entry(entry, incomplete=True) | ||||
|                 if reason is not None: | ||||
|                     self.to_screen('[download] ' + reason) | ||||
|                     continue | ||||
| @@ -884,10 +917,17 @@ class YoutubeDL(object): | ||||
|         if not available_formats: | ||||
|             return None | ||||
|  | ||||
|         if format_spec == 'best' or format_spec is None: | ||||
|             return available_formats[-1] | ||||
|         elif format_spec == 'worst': | ||||
|             return available_formats[0] | ||||
|         if format_spec in ['best', 'worst', None]: | ||||
|             format_idx = 0 if format_spec == 'worst' else -1 | ||||
|             audiovideo_formats = [ | ||||
|                 f for f in available_formats | ||||
|                 if f.get('vcodec') != 'none' and f.get('acodec') != 'none'] | ||||
|             if audiovideo_formats: | ||||
|                 return audiovideo_formats[format_idx] | ||||
|             # for audio only (soundcloud) or video only (imgur) urls, select the best/worst audio format | ||||
|             elif (all(f.get('acodec') != 'none' for f in available_formats) or | ||||
|                   all(f.get('vcodec') != 'none' for f in available_formats)): | ||||
|                 return available_formats[format_idx] | ||||
|         elif format_spec == 'bestaudio': | ||||
|             audio_formats = [ | ||||
|                 f for f in available_formats | ||||
| @@ -937,30 +977,9 @@ class YoutubeDL(object): | ||||
|         return res | ||||
|  | ||||
|     def _calc_cookies(self, info_dict): | ||||
|         class _PseudoRequest(object): | ||||
|             def __init__(self, url): | ||||
|                 self.url = url | ||||
|                 self.headers = {} | ||||
|                 self.unverifiable = False | ||||
|  | ||||
|             def add_unredirected_header(self, k, v): | ||||
|                 self.headers[k] = v | ||||
|  | ||||
|             def get_full_url(self): | ||||
|                 return self.url | ||||
|  | ||||
|             def is_unverifiable(self): | ||||
|                 return self.unverifiable | ||||
|  | ||||
|             def has_header(self, h): | ||||
|                 return h in self.headers | ||||
|  | ||||
|             def get_header(self, h, default=None): | ||||
|                 return self.headers.get(h, default) | ||||
|  | ||||
|         pr = _PseudoRequest(info_dict['url']) | ||||
|         pr = compat_urllib_request.Request(info_dict['url']) | ||||
|         self.cookiejar.add_cookie_header(pr) | ||||
|         return pr.headers.get('Cookie') | ||||
|         return pr.get_header('Cookie') | ||||
|  | ||||
|     def process_video_result(self, info_dict, download=True): | ||||
|         assert info_dict.get('_type', 'video') == 'video' | ||||
| @@ -997,13 +1016,22 @@ class YoutubeDL(object): | ||||
|             info_dict['display_id'] = info_dict['id'] | ||||
|  | ||||
|         if info_dict.get('upload_date') is None and info_dict.get('timestamp') is not None: | ||||
|             # Working around negative timestamps in Windows | ||||
|             # (see http://bugs.python.org/issue1646728) | ||||
|             if info_dict['timestamp'] < 0 and os.name == 'nt': | ||||
|                 info_dict['timestamp'] = 0 | ||||
|             upload_date = datetime.datetime.utcfromtimestamp( | ||||
|                 info_dict['timestamp']) | ||||
|             info_dict['upload_date'] = upload_date.strftime('%Y%m%d') | ||||
|             # Working around out-of-range timestamp values (e.g. negative ones on Windows, | ||||
|             # see http://bugs.python.org/issue1646728) | ||||
|             try: | ||||
|                 upload_date = datetime.datetime.utcfromtimestamp(info_dict['timestamp']) | ||||
|                 info_dict['upload_date'] = upload_date.strftime('%Y%m%d') | ||||
|             except (ValueError, OverflowError, OSError): | ||||
|                 pass | ||||
|  | ||||
|         if self.params.get('listsubtitles', False): | ||||
|             if 'automatic_captions' in info_dict: | ||||
|                 self.list_subtitles(info_dict['id'], info_dict.get('automatic_captions'), 'automatic captions') | ||||
|             self.list_subtitles(info_dict['id'], info_dict.get('subtitles'), 'subtitles') | ||||
|             return | ||||
|         info_dict['requested_subtitles'] = self.process_subtitles( | ||||
|             info_dict['id'], info_dict.get('subtitles'), | ||||
|             info_dict.get('automatic_captions')) | ||||
|  | ||||
|         # This extractors handle format selection themselves | ||||
|         if info_dict['extractor'] in ['Youku']: | ||||
| @@ -1021,6 +1049,8 @@ class YoutubeDL(object): | ||||
|         if not formats: | ||||
|             raise ExtractorError('No video formats found!') | ||||
|  | ||||
|         formats_dict = {} | ||||
|  | ||||
|         # We check that all the formats have the format and format_id fields | ||||
|         for i, format in enumerate(formats): | ||||
|             if 'url' not in format: | ||||
| @@ -1028,6 +1058,18 @@ class YoutubeDL(object): | ||||
|  | ||||
|             if format.get('format_id') is None: | ||||
|                 format['format_id'] = compat_str(i) | ||||
|             format_id = format['format_id'] | ||||
|             if format_id not in formats_dict: | ||||
|                 formats_dict[format_id] = [] | ||||
|             formats_dict[format_id].append(format) | ||||
|  | ||||
|         # Make sure all formats have unique format_id | ||||
|         for format_id, ambiguous_formats in formats_dict.items(): | ||||
|             if len(ambiguous_formats) > 1: | ||||
|                 for i, format in enumerate(ambiguous_formats): | ||||
|                     format['format_id'] = '%s-%d' % (format_id, i) | ||||
|  | ||||
|         for i, format in enumerate(formats): | ||||
|             if format.get('format') is None: | ||||
|                 format['format'] = '{id} - {res}{note}'.format( | ||||
|                     id=format['format_id'], | ||||
| @@ -1043,12 +1085,6 @@ class YoutubeDL(object): | ||||
|             full_format_info.update(format) | ||||
|             format['http_headers'] = self._calc_headers(full_format_info) | ||||
|  | ||||
|         format_limit = self.params.get('format_limit', None) | ||||
|         if format_limit: | ||||
|             formats = list(takewhile_inclusive( | ||||
|                 lambda f: f['format_id'] != format_limit, formats | ||||
|             )) | ||||
|  | ||||
|         # TODO Central sorting goes here | ||||
|  | ||||
|         if formats[0] is not info_dict: | ||||
| @@ -1066,10 +1102,16 @@ class YoutubeDL(object): | ||||
|  | ||||
|         req_format = self.params.get('format') | ||||
|         if req_format is None: | ||||
|             req_format = 'best' | ||||
|             req_format_list = [] | ||||
|             if (self.params.get('outtmpl', DEFAULT_OUTTMPL) != '-' and | ||||
|                     info_dict['extractor'] in ['youtube', 'ted']): | ||||
|                 merger = FFmpegMergerPP(self) | ||||
|                 if merger.available and merger.can_merge(): | ||||
|                     req_format_list.append('bestvideo+bestaudio') | ||||
|             req_format_list.append('best') | ||||
|             req_format = '/'.join(req_format_list) | ||||
|         formats_to_download = [] | ||||
|         # The -1 is for supporting YoutubeIE | ||||
|         if req_format in ('-1', 'all'): | ||||
|         if req_format == 'all': | ||||
|             formats_to_download = formats | ||||
|         else: | ||||
|             for rfstr in req_format.split(','): | ||||
| @@ -1133,6 +1175,55 @@ class YoutubeDL(object): | ||||
|         info_dict.update(formats_to_download[-1]) | ||||
|         return info_dict | ||||
|  | ||||
|     def process_subtitles(self, video_id, normal_subtitles, automatic_captions): | ||||
|         """Select the requested subtitles and their format""" | ||||
|         available_subs = {} | ||||
|         if normal_subtitles and self.params.get('writesubtitles'): | ||||
|             available_subs.update(normal_subtitles) | ||||
|         if automatic_captions and self.params.get('writeautomaticsub'): | ||||
|             for lang, cap_info in automatic_captions.items(): | ||||
|                 if lang not in available_subs: | ||||
|                     available_subs[lang] = cap_info | ||||
|  | ||||
|         if (not self.params.get('writesubtitles') and not | ||||
|                 self.params.get('writeautomaticsub') or not | ||||
|                 available_subs): | ||||
|             return None | ||||
|  | ||||
|         if self.params.get('allsubtitles', False): | ||||
|             requested_langs = available_subs.keys() | ||||
|         else: | ||||
|             if self.params.get('subtitleslangs', False): | ||||
|                 requested_langs = self.params.get('subtitleslangs') | ||||
|             elif 'en' in available_subs: | ||||
|                 requested_langs = ['en'] | ||||
|             else: | ||||
|                 requested_langs = [list(available_subs.keys())[0]] | ||||
|  | ||||
|         formats_query = self.params.get('subtitlesformat', 'best') | ||||
|         formats_preference = formats_query.split('/') if formats_query else [] | ||||
|         subs = {} | ||||
|         for lang in requested_langs: | ||||
|             formats = available_subs.get(lang) | ||||
|             if formats is None: | ||||
|                 self.report_warning('%s subtitles not available for %s' % (lang, video_id)) | ||||
|                 continue | ||||
|             for ext in formats_preference: | ||||
|                 if ext == 'best': | ||||
|                     f = formats[-1] | ||||
|                     break | ||||
|                 matches = list(filter(lambda f: f['ext'] == ext, formats)) | ||||
|                 if matches: | ||||
|                     f = matches[-1] | ||||
|                     break | ||||
|             else: | ||||
|                 f = formats[-1] | ||||
|                 self.report_warning( | ||||
|                     'No subtitle format found matching "%s" for language %s, ' | ||||
|                     'using %s' % (formats_query, lang, f['ext'])) | ||||
|             subs[lang] = f | ||||
|         return subs | ||||
|  | ||||
|     def process_info(self, info_dict): | ||||
|         """Process a single resolved IE result.""" | ||||
|  | ||||
| @@ -1147,13 +1238,10 @@ class YoutubeDL(object): | ||||
|         if len(info_dict['title']) > 200: | ||||
|             info_dict['title'] = info_dict['title'][:197] + '...' | ||||
|  | ||||
|         # Keep for backwards compatibility | ||||
|         info_dict['stitle'] = info_dict['title'] | ||||
|  | ||||
|         if 'format' not in info_dict: | ||||
|             info_dict['format'] = info_dict['ext'] | ||||
|  | ||||
|         reason = self._match_entry(info_dict) | ||||
|         reason = self._match_entry(info_dict, incomplete=False) | ||||
|         if reason is not None: | ||||
|             self.to_screen('[download] ' + reason) | ||||
|             return | ||||
| @@ -1195,7 +1283,7 @@ class YoutubeDL(object): | ||||
|             return | ||||
|  | ||||
|         try: | ||||
|             dn = os.path.dirname(encodeFilename(filename)) | ||||
|             dn = os.path.dirname(sanitize_path(encodeFilename(filename))) | ||||
|             if dn and not os.path.exists(dn): | ||||
|                 os.makedirs(dn) | ||||
|         except (OSError, IOError) as err: | ||||
| @@ -1203,7 +1291,7 @@ class YoutubeDL(object): | ||||
|             return | ||||
|  | ||||
|         if self.params.get('writedescription', False): | ||||
|             descfn = filename + '.description' | ||||
|             descfn = replace_extension(filename, 'description', info_dict.get('ext')) | ||||
|             if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(descfn)): | ||||
|                 self.to_screen('[info] Video description is already present') | ||||
|             elif info_dict.get('description') is None: | ||||
| @@ -1218,7 +1306,7 @@ class YoutubeDL(object): | ||||
|                     return | ||||
|  | ||||
|         if self.params.get('writeannotations', False): | ||||
|             annofn = filename + '.annotations.xml' | ||||
|             annofn = replace_extension(filename, 'annotations.xml', info_dict.get('ext')) | ||||
|             if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(annofn)): | ||||
|                 self.to_screen('[info] Video annotations are already present') | ||||
|             else: | ||||
| @@ -1235,15 +1323,23 @@ class YoutubeDL(object): | ||||
|         subtitles_are_requested = any([self.params.get('writesubtitles', False), | ||||
|                                        self.params.get('writeautomaticsub')]) | ||||
|  | ||||
|         if subtitles_are_requested and 'subtitles' in info_dict and info_dict['subtitles']: | ||||
|         if subtitles_are_requested and info_dict.get('requested_subtitles'): | ||||
|             # subtitles download errors are already managed as troubles in relevant IE | ||||
|             # that way it will silently go on when used with unsupporting IE | ||||
|             subtitles = info_dict['subtitles'] | ||||
|             sub_format = self.params.get('subtitlesformat', 'srt') | ||||
|             for sub_lang in subtitles.keys(): | ||||
|                 sub = subtitles[sub_lang] | ||||
|                 if sub is None: | ||||
|                     continue | ||||
|             subtitles = info_dict['requested_subtitles'] | ||||
|             ie = self.get_info_extractor(info_dict['extractor_key']) | ||||
|             for sub_lang, sub_info in subtitles.items(): | ||||
|                 sub_format = sub_info['ext'] | ||||
|                 if sub_info.get('data') is not None: | ||||
|                     sub_data = sub_info['data'] | ||||
|                 else: | ||||
|                     try: | ||||
|                         sub_data = ie._download_webpage( | ||||
|                             sub_info['url'], info_dict['id'], note=False) | ||||
|                     except ExtractorError as err: | ||||
|                         self.report_warning('Unable to download subtitle for "%s": %s' % | ||||
|                                             (sub_lang, compat_str(err.cause))) | ||||
|                         continue | ||||
|                 try: | ||||
|                     sub_filename = subtitles_filename(filename, sub_lang, sub_format) | ||||
|                     if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(sub_filename)): | ||||
| @@ -1251,19 +1347,19 @@ class YoutubeDL(object): | ||||
|                     else: | ||||
|                         self.to_screen('[info] Writing video subtitles to: ' + sub_filename) | ||||
|                         with io.open(encodeFilename(sub_filename), 'w', encoding='utf-8') as subfile: | ||||
|                             subfile.write(sub) | ||||
|                             subfile.write(sub_data) | ||||
|                 except (OSError, IOError): | ||||
|                     self.report_error('Cannot write subtitles file ' + sub_filename) | ||||
|                     return | ||||
|  | ||||
|         if self.params.get('writeinfojson', False): | ||||
|             infofn = os.path.splitext(filename)[0] + '.info.json' | ||||
|             infofn = replace_extension(filename, 'info.json', info_dict.get('ext')) | ||||
|             if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(infofn)): | ||||
|                 self.to_screen('[info] Video description metadata is already present') | ||||
|             else: | ||||
|                 self.to_screen('[info] Writing video description metadata as JSON to: ' + infofn) | ||||
|                 try: | ||||
|                     write_json_file(info_dict, infofn) | ||||
|                     write_json_file(self.filter_requested_info(info_dict), infofn) | ||||
|                 except (OSError, IOError): | ||||
|                     self.report_error('Cannot write metadata to JSON file ' + infofn) | ||||
|                     return | ||||
| @@ -1283,24 +1379,57 @@ class YoutubeDL(object): | ||||
|                 if info_dict.get('requested_formats') is not None: | ||||
|                     downloaded = [] | ||||
|                     success = True | ||||
|                     merger = FFmpegMergerPP(self, not self.params.get('keepvideo')) | ||||
|                     if not merger._executable: | ||||
|                     merger = FFmpegMergerPP(self) | ||||
|                     if not merger.available: | ||||
|                         postprocessors = [] | ||||
|                         self.report_warning('You have requested multiple ' | ||||
|                                             'formats but ffmpeg or avconv are not installed.' | ||||
|                                             ' The formats won\'t be merged') | ||||
|                                             ' The formats won\'t be merged.') | ||||
|                     else: | ||||
|                         postprocessors = [merger] | ||||
|                     for f in info_dict['requested_formats']: | ||||
|                         new_info = dict(info_dict) | ||||
|                         new_info.update(f) | ||||
|                         fname = self.prepare_filename(new_info) | ||||
|                         fname = prepend_extension(fname, 'f%s' % f['format_id']) | ||||
|                         downloaded.append(fname) | ||||
|                         partial_success = dl(fname, new_info) | ||||
|                         success = success and partial_success | ||||
|                     info_dict['__postprocessors'] = postprocessors | ||||
|                     info_dict['__files_to_merge'] = downloaded | ||||
|  | ||||
|                     def compatible_formats(formats): | ||||
|                         video, audio = formats | ||||
|                         # Check extension | ||||
|                         video_ext, audio_ext = audio.get('ext'), video.get('ext') | ||||
|                         if video_ext and audio_ext: | ||||
|                             COMPATIBLE_EXTS = ( | ||||
|                                 ('mp3', 'mp4', 'm4a', 'm4p', 'm4b', 'm4r', 'm4v'), | ||||
|                                 ('webm') | ||||
|                             ) | ||||
|                             for exts in COMPATIBLE_EXTS: | ||||
|                                 if video_ext in exts and audio_ext in exts: | ||||
|                                     return True | ||||
|                         # TODO: Check acodec/vcodec | ||||
|                         return False | ||||
|  | ||||
|                     filename_real_ext = os.path.splitext(filename)[1][1:] | ||||
|                     filename_wo_ext = ( | ||||
|                         os.path.splitext(filename)[0] | ||||
|                         if filename_real_ext == info_dict['ext'] | ||||
|                         else filename) | ||||
|                     requested_formats = info_dict['requested_formats'] | ||||
|                     if self.params.get('merge_output_format') is None and not compatible_formats(requested_formats): | ||||
|                         info_dict['ext'] = 'mkv' | ||||
|                         self.report_warning( | ||||
|                             'Requested formats are incompatible for merge and will be merged into mkv.') | ||||
|                     # Ensure filename always has a correct extension for successful merge | ||||
|                     filename = '%s.%s' % (filename_wo_ext, info_dict['ext']) | ||||
|                     if os.path.exists(encodeFilename(filename)): | ||||
|                         self.to_screen( | ||||
|                             '[download] %s has already been downloaded and ' | ||||
|                             'merged' % filename) | ||||
|                     else: | ||||
|                         for f in requested_formats: | ||||
|                             new_info = dict(info_dict) | ||||
|                             new_info.update(f) | ||||
|                             fname = self.prepare_filename(new_info) | ||||
|                             fname = prepend_extension(fname, 'f%s' % f['format_id'], new_info['ext']) | ||||
|                             downloaded.append(fname) | ||||
|                             partial_success = dl(fname, new_info) | ||||
|                             success = success and partial_success | ||||
|                         info_dict['__postprocessors'] = postprocessors | ||||
|                         info_dict['__files_to_merge'] = downloaded | ||||
|                 else: | ||||
|                     # Just a single file | ||||
|                     success = dl(filename, info_dict) | ||||
| @@ -1363,8 +1492,8 @@ class YoutubeDL(object): | ||||
|         """Download a given list of URLs.""" | ||||
|         outtmpl = self.params.get('outtmpl', DEFAULT_OUTTMPL) | ||||
|         if (len(url_list) > 1 and | ||||
|                 '%' not in outtmpl | ||||
|                 and self.params.get('max_downloads') != 1): | ||||
|                 '%' not in outtmpl and | ||||
|                 self.params.get('max_downloads') != 1): | ||||
|             raise SameFileError(outtmpl) | ||||
|  | ||||
|         for url in url_list: | ||||
| @@ -1383,8 +1512,11 @@ class YoutubeDL(object): | ||||
|         return self._download_retcode | ||||
|  | ||||
|     def download_with_info_file(self, info_filename): | ||||
|         with io.open(info_filename, 'r', encoding='utf-8') as f: | ||||
|             info = json.load(f) | ||||
|         with contextlib.closing(fileinput.FileInput( | ||||
|                 [info_filename], mode='r', | ||||
|                 openhook=fileinput.hook_encoded('utf-8'))) as f: | ||||
|             # FileInput doesn't have a read method, we can't call json.load | ||||
|             info = self.filter_requested_info(json.loads('\n'.join(f))) | ||||
|         try: | ||||
|             self.process_ie_result(info, download=True) | ||||
|         except DownloadError: | ||||
| @@ -1396,6 +1528,12 @@ class YoutubeDL(object): | ||||
|                 raise | ||||
|         return self._download_retcode | ||||
|  | ||||
|     @staticmethod | ||||
|     def filter_requested_info(info_dict): | ||||
|         return dict( | ||||
|             (k, v) for k, v in info_dict.items() | ||||
|             if k not in ['requested_formats', 'requested_subtitles']) | ||||
|  | ||||
|     def post_process(self, filename, ie_info): | ||||
|         """Run all the postprocessors on the given file.""" | ||||
|         info = dict(ie_info) | ||||
| @@ -1405,24 +1543,18 @@ class YoutubeDL(object): | ||||
|             pps_chain.extend(ie_info['__postprocessors']) | ||||
|         pps_chain.extend(self._pps) | ||||
|         for pp in pps_chain: | ||||
|             keep_video = None | ||||
|             old_filename = info['filepath'] | ||||
|             files_to_delete = [] | ||||
|             try: | ||||
|                 keep_video_wish, info = pp.run(info) | ||||
|                 if keep_video_wish is not None: | ||||
|                     if keep_video_wish: | ||||
|                         keep_video = keep_video_wish | ||||
|                     elif keep_video is None: | ||||
|                         # No clear decision yet, let IE decide | ||||
|                         keep_video = keep_video_wish | ||||
|                 files_to_delete, info = pp.run(info) | ||||
|             except PostProcessingError as e: | ||||
|                 self.report_error(e.msg) | ||||
|             if keep_video is False and not self.params.get('keepvideo', False): | ||||
|                 try: | ||||
|             if files_to_delete and not self.params.get('keepvideo', False): | ||||
|                 for old_filename in files_to_delete: | ||||
|                     self.to_screen('Deleting original file %s (pass -k to keep)' % old_filename) | ||||
|                     os.remove(encodeFilename(old_filename)) | ||||
|                 except (IOError, OSError): | ||||
|                     self.report_warning('Unable to remove downloaded video file') | ||||
|                     try: | ||||
|                         os.remove(encodeFilename(old_filename)) | ||||
|                     except (IOError, OSError): | ||||
|                         self.report_warning('Unable to remove downloaded original file') | ||||
|  | ||||
|     def _make_archive_id(self, info_dict): | ||||
|         # Future-proof against any change in case | ||||
| @@ -1531,30 +1663,18 @@ class YoutubeDL(object): | ||||
|         return res | ||||
|  | ||||
|     def list_formats(self, info_dict): | ||||
|         def line(format, idlen=20): | ||||
|             return (('%-' + compat_str(idlen + 1) + 's%-10s%-12s%s') % ( | ||||
|                 format['format_id'], | ||||
|                 format['ext'], | ||||
|                 self.format_resolution(format), | ||||
|                 self._format_note(format), | ||||
|             )) | ||||
|  | ||||
|         formats = info_dict.get('formats', [info_dict]) | ||||
|         idlen = max(len('format code'), | ||||
|                     max(len(f['format_id']) for f in formats)) | ||||
|         formats_s = [ | ||||
|             line(f, idlen) for f in formats | ||||
|         table = [ | ||||
|             [f['format_id'], f['ext'], self.format_resolution(f), self._format_note(f)] | ||||
|             for f in formats | ||||
|             if f.get('preference') is None or f['preference'] >= -1000] | ||||
|         if len(formats) > 1: | ||||
|             formats_s[0] += (' ' if self._format_note(formats[0]) else '') + '(worst)' | ||||
|             formats_s[-1] += (' ' if self._format_note(formats[-1]) else '') + '(best)' | ||||
|             table[-1][-1] += (' ' if table[-1][-1] else '') + '(best)' | ||||
|  | ||||
|         header_line = line({ | ||||
|             'format_id': 'format code', 'ext': 'extension', | ||||
|             'resolution': 'resolution', 'format_note': 'note'}, idlen=idlen) | ||||
|         header_line = ['format code', 'extension', 'resolution', 'note'] | ||||
|         self.to_screen( | ||||
|             '[info] Available formats for %s:\n%s\n%s' % | ||||
|             (info_dict['id'], header_line, '\n'.join(formats_s))) | ||||
|             '[info] Available formats for %s:\n%s' % | ||||
|             (info_dict['id'], render_table(header_line, table))) | ||||
|  | ||||
|     def list_thumbnails(self, info_dict): | ||||
|         thumbnails = info_dict.get('thumbnails') | ||||
| @@ -1573,6 +1693,17 @@ class YoutubeDL(object): | ||||
|             ['ID', 'width', 'height', 'URL'], | ||||
|             [[t['id'], t.get('width', 'unknown'), t.get('height', 'unknown'), t['url']] for t in thumbnails])) | ||||
|  | ||||
|     def list_subtitles(self, video_id, subtitles, name='subtitles'): | ||||
|         if not subtitles: | ||||
|             self.to_screen('%s has no %s' % (video_id, name)) | ||||
|             return | ||||
|         self.to_screen( | ||||
|             'Available %s for %s:' % (name, video_id)) | ||||
|         self.to_screen(render_table( | ||||
|             ['Language', 'formats'], | ||||
|             [[lang, ', '.join(f['ext'] for f in reversed(formats))] | ||||
|                 for lang, formats in subtitles.items()])) | ||||
|  | ||||
|     def urlopen(self, req): | ||||
|         """ Start an HTTP download """ | ||||
|  | ||||
| @@ -1591,7 +1722,8 @@ class YoutubeDL(object): | ||||
|             if req_is_string: | ||||
|                 req = url_escaped | ||||
|             else: | ||||
|                 req = compat_urllib_request.Request( | ||||
|                 req_type = HEADRequest if req.get_method() == 'HEAD' else compat_urllib_request.Request | ||||
|                 req = req_type( | ||||
|                     url_escaped, data=req.data, headers=req.headers, | ||||
|                     origin_req_host=req.origin_req_host, unverifiable=req.unverifiable) | ||||
|  | ||||
| @@ -1626,15 +1758,15 @@ class YoutubeDL(object): | ||||
|             out = out.decode().strip() | ||||
|             if re.match('[0-9a-f]+', out): | ||||
|                 self._write_string('[debug] Git HEAD: ' + out + '\n') | ||||
|         except: | ||||
|         except Exception: | ||||
|             try: | ||||
|                 sys.exc_clear() | ||||
|             except: | ||||
|             except Exception: | ||||
|                 pass | ||||
|         self._write_string('[debug] Python version %s - %s\n' % ( | ||||
|             platform.python_version(), platform_name())) | ||||
|  | ||||
|         exe_versions = FFmpegPostProcessor.get_versions() | ||||
|         exe_versions = FFmpegPostProcessor.get_versions(self) | ||||
|         exe_versions['rtmpdump'] = rtmpdump_version() | ||||
|         exe_str = ', '.join( | ||||
|             '%s %s' % (exe, v) | ||||
| @@ -1689,13 +1821,14 @@ class YoutubeDL(object): | ||||
|             # Set HTTPS proxy to HTTP one if given (https://github.com/rg3/youtube-dl/issues/805) | ||||
|             if 'http' in proxies and 'https' not in proxies: | ||||
|                 proxies['https'] = proxies['http'] | ||||
|         proxy_handler = compat_urllib_request.ProxyHandler(proxies) | ||||
|         proxy_handler = PerRequestProxyHandler(proxies) | ||||
|  | ||||
|         debuglevel = 1 if self.params.get('debug_printtraffic') else 0 | ||||
|         https_handler = make_HTTPS_handler(self.params, debuglevel=debuglevel) | ||||
|         ydlh = YoutubeDLHandler(self.params, debuglevel=debuglevel) | ||||
|         opener = compat_urllib_request.build_opener( | ||||
|             https_handler, proxy_handler, cookie_processor, ydlh) | ||||
|             proxy_handler, https_handler, cookie_processor, ydlh) | ||||
|  | ||||
|         # Delete the default user-agent header, which would otherwise apply in | ||||
|         # cases where our custom HTTP handler doesn't come into play | ||||
|         # (See https://github.com/rg3/youtube-dl/issues/1309 for details) | ||||
| @@ -1736,7 +1869,7 @@ class YoutubeDL(object): | ||||
|             thumb_ext = determine_ext(t['url'], 'jpg') | ||||
|             suffix = '_%s' % t['id'] if len(thumbnails) > 1 else '' | ||||
|             thumb_display_id = '%s ' % t['id'] if len(thumbnails) > 1 else '' | ||||
|             thumb_filename = os.path.splitext(filename)[0] + suffix + '.' + thumb_ext | ||||
|             t['filename'] = thumb_filename = os.path.splitext(filename)[0] + suffix + '.' + thumb_ext | ||||
|  | ||||
|             if self.params.get('nooverwrites', False) and os.path.exists(encodeFilename(thumb_filename)): | ||||
|                 self.to_screen('[%s] %s: Thumbnail %sis already present' % | ||||
|   | ||||
| @@ -9,6 +9,7 @@ import codecs | ||||
| import io | ||||
| import os | ||||
| import random | ||||
| import shlex | ||||
| import sys | ||||
|  | ||||
|  | ||||
| @@ -23,9 +24,10 @@ from .compat import ( | ||||
| ) | ||||
| from .utils import ( | ||||
|     DateRange, | ||||
|     DEFAULT_OUTTMPL, | ||||
|     decodeOption, | ||||
|     DEFAULT_OUTTMPL, | ||||
|     DownloadError, | ||||
|     match_filter_func, | ||||
|     MaxDownloadsReached, | ||||
|     preferredencoding, | ||||
|     read_batch_urls, | ||||
| @@ -169,6 +171,9 @@ def _real_main(argv=None): | ||||
|     if opts.recodevideo is not None: | ||||
|         if opts.recodevideo not in ['mp4', 'flv', 'webm', 'ogg', 'mkv']: | ||||
|             parser.error('invalid video recode format specified') | ||||
|     if opts.convertsubtitles is not None: | ||||
|         if opts.convertsubtitles not in ['srt', 'vtt', 'ass']: | ||||
|             parser.error('invalid subtitle format specified') | ||||
|  | ||||
|     if opts.date is not None: | ||||
|         date = DateRange.day(opts.date) | ||||
| @@ -184,18 +189,14 @@ def _real_main(argv=None): | ||||
|     if opts.allsubtitles and not opts.writeautomaticsub: | ||||
|         opts.writesubtitles = True | ||||
|  | ||||
|     if sys.version_info < (3,): | ||||
|         # In Python 2, sys.argv is a bytestring (also note http://bugs.python.org/issue2128 for Windows systems) | ||||
|         if opts.outtmpl is not None: | ||||
|             opts.outtmpl = opts.outtmpl.decode(preferredencoding()) | ||||
|     outtmpl = ((opts.outtmpl is not None and opts.outtmpl) | ||||
|                or (opts.format == '-1' and opts.usetitle and '%(title)s-%(id)s-%(format)s.%(ext)s') | ||||
|                or (opts.format == '-1' and '%(id)s-%(format)s.%(ext)s') | ||||
|                or (opts.usetitle and opts.autonumber and '%(autonumber)s-%(title)s-%(id)s.%(ext)s') | ||||
|                or (opts.usetitle and '%(title)s-%(id)s.%(ext)s') | ||||
|                or (opts.useid and '%(id)s.%(ext)s') | ||||
|                or (opts.autonumber and '%(autonumber)s-%(id)s.%(ext)s') | ||||
|                or DEFAULT_OUTTMPL) | ||||
|     outtmpl = ((opts.outtmpl is not None and opts.outtmpl) or | ||||
|                (opts.format == '-1' and opts.usetitle and '%(title)s-%(id)s-%(format)s.%(ext)s') or | ||||
|                (opts.format == '-1' and '%(id)s-%(format)s.%(ext)s') or | ||||
|                (opts.usetitle and opts.autonumber and '%(autonumber)s-%(title)s-%(id)s.%(ext)s') or | ||||
|                (opts.usetitle and '%(title)s-%(id)s.%(ext)s') or | ||||
|                (opts.useid and '%(id)s.%(ext)s') or | ||||
|                (opts.autonumber and '%(autonumber)s-%(id)s.%(ext)s') or | ||||
|                DEFAULT_OUTTMPL) | ||||
|     if not os.path.splitext(outtmpl)[1] and opts.extractaudio: | ||||
|         parser.error('Cannot download a video and extract audio into the same' | ||||
|                      ' file! Use "{0}.%(ext)s" instead of "{0}" as the output' | ||||
| @@ -208,6 +209,11 @@ def _real_main(argv=None): | ||||
|     # PostProcessors | ||||
|     postprocessors = [] | ||||
|     # Add the metadata pp first, the other pps will copy it | ||||
|     if opts.metafromtitle: | ||||
|         postprocessors.append({ | ||||
|             'key': 'MetadataFromTitle', | ||||
|             'titleformat': opts.metafromtitle | ||||
|         }) | ||||
|     if opts.addmetadata: | ||||
|         postprocessors.append({'key': 'FFmpegMetadata'}) | ||||
|     if opts.extractaudio: | ||||
| @@ -222,23 +228,30 @@ def _real_main(argv=None): | ||||
|             'key': 'FFmpegVideoConvertor', | ||||
|             'preferedformat': opts.recodevideo, | ||||
|         }) | ||||
|     if opts.convertsubtitles: | ||||
|         postprocessors.append({ | ||||
|             'key': 'FFmpegSubtitlesConvertor', | ||||
|             'format': opts.convertsubtitles, | ||||
|         }) | ||||
|     if opts.embedsubtitles: | ||||
|         postprocessors.append({ | ||||
|             'key': 'FFmpegEmbedSubtitle', | ||||
|             'subtitlesformat': opts.subtitlesformat, | ||||
|         }) | ||||
|     if opts.xattrs: | ||||
|         postprocessors.append({'key': 'XAttrMetadata'}) | ||||
|     if opts.embedthumbnail: | ||||
|         if not opts.addmetadata: | ||||
|             postprocessors.append({'key': 'FFmpegAudioFix'}) | ||||
|         postprocessors.append({'key': 'AtomicParsley'}) | ||||
|         already_have_thumbnail = opts.writethumbnail or opts.write_all_thumbnails | ||||
|         postprocessors.append({ | ||||
|             'key': 'EmbedThumbnail', | ||||
|             'already_have_thumbnail': already_have_thumbnail | ||||
|         }) | ||||
|         if not already_have_thumbnail: | ||||
|             opts.writethumbnail = True | ||||
|     # Please keep ExecAfterDownload towards the bottom as it allows the user to modify the final file in any way. | ||||
|     # So if the user is able to remove the file before your postprocessor runs it might cause a few problems. | ||||
|     if opts.exec_cmd: | ||||
|         postprocessors.append({ | ||||
|             'key': 'ExecAfterDownload', | ||||
|             'verboseOutput': opts.verbose, | ||||
|             'exec_cmd': opts.exec_cmd, | ||||
|         }) | ||||
|     if opts.xattr_set_filesize: | ||||
| @@ -247,6 +260,12 @@ def _real_main(argv=None): | ||||
|             xattr  # Confuse flake8 | ||||
|         except ImportError: | ||||
|             parser.error('setting filesize xattr requested but python-xattr is not available') | ||||
|     external_downloader_args = None | ||||
|     if opts.external_downloader_args: | ||||
|         external_downloader_args = shlex.split(opts.external_downloader_args) | ||||
|     match_filter = ( | ||||
|         None if opts.match_filter is None | ||||
|         else match_filter_func(opts.match_filter)) | ||||
|  | ||||
|     ydl_opts = { | ||||
|         'usenetrc': opts.usenetrc, | ||||
| @@ -269,7 +288,6 @@ def _real_main(argv=None): | ||||
|         'simulate': opts.simulate or any_getting, | ||||
|         'skip_download': opts.skip_download, | ||||
|         'format': opts.format, | ||||
|         'format_limit': opts.format_limit, | ||||
|         'listformats': opts.listformats, | ||||
|         'outtmpl': outtmpl, | ||||
|         'autonumber_size': opts.autonumber_size, | ||||
| @@ -332,7 +350,6 @@ def _real_main(argv=None): | ||||
|         'default_search': opts.default_search, | ||||
|         'youtube_include_dash_manifest': opts.youtube_include_dash_manifest, | ||||
|         'encoding': opts.encoding, | ||||
|         'exec_cmd': opts.exec_cmd, | ||||
|         'extract_flat': opts.extract_flat, | ||||
|         'merge_output_format': opts.merge_output_format, | ||||
|         'postprocessors': postprocessors, | ||||
| @@ -344,6 +361,12 @@ def _real_main(argv=None): | ||||
|         'list_thumbnails': opts.list_thumbnails, | ||||
|         'playlist_items': opts.playlist_items, | ||||
|         'xattr_set_filesize': opts.xattr_set_filesize, | ||||
|         'match_filter': match_filter, | ||||
|         'no_color': opts.no_color, | ||||
|         'ffmpeg_location': opts.ffmpeg_location, | ||||
|         'hls_prefer_native': opts.hls_prefer_native, | ||||
|         'external_downloader_args': external_downloader_args, | ||||
|         'cn_verification_proxy': opts.cn_verification_proxy, | ||||
|     } | ||||
|  | ||||
|     with YoutubeDL(ydl_opts) as ydl: | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| __all__ = ['aes_encrypt', 'key_expansion', 'aes_ctr_decrypt', 'aes_cbc_decrypt', 'aes_decrypt_text'] | ||||
|  | ||||
| import base64 | ||||
| from math import ceil | ||||
|  | ||||
| @@ -154,7 +152,7 @@ def aes_decrypt_text(data, password, key_size_bytes): | ||||
|     """ | ||||
|     NONCE_LENGTH_BYTES = 8 | ||||
|  | ||||
|     data = bytes_to_intlist(base64.b64decode(data)) | ||||
|     data = bytes_to_intlist(base64.b64decode(data.encode('utf-8'))) | ||||
|     password = bytes_to_intlist(password.encode('utf-8')) | ||||
|  | ||||
|     key = password[:key_size_bytes] + [0] * (key_size_bytes - len(password)) | ||||
| @@ -329,3 +327,5 @@ def inc(data): | ||||
|             data[i] = data[i] + 1 | ||||
|             break | ||||
|     return data | ||||
|  | ||||
| __all__ = ['aes_encrypt', 'key_expansion', 'aes_ctr_decrypt', 'aes_cbc_decrypt', 'aes_decrypt_text'] | ||||
|   | ||||
| @@ -1,9 +1,11 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import collections | ||||
| import getpass | ||||
| import optparse | ||||
| import os | ||||
| import re | ||||
| import shutil | ||||
| import socket | ||||
| import subprocess | ||||
| import sys | ||||
| @@ -44,11 +46,6 @@ try: | ||||
| except ImportError:  # Python 2 | ||||
|     import htmlentitydefs as compat_html_entities | ||||
|  | ||||
| try: | ||||
|     import html.parser as compat_html_parser | ||||
| except ImportError:  # Python 2 | ||||
|     import HTMLParser as compat_html_parser | ||||
|  | ||||
| try: | ||||
|     import http.client as compat_http_client | ||||
| except ImportError:  # Python 2 | ||||
| @@ -364,6 +361,33 @@ def workaround_optparse_bug9161(): | ||||
|             return real_add_option(self, *bargs, **bkwargs) | ||||
|         optparse.OptionGroup.add_option = _compat_add_option | ||||
|  | ||||
| if hasattr(shutil, 'get_terminal_size'):  # Python >= 3.3 | ||||
|     compat_get_terminal_size = shutil.get_terminal_size | ||||
| else: | ||||
|     _terminal_size = collections.namedtuple('terminal_size', ['columns', 'lines']) | ||||
|  | ||||
|     def compat_get_terminal_size(): | ||||
|         columns = compat_getenv('COLUMNS', None) | ||||
|         if columns: | ||||
|             columns = int(columns) | ||||
|         else: | ||||
|             columns = None | ||||
|         lines = compat_getenv('LINES', None) | ||||
|         if lines: | ||||
|             lines = int(lines) | ||||
|         else: | ||||
|             lines = None | ||||
|  | ||||
|         try: | ||||
|             sp = subprocess.Popen( | ||||
|                 ['stty', 'size'], | ||||
|                 stdout=subprocess.PIPE, stderr=subprocess.PIPE) | ||||
|             out, err = sp.communicate() | ||||
|             lines, columns = map(int, out.split()) | ||||
|         except Exception: | ||||
|             pass | ||||
|         return _terminal_size(columns, lines) | ||||
|  | ||||
|  | ||||
| __all__ = [ | ||||
|     'compat_HTTPError', | ||||
| @@ -371,10 +395,10 @@ __all__ = [ | ||||
|     'compat_chr', | ||||
|     'compat_cookiejar', | ||||
|     'compat_expanduser', | ||||
|     'compat_get_terminal_size', | ||||
|     'compat_getenv', | ||||
|     'compat_getpass', | ||||
|     'compat_html_entities', | ||||
|     'compat_html_parser', | ||||
|     'compat_http_client', | ||||
|     'compat_http_server', | ||||
|     'compat_kwargs', | ||||
|   | ||||
| @@ -6,7 +6,7 @@ from .f4m import F4mFD | ||||
| from .hls import HlsFD | ||||
| from .hls import NativeHlsFD | ||||
| from .http import HttpFD | ||||
| from .mplayer import MplayerFD | ||||
| from .rtsp import RtspFD | ||||
| from .rtmp import RtmpFD | ||||
|  | ||||
| from ..utils import ( | ||||
| @@ -17,8 +17,8 @@ PROTOCOL_MAP = { | ||||
|     'rtmp': RtmpFD, | ||||
|     'm3u8_native': NativeHlsFD, | ||||
|     'm3u8': HlsFD, | ||||
|     'mms': MplayerFD, | ||||
|     'rtsp': MplayerFD, | ||||
|     'mms': RtspFD, | ||||
|     'rtsp': RtspFD, | ||||
|     'f4m': F4mFD, | ||||
| } | ||||
|  | ||||
| @@ -34,6 +34,9 @@ def get_suitable_downloader(info_dict, params={}): | ||||
|         if ed.supports(info_dict): | ||||
|             return ed | ||||
|  | ||||
|     if protocol == 'm3u8' and params.get('hls_prefer_native'): | ||||
|         return NativeHlsFD | ||||
|  | ||||
|     return PROTOCOL_MAP.get(protocol, HttpFD) | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| from __future__ import unicode_literals | ||||
| from __future__ import division, unicode_literals | ||||
|  | ||||
| import os | ||||
| import re | ||||
| @@ -8,6 +8,7 @@ import time | ||||
| from ..compat import compat_str | ||||
| from ..utils import ( | ||||
|     encodeFilename, | ||||
|     decodeArgument, | ||||
|     format_bytes, | ||||
|     timeconvert, | ||||
| ) | ||||
| @@ -42,6 +43,8 @@ class FileDownloader(object): | ||||
|     max_filesize:       Skip files larger than this size | ||||
|     xattr_set_filesize: Set ytdl.filesize user xattribute with expected size. | ||||
|                         (experimenatal) | ||||
|     external_downloader_args:  A list of additional command-line arguments for the | ||||
|                         external downloader. | ||||
|  | ||||
|     Subclasses of this one must re-define the real_download method. | ||||
|     """ | ||||
| @@ -54,6 +57,7 @@ class FileDownloader(object): | ||||
|         self.ydl = ydl | ||||
|         self._progress_hooks = [] | ||||
|         self.params = params | ||||
|         self.add_progress_hook(self.report_progress) | ||||
|  | ||||
|     @staticmethod | ||||
|     def format_seconds(seconds): | ||||
| @@ -201,7 +205,7 @@ class FileDownloader(object): | ||||
|             return | ||||
|         try: | ||||
|             os.utime(filename, (time.time(), filetime)) | ||||
|         except: | ||||
|         except Exception: | ||||
|             pass | ||||
|         return filetime | ||||
|  | ||||
| @@ -226,42 +230,64 @@ class FileDownloader(object): | ||||
|             self.to_screen(clear_line + fullmsg, skip_eol=not is_last_line) | ||||
|         self.to_console_title('youtube-dl ' + msg) | ||||
|  | ||||
|     def report_progress(self, percent, data_len_str, speed, eta): | ||||
|         """Report download progress.""" | ||||
|         if self.params.get('noprogress', False): | ||||
|     def report_progress(self, s): | ||||
|         if s['status'] == 'finished': | ||||
|             if self.params.get('noprogress', False): | ||||
|                 self.to_screen('[download] Download completed') | ||||
|             else: | ||||
|                 s['_total_bytes_str'] = format_bytes(s['total_bytes']) | ||||
|                 if s.get('elapsed') is not None: | ||||
|                     s['_elapsed_str'] = self.format_seconds(s['elapsed']) | ||||
|                     msg_template = '100%% of %(_total_bytes_str)s in %(_elapsed_str)s' | ||||
|                 else: | ||||
|                     msg_template = '100%% of %(_total_bytes_str)s' | ||||
|                 self._report_progress_status( | ||||
|                     msg_template % s, is_last_line=True) | ||||
|  | ||||
|         if self.params.get('noprogress'): | ||||
|             return | ||||
|         if eta is not None: | ||||
|             eta_str = self.format_eta(eta) | ||||
|         else: | ||||
|             eta_str = 'Unknown ETA' | ||||
|         if percent is not None: | ||||
|             percent_str = self.format_percent(percent) | ||||
|         else: | ||||
|             percent_str = 'Unknown %' | ||||
|         speed_str = self.format_speed(speed) | ||||
|  | ||||
|         msg = ('%s of %s at %s ETA %s' % | ||||
|                (percent_str, data_len_str, speed_str, eta_str)) | ||||
|         self._report_progress_status(msg) | ||||
|  | ||||
|     def report_progress_live_stream(self, downloaded_data_len, speed, elapsed): | ||||
|         if self.params.get('noprogress', False): | ||||
|         if s['status'] != 'downloading': | ||||
|             return | ||||
|         downloaded_str = format_bytes(downloaded_data_len) | ||||
|         speed_str = self.format_speed(speed) | ||||
|         elapsed_str = FileDownloader.format_seconds(elapsed) | ||||
|         msg = '%s at %s (%s)' % (downloaded_str, speed_str, elapsed_str) | ||||
|         self._report_progress_status(msg) | ||||
|  | ||||
|     def report_finish(self, data_len_str, tot_time): | ||||
|         """Report download finished.""" | ||||
|         if self.params.get('noprogress', False): | ||||
|             self.to_screen('[download] Download completed') | ||||
|         if s.get('eta') is not None: | ||||
|             s['_eta_str'] = self.format_eta(s['eta']) | ||||
|         else: | ||||
|             self._report_progress_status( | ||||
|                 ('100%% of %s in %s' % | ||||
|                  (data_len_str, self.format_seconds(tot_time))), | ||||
|                 is_last_line=True) | ||||
|             s['_eta_str'] = 'Unknown ETA' | ||||
|  | ||||
|         if s.get('total_bytes') and s.get('downloaded_bytes') is not None: | ||||
|             s['_percent_str'] = self.format_percent(100 * s['downloaded_bytes'] / s['total_bytes']) | ||||
|         elif s.get('total_bytes_estimate') and s.get('downloaded_bytes') is not None: | ||||
|             s['_percent_str'] = self.format_percent(100 * s['downloaded_bytes'] / s['total_bytes_estimate']) | ||||
|         else: | ||||
|             if s.get('downloaded_bytes') == 0: | ||||
|                 s['_percent_str'] = self.format_percent(0) | ||||
|             else: | ||||
|                 s['_percent_str'] = 'Unknown %' | ||||
|  | ||||
|         if s.get('speed') is not None: | ||||
|             s['_speed_str'] = self.format_speed(s['speed']) | ||||
|         else: | ||||
|             s['_speed_str'] = 'Unknown speed' | ||||
|  | ||||
|         if s.get('total_bytes') is not None: | ||||
|             s['_total_bytes_str'] = format_bytes(s['total_bytes']) | ||||
|             msg_template = '%(_percent_str)s of %(_total_bytes_str)s at %(_speed_str)s ETA %(_eta_str)s' | ||||
|         elif s.get('total_bytes_estimate') is not None: | ||||
|             s['_total_bytes_estimate_str'] = format_bytes(s['total_bytes_estimate']) | ||||
|             msg_template = '%(_percent_str)s of ~%(_total_bytes_estimate_str)s at %(_speed_str)s ETA %(_eta_str)s' | ||||
|         else: | ||||
|             if s.get('downloaded_bytes') is not None: | ||||
|                 s['_downloaded_bytes_str'] = format_bytes(s['downloaded_bytes']) | ||||
|                 if s.get('elapsed'): | ||||
|                     s['_elapsed_str'] = self.format_seconds(s['elapsed']) | ||||
|                     msg_template = '%(_downloaded_bytes_str)s at %(_speed_str)s (%(_elapsed_str)s)' | ||||
|                 else: | ||||
|                     msg_template = '%(_downloaded_bytes_str)s at %(_speed_str)s' | ||||
|             else: | ||||
|                 msg_template = '%(_percent_str)s % at %(_speed_str)s ETA %(_eta_str)s' | ||||
|  | ||||
|         self._report_progress_status(msg_template % s) | ||||
|  | ||||
|     def report_resuming_byte(self, resume_len): | ||||
|         """Report attempt to resume at given byte.""" | ||||
| @@ -288,14 +314,14 @@ class FileDownloader(object): | ||||
|         """ | ||||
|  | ||||
|         nooverwrites_and_exists = ( | ||||
|             self.params.get('nooverwrites', False) | ||||
|             and os.path.exists(encodeFilename(filename)) | ||||
|             self.params.get('nooverwrites', False) and | ||||
|             os.path.exists(encodeFilename(filename)) | ||||
|         ) | ||||
|  | ||||
|         continuedl_and_exists = ( | ||||
|             self.params.get('continuedl', False) | ||||
|             and os.path.isfile(encodeFilename(filename)) | ||||
|             and not self.params.get('nopart', False) | ||||
|             self.params.get('continuedl', True) and | ||||
|             os.path.isfile(encodeFilename(filename)) and | ||||
|             not self.params.get('nopart', False) | ||||
|         ) | ||||
|  | ||||
|         # Check file already present | ||||
| @@ -328,19 +354,15 @@ class FileDownloader(object): | ||||
|         # this interface | ||||
|         self._progress_hooks.append(ph) | ||||
|  | ||||
|     def _debug_cmd(self, args, subprocess_encoding, exe=None): | ||||
|     def _debug_cmd(self, args, exe=None): | ||||
|         if not self.params.get('verbose', False): | ||||
|             return | ||||
|  | ||||
|         if exe is None: | ||||
|             exe = os.path.basename(args[0]) | ||||
|         str_args = [decodeArgument(a) for a in args] | ||||
|  | ||||
|         if exe is None: | ||||
|             exe = os.path.basename(str_args[0]) | ||||
|  | ||||
|         if subprocess_encoding: | ||||
|             str_args = [ | ||||
|                 a.decode(subprocess_encoding) if isinstance(a, bytes) else a | ||||
|                 for a in args] | ||||
|         else: | ||||
|             str_args = args | ||||
|         try: | ||||
|             import pipes | ||||
|             shell_quote = lambda args: ' '.join(map(pipes.quote, str_args)) | ||||
|   | ||||
| @@ -2,11 +2,11 @@ from __future__ import unicode_literals | ||||
|  | ||||
| import os.path | ||||
| import subprocess | ||||
| import sys | ||||
|  | ||||
| from .common import FileDownloader | ||||
| from ..utils import ( | ||||
|     encodeFilename, | ||||
|     encodeArgument, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -51,19 +51,18 @@ class ExternalFD(FileDownloader): | ||||
|             return [] | ||||
|         return [command_option, source_address] | ||||
|  | ||||
|     def _configuration_args(self, default=[]): | ||||
|         ex_args = self.params.get('external_downloader_args') | ||||
|         if ex_args is None: | ||||
|             return default | ||||
|         assert isinstance(ex_args, list) | ||||
|         return ex_args | ||||
|  | ||||
|     def _call_downloader(self, tmpfilename, info_dict): | ||||
|         """ Either overwrite this or implement _make_cmd """ | ||||
|         cmd = self._make_cmd(tmpfilename, info_dict) | ||||
|         cmd = [encodeArgument(a) for a in self._make_cmd(tmpfilename, info_dict)] | ||||
|  | ||||
|         if sys.platform == 'win32' and sys.version_info < (3, 0): | ||||
|             # Windows subprocess module does not actually support Unicode | ||||
|             # on Python 2.x | ||||
|             # See http://stackoverflow.com/a/9951851/35070 | ||||
|             subprocess_encoding = sys.getfilesystemencoding() | ||||
|             cmd = [a.encode(subprocess_encoding, 'ignore') for a in cmd] | ||||
|         else: | ||||
|             subprocess_encoding = None | ||||
|         self._debug_cmd(cmd, subprocess_encoding) | ||||
|         self._debug_cmd(cmd) | ||||
|  | ||||
|         p = subprocess.Popen( | ||||
|             cmd, stderr=subprocess.PIPE) | ||||
| @@ -75,10 +74,11 @@ class ExternalFD(FileDownloader): | ||||
|  | ||||
| class CurlFD(ExternalFD): | ||||
|     def _make_cmd(self, tmpfilename, info_dict): | ||||
|         cmd = [self.exe, '-o', tmpfilename] | ||||
|         cmd = [self.exe, '--location', '-o', tmpfilename] | ||||
|         for key, val in info_dict['http_headers'].items(): | ||||
|             cmd += ['--header', '%s: %s' % (key, val)] | ||||
|         cmd += self._source_address('--interface') | ||||
|         cmd += self._configuration_args() | ||||
|         cmd += ['--', info_dict['url']] | ||||
|         return cmd | ||||
|  | ||||
| @@ -89,15 +89,16 @@ class WgetFD(ExternalFD): | ||||
|         for key, val in info_dict['http_headers'].items(): | ||||
|             cmd += ['--header', '%s: %s' % (key, val)] | ||||
|         cmd += self._source_address('--bind-address') | ||||
|         cmd += self._configuration_args() | ||||
|         cmd += ['--', info_dict['url']] | ||||
|         return cmd | ||||
|  | ||||
|  | ||||
| class Aria2cFD(ExternalFD): | ||||
|     def _make_cmd(self, tmpfilename, info_dict): | ||||
|         cmd = [ | ||||
|             self.exe, '-c', | ||||
|             '--min-split-size', '1M', '--max-connection-per-server', '4'] | ||||
|         cmd = [self.exe, '-c'] | ||||
|         cmd += self._configuration_args([ | ||||
|             '--min-split-size', '1M', '--max-connection-per-server', '4']) | ||||
|         dn = os.path.dirname(tmpfilename) | ||||
|         if dn: | ||||
|             cmd += ['--dir', dn] | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| from __future__ import unicode_literals | ||||
| from __future__ import division, unicode_literals | ||||
|  | ||||
| import base64 | ||||
| import io | ||||
| @@ -11,11 +11,11 @@ from .common import FileDownloader | ||||
| from .http import HttpFD | ||||
| from ..compat import ( | ||||
|     compat_urlparse, | ||||
|     compat_urllib_error, | ||||
| ) | ||||
| from ..utils import ( | ||||
|     struct_pack, | ||||
|     struct_unpack, | ||||
|     format_bytes, | ||||
|     encodeFilename, | ||||
|     sanitize_open, | ||||
|     xpath_text, | ||||
| @@ -122,7 +122,8 @@ class FlvReader(io.BytesIO): | ||||
|  | ||||
|         self.read_unsigned_int()  # BootstrapinfoVersion | ||||
|         # Profile,Live,Update,Reserved | ||||
|         self.read(1) | ||||
|         flags = self.read_unsigned_char() | ||||
|         live = flags & 0x20 != 0 | ||||
|         # time scale | ||||
|         self.read_unsigned_int() | ||||
|         # CurrentMediaTime | ||||
| @@ -161,6 +162,7 @@ class FlvReader(io.BytesIO): | ||||
|         return { | ||||
|             'segments': segments, | ||||
|             'fragments': fragments, | ||||
|             'live': live, | ||||
|         } | ||||
|  | ||||
|     def read_bootstrap_info(self): | ||||
| @@ -183,6 +185,10 @@ def build_fragments_list(boot_info): | ||||
|     for segment, fragments_count in segment_run_table['segment_run']: | ||||
|         for _ in range(fragments_count): | ||||
|             res.append((segment, next(fragments_counter))) | ||||
|  | ||||
|     if boot_info['live']: | ||||
|         res = res[-2:] | ||||
|  | ||||
|     return res | ||||
|  | ||||
|  | ||||
| @@ -247,22 +253,43 @@ class F4mFD(FileDownloader): | ||||
|             self.report_error('Unsupported DRM') | ||||
|         return media | ||||
|  | ||||
|     def _get_bootstrap_from_url(self, bootstrap_url): | ||||
|         bootstrap = self.ydl.urlopen(bootstrap_url).read() | ||||
|         return read_bootstrap_info(bootstrap) | ||||
|  | ||||
|     def _update_live_fragments(self, bootstrap_url, latest_fragment): | ||||
|         fragments_list = [] | ||||
|         retries = 30 | ||||
|         while (not fragments_list) and (retries > 0): | ||||
|             boot_info = self._get_bootstrap_from_url(bootstrap_url) | ||||
|             fragments_list = build_fragments_list(boot_info) | ||||
|             fragments_list = [f for f in fragments_list if f[1] > latest_fragment] | ||||
|             if not fragments_list: | ||||
|                 # Retry after a while | ||||
|                 time.sleep(5.0) | ||||
|                 retries -= 1 | ||||
|  | ||||
|         if not fragments_list: | ||||
|             self.report_error('Failed to update fragments') | ||||
|  | ||||
|         return fragments_list | ||||
|  | ||||
|     def _parse_bootstrap_node(self, node, base_url): | ||||
|         if node.text is None: | ||||
|             bootstrap_url = compat_urlparse.urljoin( | ||||
|                 base_url, node.attrib['url']) | ||||
|             boot_info = self._get_bootstrap_from_url(bootstrap_url) | ||||
|         else: | ||||
|             bootstrap_url = None | ||||
|             bootstrap = base64.b64decode(node.text.encode('ascii')) | ||||
|             boot_info = read_bootstrap_info(bootstrap) | ||||
|         return (boot_info, bootstrap_url) | ||||
|  | ||||
|     def real_download(self, filename, info_dict): | ||||
|         man_url = info_dict['url'] | ||||
|         requested_bitrate = info_dict.get('tbr') | ||||
|         self.to_screen('[download] Downloading f4m manifest') | ||||
|         manifest = self.ydl.urlopen(man_url).read() | ||||
|         self.report_destination(filename) | ||||
|         http_dl = HttpQuietDownloader( | ||||
|             self.ydl, | ||||
|             { | ||||
|                 'continuedl': True, | ||||
|                 'quiet': True, | ||||
|                 'noprogress': True, | ||||
|                 'ratelimit': self.params.get('ratelimit', None), | ||||
|                 'test': self.params.get('test', False), | ||||
|             } | ||||
|         ) | ||||
|  | ||||
|         doc = etree.fromstring(manifest) | ||||
|         formats = [(int(f.attrib.get('bitrate', -1)), f) | ||||
| @@ -277,18 +304,13 @@ class F4mFD(FileDownloader): | ||||
|  | ||||
|         base_url = compat_urlparse.urljoin(man_url, media.attrib['url']) | ||||
|         bootstrap_node = doc.find(_add_ns('bootstrapInfo')) | ||||
|         if bootstrap_node.text is None: | ||||
|             bootstrap_url = compat_urlparse.urljoin( | ||||
|                 base_url, bootstrap_node.attrib['url']) | ||||
|             bootstrap = self.ydl.urlopen(bootstrap_url).read() | ||||
|         else: | ||||
|             bootstrap = base64.b64decode(bootstrap_node.text) | ||||
|         boot_info, bootstrap_url = self._parse_bootstrap_node(bootstrap_node, base_url) | ||||
|         live = boot_info['live'] | ||||
|         metadata_node = media.find(_add_ns('metadata')) | ||||
|         if metadata_node is not None: | ||||
|             metadata = base64.b64decode(metadata_node.text) | ||||
|             metadata = base64.b64decode(metadata_node.text.encode('ascii')) | ||||
|         else: | ||||
|             metadata = None | ||||
|         boot_info = read_bootstrap_info(bootstrap) | ||||
|  | ||||
|         fragments_list = build_fragments_list(boot_info) | ||||
|         if self.params.get('test', False): | ||||
| @@ -298,64 +320,114 @@ class F4mFD(FileDownloader): | ||||
|         # For some akamai manifests we'll need to add a query to the fragment url | ||||
|         akamai_pv = xpath_text(doc, _add_ns('pv-2.0')) | ||||
|  | ||||
|         self.report_destination(filename) | ||||
|         http_dl = HttpQuietDownloader( | ||||
|             self.ydl, | ||||
|             { | ||||
|                 'continuedl': True, | ||||
|                 'quiet': True, | ||||
|                 'noprogress': True, | ||||
|                 'ratelimit': self.params.get('ratelimit', None), | ||||
|                 'test': self.params.get('test', False), | ||||
|             } | ||||
|         ) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
|         (dest_stream, tmpfilename) = sanitize_open(tmpfilename, 'wb') | ||||
|  | ||||
|         write_flv_header(dest_stream) | ||||
|         write_metadata_tag(dest_stream, metadata) | ||||
|         if not live: | ||||
|             write_metadata_tag(dest_stream, metadata) | ||||
|  | ||||
|         # This dict stores the download progress, it's updated by the progress | ||||
|         # hook | ||||
|         state = { | ||||
|             'status': 'downloading', | ||||
|             'downloaded_bytes': 0, | ||||
|             'frag_counter': 0, | ||||
|             'frag_index': 0, | ||||
|             'frag_count': total_frags, | ||||
|             'filename': filename, | ||||
|             'tmpfilename': tmpfilename, | ||||
|         } | ||||
|         start = time.time() | ||||
|  | ||||
|         def frag_progress_hook(status): | ||||
|             frag_total_bytes = status.get('total_bytes', 0) | ||||
|             estimated_size = (state['downloaded_bytes'] + | ||||
|                               (total_frags - state['frag_counter']) * frag_total_bytes) | ||||
|             if status['status'] == 'finished': | ||||
|         def frag_progress_hook(s): | ||||
|             if s['status'] not in ('downloading', 'finished'): | ||||
|                 return | ||||
|  | ||||
|             frag_total_bytes = s.get('total_bytes', 0) | ||||
|             if s['status'] == 'finished': | ||||
|                 state['downloaded_bytes'] += frag_total_bytes | ||||
|                 state['frag_counter'] += 1 | ||||
|                 progress = self.calc_percent(state['frag_counter'], total_frags) | ||||
|                 byte_counter = state['downloaded_bytes'] | ||||
|                 state['frag_index'] += 1 | ||||
|  | ||||
|             estimated_size = ( | ||||
|                 (state['downloaded_bytes'] + frag_total_bytes) / | ||||
|                 (state['frag_index'] + 1) * total_frags) | ||||
|             time_now = time.time() | ||||
|             state['total_bytes_estimate'] = estimated_size | ||||
|             state['elapsed'] = time_now - start | ||||
|  | ||||
|             if s['status'] == 'finished': | ||||
|                 progress = self.calc_percent(state['frag_index'], total_frags) | ||||
|             else: | ||||
|                 frag_downloaded_bytes = status['downloaded_bytes'] | ||||
|                 byte_counter = state['downloaded_bytes'] + frag_downloaded_bytes | ||||
|                 frag_downloaded_bytes = s['downloaded_bytes'] | ||||
|                 frag_progress = self.calc_percent(frag_downloaded_bytes, | ||||
|                                                   frag_total_bytes) | ||||
|                 progress = self.calc_percent(state['frag_counter'], total_frags) | ||||
|                 progress = self.calc_percent(state['frag_index'], total_frags) | ||||
|                 progress += frag_progress / float(total_frags) | ||||
|  | ||||
|             eta = self.calc_eta(start, time.time(), estimated_size, byte_counter) | ||||
|             self.report_progress(progress, format_bytes(estimated_size), | ||||
|                                  status.get('speed'), eta) | ||||
|                 state['eta'] = self.calc_eta( | ||||
|                     start, time_now, estimated_size, state['downloaded_bytes'] + frag_downloaded_bytes) | ||||
|                 state['speed'] = s.get('speed') | ||||
|             self._hook_progress(state) | ||||
|  | ||||
|         http_dl.add_progress_hook(frag_progress_hook) | ||||
|  | ||||
|         frags_filenames = [] | ||||
|         for (seg_i, frag_i) in fragments_list: | ||||
|         while fragments_list: | ||||
|             seg_i, frag_i = fragments_list.pop(0) | ||||
|             name = 'Seg%d-Frag%d' % (seg_i, frag_i) | ||||
|             url = base_url + name | ||||
|             if akamai_pv: | ||||
|                 url += '?' + akamai_pv.strip(';') | ||||
|             if info_dict.get('extra_param_to_segment_url'): | ||||
|                 url += info_dict.get('extra_param_to_segment_url') | ||||
|             frag_filename = '%s-%s' % (tmpfilename, name) | ||||
|             success = http_dl.download(frag_filename, {'url': url}) | ||||
|             if not success: | ||||
|                 return False | ||||
|             with open(frag_filename, 'rb') as down: | ||||
|                 down_data = down.read() | ||||
|                 reader = FlvReader(down_data) | ||||
|                 while True: | ||||
|                     _, box_type, box_data = reader.read_box_info() | ||||
|                     if box_type == b'mdat': | ||||
|                         dest_stream.write(box_data) | ||||
|                         break | ||||
|             frags_filenames.append(frag_filename) | ||||
|             try: | ||||
|                 success = http_dl.download(frag_filename, {'url': url}) | ||||
|                 if not success: | ||||
|                     return False | ||||
|                 with open(frag_filename, 'rb') as down: | ||||
|                     down_data = down.read() | ||||
|                     reader = FlvReader(down_data) | ||||
|                     while True: | ||||
|                         _, box_type, box_data = reader.read_box_info() | ||||
|                         if box_type == b'mdat': | ||||
|                             dest_stream.write(box_data) | ||||
|                             break | ||||
|                 if live: | ||||
|                     os.remove(frag_filename) | ||||
|                 else: | ||||
|                     frags_filenames.append(frag_filename) | ||||
|             except (compat_urllib_error.HTTPError, ) as err: | ||||
|                 if live and (err.code == 404 or err.code == 410): | ||||
|                     # We didn't keep up with the live window. Continue | ||||
|                     # with the next available fragment. | ||||
|                     msg = 'Fragment %d unavailable' % frag_i | ||||
|                     self.report_warning(msg) | ||||
|                     fragments_list = [] | ||||
|                 else: | ||||
|                     raise | ||||
|  | ||||
|             if not fragments_list and live and bootstrap_url: | ||||
|                 fragments_list = self._update_live_fragments(bootstrap_url, frag_i) | ||||
|                 total_frags += len(fragments_list) | ||||
|                 if fragments_list and (fragments_list[0][1] > frag_i + 1): | ||||
|                     msg = 'Missed %d fragments' % (fragments_list[0][1] - (frag_i + 1)) | ||||
|                     self.report_warning(msg) | ||||
|  | ||||
|         dest_stream.close() | ||||
|         self.report_finish(format_bytes(state['downloaded_bytes']), time.time() - start) | ||||
|  | ||||
|         elapsed = time.time() - start | ||||
|         self.try_rename(tmpfilename, filename) | ||||
|         for frag_file in frags_filenames: | ||||
|             os.remove(frag_file) | ||||
| @@ -366,6 +438,7 @@ class F4mFD(FileDownloader): | ||||
|             'total_bytes': fsize, | ||||
|             'filename': filename, | ||||
|             'status': 'finished', | ||||
|             'elapsed': elapsed, | ||||
|         }) | ||||
|  | ||||
|         return True | ||||
|   | ||||
| @@ -23,15 +23,14 @@ class HlsFD(FileDownloader): | ||||
|         tmpfilename = self.temp_name(filename) | ||||
|  | ||||
|         ffpp = FFmpegPostProcessor(downloader=self) | ||||
|         program = ffpp._executable | ||||
|         if program is None: | ||||
|         if not ffpp.available: | ||||
|             self.report_error('m3u8 download detected but ffmpeg or avconv could not be found. Please install one.') | ||||
|             return False | ||||
|         ffpp.check_version() | ||||
|  | ||||
|         args = [ | ||||
|             encodeArgument(opt) | ||||
|             for opt in (program, '-y', '-i', url, '-f', 'mp4', '-c', 'copy', '-bsf:a', 'aac_adtstoasc')] | ||||
|             for opt in (ffpp.executable, '-y', '-i', url, '-f', 'mp4', '-c', 'copy', '-bsf:a', 'aac_adtstoasc')] | ||||
|         args.append(encodeFilename(tmpfilename, True)) | ||||
|  | ||||
|         retval = subprocess.call(args) | ||||
| @@ -48,7 +47,7 @@ class HlsFD(FileDownloader): | ||||
|             return True | ||||
|         else: | ||||
|             self.to_stderr('\n') | ||||
|             self.report_error('%s exited with code %d' % (program, retval)) | ||||
|             self.report_error('%s exited with code %d' % (ffpp.basename, retval)) | ||||
|             return False | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -1,10 +1,9 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import os | ||||
| import time | ||||
|  | ||||
| from socket import error as SocketError | ||||
| import errno | ||||
| import os | ||||
| import socket | ||||
| import time | ||||
|  | ||||
| from .common import FileDownloader | ||||
| from ..compat import ( | ||||
| @@ -15,7 +14,6 @@ from ..utils import ( | ||||
|     ContentTooShortError, | ||||
|     encodeFilename, | ||||
|     sanitize_open, | ||||
|     format_bytes, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -30,13 +28,8 @@ class HttpFD(FileDownloader): | ||||
|         add_headers = info_dict.get('http_headers') | ||||
|         if add_headers: | ||||
|             headers.update(add_headers) | ||||
|         data = info_dict.get('http_post_data') | ||||
|         http_method = info_dict.get('http_method') | ||||
|         basic_request = compat_urllib_request.Request(url, data, headers) | ||||
|         request = compat_urllib_request.Request(url, data, headers) | ||||
|         if http_method is not None: | ||||
|             basic_request.get_method = lambda: http_method | ||||
|             request.get_method = lambda: http_method | ||||
|         basic_request = compat_urllib_request.Request(url, None, headers) | ||||
|         request = compat_urllib_request.Request(url, None, headers) | ||||
|  | ||||
|         is_test = self.params.get('test', False) | ||||
|  | ||||
| @@ -51,7 +44,7 @@ class HttpFD(FileDownloader): | ||||
|  | ||||
|         open_mode = 'wb' | ||||
|         if resume_len != 0: | ||||
|             if self.params.get('continuedl', False): | ||||
|             if self.params.get('continuedl', True): | ||||
|                 self.report_resuming_byte(resume_len) | ||||
|                 request.add_header('Range', 'bytes=%d-' % resume_len) | ||||
|                 open_mode = 'ab' | ||||
| @@ -94,6 +87,8 @@ class HttpFD(FileDownloader): | ||||
|                             self._hook_progress({ | ||||
|                                 'filename': filename, | ||||
|                                 'status': 'finished', | ||||
|                                 'downloaded_bytes': resume_len, | ||||
|                                 'total_bytes': resume_len, | ||||
|                             }) | ||||
|                             return True | ||||
|                         else: | ||||
| @@ -102,7 +97,7 @@ class HttpFD(FileDownloader): | ||||
|                             resume_len = 0 | ||||
|                             open_mode = 'wb' | ||||
|                             break | ||||
|             except SocketError as e: | ||||
|             except socket.error as e: | ||||
|                 if e.errno != errno.ECONNRESET: | ||||
|                     # Connection reset is no problem, just retry | ||||
|                     raise | ||||
| @@ -137,7 +132,6 @@ class HttpFD(FileDownloader): | ||||
|                 self.to_screen('\r[download] File is larger than max-filesize (%s bytes > %s bytes). Aborting.' % (data_len, max_data_len)) | ||||
|                 return False | ||||
|  | ||||
|         data_len_str = format_bytes(data_len) | ||||
|         byte_counter = 0 + resume_len | ||||
|         block_size = self.params.get('buffersize', 1024) | ||||
|         start = time.time() | ||||
| @@ -196,20 +190,19 @@ class HttpFD(FileDownloader): | ||||
|             # Progress message | ||||
|             speed = self.calc_speed(start, now, byte_counter - resume_len) | ||||
|             if data_len is None: | ||||
|                 eta = percent = None | ||||
|                 eta = None | ||||
|             else: | ||||
|                 percent = self.calc_percent(byte_counter, data_len) | ||||
|                 eta = self.calc_eta(start, time.time(), data_len - resume_len, byte_counter - resume_len) | ||||
|             self.report_progress(percent, data_len_str, speed, eta) | ||||
|  | ||||
|             self._hook_progress({ | ||||
|                 'status': 'downloading', | ||||
|                 'downloaded_bytes': byte_counter, | ||||
|                 'total_bytes': data_len, | ||||
|                 'tmpfilename': tmpfilename, | ||||
|                 'filename': filename, | ||||
|                 'status': 'downloading', | ||||
|                 'eta': eta, | ||||
|                 'speed': speed, | ||||
|                 'elapsed': now - start, | ||||
|             }) | ||||
|  | ||||
|             if is_test and byte_counter == data_len: | ||||
| @@ -221,7 +214,7 @@ class HttpFD(FileDownloader): | ||||
|             return False | ||||
|         if tmpfilename != '-': | ||||
|             stream.close() | ||||
|         self.report_finish(data_len_str, (time.time() - start)) | ||||
|  | ||||
|         if data_len is not None and byte_counter != data_len: | ||||
|             raise ContentTooShortError(byte_counter, int(data_len)) | ||||
|         self.try_rename(tmpfilename, filename) | ||||
| @@ -235,6 +228,7 @@ class HttpFD(FileDownloader): | ||||
|             'total_bytes': byte_counter, | ||||
|             'filename': filename, | ||||
|             'status': 'finished', | ||||
|             'elapsed': time.time() - start, | ||||
|         }) | ||||
|  | ||||
|         return True | ||||
|   | ||||
| @@ -3,7 +3,6 @@ from __future__ import unicode_literals | ||||
| import os | ||||
| import re | ||||
| import subprocess | ||||
| import sys | ||||
| import time | ||||
|  | ||||
| from .common import FileDownloader | ||||
| @@ -11,7 +10,7 @@ from ..compat import compat_str | ||||
| from ..utils import ( | ||||
|     check_executable, | ||||
|     encodeFilename, | ||||
|     format_bytes, | ||||
|     encodeArgument, | ||||
|     get_exe_version, | ||||
| ) | ||||
|  | ||||
| @@ -51,23 +50,23 @@ class RtmpFD(FileDownloader): | ||||
|                     if not resume_percent: | ||||
|                         resume_percent = percent | ||||
|                         resume_downloaded_data_len = downloaded_data_len | ||||
|                     eta = self.calc_eta(start, time.time(), 100 - resume_percent, percent - resume_percent) | ||||
|                     speed = self.calc_speed(start, time.time(), downloaded_data_len - resume_downloaded_data_len) | ||||
|                     time_now = time.time() | ||||
|                     eta = self.calc_eta(start, time_now, 100 - resume_percent, percent - resume_percent) | ||||
|                     speed = self.calc_speed(start, time_now, downloaded_data_len - resume_downloaded_data_len) | ||||
|                     data_len = None | ||||
|                     if percent > 0: | ||||
|                         data_len = int(downloaded_data_len * 100 / percent) | ||||
|                     data_len_str = '~' + format_bytes(data_len) | ||||
|                     self.report_progress(percent, data_len_str, speed, eta) | ||||
|                     cursor_in_new_line = False | ||||
|                     self._hook_progress({ | ||||
|                         'status': 'downloading', | ||||
|                         'downloaded_bytes': downloaded_data_len, | ||||
|                         'total_bytes': data_len, | ||||
|                         'total_bytes_estimate': data_len, | ||||
|                         'tmpfilename': tmpfilename, | ||||
|                         'filename': filename, | ||||
|                         'status': 'downloading', | ||||
|                         'eta': eta, | ||||
|                         'elapsed': time_now - start, | ||||
|                         'speed': speed, | ||||
|                     }) | ||||
|                     cursor_in_new_line = False | ||||
|                 else: | ||||
|                     # no percent for live streams | ||||
|                     mobj = re.search(r'([0-9]+\.[0-9]{3}) kB / [0-9]+\.[0-9]{2} sec', line) | ||||
| @@ -75,15 +74,15 @@ class RtmpFD(FileDownloader): | ||||
|                         downloaded_data_len = int(float(mobj.group(1)) * 1024) | ||||
|                         time_now = time.time() | ||||
|                         speed = self.calc_speed(start, time_now, downloaded_data_len) | ||||
|                         self.report_progress_live_stream(downloaded_data_len, speed, time_now - start) | ||||
|                         cursor_in_new_line = False | ||||
|                         self._hook_progress({ | ||||
|                             'downloaded_bytes': downloaded_data_len, | ||||
|                             'tmpfilename': tmpfilename, | ||||
|                             'filename': filename, | ||||
|                             'status': 'downloading', | ||||
|                             'elapsed': time_now - start, | ||||
|                             'speed': speed, | ||||
|                         }) | ||||
|                         cursor_in_new_line = False | ||||
|                     elif self.params.get('verbose', False): | ||||
|                         if not cursor_in_new_line: | ||||
|                             self.to_screen('') | ||||
| @@ -106,7 +105,7 @@ class RtmpFD(FileDownloader): | ||||
|         protocol = info_dict.get('rtmp_protocol', None) | ||||
|         real_time = info_dict.get('rtmp_real_time', False) | ||||
|         no_resume = info_dict.get('no_resume', False) | ||||
|         continue_dl = info_dict.get('continuedl', False) | ||||
|         continue_dl = info_dict.get('continuedl', True) | ||||
|  | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
| @@ -120,7 +119,9 @@ class RtmpFD(FileDownloader): | ||||
|         # Download using rtmpdump. rtmpdump returns exit code 2 when | ||||
|         # the connection was interrumpted and resuming appears to be | ||||
|         # possible. This is part of rtmpdump's normal usage, AFAIK. | ||||
|         basic_args = ['rtmpdump', '--verbose', '-r', url, '-o', tmpfilename] | ||||
|         basic_args = [ | ||||
|             'rtmpdump', '--verbose', '-r', url, | ||||
|             '-o', tmpfilename] | ||||
|         if player_url is not None: | ||||
|             basic_args += ['--swfVfy', player_url] | ||||
|         if page_url is not None: | ||||
| @@ -130,7 +131,7 @@ class RtmpFD(FileDownloader): | ||||
|         if play_path is not None: | ||||
|             basic_args += ['--playpath', play_path] | ||||
|         if tc_url is not None: | ||||
|             basic_args += ['--tcUrl', url] | ||||
|             basic_args += ['--tcUrl', tc_url] | ||||
|         if test: | ||||
|             basic_args += ['--stop', '1'] | ||||
|         if flash_version is not None: | ||||
| @@ -153,16 +154,9 @@ class RtmpFD(FileDownloader): | ||||
|         if not live and continue_dl: | ||||
|             args += ['--skip', '1'] | ||||
|  | ||||
|         if sys.platform == 'win32' and sys.version_info < (3, 0): | ||||
|             # Windows subprocess module does not actually support Unicode | ||||
|             # on Python 2.x | ||||
|             # See http://stackoverflow.com/a/9951851/35070 | ||||
|             subprocess_encoding = sys.getfilesystemencoding() | ||||
|             args = [a.encode(subprocess_encoding, 'ignore') for a in args] | ||||
|         else: | ||||
|             subprocess_encoding = None | ||||
|         args = [encodeArgument(a) for a in args] | ||||
|  | ||||
|         self._debug_cmd(args, subprocess_encoding, exe='rtmpdump') | ||||
|         self._debug_cmd(args, exe='rtmpdump') | ||||
|  | ||||
|         RD_SUCCESS = 0 | ||||
|         RD_FAILED = 1 | ||||
| @@ -179,7 +173,11 @@ class RtmpFD(FileDownloader): | ||||
|             prevsize = os.path.getsize(encodeFilename(tmpfilename)) | ||||
|             self.to_screen('[rtmpdump] %s bytes' % prevsize) | ||||
|             time.sleep(5.0)  # This seems to be needed | ||||
|             retval = run_rtmpdump(basic_args + ['-e'] + [[], ['-k', '1']][retval == RD_FAILED]) | ||||
|             args = basic_args + ['--resume'] | ||||
|             if retval == RD_FAILED: | ||||
|                 args += ['--skip', '1'] | ||||
|             args = [encodeArgument(a) for a in args] | ||||
|             retval = run_rtmpdump(args) | ||||
|             cursize = os.path.getsize(encodeFilename(tmpfilename)) | ||||
|             if prevsize == cursize and retval == RD_FAILED: | ||||
|                 break | ||||
|   | ||||
| @@ -10,21 +10,23 @@ from ..utils import ( | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| class MplayerFD(FileDownloader): | ||||
| class RtspFD(FileDownloader): | ||||
|     def real_download(self, filename, info_dict): | ||||
|         url = info_dict['url'] | ||||
|         self.report_destination(filename) | ||||
|         tmpfilename = self.temp_name(filename) | ||||
| 
 | ||||
|         args = [ | ||||
|             'mplayer', '-really-quiet', '-vo', 'null', '-vc', 'dummy', | ||||
|             '-dumpstream', '-dumpfile', tmpfilename, url] | ||||
|         # Check for mplayer first | ||||
|         if not check_executable('mplayer', ['-h']): | ||||
|             self.report_error('MMS or RTSP download detected but "%s" could not be run' % args[0]) | ||||
|         if check_executable('mplayer', ['-h']): | ||||
|             args = [ | ||||
|                 'mplayer', '-really-quiet', '-vo', 'null', '-vc', 'dummy', | ||||
|                 '-dumpstream', '-dumpfile', tmpfilename, url] | ||||
|         elif check_executable('mpv', ['-h']): | ||||
|             args = [ | ||||
|                 'mpv', '-really-quiet', '--vo=null', '--stream-dump=' + tmpfilename, url] | ||||
|         else: | ||||
|             self.report_error('MMS or RTSP download detected but neither "mplayer" nor "mpv" could be run. Please install any.') | ||||
|             return False | ||||
| 
 | ||||
|         # Download using mplayer. | ||||
|         retval = subprocess.call(args) | ||||
|         if retval == 0: | ||||
|             fsize = os.path.getsize(encodeFilename(tmpfilename)) | ||||
| @@ -39,5 +41,5 @@ class MplayerFD(FileDownloader): | ||||
|             return True | ||||
|         else: | ||||
|             self.to_stderr('\n') | ||||
|             self.report_error('mplayer exited with code %d' % retval) | ||||
|             self.report_error('%s exited with code %d' % (args[0], retval)) | ||||
|             return False | ||||
| @@ -8,6 +8,7 @@ from .adobetv import AdobeTVIE | ||||
| from .adultswim import AdultSwimIE | ||||
| from .aftenposten import AftenpostenIE | ||||
| from .aftonbladet import AftonbladetIE | ||||
| from .airmozilla import AirMozillaIE | ||||
| from .aljazeera import AlJazeeraIE | ||||
| from .alphaporno import AlphaPornoIE | ||||
| from .anitube import AnitubeIE | ||||
| @@ -31,11 +32,13 @@ from .atresplayer import AtresPlayerIE | ||||
| from .atttechchannel import ATTTechChannelIE | ||||
| from .audiomack import AudiomackIE, AudiomackAlbumIE | ||||
| from .azubu import AzubuIE | ||||
| from .baidu import BaiduVideoIE | ||||
| from .bambuser import BambuserIE, BambuserChannelIE | ||||
| from .bandcamp import BandcampIE, BandcampAlbumIE | ||||
| from .bbccouk import BBCCoUkIE | ||||
| from .beeg import BeegIE | ||||
| from .behindkink import BehindKinkIE | ||||
| from .beatportpro import BeatportProIE | ||||
| from .bet import BetIE | ||||
| from .bild import BildIE | ||||
| from .bilibili import BiliBiliIE | ||||
| @@ -49,15 +52,26 @@ from .brightcove import BrightcoveIE | ||||
| from .buzzfeed import BuzzFeedIE | ||||
| from .byutv import BYUtvIE | ||||
| from .c56 import C56IE | ||||
| from .camdemy import ( | ||||
|     CamdemyIE, | ||||
|     CamdemyFolderIE | ||||
| ) | ||||
| from .canal13cl import Canal13clIE | ||||
| from .canalplus import CanalplusIE | ||||
| from .canalc2 import Canalc2IE | ||||
| from .cbs import CBSIE | ||||
| from .cbsnews import CBSNewsIE | ||||
| from .cbssports import CBSSportsIE | ||||
| from .ccc import CCCIE | ||||
| from .ceskatelevize import CeskaTelevizeIE | ||||
| from .channel9 import Channel9IE | ||||
| from .chilloutzone import ChilloutzoneIE | ||||
| from .chirbit import ( | ||||
|     ChirbitIE, | ||||
|     ChirbitProfileIE, | ||||
| ) | ||||
| from .cinchcast import CinchcastIE | ||||
| from .cinemassacre import CinemassacreIE | ||||
| from .clipfish import ClipfishIE | ||||
| from .cliphunter import CliphunterIE | ||||
| from .clipsyndicate import ClipsyndicateIE | ||||
| @@ -74,10 +88,11 @@ from .collegehumor import CollegeHumorIE | ||||
| from .collegerama import CollegeRamaIE | ||||
| from .comedycentral import ComedyCentralIE, ComedyCentralShowsIE | ||||
| from .comcarcoff import ComCarCoffIE | ||||
| from .commonmistakes import CommonMistakesIE | ||||
| from .commonmistakes import CommonMistakesIE, UnicodeBOMIE | ||||
| from .condenast import CondeNastIE | ||||
| from .cracked import CrackedIE | ||||
| from .criterion import CriterionIE | ||||
| from .crooksandliars import CrooksAndLiarsIE | ||||
| from .crunchyroll import ( | ||||
|     CrunchyrollIE, | ||||
|     CrunchyrollShowPlaylistIE | ||||
| @@ -94,17 +109,25 @@ from .dbtv import DBTVIE | ||||
| from .dctp import DctpTvIE | ||||
| from .deezer import DeezerPlaylistIE | ||||
| from .dfb import DFBIE | ||||
| from .dhm import DHMIE | ||||
| from .dotsub import DotsubIE | ||||
| from .douyutv import DouyuTVIE | ||||
| from .dramafever import ( | ||||
|     DramaFeverIE, | ||||
|     DramaFeverSeriesIE, | ||||
| ) | ||||
| from .dreisat import DreiSatIE | ||||
| from .drbonanza import DRBonanzaIE | ||||
| from .drtuber import DrTuberIE | ||||
| from .drtv import DRTVIE | ||||
| from .dvtv import DVTVIE | ||||
| from .dump import DumpIE | ||||
| from .dumpert import DumpertIE | ||||
| from .defense import DefenseGouvFrIE | ||||
| from .discovery import DiscoveryIE | ||||
| from .divxstage import DivxStageIE | ||||
| from .dropbox import DropboxIE | ||||
| from .eagleplatform import EaglePlatformIE | ||||
| from .ebaumsworld import EbaumsWorldIE | ||||
| from .echomsk import EchoMskIE | ||||
| from .ehow import EHowIE | ||||
| @@ -116,11 +139,13 @@ from .ellentv import ( | ||||
|     EllenTVClipsIE, | ||||
| ) | ||||
| from .elpais import ElPaisIE | ||||
| from .embedly import EmbedlyIE | ||||
| from .empflix import EMPFlixIE | ||||
| from .engadget import EngadgetIE | ||||
| from .eporner import EpornerIE | ||||
| from .eroprofile import EroProfileIE | ||||
| from .escapist import EscapistIE | ||||
| from .espn import ESPNIE | ||||
| from .everyonesmixtape import EveryonesMixtapeIE | ||||
| from .exfm import ExfmIE | ||||
| from .expotv import ExpoTVIE | ||||
| @@ -128,19 +153,21 @@ from .extremetube import ExtremeTubeIE | ||||
| from .facebook import FacebookIE | ||||
| from .faz import FazIE | ||||
| from .fc2 import FC2IE | ||||
| from .firedrive import FiredriveIE | ||||
| from .firstpost import FirstpostIE | ||||
| from .firsttv import FirstTVIE | ||||
| from .fivemin import FiveMinIE | ||||
| from .fivetv import FiveTVIE | ||||
| from .fktv import ( | ||||
|     FKTVIE, | ||||
|     FKTVPosteckeIE, | ||||
| ) | ||||
| from .flickr import FlickrIE | ||||
| from .folketinget import FolketingetIE | ||||
| from .footyroom import FootyRoomIE | ||||
| from .fourtube import FourTubeIE | ||||
| from .foxgay import FoxgayIE | ||||
| from .foxnews import FoxNewsIE | ||||
| from .foxsports import FoxSportsIE | ||||
| from .franceculture import FranceCultureIE | ||||
| from .franceinter import FranceInterIE | ||||
| from .francetv import ( | ||||
| @@ -159,11 +186,14 @@ from .gameone import ( | ||||
|     GameOneIE, | ||||
|     GameOnePlaylistIE, | ||||
| ) | ||||
| from .gamersyde import GamersydeIE | ||||
| from .gamespot import GameSpotIE | ||||
| from .gamestar import GameStarIE | ||||
| from .gametrailers import GametrailersIE | ||||
| from .gazeta import GazetaIE | ||||
| from .gdcvault import GDCVaultIE | ||||
| from .generic import GenericIE | ||||
| from .gfycat import GfycatIE | ||||
| from .giantbomb import GiantBombIE | ||||
| from .giga import GigaIE | ||||
| from .glide import GlideIE | ||||
| @@ -175,7 +205,6 @@ from .googleplus import GooglePlusIE | ||||
| from .googlesearch import GoogleSearchIE | ||||
| from .gorillavid import GorillaVidIE | ||||
| from .goshgay import GoshgayIE | ||||
| from .grooveshark import GroovesharkIE | ||||
| from .groupon import GrouponIE | ||||
| from .hark import HarkIE | ||||
| from .hearthisat import HearThisAtIE | ||||
| @@ -184,6 +213,7 @@ from .hellporno import HellPornoIE | ||||
| from .helsinki import HelsinkiIE | ||||
| from .hentaistigma import HentaiStigmaIE | ||||
| from .historicfilms import HistoricFilmsIE | ||||
| from .history import HistoryIE | ||||
| from .hitbox import HitboxIE, HitboxLiveIE | ||||
| from .hornbunny import HornBunnyIE | ||||
| from .hostingbulk import HostingBulkIE | ||||
| @@ -198,11 +228,13 @@ from .imdb import ( | ||||
|     ImdbIE, | ||||
|     ImdbListIE | ||||
| ) | ||||
| from .imgur import ImgurIE | ||||
| from .ina import InaIE | ||||
| from .infoq import InfoQIE | ||||
| from .instagram import InstagramIE, InstagramUserIE | ||||
| from .internetvideoarchive import InternetVideoArchiveIE | ||||
| from .iprima import IPrimaIE | ||||
| from .iqiyi import IqiyiIE | ||||
| from .ivi import ( | ||||
|     IviIE, | ||||
|     IviCompilationIE | ||||
| @@ -213,8 +245,11 @@ from .jeuxvideo import JeuxVideoIE | ||||
| from .jove import JoveIE | ||||
| from .jukebox import JukeboxIE | ||||
| from .jpopsukitv import JpopsukiIE | ||||
| from .kaltura import KalturaIE | ||||
| from .kanalplay import KanalPlayIE | ||||
| from .kankan import KankanIE | ||||
| from .karaoketv import KaraoketvIE | ||||
| from .karrierevideos import KarriereVideosIE | ||||
| from .keezmovies import KeezMoviesIE | ||||
| from .khanacademy import KhanAcademyIE | ||||
| from .kickstarter import KickStarterIE | ||||
| @@ -224,7 +259,16 @@ from .krasview import KrasViewIE | ||||
| from .ku6 import Ku6IE | ||||
| from .la7 import LA7IE | ||||
| from .laola1tv import Laola1TvIE | ||||
| from .lifenews import LifeNewsIE | ||||
| from .letv import ( | ||||
|     LetvIE, | ||||
|     LetvTvIE, | ||||
|     LetvPlaylistIE | ||||
| ) | ||||
| from .libsyn import LibsynIE | ||||
| from .lifenews import ( | ||||
|     LifeNewsIE, | ||||
|     LifeEmbedIE, | ||||
| ) | ||||
| from .liveleak import LiveLeakIE | ||||
| from .livestream import ( | ||||
|     LivestreamIE, | ||||
| @@ -242,11 +286,13 @@ from .macgamestore import MacGameStoreIE | ||||
| from .mailru import MailRuIE | ||||
| from .malemotion import MalemotionIE | ||||
| from .mdr import MDRIE | ||||
| from .megavideoz import MegaVideozIE | ||||
| from .metacafe import MetacafeIE | ||||
| from .metacritic import MetacriticIE | ||||
| from .mgoon import MgoonIE | ||||
| from .minhateca import MinhatecaIE | ||||
| from .ministrygrid import MinistryGridIE | ||||
| from .miomio import MioMioIE | ||||
| from .mit import TechTVMITIE, MITIE, OCWMITIE | ||||
| from .mitele import MiTeleIE | ||||
| from .mixcloud import MixcloudIE | ||||
| @@ -276,13 +322,19 @@ from .myspace import MySpaceIE, MySpaceAlbumIE | ||||
| from .myspass import MySpassIE | ||||
| from .myvideo import MyVideoIE | ||||
| from .myvidster import MyVidsterIE | ||||
| from .nationalgeographic import NationalGeographicIE | ||||
| from .naver import NaverIE | ||||
| from .nba import NBAIE | ||||
| from .nbc import ( | ||||
|     NBCIE, | ||||
|     NBCNewsIE, | ||||
|     NBCSportsIE, | ||||
|     NBCSportsVPlayerIE, | ||||
| ) | ||||
| from .ndr import ( | ||||
|     NDRIE, | ||||
|     NJoyIE, | ||||
| ) | ||||
| from .ndr import NDRIE | ||||
| from .ndtv import NDTVIE | ||||
| from .netzkino import NetzkinoIE | ||||
| from .nerdcubed import NerdCubedFeedIE | ||||
| @@ -292,8 +344,7 @@ from .newstube import NewstubeIE | ||||
| from .nextmedia import ( | ||||
|     NextMediaIE, | ||||
|     NextMediaActionNewsIE, | ||||
|     AppleDailyRealtimeNewsIE, | ||||
|     AppleDailyAnimationNewsIE | ||||
|     AppleDailyIE, | ||||
| ) | ||||
| from .nfb import NFBIE | ||||
| from .nfl import NFLIE | ||||
| @@ -307,49 +358,80 @@ from .ninegag import NineGagIE | ||||
| from .noco import NocoIE | ||||
| from .normalboots import NormalbootsIE | ||||
| from .nosvideo import NosVideoIE | ||||
| from .nova import NovaIE | ||||
| from .novamov import NovaMovIE | ||||
| from .nowness import NownessIE | ||||
| from .nowtv import NowTVIE | ||||
| from .nowvideo import NowVideoIE | ||||
| from .npo import ( | ||||
|     NPOIE, | ||||
|     NPOLiveIE, | ||||
|     NPORadioIE, | ||||
|     NPORadioFragmentIE, | ||||
|     TegenlichtVproIE, | ||||
| ) | ||||
| from .nrk import ( | ||||
|     NRKIE, | ||||
|     NRKPlaylistIE, | ||||
|     NRKTVIE, | ||||
| ) | ||||
| from .ntvde import NTVDeIE | ||||
| from .ntvru import NTVRuIE | ||||
| from .nytimes import NYTimesIE | ||||
| from .nytimes import ( | ||||
|     NYTimesIE, | ||||
|     NYTimesArticleIE, | ||||
| ) | ||||
| from .nuvid import NuvidIE | ||||
| from .odnoklassniki import OdnoklassnikiIE | ||||
| from .oktoberfesttv import OktoberfestTVIE | ||||
| from .ooyala import OoyalaIE | ||||
| from .ooyala import ( | ||||
|     OoyalaIE, | ||||
|     OoyalaExternalIE, | ||||
| ) | ||||
| from .openfilm import OpenFilmIE | ||||
| from .orf import ( | ||||
|     ORFTVthekIE, | ||||
|     ORFOE1IE, | ||||
|     ORFFM4IE, | ||||
|     ORFIPTVIE, | ||||
| ) | ||||
| from .parliamentliveuk import ParliamentLiveUKIE | ||||
| from .patreon import PatreonIE | ||||
| from .pbs import PBSIE | ||||
| from .philharmoniedeparis import PhilharmonieDeParisIE | ||||
| from .phoenix import PhoenixIE | ||||
| from .photobucket import PhotobucketIE | ||||
| from .planetaplay import PlanetaPlayIE | ||||
| from .pladform import PladformIE | ||||
| from .played import PlayedIE | ||||
| from .playfm import PlayFMIE | ||||
| from .playvid import PlayvidIE | ||||
| from .playwire import PlaywireIE | ||||
| from .podomatic import PodomaticIE | ||||
| from .porn91 import Porn91IE | ||||
| from .pornhd import PornHdIE | ||||
| from .pornhub import PornHubIE | ||||
| from .pornhub import ( | ||||
|     PornHubIE, | ||||
|     PornHubPlaylistIE, | ||||
| ) | ||||
| from .pornotube import PornotubeIE | ||||
| from .pornovoisines import PornoVoisinesIE | ||||
| from .pornoxo import PornoXOIE | ||||
| from .primesharetv import PrimeShareTVIE | ||||
| from .promptfile import PromptFileIE | ||||
| from .prosiebensat1 import ProSiebenSat1IE | ||||
| from .puls4 import Puls4IE | ||||
| from .pyvideo import PyvideoIE | ||||
| from .qqmusic import ( | ||||
|     QQMusicIE, | ||||
|     QQMusicSingerIE, | ||||
|     QQMusicAlbumIE, | ||||
|     QQMusicToplistIE, | ||||
| ) | ||||
| from .quickvid import QuickVidIE | ||||
| from .r7 import R7IE | ||||
| from .radiode import RadioDeIE | ||||
| from .radiojavan import RadioJavanIE | ||||
| from .radiobremen import RadioBremenIE | ||||
| from .radiofrance import RadioFranceIE | ||||
| from .rai import RaiIE | ||||
| @@ -363,12 +445,11 @@ from .rottentomatoes import RottenTomatoesIE | ||||
| from .roxwel import RoxwelIE | ||||
| from .rtbf import RTBFIE | ||||
| from .rte import RteIE | ||||
| from .rtlnl import RtlXlIE | ||||
| from .rtlnow import RTLnowIE | ||||
| from .rtlnl import RtlNlIE | ||||
| from .rtl2 import RTL2IE | ||||
| from .rtp import RTPIE | ||||
| from .rts import RTSIE | ||||
| from .rtve import RTVEALaCartaIE, RTVELiveIE | ||||
| from .rtve import RTVEALaCartaIE, RTVELiveIE, RTVEInfantilIE | ||||
| from .ruhd import RUHDIE | ||||
| from .rutube import ( | ||||
|     RutubeIE, | ||||
| @@ -378,13 +459,20 @@ from .rutube import ( | ||||
|     RutubePersonIE, | ||||
| ) | ||||
| from .rutv import RUTVIE | ||||
| from .ruutu import RuutuIE | ||||
| from .sandia import SandiaIE | ||||
| from .safari import ( | ||||
|     SafariIE, | ||||
|     SafariCourseIE, | ||||
| ) | ||||
| from .sapo import SapoIE | ||||
| from .savefrom import SaveFromIE | ||||
| from .sbs import SBSIE | ||||
| from .scivee import SciVeeIE | ||||
| from .screencast import ScreencastIE | ||||
| from .screencastomatic import ScreencastOMaticIE | ||||
| from .screenwavemedia import CinemassacreIE, ScreenwaveMediaIE, TeamFourIE | ||||
| from .screenwavemedia import ScreenwaveMediaIE, TeamFourIE | ||||
| from .senateisvp import SenateISVPIE | ||||
| from .servingsys import ServingSysIE | ||||
| from .sexu import SexuIE | ||||
| from .sexykarma import SexyKarmaIE | ||||
| @@ -400,34 +488,53 @@ from .smotri import ( | ||||
|     SmotriBroadcastIE, | ||||
| ) | ||||
| from .snotr import SnotrIE | ||||
| from .sockshare import SockshareIE | ||||
| from .sohu import SohuIE | ||||
| from .soompi import ( | ||||
|     SoompiIE, | ||||
|     SoompiShowIE, | ||||
| ) | ||||
| from .soundcloud import ( | ||||
|     SoundcloudIE, | ||||
|     SoundcloudSetIE, | ||||
|     SoundcloudUserIE, | ||||
|     SoundcloudPlaylistIE | ||||
| ) | ||||
| from .soundgasm import SoundgasmIE | ||||
| from .soundgasm import ( | ||||
|     SoundgasmIE, | ||||
|     SoundgasmProfileIE | ||||
| ) | ||||
| from .southpark import ( | ||||
|     SouthParkIE, | ||||
|     SouthparkDeIE, | ||||
|     SouthParkDeIE, | ||||
|     SouthParkDkIE, | ||||
|     SouthParkEsIE, | ||||
|     SouthParkNlIE | ||||
| ) | ||||
| from .space import SpaceIE | ||||
| from .spankbang import SpankBangIE | ||||
| from .spankwire import SpankwireIE | ||||
| from .spiegel import SpiegelIE, SpiegelArticleIE | ||||
| from .spiegeltv import SpiegeltvIE | ||||
| from .spike import SpikeIE | ||||
| from .sport5 import Sport5IE | ||||
| from .sportbox import SportBoxIE | ||||
| from .sportbox import ( | ||||
|     SportBoxIE, | ||||
|     SportBoxEmbedIE, | ||||
| ) | ||||
| from .sportdeutschland import SportDeutschlandIE | ||||
| from .srf import SrfIE | ||||
| from .srmediathek import SRMediathekIE | ||||
| from .ssa import SSAIE | ||||
| from .stanfordoc import StanfordOpenClassroomIE | ||||
| from .steam import SteamIE | ||||
| from .streamcloud import StreamcloudIE | ||||
| from .streamcz import StreamCZIE | ||||
| from .streetvoice import StreetVoiceIE | ||||
| from .sunporno import SunPornoIE | ||||
| from .svt import ( | ||||
|     SVTIE, | ||||
|     SVTPlayIE, | ||||
| ) | ||||
| from .swrmediathek import SWRMediathekIE | ||||
| from .syfy import SyfyIE | ||||
| from .sztvhu import SztvHuIE | ||||
| @@ -456,7 +563,10 @@ from .thesixtyone import TheSixtyOneIE | ||||
| from .thisav import ThisAVIE | ||||
| from .tinypic import TinyPicIE | ||||
| from .tlc import TlcIE, TlcDeIE | ||||
| from .tmz import TMZIE | ||||
| from .tmz import ( | ||||
|     TMZIE, | ||||
|     TMZArticleIE, | ||||
| ) | ||||
| from .tnaflix import TNAFlixIE | ||||
| from .thvideo import ( | ||||
|     THVideoIE, | ||||
| @@ -468,16 +578,30 @@ from .traileraddict import TrailerAddictIE | ||||
| from .trilulilu import TriluliluIE | ||||
| from .trutube import TruTubeIE | ||||
| from .tube8 import Tube8IE | ||||
| from .tubitv import TubiTvIE | ||||
| from .tudou import TudouIE | ||||
| from .tumblr import TumblrIE | ||||
| from .tunein import TuneInIE | ||||
| from .turbo import TurboIE | ||||
| from .tutv import TutvIE | ||||
| from .tv2 import ( | ||||
|     TV2IE, | ||||
|     TV2ArticleIE, | ||||
| ) | ||||
| from .tv4 import TV4IE | ||||
| from .tvc import ( | ||||
|     TVCIE, | ||||
|     TVCArticleIE, | ||||
| ) | ||||
| from .tvigle import TvigleIE | ||||
| from .tvp import TvpIE, TvpSeriesIE | ||||
| from .tvplay import TVPlayIE | ||||
| from .tweakers import TweakersIE | ||||
| from .twentyfourvideo import TwentyFourVideoIE | ||||
| from .twentytwotracks import ( | ||||
|     TwentyTwoTracksIE, | ||||
|     TwentyTwoTracksGenreIE | ||||
| ) | ||||
| from .twitch import ( | ||||
|     TwitchVideoIE, | ||||
|     TwitchChapterIE, | ||||
| @@ -492,15 +616,23 @@ from .udemy import ( | ||||
|     UdemyIE, | ||||
|     UdemyCourseIE | ||||
| ) | ||||
| from .udn import UDNEmbedIE | ||||
| from .ultimedia import UltimediaIE | ||||
| from .unistra import UnistraIE | ||||
| from .urort import UrortIE | ||||
| from .ustream import UstreamIE, UstreamChannelIE | ||||
| from .varzesh3 import Varzesh3IE | ||||
| from .vbox7 import Vbox7IE | ||||
| from .veehd import VeeHDIE | ||||
| from .veoh import VeohIE | ||||
| from .vessel import VesselIE | ||||
| from .vesti import VestiIE | ||||
| from .vevo import VevoIE | ||||
| from .vgtv import VGTVIE | ||||
| from .vgtv import ( | ||||
|     BTArticleIE, | ||||
|     BTVestlendingenIE, | ||||
|     VGTVIE, | ||||
| ) | ||||
| from .vh1 import VH1IE | ||||
| from .vice import ViceIE | ||||
| from .viddler import ViddlerIE | ||||
| @@ -515,6 +647,7 @@ from .videoweed import VideoWeedIE | ||||
| from .vidme import VidmeIE | ||||
| from .vidzi import VidziIE | ||||
| from .vier import VierIE, VierVideosIE | ||||
| from .viewster import ViewsterIE | ||||
| from .vimeo import ( | ||||
|     VimeoIE, | ||||
|     VimeoAlbumIE, | ||||
| @@ -530,12 +663,16 @@ from .vine import ( | ||||
|     VineIE, | ||||
|     VineUserIE, | ||||
| ) | ||||
| from .viki import VikiIE | ||||
| from .viki import ( | ||||
|     VikiIE, | ||||
|     VikiChannelIE, | ||||
| ) | ||||
| from .vk import ( | ||||
|     VKIE, | ||||
|     VKUserVideosIE, | ||||
| ) | ||||
| from .vodlocker import VodlockerIE | ||||
| from .voicerepublic import VoiceRepublicIE | ||||
| from .vporn import VpornIE | ||||
| from .vrt import VRTIE | ||||
| from .vube import VubeIE | ||||
| @@ -562,14 +699,21 @@ from .xboxclips import XboxClipsIE | ||||
| from .xhamster import XHamsterIE | ||||
| from .xminus import XMinusIE | ||||
| from .xnxx import XNXXIE | ||||
| from .xvideos import XVideosIE | ||||
| from .xstream import XstreamIE | ||||
| from .xtube import XTubeUserIE, XTubeIE | ||||
| from .xuite import XuiteIE | ||||
| from .xvideos import XVideosIE | ||||
| from .xxxymovies import XXXYMoviesIE | ||||
| from .yahoo import ( | ||||
|     YahooIE, | ||||
|     YahooSearchIE, | ||||
| ) | ||||
| from .yam import YamIE | ||||
| from .yandexmusic import ( | ||||
|     YandexMusicTrackIE, | ||||
|     YandexMusicAlbumIE, | ||||
|     YandexMusicPlaylistIE, | ||||
| ) | ||||
| from .yesjapan import YesJapanIE | ||||
| from .ynet import YnetIE | ||||
| from .youjizz import YouJizzIE | ||||
| @@ -593,6 +737,7 @@ from .youtube import ( | ||||
|     YoutubeUserIE, | ||||
|     YoutubeWatchLaterIE, | ||||
| ) | ||||
| from .zapiks import ZapiksIE | ||||
| from .zdf import ZDFIE, ZDFChannelIE | ||||
| from .zingmp3 import ( | ||||
|     ZingMp3SongIE, | ||||
|   | ||||
| @@ -11,12 +11,13 @@ from ..compat import ( | ||||
| ) | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     qualities, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class AddAnimeIE(InfoExtractor): | ||||
|     _VALID_URL = r'^http://(?:\w+\.)?add-anime\.net/watch_video\.php\?(?:.*?)v=(?P<id>[\w_]+)(?:.*)' | ||||
|     _TEST = { | ||||
|     _VALID_URL = r'http://(?:\w+\.)?add-anime\.net/(?:watch_video\.php\?(?:.*?)v=|video/)(?P<id>[\w_]+)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.add-anime.net/watch_video.php?v=24MR3YO5SAS9', | ||||
|         'md5': '72954ea10bc979ab5e2eb288b21425a0', | ||||
|         'info_dict': { | ||||
| @@ -25,7 +26,10 @@ class AddAnimeIE(InfoExtractor): | ||||
|             'description': 'One Piece 606', | ||||
|             'title': 'One Piece 606', | ||||
|         } | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://add-anime.net/video/MDUGWYKNGBD8/One-Piece-687', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
| @@ -63,8 +67,10 @@ class AddAnimeIE(InfoExtractor): | ||||
|                 note='Confirming after redirect') | ||||
|             webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         FORMATS = ('normal', 'hq') | ||||
|         quality = qualities(FORMATS) | ||||
|         formats = [] | ||||
|         for format_id in ('normal', 'hq'): | ||||
|         for format_id in FORMATS: | ||||
|             rex = r"var %s_video_file = '(.*?)';" % re.escape(format_id) | ||||
|             video_url = self._search_regex(rex, webpage, 'video file URLx', | ||||
|                                            fatal=False) | ||||
| @@ -73,6 +79,7 @@ class AddAnimeIE(InfoExtractor): | ||||
|             formats.append({ | ||||
|                 'format_id': format_id, | ||||
|                 'url': video_url, | ||||
|                 'quality': quality(format_id), | ||||
|             }) | ||||
|         self._sort_formats(formats) | ||||
|         video_title = self._og_search_title(webpage) | ||||
|   | ||||
| @@ -28,7 +28,6 @@ class AdobeTVIE(InfoExtractor): | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         player = self._parse_json( | ||||
| @@ -44,8 +43,10 @@ class AdobeTVIE(InfoExtractor): | ||||
|             self._html_search_meta('datepublished', webpage, 'upload date')) | ||||
|  | ||||
|         duration = parse_duration( | ||||
|             self._html_search_meta('duration', webpage, 'duration') | ||||
|             or self._search_regex(r'Runtime:\s*(\d{2}:\d{2}:\d{2})', webpage, 'duration')) | ||||
|             self._html_search_meta('duration', webpage, 'duration') or | ||||
|             self._search_regex( | ||||
|                 r'Runtime:\s*(\d{2}:\d{2}:\d{2})', | ||||
|                 webpage, 'duration', fatal=False)) | ||||
|  | ||||
|         view_count = str_to_int(self._search_regex( | ||||
|             r'<div class="views">\s*Views?:\s*([\d,.]+)\s*</div>', | ||||
|   | ||||
| @@ -2,13 +2,12 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     xpath_text, | ||||
|     float_or_none, | ||||
|     xpath_text, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -38,6 +37,7 @@ class AdultSwimIE(InfoExtractor): | ||||
|             }, | ||||
|         ], | ||||
|         'info_dict': { | ||||
|             'id': 'rQxZvXQ4ROaSOqq-or2Mow', | ||||
|             'title': 'Rick and Morty - Pilot', | ||||
|             'description': "Rick moves in with his daughter's family and establishes himself as a bad influence on his grandson, Morty. " | ||||
|         } | ||||
| @@ -55,9 +55,28 @@ class AdultSwimIE(InfoExtractor): | ||||
|             } | ||||
|         ], | ||||
|         'info_dict': { | ||||
|             'id': '-t8CamQlQ2aYZ49ItZCFog', | ||||
|             'title': 'American Dad - Putting Francine Out of Business', | ||||
|             'description': 'Stan hatches a plan to get Francine out of the real estate business.Watch more American Dad on [adult swim].' | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'http://www.adultswim.com/videos/tim-and-eric-awesome-show-great-job/dr-steve-brule-for-your-wine/', | ||||
|         'playlist': [ | ||||
|             { | ||||
|                 'md5': '3e346a2ab0087d687a05e1e7f3b3e529', | ||||
|                 'info_dict': { | ||||
|                     'id': 'sY3cMUR_TbuE4YmdjzbIcQ-0', | ||||
|                     'ext': 'flv', | ||||
|                     'title': 'Tim and Eric Awesome Show Great Job! - Dr. Steve Brule, For Your Wine', | ||||
|                     'description': 'Dr. Brule reports live from Wine Country with a special report on wines.  \r\nWatch Tim and Eric Awesome Show Great Job! episode #20, "Embarrassed" on Adult Swim.\r\n\r\n', | ||||
|                 }, | ||||
|             } | ||||
|         ], | ||||
|         'info_dict': { | ||||
|             'id': 'sY3cMUR_TbuE4YmdjzbIcQ', | ||||
|             'title': 'Tim and Eric Awesome Show Great Job! - Dr. Steve Brule, For Your Wine', | ||||
|             'description': 'Dr. Brule reports live from Wine Country with a special report on wines.  \r\nWatch Tim and Eric Awesome Show Great Job! episode #20, "Embarrassed" on Adult Swim.\r\n\r\n', | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     @staticmethod | ||||
| @@ -78,6 +97,7 @@ class AdultSwimIE(InfoExtractor): | ||||
|             for video in collection.get('videos'): | ||||
|                 if video.get('slug') == slug: | ||||
|                     return collection, video | ||||
|         return None, None | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -88,28 +108,30 @@ class AdultSwimIE(InfoExtractor): | ||||
|         webpage = self._download_webpage(url, episode_path) | ||||
|  | ||||
|         # Extract the value of `bootstrappedData` from the Javascript in the page. | ||||
|         bootstrappedDataJS = self._search_regex(r'var bootstrappedData = ({.*});', webpage, episode_path) | ||||
|  | ||||
|         try: | ||||
|             bootstrappedData = json.loads(bootstrappedDataJS) | ||||
|         except ValueError as ve: | ||||
|             errmsg = '%s: Failed to parse JSON ' % episode_path | ||||
|             raise ExtractorError(errmsg, cause=ve) | ||||
|         bootstrapped_data = self._parse_json(self._search_regex( | ||||
|             r'var bootstrappedData = ({.*});', webpage, 'bootstraped data'), episode_path) | ||||
|  | ||||
|         # Downloading videos from a /videos/playlist/ URL needs to be handled differently. | ||||
|         # NOTE: We are only downloading one video (the current one) not the playlist | ||||
|         if is_playlist: | ||||
|             collections = bootstrappedData['playlists']['collections'] | ||||
|             collections = bootstrapped_data['playlists']['collections'] | ||||
|             collection = self.find_collection_by_linkURL(collections, show_path) | ||||
|             video_info = self.find_video_info(collection, episode_path) | ||||
|  | ||||
|             show_title = video_info['showTitle'] | ||||
|             segment_ids = [video_info['videoPlaybackID']] | ||||
|         else: | ||||
|             collections = bootstrappedData['show']['collections'] | ||||
|             collections = bootstrapped_data['show']['collections'] | ||||
|             collection, video_info = self.find_collection_containing_video(collections, episode_path) | ||||
|  | ||||
|             show = bootstrappedData['show'] | ||||
|             # Video wasn't found in the collections, let's try `slugged_video`. | ||||
|             if video_info is None: | ||||
|                 if bootstrapped_data.get('slugged_video', {}).get('slug') == episode_path: | ||||
|                     video_info = bootstrapped_data['slugged_video'] | ||||
|                 else: | ||||
|                     raise ExtractorError('Unable to find video info') | ||||
|  | ||||
|             show = bootstrapped_data['show'] | ||||
|             show_title = show['title'] | ||||
|             segment_ids = [clip['videoPlaybackID'] for clip in video_info['clips']] | ||||
|  | ||||
|   | ||||
| @@ -1,23 +1,13 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     parse_iso8601, | ||||
|     xpath_with_ns, | ||||
|     xpath_text, | ||||
|     find_xpath_attr, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class AftenpostenIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?aftenposten\.no/webtv/([^/]+/)*(?P<id>[^/]+)-\d+\.html' | ||||
|  | ||||
|     _VALID_URL = r'https?://(?:www\.)?aftenposten\.no/webtv/(?:#!/)?video/(?P<id>\d+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.aftenposten.no/webtv/serier-og-programmer/sweatshopenglish/TRAILER-SWEATSHOP---I-cant-take-any-more-7800835.html?paging=§ion=webtv_serierogprogrammer_sweatshop_sweatshopenglish', | ||||
|         'url': 'http://www.aftenposten.no/webtv/#!/video/21039/trailer-sweatshop-i-can-t-take-any-more', | ||||
|         'md5': 'fd828cd29774a729bf4d4425fe192972', | ||||
|         'info_dict': { | ||||
|             'id': '21039', | ||||
| @@ -30,74 +20,4 @@ class AftenpostenIE(InfoExtractor): | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         display_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|  | ||||
|         video_id = self._html_search_regex( | ||||
|             r'data-xs-id="(\d+)"', webpage, 'video id') | ||||
|  | ||||
|         data = self._download_xml( | ||||
|             'http://frontend.xstream.dk/ap/feed/video/?platform=web&id=%s' % video_id, video_id) | ||||
|  | ||||
|         NS_MAP = { | ||||
|             'atom': 'http://www.w3.org/2005/Atom', | ||||
|             'xt': 'http://xstream.dk/', | ||||
|             'media': 'http://search.yahoo.com/mrss/', | ||||
|         } | ||||
|  | ||||
|         entry = data.find(xpath_with_ns('./atom:entry', NS_MAP)) | ||||
|  | ||||
|         title = xpath_text( | ||||
|             entry, xpath_with_ns('./atom:title', NS_MAP), 'title') | ||||
|         description = xpath_text( | ||||
|             entry, xpath_with_ns('./atom:summary', NS_MAP), 'description') | ||||
|         timestamp = parse_iso8601(xpath_text( | ||||
|             entry, xpath_with_ns('./atom:published', NS_MAP), 'upload date')) | ||||
|  | ||||
|         formats = [] | ||||
|         media_group = entry.find(xpath_with_ns('./media:group', NS_MAP)) | ||||
|         for media_content in media_group.findall(xpath_with_ns('./media:content', NS_MAP)): | ||||
|             media_url = media_content.get('url') | ||||
|             if not media_url: | ||||
|                 continue | ||||
|             tbr = int_or_none(media_content.get('bitrate')) | ||||
|             mobj = re.search(r'^(?P<url>rtmp://[^/]+/(?P<app>[^/]+))/(?P<playpath>.+)$', media_url) | ||||
|             if mobj: | ||||
|                 formats.append({ | ||||
|                     'url': mobj.group('url'), | ||||
|                     'play_path': 'mp4:%s' % mobj.group('playpath'), | ||||
|                     'app': mobj.group('app'), | ||||
|                     'ext': 'flv', | ||||
|                     'tbr': tbr, | ||||
|                     'format_id': 'rtmp-%d' % tbr, | ||||
|                 }) | ||||
|             else: | ||||
|                 formats.append({ | ||||
|                     'url': media_url, | ||||
|                     'tbr': tbr, | ||||
|                 }) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         link = find_xpath_attr( | ||||
|             entry, xpath_with_ns('./atom:link', NS_MAP), 'rel', 'original') | ||||
|         if link is not None: | ||||
|             formats.append({ | ||||
|                 'url': link.get('href'), | ||||
|                 'format_id': link.get('rel'), | ||||
|             }) | ||||
|  | ||||
|         thumbnails = [{ | ||||
|             'url': splash.get('url'), | ||||
|             'width': int_or_none(splash.get('width')), | ||||
|             'height': int_or_none(splash.get('height')), | ||||
|         } for splash in media_group.findall(xpath_with_ns('./xt:splash', NS_MAP))] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'timestamp': timestamp, | ||||
|             'formats': formats, | ||||
|             'thumbnails': thumbnails, | ||||
|         } | ||||
|         return self.url_result('xstream:ap:%s' % self._match_id(url), 'Xstream') | ||||
|   | ||||
| @@ -2,14 +2,15 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import int_or_none | ||||
|  | ||||
|  | ||||
| class AftonbladetIE(InfoExtractor): | ||||
|     _VALID_URL = r'^http://tv\.aftonbladet\.se/webbtv.+?(?P<video_id>article[0-9]+)\.ab(?:$|[?#])' | ||||
|     _VALID_URL = r'http://tv\.aftonbladet\.se/abtv/articles/(?P<id>[0-9]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://tv.aftonbladet.se/webbtv/nyheter/vetenskap/rymden/article36015.ab', | ||||
|         'url': 'http://tv.aftonbladet.se/abtv/articles/36015', | ||||
|         'info_dict': { | ||||
|             'id': 'article36015', | ||||
|             'id': '36015', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Vulkanutbrott i rymden - nu släpper NASA bilderna', | ||||
|             'description': 'Jupiters måne mest aktiv av alla himlakroppar', | ||||
| @@ -24,8 +25,9 @@ class AftonbladetIE(InfoExtractor): | ||||
|  | ||||
|         # find internal video meta data | ||||
|         meta_url = 'http://aftonbladet-play.drlib.aptoma.no/video/%s.json' | ||||
|         internal_meta_id = self._html_search_regex( | ||||
|             r'data-aptomaId="([\w\d]+)"', webpage, 'internal_meta_id') | ||||
|         player_config = self._parse_json(self._html_search_regex( | ||||
|             r'data-player-config="([^"]+)"', webpage, 'player config'), video_id) | ||||
|         internal_meta_id = player_config['videoId'] | ||||
|         internal_meta_url = meta_url % internal_meta_id | ||||
|         internal_meta_json = self._download_json( | ||||
|             internal_meta_url, video_id, 'Downloading video meta data') | ||||
| @@ -43,9 +45,9 @@ class AftonbladetIE(InfoExtractor): | ||||
|             formats.append({ | ||||
|                 'url': 'http://%s:%d/%s/%s' % (p['address'], p['port'], p['path'], p['filename']), | ||||
|                 'ext': 'mp4', | ||||
|                 'width': fmt['width'], | ||||
|                 'height': fmt['height'], | ||||
|                 'tbr': fmt['bitrate'], | ||||
|                 'width': int_or_none(fmt.get('width')), | ||||
|                 'height': int_or_none(fmt.get('height')), | ||||
|                 'tbr': int_or_none(fmt.get('bitrate')), | ||||
|                 'protocol': 'http', | ||||
|             }) | ||||
|         self._sort_formats(formats) | ||||
| @@ -54,9 +56,9 @@ class AftonbladetIE(InfoExtractor): | ||||
|             'id': video_id, | ||||
|             'title': internal_meta_json['title'], | ||||
|             'formats': formats, | ||||
|             'thumbnail': internal_meta_json['imageUrl'], | ||||
|             'description': internal_meta_json['shortPreamble'], | ||||
|             'timestamp': internal_meta_json['timePublished'], | ||||
|             'duration': internal_meta_json['duration'], | ||||
|             'view_count': internal_meta_json['views'], | ||||
|             'thumbnail': internal_meta_json.get('imageUrl'), | ||||
|             'description': internal_meta_json.get('shortPreamble'), | ||||
|             'timestamp': int_or_none(internal_meta_json.get('timePublished')), | ||||
|             'duration': int_or_none(internal_meta_json.get('duration')), | ||||
|             'view_count': int_or_none(internal_meta_json.get('views')), | ||||
|         } | ||||
|   | ||||
							
								
								
									
										74
									
								
								youtube_dl/extractor/airmozilla.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								youtube_dl/extractor/airmozilla.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,74 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     parse_duration, | ||||
|     parse_iso8601, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class AirMozillaIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://air\.mozilla\.org/(?P<id>[0-9a-z-]+)/?' | ||||
|     _TEST = { | ||||
|         'url': 'https://air.mozilla.org/privacy-lab-a-meetup-for-privacy-minded-people-in-san-francisco/', | ||||
|         'md5': '2e3e7486ba5d180e829d453875b9b8bf', | ||||
|         'info_dict': { | ||||
|             'id': '6x4q2w', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Privacy Lab - a meetup for privacy minded people in San Francisco', | ||||
|             'thumbnail': 're:https://\w+\.cloudfront\.net/6x4q2w/poster\.jpg\?t=\d+', | ||||
|             'description': 'Brings together privacy professionals and others interested in privacy at for-profits, non-profits, and NGOs in an effort to contribute to the state of the ecosystem...', | ||||
|             'timestamp': 1422487800, | ||||
|             'upload_date': '20150128', | ||||
|             'location': 'SFO Commons', | ||||
|             'duration': 3780, | ||||
|             'view_count': int, | ||||
|             'categories': ['Main'], | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         display_id = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         video_id = self._html_search_regex(r'//vid.ly/(.*?)/embed', webpage, 'id') | ||||
|  | ||||
|         embed_script = self._download_webpage('https://vid.ly/{0}/embed'.format(video_id), video_id) | ||||
|         jwconfig = self._search_regex(r'\svar jwconfig = (\{.*?\});\s', embed_script, 'metadata') | ||||
|         metadata = self._parse_json(jwconfig, video_id) | ||||
|  | ||||
|         formats = [{ | ||||
|             'url': source['file'], | ||||
|             'ext': source['type'], | ||||
|             'format_id': self._search_regex(r'&format=(.*)$', source['file'], 'video format'), | ||||
|             'format': source['label'], | ||||
|             'height': int(source['label'].rstrip('p')), | ||||
|         } for source in metadata['playlist'][0]['sources']] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         view_count = int_or_none(self._html_search_regex( | ||||
|             r'Views since archived: ([0-9]+)', | ||||
|             webpage, 'view count', fatal=False)) | ||||
|         timestamp = parse_iso8601(self._html_search_regex( | ||||
|             r'<time datetime="(.*?)"', webpage, 'timestamp', fatal=False)) | ||||
|         duration = parse_duration(self._search_regex( | ||||
|             r'Duration:\s*(\d+\s*hours?\s*\d+\s*minutes?)', | ||||
|             webpage, 'duration', fatal=False)) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': self._og_search_title(webpage), | ||||
|             'formats': formats, | ||||
|             'url': self._og_search_url(webpage), | ||||
|             'display_id': display_id, | ||||
|             'thumbnail': metadata['playlist'][0].get('image'), | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'timestamp': timestamp, | ||||
|             'location': self._html_search_regex(r'Location: (.*)', webpage, 'location', default=None), | ||||
|             'duration': duration, | ||||
|             'view_count': view_count, | ||||
|             'categories': re.findall(r'<a href=".*?" class="channel">(.*?)</a>', webpage), | ||||
|         } | ||||
| @@ -11,9 +11,12 @@ from ..utils import ( | ||||
|  | ||||
|  | ||||
| class AppleTrailersIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?trailers\.apple\.com/trailers/(?P<company>[^/]+)/(?P<movie>[^/]+)' | ||||
|     _TEST = { | ||||
|     _VALID_URL = r'https?://(?:www\.)?trailers\.apple\.com/(?:trailers|ca)/(?P<company>[^/]+)/(?P<movie>[^/]+)' | ||||
|     _TESTS = [{ | ||||
|         "url": "http://trailers.apple.com/trailers/wb/manofsteel/", | ||||
|         'info_dict': { | ||||
|             'id': 'manofsteel', | ||||
|         }, | ||||
|         "playlist": [ | ||||
|             { | ||||
|                 "md5": "d97a8e575432dbcb81b7c3acb741f8a8", | ||||
| @@ -60,7 +63,10 @@ class AppleTrailersIE(InfoExtractor): | ||||
|                 }, | ||||
|             }, | ||||
|         ] | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://trailers.apple.com/ca/metropole/autrui/', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     _JSON_RE = r'iTunes.playURL\((.*?)\);' | ||||
|  | ||||
|   | ||||
| @@ -33,7 +33,7 @@ class ArchiveOrgIE(InfoExtractor): | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         json_url = url + ('?' if '?' in url else '&') + 'output=json' | ||||
|         json_url = url + ('&' if '?' in url else '?') + 'output=json' | ||||
|         data = self._download_json(json_url, video_id) | ||||
|  | ||||
|         def get_optional(data_dict, field): | ||||
|   | ||||
| @@ -50,6 +50,9 @@ class ARDMediathekIE(InfoExtractor): | ||||
|         if '>Der gewünschte Beitrag ist nicht mehr verfügbar.<' in webpage: | ||||
|             raise ExtractorError('Video %s is no longer available' % video_id, expected=True) | ||||
|  | ||||
|         if 'Diese Sendung ist für Jugendliche unter 12 Jahren nicht geeignet. Der Clip ist deshalb nur von 20 bis 6 Uhr verfügbar.' in webpage: | ||||
|             raise ExtractorError('This program is only suitable for those aged 12 and older. Video %s is therefore only available between 20 pm and 6 am.' % video_id, expected=True) | ||||
|  | ||||
|         if re.search(r'[\?&]rss($|[=&])', url): | ||||
|             doc = parse_xml(webpage) | ||||
|             if doc.tag == 'rss': | ||||
|   | ||||
| @@ -7,7 +7,6 @@ from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     find_xpath_attr, | ||||
|     unified_strdate, | ||||
|     get_element_by_id, | ||||
|     get_element_by_attribute, | ||||
|     int_or_none, | ||||
|     qualities, | ||||
| @@ -146,6 +145,7 @@ class ArteTVPlus7IE(InfoExtractor): | ||||
|  | ||||
|             formats.append(format) | ||||
|  | ||||
|         self._check_formats(formats, video_id) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         info_dict['formats'] = formats | ||||
| @@ -194,7 +194,9 @@ class ArteTVFutureIE(ArteTVPlus7IE): | ||||
|     def _real_extract(self, url): | ||||
|         anchor_id, lang = self._extract_url_info(url) | ||||
|         webpage = self._download_webpage(url, anchor_id) | ||||
|         row = get_element_by_id(anchor_id, webpage) | ||||
|         row = self._search_regex( | ||||
|             r'(?s)id="%s"[^>]*>.+?(<div[^>]*arte_vp_url[^>]*>)' % anchor_id, | ||||
|             webpage, 'row') | ||||
|         return self._extract_from_webpage(row, anchor_id, lang) | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -3,7 +3,7 @@ from __future__ import unicode_literals | ||||
| import time | ||||
| import hmac | ||||
|  | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_str, | ||||
|     compat_urllib_parse, | ||||
| @@ -17,8 +17,9 @@ from ..utils import ( | ||||
| ) | ||||
|  | ||||
|  | ||||
| class AtresPlayerIE(SubtitlesInfoExtractor): | ||||
| class AtresPlayerIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?atresplayer\.com/television/[^/]+/[^/]+/[^/]+/(?P<id>.+?)_\d+\.html' | ||||
|     _NETRC_MACHINE = 'atresplayer' | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.atresplayer.com/television/programas/el-club-de-la-comedia/temporada-4/capitulo-10-especial-solidario-nochebuena_2014122100174.html', | ||||
| @@ -144,13 +145,12 @@ class AtresPlayerIE(SubtitlesInfoExtractor): | ||||
|         thumbnail = xpath_text(episode, './media/asset/files/background', 'thumbnail') | ||||
|  | ||||
|         subtitles = {} | ||||
|         subtitle = xpath_text(episode, './media/asset/files/subtitle', 'subtitle') | ||||
|         if subtitle: | ||||
|             subtitles['es'] = subtitle | ||||
|  | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, subtitles) | ||||
|             return | ||||
|         subtitle_url = xpath_text(episode, './media/asset/files/subtitle', 'subtitle') | ||||
|         if subtitle_url: | ||||
|             subtitles['es'] = [{ | ||||
|                 'ext': 'srt', | ||||
|                 'url': subtitle_url, | ||||
|             }] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
| @@ -159,5 +159,5 @@ class AtresPlayerIE(SubtitlesInfoExtractor): | ||||
|             'thumbnail': thumbnail, | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|             'subtitles': self.extract_subtitles(video_id, subtitles), | ||||
|             'subtitles': subtitles, | ||||
|         } | ||||
|   | ||||
							
								
								
									
										68
									
								
								youtube_dl/extractor/baidu.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								youtube_dl/extractor/baidu.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import compat_urlparse | ||||
|  | ||||
|  | ||||
| class BaiduVideoIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://v\.baidu\.com/(?P<type>[a-z]+)/(?P<id>\d+)\.htm' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://v.baidu.com/comic/1069.htm?frp=bdbrand&q=%E4%B8%AD%E5%8D%8E%E5%B0%8F%E5%BD%93%E5%AE%B6', | ||||
|         'info_dict': { | ||||
|             'id': '1069', | ||||
|             'title': '中华小当家 TV版 (全52集)', | ||||
|             'description': 'md5:395a419e41215e531c857bb037bbaf80', | ||||
|         }, | ||||
|         'playlist_count': 52, | ||||
|     }, { | ||||
|         'url': 'http://v.baidu.com/show/11595.htm?frp=bdbrand', | ||||
|         'info_dict': { | ||||
|             'id': '11595', | ||||
|             'title': 're:^奔跑吧兄弟', | ||||
|             'description': 'md5:1bf88bad6d850930f542d51547c089b8', | ||||
|         }, | ||||
|         'playlist_mincount': 3, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         playlist_id = mobj.group('id') | ||||
|         category = category2 = mobj.group('type') | ||||
|         if category == 'show': | ||||
|             category2 = 'tvshow' | ||||
|  | ||||
|         webpage = self._download_webpage(url, playlist_id) | ||||
|  | ||||
|         playlist_title = self._html_search_regex( | ||||
|             r'title\s*:\s*(["\'])(?P<title>[^\']+)\1', webpage, | ||||
|             'playlist title', group='title') | ||||
|         playlist_description = self._html_search_regex( | ||||
|             r'<input[^>]+class="j-data-intro"[^>]+value="([^"]+)"/>', webpage, | ||||
|             playlist_id, 'playlist description') | ||||
|  | ||||
|         site = self._html_search_regex( | ||||
|             r'filterSite\s*:\s*["\']([^"]*)["\']', webpage, | ||||
|             'primary provider site') | ||||
|         api_result = self._download_json( | ||||
|             'http://v.baidu.com/%s_intro/?dtype=%sPlayUrl&id=%s&site=%s' % ( | ||||
|                 category, category2, playlist_id, site), | ||||
|             playlist_id, 'Get playlist links') | ||||
|  | ||||
|         entries = [] | ||||
|         for episode in api_result[0]['episodes']: | ||||
|             episode_id = '%s_%s' % (playlist_id, episode['episode']) | ||||
|  | ||||
|             redirect_page = self._download_webpage( | ||||
|                 compat_urlparse.urljoin(url, episode['url']), episode_id, | ||||
|                 note='Download Baidu redirect page') | ||||
|             real_url = self._html_search_regex( | ||||
|                 r'location\.replace\("([^"]+)"\)', redirect_page, 'real URL') | ||||
|  | ||||
|             entries.append(self.url_result( | ||||
|                 real_url, video_title=episode['single_title'])) | ||||
|  | ||||
|         return self.playlist_result( | ||||
|             entries, playlist_id, playlist_title, playlist_description) | ||||
| @@ -1,12 +1,18 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import json | ||||
| import itertools | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_urllib_parse, | ||||
|     compat_urllib_request, | ||||
|     compat_str, | ||||
| ) | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
|     float_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| @@ -14,6 +20,8 @@ class BambuserIE(InfoExtractor): | ||||
|     IE_NAME = 'bambuser' | ||||
|     _VALID_URL = r'https?://bambuser\.com/v/(?P<id>\d+)' | ||||
|     _API_KEY = '005f64509e19a868399060af746a00aa' | ||||
|     _LOGIN_URL = 'https://bambuser.com/user' | ||||
|     _NETRC_MACHINE = 'bambuser' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://bambuser.com/v/4050584', | ||||
| @@ -26,6 +34,9 @@ class BambuserIE(InfoExtractor): | ||||
|             'duration': 3741, | ||||
|             'uploader': 'pixelversity', | ||||
|             'uploader_id': '344706', | ||||
|             'timestamp': 1382976692, | ||||
|             'upload_date': '20131028', | ||||
|             'view_count': int, | ||||
|         }, | ||||
|         'params': { | ||||
|             # It doesn't respect the 'Range' header, it would download the whole video | ||||
| @@ -34,23 +45,60 @@ class BambuserIE(InfoExtractor): | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _login(self): | ||||
|         (username, password) = self._get_login_info() | ||||
|         if username is None: | ||||
|             return | ||||
|  | ||||
|         login_form = { | ||||
|             'form_id': 'user_login', | ||||
|             'op': 'Log in', | ||||
|             'name': username, | ||||
|             'pass': password, | ||||
|         } | ||||
|  | ||||
|         request = compat_urllib_request.Request( | ||||
|             self._LOGIN_URL, compat_urllib_parse.urlencode(login_form).encode('utf-8')) | ||||
|         request.add_header('Referer', self._LOGIN_URL) | ||||
|         response = self._download_webpage( | ||||
|             request, None, 'Logging in as %s' % username) | ||||
|  | ||||
|         login_error = self._html_search_regex( | ||||
|             r'(?s)<div class="messages error">(.+?)</div>', | ||||
|             response, 'login error', default=None) | ||||
|         if login_error: | ||||
|             raise ExtractorError( | ||||
|                 'Unable to login: %s' % login_error, expected=True) | ||||
|  | ||||
|     def _real_initialize(self): | ||||
|         self._login() | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         info_url = ('http://player-c.api.bambuser.com/getVideo.json?' | ||||
|                     '&api_key=%s&vid=%s' % (self._API_KEY, video_id)) | ||||
|         info_json = self._download_webpage(info_url, video_id) | ||||
|         info = json.loads(info_json)['result'] | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         info = self._download_json( | ||||
|             'http://player-c.api.bambuser.com/getVideo.json?api_key=%s&vid=%s' | ||||
|             % (self._API_KEY, video_id), video_id) | ||||
|  | ||||
|         error = info.get('error') | ||||
|         if error: | ||||
|             raise ExtractorError( | ||||
|                 '%s returned error: %s' % (self.IE_NAME, error), expected=True) | ||||
|  | ||||
|         result = info['result'] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': info['title'], | ||||
|             'url': info['url'], | ||||
|             'thumbnail': info.get('preview'), | ||||
|             'duration': int(info['length']), | ||||
|             'view_count': int(info['views_total']), | ||||
|             'uploader': info['username'], | ||||
|             'uploader_id': info['uid'], | ||||
|             'title': result['title'], | ||||
|             'url': result['url'], | ||||
|             'thumbnail': result.get('preview'), | ||||
|             'duration': int_or_none(result.get('length')), | ||||
|             'uploader': result.get('username'), | ||||
|             'uploader_id': compat_str(result.get('owner', {}).get('uid')), | ||||
|             'timestamp': int_or_none(result.get('created')), | ||||
|             'fps': float_or_none(result.get('framerate')), | ||||
|             'view_count': int_or_none(result.get('views_total')), | ||||
|             'comment_count': int_or_none(result.get('comment_count')), | ||||
|         } | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -72,26 +72,29 @@ class BandcampIE(InfoExtractor): | ||||
|  | ||||
|         download_link = m_download.group(1) | ||||
|         video_id = self._search_regex( | ||||
|             r'var TralbumData = {.*?id: (?P<id>\d+),?$', | ||||
|             webpage, 'video id', flags=re.MULTILINE | re.DOTALL) | ||||
|             r'(?ms)var TralbumData = .*?[{,]\s*id: (?P<id>\d+),?$', | ||||
|             webpage, 'video id') | ||||
|  | ||||
|         download_webpage = self._download_webpage(download_link, video_id, 'Downloading free downloads page') | ||||
|         # We get the dictionary of the track from some javascript code | ||||
|         info = re.search(r'items: (.*?),$', download_webpage, re.MULTILINE).group(1) | ||||
|         info = json.loads(info)[0] | ||||
|         all_info = self._parse_json(self._search_regex( | ||||
|             r'(?sm)items: (.*?),$', download_webpage, 'items'), video_id) | ||||
|         info = all_info[0] | ||||
|         # We pick mp3-320 for now, until format selection can be easily implemented. | ||||
|         mp3_info = info['downloads']['mp3-320'] | ||||
|         # If we try to use this url it says the link has expired | ||||
|         initial_url = mp3_info['url'] | ||||
|         re_url = r'(?P<server>http://(.*?)\.bandcamp\.com)/download/track\?enc=mp3-320&fsig=(?P<fsig>.*?)&id=(?P<id>.*?)&ts=(?P<ts>.*)$' | ||||
|         m_url = re.match(re_url, initial_url) | ||||
|         m_url = re.match( | ||||
|             r'(?P<server>http://(.*?)\.bandcamp\.com)/download/track\?enc=mp3-320&fsig=(?P<fsig>.*?)&id=(?P<id>.*?)&ts=(?P<ts>.*)$', | ||||
|             initial_url) | ||||
|         # We build the url we will use to get the final track url | ||||
|         # This url is build in Bandcamp in the script download_bunde_*.js | ||||
|         request_url = '%s/statdownload/track?enc=mp3-320&fsig=%s&id=%s&ts=%s&.rand=665028774616&.vrs=1' % (m_url.group('server'), m_url.group('fsig'), video_id, m_url.group('ts')) | ||||
|         final_url_webpage = self._download_webpage(request_url, video_id, 'Requesting download url') | ||||
|         # If we could correctly generate the .rand field the url would be | ||||
|         # in the "download_url" key | ||||
|         final_url = re.search(r'"retry_url":"(.*?)"', final_url_webpage).group(1) | ||||
|         final_url = self._search_regex( | ||||
|             r'"retry_url":"(.*?)"', final_url_webpage, 'final video URL') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
| @@ -106,7 +109,7 @@ class BandcampIE(InfoExtractor): | ||||
|  | ||||
| class BandcampAlbumIE(InfoExtractor): | ||||
|     IE_NAME = 'Bandcamp:album' | ||||
|     _VALID_URL = r'https?://(?:(?P<subdomain>[^.]+)\.)?bandcamp\.com(?:/album/(?P<title>[^?#]+)|/?(?:$|[?#]))' | ||||
|     _VALID_URL = r'https?://(?:(?P<subdomain>[^.]+)\.)?bandcamp\.com(?:/album/(?P<album_id>[^?#]+)|/?(?:$|[?#]))' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://blazo.bandcamp.com/album/jazz-format-mixtape-vol-1', | ||||
| @@ -130,31 +133,37 @@ class BandcampAlbumIE(InfoExtractor): | ||||
|         ], | ||||
|         'info_dict': { | ||||
|             'title': 'Jazz Format Mixtape vol.1', | ||||
|             'id': 'jazz-format-mixtape-vol-1', | ||||
|             'uploader_id': 'blazo', | ||||
|         }, | ||||
|         'params': { | ||||
|             'playlistend': 2 | ||||
|         }, | ||||
|         'skip': 'Bandcamp imposes download limits. See test_playlists:test_bandcamp_album for the playlist test' | ||||
|         'skip': 'Bandcamp imposes download limits.' | ||||
|     }, { | ||||
|         'url': 'http://nightbringer.bandcamp.com/album/hierophany-of-the-open-grave', | ||||
|         'info_dict': { | ||||
|             'title': 'Hierophany of the Open Grave', | ||||
|             'uploader_id': 'nightbringer', | ||||
|             'id': 'hierophany-of-the-open-grave', | ||||
|         }, | ||||
|         'playlist_mincount': 9, | ||||
|     }, { | ||||
|         'url': 'http://dotscale.bandcamp.com', | ||||
|         'info_dict': { | ||||
|             'title': 'Loom', | ||||
|             'id': 'dotscale', | ||||
|             'uploader_id': 'dotscale', | ||||
|         }, | ||||
|         'playlist_mincount': 7, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         playlist_id = mobj.group('subdomain') | ||||
|         title = mobj.group('title') | ||||
|         display_id = title or playlist_id | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         uploader_id = mobj.group('subdomain') | ||||
|         album_id = mobj.group('album_id') | ||||
|         playlist_id = album_id or uploader_id | ||||
|         webpage = self._download_webpage(url, playlist_id) | ||||
|         tracks_paths = re.findall(r'<a href="(.*?)" itemprop="url">', webpage) | ||||
|         if not tracks_paths: | ||||
|             raise ExtractorError('The page doesn\'t contain any tracks') | ||||
| @@ -165,8 +174,8 @@ class BandcampAlbumIE(InfoExtractor): | ||||
|             r'album_title\s*:\s*"(.*?)"', webpage, 'title', fatal=False) | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'uploader_id': uploader_id, | ||||
|             'id': playlist_id, | ||||
|             'display_id': display_id, | ||||
|             'title': title, | ||||
|             'entries': entries, | ||||
|         } | ||||
|   | ||||
| @@ -2,12 +2,15 @@ from __future__ import unicode_literals | ||||
|  | ||||
| import xml.etree.ElementTree | ||||
|  | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from ..utils import ExtractorError | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     int_or_none, | ||||
| ) | ||||
| from ..compat import compat_HTTPError | ||||
|  | ||||
|  | ||||
| class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
| class BBCCoUkIE(InfoExtractor): | ||||
|     IE_NAME = 'bbc.co.uk' | ||||
|     IE_DESC = 'BBC iPlayer' | ||||
|     _VALID_URL = r'https?://(?:www\.)?bbc\.co\.uk/(?:(?:(?:programmes|iplayer(?:/[^/]+)?/(?:episode|playlist))/)|music/clips[/#])(?P<id>[\da-z]{8})' | ||||
| @@ -112,6 +115,20 @@ class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
|                 # rtmp download | ||||
|                 'skip_download': True, | ||||
|             } | ||||
|         }, { | ||||
|             'url': 'http://www.bbc.co.uk/iplayer/episode/b054fn09/ad/natural-world-20152016-2-super-powered-owls', | ||||
|             'info_dict': { | ||||
|                 'id': 'p02n76xf', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'Natural World, 2015-2016: 2. Super Powered Owls', | ||||
|                 'description': 'md5:e4db5c937d0e95a7c6b5e654d429183d', | ||||
|                 'duration': 3540, | ||||
|             }, | ||||
|             'params': { | ||||
|                 # rtmp download | ||||
|                 'skip_download': True, | ||||
|             }, | ||||
|             'skip': 'geolocation', | ||||
|         }, { | ||||
|             'url': 'http://www.bbc.co.uk/iplayer/playlist/p01dvks4', | ||||
|             'only_matching': True, | ||||
| @@ -215,17 +232,32 @@ class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
|             formats.extend(conn_formats) | ||||
|         return formats | ||||
|  | ||||
|     def _extract_captions(self, media, programme_id): | ||||
|     def _get_subtitles(self, media, programme_id): | ||||
|         subtitles = {} | ||||
|         for connection in self._extract_connections(media): | ||||
|             captions = self._download_xml(connection.get('href'), programme_id, 'Downloading captions') | ||||
|             lang = captions.get('{http://www.w3.org/XML/1998/namespace}lang', 'en') | ||||
|             ps = captions.findall('./{0}body/{0}div/{0}p'.format('{http://www.w3.org/2006/10/ttaf1}')) | ||||
|             srt = '' | ||||
|  | ||||
|             def _extract_text(p): | ||||
|                 if p.text is not None: | ||||
|                     stripped_text = p.text.strip() | ||||
|                     if stripped_text: | ||||
|                         return stripped_text | ||||
|                 return ' '.join(span.text.strip() for span in p.findall('{http://www.w3.org/2006/10/ttaf1}span')) | ||||
|             for pos, p in enumerate(ps): | ||||
|                 srt += '%s\r\n%s --> %s\r\n%s\r\n\r\n' % (str(pos), p.get('begin'), p.get('end'), | ||||
|                                                           p.text.strip() if p.text is not None else '') | ||||
|             subtitles[lang] = srt | ||||
|                 srt += '%s\r\n%s --> %s\r\n%s\r\n\r\n' % (str(pos), p.get('begin'), p.get('end'), _extract_text(p)) | ||||
|             subtitles[lang] = [ | ||||
|                 { | ||||
|                     'url': connection.get('href'), | ||||
|                     'ext': 'ttml', | ||||
|                 }, | ||||
|                 { | ||||
|                     'data': srt, | ||||
|                     'ext': 'srt', | ||||
|                 }, | ||||
|             ] | ||||
|         return subtitles | ||||
|  | ||||
|     def _download_media_selector(self, programme_id): | ||||
| @@ -249,7 +281,7 @@ class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
|             elif kind == 'video': | ||||
|                 formats.extend(self._extract_video(media, programme_id)) | ||||
|             elif kind == 'captions': | ||||
|                 subtitles = self._extract_captions(media, programme_id) | ||||
|                 subtitles = self.extract_subtitles(media, programme_id) | ||||
|  | ||||
|         return formats, subtitles | ||||
|  | ||||
| @@ -273,7 +305,7 @@ class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
|                     formats, subtitles = self._download_media_selector(programme_id) | ||||
|                 return programme_id, title, description, duration, formats, subtitles | ||||
|         except ExtractorError as ee: | ||||
|             if not isinstance(ee.cause, compat_HTTPError) and ee.cause.code == 404: | ||||
|             if not (isinstance(ee.cause, compat_HTTPError) and ee.cause.code == 404): | ||||
|                 raise | ||||
|  | ||||
|         # fallback to legacy playlist | ||||
| @@ -311,29 +343,37 @@ class BBCCoUkIE(SubtitlesInfoExtractor): | ||||
|  | ||||
|         webpage = self._download_webpage(url, group_id, 'Downloading video page') | ||||
|  | ||||
|         programme_id = self._search_regex( | ||||
|             r'"vpid"\s*:\s*"([\da-z]{8})"', webpage, 'vpid', fatal=False, default=None) | ||||
|         programme_id = None | ||||
|  | ||||
|         tviplayer = self._search_regex( | ||||
|             r'mediator\.bind\(({.+?})\s*,\s*document\.getElementById', | ||||
|             webpage, 'player', default=None) | ||||
|  | ||||
|         if tviplayer: | ||||
|             player = self._parse_json(tviplayer, group_id).get('player', {}) | ||||
|             duration = int_or_none(player.get('duration')) | ||||
|             programme_id = player.get('vpid') | ||||
|  | ||||
|         if not programme_id: | ||||
|             programme_id = self._search_regex( | ||||
|                 r'"vpid"\s*:\s*"([\da-z]{8})"', webpage, 'vpid', fatal=False, default=None) | ||||
|  | ||||
|         if programme_id: | ||||
|             player = self._download_json( | ||||
|                 'http://www.bbc.co.uk/iplayer/episode/%s.json' % group_id, | ||||
|                 group_id)['jsConf']['player'] | ||||
|             title = player['title'] | ||||
|             description = player['subtitle'] | ||||
|             duration = player['duration'] | ||||
|             formats, subtitles = self._download_media_selector(programme_id) | ||||
|             title = self._og_search_title(webpage) | ||||
|             description = self._search_regex( | ||||
|                 r'<p class="medium-description">([^<]+)</p>', | ||||
|                 webpage, 'description', fatal=False) | ||||
|         else: | ||||
|             programme_id, title, description, duration, formats, subtitles = self._download_playlist(group_id) | ||||
|  | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(programme_id, subtitles) | ||||
|             return | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': programme_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': self._og_search_thumbnail(webpage, default=None), | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|             'subtitles': subtitles, | ||||
|   | ||||
							
								
								
									
										103
									
								
								youtube_dl/extractor/beatportpro.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								youtube_dl/extractor/beatportpro.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,103 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import compat_str | ||||
| from ..utils import int_or_none | ||||
|  | ||||
|  | ||||
| class BeatportProIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://pro\.beatport\.com/track/(?P<display_id>[^/]+)/(?P<id>[0-9]+)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'https://pro.beatport.com/track/synesthesia-original-mix/5379371', | ||||
|         'md5': 'b3c34d8639a2f6a7f734382358478887', | ||||
|         'info_dict': { | ||||
|             'id': '5379371', | ||||
|             'display_id': 'synesthesia-original-mix', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Froxic - Synesthesia (Original Mix)', | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'https://pro.beatport.com/track/love-and-war-original-mix/3756896', | ||||
|         'md5': 'e44c3025dfa38c6577fbaeb43da43514', | ||||
|         'info_dict': { | ||||
|             'id': '3756896', | ||||
|             'display_id': 'love-and-war-original-mix', | ||||
|             'ext': 'mp3', | ||||
|             'title': 'Wolfgang Gartner - Love & War (Original Mix)', | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'https://pro.beatport.com/track/birds-original-mix/4991738', | ||||
|         'md5': 'a1fd8e8046de3950fd039304c186c05f', | ||||
|         'info_dict': { | ||||
|             'id': '4991738', | ||||
|             'display_id': 'birds-original-mix', | ||||
|             'ext': 'mp4', | ||||
|             'title': "Tos, Middle Milk, Mumblin' Johnsson - Birds (Original Mix)", | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         track_id = mobj.group('id') | ||||
|         display_id = mobj.group('display_id') | ||||
|  | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|  | ||||
|         playables = self._parse_json( | ||||
|             self._search_regex( | ||||
|                 r'window\.Playables\s*=\s*({.+?});', webpage, | ||||
|                 'playables info', flags=re.DOTALL), | ||||
|             track_id) | ||||
|  | ||||
|         track = next(t for t in playables['tracks'] if t['id'] == int(track_id)) | ||||
|  | ||||
|         title = ', '.join((a['name'] for a in track['artists'])) + ' - ' + track['name'] | ||||
|         if track['mix']: | ||||
|             title += ' (' + track['mix'] + ')' | ||||
|  | ||||
|         formats = [] | ||||
|         for ext, info in track['preview'].items(): | ||||
|             if not info['url']: | ||||
|                 continue | ||||
|             fmt = { | ||||
|                 'url': info['url'], | ||||
|                 'ext': ext, | ||||
|                 'format_id': ext, | ||||
|                 'vcodec': 'none', | ||||
|             } | ||||
|             if ext == 'mp3': | ||||
|                 fmt['preference'] = 0 | ||||
|                 fmt['acodec'] = 'mp3' | ||||
|                 fmt['abr'] = 96 | ||||
|                 fmt['asr'] = 44100 | ||||
|             elif ext == 'mp4': | ||||
|                 fmt['preference'] = 1 | ||||
|                 fmt['acodec'] = 'aac' | ||||
|                 fmt['abr'] = 96 | ||||
|                 fmt['asr'] = 44100 | ||||
|             formats.append(fmt) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         images = [] | ||||
|         for name, info in track['images'].items(): | ||||
|             image_url = info.get('url') | ||||
|             if name == 'dynamic' or not image_url: | ||||
|                 continue | ||||
|             image = { | ||||
|                 'id': name, | ||||
|                 'url': image_url, | ||||
|                 'height': int_or_none(info.get('height')), | ||||
|                 'width': int_or_none(info.get('width')), | ||||
|             } | ||||
|             images.append(image) | ||||
|  | ||||
|         return { | ||||
|             'id': compat_str(track.get('id')) or track_id, | ||||
|             'display_id': track.get('slug') or display_id, | ||||
|             'title': title, | ||||
|             'formats': formats, | ||||
|             'thumbnails': images, | ||||
|         } | ||||
| @@ -9,7 +9,7 @@ class BeegIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?beeg\.com/(?P<id>\d+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://beeg.com/5416503', | ||||
|         'md5': '634526ae978711f6b748fe0dd6c11f57', | ||||
|         'md5': '1bff67111adb785c51d1b42959ec10e5', | ||||
|         'info_dict': { | ||||
|             'id': '5416503', | ||||
|             'ext': 'mp4', | ||||
|   | ||||
| @@ -16,11 +16,11 @@ class BetIE(InfoExtractor): | ||||
|         { | ||||
|             'url': 'http://www.bet.com/news/politics/2014/12/08/in-bet-exclusive-obama-talks-race-and-racism.html', | ||||
|             'info_dict': { | ||||
|                 'id': '740ab250-bb94-4a8a-8787-fe0de7c74471', | ||||
|                 'id': 'news/national/2014/a-conversation-with-president-obama', | ||||
|                 'display_id': 'in-bet-exclusive-obama-talks-race-and-racism', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'BET News Presents: A Conversation With President Obama', | ||||
|                 'description': 'md5:5a88d8ae912c1b33e090290af7ec33c6', | ||||
|                 'title': 'A Conversation With President Obama', | ||||
|                 'description': 'md5:699d0652a350cf3e491cd15cc745b5da', | ||||
|                 'duration': 1534, | ||||
|                 'timestamp': 1418075340, | ||||
|                 'upload_date': '20141208', | ||||
| @@ -35,7 +35,7 @@ class BetIE(InfoExtractor): | ||||
|         { | ||||
|             'url': 'http://www.bet.com/video/news/national/2014/justice-for-ferguson-a-community-reacts.html', | ||||
|             'info_dict': { | ||||
|                 'id': 'bcd1b1df-673a-42cf-8d01-b282db608f2d', | ||||
|                 'id': 'news/national/2014/justice-for-ferguson-a-community-reacts', | ||||
|                 'display_id': 'justice-for-ferguson-a-community-reacts', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'Justice for Ferguson: A Community Reacts', | ||||
| @@ -61,6 +61,9 @@ class BetIE(InfoExtractor): | ||||
|             [r'mediaURL\s*:\s*"([^"]+)"', r"var\s+mrssMediaUrl\s*=\s*'([^']+)'"], | ||||
|             webpage, 'media URL')) | ||||
|  | ||||
|         video_id = self._search_regex( | ||||
|             r'/video/(.*)/_jcr_content/', media_url, 'video id') | ||||
|  | ||||
|         mrss = self._download_xml(media_url, display_id) | ||||
|  | ||||
|         item = mrss.find('./channel/item') | ||||
| @@ -75,8 +78,6 @@ class BetIE(InfoExtractor): | ||||
|         description = xpath_text( | ||||
|             item, './description', 'description', fatal=False) | ||||
|  | ||||
|         video_id = xpath_text(item, './guid', 'video id', fatal=False) | ||||
|  | ||||
|         timestamp = parse_iso8601(xpath_text( | ||||
|             item, xpath_with_ns('./dc:date', NS_MAP), | ||||
|             'upload date', fatal=False)) | ||||
|   | ||||
| @@ -2,7 +2,10 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import int_or_none | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     fix_xml_ampersands, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BildIE(InfoExtractor): | ||||
| @@ -15,7 +18,7 @@ class BildIE(InfoExtractor): | ||||
|             'id': '38184146', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'BILD hat sie getestet', | ||||
|             'thumbnail': 'http://bilder.bild.de/fotos/stand-das-koennen-die-neuen-ipads-38184138/Bild/1.bild.jpg', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'duration': 196, | ||||
|             'description': 'Mit dem iPad Air 2 und dem iPad Mini 3 hat Apple zwei neue Tablet-Modelle präsentiert. BILD-Reporter Sven Stein durfte die Geräte bereits testen. ', | ||||
|         } | ||||
| @@ -25,7 +28,7 @@ class BildIE(InfoExtractor): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         xml_url = url.split(".bild.html")[0] + ",view=xml.bild.xml" | ||||
|         doc = self._download_xml(xml_url, video_id) | ||||
|         doc = self._download_xml(xml_url, video_id, transform_source=fix_xml_ampersands) | ||||
|  | ||||
|         duration = int_or_none(doc.attrib.get('duration'), scale=1000) | ||||
|  | ||||
|   | ||||
| @@ -2,34 +2,47 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import itertools | ||||
| import json | ||||
| import xml.etree.ElementTree as ET | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     unified_strdate, | ||||
|     ExtractorError, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BiliBiliIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www\.bilibili\.(?:tv|com)/video/av(?P<id>[0-9]+)/' | ||||
|  | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.bilibili.tv/video/av1074402/', | ||||
|         'md5': '2c301e4dab317596e837c3e7633e7d86', | ||||
|         'info_dict': { | ||||
|             'id': '1074402', | ||||
|             'id': '1074402_part1', | ||||
|             'ext': 'flv', | ||||
|             'title': '【金坷垃】金泡沫', | ||||
|             'duration': 308, | ||||
|             'upload_date': '20140420', | ||||
|             'thumbnail': 're:^https?://.+\.jpg', | ||||
|         }, | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://www.bilibili.com/video/av1041170/', | ||||
|         'info_dict': { | ||||
|             'id': '1041170', | ||||
|             'title': '【BD1080P】刀语【诸神&异域】', | ||||
|         }, | ||||
|         'playlist_count': 9, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         if self._search_regex(r'(此视频不存在或被删除)', webpage, 'error message', default=None): | ||||
|             raise ExtractorError('The video does not exist or was deleted', expected=True) | ||||
|         video_code = self._search_regex( | ||||
|             r'(?s)<div itemprop="video".*?>(.*?)</div>', webpage, 'video code') | ||||
|  | ||||
| @@ -54,19 +67,22 @@ class BiliBiliIE(InfoExtractor): | ||||
|  | ||||
|         cid = self._search_regex(r'cid=(\d+)', webpage, 'cid') | ||||
|  | ||||
|         lq_doc = self._download_xml( | ||||
|         entries = [] | ||||
|  | ||||
|         lq_page = self._download_webpage( | ||||
|             'http://interface.bilibili.com/v_cdn_play?appkey=1&cid=%s' % cid, | ||||
|             video_id, | ||||
|             note='Downloading LQ video info' | ||||
|         ) | ||||
|         lq_durl = lq_doc.find('./durl') | ||||
|         formats = [{ | ||||
|             'format_id': 'lq', | ||||
|             'quality': 1, | ||||
|             'url': lq_durl.find('./url').text, | ||||
|             'filesize': int_or_none( | ||||
|                 lq_durl.find('./size'), get_attr='text'), | ||||
|         }] | ||||
|         try: | ||||
|             err_info = json.loads(lq_page) | ||||
|             raise ExtractorError( | ||||
|                 'BiliBili said: ' + err_info['error_text'], expected=True) | ||||
|         except ValueError: | ||||
|             pass | ||||
|  | ||||
|         lq_doc = ET.fromstring(lq_page) | ||||
|         lq_durls = lq_doc.findall('./durl') | ||||
|  | ||||
|         hq_doc = self._download_xml( | ||||
|             'http://interface.bilibili.com/playurl?appkey=1&cid=%s' % cid, | ||||
| @@ -75,22 +91,45 @@ class BiliBiliIE(InfoExtractor): | ||||
|             fatal=False, | ||||
|         ) | ||||
|         if hq_doc is not False: | ||||
|             hq_durl = hq_doc.find('./durl') | ||||
|             formats.append({ | ||||
|                 'format_id': 'hq', | ||||
|                 'quality': 2, | ||||
|                 'ext': 'flv', | ||||
|                 'url': hq_durl.find('./url').text, | ||||
|             hq_durls = hq_doc.findall('./durl') | ||||
|             assert len(lq_durls) == len(hq_durls) | ||||
|         else: | ||||
|             hq_durls = itertools.repeat(None) | ||||
|  | ||||
|         i = 1 | ||||
|         for lq_durl, hq_durl in zip(lq_durls, hq_durls): | ||||
|             formats = [{ | ||||
|                 'format_id': 'lq', | ||||
|                 'quality': 1, | ||||
|                 'url': lq_durl.find('./url').text, | ||||
|                 'filesize': int_or_none( | ||||
|                     hq_durl.find('./size'), get_attr='text'), | ||||
|                     lq_durl.find('./size'), get_attr='text'), | ||||
|             }] | ||||
|             if hq_durl is not None: | ||||
|                 formats.append({ | ||||
|                     'format_id': 'hq', | ||||
|                     'quality': 2, | ||||
|                     'ext': 'flv', | ||||
|                     'url': hq_durl.find('./url').text, | ||||
|                     'filesize': int_or_none( | ||||
|                         hq_durl.find('./size'), get_attr='text'), | ||||
|                 }) | ||||
|             self._sort_formats(formats) | ||||
|  | ||||
|             entries.append({ | ||||
|                 'id': '%s_part%d' % (video_id, i), | ||||
|                 'title': title, | ||||
|                 'formats': formats, | ||||
|                 'duration': duration, | ||||
|                 'upload_date': upload_date, | ||||
|                 'thumbnail': thumbnail, | ||||
|             }) | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|             i += 1 | ||||
|  | ||||
|         return { | ||||
|             '_type': 'multi_video', | ||||
|             'entries': entries, | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'formats': formats, | ||||
|             'duration': duration, | ||||
|             'upload_date': upload_date, | ||||
|             'thumbnail': thumbnail, | ||||
|             'title': title | ||||
|         } | ||||
|   | ||||
| @@ -1,40 +1,35 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import json | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import remove_start | ||||
| from ..utils import ( | ||||
|     remove_start, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BlinkxIE(InfoExtractor): | ||||
|     _VALID_URL = r'^(?:https?://(?:www\.)blinkx\.com/#?ce/|blinkx:)(?P<id>[^?]+)' | ||||
|     _VALID_URL = r'(?:https?://(?:www\.)blinkx\.com/#?ce/|blinkx:)(?P<id>[^?]+)' | ||||
|     IE_NAME = 'blinkx' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.blinkx.com/ce/8aQUy7GVFYgFzpKhT0oqsilwOGFRVXk3R1ZGWWdGenBLaFQwb3FzaWx3OGFRVXk3R1ZGWWdGenB', | ||||
|         'md5': '2e9a07364af40163a908edbf10bb2492', | ||||
|         'url': 'http://www.blinkx.com/ce/Da0Gw3xc5ucpNduzLuDDlv4WC9PuI4fDi1-t6Y3LyfdY2SZS5Urbvn-UPJvrvbo8LTKTc67Wu2rPKSQDJyZeeORCR8bYkhs8lI7eqddznH2ofh5WEEdjYXnoRtj7ByQwt7atMErmXIeYKPsSDuMAAqJDlQZ-3Ff4HJVeH_s3Gh8oQ', | ||||
|         'md5': '337cf7a344663ec79bf93a526a2e06c7', | ||||
|         'info_dict': { | ||||
|             'id': '8aQUy7GV', | ||||
|             'id': 'Da0Gw3xc', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Police Car Rolls Away', | ||||
|             'uploader': 'stupidvideos.com', | ||||
|             'upload_date': '20131215', | ||||
|             'timestamp': 1387068000, | ||||
|             'description': 'A police car gently rolls away from a fight. Maybe it felt weird being around a confrontation and just had to get out of there!', | ||||
|             'duration': 14.886, | ||||
|             'thumbnails': [{ | ||||
|                 'width': 100, | ||||
|                 'height': 76, | ||||
|                 'resolution': '100x76', | ||||
|                 'url': 'http://cdn.blinkx.com/stream/b/41/StupidVideos/20131215/1873969261/1873969261_tn_0.jpg', | ||||
|             }], | ||||
|             'title': 'No Daily Show for John Oliver; HBO Show Renewed - IGN News', | ||||
|             'uploader': 'IGN News', | ||||
|             'upload_date': '20150217', | ||||
|             'timestamp': 1424215740, | ||||
|             'description': 'HBO has renewed Last Week Tonight With John Oliver for two more seasons.', | ||||
|             'duration': 47.743333, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, rl): | ||||
|         m = re.match(self._VALID_URL, rl) | ||||
|         video_id = m.group('id') | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         display_id = video_id[:8] | ||||
|  | ||||
|         api_url = ('https://apib4.blinkx.com/api.php?action=play_video&' + | ||||
| @@ -60,18 +55,20 @@ class BlinkxIE(InfoExtractor): | ||||
|             elif m['type'] in ('flv', 'mp4'): | ||||
|                 vcodec = remove_start(m['vcodec'], 'ff') | ||||
|                 acodec = remove_start(m['acodec'], 'ff') | ||||
|                 tbr = (int(m['vbr']) + int(m['abr'])) // 1000 | ||||
|                 vbr = int_or_none(m.get('vbr') or m.get('vbitrate'), 1000) | ||||
|                 abr = int_or_none(m.get('abr') or m.get('abitrate'), 1000) | ||||
|                 tbr = vbr + abr if vbr and abr else None | ||||
|                 format_id = '%s-%sk-%s' % (vcodec, tbr, m['w']) | ||||
|                 formats.append({ | ||||
|                     'format_id': format_id, | ||||
|                     'url': m['link'], | ||||
|                     'vcodec': vcodec, | ||||
|                     'acodec': acodec, | ||||
|                     'abr': int(m['abr']) // 1000, | ||||
|                     'vbr': int(m['vbr']) // 1000, | ||||
|                     'abr': abr, | ||||
|                     'vbr': vbr, | ||||
|                     'tbr': tbr, | ||||
|                     'width': int(m['w']), | ||||
|                     'height': int(m['h']), | ||||
|                     'width': int_or_none(m.get('w')), | ||||
|                     'height': int_or_none(m.get('h')), | ||||
|                 }) | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|   | ||||
| @@ -3,7 +3,6 @@ from __future__ import unicode_literals | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
|  | ||||
| from ..compat import ( | ||||
|     compat_str, | ||||
| @@ -18,7 +17,7 @@ from ..utils import ( | ||||
| ) | ||||
|  | ||||
|  | ||||
| class BlipTVIE(SubtitlesInfoExtractor): | ||||
| class BlipTVIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:\w+\.)?blip\.tv/(?:(?:.+-|rss/flash/)(?P<id>\d+)|((?:play/|api\.swf#)(?P<lookup_id>[\da-zA-Z+_]+)))' | ||||
|  | ||||
|     _TESTS = [ | ||||
| @@ -103,6 +102,15 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     @staticmethod | ||||
|     def _extract_url(webpage): | ||||
|         mobj = re.search(r'<meta\s[^>]*https?://api\.blip\.tv/\w+/redirect/\w+/(\d+)', webpage) | ||||
|         if mobj: | ||||
|             return 'http://blip.tv/a/a-' + mobj.group(1) | ||||
|         mobj = re.search(r'<(?:iframe|embed|object)\s[^>]*(https?://(?:\w+\.)?blip\.tv/(?:play/|api\.swf#)[a-zA-Z0-9_]+)', webpage) | ||||
|         if mobj: | ||||
|             return mobj.group(1) | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         lookup_id = mobj.group('lookup_id') | ||||
| @@ -143,7 +151,7 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|         categories = [category.text for category in item.findall('category')] | ||||
|  | ||||
|         formats = [] | ||||
|         subtitles = {} | ||||
|         subtitles_urls = {} | ||||
|  | ||||
|         media_group = item.find(media('group')) | ||||
|         for media_content in media_group.findall(media('content')): | ||||
| @@ -161,7 +169,7 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|                 } | ||||
|                 lang = role.rpartition('-')[-1].strip().lower() | ||||
|                 langcode = LANGS.get(lang, lang) | ||||
|                 subtitles[langcode] = url | ||||
|                 subtitles_urls[langcode] = url | ||||
|             elif media_type.startswith('video/'): | ||||
|                 formats.append({ | ||||
|                     'url': real_url, | ||||
| @@ -173,13 +181,10 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|                     'width': int_or_none(media_content.get('width')), | ||||
|                     'height': int_or_none(media_content.get('height')), | ||||
|                 }) | ||||
|         self._check_formats(formats, video_id) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         # subtitles | ||||
|         video_subtitles = self.extract_subtitles(video_id, subtitles) | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, subtitles) | ||||
|             return | ||||
|         subtitles = self.extract_subtitles(video_id, subtitles_urls) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
| @@ -192,15 +197,22 @@ class BlipTVIE(SubtitlesInfoExtractor): | ||||
|             'thumbnail': thumbnail, | ||||
|             'categories': categories, | ||||
|             'formats': formats, | ||||
|             'subtitles': video_subtitles, | ||||
|             'subtitles': subtitles, | ||||
|         } | ||||
|  | ||||
|     def _download_subtitle_url(self, sub_lang, url): | ||||
|         # For some weird reason, blip.tv serves a video instead of subtitles | ||||
|         # when we request with a common UA | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('User-Agent', 'youtube-dl') | ||||
|         return self._download_webpage(req, None, note=False) | ||||
|     def _get_subtitles(self, video_id, subtitles_urls): | ||||
|         subtitles = {} | ||||
|         for lang, url in subtitles_urls.items(): | ||||
|             # For some weird reason, blip.tv serves a video instead of subtitles | ||||
|             # when we request with a common UA | ||||
|             req = compat_urllib_request.Request(url) | ||||
|             req.add_header('User-Agent', 'youtube-dl') | ||||
|             subtitles[lang] = [{ | ||||
|                 # The extension is 'srt' but it's actually an 'ass' file | ||||
|                 'ext': 'ass', | ||||
|                 'data': self._download_webpage(req, None, note=False), | ||||
|             }] | ||||
|         return subtitles | ||||
|  | ||||
|  | ||||
| class BlipTVUserIE(InfoExtractor): | ||||
|   | ||||
| @@ -6,32 +6,39 @@ from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class BloombergIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.bloomberg\.com/video/(?P<name>.+?)\.html' | ||||
|     _VALID_URL = r'https?://www\.bloomberg\.com/news/videos/[^/]+/(?P<id>[^/?#]+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.bloomberg.com/video/shah-s-presentation-on-foreign-exchange-strategies-qurhIVlJSB6hzkVi229d8g.html', | ||||
|         'url': 'http://www.bloomberg.com/news/videos/b/aaeae121-5949-481e-a1ce-4562db6f5df2', | ||||
|         # The md5 checksum changes | ||||
|         'info_dict': { | ||||
|             'id': 'qurhIVlJSB6hzkVi229d8g', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Shah\'s Presentation on Foreign-Exchange Strategies', | ||||
|             'description': 'md5:0681e0d30dcdfc6abf34594961d8ea88', | ||||
|             'description': 'md5:a8ba0302912d03d246979735c17d2761', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         name = mobj.group('name') | ||||
|         name = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, name) | ||||
|         f4m_url = self._search_regex( | ||||
|             r'<source src="(https?://[^"]+\.f4m.*?)"', webpage, | ||||
|             'f4m url') | ||||
|         video_id = self._search_regex(r'"bmmrId":"(.+?)"', webpage, 'id') | ||||
|         title = re.sub(': Video$', '', self._og_search_title(webpage)) | ||||
|  | ||||
|         embed_info = self._download_json( | ||||
|             'http://www.bloomberg.com/api/embed?id=%s' % video_id, video_id) | ||||
|         formats = [] | ||||
|         for stream in embed_info['streams']: | ||||
|             if stream["muxing_format"] == "TS": | ||||
|                 formats.extend(self._extract_m3u8_formats(stream['url'], video_id)) | ||||
|             else: | ||||
|                 formats.extend(self._extract_f4m_formats(stream['url'], video_id)) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': name.split('-')[-1], | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'formats': self._extract_f4m_formats(f4m_url, name), | ||||
|             'formats': formats, | ||||
|             'description': self._og_search_description(webpage), | ||||
|             'thumbnail': self._og_search_thumbnail(webpage), | ||||
|         } | ||||
|   | ||||
| @@ -16,27 +16,38 @@ class BRIE(InfoExtractor): | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.br.de/mediathek/video/sendungen/heimatsound/heimatsound-festival-2014-trailer-100.html', | ||||
|             'md5': '93556dd2bcb2948d9259f8670c516d59', | ||||
|             'url': 'http://www.br.de/mediathek/video/sendungen/abendschau/betriebliche-altersvorsorge-104.html', | ||||
|             'md5': '83a0477cf0b8451027eb566d88b51106', | ||||
|             'info_dict': { | ||||
|                 'id': '25e279aa-1ffd-40fd-9955-5325bd48a53a', | ||||
|                 'id': '48f656ef-287e-486f-be86-459122db22cc', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Wenn das Traditions-Theater wackelt', | ||||
|                 'description': 'Heimatsound-Festival 2014: Wenn das Traditions-Theater wackelt', | ||||
|                 'duration': 34, | ||||
|                 'uploader': 'BR', | ||||
|                 'upload_date': '20140802', | ||||
|                 'title': 'Die böse Überraschung', | ||||
|                 'description': 'Betriebliche Altersvorsorge: Die böse Überraschung', | ||||
|                 'duration': 180, | ||||
|                 'uploader': 'Reinhard Weber', | ||||
|                 'upload_date': '20150422', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.br.de/nachrichten/schaeuble-haushaltsentwurf-bundestag-100.html', | ||||
|             'md5': '3db0df1a9a9cd9fa0c70e6ea8aa8e820', | ||||
|             'url': 'http://www.br.de/nachrichten/oberbayern/inhalt/muenchner-polizeipraesident-schreiber-gestorben-100.html', | ||||
|             'md5': 'a44396d73ab6a68a69a568fae10705bb', | ||||
|             'info_dict': { | ||||
|                 'id': 'c6aae3de-2cf9-43f2-957f-f17fef9afaab', | ||||
|                 'id': 'a4b83e34-123d-4b81-9f4e-c0d3121a4e05', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Manfred Schreiber ist tot', | ||||
|                 'description': 'Abendschau kompakt: Manfred Schreiber ist tot', | ||||
|                 'duration': 26, | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.br.de/radio/br-klassik/sendungen/allegro/premiere-urauffuehrung-the-land-2015-dance-festival-muenchen-100.html', | ||||
|             'md5': '8b5b27c0b090f3b35eac4ab3f7a73d3d', | ||||
|             'info_dict': { | ||||
|                 'id': '74c603c9-26d3-48bb-b85b-079aeed66e0b', | ||||
|                 'ext': 'aac', | ||||
|                 'title': '"Keine neuen Schulden im nächsten Jahr"', | ||||
|                 'description': 'Haushaltsentwurf: "Keine neuen Schulden im nächsten Jahr"', | ||||
|                 'duration': 64, | ||||
|                 'title': 'Kurzweilig und sehr bewegend', | ||||
|                 'description': '"The Land" von Peeping Tom: Kurzweilig und sehr bewegend', | ||||
|                 'duration': 296, | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|   | ||||
| @@ -41,7 +41,7 @@ class BreakIE(InfoExtractor): | ||||
|             'tbr': media['bitRate'], | ||||
|             'width': media['width'], | ||||
|             'height': media['height'], | ||||
|         } for media in info['media']] | ||||
|         } for media in info['media'] if media.get('mediaPurpose') == 'play'] | ||||
|  | ||||
|         if not formats: | ||||
|             formats.append({ | ||||
|   | ||||
| @@ -95,6 +95,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|             'url': 'http://c.brightcove.com/services/viewer/htmlFederated?playerID=3550052898001&playerKey=AQ%7E%7E%2CAAABmA9XpXk%7E%2C-Kp7jNgisre1fG5OdqpAFUTcs0lP_ZoL', | ||||
|             'info_dict': { | ||||
|                 'title': 'Sealife', | ||||
|                 'id': '3550319591001', | ||||
|             }, | ||||
|             'playlist_mincount': 7, | ||||
|         }, | ||||
| @@ -116,7 +117,10 @@ class BrightcoveIE(InfoExtractor): | ||||
|         object_str = re.sub(r'(<object[^>]*)(xmlns=".*?")', r'\1', object_str) | ||||
|         object_str = fix_xml_ampersands(object_str) | ||||
|  | ||||
|         object_doc = xml.etree.ElementTree.fromstring(object_str.encode('utf-8')) | ||||
|         try: | ||||
|             object_doc = xml.etree.ElementTree.fromstring(object_str.encode('utf-8')) | ||||
|         except xml.etree.ElementTree.ParseError: | ||||
|             return | ||||
|  | ||||
|         fv_el = find_xpath_attr(object_doc, './param', 'name', 'flashVars') | ||||
|         if fv_el is not None: | ||||
| @@ -152,6 +156,28 @@ class BrightcoveIE(InfoExtractor): | ||||
|         linkBase = find_param('linkBaseURL') | ||||
|         if linkBase is not None: | ||||
|             params['linkBaseURL'] = linkBase | ||||
|         return cls._make_brightcove_url(params) | ||||
|  | ||||
|     @classmethod | ||||
|     def _build_brighcove_url_from_js(cls, object_js): | ||||
|         # The layout of JS is as follows: | ||||
|         # customBC.createVideo = function (width, height, playerID, playerKey, videoPlayer, VideoRandomID) { | ||||
|         #   // build Brightcove <object /> XML | ||||
|         # } | ||||
|         m = re.search( | ||||
|             r'''(?x)customBC.\createVideo\( | ||||
|                 .*?                                                  # skipping width and height | ||||
|                 ["\'](?P<playerID>\d+)["\']\s*,\s*                   # playerID | ||||
|                 ["\'](?P<playerKey>AQ[^"\']{48})[^"\']*["\']\s*,\s*  # playerKey begins with AQ and is 50 characters | ||||
|                                                                      # in length, however it's appended to itself | ||||
|                                                                      # in places, so truncate | ||||
|                 ["\'](?P<videoID>\d+)["\']                           # @videoPlayer | ||||
|             ''', object_js) | ||||
|         if m: | ||||
|             return cls._make_brightcove_url(m.groupdict()) | ||||
|  | ||||
|     @classmethod | ||||
|     def _make_brightcove_url(cls, params): | ||||
|         data = compat_urllib_parse.urlencode(params) | ||||
|         return cls._FEDERATED_URL_TEMPLATE % data | ||||
|  | ||||
| @@ -168,7 +194,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|         """Return a list of all Brightcove URLs from the webpage """ | ||||
|  | ||||
|         url_m = re.search( | ||||
|             r'<meta\s+property="og:video"\s+content="(https?://(?:secure|c)\.brightcove.com/[^"]+)"', | ||||
|             r'<meta\s+property=[\'"]og:video[\'"]\s+content=[\'"](https?://(?:secure|c)\.brightcove.com/[^\'"]+)[\'"]', | ||||
|             webpage) | ||||
|         if url_m: | ||||
|             url = unescapeHTML(url_m.group(1)) | ||||
| @@ -182,9 +208,14 @@ class BrightcoveIE(InfoExtractor): | ||||
|             (?: | ||||
|                 [^>]+?class=[\'"][^>]*?BrightcoveExperience.*?[\'"] | | ||||
|                 [^>]*?>\s*<param\s+name="movie"\s+value="https?://[^/]*brightcove\.com/ | ||||
|             ).+?</object>''', | ||||
|             ).+?>\s*</object>''', | ||||
|             webpage) | ||||
|         return [cls._build_brighcove_url(m) for m in matches] | ||||
|         if matches: | ||||
|             return list(filter(None, [cls._build_brighcove_url(m) for m in matches])) | ||||
|  | ||||
|         return list(filter(None, [ | ||||
|             cls._build_brighcove_url_from_js(custom_bc) | ||||
|             for custom_bc in re.findall(r'(customBC\.createVideo\(.+?\);)', webpage)])) | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         url, smuggled_data = unsmuggle_url(url, {}) | ||||
| @@ -247,7 +278,7 @@ class BrightcoveIE(InfoExtractor): | ||||
|         playlist_info = json_data['videoList'] | ||||
|         videos = [self._extract_video_info(video_info) for video_info in playlist_info['mediaCollectionDTO']['videoDTOs']] | ||||
|  | ||||
|         return self.playlist_result(videos, playlist_id=playlist_info['id'], | ||||
|         return self.playlist_result(videos, playlist_id='%s' % playlist_info['id'], | ||||
|                                     playlist_title=playlist_info['mediaCollectionDTO']['displayName']) | ||||
|  | ||||
|     def _extract_video_info(self, video_info): | ||||
|   | ||||
| @@ -33,6 +33,7 @@ class BuzzFeedIE(InfoExtractor): | ||||
|             'skip_download': True,  # Got enough YouTube download tests | ||||
|         }, | ||||
|         'info_dict': { | ||||
|             'id': 'look-at-this-cute-dog-omg', | ||||
|             'description': 're:Munchkin the Teddy Bear is back ?!', | ||||
|             'title': 'You Need To Stop What You\'re Doing And Watching This Dog Walk On A Treadmill', | ||||
|         }, | ||||
| @@ -42,8 +43,8 @@ class BuzzFeedIE(InfoExtractor): | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20141124', | ||||
|                 'uploader_id': 'CindysMunchkin', | ||||
|                 'description': 're:© 2014 Munchkin the Shih Tzu', | ||||
|                 'uploader': 'Munchkin the Shih Tzu', | ||||
|                 'description': 're:© 2014 Munchkin the', | ||||
|                 'uploader': 're:^Munchkin the', | ||||
|                 'title': 're:Munchkin the Teddy Bear gets her exercise', | ||||
|             }, | ||||
|         }] | ||||
|   | ||||
| @@ -16,7 +16,7 @@ class BYUtvIE(InfoExtractor): | ||||
|             'ext': 'mp4', | ||||
|             'description': 'md5:5438d33774b6bdc662f9485a340401cc', | ||||
|             'title': 'Season 5 Episode 5', | ||||
|             'thumbnail': 're:^https?://.*promo.*' | ||||
|             'thumbnail': 're:^https?://.*\.jpg$' | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True, | ||||
|   | ||||
							
								
								
									
										153
									
								
								youtube_dl/extractor/camdemy.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										153
									
								
								youtube_dl/extractor/camdemy.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,153 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import datetime | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_urllib_parse, | ||||
|     compat_urlparse, | ||||
| ) | ||||
| from ..utils import ( | ||||
|     parse_iso8601, | ||||
|     str_to_int, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CamdemyIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?camdemy\.com/media/(?P<id>\d+)' | ||||
|     _TESTS = [{ | ||||
|         # single file | ||||
|         'url': 'http://www.camdemy.com/media/5181/', | ||||
|         'md5': '5a5562b6a98b37873119102e052e311b', | ||||
|         'info_dict': { | ||||
|             'id': '5181', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Ch1-1 Introduction, Signals (02-23-2012)', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'description': '', | ||||
|             'creator': 'ss11spring', | ||||
|             'upload_date': '20130114', | ||||
|             'timestamp': 1358154556, | ||||
|             'view_count': int, | ||||
|         } | ||||
|     }, { | ||||
|         # With non-empty description | ||||
|         'url': 'http://www.camdemy.com/media/13885', | ||||
|         'md5': '4576a3bb2581f86c61044822adbd1249', | ||||
|         'info_dict': { | ||||
|             'id': '13885', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'EverCam + Camdemy QuickStart', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'description': 'md5:050b62f71ed62928f8a35f1a41e186c9', | ||||
|             'creator': 'evercam', | ||||
|             'upload_date': '20140620', | ||||
|             'timestamp': 1403271569, | ||||
|         } | ||||
|     }, { | ||||
|         # External source | ||||
|         'url': 'http://www.camdemy.com/media/14842', | ||||
|         'md5': '50e1c3c3aa233d3d7b7daa2fa10b1cf7', | ||||
|         'info_dict': { | ||||
|             'id': '2vsYQzNIsJo', | ||||
|             'ext': 'mp4', | ||||
|             'upload_date': '20130211', | ||||
|             'uploader': 'Hun Kim', | ||||
|             'description': 'Excel 2013 Tutorial for Beginners - How to add Password Protection', | ||||
|             'uploader_id': 'hunkimtutorials', | ||||
|             'title': 'Excel 2013 Tutorial - How to add Password Protection', | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         page = self._download_webpage(url, video_id) | ||||
|  | ||||
|         src_from = self._html_search_regex( | ||||
|             r"<div class='srcFrom'>Source: <a title='([^']+)'", page, | ||||
|             'external source', default=None) | ||||
|         if src_from: | ||||
|             return self.url_result(src_from) | ||||
|  | ||||
|         oembed_obj = self._download_json( | ||||
|             'http://www.camdemy.com/oembed/?format=json&url=' + url, video_id) | ||||
|  | ||||
|         thumb_url = oembed_obj['thumbnail_url'] | ||||
|         video_folder = compat_urlparse.urljoin(thumb_url, 'video/') | ||||
|         file_list_doc = self._download_xml( | ||||
|             compat_urlparse.urljoin(video_folder, 'fileList.xml'), | ||||
|             video_id, 'Filelist XML') | ||||
|         file_name = file_list_doc.find('./video/item/fileName').text | ||||
|         video_url = compat_urlparse.urljoin(video_folder, file_name) | ||||
|  | ||||
|         timestamp = parse_iso8601(self._html_search_regex( | ||||
|             r"<div class='title'>Posted\s*:</div>\s*<div class='value'>([^<>]+)<", | ||||
|             page, 'creation time', fatal=False), | ||||
|             delimiter=' ', timezone=datetime.timedelta(hours=8)) | ||||
|         view_count = str_to_int(self._html_search_regex( | ||||
|             r"<div class='title'>Views\s*:</div>\s*<div class='value'>([^<>]+)<", | ||||
|             page, 'view count', fatal=False)) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': oembed_obj['title'], | ||||
|             'thumbnail': thumb_url, | ||||
|             'description': self._html_search_meta('description', page), | ||||
|             'creator': oembed_obj['author_name'], | ||||
|             'duration': oembed_obj['duration'], | ||||
|             'timestamp': timestamp, | ||||
|             'view_count': view_count, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class CamdemyFolderIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www.camdemy.com/folder/(?P<id>\d+)' | ||||
|     _TESTS = [{ | ||||
|         # links with trailing slash | ||||
|         'url': 'http://www.camdemy.com/folder/450', | ||||
|         'info_dict': { | ||||
|             'id': '450', | ||||
|             'title': '信號與系統 2012 & 2011 (Signals and Systems)', | ||||
|         }, | ||||
|         'playlist_mincount': 145 | ||||
|     }, { | ||||
|         # links without trailing slash | ||||
|         # and multi-page | ||||
|         'url': 'http://www.camdemy.com/folder/853', | ||||
|         'info_dict': { | ||||
|             'id': '853', | ||||
|             'title': '科學計算 - 使用 Matlab' | ||||
|         }, | ||||
|         'playlist_mincount': 20 | ||||
|     }, { | ||||
|         # with displayMode parameter. For testing the codes to add parameters | ||||
|         'url': 'http://www.camdemy.com/folder/853/?displayMode=defaultOrderByOrg', | ||||
|         'info_dict': { | ||||
|             'id': '853', | ||||
|             'title': '科學計算 - 使用 Matlab' | ||||
|         }, | ||||
|         'playlist_mincount': 20 | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         folder_id = self._match_id(url) | ||||
|  | ||||
|         # Add displayMode=list so that all links are displayed in a single page | ||||
|         parsed_url = list(compat_urlparse.urlparse(url)) | ||||
|         query = dict(compat_urlparse.parse_qsl(parsed_url[4])) | ||||
|         query.update({'displayMode': 'list'}) | ||||
|         parsed_url[4] = compat_urllib_parse.urlencode(query) | ||||
|         final_url = compat_urlparse.urlunparse(parsed_url) | ||||
|  | ||||
|         page = self._download_webpage(final_url, folder_id) | ||||
|         matches = re.findall(r"href='(/media/\d+/?)'", page) | ||||
|  | ||||
|         entries = [self.url_result('http://www.camdemy.com' + media_path) | ||||
|                    for media_path in matches] | ||||
|  | ||||
|         folder_title = self._html_search_meta('keywords', page) | ||||
|  | ||||
|         return self.playlist_result(entries, folder_id, folder_title) | ||||
| @@ -15,23 +15,24 @@ from ..utils import ( | ||||
|  | ||||
| class CanalplusIE(InfoExtractor): | ||||
|     IE_DESC = 'canalplus.fr, piwiplus.fr and d8.tv' | ||||
|     _VALID_URL = r'https?://(?:www\.(?P<site>canalplus\.fr|piwiplus\.fr|d8\.tv)/.*?/(?P<path>.*)|player\.canalplus\.fr/#/(?P<id>[0-9]+))' | ||||
|     _VALID_URL = r'https?://(?:www\.(?P<site>canalplus\.fr|piwiplus\.fr|d8\.tv|itele\.fr)/.*?/(?P<path>.*)|player\.canalplus\.fr/#/(?P<id>[0-9]+))' | ||||
|     _VIDEO_INFO_TEMPLATE = 'http://service.canal-plus.com/video/rest/getVideosLiees/%s/%s' | ||||
|     _SITE_ID_MAP = { | ||||
|         'canalplus.fr': 'cplus', | ||||
|         'piwiplus.fr': 'teletoon', | ||||
|         'd8.tv': 'd8', | ||||
|         'itele.fr': 'itele', | ||||
|     } | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.canalplus.fr/c-infos-documentaires/pid1830-c-zapping.html?vid=922470', | ||||
|         'md5': '3db39fb48b9685438ecf33a1078023e4', | ||||
|         'url': 'http://www.canalplus.fr/c-emissions/pid1830-c-zapping.html?vid=1263092', | ||||
|         'md5': 'b3481d7ca972f61e37420798d0a9d934', | ||||
|         'info_dict': { | ||||
|             'id': '922470', | ||||
|             'id': '1263092', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Zapping - 26/08/13', | ||||
|             'description': 'Le meilleur de toutes les chaînes, tous les jours.\nEmission du 26 août 2013', | ||||
|             'upload_date': '20130826', | ||||
|             'title': 'Le Zapping - 13/05/15', | ||||
|             'description': 'md5:09738c0d06be4b5d06a0940edb0da73f', | ||||
|             'upload_date': '20150513', | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'http://www.piwiplus.fr/videos-piwi/pid1405-le-labyrinthe-boing-super-ranger.html?vid=1108190', | ||||
| @@ -53,6 +54,16 @@ class CanalplusIE(InfoExtractor): | ||||
|             'upload_date': '20131108', | ||||
|         }, | ||||
|         'skip': 'videos get deleted after a while', | ||||
|     }, { | ||||
|         'url': 'http://www.itele.fr/france/video/aubervilliers-un-lycee-en-colere-111559', | ||||
|         'md5': 'f3a46edcdf28006598ffaf5b30e6a2d4', | ||||
|         'info_dict': { | ||||
|             'id': '1213714', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Aubervilliers : un lycée en colère - Le 11/02/2015 à 06h45', | ||||
|             'description': 'md5:8216206ec53426ea6321321f3b3c16db', | ||||
|             'upload_date': '20150211', | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|   | ||||
| @@ -1,17 +1,16 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class CBSIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?cbs\.com/shows/[^/]+/(?:video|artist)/(?P<id>[^/]+)/.*' | ||||
|     _VALID_URL = r'https?://(?:www\.)?(?:cbs\.com/shows/[^/]+/(?:video|artist)|colbertlateshow\.com/(?:video|podcasts))/[^/]+/(?P<id>[^/]+)' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.cbs.com/shows/garth-brooks/video/_u7W953k6la293J7EPTd9oHkSPs6Xn6_/connect-chat-feat-garth-brooks/', | ||||
|         'info_dict': { | ||||
|             'id': '4JUVEwq3wUT7', | ||||
|             'display_id': 'connect-chat-feat-garth-brooks', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Connect Chat feat. Garth Brooks', | ||||
|             'description': 'Connect with country music singer Garth Brooks, as he chats with fans on Wednesday November 27, 2013. Be sure to tune in to Garth Brooks: Live from Las Vegas, Friday November 29, at 9/8c on CBS!', | ||||
| @@ -26,6 +25,7 @@ class CBSIE(InfoExtractor): | ||||
|         'url': 'http://www.cbs.com/shows/liveonletterman/artist/221752/st-vincent/', | ||||
|         'info_dict': { | ||||
|             'id': 'WWF_5KqY3PK1', | ||||
|             'display_id': 'st-vincent', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Live on Letterman - St. Vincent', | ||||
|             'description': 'Live On Letterman: St. Vincent in concert from New York\'s Ed Sullivan Theater on Tuesday, July 16, 2014.', | ||||
| @@ -36,13 +36,23 @@ class CBSIE(InfoExtractor): | ||||
|             'skip_download': True, | ||||
|         }, | ||||
|         '_skip': 'Blocked outside the US', | ||||
|     }, { | ||||
|         'url': 'http://colbertlateshow.com/video/8GmB0oY0McANFvp2aEffk9jZZZ2YyXxy/the-colbeard/', | ||||
|         'only_matching': True, | ||||
|     }, { | ||||
|         'url': 'http://www.colbertlateshow.com/podcasts/dYSwjqPs_X1tvbV_P2FcPWRa_qT6akTC/in-the-bad-room-with-stephen/', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         display_id = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|         real_id = self._search_regex( | ||||
|             r"video\.settings\.pid\s*=\s*'([^']+)';", | ||||
|             [r"video\.settings\.pid\s*=\s*'([^']+)';", r"cbsplayer\.pid\s*=\s*'([^']+)';"], | ||||
|             webpage, 'real video ID') | ||||
|         return self.url_result('theplatform:%s' % real_id) | ||||
|         return { | ||||
|             '_type': 'url_transparent', | ||||
|             'ie_key': 'ThePlatform', | ||||
|             'url': 'theplatform:%s' % real_id, | ||||
|             'display_id': display_id, | ||||
|         } | ||||
|   | ||||
| @@ -32,7 +32,7 @@ class CBSNewsIE(InfoExtractor): | ||||
|                 'id': 'fort-hood-shooting-army-downplays-mental-illness-as-cause-of-attack', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'Fort Hood shooting: Army downplays mental illness as cause of attack', | ||||
|                 'thumbnail': 'http://cbsnews2.cbsistatic.com/hub/i/r/2014/04/04/0c9fbc66-576b-41ca-8069-02d122060dd2/thumbnail/140x90/6dad7a502f88875ceac38202984b6d58/en-0404-werner-replace-640x360.jpg', | ||||
|                 'thumbnail': 're:^https?://.*\.jpg$', | ||||
|                 'duration': 205, | ||||
|             }, | ||||
|             'params': { | ||||
|   | ||||
							
								
								
									
										30
									
								
								youtube_dl/extractor/cbssports.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								youtube_dl/extractor/cbssports.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class CBSSportsIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www\.cbssports\.com/video/player/(?P<section>[^/]+)/(?P<id>[^/]+)' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.cbssports.com/video/player/tennis/318462531970/0/us-open-flashbacks-1990s', | ||||
|         'info_dict': { | ||||
|             'id': '_d5_GbO8p1sT', | ||||
|             'ext': 'flv', | ||||
|             'title': 'US Open flashbacks: 1990s', | ||||
|             'description': 'Bill Macatee relives the best moments in US Open history from the 1990s.', | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         section = mobj.group('section') | ||||
|         video_id = mobj.group('id') | ||||
|         all_videos = self._download_json( | ||||
|             'http://www.cbssports.com/data/video/player/getVideos/%s?as=json' % section, | ||||
|             video_id) | ||||
|         # The json file contains the info of all the videos in the section | ||||
|         video_info = next(v for v in all_videos if v['pcid'] == video_id) | ||||
|         return self.url_result('theplatform:%s' % video_info['pid'], 'ThePlatform') | ||||
							
								
								
									
										99
									
								
								youtube_dl/extractor/ccc.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								youtube_dl/extractor/ccc.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     qualities, | ||||
|     unified_strdate, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CCCIE(InfoExtractor): | ||||
|     IE_NAME = 'media.ccc.de' | ||||
|     _VALID_URL = r'https?://(?:www\.)?media\.ccc\.de/[^?#]+/[^?#/]*?_(?P<id>[0-9]{8,})._[^?#/]*\.html' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://media.ccc.de/browse/congress/2013/30C3_-_5443_-_en_-_saal_g_-_201312281830_-_introduction_to_processor_design_-_byterazor.html#video', | ||||
|         'md5': '3a1eda8f3a29515d27f5adb967d7e740', | ||||
|         'info_dict': { | ||||
|             'id': '20131228183', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Introduction to Processor Design', | ||||
|             'description': 'md5:5ddbf8c734800267f2cee4eab187bc1b', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'view_count': int, | ||||
|             'upload_date': '20131229', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         if self._downloader.params.get('prefer_free_formats'): | ||||
|             preference = qualities(['mp3', 'opus', 'mp4-lq', 'webm-lq', 'h264-sd', 'mp4-sd', 'webm-sd', 'mp4', 'webm', 'mp4-hd', 'h264-hd', 'webm-hd']) | ||||
|         else: | ||||
|             preference = qualities(['opus', 'mp3', 'webm-lq', 'mp4-lq', 'webm-sd', 'h264-sd', 'mp4-sd', 'webm', 'mp4', 'webm-hd', 'mp4-hd', 'h264-hd']) | ||||
|  | ||||
|         title = self._html_search_regex( | ||||
|             r'(?s)<h1>(.*?)</h1>', webpage, 'title') | ||||
|         description = self._html_search_regex( | ||||
|             r"(?s)<p class='description'>(.*?)</p>", | ||||
|             webpage, 'description', fatal=False) | ||||
|         upload_date = unified_strdate(self._html_search_regex( | ||||
|             r"(?s)<span class='[^']*fa-calendar-o'></span>(.*?)</li>", | ||||
|             webpage, 'upload date', fatal=False)) | ||||
|         view_count = int_or_none(self._html_search_regex( | ||||
|             r"(?s)<span class='[^']*fa-eye'></span>(.*?)</li>", | ||||
|             webpage, 'view count', fatal=False)) | ||||
|  | ||||
|         matches = re.finditer(r'''(?xs) | ||||
|             <(?:span|div)\s+class='label\s+filetype'>(?P<format>.*?)</(?:span|div)>\s* | ||||
|             <a\s+download\s+href='(?P<http_url>[^']+)'>\s* | ||||
|             (?: | ||||
|                 .*? | ||||
|                 <a\s+href='(?P<torrent_url>[^']+\.torrent)' | ||||
|             )?''', webpage) | ||||
|         formats = [] | ||||
|         for m in matches: | ||||
|             format = m.group('format') | ||||
|             format_id = self._search_regex( | ||||
|                 r'.*/([a-z0-9_-]+)/[^/]*$', | ||||
|                 m.group('http_url'), 'format id', default=None) | ||||
|             vcodec = 'h264' if 'h264' in format_id else ( | ||||
|                 'none' if format_id in ('mp3', 'opus') else None | ||||
|             ) | ||||
|             formats.append({ | ||||
|                 'format_id': format_id, | ||||
|                 'format': format, | ||||
|                 'url': m.group('http_url'), | ||||
|                 'vcodec': vcodec, | ||||
|                 'preference': preference(format_id), | ||||
|             }) | ||||
|  | ||||
|             if m.group('torrent_url'): | ||||
|                 formats.append({ | ||||
|                     'format_id': 'torrent-%s' % (format if format_id is None else format_id), | ||||
|                     'format': '%s (torrent)' % format, | ||||
|                     'proto': 'torrent', | ||||
|                     'format_note': '(unsupported; will just download the .torrent file)', | ||||
|                     'vcodec': vcodec, | ||||
|                     'preference': -100 + preference(format_id), | ||||
|                     'url': m.group('torrent_url'), | ||||
|                 }) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         thumbnail = self._html_search_regex( | ||||
|             r"<video.*?poster='([^']+)'", webpage, 'thumbnail', fatal=False) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'view_count': view_count, | ||||
|             'upload_date': upload_date, | ||||
|             'formats': formats, | ||||
|         } | ||||
| @@ -3,7 +3,7 @@ from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_urllib_request, | ||||
|     compat_urllib_parse, | ||||
| @@ -15,7 +15,7 @@ from ..utils import ( | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CeskaTelevizeIE(SubtitlesInfoExtractor): | ||||
| class CeskaTelevizeIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://www\.ceskatelevize\.cz/(porady|ivysilani)/(.+/)?(?P<id>[^?#]+)' | ||||
|  | ||||
|     _TESTS = [ | ||||
| @@ -107,13 +107,7 @@ class CeskaTelevizeIE(SubtitlesInfoExtractor): | ||||
|         subtitles = {} | ||||
|         subs = item.get('subtitles') | ||||
|         if subs: | ||||
|             subtitles['cs'] = subs[0]['url'] | ||||
|  | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, subtitles) | ||||
|             return | ||||
|  | ||||
|         subtitles = self._fix_subtitles(self.extract_subtitles(video_id, subtitles)) | ||||
|             subtitles = self.extract_subtitles(episode_id, subs) | ||||
|  | ||||
|         return { | ||||
|             'id': episode_id, | ||||
| @@ -125,11 +119,20 @@ class CeskaTelevizeIE(SubtitlesInfoExtractor): | ||||
|             'subtitles': subtitles, | ||||
|         } | ||||
|  | ||||
|     def _get_subtitles(self, episode_id, subs): | ||||
|         original_subtitles = self._download_webpage( | ||||
|             subs[0]['url'], episode_id, 'Downloading subtitles') | ||||
|         srt_subs = self._fix_subtitles(original_subtitles) | ||||
|         return { | ||||
|             'cs': [{ | ||||
|                 'ext': 'srt', | ||||
|                 'data': srt_subs, | ||||
|             }] | ||||
|         } | ||||
|  | ||||
|     @staticmethod | ||||
|     def _fix_subtitles(subtitles): | ||||
|         """ Convert millisecond-based subtitles to SRT """ | ||||
|         if subtitles is None: | ||||
|             return subtitles  # subtitles not requested | ||||
|  | ||||
|         def _msectotimecode(msec): | ||||
|             """ Helper utility to convert milliseconds to timecode """ | ||||
| @@ -149,7 +152,4 @@ class CeskaTelevizeIE(SubtitlesInfoExtractor): | ||||
|                 else: | ||||
|                     yield line | ||||
|  | ||||
|         fixed_subtitles = {} | ||||
|         for k, v in subtitles.items(): | ||||
|             fixed_subtitles[k] = "\r\n".join(_fix_subtitle(v)) | ||||
|         return fixed_subtitles | ||||
|         return "\r\n".join(_fix_subtitle(subtitles)) | ||||
|   | ||||
| @@ -57,7 +57,7 @@ class ChilloutzoneIE(InfoExtractor): | ||||
|  | ||||
|         base64_video_info = self._html_search_regex( | ||||
|             r'var cozVidData = "(.+?)";', webpage, 'video data') | ||||
|         decoded_video_info = base64.b64decode(base64_video_info).decode("utf-8") | ||||
|         decoded_video_info = base64.b64decode(base64_video_info.encode('utf-8')).decode('utf-8') | ||||
|         video_info_dict = json.loads(decoded_video_info) | ||||
|  | ||||
|         # get video information from dict | ||||
|   | ||||
							
								
								
									
										84
									
								
								youtube_dl/extractor/chirbit.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								youtube_dl/extractor/chirbit.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     parse_duration, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class ChirbitIE(InfoExtractor): | ||||
|     IE_NAME = 'chirbit' | ||||
|     _VALID_URL = r'https?://(?:www\.)?chirb\.it/(?:(?:wp|pl)/|fb_chirbit_player\.swf\?key=)?(?P<id>[\da-zA-Z]+)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://chirb.it/PrIPv5', | ||||
|         'md5': '9847b0dad6ac3e074568bf2cfb197de8', | ||||
|         'info_dict': { | ||||
|             'id': 'PrIPv5', | ||||
|             'ext': 'mp3', | ||||
|             'title': 'Фасадстрой', | ||||
|             'duration': 52, | ||||
|             'view_count': int, | ||||
|             'comment_count': int, | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'https://chirb.it/fb_chirbit_player.swf?key=PrIPv5', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         audio_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage( | ||||
|             'http://chirb.it/%s' % audio_id, audio_id) | ||||
|  | ||||
|         audio_url = self._search_regex( | ||||
|             r'"setFile"\s*,\s*"([^"]+)"', webpage, 'audio url') | ||||
|  | ||||
|         title = self._search_regex( | ||||
|             r'itemprop="name">([^<]+)', webpage, 'title') | ||||
|         duration = parse_duration(self._html_search_meta( | ||||
|             'duration', webpage, 'duration', fatal=False)) | ||||
|         view_count = int_or_none(self._search_regex( | ||||
|             r'itemprop="playCount"\s*>(\d+)', webpage, | ||||
|             'listen count', fatal=False)) | ||||
|         comment_count = int_or_none(self._search_regex( | ||||
|             r'>(\d+) Comments?:', webpage, | ||||
|             'comment count', fatal=False)) | ||||
|  | ||||
|         return { | ||||
|             'id': audio_id, | ||||
|             'url': audio_url, | ||||
|             'title': title, | ||||
|             'duration': duration, | ||||
|             'view_count': view_count, | ||||
|             'comment_count': comment_count, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class ChirbitProfileIE(InfoExtractor): | ||||
|     IE_NAME = 'chirbit:profile' | ||||
|     _VALID_URL = r'https?://(?:www\.)?chirbit.com/(?:rss/)?(?P<id>[^/]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://chirbit.com/ScarletBeauty', | ||||
|         'info_dict': { | ||||
|             'id': 'ScarletBeauty', | ||||
|             'title': 'Chirbits by ScarletBeauty', | ||||
|         }, | ||||
|         'playlist_mincount': 3, | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         profile_id = self._match_id(url) | ||||
|  | ||||
|         rss = self._download_xml( | ||||
|             'http://chirbit.com/rss/%s' % profile_id, profile_id) | ||||
|  | ||||
|         entries = [ | ||||
|             self.url_result(audio_url.text, 'Chirbit') | ||||
|             for audio_url in rss.findall('./channel/item/link')] | ||||
|  | ||||
|         title = rss.find('./channel/title').text | ||||
|  | ||||
|         return self.playlist_result(entries, profile_id, title) | ||||
							
								
								
									
										110
									
								
								youtube_dl/extractor/cinemassacre.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								youtube_dl/extractor/cinemassacre.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ExtractorError | ||||
| from .bliptv import BlipTVIE | ||||
|  | ||||
|  | ||||
| class CinemassacreIE(InfoExtractor): | ||||
|     _VALID_URL = 'https?://(?:www\.)?cinemassacre\.com/(?P<date_y>[0-9]{4})/(?P<date_m>[0-9]{2})/(?P<date_d>[0-9]{2})/(?P<display_id>[^?#/]+)' | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://cinemassacre.com/2012/11/10/avgn-the-movie-trailer/', | ||||
|             'md5': 'fde81fbafaee331785f58cd6c0d46190', | ||||
|             'info_dict': { | ||||
|                 'id': 'Cinemassacre-19911', | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20121110', | ||||
|                 'title': '“Angry Video Game Nerd: The Movie” – Trailer', | ||||
|                 'description': 'md5:fb87405fcb42a331742a0dce2708560b', | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://cinemassacre.com/2013/10/02/the-mummys-hand-1940', | ||||
|             'md5': 'd72f10cd39eac4215048f62ab477a511', | ||||
|             'info_dict': { | ||||
|                 'id': 'Cinemassacre-521be8ef82b16', | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20131002', | ||||
|                 'title': 'The Mummy’s Hand (1940)', | ||||
|             }, | ||||
|         }, | ||||
|         { | ||||
|             # blip.tv embedded video | ||||
|             'url': 'http://cinemassacre.com/2006/12/07/chronologically-confused-about-bad-movie-and-video-game-sequel-titles/', | ||||
|             'md5': 'ca9b3c8dd5a66f9375daeb5135f5a3de', | ||||
|             'info_dict': { | ||||
|                 'id': '4065369', | ||||
|                 'ext': 'flv', | ||||
|                 'title': 'AVGN: Chronologically Confused about Bad Movie and Video Game Sequel Titles', | ||||
|                 'upload_date': '20061207', | ||||
|                 'uploader': 'cinemassacre', | ||||
|                 'uploader_id': '250778', | ||||
|                 'timestamp': 1283233867, | ||||
|                 'description': 'md5:0a108c78d130676b207d0f6d029ecffd', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             # Youtube embedded video | ||||
|             'url': 'http://cinemassacre.com/2006/09/01/mckids/', | ||||
|             'md5': '6eb30961fa795fedc750eac4881ad2e1', | ||||
|             'info_dict': { | ||||
|                 'id': 'FnxsNhuikpo', | ||||
|                 'ext': 'mp4', | ||||
|                 'upload_date': '20060901', | ||||
|                 'uploader': 'Cinemassacre Extras', | ||||
|                 'description': 'md5:de9b751efa9e45fbaafd9c8a1123ed53', | ||||
|                 'uploader_id': 'Cinemassacre', | ||||
|                 'title': 'AVGN: McKids', | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://cinemassacre.com/2015/05/25/mario-kart-64-nintendo-64-james-mike-mondays/', | ||||
|             'md5': '1376908e49572389e7b06251a53cdd08', | ||||
|             'info_dict': { | ||||
|                 'id': 'Cinemassacre-555779690c440', | ||||
|                 'ext': 'mp4', | ||||
|                 'description': 'Let’s Play Mario Kart 64 !! Mario Kart 64 is a classic go-kart racing game released for the Nintendo 64 (N64). Today James & Mike do 4 player Battle Mode with Kyle and Bootsy!', | ||||
|                 'title': 'Mario Kart 64 (Nintendo 64) James & Mike Mondays', | ||||
|                 'upload_date': '20150525', | ||||
|             } | ||||
|         } | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         display_id = mobj.group('display_id') | ||||
|         video_date = mobj.group('date_y') + mobj.group('date_m') + mobj.group('date_d') | ||||
|  | ||||
|         webpage = self._download_webpage(url, display_id) | ||||
|  | ||||
|         playerdata_url = self._search_regex( | ||||
|             [ | ||||
|                 r'src="(http://(?:player2\.screenwavemedia\.com|player\.screenwavemedia\.com/play)/[a-zA-Z]+\.php\?[^"]*\bid=.+?)"', | ||||
|                 r'<iframe[^>]+src="((?:https?:)?//(?:[^.]+\.)?youtube\.com/.+?)"', | ||||
|             ], | ||||
|             webpage, 'player data URL', default=None) | ||||
|         if not playerdata_url: | ||||
|             playerdata_url = BlipTVIE._extract_url(webpage) | ||||
|         if not playerdata_url: | ||||
|             raise ExtractorError('Unable to find player data') | ||||
|  | ||||
|         video_title = self._html_search_regex( | ||||
|             r'<title>(?P<title>.+?)\|', webpage, 'title') | ||||
|         video_description = self._html_search_regex( | ||||
|             r'<div class="entry-content">(?P<description>.+?)</div>', | ||||
|             webpage, 'description', flags=re.DOTALL, fatal=False) | ||||
|         video_thumbnail = self._og_search_thumbnail(webpage) | ||||
|  | ||||
|         return { | ||||
|             '_type': 'url_transparent', | ||||
|             'display_id': display_id, | ||||
|             'title': video_title, | ||||
|             'description': video_description, | ||||
|             'upload_date': video_date, | ||||
|             'thumbnail': video_thumbnail, | ||||
|             'url': playerdata_url, | ||||
|         } | ||||
| @@ -105,6 +105,7 @@ class CloudyIE(InfoExtractor): | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         file_key = self._search_regex( | ||||
|             r'filekey\s*=\s*"([^"]+)"', webpage, 'file_key') | ||||
|             [r'key\s*:\s*"([^"]+)"', r'filekey\s*=\s*"([^"]+)"'], | ||||
|             webpage, 'file_key') | ||||
|  | ||||
|         return self._extract_video(video_host, video_id, file_key) | ||||
|   | ||||
| @@ -11,7 +11,7 @@ from ..utils import ( | ||||
|  | ||||
| class CNETIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?cnet\.com/videos/(?P<id>[^/]+)/' | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.cnet.com/videos/hands-on-with-microsofts-windows-8-1-update/', | ||||
|         'info_dict': { | ||||
|             'id': '56f4ea68-bd21-4852-b08c-4de5b8354c60', | ||||
| @@ -25,7 +25,20 @@ class CNETIE(InfoExtractor): | ||||
|         'params': { | ||||
|             'skip_download': 'requires rtmpdump', | ||||
|         } | ||||
|     } | ||||
|     }, { | ||||
|         'url': 'http://www.cnet.com/videos/whiny-pothole-tweets-at-local-government-when-hit-by-cars-tomorrow-daily-187/', | ||||
|         'info_dict': { | ||||
|             'id': '56527b93-d25d-44e3-b738-f989ce2e49ba', | ||||
|             'ext': 'flv', | ||||
|             'description': 'Khail and Ashley wonder what other civic woes can be solved by self-tweeting objects, investigate a new kind of VR camera and watch an origami robot self-assemble, walk, climb, dig and dissolve. #TDPothole', | ||||
|             'uploader_id': 'b163284d-6b73-44fc-b3e6-3da66c392d40', | ||||
|             'uploader': 'Ashley Esqueda', | ||||
|             'title': 'Whiny potholes tweet at local government when hit by cars (Tomorrow Daily 187)', | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True,  # requires rtmpdump | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         display_id = self._match_id(url) | ||||
| @@ -42,7 +55,7 @@ class CNETIE(InfoExtractor): | ||||
|             raise ExtractorError('Cannot find video data') | ||||
|  | ||||
|         mpx_account = data['config']['players']['default']['mpx_account'] | ||||
|         vid = vdata['files']['rtmp'] | ||||
|         vid = vdata['files'].get('rtmp', vdata['files']['hds']) | ||||
|         tp_link = 'http://link.theplatform.com/s/%s/%s' % (mpx_account, vid) | ||||
|  | ||||
|         video_id = vdata['id'] | ||||
|   | ||||
| @@ -12,7 +12,7 @@ from ..utils import ( | ||||
|  | ||||
| class CNNIE(InfoExtractor): | ||||
|     _VALID_URL = r'''(?x)https?://(?:(?:edition|www)\.)?cnn\.com/video/(?:data/.+?|\?)/ | ||||
|         (?P<path>.+?/(?P<title>[^/]+?)(?:\.(?:cnn|hln)(?:-ap)?|(?=&)))''' | ||||
|         (?P<path>.+?/(?P<title>[^/]+?)(?:\.(?:[a-z\-]+)|(?=&)))''' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://edition.cnn.com/video/?/video/sports/2013/06/09/nadal-1-on-1.cnn', | ||||
| @@ -45,6 +45,12 @@ class CNNIE(InfoExtractor): | ||||
|             'description': 'md5:e7223a503315c9f150acac52e76de086', | ||||
|             'upload_date': '20141222', | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'http://cnn.com/video/?/video/politics/2015/03/27/pkg-arizona-senator-church-attendance-mandatory.ktvk', | ||||
|         'only_matching': True, | ||||
|     }, { | ||||
|         'url': 'http://cnn.com/video/?/video/us/2015/04/06/dnt-baker-refuses-anti-gay-order.wkmg', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|   | ||||
| @@ -201,7 +201,7 @@ class ComedyCentralShowsIE(MTVServicesInfoExtractor): | ||||
|  | ||||
|         uri = mMovieParams[0][1] | ||||
|         # Correct cc.com in uri | ||||
|         uri = re.sub(r'(episode:[^.]+)(\.cc)?\.com', r'\1.cc.com', uri) | ||||
|         uri = re.sub(r'(episode:[^.]+)(\.cc)?\.com', r'\1.com', uri) | ||||
|  | ||||
|         index_url = 'http://%s.cc.com/feeds/mrss?%s' % (show_name, compat_urllib_parse.urlencode({'uri': uri})) | ||||
|         idoc = self._download_xml( | ||||
| @@ -250,6 +250,8 @@ class ComedyCentralShowsIE(MTVServicesInfoExtractor): | ||||
|                 }) | ||||
|                 self._sort_formats(formats) | ||||
|  | ||||
|             subtitles = self._extract_subtitles(cdoc, guid) | ||||
|  | ||||
|             virtual_id = show_name + ' ' + epTitle + ' part ' + compat_str(part_num + 1) | ||||
|             entries.append({ | ||||
|                 'id': guid, | ||||
| @@ -260,6 +262,7 @@ class ComedyCentralShowsIE(MTVServicesInfoExtractor): | ||||
|                 'duration': duration, | ||||
|                 'thumbnail': thumbnail, | ||||
|                 'description': description, | ||||
|                 'subtitles': subtitles, | ||||
|             }) | ||||
|  | ||||
|         return { | ||||
|   | ||||
| @@ -23,11 +23,11 @@ from ..compat import ( | ||||
| ) | ||||
| from ..utils import ( | ||||
|     age_restricted, | ||||
|     bug_reports_message, | ||||
|     clean_html, | ||||
|     compiled_regex_type, | ||||
|     ExtractorError, | ||||
|     float_or_none, | ||||
|     HEADRequest, | ||||
|     int_or_none, | ||||
|     RegexNotFoundError, | ||||
|     sanitize_filename, | ||||
| @@ -47,7 +47,7 @@ class InfoExtractor(object): | ||||
|     information possibly downloading the video to the file system, among | ||||
|     other possible outcomes. | ||||
|  | ||||
|     The type field determines the the type of the result. | ||||
|     The type field determines the type of the result. | ||||
|     By far the most common value (and the default if _type is missing) is | ||||
|     "video", which indicates a single video. | ||||
|  | ||||
| @@ -111,11 +111,8 @@ class InfoExtractor(object): | ||||
|                                   (quality takes higher priority) | ||||
|                                  -1 for default (order by other properties), | ||||
|                                  -2 or smaller for less than default. | ||||
|                     * http_method  HTTP method to use for the download. | ||||
|                     * http_headers  A dictionary of additional HTTP headers | ||||
|                                  to add to the request. | ||||
|                     * http_post_data  Additional data to send with a POST | ||||
|                                  request. | ||||
|                     * stretched_ratio  If given and not 1, indicates that the | ||||
|                                  video's pixels are not square. | ||||
|                                  width : height ratio as float. | ||||
| @@ -151,12 +148,19 @@ class InfoExtractor(object): | ||||
|                     If not explicitly set, calculated from timestamp. | ||||
|     uploader_id:    Nickname or id of the video uploader. | ||||
|     location:       Physical location where the video was filmed. | ||||
|     subtitles:      The subtitle file contents as a dictionary in the format | ||||
|                     {language: subtitles}. | ||||
|     subtitles:      The available subtitles as a dictionary in the format | ||||
|                     {language: subformats}. "subformats" is a list sorted from | ||||
|                     lower to higher preference, each element is a dictionary | ||||
|                     with the "ext" entry and one of: | ||||
|                         * "data": The subtitles file contents | ||||
|                         * "url": A url pointing to the subtitles file | ||||
|     automatic_captions: Like 'subtitles', used by the YoutubeIE for | ||||
|                     automatically generated captions | ||||
|     duration:       Length of the video in seconds, as an integer. | ||||
|     view_count:     How many users have watched the video on the platform. | ||||
|     like_count:     Number of positive ratings of the video | ||||
|     dislike_count:  Number of negative ratings of the video | ||||
|     average_rating: Average rating give by users, the scale used depends on the webpage | ||||
|     comment_count:  Number of comments on the video | ||||
|     comments:       A list of comments, each with one or more of the following | ||||
|                     properties (all but one of text or html optional): | ||||
| @@ -264,8 +268,15 @@ class InfoExtractor(object): | ||||
|  | ||||
|     def extract(self, url): | ||||
|         """Extracts URL information and returns it in list of dicts.""" | ||||
|         self.initialize() | ||||
|         return self._real_extract(url) | ||||
|         try: | ||||
|             self.initialize() | ||||
|             return self._real_extract(url) | ||||
|         except ExtractorError: | ||||
|             raise | ||||
|         except compat_http_client.IncompleteRead as e: | ||||
|             raise ExtractorError('A network error has occured.', cause=e, expected=True) | ||||
|         except (KeyError, StopIteration) as e: | ||||
|             raise ExtractorError('An extractor error has occured.', cause=e) | ||||
|  | ||||
|     def set_downloader(self, downloader): | ||||
|         """Sets the downloader for this IE.""" | ||||
| @@ -311,7 +322,7 @@ class InfoExtractor(object): | ||||
|                 self._downloader.report_warning(errmsg) | ||||
|                 return False | ||||
|  | ||||
|     def _download_webpage_handle(self, url_or_request, video_id, note=None, errnote=None, fatal=True): | ||||
|     def _download_webpage_handle(self, url_or_request, video_id, note=None, errnote=None, fatal=True, encoding=None): | ||||
|         """ Returns a tuple (page content as string, URL handle) """ | ||||
|         # Strip hashes from the URL (#1038) | ||||
|         if isinstance(url_or_request, (compat_str, str)): | ||||
| @@ -321,14 +332,11 @@ class InfoExtractor(object): | ||||
|         if urlh is False: | ||||
|             assert not fatal | ||||
|             return False | ||||
|         content = self._webpage_read_content(urlh, url_or_request, video_id, note, errnote, fatal) | ||||
|         content = self._webpage_read_content(urlh, url_or_request, video_id, note, errnote, fatal, encoding=encoding) | ||||
|         return (content, urlh) | ||||
|  | ||||
|     def _webpage_read_content(self, urlh, url_or_request, video_id, note=None, errnote=None, fatal=True, prefix=None): | ||||
|         content_type = urlh.headers.get('Content-Type', '') | ||||
|         webpage_bytes = urlh.read() | ||||
|         if prefix is not None: | ||||
|             webpage_bytes = prefix + webpage_bytes | ||||
|     @staticmethod | ||||
|     def _guess_encoding_from_content(content_type, webpage_bytes): | ||||
|         m = re.match(r'[a-zA-Z0-9_.-]+/[a-zA-Z0-9_.-]+\s*;\s*charset=(.+)', content_type) | ||||
|         if m: | ||||
|             encoding = m.group(1) | ||||
| @@ -341,6 +349,16 @@ class InfoExtractor(object): | ||||
|                 encoding = 'utf-16' | ||||
|             else: | ||||
|                 encoding = 'utf-8' | ||||
|  | ||||
|         return encoding | ||||
|  | ||||
|     def _webpage_read_content(self, urlh, url_or_request, video_id, note=None, errnote=None, fatal=True, prefix=None, encoding=None): | ||||
|         content_type = urlh.headers.get('Content-Type', '') | ||||
|         webpage_bytes = urlh.read() | ||||
|         if prefix is not None: | ||||
|             webpage_bytes = prefix + webpage_bytes | ||||
|         if not encoding: | ||||
|             encoding = self._guess_encoding_from_content(content_type, webpage_bytes) | ||||
|         if self._downloader.params.get('dump_intermediate_pages', False): | ||||
|             try: | ||||
|                 url = url_or_request.get_full_url() | ||||
| @@ -384,16 +402,26 @@ class InfoExtractor(object): | ||||
|             if blocked_iframe: | ||||
|                 msg += ' Visit %s for more details' % blocked_iframe | ||||
|             raise ExtractorError(msg, expected=True) | ||||
|         if '<title>The URL you requested has been blocked</title>' in content[:512]: | ||||
|             msg = ( | ||||
|                 'Access to this webpage has been blocked by Indian censorship. ' | ||||
|                 'Use a VPN or proxy server (with --proxy) to route around it.') | ||||
|             block_msg = self._html_search_regex( | ||||
|                 r'</h1><p>(.*?)</p>', | ||||
|                 content, 'block message', default=None) | ||||
|             if block_msg: | ||||
|                 msg += ' (Message: "%s")' % block_msg.replace('\n', ' ') | ||||
|             raise ExtractorError(msg, expected=True) | ||||
|  | ||||
|         return content | ||||
|  | ||||
|     def _download_webpage(self, url_or_request, video_id, note=None, errnote=None, fatal=True, tries=1, timeout=5): | ||||
|     def _download_webpage(self, url_or_request, video_id, note=None, errnote=None, fatal=True, tries=1, timeout=5, encoding=None): | ||||
|         """ Returns the data of the page as a string """ | ||||
|         success = False | ||||
|         try_count = 0 | ||||
|         while success is False: | ||||
|             try: | ||||
|                 res = self._download_webpage_handle(url_or_request, video_id, note, errnote, fatal) | ||||
|                 res = self._download_webpage_handle(url_or_request, video_id, note, errnote, fatal, encoding=encoding) | ||||
|                 success = True | ||||
|             except compat_http_client.IncompleteRead as e: | ||||
|                 try_count += 1 | ||||
| @@ -408,10 +436,10 @@ class InfoExtractor(object): | ||||
|  | ||||
|     def _download_xml(self, url_or_request, video_id, | ||||
|                       note='Downloading XML', errnote='Unable to download XML', | ||||
|                       transform_source=None, fatal=True): | ||||
|                       transform_source=None, fatal=True, encoding=None): | ||||
|         """Return the xml as an xml.etree.ElementTree.Element""" | ||||
|         xml_string = self._download_webpage( | ||||
|             url_or_request, video_id, note, errnote, fatal=fatal) | ||||
|             url_or_request, video_id, note, errnote, fatal=fatal, encoding=encoding) | ||||
|         if xml_string is False: | ||||
|             return xml_string | ||||
|         if transform_source: | ||||
| @@ -422,9 +450,10 @@ class InfoExtractor(object): | ||||
|                        note='Downloading JSON metadata', | ||||
|                        errnote='Unable to download JSON metadata', | ||||
|                        transform_source=None, | ||||
|                        fatal=True): | ||||
|                        fatal=True, encoding=None): | ||||
|         json_string = self._download_webpage( | ||||
|             url_or_request, video_id, note, errnote, fatal=fatal) | ||||
|             url_or_request, video_id, note, errnote, fatal=fatal, | ||||
|             encoding=encoding) | ||||
|         if (not fatal) and json_string is False: | ||||
|             return None | ||||
|         return self._parse_json( | ||||
| @@ -469,7 +498,7 @@ class InfoExtractor(object): | ||||
|  | ||||
|     # Methods for following #608 | ||||
|     @staticmethod | ||||
|     def url_result(url, ie=None, video_id=None): | ||||
|     def url_result(url, ie=None, video_id=None, video_title=None): | ||||
|         """Returns a url that points to a page that should be processed""" | ||||
|         # TODO: ie should be the class used for getting the info | ||||
|         video_info = {'_type': 'url', | ||||
| @@ -477,6 +506,8 @@ class InfoExtractor(object): | ||||
|                       'ie_key': ie} | ||||
|         if video_id is not None: | ||||
|             video_info['id'] = video_id | ||||
|         if video_title is not None: | ||||
|             video_info['title'] = video_title | ||||
|         return video_info | ||||
|  | ||||
|     @staticmethod | ||||
| @@ -507,7 +538,7 @@ class InfoExtractor(object): | ||||
|                 if mobj: | ||||
|                     break | ||||
|  | ||||
|         if os.name != 'nt' and sys.stderr.isatty(): | ||||
|         if not self._downloader.params.get('no_color') and os.name != 'nt' and sys.stderr.isatty(): | ||||
|             _name = '\033[0;34m%s\033[0m' % name | ||||
|         else: | ||||
|             _name = name | ||||
| @@ -523,8 +554,7 @@ class InfoExtractor(object): | ||||
|         elif fatal: | ||||
|             raise RegexNotFoundError('Unable to extract %s' % _name) | ||||
|         else: | ||||
|             self._downloader.report_warning('unable to extract %s; ' | ||||
|                                             'please report this issue on http://yt-dl.org/bug' % _name) | ||||
|             self._downloader.report_warning('unable to extract %s' % _name + bug_reports_message()) | ||||
|             return None | ||||
|  | ||||
|     def _html_search_regex(self, pattern, string, name, default=_NO_DEFAULT, fatal=True, flags=0, group=None): | ||||
| @@ -539,7 +569,7 @@ class InfoExtractor(object): | ||||
|  | ||||
|     def _get_login_info(self): | ||||
|         """ | ||||
|         Get the the login info as (username, password) | ||||
|         Get the login info as (username, password) | ||||
|         It will look in the netrc file using the _NETRC_MACHINE value | ||||
|         If there's no info available, return (None, None) | ||||
|         """ | ||||
| @@ -657,7 +687,7 @@ class InfoExtractor(object): | ||||
|         return RATING_TABLE.get(rating.lower(), None) | ||||
|  | ||||
|     def _family_friendly_search(self, html): | ||||
|         # See http://schema.org/VideoObj | ||||
|         # See http://schema.org/VideoObject | ||||
|         family_friendly = self._html_search_meta('isFamilyFriendly', html) | ||||
|  | ||||
|         if not family_friendly: | ||||
| @@ -675,7 +705,7 @@ class InfoExtractor(object): | ||||
|         return self._html_search_meta('twitter:player', html, | ||||
|                                       'twitter card player') | ||||
|  | ||||
|     def _sort_formats(self, formats): | ||||
|     def _sort_formats(self, formats, field_preference=None): | ||||
|         if not formats: | ||||
|             raise ExtractorError('No video formats found') | ||||
|  | ||||
| @@ -685,6 +715,9 @@ class InfoExtractor(object): | ||||
|             if not f.get('ext') and 'url' in f: | ||||
|                 f['ext'] = determine_ext(f['url']) | ||||
|  | ||||
|             if isinstance(field_preference, (list, tuple)): | ||||
|                 return tuple(f.get(field) if f.get(field) is not None else -1 for field in field_preference) | ||||
|  | ||||
|             preference = f.get('preference') | ||||
|             if preference is None: | ||||
|                 proto = f.get('protocol') | ||||
| @@ -721,6 +754,7 @@ class InfoExtractor(object): | ||||
|                 f.get('language_preference') if f.get('language_preference') is not None else -1, | ||||
|                 f.get('quality') if f.get('quality') is not None else -1, | ||||
|                 f.get('tbr') if f.get('tbr') is not None else -1, | ||||
|                 f.get('filesize') if f.get('filesize') is not None else -1, | ||||
|                 f.get('vbr') if f.get('vbr') is not None else -1, | ||||
|                 f.get('height') if f.get('height') is not None else -1, | ||||
|                 f.get('width') if f.get('width') is not None else -1, | ||||
| @@ -728,10 +762,9 @@ class InfoExtractor(object): | ||||
|                 f.get('abr') if f.get('abr') is not None else -1, | ||||
|                 audio_ext_preference, | ||||
|                 f.get('fps') if f.get('fps') is not None else -1, | ||||
|                 f.get('filesize') if f.get('filesize') is not None else -1, | ||||
|                 f.get('filesize_approx') if f.get('filesize_approx') is not None else -1, | ||||
|                 f.get('source_preference') if f.get('source_preference') is not None else -1, | ||||
|                 f.get('format_id'), | ||||
|                 f.get('format_id') if f.get('format_id') is not None else '', | ||||
|             ) | ||||
|         formats.sort(key=_formats_key) | ||||
|  | ||||
| @@ -744,15 +777,17 @@ class InfoExtractor(object): | ||||
|                 formats) | ||||
|  | ||||
|     def _is_valid_url(self, url, video_id, item='video'): | ||||
|         url = self._proto_relative_url(url, scheme='http:') | ||||
|         # For now assume non HTTP(S) URLs always valid | ||||
|         if not (url.startswith('http://') or url.startswith('https://')): | ||||
|             return True | ||||
|         try: | ||||
|             self._request_webpage( | ||||
|                 HEADRequest(url), video_id, | ||||
|                 'Checking %s URL' % item) | ||||
|             self._request_webpage(url, video_id, 'Checking %s URL' % item) | ||||
|             return True | ||||
|         except ExtractorError as e: | ||||
|             if isinstance(e.cause, compat_HTTPError): | ||||
|                 self.report_warning( | ||||
|                     '%s URL is invalid, skipping' % item, video_id) | ||||
|                 self.to_screen( | ||||
|                     '%s: %s URL is invalid, skipping' % (video_id, item)) | ||||
|                 return False | ||||
|             raise | ||||
|  | ||||
| @@ -793,11 +828,11 @@ class InfoExtractor(object): | ||||
|             media_nodes = manifest.findall('{http://ns.adobe.com/f4m/2.0}media') | ||||
|         for i, media_el in enumerate(media_nodes): | ||||
|             if manifest_version == '2.0': | ||||
|                 manifest_url = ('/'.join(manifest_url.split('/')[:-1]) + '/' | ||||
|                                 + (media_el.attrib.get('href') or media_el.attrib.get('url'))) | ||||
|                 manifest_url = ('/'.join(manifest_url.split('/')[:-1]) + '/' + | ||||
|                                 (media_el.attrib.get('href') or media_el.attrib.get('url'))) | ||||
|             tbr = int_or_none(media_el.attrib.get('bitrate')) | ||||
|             formats.append({ | ||||
|                 'format_id': '-'.join(filter(None, [f4m_id, 'f4m-%d' % (i if tbr is None else tbr)])), | ||||
|                 'format_id': '-'.join(filter(None, [f4m_id, compat_str(i if tbr is None else tbr)])), | ||||
|                 'url': manifest_url, | ||||
|                 'ext': 'flv', | ||||
|                 'tbr': tbr, | ||||
| @@ -811,14 +846,14 @@ class InfoExtractor(object): | ||||
|  | ||||
|     def _extract_m3u8_formats(self, m3u8_url, video_id, ext=None, | ||||
|                               entry_protocol='m3u8', preference=None, | ||||
|                               m3u8_id=None): | ||||
|                               m3u8_id=None, note=None, errnote=None): | ||||
|  | ||||
|         formats = [{ | ||||
|             'format_id': '-'.join(filter(None, [m3u8_id, 'm3u8-meta'])), | ||||
|             'format_id': '-'.join(filter(None, [m3u8_id, 'meta'])), | ||||
|             'url': m3u8_url, | ||||
|             'ext': ext, | ||||
|             'protocol': 'm3u8', | ||||
|             'preference': -1, | ||||
|             'preference': preference - 1 if preference else -1, | ||||
|             'resolution': 'multiple', | ||||
|             'format_note': 'Quality selection URL', | ||||
|         }] | ||||
| @@ -830,9 +865,10 @@ class InfoExtractor(object): | ||||
|  | ||||
|         m3u8_doc = self._download_webpage( | ||||
|             m3u8_url, video_id, | ||||
|             note='Downloading m3u8 information', | ||||
|             errnote='Failed to download m3u8 information') | ||||
|             note=note or 'Downloading m3u8 information', | ||||
|             errnote=errnote or 'Failed to download m3u8 information') | ||||
|         last_info = None | ||||
|         last_media = None | ||||
|         kv_rex = re.compile( | ||||
|             r'(?P<key>[a-zA-Z_-]+)=(?P<val>"[^"]+"|[^",]+)(?:,|$)') | ||||
|         for line in m3u8_doc.splitlines(): | ||||
| @@ -843,6 +879,13 @@ class InfoExtractor(object): | ||||
|                     if v.startswith('"'): | ||||
|                         v = v[1:-1] | ||||
|                     last_info[m.group('key')] = v | ||||
|             elif line.startswith('#EXT-X-MEDIA:'): | ||||
|                 last_media = {} | ||||
|                 for m in kv_rex.finditer(line): | ||||
|                     v = m.group('val') | ||||
|                     if v.startswith('"'): | ||||
|                         v = v[1:-1] | ||||
|                     last_media[m.group('key')] = v | ||||
|             elif line.startswith('#') or not line.strip(): | ||||
|                 continue | ||||
|             else: | ||||
| @@ -850,8 +893,13 @@ class InfoExtractor(object): | ||||
|                     formats.append({'url': format_url(line)}) | ||||
|                     continue | ||||
|                 tbr = int_or_none(last_info.get('BANDWIDTH'), scale=1000) | ||||
|                 format_id = [] | ||||
|                 if m3u8_id: | ||||
|                     format_id.append(m3u8_id) | ||||
|                 last_media_name = last_media.get('NAME') if last_media and last_media.get('TYPE') != 'SUBTITLES' else None | ||||
|                 format_id.append(last_media_name if last_media_name else '%d' % (tbr if tbr else len(formats))) | ||||
|                 f = { | ||||
|                     'format_id': '-'.join(filter(None, [m3u8_id, 'm3u8-%d' % (tbr if tbr else len(formats))])), | ||||
|                     'format_id': '-'.join(format_id), | ||||
|                     'url': format_url(line.strip()), | ||||
|                     'tbr': tbr, | ||||
|                     'ext': ext, | ||||
| @@ -871,6 +919,9 @@ class InfoExtractor(object): | ||||
|                     width_str, height_str = resolution.split('x') | ||||
|                     f['width'] = int(width_str) | ||||
|                     f['height'] = int(height_str) | ||||
|                 if last_media is not None: | ||||
|                     f['m3u8_media'] = last_media | ||||
|                     last_media = None | ||||
|                 formats.append(f) | ||||
|                 last_info = {} | ||||
|         self._sort_formats(formats) | ||||
| @@ -889,39 +940,57 @@ class InfoExtractor(object): | ||||
|  | ||||
|         formats = [] | ||||
|         rtmp_count = 0 | ||||
|         for video in smil.findall('./body/switch/video'): | ||||
|             src = video.get('src') | ||||
|             if not src: | ||||
|                 continue | ||||
|             bitrate = int_or_none(video.get('system-bitrate') or video.get('systemBitrate'), 1000) | ||||
|             width = int_or_none(video.get('width')) | ||||
|             height = int_or_none(video.get('height')) | ||||
|             proto = video.get('proto') | ||||
|             if not proto: | ||||
|                 if base: | ||||
|                     if base.startswith('rtmp'): | ||||
|                         proto = 'rtmp' | ||||
|                     elif base.startswith('http'): | ||||
|                         proto = 'http' | ||||
|             ext = video.get('ext') | ||||
|             if proto == 'm3u8': | ||||
|                 formats.extend(self._extract_m3u8_formats(src, video_id, ext)) | ||||
|             elif proto == 'rtmp': | ||||
|                 rtmp_count += 1 | ||||
|                 streamer = video.get('streamer') or base | ||||
|                 formats.append({ | ||||
|                     'url': streamer, | ||||
|                     'play_path': src, | ||||
|                     'ext': 'flv', | ||||
|                     'format_id': 'rtmp-%d' % (rtmp_count if bitrate is None else bitrate), | ||||
|                     'tbr': bitrate, | ||||
|                     'width': width, | ||||
|                     'height': height, | ||||
|                 }) | ||||
|         if smil.findall('./body/seq/video'): | ||||
|             video = smil.findall('./body/seq/video')[0] | ||||
|             fmts, rtmp_count = self._parse_smil_video(video, video_id, base, rtmp_count) | ||||
|             formats.extend(fmts) | ||||
|         else: | ||||
|             for video in smil.findall('./body/switch/video'): | ||||
|                 fmts, rtmp_count = self._parse_smil_video(video, video_id, base, rtmp_count) | ||||
|                 formats.extend(fmts) | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return formats | ||||
|  | ||||
|     def _parse_smil_video(self, video, video_id, base, rtmp_count): | ||||
|         src = video.get('src') | ||||
|         if not src: | ||||
|             return ([], rtmp_count) | ||||
|         bitrate = int_or_none(video.get('system-bitrate') or video.get('systemBitrate'), 1000) | ||||
|         width = int_or_none(video.get('width')) | ||||
|         height = int_or_none(video.get('height')) | ||||
|         proto = video.get('proto') | ||||
|         if not proto: | ||||
|             if base: | ||||
|                 if base.startswith('rtmp'): | ||||
|                     proto = 'rtmp' | ||||
|                 elif base.startswith('http'): | ||||
|                     proto = 'http' | ||||
|         ext = video.get('ext') | ||||
|         if proto == 'm3u8': | ||||
|             return (self._extract_m3u8_formats(src, video_id, ext), rtmp_count) | ||||
|         elif proto == 'rtmp': | ||||
|             rtmp_count += 1 | ||||
|             streamer = video.get('streamer') or base | ||||
|             return ([{ | ||||
|                 'url': streamer, | ||||
|                 'play_path': src, | ||||
|                 'ext': 'flv', | ||||
|                 'format_id': 'rtmp-%d' % (rtmp_count if bitrate is None else bitrate), | ||||
|                 'tbr': bitrate, | ||||
|                 'width': width, | ||||
|                 'height': height, | ||||
|             }], rtmp_count) | ||||
|         elif proto.startswith('http'): | ||||
|             return ([{ | ||||
|                 'url': base + src, | ||||
|                 'ext': ext or 'flv', | ||||
|                 'tbr': bitrate, | ||||
|                 'width': width, | ||||
|                 'height': height, | ||||
|             }], rtmp_count) | ||||
|  | ||||
|     def _live_title(self, name): | ||||
|         """ Generate the title for a live video """ | ||||
|         now = datetime.datetime.now() | ||||
| @@ -985,6 +1054,24 @@ class InfoExtractor(object): | ||||
|             any_restricted = any_restricted or is_restricted | ||||
|         return not any_restricted | ||||
|  | ||||
|     def extract_subtitles(self, *args, **kwargs): | ||||
|         if (self._downloader.params.get('writesubtitles', False) or | ||||
|                 self._downloader.params.get('listsubtitles')): | ||||
|             return self._get_subtitles(*args, **kwargs) | ||||
|         return {} | ||||
|  | ||||
|     def _get_subtitles(self, *args, **kwargs): | ||||
|         raise NotImplementedError("This method must be implemented by subclasses") | ||||
|  | ||||
|     def extract_automatic_captions(self, *args, **kwargs): | ||||
|         if (self._downloader.params.get('writeautomaticsub', False) or | ||||
|                 self._downloader.params.get('listsubtitles')): | ||||
|             return self._get_automatic_captions(*args, **kwargs) | ||||
|         return {} | ||||
|  | ||||
|     def _get_automatic_captions(self, *args, **kwargs): | ||||
|         raise NotImplementedError("This method must be implemented by subclasses") | ||||
|  | ||||
|  | ||||
| class SearchInfoExtractor(InfoExtractor): | ||||
|     """ | ||||
|   | ||||
| @@ -24,6 +24,23 @@ class CommonMistakesIE(InfoExtractor): | ||||
|             'That doesn\'t make any sense. ' | ||||
|             'Simply remove the parameter in your command or configuration.' | ||||
|         ) % url | ||||
|         if self._downloader.params.get('verbose'): | ||||
|         if not self._downloader.params.get('verbose'): | ||||
|             msg += ' Add -v to the command line to see what arguments and configuration youtube-dl got.' | ||||
|         raise ExtractorError(msg, expected=True) | ||||
|  | ||||
|  | ||||
| class UnicodeBOMIE(InfoExtractor): | ||||
|         IE_DESC = False | ||||
|         _VALID_URL = r'(?P<bom>\ufeff)(?P<id>.*)$' | ||||
|  | ||||
|         _TESTS = [{ | ||||
|             'url': '\ufeffhttp://www.youtube.com/watch?v=BaW_jenozKc', | ||||
|             'only_matching': True, | ||||
|         }] | ||||
|  | ||||
|         def _real_extract(self, url): | ||||
|             real_url = self._match_id(url) | ||||
|             self.report_warning( | ||||
|                 'Your URL starts with a Byte Order Mark (BOM). ' | ||||
|                 'Removing the BOM and looking for "%s" ...' % real_url) | ||||
|             return self.url_result(real_url) | ||||
|   | ||||
| @@ -11,39 +11,65 @@ from ..utils import ( | ||||
|  | ||||
| class CrackedIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?cracked\.com/video_(?P<id>\d+)_[\da-z-]+\.html' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.cracked.com/video_19006_4-plot-holes-you-didnt-notice-in-your-favorite-movies.html', | ||||
|         'md5': '4b29a5eeec292cd5eca6388c7558db9e', | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.cracked.com/video_19070_if-animal-actors-got-e21-true-hollywood-stories.html', | ||||
|         'md5': '89b90b9824e3806ca95072c4d78f13f7', | ||||
|         'info_dict': { | ||||
|             'id': '19006', | ||||
|             'id': '19070', | ||||
|             'ext': 'mp4', | ||||
|             'title': '4 Plot Holes You Didn\'t Notice in Your Favorite Movies', | ||||
|             'description': 'md5:3b909e752661db86007d10e5ec2df769', | ||||
|             'timestamp': 1405659600, | ||||
|             'upload_date': '20140718', | ||||
|             'title': 'If Animal Actors Got E! True Hollywood Stories', | ||||
|             'timestamp': 1404954000, | ||||
|             'upload_date': '20140710', | ||||
|         } | ||||
|     } | ||||
|     }, { | ||||
|         # youtube embed | ||||
|         'url': 'http://www.cracked.com/video_19006_4-plot-holes-you-didnt-notice-in-your-favorite-movies.html', | ||||
|         'md5': 'ccd52866b50bde63a6ef3b35016ba8c7', | ||||
|         'info_dict': { | ||||
|             'id': 'EjI00A3rZD0', | ||||
|             'ext': 'mp4', | ||||
|             'title': "4 Plot Holes You Didn't Notice in Your Favorite Movies - The Spit Take", | ||||
|             'description': 'md5:c603708c718b796fe6079e2b3351ffc7', | ||||
|             'upload_date': '20140725', | ||||
|             'uploader_id': 'Cracked', | ||||
|             'uploader': 'Cracked', | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         youtube_url = self._search_regex( | ||||
|             r'<iframe[^>]+src="((?:https?:)?//www\.youtube\.com/embed/[^"]+)"', | ||||
|             webpage, 'youtube url', default=None) | ||||
|         if youtube_url: | ||||
|             return self.url_result(youtube_url, 'Youtube') | ||||
|  | ||||
|         video_url = self._html_search_regex( | ||||
|             [r'var\s+CK_vidSrc\s*=\s*"([^"]+)"', r'<video\s+src="([^"]+)"'], webpage, 'video URL') | ||||
|             [r'var\s+CK_vidSrc\s*=\s*"([^"]+)"', r'<video\s+src="([^"]+)"'], | ||||
|             webpage, 'video URL') | ||||
|  | ||||
|         title = self._og_search_title(webpage) | ||||
|         description = self._og_search_description(webpage) | ||||
|         title = self._search_regex( | ||||
|             [r'property="?og:title"?\s+content="([^"]+)"', r'class="?title"?>([^<]+)'], | ||||
|             webpage, 'title') | ||||
|  | ||||
|         timestamp = self._html_search_regex(r'<time datetime="([^"]+)"', webpage, 'upload date', fatal=False) | ||||
|         description = self._search_regex( | ||||
|             r'name="?(?:og:)?description"?\s+content="([^"]+)"', | ||||
|             webpage, 'description', default=None) | ||||
|  | ||||
|         timestamp = self._html_search_regex( | ||||
|             r'"date"\s*:\s*"([^"]+)"', webpage, 'upload date', fatal=False) | ||||
|         if timestamp: | ||||
|             timestamp = parse_iso8601(timestamp[:-6]) | ||||
|  | ||||
|         view_count = str_to_int(self._html_search_regex( | ||||
|             r'<span class="views" id="viewCounts">([\d,\.]+) Views</span>', webpage, 'view count', fatal=False)) | ||||
|             r'<span\s+class="?views"? id="?viewCounts"?>([\d,\.]+) Views</span>', | ||||
|             webpage, 'view count', fatal=False)) | ||||
|         comment_count = str_to_int(self._html_search_regex( | ||||
|             r'<span id="commentCounts">([\d,\.]+)</span>', webpage, 'comment count', fatal=False)) | ||||
|             r'<span\s+id="?commentCounts"?>([\d,\.]+)</span>', | ||||
|             webpage, 'comment count', fatal=False)) | ||||
|  | ||||
|         m = re.search(r'_(?P<width>\d+)X(?P<height>\d+)\.mp4$', video_url) | ||||
|         if m: | ||||
|   | ||||
							
								
								
									
										60
									
								
								youtube_dl/extractor/crooksandliars.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								youtube_dl/extractor/crooksandliars.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     int_or_none, | ||||
|     qualities, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class CrooksAndLiarsIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://embed\.crooksandliars\.com/(?:embed|v)/(?P<id>[A-Za-z0-9]+)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'https://embed.crooksandliars.com/embed/8RUoRhRi', | ||||
|         'info_dict': { | ||||
|             'id': '8RUoRhRi', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Fox & Friends Says Protecting Atheists From Discrimination Is Anti-Christian!', | ||||
|             'description': 'md5:e1a46ad1650e3a5ec7196d432799127f', | ||||
|             'thumbnail': 're:^https?://.*\.jpg', | ||||
|             'timestamp': 1428207000, | ||||
|             'upload_date': '20150405', | ||||
|             'uploader': 'Heather', | ||||
|             'duration': 236, | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'http://embed.crooksandliars.com/v/MTE3MjUtMzQ2MzA', | ||||
|         'only_matching': True, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage( | ||||
|             'http://embed.crooksandliars.com/embed/%s' % video_id, video_id) | ||||
|  | ||||
|         manifest = self._parse_json( | ||||
|             self._search_regex( | ||||
|                 r'var\s+manifest\s*=\s*({.+?})\n', webpage, 'manifest JSON'), | ||||
|             video_id) | ||||
|  | ||||
|         quality = qualities(('webm_low', 'mp4_low', 'webm_high', 'mp4_high')) | ||||
|  | ||||
|         formats = [{ | ||||
|             'url': item['url'], | ||||
|             'format_id': item['type'], | ||||
|             'quality': quality(item['type']), | ||||
|         } for item in manifest['flavors'] if item['mime'].startswith('video/')] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'url': url, | ||||
|             'id': video_id, | ||||
|             'title': manifest['title'], | ||||
|             'description': manifest.get('description'), | ||||
|             'thumbnail': self._proto_relative_url(manifest.get('poster')), | ||||
|             'timestamp': int_or_none(manifest.get('created')), | ||||
|             'uploader': manifest.get('author'), | ||||
|             'duration': int_or_none(manifest.get('duration')), | ||||
|             'formats': formats, | ||||
|         } | ||||
| @@ -9,7 +9,7 @@ import xml.etree.ElementTree | ||||
|  | ||||
| from hashlib import sha1 | ||||
| from math import pow, sqrt, floor | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_urllib_parse, | ||||
|     compat_urllib_request, | ||||
| @@ -23,13 +23,12 @@ from ..utils import ( | ||||
| ) | ||||
| from ..aes import ( | ||||
|     aes_cbc_decrypt, | ||||
|     inc, | ||||
| ) | ||||
| from .common import InfoExtractor | ||||
|  | ||||
|  | ||||
| class CrunchyrollIE(SubtitlesInfoExtractor): | ||||
| class CrunchyrollIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:(?P<prefix>www|m)\.)?(?P<url>crunchyroll\.(?:com|fr)/(?:[^/]*/[^/?&]*?|media/\?id=)(?P<video_id>[0-9]+))(?:[/?&]|$)' | ||||
|     _NETRC_MACHINE = 'crunchyroll' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.crunchyroll.com/wanna-be-the-strongest-in-the-world/episode-1-an-idol-wrestler-is-born-645513', | ||||
|         'info_dict': { | ||||
| @@ -77,8 +76,8 @@ class CrunchyrollIE(SubtitlesInfoExtractor): | ||||
|         self._login() | ||||
|  | ||||
|     def _decrypt_subtitles(self, data, iv, id): | ||||
|         data = bytes_to_intlist(data) | ||||
|         iv = bytes_to_intlist(iv) | ||||
|         data = bytes_to_intlist(base64.b64decode(data.encode('utf-8'))) | ||||
|         iv = bytes_to_intlist(base64.b64decode(iv.encode('utf-8'))) | ||||
|         id = int(id) | ||||
|  | ||||
|         def obfuscate_key_aux(count, modulo, start): | ||||
| @@ -102,13 +101,6 @@ class CrunchyrollIE(SubtitlesInfoExtractor): | ||||
|  | ||||
|         key = obfuscate_key(id) | ||||
|  | ||||
|         class Counter: | ||||
|             __value = iv | ||||
|  | ||||
|             def next_value(self): | ||||
|                 temp = self.__value | ||||
|                 self.__value = inc(self.__value) | ||||
|                 return temp | ||||
|         decrypted_data = intlist_to_bytes(aes_cbc_decrypt(data, key, iv)) | ||||
|         return zlib.decompress(decrypted_data) | ||||
|  | ||||
| @@ -187,6 +179,34 @@ Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text | ||||
|  | ||||
|         return output | ||||
|  | ||||
|     def _extract_subtitles(self, subtitle): | ||||
|         sub_root = xml.etree.ElementTree.fromstring(subtitle) | ||||
|         return [{ | ||||
|             'ext': 'srt', | ||||
|             'data': self._convert_subtitles_to_srt(sub_root), | ||||
|         }, { | ||||
|             'ext': 'ass', | ||||
|             'data': self._convert_subtitles_to_ass(sub_root), | ||||
|         }] | ||||
|  | ||||
|     def _get_subtitles(self, video_id, webpage): | ||||
|         subtitles = {} | ||||
|         for sub_id, sub_name in re.findall(r'\?ssid=([0-9]+)" title="([^"]+)', webpage): | ||||
|             sub_page = self._download_webpage( | ||||
|                 'http://www.crunchyroll.com/xml/?req=RpcApiSubtitle_GetXml&subtitle_script_id=' + sub_id, | ||||
|                 video_id, note='Downloading subtitles for ' + sub_name) | ||||
|             id = self._search_regex(r'id=\'([0-9]+)', sub_page, 'subtitle_id', fatal=False) | ||||
|             iv = self._search_regex(r'<iv>([^<]+)', sub_page, 'subtitle_iv', fatal=False) | ||||
|             data = self._search_regex(r'<data>([^<]+)', sub_page, 'subtitle_data', fatal=False) | ||||
|             if not id or not iv or not data: | ||||
|                 continue | ||||
|             subtitle = self._decrypt_subtitles(data, iv, id).decode('utf-8') | ||||
|             lang_code = self._search_regex(r'lang_code=["\']([^"\']+)', subtitle, 'subtitle_lang_code', fatal=False) | ||||
|             if not lang_code: | ||||
|                 continue | ||||
|             subtitles[lang_code] = self._extract_subtitles(subtitle) | ||||
|         return subtitles | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('video_id') | ||||
| @@ -239,8 +259,8 @@ Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text | ||||
|             streamdata = self._download_xml( | ||||
|                 streamdata_req, video_id, | ||||
|                 note='Downloading media info for %s' % video_format) | ||||
|             video_url = streamdata.find('.//host').text | ||||
|             video_play_path = streamdata.find('.//file').text | ||||
|             video_url = streamdata.find('./host').text | ||||
|             video_play_path = streamdata.find('./file').text | ||||
|             formats.append({ | ||||
|                 'url': video_url, | ||||
|                 'play_path': video_play_path, | ||||
| @@ -249,34 +269,7 @@ Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text | ||||
|                 'format_id': video_format, | ||||
|             }) | ||||
|  | ||||
|         subtitles = {} | ||||
|         sub_format = self._downloader.params.get('subtitlesformat', 'srt') | ||||
|         for sub_id, sub_name in re.findall(r'\?ssid=([0-9]+)" title="([^"]+)', webpage): | ||||
|             sub_page = self._download_webpage( | ||||
|                 'http://www.crunchyroll.com/xml/?req=RpcApiSubtitle_GetXml&subtitle_script_id=' + sub_id, | ||||
|                 video_id, note='Downloading subtitles for ' + sub_name) | ||||
|             id = self._search_regex(r'id=\'([0-9]+)', sub_page, 'subtitle_id', fatal=False) | ||||
|             iv = self._search_regex(r'<iv>([^<]+)', sub_page, 'subtitle_iv', fatal=False) | ||||
|             data = self._search_regex(r'<data>([^<]+)', sub_page, 'subtitle_data', fatal=False) | ||||
|             if not id or not iv or not data: | ||||
|                 continue | ||||
|             id = int(id) | ||||
|             iv = base64.b64decode(iv) | ||||
|             data = base64.b64decode(data) | ||||
|  | ||||
|             subtitle = self._decrypt_subtitles(data, iv, id).decode('utf-8') | ||||
|             lang_code = self._search_regex(r'lang_code=["\']([^"\']+)', subtitle, 'subtitle_lang_code', fatal=False) | ||||
|             if not lang_code: | ||||
|                 continue | ||||
|             sub_root = xml.etree.ElementTree.fromstring(subtitle) | ||||
|             if sub_format == 'ass': | ||||
|                 subtitles[lang_code] = self._convert_subtitles_to_ass(sub_root) | ||||
|             else: | ||||
|                 subtitles[lang_code] = self._convert_subtitles_to_srt(sub_root) | ||||
|  | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, subtitles) | ||||
|             return | ||||
|         subtitles = self.extract_subtitles(video_id, webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|   | ||||
| @@ -7,7 +7,10 @@ from ..utils import ( | ||||
|     int_or_none, | ||||
|     unescapeHTML, | ||||
|     find_xpath_attr, | ||||
|     smuggle_url, | ||||
|     determine_ext, | ||||
| ) | ||||
| from .senateisvp import SenateISVPIE | ||||
|  | ||||
|  | ||||
| class CSpanIE(InfoExtractor): | ||||
| @@ -35,11 +38,22 @@ class CSpanIE(InfoExtractor): | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'http://www.c-span.org/video/?318608-1/gm-ignition-switch-recall', | ||||
|         'md5': '446562a736c6bf97118e389433ed88d4', | ||||
|         'info_dict': { | ||||
|             'id': '342759', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'General Motors Ignition Switch Recall', | ||||
|             'duration': 14848, | ||||
|             'description': 'md5:70c7c3b8fa63fa60d42772440596034c' | ||||
|         }, | ||||
|         'playlist_duration_sum': 14855, | ||||
|     }, { | ||||
|         # Video from senate.gov | ||||
|         'url': 'http://www.c-span.org/video/?104517-1/immigration-reforms-needed-protect-skilled-american-workers', | ||||
|         'info_dict': { | ||||
|             'id': 'judiciary031715', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Immigration Reforms Needed to Protect Skilled American Workers', | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
| @@ -56,7 +70,7 @@ class CSpanIE(InfoExtractor): | ||||
|                 # present, otherwise this is a stripped version | ||||
|                 r'<p class=\'initial\'>(.*?)</p>' | ||||
|             ], | ||||
|             webpage, 'description', flags=re.DOTALL) | ||||
|             webpage, 'description', flags=re.DOTALL, default=None) | ||||
|  | ||||
|         info_url = 'http://c-spanvideo.org/videoLibrary/assets/player/ajax-player.php?os=android&html5=program&id=' + video_id | ||||
|         data = self._download_json(info_url, video_id) | ||||
| @@ -68,7 +82,16 @@ class CSpanIE(InfoExtractor): | ||||
|         title = find_xpath_attr(doc, './/string', 'name', 'title').text | ||||
|         thumbnail = find_xpath_attr(doc, './/string', 'name', 'poster').text | ||||
|  | ||||
|         senate_isvp_url = SenateISVPIE._search_iframe_url(webpage) | ||||
|         if senate_isvp_url: | ||||
|             surl = smuggle_url(senate_isvp_url, {'force_title': title}) | ||||
|             return self.url_result(surl, 'SenateISVP', video_id, title) | ||||
|  | ||||
|         files = data['video']['files'] | ||||
|         try: | ||||
|             capfile = data['video']['capfile']['#text'] | ||||
|         except KeyError: | ||||
|             capfile = None | ||||
|  | ||||
|         entries = [{ | ||||
|             'id': '%s_%d' % (video_id, partnum + 1), | ||||
| @@ -79,11 +102,22 @@ class CSpanIE(InfoExtractor): | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'duration': int_or_none(f.get('length', {}).get('#text')), | ||||
|             'subtitles': { | ||||
|                 'en': [{ | ||||
|                     'url': capfile, | ||||
|                     'ext': determine_ext(capfile, 'dfxp') | ||||
|                 }], | ||||
|             } if capfile else None, | ||||
|         } for partnum, f in enumerate(files)] | ||||
|  | ||||
|         return { | ||||
|             '_type': 'playlist', | ||||
|             'entries': entries, | ||||
|             'title': title, | ||||
|             'id': video_id, | ||||
|         } | ||||
|         if len(entries) == 1: | ||||
|             entry = dict(entries[0]) | ||||
|             entry['id'] = video_id | ||||
|             return entry | ||||
|         else: | ||||
|             return { | ||||
|                 '_type': 'playlist', | ||||
|                 'entries': entries, | ||||
|                 'title': title, | ||||
|                 'id': video_id, | ||||
|             } | ||||
|   | ||||
| @@ -6,7 +6,6 @@ import json | ||||
| import itertools | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
|  | ||||
| from ..compat import ( | ||||
|     compat_str, | ||||
| @@ -26,12 +25,11 @@ class DailymotionBaseInfoExtractor(InfoExtractor): | ||||
|     def _build_request(url): | ||||
|         """Build a request with the family filter disabled""" | ||||
|         request = compat_urllib_request.Request(url) | ||||
|         request.add_header('Cookie', 'family_filter=off') | ||||
|         request.add_header('Cookie', 'ff=off') | ||||
|         request.add_header('Cookie', 'family_filter=off; ff=off') | ||||
|         return request | ||||
|  | ||||
|  | ||||
| class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
| class DailymotionIE(DailymotionBaseInfoExtractor): | ||||
|     """Information Extractor for Dailymotion""" | ||||
|  | ||||
|     _VALID_URL = r'(?i)(?:https?://)?(?:(www|touch)\.)?dailymotion\.[a-z]{2,3}/(?:(embed|#)/)?video/(?P<id>[^/?_]+)' | ||||
| @@ -47,13 +45,14 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|  | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.dailymotion.com/video/x33vw9_tutoriel-de-youtubeur-dl-des-video_tech', | ||||
|             'md5': '392c4b85a60a90dc4792da41ce3144eb', | ||||
|             'url': 'https://www.dailymotion.com/video/x2iuewm_steam-machine-models-pricing-listed-on-steam-store-ign-news_videogames', | ||||
|             'md5': '2137c41a8e78554bb09225b8eb322406', | ||||
|             'info_dict': { | ||||
|                 'id': 'x33vw9', | ||||
|                 'id': 'x2iuewm', | ||||
|                 'ext': 'mp4', | ||||
|                 'uploader': 'Amphora Alex and Van .', | ||||
|                 'title': 'Tutoriel de Youtubeur"DL DES VIDEO DE YOUTUBE"', | ||||
|                 'uploader': 'IGN', | ||||
|                 'title': 'Steam Machine Models, Pricing Listed on Steam Store - IGN News', | ||||
|                 'upload_date': '20150306', | ||||
|             } | ||||
|         }, | ||||
|         # Vevo video | ||||
| @@ -87,7 +86,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         url = 'http://www.dailymotion.com/video/%s' % video_id | ||||
|         url = 'https://www.dailymotion.com/video/%s' % video_id | ||||
|  | ||||
|         # Retrieve video webpage to extract further information | ||||
|         request = self._build_request(url) | ||||
| @@ -108,13 +107,14 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|         age_limit = self._rta_search(webpage) | ||||
|  | ||||
|         video_upload_date = None | ||||
|         mobj = re.search(r'<div class="[^"]*uploaded_cont[^"]*" title="[^"]*">([0-9]{2})-([0-9]{2})-([0-9]{4})</div>', webpage) | ||||
|         mobj = re.search(r'<meta property="video:release_date" content="([0-9]{4})-([0-9]{2})-([0-9]{2}).+?"/>', webpage) | ||||
|         if mobj is not None: | ||||
|             video_upload_date = mobj.group(3) + mobj.group(2) + mobj.group(1) | ||||
|             video_upload_date = mobj.group(1) + mobj.group(2) + mobj.group(3) | ||||
|  | ||||
|         embed_url = 'http://www.dailymotion.com/embed/video/%s' % video_id | ||||
|         embed_page = self._download_webpage(embed_url, video_id, | ||||
|                                             'Downloading embed page') | ||||
|         embed_url = 'https://www.dailymotion.com/embed/video/%s' % video_id | ||||
|         embed_request = self._build_request(embed_url) | ||||
|         embed_page = self._download_webpage( | ||||
|             embed_request, video_id, 'Downloading embed page') | ||||
|         info = self._search_regex(r'var info = ({.*?}),$', embed_page, | ||||
|                                   'video info', flags=re.MULTILINE) | ||||
|         info = json.loads(info) | ||||
| @@ -143,9 +143,6 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|  | ||||
|         # subtitles | ||||
|         video_subtitles = self.extract_subtitles(video_id, webpage) | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, webpage) | ||||
|             return | ||||
|  | ||||
|         view_count = str_to_int(self._search_regex( | ||||
|             r'video_views_count[^>]+>\s+([\d\.,]+)', | ||||
| @@ -169,7 +166,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             'view_count': view_count, | ||||
|         } | ||||
|  | ||||
|     def _get_available_subtitles(self, video_id, webpage): | ||||
|     def _get_subtitles(self, video_id, webpage): | ||||
|         try: | ||||
|             sub_list = self._download_webpage( | ||||
|                 'https://api.dailymotion.com/video/%s/subtitles?fields=id,language,url' % video_id, | ||||
| @@ -179,7 +176,7 @@ class DailymotionIE(DailymotionBaseInfoExtractor, SubtitlesInfoExtractor): | ||||
|             return {} | ||||
|         info = json.loads(sub_list) | ||||
|         if (info['total'] > 0): | ||||
|             sub_lang_list = dict((l['language'], l['url']) for l in info['list']) | ||||
|             sub_lang_list = dict((l['language'], [{'url': l['url'], 'ext': 'srt'}]) for l in info['list']) | ||||
|             return sub_lang_list | ||||
|         self._downloader.report_warning('video doesn\'t have subtitles') | ||||
|         return {} | ||||
| @@ -194,6 +191,7 @@ class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|         'url': 'http://www.dailymotion.com/playlist/xv4bw_nqtv_sport/1#video=xl8v3q', | ||||
|         'info_dict': { | ||||
|             'title': 'SPORT', | ||||
|             'id': 'xv4bw_nqtv_sport', | ||||
|         }, | ||||
|         'playlist_mincount': 20, | ||||
|     }] | ||||
| @@ -227,7 +225,7 @@ class DailymotionPlaylistIE(DailymotionBaseInfoExtractor): | ||||
|  | ||||
| class DailymotionUserIE(DailymotionPlaylistIE): | ||||
|     IE_NAME = 'dailymotion:user' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dailymotion\.[a-z]{2,3}/user/(?P<user>[^/]+)' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dailymotion\.[a-z]{2,3}/(?:(?:old/)?user/)?(?P<user>[^/]+)$' | ||||
|     _PAGE_TEMPLATE = 'http://www.dailymotion.com/user/%s/%s' | ||||
|     _TESTS = [{ | ||||
|         'url': 'https://www.dailymotion.com/user/nqtv', | ||||
| @@ -241,7 +239,8 @@ class DailymotionUserIE(DailymotionPlaylistIE): | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         user = mobj.group('user') | ||||
|         webpage = self._download_webpage(url, user) | ||||
|         webpage = self._download_webpage( | ||||
|             'https://www.dailymotion.com/user/%s' % user, user) | ||||
|         full_user = unescapeHTML(self._html_search_regex( | ||||
|             r'<a class="nav-image" title="([^"]+)" href="/%s">' % re.escape(user), | ||||
|             webpage, 'user')) | ||||
|   | ||||
| @@ -14,6 +14,10 @@ class DctpTvIE(InfoExtractor): | ||||
|             'display_id': 'videoinstallation-fuer-eine-kaufhausfassade', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Videoinstallation für eine Kaufhausfassade' | ||||
|         }, | ||||
|         'params': { | ||||
|             # rtmp download | ||||
|             'skip_download': True, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -25,8 +25,9 @@ class DefenseGouvFrIE(InfoExtractor): | ||||
|             r"flashvars.pvg_id=\"(\d+)\";", | ||||
|             webpage, 'ID') | ||||
|  | ||||
|         json_url = ('http://static.videos.gouv.fr/brightcovehub/export/json/' | ||||
|                     + video_id) | ||||
|         json_url = ( | ||||
|             'http://static.videos.gouv.fr/brightcovehub/export/json/%s' % | ||||
|             video_id) | ||||
|         info = self._download_json(json_url, title, 'Downloading JSON config') | ||||
|         video_url = info['renditions'][0]['url'] | ||||
|  | ||||
|   | ||||
							
								
								
									
										73
									
								
								youtube_dl/extractor/dhm.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								youtube_dl/extractor/dhm.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     xpath_text, | ||||
|     parse_duration, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class DHMIE(InfoExtractor): | ||||
|     IE_DESC = 'Filmarchiv - Deutsches Historisches Museum' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dhm\.de/filmarchiv/(?:[^/]+/)+(?P<id>[^/]+)' | ||||
|  | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.dhm.de/filmarchiv/die-filme/the-marshallplan-at-work-in-west-germany/', | ||||
|         'md5': '11c475f670209bf6acca0b2b7ef51827', | ||||
|         'info_dict': { | ||||
|             'id': 'the-marshallplan-at-work-in-west-germany', | ||||
|             'ext': 'flv', | ||||
|             'title': 'MARSHALL PLAN AT WORK IN WESTERN GERMANY, THE', | ||||
|             'description': 'md5:1fabd480c153f97b07add61c44407c82', | ||||
|             'duration': 660, | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|         }, | ||||
|     }, { | ||||
|         'url': 'http://www.dhm.de/filmarchiv/02-mapping-the-wall/peter-g/rolle-1/', | ||||
|         'md5': '09890226332476a3e3f6f2cb74734aa5', | ||||
|         'info_dict': { | ||||
|             'id': 'rolle-1', | ||||
|             'ext': 'flv', | ||||
|             'title': 'ROLLE 1', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|         }, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         playlist_url = self._search_regex( | ||||
|             r"file\s*:\s*'([^']+)'", webpage, 'playlist url') | ||||
|  | ||||
|         playlist = self._download_xml(playlist_url, video_id) | ||||
|  | ||||
|         track = playlist.find( | ||||
|             './{http://xspf.org/ns/0/}trackList/{http://xspf.org/ns/0/}track') | ||||
|  | ||||
|         video_url = xpath_text( | ||||
|             track, './{http://xspf.org/ns/0/}location', | ||||
|             'video url', fatal=True) | ||||
|         thumbnail = xpath_text( | ||||
|             track, './{http://xspf.org/ns/0/}image', | ||||
|             'thumbnail') | ||||
|  | ||||
|         title = self._search_regex( | ||||
|             [r'dc:title="([^"]+)"', r'<title> »([^<]+)</title>'], | ||||
|             webpage, 'title').strip() | ||||
|         description = self._html_search_regex( | ||||
|             r'<p><strong>Description:</strong>(.+?)</p>', | ||||
|             webpage, 'description', default=None) | ||||
|         duration = parse_duration(self._search_regex( | ||||
|             r'<em>Length\s*</em>\s*:\s*</strong>([^<]+)', | ||||
|             webpage, 'duration', default=None)) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': video_url, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'duration': duration, | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
| @@ -2,19 +2,19 @@ from __future__ import unicode_literals | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     parse_duration, | ||||
|     parse_iso8601, | ||||
|     int_or_none, | ||||
| ) | ||||
| from ..compat import compat_str | ||||
|  | ||||
|  | ||||
| class DiscoveryIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://www\.discovery\.com\/[a-zA-Z0-9\-]*/[a-zA-Z0-9\-]*/videos/(?P<id>[a-zA-Z0-9_\-]*)(?:\.htm)?' | ||||
|     _TEST = { | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.discovery.com/tv-shows/mythbusters/videos/mission-impossible-outtakes.htm', | ||||
|         'md5': '3c69d77d9b0d82bfd5e5932a60f26504', | ||||
|         'info_dict': { | ||||
|             'id': 'mission-impossible-outtakes', | ||||
|             'ext': 'flv', | ||||
|             'id': '20769', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Mission Impossible Outtakes', | ||||
|             'description': ('Watch Jamie Hyneman and Adam Savage practice being' | ||||
|                             ' each other -- to the point of confusing Jamie\'s dog -- and ' | ||||
| @@ -24,22 +24,36 @@ class DiscoveryIE(InfoExtractor): | ||||
|             'timestamp': 1303099200, | ||||
|             'upload_date': '20110418', | ||||
|         }, | ||||
|     } | ||||
|         'params': { | ||||
|             'skip_download': True,  # requires ffmpeg | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'http://www.discovery.com/tv-shows/mythbusters/videos/mythbusters-the-simpsons', | ||||
|         'info_dict': { | ||||
|             'id': 'mythbusters-the-simpsons', | ||||
|             'title': 'MythBusters: The Simpsons', | ||||
|         }, | ||||
|         'playlist_count': 9, | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|         info = self._download_json(url + '?flat=1', video_id) | ||||
|  | ||||
|         info = self._parse_json(self._search_regex( | ||||
|             r'(?s)<script type="application/ld\+json">(.*?)</script>', | ||||
|             webpage, 'video info'), video_id) | ||||
|         video_title = info.get('playlist_title') or info.get('video_title') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': info['name'], | ||||
|             'url': info['contentURL'], | ||||
|             'description': info.get('description'), | ||||
|             'thumbnail': info.get('thumbnailUrl'), | ||||
|             'timestamp': parse_iso8601(info.get('uploadDate')), | ||||
|             'duration': int_or_none(info.get('duration')), | ||||
|         } | ||||
|         entries = [{ | ||||
|             'id': compat_str(video_info['id']), | ||||
|             'formats': self._extract_m3u8_formats( | ||||
|                 video_info['src'], video_id, ext='mp4', | ||||
|                 note='Download m3u8 information for video %d' % (idx + 1)), | ||||
|             'title': video_info['title'], | ||||
|             'description': video_info.get('description'), | ||||
|             'duration': parse_duration(video_info.get('video_length')), | ||||
|             'webpage_url': video_info.get('href'), | ||||
|             'thumbnail': video_info.get('thumbnailURL'), | ||||
|             'alt_title': video_info.get('secondary_title'), | ||||
|             'timestamp': parse_iso8601(video_info.get('publishedDate')), | ||||
|         } for idx, video_info in enumerate(info['playlist'])] | ||||
|  | ||||
|         return self.playlist_result(entries, video_id, video_title) | ||||
|   | ||||
| @@ -1,13 +1,14 @@ | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import re | ||||
| import time | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     float_or_none, | ||||
|     int_or_none, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class DotsubIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?dotsub\.com/view/(?P<id>[^/]+)' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dotsub\.com/view/(?P<id>[^/]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://dotsub.com/view/aed3b8b2-1889-4df5-ae63-ad85f5572f27', | ||||
|         'md5': '0914d4d69605090f623b7ac329fea66e', | ||||
| @@ -15,28 +16,38 @@ class DotsubIE(InfoExtractor): | ||||
|             'id': 'aed3b8b2-1889-4df5-ae63-ad85f5572f27', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Pyramids of Waste (2010), AKA The Lightbulb Conspiracy - Planned obsolescence documentary', | ||||
|             'description': 'md5:699a0f7f50aeec6042cb3b1db2d0d074', | ||||
|             'thumbnail': 're:^https?://dotsub.com/media/aed3b8b2-1889-4df5-ae63-ad85f5572f27/p', | ||||
|             'duration': 3169, | ||||
|             'uploader': '4v4l0n42', | ||||
|             'description': 'Pyramids of Waste (2010) also known as "The lightbulb conspiracy" is a documentary about how our economic system based on consumerism  and planned obsolescence is breaking our planet down.\r\n\r\nSolutions to this can be found at:\r\nhttp://robotswillstealyourjob.com\r\nhttp://www.federicopistono.org\r\n\r\nhttp://opensourceecology.org\r\nhttp://thezeitgeistmovement.com', | ||||
|             'thumbnail': 'http://dotsub.com/media/aed3b8b2-1889-4df5-ae63-ad85f5572f27/p', | ||||
|             'timestamp': 1292248482.625, | ||||
|             'upload_date': '20101213', | ||||
|             'view_count': int, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
|         video_id = mobj.group('id') | ||||
|         info_url = "https://dotsub.com/api/media/%s/metadata" % video_id | ||||
|         info = self._download_json(info_url, video_id) | ||||
|         date = time.gmtime(info['dateCreated'] / 1000)  # The timestamp is in miliseconds | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         info = self._download_json( | ||||
|             'https://dotsub.com/api/media/%s/metadata' % video_id, video_id) | ||||
|         video_url = info.get('mediaURI') | ||||
|  | ||||
|         if not video_url: | ||||
|             webpage = self._download_webpage(url, video_id) | ||||
|             video_url = self._search_regex( | ||||
|                 [r'<source[^>]+src="([^"]+)"', r'"file"\s*:\s*\'([^\']+)'], | ||||
|                 webpage, 'video url') | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'url': info['mediaURI'], | ||||
|             'url': video_url, | ||||
|             'ext': 'flv', | ||||
|             'title': info['title'], | ||||
|             'thumbnail': info['screenshotURI'], | ||||
|             'description': info['description'], | ||||
|             'uploader': info['user'], | ||||
|             'view_count': info['numberOfViews'], | ||||
|             'upload_date': '%04i%02i%02i' % (date.tm_year, date.tm_mon, date.tm_mday), | ||||
|             'description': info.get('description'), | ||||
|             'thumbnail': info.get('screenshotURI'), | ||||
|             'duration': int_or_none(info.get('duration'), 1000), | ||||
|             'uploader': info.get('user'), | ||||
|             'timestamp': float_or_none(info.get('dateCreated'), 1000), | ||||
|             'view_count': int_or_none(info.get('numberOfViews')), | ||||
|         } | ||||
|   | ||||
							
								
								
									
										112
									
								
								youtube_dl/extractor/douyutv.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								youtube_dl/extractor/douyutv.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import hashlib | ||||
| import time | ||||
| from .common import InfoExtractor | ||||
| from ..utils import (ExtractorError, unescapeHTML) | ||||
| from ..compat import (compat_str, compat_basestring) | ||||
|  | ||||
|  | ||||
| class DouyuTVIE(InfoExtractor): | ||||
|     _VALID_URL = r'http://(?:www\.)?douyutv\.com/(?P<id>[A-Za-z0-9]+)' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.douyutv.com/iseven', | ||||
|         'info_dict': { | ||||
|             'id': '17732', | ||||
|             'display_id': 'iseven', | ||||
|             'ext': 'flv', | ||||
|             'title': 're:^清晨醒脑!T-ara根本停不下来! [0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}$', | ||||
|             'description': 'md5:c93d6692dde6fe33809a46edcbecca44', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'uploader': '7师傅', | ||||
|             'uploader_id': '431925', | ||||
|             'is_live': True, | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True, | ||||
|         } | ||||
|     }, { | ||||
|         'url': 'http://www.douyutv.com/85982', | ||||
|         'info_dict': { | ||||
|             'id': '85982', | ||||
|             'display_id': '85982', | ||||
|             'ext': 'flv', | ||||
|             'title': 're:^小漠从零单排记!——CSOL2躲猫猫 [0-9]{4}-[0-9]{2}-[0-9]{2} [0-9]{2}:[0-9]{2}$', | ||||
|             'description': 'md5:746a2f7a253966a06755a912f0acc0d2', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|             'uploader': 'douyu小漠', | ||||
|             'uploader_id': '3769985', | ||||
|             'is_live': True, | ||||
|         }, | ||||
|         'params': { | ||||
|             'skip_download': True, | ||||
|         } | ||||
|     }] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         if video_id.isdigit(): | ||||
|             room_id = video_id | ||||
|         else: | ||||
|             page = self._download_webpage(url, video_id) | ||||
|             room_id = self._html_search_regex( | ||||
|                 r'"room_id"\s*:\s*(\d+),', page, 'room id') | ||||
|  | ||||
|         prefix = 'room/%s?aid=android&client_sys=android&time=%d' % ( | ||||
|             room_id, int(time.time())) | ||||
|  | ||||
|         auth = hashlib.md5((prefix + '1231').encode('ascii')).hexdigest() | ||||
|         config = self._download_json( | ||||
|             'http://www.douyutv.com/api/v1/%s&auth=%s' % (prefix, auth), | ||||
|             video_id) | ||||
|  | ||||
|         data = config['data'] | ||||
|  | ||||
|         error_code = config.get('error', 0) | ||||
|         if error_code is not 0: | ||||
|             error_desc = 'Server reported error %i' % error_code | ||||
|             if isinstance(data, (compat_str, compat_basestring)): | ||||
|                 error_desc += ': ' + data | ||||
|             raise ExtractorError(error_desc, expected=True) | ||||
|  | ||||
|         show_status = data.get('show_status') | ||||
|         # 1 = live, 2 = offline | ||||
|         if show_status == '2': | ||||
|             raise ExtractorError( | ||||
|                 'Live stream is offline', expected=True) | ||||
|  | ||||
|         base_url = data['rtmp_url'] | ||||
|         live_path = data['rtmp_live'] | ||||
|  | ||||
|         title = self._live_title(unescapeHTML(data['room_name'])) | ||||
|         description = data.get('show_details') | ||||
|         thumbnail = data.get('room_src') | ||||
|  | ||||
|         uploader = data.get('nickname') | ||||
|         uploader_id = data.get('owner_uid') | ||||
|  | ||||
|         multi_formats = data.get('rtmp_multi_bitrate') | ||||
|         if not isinstance(multi_formats, dict): | ||||
|             multi_formats = {} | ||||
|         multi_formats['live'] = live_path | ||||
|  | ||||
|         formats = [{ | ||||
|             'url': '%s/%s' % (base_url, format_path), | ||||
|             'format_id': format_id, | ||||
|             'preference': 1 if format_id == 'live' else 0, | ||||
|         } for format_id, format_path in multi_formats.items()] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         return { | ||||
|             'id': room_id, | ||||
|             'display_id': video_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'uploader': uploader, | ||||
|             'uploader_id': uploader_id, | ||||
|             'formats': formats, | ||||
|             'is_live': True, | ||||
|         } | ||||
							
								
								
									
										160
									
								
								youtube_dl/extractor/dramafever.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								youtube_dl/extractor/dramafever.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,160 @@ | ||||
| # encoding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import itertools | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import ( | ||||
|     compat_HTTPError, | ||||
|     compat_urlparse, | ||||
| ) | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     clean_html, | ||||
|     determine_ext, | ||||
|     int_or_none, | ||||
|     parse_iso8601, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class DramaFeverIE(InfoExtractor): | ||||
|     IE_NAME = 'dramafever' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dramafever\.com/drama/(?P<id>[0-9]+/[0-9]+)(?:/|$)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.dramafever.com/drama/4512/1/Cooking_with_Shin/', | ||||
|         'info_dict': { | ||||
|             'id': '4512.1', | ||||
|             'ext': 'flv', | ||||
|             'title': 'Cooking with Shin 4512.1', | ||||
|             'description': 'md5:a8eec7942e1664a6896fcd5e1287bfd0', | ||||
|             'thumbnail': 're:^https?://.*\.jpg', | ||||
|             'timestamp': 1404336058, | ||||
|             'upload_date': '20140702', | ||||
|             'duration': 343, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url).replace('/', '.') | ||||
|  | ||||
|         try: | ||||
|             feed = self._download_json( | ||||
|                 'http://www.dramafever.com/amp/episode/feed.json?guid=%s' % video_id, | ||||
|                 video_id, 'Downloading episode JSON')['channel']['item'] | ||||
|         except ExtractorError as e: | ||||
|             if isinstance(e.cause, compat_HTTPError): | ||||
|                 raise ExtractorError( | ||||
|                     'Currently unavailable in your country.', expected=True) | ||||
|             raise | ||||
|  | ||||
|         media_group = feed.get('media-group', {}) | ||||
|  | ||||
|         formats = [] | ||||
|         for media_content in media_group['media-content']: | ||||
|             src = media_content.get('@attributes', {}).get('url') | ||||
|             if not src: | ||||
|                 continue | ||||
|             ext = determine_ext(src) | ||||
|             if ext == 'f4m': | ||||
|                 formats.extend(self._extract_f4m_formats( | ||||
|                     src, video_id, f4m_id='hds')) | ||||
|             elif ext == 'm3u8': | ||||
|                 formats.extend(self._extract_m3u8_formats( | ||||
|                     src, video_id, 'mp4', m3u8_id='hls')) | ||||
|             else: | ||||
|                 formats.append({ | ||||
|                     'url': src, | ||||
|                 }) | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         title = media_group.get('media-title') | ||||
|         description = media_group.get('media-description') | ||||
|         duration = int_or_none(media_group['media-content'][0].get('@attributes', {}).get('duration')) | ||||
|         thumbnail = self._proto_relative_url( | ||||
|             media_group.get('media-thumbnail', {}).get('@attributes', {}).get('url')) | ||||
|         timestamp = parse_iso8601(feed.get('pubDate'), ' ') | ||||
|  | ||||
|         subtitles = {} | ||||
|         for media_subtitle in media_group.get('media-subTitle', []): | ||||
|             lang = media_subtitle.get('@attributes', {}).get('lang') | ||||
|             href = media_subtitle.get('@attributes', {}).get('href') | ||||
|             if not lang or not href: | ||||
|                 continue | ||||
|             subtitles[lang] = [{ | ||||
|                 'ext': 'ttml', | ||||
|                 'url': href, | ||||
|             }] | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'timestamp': timestamp, | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|             'subtitles': subtitles, | ||||
|         } | ||||
|  | ||||
|  | ||||
| class DramaFeverSeriesIE(InfoExtractor): | ||||
|     IE_NAME = 'dramafever:series' | ||||
|     _VALID_URL = r'https?://(?:www\.)?dramafever\.com/drama/(?P<id>[0-9]+)(?:/(?:(?!\d+(?:/|$)).+)?)?$' | ||||
|     _TESTS = [{ | ||||
|         'url': 'http://www.dramafever.com/drama/4512/Cooking_with_Shin/', | ||||
|         'info_dict': { | ||||
|             'id': '4512', | ||||
|             'title': 'Cooking with Shin', | ||||
|             'description': 'md5:84a3f26e3cdc3fb7f500211b3593b5c1', | ||||
|         }, | ||||
|         'playlist_count': 4, | ||||
|     }, { | ||||
|         'url': 'http://www.dramafever.com/drama/124/IRIS/', | ||||
|         'info_dict': { | ||||
|             'id': '124', | ||||
|             'title': 'IRIS', | ||||
|             'description': 'md5:b3a30e587cf20c59bd1c01ec0ee1b862', | ||||
|         }, | ||||
|         'playlist_count': 20, | ||||
|     }] | ||||
|  | ||||
|     _CONSUMER_SECRET = 'DA59dtVXYLxajktV' | ||||
|     _PAGE_SIZE = 60  # max is 60 (see http://api.drama9.com/#get--api-4-episode-series-) | ||||
|  | ||||
|     def _get_consumer_secret(self, video_id): | ||||
|         mainjs = self._download_webpage( | ||||
|             'http://www.dramafever.com/static/51afe95/df2014/scripts/main.js', | ||||
|             video_id, 'Downloading main.js', fatal=False) | ||||
|         if not mainjs: | ||||
|             return self._CONSUMER_SECRET | ||||
|         return self._search_regex( | ||||
|             r"var\s+cs\s*=\s*'([^']+)'", mainjs, | ||||
|             'consumer secret', default=self._CONSUMER_SECRET) | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         series_id = self._match_id(url) | ||||
|  | ||||
|         consumer_secret = self._get_consumer_secret(series_id) | ||||
|  | ||||
|         series = self._download_json( | ||||
|             'http://www.dramafever.com/api/4/series/query/?cs=%s&series_id=%s' | ||||
|             % (consumer_secret, series_id), | ||||
|             series_id, 'Downloading series JSON')['series'][series_id] | ||||
|  | ||||
|         title = clean_html(series['name']) | ||||
|         description = clean_html(series.get('description') or series.get('description_short')) | ||||
|  | ||||
|         entries = [] | ||||
|         for page_num in itertools.count(1): | ||||
|             episodes = self._download_json( | ||||
|                 'http://www.dramafever.com/api/4/episode/series/?cs=%s&series_id=%s&page_size=%d&page_number=%d' | ||||
|                 % (consumer_secret, series_id, self._PAGE_SIZE, page_num), | ||||
|                 series_id, 'Downloading episodes JSON page #%d' % page_num) | ||||
|             for episode in episodes.get('value', []): | ||||
|                 entries.append(self.url_result( | ||||
|                     compat_urlparse.urljoin(url, episode['episode_url']), | ||||
|                     'DramaFever', episode.get('guid'))) | ||||
|             if page_num == episodes['num_pages']: | ||||
|                 break | ||||
|  | ||||
|         return self.playlist_result(entries, series_id, title, description) | ||||
| @@ -3,24 +3,33 @@ from __future__ import unicode_literals | ||||
| import re | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..utils import unified_strdate | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     unified_strdate, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class DreiSatIE(InfoExtractor): | ||||
|     IE_NAME = '3sat' | ||||
|     _VALID_URL = r'(?:http://)?(?:www\.)?3sat\.de/mediathek/(?:index\.php)?\?(?:(?:mode|display)=[^&]+&)*obj=(?P<id>[0-9]+)$' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.3sat.de/mediathek/index.php?obj=36983', | ||||
|         'md5': '9dcfe344732808dbfcc901537973c922', | ||||
|         'info_dict': { | ||||
|             'id': '36983', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Kaffeeland Schweiz', | ||||
|             'description': 'md5:cc4424b18b75ae9948b13929a0814033', | ||||
|             'uploader': '3sat', | ||||
|             'upload_date': '20130622' | ||||
|         } | ||||
|     } | ||||
|     _VALID_URL = r'(?:http://)?(?:www\.)?3sat\.de/mediathek/(?:index\.php|mediathek\.php)?\?(?:(?:mode|display)=[^&]+&)*obj=(?P<id>[0-9]+)$' | ||||
|     _TESTS = [ | ||||
|         { | ||||
|             'url': 'http://www.3sat.de/mediathek/index.php?mode=play&obj=45918', | ||||
|             'md5': 'be37228896d30a88f315b638900a026e', | ||||
|             'info_dict': { | ||||
|                 'id': '45918', | ||||
|                 'ext': 'mp4', | ||||
|                 'title': 'Waidmannsheil', | ||||
|                 'description': 'md5:cce00ca1d70e21425e72c86a98a56817', | ||||
|                 'uploader': '3sat', | ||||
|                 'upload_date': '20140913' | ||||
|             } | ||||
|         }, | ||||
|         { | ||||
|             'url': 'http://www.3sat.de/mediathek/mediathek.php?mode=play&obj=51066', | ||||
|             'only_matching': True, | ||||
|         }, | ||||
|     ] | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         mobj = re.match(self._VALID_URL, url) | ||||
| @@ -28,6 +37,15 @@ class DreiSatIE(InfoExtractor): | ||||
|         details_url = 'http://www.3sat.de/mediathek/xmlservice/web/beitragsDetails?ak=web&id=%s' % video_id | ||||
|         details_doc = self._download_xml(details_url, video_id, 'Downloading video details') | ||||
|  | ||||
|         status_code = details_doc.find('./status/statuscode') | ||||
|         if status_code is not None and status_code.text != 'ok': | ||||
|             code = status_code.text | ||||
|             if code == 'notVisibleAnymore': | ||||
|                 message = 'Video %s is not available' % video_id | ||||
|             else: | ||||
|                 message = '%s returned error: %s' % (self.IE_NAME, code) | ||||
|             raise ExtractorError(message, expected=True) | ||||
|  | ||||
|         thumbnail_els = details_doc.findall('.//teaserimage') | ||||
|         thumbnails = [{ | ||||
|             'width': int(te.attrib['key'].partition('x')[0]), | ||||
|   | ||||
| @@ -15,7 +15,7 @@ class DrTuberIE(InfoExtractor): | ||||
|             'id': '1740434', | ||||
|             'display_id': 'hot-perky-blonde-naked-golf', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Hot Perky Blonde Naked Golf', | ||||
|             'title': 'hot perky blonde naked golf', | ||||
|             'like_count': int, | ||||
|             'dislike_count': int, | ||||
|             'comment_count': int, | ||||
| @@ -36,7 +36,8 @@ class DrTuberIE(InfoExtractor): | ||||
|             r'<source src="([^"]+)"', webpage, 'video URL') | ||||
|  | ||||
|         title = self._html_search_regex( | ||||
|             r'<title>([^<]+)\s*-\s*Free', webpage, 'title') | ||||
|             [r'class="hd_title" style="[^"]+">([^<]+)</h1>', r'<title>([^<]+) - \d+'], | ||||
|             webpage, 'title') | ||||
|  | ||||
|         thumbnail = self._html_search_regex( | ||||
|             r'poster="([^"]+)"', | ||||
|   | ||||
| @@ -1,24 +1,27 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| from .subtitles import SubtitlesInfoExtractor | ||||
| from .common import ExtractorError | ||||
| from ..utils import parse_iso8601 | ||||
| from .common import InfoExtractor | ||||
| from ..utils import ( | ||||
|     ExtractorError, | ||||
|     parse_iso8601, | ||||
| ) | ||||
|  | ||||
|  | ||||
| class DRTVIE(SubtitlesInfoExtractor): | ||||
| class DRTVIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?dr\.dk/tv/se/(?:[^/]+/)*(?P<id>[\da-z-]+)(?:[/#?]|$)' | ||||
|  | ||||
|     _TEST = { | ||||
|         'url': 'http://www.dr.dk/tv/se/partiets-mand/partiets-mand-7-8', | ||||
|         'md5': '4a7e1dd65cdb2643500a3f753c942f25', | ||||
|         'url': 'https://www.dr.dk/tv/se/boern/ultra/panisk-paske/panisk-paske-5', | ||||
|         'md5': 'dc515a9ab50577fa14cc4e4b0265168f', | ||||
|         'info_dict': { | ||||
|             'id': 'partiets-mand-7-8', | ||||
|             'id': 'panisk-paske-5', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Partiets mand (7:8)', | ||||
|             'description': 'md5:a684b90a8f9336cd4aab94b7647d7862', | ||||
|             'timestamp': 1403047940, | ||||
|             'upload_date': '20140617', | ||||
|             'duration': 1299.040, | ||||
|             'title': 'Panisk Påske (5)', | ||||
|             'description': 'md5:ca14173c5ab24cd26b0fcc074dff391c', | ||||
|             'timestamp': 1426984612, | ||||
|             'upload_date': '20150322', | ||||
|             'duration': 1455, | ||||
|         }, | ||||
|     } | ||||
|  | ||||
| @@ -27,6 +30,10 @@ class DRTVIE(SubtitlesInfoExtractor): | ||||
|  | ||||
|         webpage = self._download_webpage(url, video_id) | ||||
|  | ||||
|         if '>Programmet er ikke længere tilgængeligt' in webpage: | ||||
|             raise ExtractorError( | ||||
|                 'Video %s is not available' % video_id, expected=True) | ||||
|  | ||||
|         video_id = self._search_regex( | ||||
|             r'data-(?:material-identifier|episode-slug)="([^"]+)"', | ||||
|             webpage, 'video id') | ||||
| @@ -56,19 +63,31 @@ class DRTVIE(SubtitlesInfoExtractor): | ||||
|                 restricted_to_denmark = asset['RestrictedToDenmark'] | ||||
|                 spoken_subtitles = asset['Target'] == 'SpokenSubtitles' | ||||
|                 for link in asset['Links']: | ||||
|                     target = link['Target'] | ||||
|                     uri = link['Uri'] | ||||
|                     target = link['Target'] | ||||
|                     format_id = target | ||||
|                     preference = -1 if target == 'HDS' else -2 | ||||
|                     preference = None | ||||
|                     if spoken_subtitles: | ||||
|                         preference -= 2 | ||||
|                         preference = -1 | ||||
|                         format_id += '-spoken-subtitles' | ||||
|                     formats.append({ | ||||
|                         'url': uri + '?hdcore=3.3.0&plugin=aasp-3.3.0.99.43' if target == 'HDS' else uri, | ||||
|                         'format_id': format_id, | ||||
|                         'ext': link['FileFormat'], | ||||
|                         'preference': preference, | ||||
|                     }) | ||||
|                     if target == 'HDS': | ||||
|                         formats.extend(self._extract_f4m_formats( | ||||
|                             uri + '?hdcore=3.3.0&plugin=aasp-3.3.0.99.43', | ||||
|                             video_id, preference, f4m_id=format_id)) | ||||
|                     elif target == 'HLS': | ||||
|                         formats.extend(self._extract_m3u8_formats( | ||||
|                             uri, video_id, 'mp4', preference=preference, | ||||
|                             m3u8_id=format_id)) | ||||
|                     else: | ||||
|                         bitrate = link.get('Bitrate') | ||||
|                         if bitrate: | ||||
|                             format_id += '-%s' % bitrate | ||||
|                         formats.append({ | ||||
|                             'url': uri, | ||||
|                             'format_id': format_id, | ||||
|                             'tbr': bitrate, | ||||
|                             'ext': link.get('FileFormat'), | ||||
|                         }) | ||||
|                 subtitles_list = asset.get('SubtitlesList') | ||||
|                 if isinstance(subtitles_list, list): | ||||
|                     LANGS = { | ||||
| @@ -76,7 +95,7 @@ class DRTVIE(SubtitlesInfoExtractor): | ||||
|                     } | ||||
|                     for subs in subtitles_list: | ||||
|                         lang = subs['Language'] | ||||
|                         subtitles[LANGS.get(lang, lang)] = subs['Uri'] | ||||
|                         subtitles[LANGS.get(lang, lang)] = [{'url': subs['Uri'], 'ext': 'vtt'}] | ||||
|  | ||||
|         if not formats and restricted_to_denmark: | ||||
|             raise ExtractorError( | ||||
| @@ -84,10 +103,6 @@ class DRTVIE(SubtitlesInfoExtractor): | ||||
|  | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         if self._downloader.params.get('listsubtitles', False): | ||||
|             self._list_available_subtitles(video_id, subtitles) | ||||
|             return | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
| @@ -96,5 +111,5 @@ class DRTVIE(SubtitlesInfoExtractor): | ||||
|             'timestamp': timestamp, | ||||
|             'duration': duration, | ||||
|             'formats': formats, | ||||
|             'subtitles': self.extract_subtitles(video_id, subtitles), | ||||
|             'subtitles': subtitles, | ||||
|         } | ||||
|   | ||||
| @@ -28,12 +28,12 @@ class DumpIE(InfoExtractor): | ||||
|         video_url = self._search_regex( | ||||
|             r's1.addVariable\("file",\s*"([^"]+)"', webpage, 'video URL') | ||||
|  | ||||
|         thumb = self._og_search_thumbnail(webpage) | ||||
|         title = self._search_regex(r'<b>([^"]+)</b>', webpage, 'title') | ||||
|         title = self._og_search_title(webpage) | ||||
|         thumbnail = self._og_search_thumbnail(webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'url': video_url, | ||||
|             'thumbnail': thumb, | ||||
|             'thumbnail': thumbnail, | ||||
|         } | ||||
|   | ||||
							
								
								
									
										60
									
								
								youtube_dl/extractor/dumpert.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								youtube_dl/extractor/dumpert.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | ||||
| # coding: utf-8 | ||||
| from __future__ import unicode_literals | ||||
|  | ||||
| import base64 | ||||
|  | ||||
| from .common import InfoExtractor | ||||
| from ..compat import compat_urllib_request | ||||
| from ..utils import qualities | ||||
|  | ||||
|  | ||||
| class DumpertIE(InfoExtractor): | ||||
|     _VALID_URL = r'https?://(?:www\.)?dumpert\.nl/mediabase/(?P<id>[0-9]+/[0-9a-zA-Z]+)' | ||||
|     _TEST = { | ||||
|         'url': 'http://www.dumpert.nl/mediabase/6646981/951bc60f/', | ||||
|         'md5': '1b9318d7d5054e7dcb9dc7654f21d643', | ||||
|         'info_dict': { | ||||
|             'id': '6646981/951bc60f', | ||||
|             'ext': 'mp4', | ||||
|             'title': 'Ik heb nieuws voor je', | ||||
|             'description': 'Niet schrikken hoor', | ||||
|             'thumbnail': 're:^https?://.*\.jpg$', | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     def _real_extract(self, url): | ||||
|         video_id = self._match_id(url) | ||||
|  | ||||
|         req = compat_urllib_request.Request(url) | ||||
|         req.add_header('Cookie', 'nsfw=1; cpc=10') | ||||
|         webpage = self._download_webpage(req, video_id) | ||||
|  | ||||
|         files_base64 = self._search_regex( | ||||
|             r'data-files="([^"]+)"', webpage, 'data files') | ||||
|  | ||||
|         files = self._parse_json( | ||||
|             base64.b64decode(files_base64.encode('utf-8')).decode('utf-8'), | ||||
|             video_id) | ||||
|  | ||||
|         quality = qualities(['flv', 'mobile', 'tablet', '720p']) | ||||
|  | ||||
|         formats = [{ | ||||
|             'url': video_url, | ||||
|             'format_id': format_id, | ||||
|             'quality': quality(format_id), | ||||
|         } for format_id, video_url in files.items() if format_id != 'still'] | ||||
|         self._sort_formats(formats) | ||||
|  | ||||
|         title = self._html_search_meta( | ||||
|             'title', webpage) or self._og_search_title(webpage) | ||||
|         description = self._html_search_meta( | ||||
|             'description', webpage) or self._og_search_description(webpage) | ||||
|         thumbnail = files.get('still') or self._og_search_thumbnail(webpage) | ||||
|  | ||||
|         return { | ||||
|             'id': video_id, | ||||
|             'title': title, | ||||
|             'description': description, | ||||
|             'thumbnail': thumbnail, | ||||
|             'formats': formats | ||||
|         } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user